IDDeepNodeListImpl.cpp
上传用户:huihehuasu
上传日期:2007-01-10
资源大小:6948k
文件大小:9k
源码类别:

xml/soap/webservice

开发平台:

C/C++

  1. /*
  2.  * The Apache Software License, Version 1.1
  3.  *
  4.  * Copyright (c) 2001 The Apache Software Foundation.  All rights
  5.  * reserved.
  6.  *
  7.  * Redistribution and use in source and binary forms, with or without
  8.  * modification, are permitted provided that the following conditions
  9.  * are met:
  10.  *
  11.  * 1. Redistributions of source code must retain the above copyright
  12.  *    notice, this list of conditions and the following disclaimer.
  13.  *
  14.  * 2. Redistributions in binary form must reproduce the above copyright
  15.  *    notice, this list of conditions and the following disclaimer in
  16.  *    the documentation and/or other materials provided with the
  17.  *    distribution.
  18.  *
  19.  * 3. The end-user documentation included with the redistribution,
  20.  *    if any, must include the following acknowledgment:
  21.  *       "This product includes software developed by the
  22.  *        Apache Software Foundation (http://www.apache.org/)."
  23.  *    Alternately, this acknowledgment may appear in the software itself,
  24.  *    if and wherever such third-party acknowledgments normally appear.
  25.  *
  26.  * 4. The names "Xerces" and "Apache Software Foundation" must
  27.  *    not be used to endorse or promote products derived from this
  28.  *    software without prior written permission. For written
  29.  *    permission, please contact apache@apache.org.
  30.  *
  31.  * 5. Products derived from this software may not be called "Apache",
  32.  *    nor may "Apache" appear in their name, without prior written
  33.  *    permission of the Apache Software Foundation.
  34.  *
  35.  * THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED
  36.  * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
  37.  * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
  38.  * DISCLAIMED.  IN NO EVENT SHALL THE APACHE SOFTWARE FOUNDATION OR
  39.  * ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
  40.  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
  41.  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF
  42.  * USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
  43.  * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
  44.  * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
  45.  * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
  46.  * SUCH DAMAGE.
  47.  * ====================================================================
  48.  *
  49.  * This software consists of voluntary contributions made by many
  50.  * individuals on behalf of the Apache Software Foundation, and was
  51.  * originally based on software copyright (c) 2001, International
  52.  * Business Machines, Inc., http://www.ibm.com .  For more information
  53.  * on the Apache Software Foundation, please see
  54.  * <http://www.apache.org/>.
  55.  */
  56. /*
  57.  * $Log: IDDeepNodeListImpl.cpp,v $
  58.  * Revision 1.8  2001/11/28 20:06:52  tng
  59.  * [Bug 4544] DOM_NodeList::getLength incorrect when called twice for empty list .
  60.  *
  61.  * Revision 1.7  2001/10/03 15:49:01  tng
  62.  * [Bug 3867] IDOM_Element::getElementsByTagName() threading problem.
  63.  *
  64.  * Revision 1.6  2001/08/09 16:52:59  tng
  65.  * [Bug 2947]  IDOM segfault calling getElementsByTagName() using a DOM_Document().
  66.  *
  67.  * Revision 1.5  2001/08/07 17:01:09  tng
  68.  * [Bug 2676] IDOM: pure virtual called in IDDeepNodeListImpl::item() .
  69.  *
  70.  * Revision 1.4  2001/06/04 20:11:52  tng
  71.  * IDOM: Complete IDNodeIterator, IDTreeWalker, IDNodeFilter.
  72.  *
  73.  * Revision 1.3  2001/06/04 14:55:32  tng
  74.  * IDOM: Add IRange and IDeepNodeList Support.
  75.  *
  76.  * Revision 1.2  2001/05/11 13:25:40  tng
  77.  * Copyright update.
  78.  *
  79.  * Revision 1.1.1.1  2001/04/03 00:14:20  andyh
  80.  * IDOM
  81.  *
  82.  */
  83. #include "IDDeepNodeListImpl.hpp"
  84. #include "IDElementImpl.hpp"
  85. #include "IDDocumentImpl.hpp"
  86. #include "IDCasts.hpp"
  87. #include "IDNodeImpl.hpp"
  88. #include <util/XMLUniDefs.hpp>
  89. #include <limits.h>
  90. static const XMLCh kAstr[] = {chAsterisk, chNull};
  91. IDDeepNodeListImpl::IDDeepNodeListImpl(const IDOM_Node *rootNode,
  92.                                        const XMLCh *tagName)
  93.     : fRootNode(rootNode)
  94.     , fChanges(0)
  95.     , fCurrentNode(0)
  96.     , fCurrentIndexPlus1(0)
  97.     , fNamespaceURI(0)
  98.     , fMatchAllURI(false)
  99.     , fMatchURIandTagname(false)
  100. {
  101.     fTagName = ((IDDocumentImpl *)(castToNodeImpl(rootNode)->getOwnerDocument()))->getPooledString(tagName);
  102.     fMatchAll = (XMLString::compareString(fTagName, kAstr) == 0);
  103. }
  104. //DOM Level 2
  105. IDDeepNodeListImpl::IDDeepNodeListImpl(const IDOM_Node *rootNode,
  106.                                        const XMLCh *namespaceURI,
  107.                                        const XMLCh *localName)
  108.     : fRootNode(rootNode)
  109.     , fChanges(0)
  110.     , fCurrentNode(0)
  111.     , fCurrentIndexPlus1(0)
  112.     , fMatchAllURI(false)
  113.     , fMatchURIandTagname(true)
  114. {
  115.     fTagName = ((IDDocumentImpl *)(castToNodeImpl(rootNode)->getOwnerDocument()))->getPooledString(localName);
  116.     fMatchAll = (XMLString::compareString(fTagName, kAstr) == 0);
  117.     fMatchAllURI = (XMLString::compareString(namespaceURI, kAstr) == 0);
  118.     fNamespaceURI = ((IDDocumentImpl *)(castToNodeImpl(rootNode)->getOwnerDocument()))->getPooledString(namespaceURI);
  119. }
  120. IDDeepNodeListImpl::~IDDeepNodeListImpl()
  121. {
  122. }
  123. unsigned int IDDeepNodeListImpl::getLength()
  124. {
  125.     // Reset cache to beginning of list
  126.     item(0);
  127.     // Preload all matching elements. (Stops when we run out of subtree!)
  128.     item(INT_MAX);
  129.     return fCurrentIndexPlus1;
  130. }
  131. // Start from the first child and count forward, 0-based. index>length-1
  132. // should return 0.
  133. //
  134. // Attempts to do only work actually requested, cache work already
  135. // done, and to flush that cache when the tree has changed.
  136. //
  137. // LIMITATION: ????? Unable to tell relevant tree-changes from
  138. // irrelevant ones.  Doing so in a really useful manner would seem
  139. // to involve a tree-walk in its own right, or maintaining our data
  140. // in a parallel tree.
  141. IDOM_Node *IDDeepNodeListImpl::item(unsigned int index)
  142. {
  143.     unsigned int currentIndexPlus1 = fCurrentIndexPlus1;
  144.     IDOM_Node *currentNode = fCurrentNode;
  145.     if (castToParentImpl(fRootNode)->changes() != fChanges)
  146.     {
  147.         // Tree changed. Do it all from scratch!
  148.         currentIndexPlus1 = 0;
  149.         currentNode = (IDOM_Node *)fRootNode;
  150.         fChanges = castToParentImpl(fRootNode)->changes();
  151.     }
  152.     else if (currentIndexPlus1 > index+1)
  153.     {
  154.         // Interested in something before cached node.  Do it all from scratch!
  155.         currentIndexPlus1 = 0;
  156.         currentNode = (IDOM_Node *)fRootNode;
  157.     }
  158.     else if (index+1 == currentIndexPlus1)
  159.     {
  160.         // What luck!  User is interested in cached node.
  161.         return currentNode;
  162.     }
  163.     IDOM_Node *nextNode = 0;
  164. // idom_revisit - ???? How efficient is this loop? ????
  165.     // Start at the place in the tree at which we're
  166.     // currently pointing and count off nodes until we
  167.     // reach the node of interest or the end of the tree.
  168.     while (currentIndexPlus1 < index+1 && currentNode != 0)
  169.     {
  170.         nextNode = nextMatchingElementAfter(currentNode);
  171.         if (nextNode == 0)
  172.             break;
  173.         currentNode = nextNode;
  174.         currentIndexPlus1++;
  175.     }
  176.     fCurrentNode = currentNode;
  177.     fCurrentIndexPlus1 = currentIndexPlus1;
  178.     // If we found a node at the requested index, make that the current node
  179.     if (nextNode != 0)
  180.     {
  181.         return currentNode;
  182.     }
  183.     // If we didn't find a node at the requested index, return 0
  184.     return 0;
  185. }
  186. /* Iterative tree-walker. When you have a Parent link, there's often no
  187. need to resort to recursion. NOTE THAT only Element nodes are matched
  188. since we're specifically supporting getElementsByTagName().
  189. */
  190. IDOM_Node *IDDeepNodeListImpl::nextMatchingElementAfter(IDOM_Node *current)
  191. {
  192.     IDOM_Node *next;
  193.     while (current != 0)
  194.     {
  195.         // Look down to first child.
  196.         if (current->hasChildNodes())
  197.         {
  198.             current = current->getFirstChild();
  199.         }
  200.         // Look right to sibling (but not from root!)
  201.         else
  202.         {
  203.             if (current != fRootNode && 0 != (next = current->getNextSibling()))
  204.             {
  205.                 current = next;
  206.             }
  207.             // Look up and right (but not past root!)
  208.             else
  209.             {
  210.                 next = 0;
  211.                 for (;
  212.                      current != fRootNode; // Stop on return to starting point
  213.                      current = current->getParentNode())
  214.                 {
  215.                     next = current->getNextSibling();
  216.                     if (next != 0)
  217.                         break;
  218.                 }
  219.                 current = next;
  220.             }
  221.         }
  222.         // Have we found an Element with the right tagName?
  223.         // ("*" matches anything.)
  224.         if (current != 0 && current != fRootNode &&
  225.             current->getNodeType() == IDOM_Node::ELEMENT_NODE) {
  226.             IDOM_Element *currElement = (IDOM_Element *)current;
  227.             if (!fMatchURIandTagname) {        //DOM Level 1
  228.                 if (fMatchAll ||
  229.                     (XMLString::compareString(currElement->getTagName(),
  230.                                               fTagName) == 0))
  231.                     return current;
  232.             } else {        //DOM Level 2
  233.                 if (!fMatchAllURI &&
  234.                     (XMLString::compareString(current -> getNamespaceURI(),
  235.                                               fNamespaceURI) != 0))
  236.                     continue;
  237.                 if (fMatchAll ||
  238.                     (XMLString::compareString(current -> getLocalName(),
  239.                                               fTagName) == 0))
  240.                     return current;
  241.             }
  242.         }
  243.         // Otherwise continue walking the tree
  244.     }
  245.     // Fell out of tree-walk; no more instances found
  246.     return 0;
  247. }