arabica/include/XPath/impl/xpath_axis_enumerator.hpp

600 lines
22 KiB
C++
Raw Normal View History

2005-08-04 22:42:30 +02:00
#ifndef ARABICA_XPATHIC_XPATH_AXIS_ENUMERATOR_H
#define ARABICA_XPATHIC_XPATH_AXIS_ENUMERATOR_H
2007-09-05 00:55:47 +02:00
#include <DOM/Node.hpp>
#include <DOM/Document.hpp>
#include <DOM/NamedNodeMap.hpp>
2005-08-04 22:42:30 +02:00
#include "xpath_namespace_node.hpp"
namespace Arabica
{
namespace XPath
{
enum Axis
{
ANCESTOR,
ANCESTOR_OR_SELF,
ATTRIBUTE,
CHILD,
DESCENDANT,
DESCENDANT_OR_SELF,
FOLLOWING,
FOLLOWING_SIBLING,
NAMESPACE,
PARENT,
PRECEDING,
PRECEDING_SIBLING,
SELF
}; // Axis
2005-08-21 16:25:14 +02:00
namespace impl
2005-08-04 22:42:30 +02:00
{
template<class string_type, class string_adaptor> class AxisWalker;
2005-08-04 22:42:30 +02:00
template<class axis_walker, class string_type, class string_adaptor>
AxisWalker<string_type, string_adaptor>* CreateAxis(const DOM::Node<string_type, string_adaptor>& context) { return new axis_walker(context); }
2005-08-21 16:25:14 +02:00
} // namespace impl
2005-08-04 22:42:30 +02:00
2005-08-21 16:25:14 +02:00
template<class string_type, class string_adaptor>
class AxisEnumerator
{
typedef impl::AxisWalker<string_type, string_adaptor>* (*CreateAxisPtr)(const DOM::Node<string_type, string_adaptor>& context);
2005-08-04 22:42:30 +02:00
struct NamedAxis { Axis name; CreateAxisPtr creator; };
static const NamedAxis AxisLookupTable[];
public:
AxisEnumerator(const DOM::Node<string_type, string_adaptor>& context, Axis axis) :
2005-08-04 22:42:30 +02:00
walker_(0)
{
for(const NamedAxis* ax = AxisLookupTable; ax->creator != 0; ++ax)
if(axis == ax->name)
walker_ = ax->creator(context);
if(!walker_)
throw std::runtime_error("Unknown Axis specifier");
} // AxisEnumerator
AxisEnumerator(const AxisEnumerator& rhs) :
walker_(rhs.walker_->clone())
{
} // AxisEnumerator
AxisEnumerator& operator=(const AxisEnumerator& rhs)
{
impl::AxisWalker<string_type, string_adaptor>* newwalker = rhs.walker_->clone();
2005-08-04 22:42:30 +02:00
delete walker_;
walker_ = newwalker;
return *this;
} // operator=
~AxisEnumerator()
{
delete walker_;
} // ~AxisEnumerator
bool forward() const { return walker_->forward(); }
bool reverse() const { return !walker_->forward(); }
const DOM::Node<string_type, string_adaptor>& operator*() const { return walker_->get(); }
const DOM::Node<string_type, string_adaptor>* const operator->() const { return &(walker_->get()); }
2005-08-04 22:42:30 +02:00
AxisEnumerator& operator++() { walker_->advance(); return *this; }
AxisEnumerator operator++(int) { AxisEnumerator copy(*this); walker_->advance(); return copy; }
private:
impl::AxisWalker<string_type, string_adaptor>* walker_;
2005-08-04 22:42:30 +02:00
AxisEnumerator();
2005-08-21 16:25:14 +02:00
}; // class AxisEnumerator
2005-08-04 22:42:30 +02:00
2005-08-21 16:25:14 +02:00
////////////////////////////////////////////////////
namespace impl
{
2005-08-04 22:42:30 +02:00
template<class string_type, class string_adaptor>
2005-08-21 16:25:14 +02:00
class AxisWalker
{
public:
virtual ~AxisWalker() { }
const DOM::Node<string_type, string_adaptor>& get() const { return current_; }
2005-08-21 16:25:14 +02:00
virtual void advance() = 0;
bool forward() { return forward_; }
virtual AxisWalker<string_type, string_adaptor>* clone() const = 0;
2005-08-21 16:25:14 +02:00
protected:
AxisWalker(bool forward) : forward_(forward) { }
AxisWalker(const AxisWalker& rhs) : current_(rhs.current_), forward_(rhs.forward_) { }
void set(const DOM::Node<string_type, string_adaptor>& current) { current_ = current; }
2005-08-21 16:25:14 +02:00
void end() { current_ = 0; }
2005-08-04 22:42:30 +02:00
static DOM::Node<string_type, string_adaptor> walkDown(const DOM::Node<string_type, string_adaptor>& context, const DOM::Node<string_type, string_adaptor>& origin)
2005-08-04 22:42:30 +02:00
{
if(context.getNodeType() == DOM::Node_base::ATTRIBUTE_NODE)
2005-08-21 16:25:14 +02:00
return 0;
DOM::Node<string_type, string_adaptor> next = context.getFirstChild();
2005-08-21 16:25:14 +02:00
if((next == 0) && (context == origin)) // node with no children
return 0;
2005-08-04 22:42:30 +02:00
2005-08-21 16:25:14 +02:00
if(next != 0)
return next;
next = context.getNextSibling();
if(next != 0)
return next;
DOM::Node<string_type, string_adaptor> parent = context.getParentNode();
2005-08-21 16:25:14 +02:00
while(parent != origin && next == 0)
2005-08-04 22:42:30 +02:00
{
2005-08-21 16:25:14 +02:00
next = parent.getNextSibling();
parent = parent.getParentNode();
} // while ...
return next;
} // walkDown
2005-08-04 22:42:30 +02:00
2005-08-21 16:25:14 +02:00
private:
DOM::Node<string_type, string_adaptor> current_;
2005-08-21 16:25:14 +02:00
bool forward_;
2005-08-04 22:42:30 +02:00
2005-08-21 16:25:14 +02:00
AxisWalker& operator=(const AxisWalker&);
bool operator==(const AxisWalker&);
}; // AxisWalker
2005-08-04 22:42:30 +02:00
template<class string_type, class string_adaptor>
class AncestorAxisWalker : public AxisWalker<string_type, string_adaptor>
2005-08-21 16:25:14 +02:00
{
public:
AncestorAxisWalker(const DOM::Node<string_type, string_adaptor>& context) : AxisWalker<string_type, string_adaptor>(false)
2005-08-04 22:42:30 +02:00
{
2005-08-21 16:25:14 +02:00
if(context == 0)
return;
2005-08-04 22:42:30 +02:00
if(context.getNodeType() != DOM::Node_base::ATTRIBUTE_NODE)
AxisWalker<string_type, string_adaptor>::set(context.getParentNode());
2005-08-21 16:25:14 +02:00
else
AxisWalker<string_type, string_adaptor>::set((static_cast<DOM::Attr<string_type, string_adaptor> >(context)).getOwnerElement());
2005-08-21 16:25:14 +02:00
} // AncestorAxisWalker
2005-08-04 22:42:30 +02:00
2005-08-21 16:25:14 +02:00
virtual void advance()
{
if(AxisWalker<string_type, string_adaptor>::get() != 0)
AxisWalker<string_type, string_adaptor>::set(AxisWalker<string_type, string_adaptor>::get().getParentNode());
2005-08-21 16:25:14 +02:00
} // advance
virtual AxisWalker<string_type, string_adaptor>* clone() const { return new AncestorAxisWalker(*this); }
2005-08-04 22:42:30 +02:00
2005-08-21 16:25:14 +02:00
private:
AncestorAxisWalker(const AncestorAxisWalker& rhs) : AxisWalker<string_type, string_adaptor>(rhs) { }
2005-08-21 16:25:14 +02:00
}; // class AncestorAxisWalker
2005-08-04 22:42:30 +02:00
template<class string_type, class string_adaptor>
class AncestorOrSelfAxisWalker : public AxisWalker<string_type, string_adaptor>
2005-08-21 16:25:14 +02:00
{
public:
AncestorOrSelfAxisWalker(const DOM::Node<string_type, string_adaptor>& context) : AxisWalker<string_type, string_adaptor>(false)
2005-08-21 16:25:14 +02:00
{
if(context != 0)
AxisWalker<string_type, string_adaptor>::set(context);
2005-08-21 16:25:14 +02:00
} // AncestorAxisWalker
2005-08-04 22:42:30 +02:00
2005-08-21 16:25:14 +02:00
virtual void advance()
{
if(AxisWalker<string_type, string_adaptor>::get() == 0)
2005-08-21 16:25:14 +02:00
return;
2005-08-04 22:42:30 +02:00
if(AxisWalker<string_type, string_adaptor>::get().getNodeType() != DOM::Node_base::ATTRIBUTE_NODE)
AxisWalker<string_type, string_adaptor>::set(AxisWalker<string_type, string_adaptor>::get().getParentNode());
2005-08-21 16:25:14 +02:00
else
AxisWalker<string_type, string_adaptor>::set((static_cast<DOM::Attr<string_type, string_adaptor> >(AxisWalker<string_type, string_adaptor>::get())).getOwnerElement());
2005-08-21 16:25:14 +02:00
} // advance
virtual AxisWalker<string_type, string_adaptor>* clone() const { return new AncestorOrSelfAxisWalker(*this); }
2005-08-04 22:42:30 +02:00
2005-08-21 16:25:14 +02:00
private:
AncestorOrSelfAxisWalker(const AncestorOrSelfAxisWalker& rhs) : AxisWalker<string_type, string_adaptor>(rhs) { }
2005-08-21 16:25:14 +02:00
}; // class AncestorOrSelfAxisWalker
2005-08-04 22:42:30 +02:00
template<class string_type, class string_adaptor>
class AttributeAxisWalker : public AxisWalker<string_type, string_adaptor>
2005-08-21 16:25:14 +02:00
{
public:
AttributeAxisWalker(const DOM::Node<string_type, string_adaptor>& context) : AxisWalker<string_type, string_adaptor>(true),
2005-08-21 16:25:14 +02:00
index_(0), count_(0)
{
if((context != 0) && (context.hasAttributes()))
2005-08-04 22:42:30 +02:00
{
2005-08-21 16:25:14 +02:00
attrs_ = context.getAttributes();
count_ = attrs_.getLength();
set_next();
} // if ...
} // AttributeAxisWalker
virtual void advance()
2005-08-04 22:42:30 +02:00
{
if(AxisWalker<string_type, string_adaptor>::get() == 0)
2005-08-21 16:25:14 +02:00
return;
set_next();
} // advance
2005-08-04 22:42:30 +02:00
virtual AxisWalker<string_type, string_adaptor>* clone() const { return new AttributeAxisWalker(*this); }
2005-08-04 22:42:30 +02:00
2005-08-21 16:25:14 +02:00
private:
AttributeAxisWalker(const AttributeAxisWalker& rhs) :
AxisWalker<string_type, string_adaptor>(rhs),
2005-08-21 16:25:14 +02:00
attrs_(rhs.attrs_),
index_(rhs.index_),
count_(rhs.count_) { }
2005-08-04 22:42:30 +02:00
DOM::NamedNodeMap<string_type, string_adaptor> attrs_;
2005-08-21 16:25:14 +02:00
unsigned int index_;
unsigned int count_;
void set_next()
{
if(index_ == count_)
2005-08-04 22:42:30 +02:00
{
AxisWalker<string_type, string_adaptor>::end();
2005-08-21 16:25:14 +02:00
return;
} // if ...
2005-08-04 22:42:30 +02:00
DOM::Node<string_type, string_adaptor> a;
2005-08-21 16:25:14 +02:00
do
{
a = attrs_.item(index_++);
2005-09-30 23:36:11 +02:00
} while ((a != 0) && (a.getNamespaceURI() == string_adaptor::construct_from_utf8("http://www.w3.org/2000/xmlns/")));
2005-08-04 22:42:30 +02:00
AxisWalker<string_type, string_adaptor>::set(a);
2005-08-21 16:25:14 +02:00
} // set_next
}; // class AttributeAxisEnumerator
2005-08-04 22:42:30 +02:00
template<class string_type, class string_adaptor>
class ChildAxisWalker : public AxisWalker<string_type, string_adaptor>
2005-08-21 16:25:14 +02:00
{
public:
ChildAxisWalker(const DOM::Node<string_type, string_adaptor>& context) : AxisWalker<string_type, string_adaptor>(true)
2005-08-04 22:42:30 +02:00
{
2005-08-21 16:25:14 +02:00
if(context != 0)
AxisWalker<string_type, string_adaptor>::set(context.getFirstChild());
2005-08-21 16:25:14 +02:00
} // ChildAxisWalker
2005-08-04 22:42:30 +02:00
2005-08-21 16:25:14 +02:00
virtual void advance()
{
if(AxisWalker<string_type, string_adaptor>::get() != 0)
AxisWalker<string_type, string_adaptor>::set(AxisWalker<string_type, string_adaptor>::get().getNextSibling());
2005-08-21 16:25:14 +02:00
} // advance
virtual AxisWalker<string_type, string_adaptor>* clone() const { return new ChildAxisWalker(*this); }
2005-08-04 22:42:30 +02:00
2005-08-21 16:25:14 +02:00
private:
ChildAxisWalker(const ChildAxisWalker& rhs) : AxisWalker<string_type, string_adaptor>(rhs) { }
2005-08-21 16:25:14 +02:00
}; // class ChildAxisWalker
2005-08-04 22:42:30 +02:00
template<class string_type, class string_adaptor>
class DescendantAxisWalker : public AxisWalker<string_type, string_adaptor>
2005-08-21 16:25:14 +02:00
{
public:
DescendantAxisWalker(const DOM::Node<string_type, string_adaptor>& context) : AxisWalker<string_type, string_adaptor>(true),
2005-08-21 16:25:14 +02:00
origin_(context)
{
if((context != 0) && (context.getNodeType() != DOM::Node_base::ATTRIBUTE_NODE))
AxisWalker<string_type, string_adaptor>::set(context.getFirstChild());
2005-08-21 16:25:14 +02:00
} // DescendantAxisWalker
2005-08-04 22:42:30 +02:00
2005-08-21 16:25:14 +02:00
virtual void advance()
2005-08-04 22:42:30 +02:00
{
AxisWalker<string_type, string_adaptor>::set(nextDescendant());
2005-08-21 16:25:14 +02:00
} // advance
2005-08-04 22:42:30 +02:00
virtual AxisWalker<string_type, string_adaptor>* clone() const { return new DescendantAxisWalker(*this); }
2005-08-04 22:42:30 +02:00
2005-08-21 16:25:14 +02:00
private:
DOM::Node<string_type, string_adaptor> nextDescendant()
2005-08-21 16:25:14 +02:00
{
DOM::Node<string_type, string_adaptor> next = AxisWalker<string_type, string_adaptor>::get().getFirstChild();
2005-08-21 16:25:14 +02:00
if(next == 0)
next = AxisWalker<string_type, string_adaptor>::get().getNextSibling();
2005-08-21 16:25:14 +02:00
if(next != 0)
return next;
DOM::Node<string_type, string_adaptor> parent = AxisWalker<string_type, string_adaptor>::get().getParentNode();
2005-08-21 16:25:14 +02:00
while(parent != origin_ && next == 0)
2005-08-04 22:42:30 +02:00
{
2005-08-21 16:25:14 +02:00
next = parent.getNextSibling();
parent = parent.getParentNode();
} // while ...
2005-08-04 22:42:30 +02:00
2005-08-21 16:25:14 +02:00
return next;
} // nextDescendant
2005-08-04 22:42:30 +02:00
DescendantAxisWalker(const DescendantAxisWalker& rhs) : AxisWalker<string_type, string_adaptor>(rhs), origin_(rhs.origin_) { }
const DOM::Node<string_type, string_adaptor> origin_;
2005-08-21 16:25:14 +02:00
}; // class DescendantAxisWalker
2005-08-04 22:42:30 +02:00
template<class string_type, class string_adaptor>
class DescendantOrSelfAxisWalker : public AxisWalker<string_type, string_adaptor>
2005-08-21 16:25:14 +02:00
{
public:
DescendantOrSelfAxisWalker(const DOM::Node<string_type, string_adaptor>& context) : AxisWalker<string_type, string_adaptor>(true),
2005-08-21 16:25:14 +02:00
origin_(context)
{
if(context != 0)
AxisWalker<string_type, string_adaptor>::set(context);
2005-08-21 16:25:14 +02:00
} // DescendantAxisWalker
2005-08-04 22:42:30 +02:00
2005-08-21 16:25:14 +02:00
virtual void advance()
2005-08-04 22:42:30 +02:00
{
AxisWalker<string_type, string_adaptor>::set(AxisWalker<string_type, string_adaptor>::walkDown(AxisWalker<string_type, string_adaptor>::get(), origin_));
2005-08-21 16:25:14 +02:00
} // advance
2005-08-04 22:42:30 +02:00
virtual AxisWalker<string_type, string_adaptor>* clone() const { return new DescendantOrSelfAxisWalker(*this); }
2005-08-04 22:42:30 +02:00
2005-08-21 16:25:14 +02:00
private:
DescendantOrSelfAxisWalker(const DescendantOrSelfAxisWalker& rhs) : AxisWalker<string_type, string_adaptor>(rhs), origin_(rhs.origin_) { }
const DOM::Node<string_type, string_adaptor> origin_;
2005-08-21 16:25:14 +02:00
}; // class DescendantOrSelfAxisWalker
2005-08-04 22:42:30 +02:00
template<class string_type, class string_adaptor>
class FollowingAxisWalker : public AxisWalker<string_type, string_adaptor>
2005-08-21 16:25:14 +02:00
{
public:
FollowingAxisWalker(const DOM::Node<string_type, string_adaptor>& context) : AxisWalker<string_type, string_adaptor>(true)
2005-08-04 22:42:30 +02:00
{
AxisWalker<string_type, string_adaptor>::set(firstFollowing(context));
2005-08-21 16:25:14 +02:00
} // FollowingAxisWalker
virtual void advance()
2005-08-04 22:42:30 +02:00
{
AxisWalker<string_type, string_adaptor>::set(AxisWalker<string_type, string_adaptor>::walkDown(AxisWalker<string_type, string_adaptor>::get(), AxisWalker<string_type, string_adaptor>::get().getOwnerDocument()));
2005-08-21 16:25:14 +02:00
} // advance
virtual AxisWalker<string_type, string_adaptor>* clone() const { return new FollowingAxisWalker(*this); }
2005-08-04 22:42:30 +02:00
2005-08-21 16:25:14 +02:00
private:
DOM::Node<string_type, string_adaptor> firstFollowing(const DOM::Node<string_type, string_adaptor>& context) const
2005-08-21 16:25:14 +02:00
{
if(context.getNodeType() == DOM::Node_base::ATTRIBUTE_NODE)
2007-07-19 19:01:31 +02:00
{
DOM::Node<string_type, string_adaptor> owner = static_cast<DOM::Attr<string_type, string_adaptor> >(context).getOwnerElement();
2007-07-19 19:01:31 +02:00
if(owner.hasChildNodes())
return owner.getFirstChild();
return firstFollowing(owner);
} // if attribute
2005-08-04 22:42:30 +02:00
DOM::Node<string_type, string_adaptor> next = context.getNextSibling();
2007-07-19 19:01:31 +02:00
2005-08-21 16:25:14 +02:00
if(next != 0)
return next;
DOM::Node<string_type, string_adaptor> parent = context.getParentNode();
2005-08-21 16:25:14 +02:00
while(parent != context.getOwnerDocument() && next == 0)
2005-08-04 22:42:30 +02:00
{
2005-08-21 16:25:14 +02:00
next = parent.getNextSibling();
parent = parent.getParentNode();
} // while ...
return next;
} // firstFollowing
2005-08-04 22:42:30 +02:00
FollowingAxisWalker(const FollowingAxisWalker& rhs) : AxisWalker<string_type, string_adaptor>(rhs) { }
2005-08-21 16:25:14 +02:00
}; // class FollowingAxisWalker
2005-08-04 22:42:30 +02:00
template<class string_type, class string_adaptor>
class FollowingSiblingAxisWalker : public AxisWalker<string_type, string_adaptor>
2005-08-21 16:25:14 +02:00
{
public:
FollowingSiblingAxisWalker(const DOM::Node<string_type, string_adaptor>& context) : AxisWalker<string_type, string_adaptor>(true)
2005-08-04 22:42:30 +02:00
{
2005-08-21 16:25:14 +02:00
if(context != 0)
AxisWalker<string_type, string_adaptor>::set(context.getNextSibling());
2005-08-21 16:25:14 +02:00
} // FollowingSiblingAxisWalker
2005-08-04 22:42:30 +02:00
2005-08-21 16:25:14 +02:00
virtual void advance()
{
if(AxisWalker<string_type, string_adaptor>::get() != 0)
AxisWalker<string_type, string_adaptor>::set(AxisWalker<string_type, string_adaptor>::get().getNextSibling());
2005-08-21 16:25:14 +02:00
} // advance
virtual AxisWalker<string_type, string_adaptor>* clone() const { return new FollowingSiblingAxisWalker(*this); }
2005-08-04 22:42:30 +02:00
2005-08-21 16:25:14 +02:00
private:
FollowingSiblingAxisWalker(const FollowingSiblingAxisWalker& rhs) : AxisWalker<string_type, string_adaptor>(rhs) { }
2005-08-21 16:25:14 +02:00
}; // class FollowingSiblingAxisWalker
2005-08-04 22:42:30 +02:00
2005-08-21 16:25:14 +02:00
template<class string_type, class string_adaptor>
class NamespaceAxisWalker : public AxisWalker<string_type, string_adaptor>
2005-08-21 16:25:14 +02:00
{
public:
NamespaceAxisWalker(const DOM::Node<string_type, string_adaptor>& context) : AxisWalker<string_type, string_adaptor>(true),
2005-09-30 23:36:11 +02:00
xmlns_prefix_(string_adaptor::construct_from_utf8("xmlns")),
2005-08-21 16:25:14 +02:00
index_(0)
{
DOM::Node<string_type, string_adaptor> current = context;
2007-07-19 19:01:31 +02:00
if(current.getNodeType() != DOM::Node_base::ATTRIBUTE_NODE)
list_.push_back(DOM::Node<string_type, string_adaptor>(
2007-07-19 19:01:31 +02:00
new NamespaceNodeImpl<string_type, string_adaptor>(string_adaptor::construct_from_utf8("xml"),
string_adaptor::construct_from_utf8("http://www.w3.org/XML/1998/namespace"))
)
);
while(current.getNodeType() == DOM::Node_base::ELEMENT_NODE)
2005-08-21 16:25:14 +02:00
{
for(unsigned int a = 0, ae = current.getAttributes().getLength(); a != ae; ++a)
{
DOM::Node<string_type, string_adaptor> attr = current.getAttributes().item(a);
2005-08-21 16:25:14 +02:00
if(attr.getPrefix() == xmlns_prefix_)
list_.push_back(DOM::Node<string_type, string_adaptor>(
new NamespaceNodeImpl<string_type, string_adaptor>(attr.getLocalName(),
attr.getNodeValue())
2005-08-21 16:25:14 +02:00
)
);
} // for ...
current = current.getParentNode();
} // while
list_.push_back(DOM::Node<string_type, string_adaptor>(0));
AxisWalker<string_type, string_adaptor>::set(list_[index_]);
2005-08-21 16:25:14 +02:00
} // NamespaceAxisWalker
virtual void advance()
{
if(index_ != list_.size())
AxisWalker<string_type, string_adaptor>::set(list_[++index_]);
2005-08-21 16:25:14 +02:00
} // advance
virtual AxisWalker<string_type, string_adaptor>* clone() const { return new NamespaceAxisWalker(*this); }
2005-08-04 22:42:30 +02:00
2005-08-21 16:25:14 +02:00
private:
NamespaceAxisWalker(const NamespaceAxisWalker& rhs) : AxisWalker<string_type, string_adaptor>(rhs) { }
std::vector<DOM::Node<string_type, string_adaptor> > list_;
2005-08-21 16:25:14 +02:00
const string_type xmlns_prefix_;
unsigned int index_;
}; // class NamespaceAxisWalker
template<class string_type, class string_adaptor>
class ParentAxisWalker : public AxisWalker<string_type, string_adaptor>
2005-08-21 16:25:14 +02:00
{
public:
ParentAxisWalker(const DOM::Node<string_type, string_adaptor>& context) : AxisWalker<string_type, string_adaptor>(false)
2005-08-21 16:25:14 +02:00
{
if(context == 0)
return;
2005-08-04 22:42:30 +02:00
if(context.getNodeType() != DOM::Node_base::ATTRIBUTE_NODE)
AxisWalker<string_type, string_adaptor>::set(context.getParentNode());
2005-08-21 16:25:14 +02:00
else
AxisWalker<string_type, string_adaptor>::set((static_cast<DOM::Attr<string_type, string_adaptor> >(context)).getOwnerElement());
2005-08-21 16:25:14 +02:00
} // ParentAxisWalker
2005-08-04 22:42:30 +02:00
2005-08-21 16:25:14 +02:00
virtual void advance()
{
if(AxisWalker<string_type, string_adaptor>::get() != 0)
AxisWalker<string_type, string_adaptor>::set(0);
2005-08-21 16:25:14 +02:00
} // advance
virtual AxisWalker<string_type, string_adaptor>* clone() const { return new ParentAxisWalker(*this); }
2005-08-04 22:42:30 +02:00
2005-08-21 16:25:14 +02:00
private:
ParentAxisWalker(const ParentAxisWalker& rhs) : AxisWalker<string_type, string_adaptor>(rhs) { }
2005-08-21 16:25:14 +02:00
}; // class ParentAxisWalker
2005-08-04 22:42:30 +02:00
template<class string_type, class string_adaptor>
class PrecedingAxisWalker : public AxisWalker<string_type, string_adaptor>
2005-08-21 16:25:14 +02:00
{
public:
PrecedingAxisWalker(const DOM::Node<string_type, string_adaptor>& context) : AxisWalker<string_type, string_adaptor>(false)
2005-08-21 16:25:14 +02:00
{
if(context.getNodeType() != DOM::Node_base::ATTRIBUTE_NODE)
2007-07-19 19:01:31 +02:00
firstPreceding(context);
else
firstPreceding((static_cast<DOM::Attr<string_type, string_adaptor> >(context)).getOwnerElement());
2005-08-21 16:25:14 +02:00
} // PrecedingAxisWalker
2005-08-04 22:42:30 +02:00
2005-08-21 16:25:14 +02:00
virtual void advance()
{
AxisWalker<string_type, string_adaptor>::set(previousInDocument(AxisWalker<string_type, string_adaptor>::get()));
2005-08-21 16:25:14 +02:00
} // advance
virtual AxisWalker<string_type, string_adaptor>* clone() const { return new PrecedingAxisWalker(*this); }
2005-08-04 22:42:30 +02:00
2005-08-21 16:25:14 +02:00
private:
void firstPreceding(const DOM::Node<string_type, string_adaptor>& context)
2007-07-19 19:01:31 +02:00
{
nextAncestor_ = context.getParentNode();
AxisWalker<string_type, string_adaptor>::set(previousInDocument(context));
2007-07-19 19:01:31 +02:00
} // firstPreceding
DOM::Node<string_type, string_adaptor> previousInDocument(const DOM::Node<string_type, string_adaptor>& context)
2005-08-04 22:42:30 +02:00
{
DOM::Node<string_type, string_adaptor> next = context.getPreviousSibling();
2005-08-21 16:25:14 +02:00
if(next != 0)
return getLastDescendant(next);
2005-08-04 22:42:30 +02:00
2005-08-21 16:25:14 +02:00
next = context.getParentNode();
if(next != nextAncestor_)
return next;
2005-08-04 22:42:30 +02:00
2005-08-21 16:25:14 +02:00
// ancestor collision!! woorp, woorp!
if(nextAncestor_ != 0)
2007-07-19 19:01:31 +02:00
{
nextAncestor_ = nextAncestor_.getParentNode();
if(nextAncestor_ != 0)
return previousInDocument(next);
} //
2005-08-04 22:42:30 +02:00
2005-08-21 16:25:14 +02:00
return 0;
} // previousInDocument
DOM::Node<string_type, string_adaptor> getLastDescendant(const DOM::Node<string_type, string_adaptor>& context)
2005-08-04 22:42:30 +02:00
{
2005-08-21 16:25:14 +02:00
if(context.getFirstChild() == 0)
return context;
2005-08-04 22:42:30 +02:00
DOM::Node<string_type, string_adaptor> c = context.getFirstChild();
2005-08-21 16:25:14 +02:00
while(c.getNextSibling() != 0)
c = c.getNextSibling();
2005-08-04 22:42:30 +02:00
2005-08-21 16:25:14 +02:00
return getLastDescendant(c);
} // getLastDescendant
2005-08-04 22:42:30 +02:00
PrecedingAxisWalker(const PrecedingAxisWalker& rhs) : AxisWalker<string_type, string_adaptor>(rhs), nextAncestor_(rhs.nextAncestor_) { }
DOM::Node<string_type, string_adaptor> nextAncestor_;
2005-08-21 16:25:14 +02:00
}; // PrecedingAxisWalker
template<class string_type, class string_adaptor>
class PrecedingSiblingAxisWalker : public AxisWalker<string_type, string_adaptor>
2005-08-21 16:25:14 +02:00
{
public:
PrecedingSiblingAxisWalker(const DOM::Node<string_type, string_adaptor>& context) : AxisWalker<string_type, string_adaptor>(false)
2005-08-04 22:42:30 +02:00
{
2005-08-21 16:25:14 +02:00
if(context != 0)
AxisWalker<string_type, string_adaptor>::set(context.getPreviousSibling());
2005-08-21 16:25:14 +02:00
} // PrecedingSiblingAxisWalker
2005-08-04 22:42:30 +02:00
2005-08-21 16:25:14 +02:00
virtual void advance()
{
if(AxisWalker<string_type, string_adaptor>::get() != 0)
AxisWalker<string_type, string_adaptor>::set(AxisWalker<string_type, string_adaptor>::get().getPreviousSibling());
2005-08-21 16:25:14 +02:00
} // advance
virtual AxisWalker<string_type, string_adaptor>* clone() const { return new PrecedingSiblingAxisWalker(*this); }
2005-08-04 22:42:30 +02:00
2005-08-21 16:25:14 +02:00
private:
PrecedingSiblingAxisWalker(const PrecedingSiblingAxisWalker& rhs) : AxisWalker<string_type, string_adaptor>(rhs) { }
2005-08-21 16:25:14 +02:00
}; // class PrecedingSiblingAxisWalker
2005-08-04 22:42:30 +02:00
template<class string_type, class string_adaptor>
class SelfAxisWalker : public AxisWalker<string_type, string_adaptor>
2005-08-21 16:25:14 +02:00
{
public:
SelfAxisWalker(const DOM::Node<string_type, string_adaptor>& context) : AxisWalker<string_type, string_adaptor>(true)
2005-08-21 16:25:14 +02:00
{
AxisWalker<string_type, string_adaptor>::set(context);
2005-08-21 16:25:14 +02:00
} // SelfAxisWalker
2005-08-04 22:42:30 +02:00
virtual void advance() { AxisWalker<string_type, string_adaptor>::end(); }
virtual AxisWalker<string_type, string_adaptor>* clone() const { return new SelfAxisWalker(*this); }
2005-08-21 16:25:14 +02:00
private:
SelfAxisWalker(const SelfAxisWalker& rhs) : AxisWalker<string_type, string_adaptor>(rhs) { }
2005-08-21 16:25:14 +02:00
}; // class SelfAxisWalker
} // namespace impl
2005-08-04 22:42:30 +02:00
2005-08-19 11:05:11 +02:00
template<class string_type, class string_adaptor>
const typename AxisEnumerator<string_type, string_adaptor>::NamedAxis
AxisEnumerator<string_type, string_adaptor>::AxisLookupTable[] =
{
{ ANCESTOR, impl::CreateAxis<impl::AncestorAxisWalker<string_type, string_adaptor>, string_type> },
{ ANCESTOR_OR_SELF, impl::CreateAxis<impl::AncestorOrSelfAxisWalker<string_type, string_adaptor>, string_type> },
{ ATTRIBUTE, impl::CreateAxis<impl::AttributeAxisWalker<string_type, string_adaptor>, string_type> },
{ CHILD, impl::CreateAxis<impl::ChildAxisWalker<string_type, string_adaptor>, string_type> },
{ DESCENDANT, impl::CreateAxis<impl::DescendantAxisWalker<string_type, string_adaptor>, string_type> },
{ DESCENDANT_OR_SELF, impl::CreateAxis<impl::DescendantOrSelfAxisWalker<string_type, string_adaptor>, string_type> },
{ FOLLOWING, impl::CreateAxis<impl::FollowingAxisWalker<string_type, string_adaptor>, string_type> },
{ FOLLOWING_SIBLING, impl::CreateAxis<impl::FollowingSiblingAxisWalker<string_type, string_adaptor>, string_type> },
2005-08-21 16:25:14 +02:00
{ NAMESPACE, impl::CreateAxis<impl::NamespaceAxisWalker<string_type, string_adaptor>, string_type> },
{ PARENT, impl::CreateAxis<impl::ParentAxisWalker<string_type, string_adaptor>, string_type> },
{ PRECEDING, impl::CreateAxis<impl::PrecedingAxisWalker<string_type, string_adaptor>, string_type> },
{ PRECEDING_SIBLING, impl::CreateAxis<impl::PrecedingSiblingAxisWalker<string_type, string_adaptor>, string_type> },
{ SELF, impl::CreateAxis<impl::SelfAxisWalker<string_type, string_adaptor>, string_type> },
2005-08-19 11:05:11 +02:00
{ static_cast<Axis>(0), 0 }
};
2005-08-04 22:42:30 +02:00
} // namespace XPath
} // namespace Arabica
#endif