Home | History | Annotate | Download | only in xpath
      1 /*
      2  * Licensed to the Apache Software Foundation (ASF) under one
      3  * or more contributor license agreements. See the NOTICE file
      4  * distributed with this work for additional information
      5  * regarding copyright ownership. The ASF licenses this file
      6  * to you under the Apache License, Version 2.0 (the  "License");
      7  * you may not use this file except in compliance with the License.
      8  * You may obtain a copy of the License at
      9  *
     10  *     http://www.apache.org/licenses/LICENSE-2.0
     11  *
     12  * Unless required by applicable law or agreed to in writing, software
     13  * distributed under the License is distributed on an "AS IS" BASIS,
     14  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
     15  * See the License for the specific language governing permissions and
     16  * limitations under the License.
     17  */
     18 /*
     19  * $Id: NodeSetDTM.java 468655 2006-10-28 07:12:06Z minchau $
     20  */
     21 package org.apache.xpath;
     22 
     23 import org.apache.xalan.res.XSLMessages;
     24 import org.apache.xml.dtm.DTM;
     25 import org.apache.xml.dtm.DTMFilter;
     26 import org.apache.xml.dtm.DTMIterator;
     27 import org.apache.xml.dtm.DTMManager;
     28 import org.apache.xml.utils.NodeVector;
     29 import org.apache.xpath.res.XPATHErrorResources;
     30 
     31 import org.w3c.dom.Node;
     32 import org.w3c.dom.NodeList;
     33 import org.w3c.dom.traversal.NodeIterator;
     34 
     35 
     36 /**
     37  * <p>The NodeSetDTM class can act as either a NodeVector,
     38  * NodeList, or NodeIterator.  However, in order for it to
     39  * act as a NodeVector or NodeList, it's required that
     40  * setShouldCacheNodes(true) be called before the first
     41  * nextNode() is called, in order that nodes can be added
     42  * as they are fetched.  Derived classes that implement iterators
     43  * must override runTo(int index), in order that they may
     44  * run the iteration to the given index. </p>
     45  *
     46  * <p>Note that we directly implement the DOM's NodeIterator
     47  * interface. We do not emulate all the behavior of the
     48  * standard NodeIterator. In particular, we do not guarantee
     49  * to present a "live view" of the document ... but in XSLT,
     50  * the source document should never be mutated, so this should
     51  * never be an issue.</p>
     52  *
     53  * <p>Thought: Should NodeSetDTM really implement NodeList and NodeIterator,
     54  * or should there be specific subclasses of it which do so? The
     55  * advantage of doing it all here is that all NodeSetDTMs will respond
     56  * to the same calls; the disadvantage is that some of them may return
     57  * less-than-enlightening results when you do so.</p>
     58  * @xsl.usage advanced
     59  */
     60 public class NodeSetDTM extends NodeVector
     61         implements /* NodeList, NodeIterator, */ DTMIterator,
     62         Cloneable
     63 {
     64     static final long serialVersionUID = 7686480133331317070L;
     65 
     66   /**
     67    * Create an empty nodelist.
     68    */
     69   public NodeSetDTM(DTMManager dtmManager)
     70   {
     71     super();
     72     m_manager = dtmManager;
     73   }
     74 
     75   /**
     76    * Create an empty, using the given block size.
     77    *
     78    * @param blocksize Size of blocks to allocate
     79    * @param dummy pass zero for right now...
     80    */
     81   public NodeSetDTM(int blocksize, int dummy, DTMManager dtmManager)
     82   {
     83     super(blocksize);
     84     m_manager = dtmManager;
     85   }
     86 
     87   // %TBD%
     88 //  /**
     89 //   * Create a NodeSetDTM, and copy the members of the
     90 //   * given nodelist into it.
     91 //   *
     92 //   * @param nodelist List of Nodes to be made members of the new set.
     93 //   */
     94 //  public NodeSetDTM(NodeList nodelist)
     95 //  {
     96 //
     97 //    super();
     98 //
     99 //    addNodes(nodelist);
    100 //  }
    101 
    102   /**
    103    * Create a NodeSetDTM, and copy the members of the
    104    * given NodeSetDTM into it.
    105    *
    106    * @param nodelist Set of Nodes to be made members of the new set.
    107    */
    108   public NodeSetDTM(NodeSetDTM nodelist)
    109   {
    110 
    111     super();
    112     m_manager = nodelist.getDTMManager();
    113     m_root = nodelist.getRoot();
    114 
    115     addNodes((DTMIterator) nodelist);
    116   }
    117 
    118   /**
    119    * Create a NodeSetDTM, and copy the members of the
    120    * given DTMIterator into it.
    121    *
    122    * @param ni Iterator which yields Nodes to be made members of the new set.
    123    */
    124   public NodeSetDTM(DTMIterator ni)
    125   {
    126 
    127     super();
    128 
    129     m_manager = ni.getDTMManager();
    130     m_root = ni.getRoot();
    131     addNodes(ni);
    132   }
    133 
    134   /**
    135    * Create a NodeSetDTM, and copy the members of the
    136    * given DTMIterator into it.
    137    *
    138    * @param iterator Iterator which yields Nodes to be made members of the new set.
    139    */
    140   public NodeSetDTM(NodeIterator iterator, XPathContext xctxt)
    141   {
    142 
    143     super();
    144 
    145     Node node;
    146     m_manager = xctxt.getDTMManager();
    147 
    148     while (null != (node = iterator.nextNode()))
    149     {
    150       int handle = xctxt.getDTMHandleFromNode(node);
    151       addNodeInDocOrder(handle, xctxt);
    152     }
    153   }
    154 
    155   /**
    156    * Create a NodeSetDTM, and copy the members of the
    157    * given DTMIterator into it.
    158    *
    159    */
    160   public NodeSetDTM(NodeList nodeList, XPathContext xctxt)
    161   {
    162 
    163     super();
    164 
    165     m_manager = xctxt.getDTMManager();
    166 
    167     int n = nodeList.getLength();
    168     for (int i = 0; i < n; i++)
    169     {
    170       Node node = nodeList.item(i);
    171       int handle = xctxt.getDTMHandleFromNode(node);
    172       // Do not reorder or strip duplicate nodes from the given DOM nodelist
    173       addNode(handle); // addNodeInDocOrder(handle, xctxt);
    174     }
    175   }
    176 
    177 
    178   /**
    179    * Create a NodeSetDTM which contains the given Node.
    180    *
    181    * @param node Single node to be added to the new set.
    182    */
    183   public NodeSetDTM(int node, DTMManager dtmManager)
    184   {
    185 
    186     super();
    187     m_manager = dtmManager;
    188 
    189     addNode(node);
    190   }
    191 
    192   /**
    193    * Set the environment in which this iterator operates, which should provide:
    194    * a node (the context node... same value as "root" defined below)
    195    * a pair of non-zero positive integers (the context position and the context size)
    196    * a set of variable bindings
    197    * a function library
    198    * the set of namespace declarations in scope for the expression.
    199    *
    200    * <p>At this time the exact implementation of this environment is application
    201    * dependent.  Probably a proper interface will be created fairly soon.</p>
    202    *
    203    * @param environment The environment object.
    204    */
    205   public void setEnvironment(Object environment)
    206   {
    207     // no-op
    208   }
    209 
    210 
    211   /**
    212    * @return The root node of the Iterator, as specified when it was created.
    213    * For non-Iterator NodeSetDTMs, this will be null.
    214    */
    215   public int getRoot()
    216   {
    217     if(DTM.NULL == m_root)
    218     {
    219       if(size() > 0)
    220         return item(0);
    221       else
    222         return DTM.NULL;
    223     }
    224     else
    225       return m_root;
    226   }
    227 
    228   /**
    229    * Initialize the context values for this expression
    230    * after it is cloned.
    231    *
    232    * @param context The XPath runtime context for this
    233    * transformation.
    234    */
    235   public void setRoot(int context, Object environment)
    236   {
    237     // no-op, I guess...  (-sb)
    238   }
    239 
    240   /**
    241    * Clone this NodeSetDTM.
    242    * At this time, we only expect this to be used with LocPathIterators;
    243    * it may not work with other kinds of NodeSetDTMs.
    244    *
    245    * @return a new NodeSetDTM of the same type, having the same state...
    246    * though unless overridden in the subclasses, it may not copy all
    247    * the state information.
    248    *
    249    * @throws CloneNotSupportedException if this subclass of NodeSetDTM
    250    * does not support the clone() operation.
    251    */
    252   public Object clone() throws CloneNotSupportedException
    253   {
    254 
    255     NodeSetDTM clone = (NodeSetDTM) super.clone();
    256 
    257     return clone;
    258   }
    259 
    260   /**
    261    * Get a cloned Iterator, and reset its state to the beginning of the
    262    * iteration.
    263    *
    264    * @return a new NodeSetDTM of the same type, having the same state...
    265    * except that the reset() operation has been called.
    266    *
    267    * @throws CloneNotSupportedException if this subclass of NodeSetDTM
    268    * does not support the clone() operation.
    269    */
    270   public DTMIterator cloneWithReset() throws CloneNotSupportedException
    271   {
    272 
    273     NodeSetDTM clone = (NodeSetDTM) clone();
    274 
    275     clone.reset();
    276 
    277     return clone;
    278   }
    279 
    280   /**
    281    * Reset the iterator. May have no effect on non-iterator Nodesets.
    282    */
    283   public void reset()
    284   {
    285     m_next = 0;
    286   }
    287 
    288   /**
    289    *  This attribute determines which node types are presented via the
    290    * iterator. The available set of constants is defined in the
    291    * <code>DTMFilter</code> interface. For NodeSetDTMs, the mask has been
    292    * hardcoded to show all nodes except EntityReference nodes, which have
    293    * no equivalent in the XPath data model.
    294    *
    295    * @return integer used as a bit-array, containing flags defined in
    296    * the DOM's DTMFilter class. The value will be
    297    * <code>SHOW_ALL & ~SHOW_ENTITY_REFERENCE</code>, meaning that
    298    * only entity references are suppressed.
    299    */
    300   public int getWhatToShow()
    301   {
    302     return DTMFilter.SHOW_ALL & ~DTMFilter.SHOW_ENTITY_REFERENCE;
    303   }
    304 
    305   /**
    306    * The filter object used to screen nodes. Filters are applied to
    307    * further reduce (and restructure) the DTMIterator's view of the
    308    * document. In our case, we will be using hardcoded filters built
    309    * into our iterators... but getFilter() is part of the DOM's
    310    * DTMIterator interface, so we have to support it.
    311    *
    312    * @return null, which is slightly misleading. True, there is no
    313    * user-written filter object, but in fact we are doing some very
    314    * sophisticated custom filtering. A DOM purist might suggest
    315    * returning a placeholder object just to indicate that this is
    316    * not going to return all nodes selected by whatToShow.
    317    */
    318   public DTMFilter getFilter()
    319   {
    320     return null;
    321   }
    322 
    323   /**
    324    *  The value of this flag determines whether the children of entity
    325    * reference nodes are visible to the iterator. If false, they will be
    326    * skipped over.
    327    * <br> To produce a view of the document that has entity references
    328    * expanded and does not expose the entity reference node itself, use the
    329    * whatToShow flags to hide the entity reference node and set
    330    * expandEntityReferences to true when creating the iterator. To produce
    331    * a view of the document that has entity reference nodes but no entity
    332    * expansion, use the whatToShow flags to show the entity reference node
    333    * and set expandEntityReferences to false.
    334    *
    335    * @return true for all iterators based on NodeSetDTM, meaning that the
    336    * contents of EntityRefrence nodes may be returned (though whatToShow
    337    * says that the EntityReferences themselves are not shown.)
    338    */
    339   public boolean getExpandEntityReferences()
    340   {
    341     return true;
    342   }
    343 
    344   /**
    345    * Get an instance of a DTM that "owns" a node handle.  Since a node
    346    * iterator may be passed without a DTMManager, this allows the
    347    * caller to easily get the DTM using just the iterator.
    348    *
    349    * @param nodeHandle the nodeHandle.
    350    *
    351    * @return a non-null DTM reference.
    352    */
    353   public DTM getDTM(int nodeHandle)
    354   {
    355 
    356     return m_manager.getDTM(nodeHandle);
    357   }
    358 
    359   /* An instance of the DTMManager. */
    360   DTMManager m_manager;
    361 
    362   /**
    363    * Get an instance of the DTMManager.  Since a node
    364    * iterator may be passed without a DTMManager, this allows the
    365    * caller to easily get the DTMManager using just the iterator.
    366    *
    367    * @return a non-null DTMManager reference.
    368    */
    369   public DTMManager getDTMManager()
    370   {
    371 
    372     return m_manager;
    373   }
    374 
    375   /**
    376    *  Returns the next node in the set and advances the position of the
    377    * iterator in the set. After a DTMIterator is created, the first call
    378    * to nextNode() returns the first node in the set.
    379    * @return  The next <code>Node</code> in the set being iterated over, or
    380    *   <code>DTM.NULL</code> if there are no more members in that set.
    381    * @throws DOMException
    382    *    INVALID_STATE_ERR: Raised if this method is called after the
    383    *   <code>detach</code> method was invoked.
    384    */
    385   public int nextNode()
    386   {
    387 
    388     if ((m_next) < this.size())
    389     {
    390       int next = this.elementAt(m_next);
    391 
    392       m_next++;
    393 
    394       return next;
    395     }
    396     else
    397       return DTM.NULL;
    398   }
    399 
    400   /**
    401    *  Returns the previous node in the set and moves the position of the
    402    * iterator backwards in the set.
    403    * @return  The previous <code>Node</code> in the set being iterated over,
    404    *   or<code>DTM.NULL</code> if there are no more members in that set.
    405    * @throws DOMException
    406    *    INVALID_STATE_ERR: Raised if this method is called after the
    407    *   <code>detach</code> method was invoked.
    408    * @throws RuntimeException thrown if this NodeSetDTM is not of
    409    * a cached type, and hence doesn't know what the previous node was.
    410    */
    411   public int previousNode()
    412   {
    413 
    414     if (!m_cacheNodes)
    415       throw new RuntimeException(
    416         XSLMessages.createXPATHMessage(XPATHErrorResources.ER_NODESETDTM_CANNOT_ITERATE, null)); //"This NodeSetDTM can not iterate to a previous node!");
    417 
    418     if ((m_next - 1) > 0)
    419     {
    420       m_next--;
    421 
    422       return this.elementAt(m_next);
    423     }
    424     else
    425       return DTM.NULL;
    426   }
    427 
    428   /**
    429    * Detaches the iterator from the set which it iterated over, releasing
    430    * any computational resources and placing the iterator in the INVALID
    431    * state. After<code>detach</code> has been invoked, calls to
    432    * <code>nextNode</code> or<code>previousNode</code> will raise the
    433    * exception INVALID_STATE_ERR.
    434    * <p>
    435    * This operation is a no-op in NodeSetDTM, and will not cause
    436    * INVALID_STATE_ERR to be raised by later operations.
    437    * </p>
    438    */
    439   public void detach(){}
    440 
    441   /**
    442    * Specify if it's OK for detach to release the iterator for reuse.
    443    *
    444    * @param allowRelease true if it is OK for detach to release this iterator
    445    * for pooling.
    446    */
    447   public void allowDetachToRelease(boolean allowRelease)
    448   {
    449     // no action for right now.
    450   }
    451 
    452 
    453   /**
    454    * Tells if this NodeSetDTM is "fresh", in other words, if
    455    * the first nextNode() that is called will return the
    456    * first node in the set.
    457    *
    458    * @return true if nextNode() would return the first node in the set,
    459    * false if it would return a later one.
    460    */
    461   public boolean isFresh()
    462   {
    463     return (m_next == 0);
    464   }
    465 
    466   /**
    467    * If an index is requested, NodeSetDTM will call this method
    468    * to run the iterator to the index.  By default this sets
    469    * m_next to the index.  If the index argument is -1, this
    470    * signals that the iterator should be run to the end.
    471    *
    472    * @param index Position to advance (or retreat) to, with
    473    * 0 requesting the reset ("fresh") position and -1 (or indeed
    474    * any out-of-bounds value) requesting the final position.
    475    * @throws RuntimeException thrown if this NodeSetDTM is not
    476    * one of the types which supports indexing/counting.
    477    */
    478   public void runTo(int index)
    479   {
    480 
    481     if (!m_cacheNodes)
    482       throw new RuntimeException(
    483         XSLMessages.createXPATHMessage(XPATHErrorResources.ER_NODESETDTM_CANNOT_INDEX, null)); //"This NodeSetDTM can not do indexing or counting functions!");
    484 
    485     if ((index >= 0) && (m_next < m_firstFree))
    486       m_next = index;
    487     else
    488       m_next = m_firstFree - 1;
    489   }
    490 
    491   /**
    492    * Returns the <code>index</code>th item in the collection. If
    493    * <code>index</code> is greater than or equal to the number of nodes in
    494    * the list, this returns <code>null</code>.
    495    *
    496    * TODO: What happens if index is out of range?
    497    *
    498    * @param index Index into the collection.
    499    * @return The node at the <code>index</code>th position in the
    500    *   <code>NodeList</code>, or <code>null</code> if that is not a valid
    501    *   index.
    502    */
    503   public int item(int index)
    504   {
    505 
    506     runTo(index);
    507 
    508     return this.elementAt(index);
    509   }
    510 
    511   /**
    512    * The number of nodes in the list. The range of valid child node indices is
    513    * 0 to <code>length-1</code> inclusive. Note that this operation requires
    514    * finding all the matching nodes, which may defeat attempts to defer
    515    * that work.
    516    *
    517    * @return integer indicating how many nodes are represented by this list.
    518    */
    519   public int getLength()
    520   {
    521 
    522     runTo(-1);
    523 
    524     return this.size();
    525   }
    526 
    527   /**
    528    * Add a node to the NodeSetDTM. Not all types of NodeSetDTMs support this
    529    * operation
    530    *
    531    * @param n Node to be added
    532    * @throws RuntimeException thrown if this NodeSetDTM is not of
    533    * a mutable type.
    534    */
    535   public void addNode(int n)
    536   {
    537 
    538     if (!m_mutable)
    539       throw new RuntimeException(XSLMessages.createXPATHMessage(XPATHErrorResources.ER_NODESETDTM_NOT_MUTABLE, null)); //"This NodeSetDTM is not mutable!");
    540 
    541     this.addElement(n);
    542   }
    543 
    544   /**
    545    * Insert a node at a given position.
    546    *
    547    * @param n Node to be added
    548    * @param pos Offset at which the node is to be inserted,
    549    * with 0 being the first position.
    550    * @throws RuntimeException thrown if this NodeSetDTM is not of
    551    * a mutable type.
    552    */
    553   public void insertNode(int n, int pos)
    554   {
    555 
    556     if (!m_mutable)
    557       throw new RuntimeException(XSLMessages.createXPATHMessage(XPATHErrorResources.ER_NODESETDTM_NOT_MUTABLE, null)); //"This NodeSetDTM is not mutable!");
    558 
    559     insertElementAt(n, pos);
    560   }
    561 
    562   /**
    563    * Remove a node.
    564    *
    565    * @param n Node to be added
    566    * @throws RuntimeException thrown if this NodeSetDTM is not of
    567    * a mutable type.
    568    */
    569   public void removeNode(int n)
    570   {
    571 
    572     if (!m_mutable)
    573       throw new RuntimeException(XSLMessages.createXPATHMessage(XPATHErrorResources.ER_NODESETDTM_NOT_MUTABLE, null)); //"This NodeSetDTM is not mutable!");
    574 
    575     this.removeElement(n);
    576   }
    577 
    578   // %TBD%
    579 //  /**
    580 //   * Copy NodeList members into this nodelist, adding in
    581 //   * document order.  If a node is null, don't add it.
    582 //   *
    583 //   * @param nodelist List of nodes which should now be referenced by
    584 //   * this NodeSetDTM.
    585 //   * @throws RuntimeException thrown if this NodeSetDTM is not of
    586 //   * a mutable type.
    587 //   */
    588 //  public void addNodes(NodeList nodelist)
    589 //  {
    590 //
    591 //    if (!m_mutable)
    592 //      throw new RuntimeException("This NodeSetDTM is not mutable!");
    593 //
    594 //    if (null != nodelist)  // defensive to fix a bug that Sanjiva reported.
    595 //    {
    596 //      int nChildren = nodelist.getLength();
    597 //
    598 //      for (int i = 0; i < nChildren; i++)
    599 //      {
    600 //        int obj = nodelist.item(i);
    601 //
    602 //        if (null != obj)
    603 //        {
    604 //          addElement(obj);
    605 //        }
    606 //      }
    607 //    }
    608 //
    609 //    // checkDups();
    610 //  }
    611 
    612   // %TBD%
    613 //  /**
    614 //   * <p>Copy NodeList members into this nodelist, adding in
    615 //   * document order.  Only genuine node references will be copied;
    616 //   * nulls appearing in the source NodeSetDTM will
    617 //   * not be added to this one. </p>
    618 //   *
    619 //   * <p> In case you're wondering why this function is needed: NodeSetDTM
    620 //   * implements both DTMIterator and NodeList. If this method isn't
    621 //   * provided, Java can't decide which of those to use when addNodes()
    622 //   * is invoked. Providing the more-explicit match avoids that
    623 //   * ambiguity.)</p>
    624 //   *
    625 //   * @param ns NodeSetDTM whose members should be merged into this NodeSetDTM.
    626 //   * @throws RuntimeException thrown if this NodeSetDTM is not of
    627 //   * a mutable type.
    628 //   */
    629 //  public void addNodes(NodeSetDTM ns)
    630 //  {
    631 //
    632 //    if (!m_mutable)
    633 //      throw new RuntimeException("This NodeSetDTM is not mutable!");
    634 //
    635 //    addNodes((DTMIterator) ns);
    636 //  }
    637 
    638   /**
    639    * Copy NodeList members into this nodelist, adding in
    640    * document order.  Null references are not added.
    641    *
    642    * @param iterator DTMIterator which yields the nodes to be added.
    643    * @throws RuntimeException thrown if this NodeSetDTM is not of
    644    * a mutable type.
    645    */
    646   public void addNodes(DTMIterator iterator)
    647   {
    648 
    649     if (!m_mutable)
    650       throw new RuntimeException(XSLMessages.createXPATHMessage(XPATHErrorResources.ER_NODESETDTM_NOT_MUTABLE, null)); //"This NodeSetDTM is not mutable!");
    651 
    652     if (null != iterator)  // defensive to fix a bug that Sanjiva reported.
    653     {
    654       int obj;
    655 
    656       while (DTM.NULL != (obj = iterator.nextNode()))
    657       {
    658         addElement(obj);
    659       }
    660     }
    661 
    662     // checkDups();
    663   }
    664 
    665   // %TBD%
    666 //  /**
    667 //   * Copy NodeList members into this nodelist, adding in
    668 //   * document order.  If a node is null, don't add it.
    669 //   *
    670 //   * @param nodelist List of nodes to be added
    671 //   * @param support The XPath runtime context.
    672 //   * @throws RuntimeException thrown if this NodeSetDTM is not of
    673 //   * a mutable type.
    674 //   */
    675 //  public void addNodesInDocOrder(NodeList nodelist, XPathContext support)
    676 //  {
    677 //
    678 //    if (!m_mutable)
    679 //      throw new RuntimeException("This NodeSetDTM is not mutable!");
    680 //
    681 //    int nChildren = nodelist.getLength();
    682 //
    683 //    for (int i = 0; i < nChildren; i++)
    684 //    {
    685 //      int node = nodelist.item(i);
    686 //
    687 //      if (null != node)
    688 //      {
    689 //        addNodeInDocOrder(node, support);
    690 //      }
    691 //    }
    692 //  }
    693 
    694   /**
    695    * Copy NodeList members into this nodelist, adding in
    696    * document order.  If a node is null, don't add it.
    697    *
    698    * @param iterator DTMIterator which yields the nodes to be added.
    699    * @param support The XPath runtime context.
    700    * @throws RuntimeException thrown if this NodeSetDTM is not of
    701    * a mutable type.
    702    */
    703   public void addNodesInDocOrder(DTMIterator iterator, XPathContext support)
    704   {
    705 
    706     if (!m_mutable)
    707       throw new RuntimeException(XSLMessages.createXPATHMessage(XPATHErrorResources.ER_NODESETDTM_NOT_MUTABLE, null)); //"This NodeSetDTM is not mutable!");
    708 
    709     int node;
    710 
    711     while (DTM.NULL != (node = iterator.nextNode()))
    712     {
    713       addNodeInDocOrder(node, support);
    714     }
    715   }
    716 
    717   // %TBD%
    718 //  /**
    719 //   * Add the node list to this node set in document order.
    720 //   *
    721 //   * @param start index.
    722 //   * @param end index.
    723 //   * @param testIndex index.
    724 //   * @param nodelist The nodelist to add.
    725 //   * @param support The XPath runtime context.
    726 //   *
    727 //   * @return false always.
    728 //   * @throws RuntimeException thrown if this NodeSetDTM is not of
    729 //   * a mutable type.
    730 //   */
    731 //  private boolean addNodesInDocOrder(int start, int end, int testIndex,
    732 //                                     NodeList nodelist, XPathContext support)
    733 //  {
    734 //
    735 //    if (!m_mutable)
    736 //      throw new RuntimeException("This NodeSetDTM is not mutable!");
    737 //
    738 //    boolean foundit = false;
    739 //    int i;
    740 //    int node = nodelist.item(testIndex);
    741 //
    742 //    for (i = end; i >= start; i--)
    743 //    {
    744 //      int child = elementAt(i);
    745 //
    746 //      if (child == node)
    747 //      {
    748 //        i = -2;  // Duplicate, suppress insert
    749 //
    750 //        break;
    751 //      }
    752 //
    753 //      if (!support.getDOMHelper().isNodeAfter(node, child))
    754 //      {
    755 //        insertElementAt(node, i + 1);
    756 //
    757 //        testIndex--;
    758 //
    759 //        if (testIndex > 0)
    760 //        {
    761 //          boolean foundPrev = addNodesInDocOrder(0, i, testIndex, nodelist,
    762 //                                                 support);
    763 //
    764 //          if (!foundPrev)
    765 //          {
    766 //            addNodesInDocOrder(i, size() - 1, testIndex, nodelist, support);
    767 //          }
    768 //        }
    769 //
    770 //        break;
    771 //      }
    772 //    }
    773 //
    774 //    if (i == -1)
    775 //    {
    776 //      insertElementAt(node, 0);
    777 //    }
    778 //
    779 //    return foundit;
    780 //  }
    781 
    782   /**
    783    * Add the node into a vector of nodes where it should occur in
    784    * document order.
    785    * @param node The node to be added.
    786    * @param test true if we should test for doc order
    787    * @param support The XPath runtime context.
    788    * @return insertIndex.
    789    * @throws RuntimeException thrown if this NodeSetDTM is not of
    790    * a mutable type.
    791    */
    792   public int addNodeInDocOrder(int node, boolean test, XPathContext support)
    793   {
    794 
    795     if (!m_mutable)
    796       throw new RuntimeException(XSLMessages.createXPATHMessage(XPATHErrorResources.ER_NODESETDTM_NOT_MUTABLE, null)); //"This NodeSetDTM is not mutable!");
    797 
    798     int insertIndex = -1;
    799 
    800     if (test)
    801     {
    802 
    803       // This needs to do a binary search, but a binary search
    804       // is somewhat tough because the sequence test involves
    805       // two nodes.
    806       int size = size(), i;
    807 
    808       for (i = size - 1; i >= 0; i--)
    809       {
    810         int child = elementAt(i);
    811 
    812         if (child == node)
    813         {
    814           i = -2;  // Duplicate, suppress insert
    815 
    816           break;
    817         }
    818 
    819         DTM dtm = support.getDTM(node);
    820         if (!dtm.isNodeAfter(node, child))
    821         {
    822           break;
    823         }
    824       }
    825 
    826       if (i != -2)
    827       {
    828         insertIndex = i + 1;
    829 
    830         insertElementAt(node, insertIndex);
    831       }
    832     }
    833     else
    834     {
    835       insertIndex = this.size();
    836 
    837       boolean foundit = false;
    838 
    839       for (int i = 0; i < insertIndex; i++)
    840       {
    841         if (i == node)
    842         {
    843           foundit = true;
    844 
    845           break;
    846         }
    847       }
    848 
    849       if (!foundit)
    850         addElement(node);
    851     }
    852 
    853     // checkDups();
    854     return insertIndex;
    855   }  // end addNodeInDocOrder(Vector v, Object obj)
    856 
    857   /**
    858    * Add the node into a vector of nodes where it should occur in
    859    * document order.
    860    * @param node The node to be added.
    861    * @param support The XPath runtime context.
    862    *
    863    * @return The index where it was inserted.
    864    * @throws RuntimeException thrown if this NodeSetDTM is not of
    865    * a mutable type.
    866    */
    867   public int addNodeInDocOrder(int node, XPathContext support)
    868   {
    869 
    870     if (!m_mutable)
    871       throw new RuntimeException(XSLMessages.createXPATHMessage(XPATHErrorResources.ER_NODESETDTM_NOT_MUTABLE, null)); //"This NodeSetDTM is not mutable!");
    872 
    873     return addNodeInDocOrder(node, true, support);
    874   }  // end addNodeInDocOrder(Vector v, Object obj)
    875 
    876   /**
    877    * Get the length of the list.
    878    *
    879    * @return The size of this node set.
    880    */
    881   public int size()
    882   {
    883     return super.size();
    884   }
    885 
    886   /**
    887    * Append a Node onto the vector.
    888    *
    889    * @param value The node to be added.
    890    * @throws RuntimeException thrown if this NodeSetDTM is not of
    891    * a mutable type.
    892    */
    893   public void addElement(int value)
    894   {
    895 
    896     if (!m_mutable)
    897       throw new RuntimeException(XSLMessages.createXPATHMessage(XPATHErrorResources.ER_NODESETDTM_NOT_MUTABLE, null)); //"This NodeSetDTM is not mutable!");
    898 
    899     super.addElement(value);
    900   }
    901 
    902   /**
    903    * Inserts the specified node in this vector at the specified index.
    904    * Each component in this vector with an index greater or equal to
    905    * the specified index is shifted upward to have an index one greater
    906    * than the value it had previously.
    907    *
    908    * @param value The node to be inserted.
    909    * @param at The index where the insert should occur.
    910    * @throws RuntimeException thrown if this NodeSetDTM is not of
    911    * a mutable type.
    912    */
    913   public void insertElementAt(int value, int at)
    914   {
    915 
    916     if (!m_mutable)
    917       throw new RuntimeException(XSLMessages.createXPATHMessage(XPATHErrorResources.ER_NODESETDTM_NOT_MUTABLE, null)); //"This NodeSetDTM is not mutable!");
    918 
    919     super.insertElementAt(value, at);
    920   }
    921 
    922   /**
    923    * Append the nodes to the list.
    924    *
    925    * @param nodes The nodes to be appended to this node set.
    926    * @throws RuntimeException thrown if this NodeSetDTM is not of
    927    * a mutable type.
    928    */
    929   public void appendNodes(NodeVector nodes)
    930   {
    931 
    932     if (!m_mutable)
    933       throw new RuntimeException(XSLMessages.createXPATHMessage(XPATHErrorResources.ER_NODESETDTM_NOT_MUTABLE, null)); //"This NodeSetDTM is not mutable!");
    934 
    935     super.appendNodes(nodes);
    936   }
    937 
    938   /**
    939    * Inserts the specified node in this vector at the specified index.
    940    * Each component in this vector with an index greater or equal to
    941    * the specified index is shifted upward to have an index one greater
    942    * than the value it had previously.
    943    * @throws RuntimeException thrown if this NodeSetDTM is not of
    944    * a mutable type.
    945    */
    946   public void removeAllElements()
    947   {
    948 
    949     if (!m_mutable)
    950       throw new RuntimeException(XSLMessages.createXPATHMessage(XPATHErrorResources.ER_NODESETDTM_NOT_MUTABLE, null)); //"This NodeSetDTM is not mutable!");
    951 
    952     super.removeAllElements();
    953   }
    954 
    955   /**
    956    * Removes the first occurrence of the argument from this vector.
    957    * If the object is found in this vector, each component in the vector
    958    * with an index greater or equal to the object's index is shifted
    959    * downward to have an index one smaller than the value it had
    960    * previously.
    961    *
    962    * @param s The node to be removed.
    963    *
    964    * @return True if the node was successfully removed
    965    * @throws RuntimeException thrown if this NodeSetDTM is not of
    966    * a mutable type.
    967    */
    968   public boolean removeElement(int s)
    969   {
    970 
    971     if (!m_mutable)
    972       throw new RuntimeException(XSLMessages.createXPATHMessage(XPATHErrorResources.ER_NODESETDTM_NOT_MUTABLE, null)); //"This NodeSetDTM is not mutable!");
    973 
    974     return super.removeElement(s);
    975   }
    976 
    977   /**
    978    * Deletes the component at the specified index. Each component in
    979    * this vector with an index greater or equal to the specified
    980    * index is shifted downward to have an index one smaller than
    981    * the value it had previously.
    982    *
    983    * @param i The index of the node to be removed.
    984    * @throws RuntimeException thrown if this NodeSetDTM is not of
    985    * a mutable type.
    986    */
    987   public void removeElementAt(int i)
    988   {
    989 
    990     if (!m_mutable)
    991       throw new RuntimeException(XSLMessages.createXPATHMessage(XPATHErrorResources.ER_NODESETDTM_NOT_MUTABLE, null)); //"This NodeSetDTM is not mutable!");
    992 
    993     super.removeElementAt(i);
    994   }
    995 
    996   /**
    997    * Sets the component at the specified index of this vector to be the
    998    * specified object. The previous component at that position is discarded.
    999    *
   1000    * The index must be a value greater than or equal to 0 and less
   1001    * than the current size of the vector.
   1002    *
   1003    * @param node  The node to be set.
   1004    * @param index The index of the node to be replaced.
   1005    * @throws RuntimeException thrown if this NodeSetDTM is not of
   1006    * a mutable type.
   1007    */
   1008   public void setElementAt(int node, int index)
   1009   {
   1010 
   1011     if (!m_mutable)
   1012       throw new RuntimeException(XSLMessages.createXPATHMessage(XPATHErrorResources.ER_NODESETDTM_NOT_MUTABLE, null)); //"This NodeSetDTM is not mutable!");
   1013 
   1014     super.setElementAt(node, index);
   1015   }
   1016 
   1017   /**
   1018    * Same as setElementAt.
   1019    *
   1020    * @param node  The node to be set.
   1021    * @param index The index of the node to be replaced.
   1022    * @throws RuntimeException thrown if this NodeSetDTM is not of
   1023    * a mutable type.
   1024    */
   1025   public void setItem(int node, int index)
   1026   {
   1027 
   1028     if (!m_mutable)
   1029       throw new RuntimeException(XSLMessages.createXPATHMessage(XPATHErrorResources.ER_NODESETDTM_NOT_MUTABLE, null)); //"This NodeSetDTM is not mutable!");
   1030 
   1031     super.setElementAt(node, index);
   1032   }
   1033 
   1034   /**
   1035    * Get the nth element.
   1036    *
   1037    * @param i The index of the requested node.
   1038    *
   1039    * @return Node at specified index.
   1040    */
   1041   public int elementAt(int i)
   1042   {
   1043 
   1044     runTo(i);
   1045 
   1046     return super.elementAt(i);
   1047   }
   1048 
   1049   /**
   1050    * Tell if the table contains the given node.
   1051    *
   1052    * @param s Node to look for
   1053    *
   1054    * @return True if the given node was found.
   1055    */
   1056   public boolean contains(int s)
   1057   {
   1058 
   1059     runTo(-1);
   1060 
   1061     return super.contains(s);
   1062   }
   1063 
   1064   /**
   1065    * Searches for the first occurence of the given argument,
   1066    * beginning the search at index, and testing for equality
   1067    * using the equals method.
   1068    *
   1069    * @param elem Node to look for
   1070    * @param index Index of where to start the search
   1071    * @return the index of the first occurrence of the object
   1072    * argument in this vector at position index or later in the
   1073    * vector; returns -1 if the object is not found.
   1074    */
   1075   public int indexOf(int elem, int index)
   1076   {
   1077 
   1078     runTo(-1);
   1079 
   1080     return super.indexOf(elem, index);
   1081   }
   1082 
   1083   /**
   1084    * Searches for the first occurence of the given argument,
   1085    * beginning the search at index, and testing for equality
   1086    * using the equals method.
   1087    *
   1088    * @param elem Node to look for
   1089    * @return the index of the first occurrence of the object
   1090    * argument in this vector at position index or later in the
   1091    * vector; returns -1 if the object is not found.
   1092    */
   1093   public int indexOf(int elem)
   1094   {
   1095 
   1096     runTo(-1);
   1097 
   1098     return super.indexOf(elem);
   1099   }
   1100 
   1101   /** If this node is being used as an iterator, the next index that nextNode()
   1102    *  will return.  */
   1103   transient protected int m_next = 0;
   1104 
   1105   /**
   1106    * Get the current position, which is one less than
   1107    * the next nextNode() call will retrieve.  i.e. if
   1108    * you call getCurrentPos() and the return is 0, the next
   1109    * fetch will take place at index 1.
   1110    *
   1111    * @return The the current position index.
   1112    */
   1113   public int getCurrentPos()
   1114   {
   1115     return m_next;
   1116   }
   1117 
   1118   /**
   1119    * Set the current position in the node set.
   1120    * @param i Must be a valid index.
   1121    * @throws RuntimeException thrown if this NodeSetDTM is not of
   1122    * a cached type, and thus doesn't permit indexed access.
   1123    */
   1124   public void setCurrentPos(int i)
   1125   {
   1126 
   1127     if (!m_cacheNodes)
   1128       throw new RuntimeException(
   1129         XSLMessages.createXPATHMessage(XPATHErrorResources.ER_NODESETDTM_CANNOT_INDEX, null)); //"This NodeSetDTM can not do indexing or counting functions!");
   1130 
   1131     m_next = i;
   1132   }
   1133 
   1134   /**
   1135    * Return the last fetched node.  Needed to support the UnionPathIterator.
   1136    *
   1137    * @return the last fetched node.
   1138    * @throws RuntimeException thrown if this NodeSetDTM is not of
   1139    * a cached type, and thus doesn't permit indexed access.
   1140    */
   1141   public int getCurrentNode()
   1142   {
   1143 
   1144     if (!m_cacheNodes)
   1145       throw new RuntimeException(
   1146         "This NodeSetDTM can not do indexing or counting functions!");
   1147 
   1148     int saved = m_next;
   1149     // because nextNode always increments
   1150     // But watch out for copy29, where the root iterator didn't
   1151     // have nextNode called on it.
   1152     int current = (m_next > 0) ? m_next-1 : m_next;
   1153     int n = (current < m_firstFree) ? elementAt(current) : DTM.NULL;
   1154     m_next = saved; // HACK: I think this is a bit of a hack.  -sb
   1155     return n;
   1156   }
   1157 
   1158   /** True if this list can be mutated.  */
   1159   transient protected boolean m_mutable = true;
   1160 
   1161   /** True if this list is cached.
   1162    *  @serial  */
   1163   transient protected boolean m_cacheNodes = true;
   1164 
   1165   /** The root of the iteration, if available. */
   1166   protected int m_root = DTM.NULL;
   1167 
   1168   /**
   1169    * Get whether or not this is a cached node set.
   1170    *
   1171    *
   1172    * @return True if this list is cached.
   1173    */
   1174   public boolean getShouldCacheNodes()
   1175   {
   1176     return m_cacheNodes;
   1177   }
   1178 
   1179   /**
   1180    * If setShouldCacheNodes(true) is called, then nodes will
   1181    * be cached.  They are not cached by default. This switch must
   1182    * be set before the first call to nextNode is made, to ensure
   1183    * that all nodes are cached.
   1184    *
   1185    * @param b true if this node set should be cached.
   1186    * @throws RuntimeException thrown if an attempt is made to
   1187    * request caching after we've already begun stepping through the
   1188    * nodes in this set.
   1189   */
   1190   public void setShouldCacheNodes(boolean b)
   1191   {
   1192 
   1193     if (!isFresh())
   1194       throw new RuntimeException(
   1195         XSLMessages.createXPATHMessage(XPATHErrorResources.ER_CANNOT_CALL_SETSHOULDCACHENODE, null)); //"Can not call setShouldCacheNodes after nextNode has been called!");
   1196 
   1197     m_cacheNodes = b;
   1198     m_mutable = true;
   1199   }
   1200 
   1201   /**
   1202    * Tells if this iterator can have nodes added to it or set via
   1203    * the <code>setItem(int node, int index)</code> method.
   1204    *
   1205    * @return True if the nodelist can be mutated.
   1206    */
   1207   public boolean isMutable()
   1208   {
   1209     return m_mutable;
   1210   }
   1211 
   1212   transient private int m_last = 0;
   1213 
   1214   public int getLast()
   1215   {
   1216     return m_last;
   1217   }
   1218 
   1219   public void setLast(int last)
   1220   {
   1221     m_last = last;
   1222   }
   1223 
   1224   /**
   1225    * Returns true if all the nodes in the iteration well be returned in document
   1226    * order.
   1227    *
   1228    * @return true as a default.
   1229    */
   1230   public boolean isDocOrdered()
   1231   {
   1232     return true;
   1233   }
   1234 
   1235   /**
   1236    * Returns the axis being iterated, if it is known.
   1237    *
   1238    * @return Axis.CHILD, etc., or -1 if the axis is not known or is of multiple
   1239    * types.
   1240    */
   1241   public int getAxis()
   1242   {
   1243     return -1;
   1244   }
   1245 
   1246 
   1247 }
   1248