/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /* ***** BEGIN LICENSE BLOCK ***** * Version: MPL 1.1/GPL 2.0/LGPL 2.1 * * The contents of this file are subject to the Mozilla Public License Version * 1.1 (the "License"); you may not use this file except in compliance with * the License. You may obtain a copy of the License at * http://www.mozilla.org/MPL/ * * Software distributed under the License is distributed on an "AS IS" basis, * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License * for the specific language governing rights and limitations under the * License. * * The Original Code is TransforMiiX XSLT processor code. * * The Initial Developer of the Original Code is * The MITRE Corporation. * Portions created by the Initial Developer are Copyright (C) 1999 * the Initial Developer. All Rights Reserved. * * Contributor(s): * Keith Visco (Original Author) * Larry Fitzpatick, OpenText * Jonas Sicking * * Alternatively, the contents of this file may be used under the terms of * either the GNU General Public License Version 2 or later (the "GPL"), or * the GNU Lesser General Public License Version 2.1 or later (the "LGPL"), * in which case the provisions of the GPL or the LGPL are applicable instead * of those above. If you wish to allow use of your version of this file only * under the terms of either the GPL or the LGPL, and not to allow others to * use your version of this file under the terms of the MPL, indicate your * decision by deleting the provisions above and replace them with the notice * and other provisions required by the GPL or the LGPL. If you do not delete * the provisions above, a recipient may use your version of this file under * the terms of any one of the MPL, the GPL or the LGPL. * * ***** END LICENSE BLOCK ***** */ #ifndef TRANSFRMX_EXPR_H #define TRANSFRMX_EXPR_H #include "txList.h" #include "nsAutoPtr.h" #include "txCore.h" #include "nsString.h" #ifdef DEBUG #define TX_TO_STRING #endif /* XPath class definitions. Much of this code was ported from XSL:P. */ class nsIAtom; class txAExprResult; class txIParseContext; class txIMatchContext; class txIEvalContext; class txNodeSet; class txXPathNode; /** * A Base Class for all XSL Expressions **/ class Expr : public TxObject { public: /** * Virtual destructor, important for subclasses **/ virtual ~Expr() { } /** * Evaluates this Expr based on the given context node and processor state * @param context the context node for evaluation of this Expr * @param ps the ContextState containing the stack information needed * for evaluation * @return the result of the evaluation **/ virtual nsresult evaluate(txIEvalContext* aContext, txAExprResult** aResult) = 0; #ifdef TX_TO_STRING /** * Returns the String representation of this Expr. * @param dest the String to use when creating the String * representation. The String representation will be appended to * any data in the destination String, to allow cascading calls to * other #toString() methods for Expressions. * @return the String representation of this Expr. **/ virtual void toString(nsAString& str) = 0; #endif }; //-- Expr #define TX_DECL_EVALUATE \ nsresult evaluate(txIEvalContext* aContext, txAExprResult** aResult) #ifndef TX_TO_STRING #define TX_DECL_EXPR TX_DECL_EVALUATE #define TX_DECL_FUNCTION TX_DECL_EVALUATE #else #define TX_DECL_EXPR \ TX_DECL_EVALUATE; \ void toString(nsAString& aDest) #define TX_DECL_FUNCTION \ TX_DECL_EVALUATE; \ nsresult getNameAtom(nsIAtom** aAtom) #endif /** * This class represents a FunctionCall as defined by the XPath 1.0 * Recommendation. **/ class FunctionCall : public Expr { public: virtual ~FunctionCall(); /** * Adds the given parameter to this FunctionCall's parameter list. * The ownership of the given Expr is passed over to the FunctionCall, * even on failure. * @param aExpr the Expr to add to this FunctionCall's parameter list * @return nsresult indicating out of memory */ nsresult addParam(Expr* aExpr); /** * Check if the number of parameters falls within a range. * * @param aParamCountMin minimum number of required parameters. * @param aParamCountMax maximum number of parameters. If aParamCountMax * is negative the maximum number is not checked. * @return boolean representing whether the number of parameters falls * within the expected range or not. * * XXX txIEvalContext should be txIParseContest, bug 143291 */ virtual PRBool requireParams(PRInt32 aParamCountMin, PRInt32 aParamCountMax, txIEvalContext* aContext); #ifdef TX_TO_STRING void toString(nsAString& aDest); #endif protected: txList params; FunctionCall(); /* * Evaluates the given Expression and converts its result to a String. * The value is appended to the given destination String */ void evaluateToString(Expr* aExpr, txIEvalContext* aContext, nsAString& aDest); /* * Evaluates the given Expression and converts its result to a number. */ static nsresult evaluateToNumber(Expr* aExpr, txIEvalContext* aContext, double* aResult); /* * Evaluates the given Expression and converts its result to a boolean. */ MBool evaluateToBoolean(Expr* aExpr, txIEvalContext* aContext); /* * Evaluates the given Expression and converts its result to a NodeSet. * If the result is not a NodeSet an error is returned. */ nsresult evaluateToNodeSet(Expr* aExpr, txIEvalContext* aContext, txNodeSet** aResult); #ifdef TX_TO_STRING /* * Returns the name of the function as an atom. */ virtual nsresult getNameAtom(nsIAtom** aAtom) = 0; #endif }; //-- FunctionCall /** * Represents an AttributeValueTemplate **/ class AttributeValueTemplate: public Expr { public: AttributeValueTemplate(); virtual ~AttributeValueTemplate(); /** * Adds the given Expr to this AttributeValueTemplate **/ void addExpr(Expr* expr); TX_DECL_EXPR; private: List expressions; }; /* * This class represents a NodeTest as defined by the XPath spec */ class txNodeTest { public: virtual ~txNodeTest() {} /* * Virtual methods * pretty much a txPattern, but not supposed to be used * standalone. The NodeTest node() is different to the * Pattern "node()" (document node isn't matched) */ virtual PRBool matches(const txXPathNode& aNode, txIMatchContext* aContext) = 0; virtual double getDefaultPriority() = 0; #ifdef TX_TO_STRING virtual void toString(nsAString& aDest) = 0; #endif }; #define TX_DECL_NODE_TEST_BASE \ PRBool matches(const txXPathNode& aNode, txIMatchContext* aContext); \ double getDefaultPriority() #ifndef TX_TO_STRING #define TX_DECL_NODE_TEST TX_DECL_NODE_TEST_BASE #else #define TX_DECL_NODE_TEST \ TX_DECL_NODE_TEST_BASE; \ void toString(nsAString& aDest) #endif /* * This class represents a NameTest as defined by the XPath spec */ class txNameTest : public txNodeTest { public: /* * Creates a new txNameTest with the given type and the given * principal node type */ txNameTest(nsIAtom* aPrefix, nsIAtom* aLocalName, PRInt32 aNSID, PRUint16 aNodeType); ~txNameTest(); TX_DECL_NODE_TEST; private: nsCOMPtr mPrefix; nsCOMPtr mLocalName; PRInt32 mNamespace; PRUint16 mNodeType; }; /* * This class represents a NodeType as defined by the XPath spec */ class txNodeTypeTest : public txNodeTest { public: enum NodeType { COMMENT_TYPE, TEXT_TYPE, PI_TYPE, NODE_TYPE }; /* * Creates a new txNodeTypeTest of the given type */ txNodeTypeTest(NodeType aNodeType); ~txNodeTypeTest(); /* * Sets the name of the node to match. Only availible for pi nodes */ void setNodeName(const nsAString& aName); TX_DECL_NODE_TEST; private: NodeType mNodeType; nsCOMPtr mNodeName; }; /** * Represents an ordered list of Predicates, * for use with Step and Filter Expressions **/ class PredicateList { public: /** * Creates a new PredicateList **/ PredicateList(); /** * Destructor, will delete all Expressions in the list, so remove * any you may need **/ virtual ~PredicateList(); /** * Adds the given Expr to the list. * The ownership of the given Expr is passed over the PredicateList, * even on failure. * @param aExpr the Expr to add to the list * @return nsresult indicating out of memory */ nsresult add(Expr* aExpr); nsresult evaluatePredicates(txNodeSet* aNodes, txIMatchContext* aContext); /** * returns true if this predicate list is empty **/ MBool isEmpty(); #ifdef TX_TO_STRING /** * Returns the String representation of this PredicateList. * @param dest the String to use when creating the String * representation. The String representation will be appended to * any data in the destination String, to allow cascading calls to * other #toString() methods for Expressions. * @return the String representation of this PredicateList. **/ virtual void toString(nsAString& dest); #endif protected: //-- list of predicates List predicates; }; //-- PredicateList class LocationStep : public PredicateList, public Expr { public: enum LocationStepType { ANCESTOR_AXIS = 0, ANCESTOR_OR_SELF_AXIS, ATTRIBUTE_AXIS, CHILD_AXIS, DESCENDANT_AXIS, DESCENDANT_OR_SELF_AXIS, FOLLOWING_AXIS, FOLLOWING_SIBLING_AXIS, NAMESPACE_AXIS, PARENT_AXIS, PRECEDING_AXIS, PRECEDING_SIBLING_AXIS, SELF_AXIS }; /** * Creates a new LocationStep using the given NodeExpr and Axis Identifier * @param nodeExpr the NodeExpr to use when matching Nodes * @param axisIdentifier the Axis Identifier in which to search for nodes **/ LocationStep(nsAutoPtr& aNodeTest, LocationStepType aAxisIdentifier) : mNodeTest(aNodeTest), mAxisIdentifier(aAxisIdentifier) { } TX_DECL_EXPR; private: void fromDescendants(const txXPathNode& aNode, txIMatchContext* aCs, txNodeSet* aNodes); void fromDescendantsRev(const txXPathNode& aNode, txIMatchContext* aCs, txNodeSet* aNodes); nsAutoPtr mNodeTest; LocationStepType mAxisIdentifier; }; class FilterExpr : public PredicateList, public Expr { public: /** * Creates a new FilterExpr using the given Expr * @param expr the Expr to use for evaluation */ FilterExpr(nsAutoPtr& aExpr) : expr(aExpr) { } TX_DECL_EXPR; private: nsAutoPtr expr; }; //-- FilterExpr class txLiteralExpr : public Expr { public: txLiteralExpr(double aDbl); txLiteralExpr(const nsAString& aStr); TX_DECL_EXPR; private: nsRefPtr mValue; }; /** * Represents an AdditiveExpr, a binary expression that * performs an additive operation between it's lvalue and rvalue: * + : add * - : subtract **/ class AdditiveExpr : public Expr { public: //-- AdditiveExpr Types //-- LF, changed from static const short to enum enum _AdditiveExprType { ADDITION = 1, SUBTRACTION }; AdditiveExpr(nsAutoPtr& aLeftExpr, nsAutoPtr& aRightExpr, short aOp) : op(aOp), leftExpr(aLeftExpr), rightExpr(aRightExpr) { } TX_DECL_EXPR; private: short op; nsAutoPtr leftExpr, rightExpr; }; //-- AdditiveExpr /** * Represents an UnaryExpr. Returns the negative value of it's expr. **/ class UnaryExpr : public Expr { public: UnaryExpr(nsAutoPtr& aExpr) : expr(aExpr) { } TX_DECL_EXPR; private: nsAutoPtr expr; }; //-- UnaryExpr /** * Represents a BooleanExpr, a binary expression that * performs a boolean operation between it's lvalue and rvalue. **/ class BooleanExpr : public Expr { public: //-- BooleanExpr Types enum _BooleanExprType { AND = 1, OR }; BooleanExpr(nsAutoPtr& aLeftExpr, nsAutoPtr& aRightExpr, short aOp) : op(aOp), leftExpr(aLeftExpr), rightExpr(aRightExpr) { } TX_DECL_EXPR; private: short op; nsAutoPtr leftExpr, rightExpr; }; //-- BooleanExpr /** * Represents a MultiplicativeExpr, a binary expression that * performs a multiplicative operation between it's lvalue and rvalue: * * : multiply * mod : modulus * div : divide * **/ class MultiplicativeExpr : public Expr { public: //-- MultiplicativeExpr Types //-- LF, changed from static const short to enum enum _MultiplicativeExprType { DIVIDE = 1, MULTIPLY, MODULUS }; MultiplicativeExpr(nsAutoPtr& aLeftExpr, nsAutoPtr& aRightExpr, short aOp) : op(aOp), leftExpr(aLeftExpr), rightExpr(aRightExpr) { } TX_DECL_EXPR; private: short op; nsAutoPtr leftExpr, rightExpr; }; //-- MultiplicativeExpr /** * Represents a RelationalExpr, an expression that compares it's lvalue * to it's rvalue using: * = : equal to * < : less than * > : greater than * <= : less than or equal to * >= : greater than or equal to * **/ class RelationalExpr : public Expr { public: enum RelationalExprType { EQUAL, NOT_EQUAL, LESS_THAN, GREATER_THAN, LESS_OR_EQUAL, GREATER_OR_EQUAL }; RelationalExpr(nsAutoPtr& aLeftExpr, nsAutoPtr& aRightExpr, RelationalExprType aOp) : mLeftExpr(aLeftExpr), mRightExpr(aRightExpr), mOp(aOp) { } TX_DECL_EXPR; private: PRBool compareResults(txIEvalContext* aContext, txAExprResult* aLeft, txAExprResult* aRight); nsAutoPtr mLeftExpr; nsAutoPtr mRightExpr; RelationalExprType mOp; }; /** * VariableRefExpr * Represents a variable reference ($refname) **/ class VariableRefExpr : public Expr { public: VariableRefExpr(nsIAtom* aPrefix, nsIAtom* aLocalName, PRInt32 aNSID); ~VariableRefExpr(); TX_DECL_EXPR; private: nsCOMPtr mPrefix; nsCOMPtr mLocalName; PRInt32 mNamespace; }; /** * Represents a PathExpr **/ class PathExpr : public Expr { public: //-- Path Operators //-- RELATIVE_OP is the default //-- LF, changed from static const short to enum enum PathOperator { RELATIVE_OP, DESCENDANT_OP }; /** * Creates a new PathExpr **/ PathExpr(); /** * Destructor, will delete all Expressions **/ virtual ~PathExpr(); /** * Adds the Expr to this PathExpr * The ownership of the given Expr is passed over the PathExpr, * even on failure. * @param aExpr the Expr to add to this PathExpr * @return nsresult indicating out of memory */ nsresult addExpr(Expr* aExpr, PathOperator pathOp); TX_DECL_EXPR; private: class PathExprItem { public: PathExprItem(Expr* aExpr, PathOperator aOp) : expr(aExpr), pathOp(aOp) {} nsAutoPtr expr; PathOperator pathOp; }; List expressions; /* * Selects from the descendants of the context node * all nodes that match the Expr */ nsresult evalDescendants(Expr* aStep, const txXPathNode& aNode, txIMatchContext* aContext, txNodeSet* resNodes); }; /** * This class represents a RootExpr, which only matches the Document node **/ class RootExpr : public Expr { public: /** * Creates a new RootExpr */ RootExpr() #ifdef TX_TO_STRING : mSerialize(PR_TRUE) #endif { } TX_DECL_EXPR; #ifdef TX_TO_STRING public: void setSerialize(PRBool aSerialize) { mSerialize = aSerialize; } private: // When a RootExpr is used in a PathExpr it shouldn't be serialized PRBool mSerialize; #endif }; //-- RootExpr /** * Represents a UnionExpr **/ class UnionExpr : public Expr { public: /** * Creates a new UnionExpr **/ UnionExpr(); /** * Destructor, will delete all Path Expressions **/ virtual ~UnionExpr(); /** * Adds the PathExpr to this UnionExpr * The ownership of the given Expr is passed over the UnionExpr, * even on failure. * @param aExpr the Expr to add to this UnionExpr * @return nsresult indicating out of memory */ nsresult addExpr(Expr* aExpr); TX_DECL_EXPR; private: List expressions; }; //-- UnionExpr /** * Expression that failed to parse */ class txErrorExpr : public Expr { public: #ifdef TX_TO_STRING txErrorExpr(const nsAString& aStr) : mStr(aStr) { } #endif TX_DECL_EXPR; #ifdef TX_TO_STRING private: nsString mStr; #endif }; #endif