#ifndef ARABICA_XPATHIC_XPATH_OBJECT_H #define ARABICA_XPATHIC_XPATH_OBJECT_H #include #include #include #include #include #include #include "xpath_execution_context.hpp" namespace Arabica { namespace XPath { class XPathExpression; typedef boost::shared_ptr XPathExpressionPtr; class XPathValue; typedef boost::shared_ptr XPathValuePtr; enum ValueType { ANY , BOOL, NUMBER, STRING, NODE_SET }; // ValueType int compareNodes(const DOM::Node& n1, const DOM::Node& n2); bool nodes_less_than(const DOM::Node& n1, const DOM::Node& n2); class NodeSet : public std::vector > { public: NodeSet() : std::vector >(), forward_(true), sorted_(false) { } // NodeSet NodeSet(bool forward) : std::vector >(), forward_(forward), sorted_(true) { } // NodeSet NodeSet(const NodeSet& rhs) : std::vector >(rhs), forward_(rhs.forward_), sorted_(rhs.sorted_) { } // NodeSet NodeSet& operator=(const NodeSet& rhs) { forward_ = rhs.forward_; sorted_ = rhs.sorted_; std::vector >::operator=(rhs); return *this; } // operator= void swap(NodeSet& rhs) { std::vector >::swap(rhs); std::swap(forward_, rhs.forward_); std::swap(sorted_, rhs.sorted_); } // swap bool forward() const { return sorted_ && forward_; } bool reverse() const { return sorted_ && !forward_; } void forward(bool forward) { forward_ = forward; sorted_ = true; } void to_document_order() { if(!sorted_) { std::sort(begin(), end(), nodes_less_than); sorted_ = true; forward_ = true; } // if(!sorted) if(!forward_) { std::reverse(begin(), end()); forward_ = true; } // if(!forward_) } // to_document_order DOM::Node top() const { if(forward_) return (*this)[0]; return (*this)[size()-1]; } // top() private: bool forward_; bool sorted_; }; // NodeSet class XPathValue { protected: XPathValue() { } public: virtual ~XPathValue() { } virtual bool asBool() const = 0; virtual double asNumber() const = 0; virtual std::string asString() const = 0; virtual const NodeSet& asNodeSet() const = 0; virtual ValueType type() const = 0; private: XPathValue(const XPathValue&); bool operator==(const XPathValue&); XPathValue& operator=(const XPathValue&); }; // class XPathValue const double NaN = std::sqrt(-2.0); const double Zero = 0.0; const double Negative_Zero = -Zero; const double Infinity = HUGE_VAL; const double Negative_Infinity = -Infinity; inline bool isNaN(double value) { return (value != value); } inline bool isInfinity(double value) { return (value == Infinity); } inline bool isNegativeInfinity(double value) { return (value == Negative_Infinity); } inline bool isInfinite(double value) { return isInfinity(value) || isNegativeInfinity(value); } inline double roundNumber(double value) { if(!(isNaN(value) || isInfinite(value) || (std::fabs(value) == 0))) if((value < 0.0) && (value > -0.5)) value = -0.0; else value = std::floor(value + 0.5); return value; } // roundNumber double stringAsNumber(const std::string& str); double nodeNumberValue(const DOM::Node& node); std::string nodeStringValue(const DOM::Node& node); bool areEqual(const XPathValuePtr& lhs, const XPathValuePtr& rhs); bool isLessThan(const XPathValuePtr& lhs, const XPathValuePtr& rhs); bool isLessThanEquals(const XPathValuePtr& lhs, const XPathValuePtr& rhs); bool isGreaterThan(const XPathValuePtr& lhs, const XPathValuePtr& rhs); bool isGreaterThanEquals(const XPathValuePtr& lhs, const XPathValuePtr& rhs); class XPathExpression { protected: XPathExpression() { } public: virtual ~XPathExpression() { } XPathValuePtr evaluate(const DOM::Node& context) const { ExecutionContext executionContext; return evaluate(context, executionContext); } // evaluate virtual bool evaluateAsBool(const DOM::Node& context) const { return evaluate(context)->asBool(); } virtual double evaluateAsNumber(const DOM::Node& context) const { return evaluate(context)->asNumber(); } virtual std::string evaluateAsString(const DOM::Node& context) const { return evaluate(context)->asString(); } virtual NodeSet evaluateAsNodeSet(const DOM::Node& context) const { return evaluate(context)->asNodeSet(); } virtual XPathValuePtr evaluate(const DOM::Node& context, const Arabica::XPath::ExecutionContext& executionContext) const = 0; private: XPathExpression(const XPathExpression&); bool operator==(const XPathExpression&); XPathExpression& operator=(const XPathExpression&); }; // class XPathExpression class UnaryExpression { public: UnaryExpression(XPathExpression* expr) : expr_(expr) { } protected: ~UnaryExpression() { delete expr_; } // ~UnaryExpression XPathExpression* expr() const { return expr_; } private: XPathExpression* expr_; }; // class UnaryExpression class BinaryExpression { public: BinaryExpression(XPathExpression* lhs, XPathExpression* rhs) : lhs_(lhs), rhs_(rhs) { } protected: ~BinaryExpression() { delete lhs_; delete rhs_; } // ~BinaryExpression XPathExpression* lhs() const { return lhs_; } XPathExpression* rhs() const { return rhs_; } private: XPathExpression* lhs_; XPathExpression* rhs_; }; // class BinaryExpression } // namespace XPath } // namespace Arabica #endif