arabica/XPath/impl/xpath_value.hpp

153 lines
5.7 KiB
C++

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