#ifndef ARABICA_XPATHIC_XPATH_VALUE_H #define ARABICA_XPATHIC_XPATH_VALUE_H #include #include #include #include #include #include "xpath_object.hpp" namespace Arabica { namespace XPath { template class Value_base : public XPathExpression_impl, public XPathValue_impl { typedef XPathValue_impl XPathValue_implT; using XPathValue_implT::asBool; using XPathValue_implT::asNumber; using XPathValue_implT::asString; using XPathValue_implT::asNodeSet; protected: Value_base() { } ~Value_base() { } public: virtual bool evaluateAsBool(const DOM::Node& /* context */) { return asBool(); } virtual double evaluateAsNumber(const DOM::Node& /* context */) { return asNumber(); } virtual string_type evaluateAsString(const DOM::Node& /* context */) { return asString(); } virtual NodeSet evaluateAsNodeSet(const DOM::Node& /* context */) { return asNodeSet(); } virtual bool evaluateAsBool(const DOM::Node& /* context */, const ExecutionContext& /* executionContext */) const { return asBool(); } virtual double evaluateAsNumber(const DOM::Node& /* context */, const ExecutionContext& /* executionContext */) const { return asNumber(); } virtual string_type evaluateAsString(const DOM::Node& /* context */, const ExecutionContext& /* executionContext */) const { return asString(); } virtual NodeSet evaluateAsNodeSet(const DOM::Node& /* context */, const ExecutionContext& /* executionContext */) const { return asNodeSet(); } }; // class Value_base template > class BoolValue : public Value_base { public: BoolValue(bool value) : value_(value) { } static XPathValue createValue(bool value) { return XPathValue(new BoolValue(value)); } virtual XPathValue evaluate(const DOM::Node& /* context */, const ExecutionContext& /* executionContext */) const { return XPathValue(new BoolValue(value_)); } // evaluate virtual bool asBool() const { return value_; } virtual double asNumber() const { return value_ ? 1 : 0; } virtual string_type asString() const { return string_adaptor::construct_from_utf8(value_ ? "true" : "false"); } virtual const NodeSet& asNodeSet() const { static NodeSet empty; return empty; } virtual ValueType type() const { return BOOL; } private: bool value_; }; // class BoolValue template > class NumericValue : public Value_base { public: NumericValue(double value) : value_(value) { } static XPathValue createValue(double value) { return XPathValue(new NumericValue(value)); } virtual XPathValue evaluate(const DOM::Node& /* context */, const ExecutionContext& /* executionContext */) const { return createValue(value_); } // evaluate virtual bool asBool() const { if(isNaN(value_)) return false; return (value_ != 0.0); } // asBool virtual double asNumber() const { return value_; } virtual string_type asString() const { if(isNaN(value_)) return string_adaptor::construct_from_utf8("NaN"); if(isInfinity(value_)) return string_adaptor::construct_from_utf8("Infinity"); if(isNegativeInfinity(value_)) return string_adaptor::construct_from_utf8("-Infinity"); return string_adaptor::construct_from_utf8(boost::lexical_cast(value_).c_str()); } // asString virtual const NodeSet& asNodeSet() const { static NodeSet empty; return empty; } virtual ValueType type() const { return NUMBER; } private: double value_; }; // class NumberValue template > class StringValue : public Value_base { public: StringValue(const char* value) : value_(string_adaptor::construct_from_utf8(value)) { } StringValue(const string_type& value) : value_(value) { } static XPathValue createValue(const string_type& value) { return XPathValue(new StringValue(value)); } virtual XPathValue evaluate(const DOM::Node& /* context */, const ExecutionContext& /* executionContext */) const { return XPathValue(new StringValue(value_)); } // evaluate virtual bool asBool() const { return !string_adaptor::empty(value_); } virtual double asNumber() const { return impl::stringAsNumber(value_); } // asNumber virtual string_type asString() const { return value_; } virtual const NodeSet& asNodeSet() const { static NodeSet empty; return empty; } virtual ValueType type() const { return STRING; } private: string_type value_; }; // class StringValue template > class NodeSetValue : public Value_base { public: NodeSetValue(const NodeSet& set) : set_(set) { } static XPathValue createValue(const NodeSet& set) { return XPathValue(new NodeSetValue(set)); } // createValue virtual XPathValue evaluate(const DOM::Node& /* context */, const ExecutionContext& /* executionContext */) const { return XPathValue(this); } // evaluate virtual bool asBool() const { return !set_.empty(); } // asBool virtual double asNumber() const { return impl::stringAsNumber(asString()); } // asNumber virtual string_type asString() const { if(set_.empty()) return string_adaptor::construct_from_utf8(""); return impl::nodeStringValue(set_.top()); } // asStringx virtual const NodeSet& asNodeSet() const { return set_; } // asNodeSet virtual ValueType type() const { return NODE_SET; } private: mutable NodeSet set_; }; // NodeSetValue } // namespace XPath } // namespace Arabica #endif