001    /*
002     * Licensed to the Apache Software Foundation (ASF) under one
003     * or more contributor license agreements. See the NOTICE file
004     * distributed with this work for additional information
005     * regarding copyright ownership. The ASF licenses this file
006     * to you under the Apache License, Version 2.0 (the  "License");
007     * you may not use this file except in compliance with the License.
008     * You may obtain a copy of the License at
009     *
010     *     http://www.apache.org/licenses/LICENSE-2.0
011     *
012     * Unless required by applicable law or agreed to in writing, software
013     * distributed under the License is distributed on an "AS IS" BASIS,
014     * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
015     * See the License for the specific language governing permissions and
016     * limitations under the License.
017     */
018    /*
019     * $Id: Operation.java 468655 2006-10-28 07:12:06Z minchau $
020     */
021    package org.apache.xpath.operations;
022    
023    import org.apache.xpath.Expression;
024    import org.apache.xpath.ExpressionOwner;
025    import org.apache.xpath.XPathContext;
026    import org.apache.xpath.XPathVisitor;
027    import org.apache.xpath.objects.XObject;
028    
029    /**
030     * The baseclass for a binary operation.
031     */
032    public class Operation extends Expression implements ExpressionOwner
033    {
034        static final long serialVersionUID = -3037139537171050430L;
035    
036      /** The left operand expression.
037       *  @serial */
038      protected Expression m_left;
039    
040      /** The right operand expression.
041       *  @serial */
042      protected Expression m_right;
043      
044      /**
045       * This function is used to fixup variables from QNames to stack frame 
046       * indexes at stylesheet build time.
047       * @param vars List of QNames that correspond to variables.  This list 
048       * should be searched backwards for the first qualified name that 
049       * corresponds to the variable reference qname.  The position of the 
050       * QName in the vector from the start of the vector will be its position 
051       * in the stack frame (but variables above the globalsTop value will need 
052       * to be offset to the current stack frame).
053       */
054      public void fixupVariables(java.util.Vector vars, int globalsSize)
055      {
056        m_left.fixupVariables(vars, globalsSize);
057        m_right.fixupVariables(vars, globalsSize);
058      }
059    
060    
061      /**
062       * Tell if this expression or it's subexpressions can traverse outside
063       * the current subtree.
064       *
065       * @return true if traversal outside the context node's subtree can occur.
066       */
067      public boolean canTraverseOutsideSubtree()
068      {
069    
070        if (null != m_left && m_left.canTraverseOutsideSubtree())
071          return true;
072    
073        if (null != m_right && m_right.canTraverseOutsideSubtree())
074          return true;
075    
076        return false;
077      }
078    
079      /**
080       * Set the left and right operand expressions for this operation.
081       *
082       *
083       * @param l The left expression operand.
084       * @param r The right expression operand.
085       */
086      public void setLeftRight(Expression l, Expression r)
087      {
088        m_left = l;
089        m_right = r;
090        l.exprSetParent(this);
091        r.exprSetParent(this);
092      }
093    
094      /**
095       * Execute a binary operation by calling execute on each of the operands,
096       * and then calling the operate method on the derived class.
097       *
098       *
099       * @param xctxt The runtime execution context.
100       *
101       * @return The XObject result of the operation.
102       *
103       * @throws javax.xml.transform.TransformerException
104       */
105      public XObject execute(XPathContext xctxt)
106              throws javax.xml.transform.TransformerException
107      {
108    
109        XObject left = m_left.execute(xctxt, true);
110        XObject right = m_right.execute(xctxt, true);
111    
112        XObject result = operate(left, right);
113        left.detach();
114        right.detach();
115        return result;
116      }
117    
118      /**
119       * Apply the operation to two operands, and return the result.
120       *
121       *
122       * @param left non-null reference to the evaluated left operand.
123       * @param right non-null reference to the evaluated right operand.
124       *
125       * @return non-null reference to the XObject that represents the result of the operation.
126       *
127       * @throws javax.xml.transform.TransformerException
128       */
129      public XObject operate(XObject left, XObject right)
130              throws javax.xml.transform.TransformerException
131      {
132        return null;  // no-op
133      }
134    
135      /** @return the left operand of binary operation, as an Expression.
136       */
137      public Expression getLeftOperand(){
138        return m_left;
139      }
140    
141      /** @return the right operand of binary operation, as an Expression.
142       */
143      public Expression getRightOperand(){
144        return m_right;
145      }
146      
147      class LeftExprOwner implements ExpressionOwner
148      {
149        /**
150         * @see ExpressionOwner#getExpression()
151         */
152        public Expression getExpression()
153        {
154          return m_left;
155        }
156    
157        /**
158         * @see ExpressionOwner#setExpression(Expression)
159         */
160        public void setExpression(Expression exp)
161        {
162            exp.exprSetParent(Operation.this);
163            m_left = exp;
164        }
165      }
166    
167      /**
168       * @see org.apache.xpath.XPathVisitable#callVisitors(ExpressionOwner, XPathVisitor)
169       */
170      public void callVisitors(ExpressionOwner owner, XPathVisitor visitor)
171      {
172            if(visitor.visitBinaryOperation(owner, this))
173            {
174                    m_left.callVisitors(new LeftExprOwner(), visitor);
175                    m_right.callVisitors(this, visitor);
176            }
177      }
178    
179      /**
180       * @see ExpressionOwner#getExpression()
181       */
182      public Expression getExpression()
183      {
184        return m_right;
185      }
186    
187      /**
188       * @see ExpressionOwner#setExpression(Expression)
189       */
190      public void setExpression(Expression exp)
191      {
192            exp.exprSetParent(this);
193            m_right = exp;
194      }
195    
196      /**
197       * @see Expression#deepEquals(Expression)
198       */
199      public boolean deepEquals(Expression expr)
200      {
201            if(!isSameClass(expr))
202                    return false;
203                    
204            if(!m_left.deepEquals(((Operation)expr).m_left))
205                    return false;
206                    
207            if(!m_right.deepEquals(((Operation)expr).m_right))
208                    return false;
209                    
210            return true;
211      }
212    }