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    */
NodeSetDTM(DTMManager dtmManager)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    */
NodeSetDTM(int blocksize, int dummy, DTMManager dtmManager)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    */
NodeSetDTM(NodeSetDTM nodelist)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    */
NodeSetDTM(DTMIterator ni)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    */
NodeSetDTM(NodeIterator iterator, XPathContext xctxt)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    */
NodeSetDTM(NodeList nodeList, XPathContext xctxt)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    */
NodeSetDTM(int node, DTMManager dtmManager)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    */
setEnvironment(Object environment)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    */
getRoot()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    */
setRoot(int context, Object environment)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    */
clone()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    */
cloneWithReset()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    */
reset()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    */
getWhatToShow()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    */
getFilter()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    */
getExpandEntityReferences()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    */
getDTM(int nodeHandle)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    */
getDTMManager()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    */
nextNode()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    */
previousNode()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    */
detach()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    */
allowDetachToRelease(boolean allowRelease)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    */
isFresh()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    */
runTo(int index)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    */
item(int index)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    */
getLength()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    */
addNode(int n)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    */
insertNode(int n, int pos)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    */
removeNode(int n)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    */
addNodes(DTMIterator iterator)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    */
addNodesInDocOrder(DTMIterator iterator, XPathContext support)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    */
addNodeInDocOrder(int node, boolean test, XPathContext support)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    */
addNodeInDocOrder(int node, XPathContext support)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    */
size()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    */
addElement(int value)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    */
insertElementAt(int value, int at)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    */
appendNodes(NodeVector nodes)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    */
removeAllElements()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    */
removeElement(int s)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    */
removeElementAt(int i)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    */
setElementAt(int node, int index)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    */
setItem(int node, int index)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    */
elementAt(int i)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    */
contains(int s)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    */
indexOf(int elem, int index)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    */
indexOf(int elem)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    */
getCurrentPos()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    */
setCurrentPos(int i)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    */
getCurrentNode()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    */
getShouldCacheNodes()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   */
setShouldCacheNodes(boolean b)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    */
isMutable()1207   public boolean isMutable()
1208   {
1209     return m_mutable;
1210   }
1211 
1212   transient private int m_last = 0;
1213 
getLast()1214   public int getLast()
1215   {
1216     return m_last;
1217   }
1218 
setLast(int last)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    */
isDocOrdered()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    */
getAxis()1241   public int getAxis()
1242   {
1243     return -1;
1244   }
1245 
1246 
1247 }
1248