#ifndef ARABICA_XPATHIC_XPATH_ARITHMETIC_HPP #define ARABICA_XPATHIC_XPATH_ARITHMETIC_HPP #include "xpath_value.hpp" namespace Arabica { namespace XPath { namespace impl { template class PlusOperator : public BinaryExpression { typedef BinaryExpression baseT; public: PlusOperator(XPathExpression_impl* lhs, XPathExpression_impl* rhs) : BinaryExpression(lhs, rhs) { } virtual ValueType type() const { return NUMBER; } virtual XPathValue evaluate(const DOM::Node& context, const ExecutionContext& executionContext) const { return NumericValue::createValue(baseT::lhs()->evaluateAsNumber(context, executionContext) + baseT::rhs()->evaluateAsNumber(context, executionContext)); } // evaluate }; // class PlusOperator template class MinusOperator : public BinaryExpression { typedef BinaryExpression baseT; public: MinusOperator(XPathExpression_impl* lhs, XPathExpression_impl* rhs) : BinaryExpression(lhs, rhs) { } virtual ValueType type() const { return NUMBER; } virtual XPathValue evaluate(const DOM::Node& context, const ExecutionContext& executionContext) const { return NumericValue::createValue(baseT::lhs()->evaluateAsNumber(context, executionContext) - baseT::rhs()->evaluateAsNumber(context, executionContext)); } // evaluate }; // class MinusOperator template class MultiplyOperator : public BinaryExpression { typedef BinaryExpression baseT; public: MultiplyOperator(XPathExpression_impl* lhs, XPathExpression_impl* rhs) : BinaryExpression(lhs, rhs) { } virtual ValueType type() const { return NUMBER; } virtual XPathValue evaluate(const DOM::Node& context, const ExecutionContext& executionContext) const { return NumericValue::createValue(baseT::lhs()->evaluateAsNumber(context, executionContext) * baseT::rhs()->evaluateAsNumber(context, executionContext)); } // evaluate }; // class MultiplyOperator template class DivideOperator : public BinaryExpression { typedef BinaryExpression baseT; public: DivideOperator(XPathExpression_impl* lhs, XPathExpression_impl* rhs) : BinaryExpression(lhs, rhs) { } virtual ValueType type() const { return NUMBER; } virtual XPathValue evaluate(const DOM::Node& context, const ExecutionContext& executionContext) const { return NumericValue::createValue(baseT::lhs()->evaluateAsNumber(context, executionContext) / baseT::rhs()->evaluateAsNumber(context, executionContext)); } // evaluate }; // class DivideOperator template class ModOperator : public BinaryExpression { typedef BinaryExpression baseT; public: ModOperator(XPathExpression_impl* lhs, XPathExpression_impl* rhs) : BinaryExpression(lhs, rhs) { } virtual ValueType type() const { return NUMBER; } virtual XPathValue evaluate(const DOM::Node& context, const ExecutionContext& executionContext) const { double l = baseT::lhs()->evaluateAsNumber(context, executionContext); double r = baseT::rhs()->evaluateAsNumber(context, executionContext); if(isNaN(l) || isNaN(r)) return NumericValue::createValue(NaN); return NumericValue::createValue(static_cast(l) % static_cast(r)); } // evaluate }; // class ModOperator template class UnaryNegative : public UnaryExpression { typedef UnaryExpression baseT; public: UnaryNegative(XPathExpression_impl* expr) : UnaryExpression(expr) { } virtual ValueType type() const { return NUMBER; } virtual XPathValue evaluate(const DOM::Node& context, const ExecutionContext& executionContext) const { return NumericValue::createValue(-baseT::expr()->evaluate(context, executionContext).asNumber()); } // evaluate }; // class UnaryNegative } // namespace impl } // namespace XPath } // namespace Arabica #endif