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

xml/soap/webservice

开发平台:

C/C++

  1. /*
  2.  * The Apache Software License, Version 1.1
  3.  *
  4.  * Copyright (c) 1999-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) 1999, 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.  * $Id: ElementImpl.cpp,v 1.33 2001/12/07 01:35:28 tng Exp $
  58.  */
  59. #include "DeepNodeListImpl.hpp"
  60. #include "DocumentImpl.hpp"
  61. #include "DocumentTypeImpl.hpp"
  62. #include "DOM_DOMException.hpp"
  63. #include "DStringPool.hpp"
  64. #include "ElementImpl.hpp"
  65. #include "ElementDefinitionImpl.hpp"
  66. #include "NamedNodeMapImpl.hpp"
  67. #include "NodeVector.hpp"
  68. static DOMString *gEmptyString = 0;
  69. static XMLRegisterCleanup emptyStringCleanup;
  70. ElementImpl::ElementImpl(DocumentImpl *ownerDoc, const DOMString &eName)
  71.     : ParentNode(ownerDoc)
  72. {
  73.     name = eName.clone();
  74.     attributes = null;
  75. setupDefaultAttributes();
  76. };
  77. ElementImpl::ElementImpl(const ElementImpl &other, bool deep)
  78.     : ParentNode(other)
  79. {
  80.     name = other.name.clone();
  81. attributes = null;
  82. setupDefaultAttributes();
  83.     if (deep)
  84.         cloneChildren(other);
  85. if (other.attributes != null)
  86. attributes = other.attributes->cloneAttrMap(this);
  87. };
  88. ElementImpl::~ElementImpl()
  89. {
  90.     if (attributes)
  91.     {
  92.         attributes->removeAll();
  93.         NamedNodeMapImpl::removeRef(attributes);
  94.     }
  95. };
  96. NodeImpl *ElementImpl::cloneNode(bool deep)
  97. {
  98.     return new ElementImpl(*this, deep);
  99. };
  100. /**
  101.  * NON-DOM
  102.  * set the ownerDocument of this node, its children, and its attributes
  103.  */
  104. void ElementImpl::setOwnerDocument(DocumentImpl *doc) {
  105.     ParentNode::setOwnerDocument(doc);
  106. if (attributes != null)
  107. attributes->setOwnerDocument(doc);
  108. }
  109. DOMString ElementImpl::getNodeName() {
  110.     return name;
  111. };
  112. short ElementImpl::getNodeType() {
  113.     return DOM_Node::ELEMENT_NODE;
  114. };
  115. DOMString ElementImpl::getAttribute(const DOMString &nam)
  116. {
  117.     AttrImpl * attr=null;
  118.     if (attributes != null)
  119. attr=(AttrImpl *)(attributes->getNamedItem(nam));
  120.     return (attr==null) ? DStringPool::getStaticString(""
  121.                                                      , &gEmptyString
  122.                                                      , reinitElementImpl
  123.                                                      , emptyStringCleanup) : attr->getValue();
  124. };
  125. AttrImpl *ElementImpl::getAttributeNode(const DOMString &nam)
  126. {
  127.     return (attributes == 0) ? null : (AttrImpl *)(attributes->getNamedItem(nam));
  128. };
  129. NamedNodeMapImpl *ElementImpl::getAttributes()
  130. {
  131.     return attributes;
  132. };
  133. DeepNodeListImpl *ElementImpl::getElementsByTagName(const DOMString &tagname)
  134. {
  135.     return new DeepNodeListImpl(this,tagname);
  136. };
  137. DOMString ElementImpl::getTagName()
  138. {
  139.     return name;
  140. }
  141. bool ElementImpl::isElementImpl()
  142. {
  143.     return true;
  144. };
  145. void ElementImpl::removeAttribute(const DOMString &nam)
  146. {
  147.     if (getOwnerDocument()->getErrorChecking() && isReadOnly()) {
  148.         throw DOM_DOMException(DOM_DOMException::NO_MODIFICATION_ALLOWED_ERR,
  149.                                null);
  150.     }
  151.     if (attributes != null)
  152.     {
  153.      AttrImpl *att = (AttrImpl *) attributes->getNamedItem(nam);
  154.      // Remove it
  155.      if (att != null)
  156.      {
  157.          attributes->removeNamedItem(nam);
  158.          if (att->nodeRefCount == 0)
  159.              NodeImpl::deleteIf(att);
  160.      }
  161.     }
  162. };
  163. AttrImpl *ElementImpl::removeAttributeNode(AttrImpl *oldAttr)
  164. {
  165.     if (getOwnerDocument()->getErrorChecking() && isReadOnly()) {
  166.         throw DOM_DOMException(DOM_DOMException::NO_MODIFICATION_ALLOWED_ERR,
  167.                                null);
  168.     }
  169.     if (attributes != null)
  170.     {
  171.     AttrImpl *found = (AttrImpl *) attributes->getNamedItem(oldAttr->getName());
  172.     // If it is in fact the right object, remove it.
  173.     if (found == oldAttr)
  174.         attributes->removeNamedItem(oldAttr->getName());
  175.     else
  176.         throw DOM_DOMException(DOM_DOMException::NOT_FOUND_ERR, null);
  177.         return found;
  178. }
  179. return null; // just to keep the compiler happy
  180. };
  181. AttrImpl *ElementImpl::setAttribute(const DOMString &nam, const DOMString &val)
  182. {
  183.     if (getOwnerDocument()->getErrorChecking() && isReadOnly()) {
  184.         throw DOM_DOMException(DOM_DOMException::NO_MODIFICATION_ALLOWED_ERR,
  185.                                null);
  186.     }
  187.     AttrImpl* newAttr = (AttrImpl*)getAttributeNode(nam);
  188.     if (!newAttr)
  189.     {
  190. if (attributes == 0)
  191. attributes = new AttrMapImpl(this, null);
  192.         newAttr = (AttrImpl*)ownerDocument->createAttribute(nam);
  193.         attributes->setNamedItem(newAttr);
  194.     }
  195.     newAttr->setNodeValue(val);       // Note that setNodeValue on attribute
  196.                                       //   nodes takes care of deleting
  197.                                       //   any previously existing children.
  198.     return newAttr;
  199. };
  200. AttrImpl * ElementImpl::setAttributeNode(AttrImpl *newAttr)
  201. {
  202.     if (getOwnerDocument()->getErrorChecking() && isReadOnly()) {
  203.         throw DOM_DOMException(DOM_DOMException::NO_MODIFICATION_ALLOWED_ERR,
  204.                                null);
  205.     }
  206.     if (!(newAttr->isAttrImpl()))
  207.         throw DOM_DOMException(DOM_DOMException::WRONG_DOCUMENT_ERR, null);
  208. if (attributes == 0)
  209. attributes = new AttrMapImpl(this, null);
  210.     AttrImpl *oldAttr =
  211.       (AttrImpl *) attributes->getNamedItem(newAttr->getName());
  212.     // This will throw INUSE if necessary
  213.     attributes->setNamedItem(newAttr);
  214.     // Attr node reference counting note:
  215.     // If oldAttr's refcount is zero at this point, here's what happens...
  216.     //
  217.     //      oldAttr is returned from this function to DOM_Attr::setAttributeNode,
  218.     //      which wraps a DOM_Attr around the returned pointer and sends it
  219.     //      up to application code, incrementing the reference count in the process.
  220.     //      When the app DOM_Attr's destructor runs, the reference count is
  221.     //      decremented back to zero and oldAttr will be deleted at that time.
  222.     return oldAttr;
  223. };
  224. void ElementImpl::setReadOnly(bool readOnl, bool deep)
  225. {
  226.     ParentNode::setReadOnly(readOnl,deep);
  227.     attributes->setReadOnly(readOnl,true);
  228. };
  229. //Introduced in DOM Level 2
  230. DOMString ElementImpl::getAttributeNS(const DOMString &fNamespaceURI,
  231. const DOMString &fLocalName)
  232. {
  233.     AttrImpl * attr= (attributes != null) ?
  234.       (AttrImpl *)(attributes->getNamedItemNS(fNamespaceURI, fLocalName))
  235.     : null;
  236.     return (attr==null) ? DOMString(null) : attr->getValue();
  237. }
  238. AttrImpl *ElementImpl::setAttributeNS(const DOMString &fNamespaceURI,
  239. const DOMString &qualifiedName, const DOMString &fValue)
  240. {
  241.     if (getOwnerDocument()->getErrorChecking() && isReadOnly()) {
  242.         throw DOM_DOMException(DOM_DOMException::NO_MODIFICATION_ALLOWED_ERR,
  243.                                null);
  244.     }
  245.     AttrImpl *newAttr =
  246.       (AttrImpl *) ownerDocument->createAttributeNS(fNamespaceURI,
  247.                                                     qualifiedName);
  248.     newAttr->setNodeValue(fValue);
  249. if (attributes == 0)
  250. attributes = new AttrMapImpl(this, null);
  251.     AttrImpl *oldAttr = (AttrImpl *)attributes->setNamedItem(newAttr);
  252.     if (oldAttr) {
  253. if (oldAttr->nodeRefCount == 0)
  254.     NodeImpl::deleteIf(oldAttr);
  255.     }
  256.     return newAttr;
  257. }
  258. void ElementImpl::removeAttributeNS(const DOMString &fNamespaceURI,
  259. const DOMString &fLocalName)
  260. {
  261.     if (getOwnerDocument()->getErrorChecking() && isReadOnly()) {
  262.         throw DOM_DOMException(DOM_DOMException::NO_MODIFICATION_ALLOWED_ERR,
  263.                                null);
  264.     }
  265.     if (attributes != null)
  266.     {
  267. AttrImpl *att =
  268.   (AttrImpl *) attributes->getNamedItemNS(fNamespaceURI, fLocalName);
  269. // Remove it
  270. if (att != null) {
  271. attributes->removeNamedItemNS(fNamespaceURI, fLocalName);
  272. if (att->nodeRefCount == 0)
  273. NodeImpl::deleteIf(att);
  274. }
  275. }
  276. }
  277. AttrImpl *ElementImpl::getAttributeNodeNS(const DOMString &fNamespaceURI,
  278. const DOMString &fLocalName)
  279. {
  280.     return (attributes == 0) ? null : (AttrImpl *)(attributes->getNamedItemNS(fNamespaceURI, fLocalName));
  281. }
  282. AttrImpl *ElementImpl::setAttributeNodeNS(AttrImpl *newAttr)
  283. {
  284.     if (getOwnerDocument()->getErrorChecking()) {
  285.         if (isReadOnly()) {
  286.             throw DOM_DOMException(
  287.                                  DOM_DOMException::NO_MODIFICATION_ALLOWED_ERR,
  288.                                  null);
  289.         }
  290.         if (newAttr->getOwnerDocument() != this->getOwnerDocument()) {
  291.             throw DOM_DOMException(DOM_DOMException::WRONG_DOCUMENT_ERR, null);
  292.         }
  293.     }
  294.     if (attributes == 0) {
  295.         attributes = new AttrMapImpl(this, null);
  296.     }
  297.     AttrImpl *oldAttr = (AttrImpl *) attributes->getNamedItemNS(newAttr->getNamespaceURI(), newAttr->getLocalName());
  298.     // This will throw INUSE if necessary
  299.     attributes->setNamedItemNS(newAttr);
  300.     // Attr node reference counting note:
  301.     // If oldAttr's refcount is zero at this point, here's what happens...
  302.     //
  303.     //      oldAttr is returned from this function to DOM_Attr::setAttributeNode,
  304.     //      which wraps a DOM_Attr around the returned pointer and sends it
  305.     //      up to application code, incrementing the reference count in the process.
  306.     //      When the app DOM_Attr's destructor runs, the reference count is
  307.     //      decremented back to zero and oldAttr will be deleted at that time.
  308.     return oldAttr;
  309. }
  310. DeepNodeListImpl *ElementImpl::getElementsByTagNameNS(const DOMString &fNamespaceURI,
  311. const DOMString &fLocalName)
  312. {
  313.     return new DeepNodeListImpl(this,fNamespaceURI, fLocalName);
  314. }
  315. // DOM_NamedNodeMap UTILITIES
  316. NamedNodeMapImpl *ElementImpl::NNM_cloneMap(NodeImpl *nnm_ownerNode)
  317. {
  318. return (getAttributes() == null) ? null : nnm_ownerNode->getAttributes()->cloneMap(nnm_ownerNode);
  319. }
  320. int ElementImpl::NNM_findNamePoint(const DOMString &nnm_name)
  321. {
  322. return (getAttributes() == null) ? -1 : getAttributes()->findNamePoint(nnm_name);
  323. }
  324. unsigned int ElementImpl::NNM_getLength()
  325. {
  326. return (getAttributes() == null) ? 0 : getAttributes()->getLength();
  327. }
  328. NodeImpl *ElementImpl::NNM_getNamedItem(const DOMString &nnm_name)
  329. {
  330. return (getAttributes() == null) ? null : getAttributes()->getNamedItem(nnm_name);
  331. }
  332. NodeImpl *ElementImpl::NNM_item(unsigned int nnm_index)
  333. {
  334. return (getAttributes() == null) ? null : getAttributes()->item(nnm_index);
  335. }
  336. void ElementImpl::NNM_removeAll()
  337. {
  338. if (getAttributes() != null)
  339. getAttributes()->removeAll();
  340. }
  341. NodeImpl *ElementImpl::NNM_removeNamedItem(const DOMString &nnm_name)
  342. {
  343. if (getAttributes() == null)
  344. throw DOM_DOMException(DOM_DOMException::NOT_FOUND_ERR, null);
  345. else
  346. return getAttributes()->removeNamedItem(nnm_name);
  347. return null;
  348. }
  349. NodeImpl *ElementImpl::NNM_setNamedItem(NodeImpl *nnm_arg)
  350. {
  351. if (getAttributes() == null)
  352. attributes = new AttrMapImpl(this);
  353. return attributes->setNamedItem(nnm_arg);
  354. }
  355. void ElementImpl::NNM_setReadOnly(bool nnm_readOnly, bool nnm_deep)
  356. {
  357. if (getAttributes() != null)
  358. getAttributes()->setReadOnly(nnm_readOnly, nnm_deep);
  359. }
  360. int ElementImpl::NNM_findNamePoint(const DOMString &nnm_namespaceURI, const DOMString &nnm_localName)
  361. {
  362. return (getAttributes() == null) ? -1 : getAttributes()->findNamePoint(nnm_namespaceURI, nnm_localName);
  363. }
  364. NodeImpl *ElementImpl::NNM_getNamedItemNS(const DOMString &nnm_namespaceURI, const DOMString &nnm_localName)
  365. {
  366. return (getAttributes() == null) ? null : getAttributes()->getNamedItemNS(nnm_namespaceURI, nnm_localName);
  367. }
  368. NodeImpl *ElementImpl::NNM_setNamedItemNS(NodeImpl *nnm_arg)
  369. {
  370. if (getAttributes() == null)
  371. attributes = new AttrMapImpl(this);
  372. return getAttributes()->setNamedItemNS(nnm_arg);
  373. }
  374. NodeImpl *ElementImpl::NNM_removeNamedItemNS(const DOMString &nnm_namespaceURI, const DOMString &nnm_localName)
  375. {
  376. if (getAttributes() == null)
  377.         throw DOM_DOMException(DOM_DOMException::NOT_FOUND_ERR, null);
  378. else
  379. return getAttributes()->removeNamedItemNS(nnm_namespaceURI, nnm_localName);
  380. return null;
  381. }
  382. void ElementImpl::NNM_setOwnerDocument(DocumentImpl *nnm_doc)
  383. {
  384. if (getAttributes() != null)
  385. getAttributes()->setOwnerDocument(nnm_doc);
  386. }
  387. // util functions for default attributes
  388. // returns the default attribute map for this node from the owner document
  389. AttrMapImpl *ElementImpl::getDefaultAttributes()
  390. {
  391. if ((ownerNode == null) || (getOwnerDocument() == null))
  392. return null;
  393. DocumentImpl *tmpdoc = getOwnerDocument();
  394. if (tmpdoc->getDoctype() == null)
  395. return null;
  396. NodeImpl *eldef = tmpdoc->getDoctype()->getElements()->getNamedItem(getNodeName());
  397. return (eldef == null) ? null : (AttrMapImpl *)(eldef->getAttributes());
  398. }
  399. // resets all attributes for this node to their default values
  400. void ElementImpl::setupDefaultAttributes()
  401. {
  402. if ((ownerNode == null) || (getOwnerDocument() == null) || (getOwnerDocument()->getDoctype() == null))
  403. return;
  404. if (attributes != 0)
  405. delete attributes;
  406. AttrMapImpl* defAttrs = getDefaultAttributes();
  407. if (defAttrs)
  408. attributes = new AttrMapImpl(this, defAttrs);
  409. }
  410. // -----------------------------------------------------------------------
  411. //  Notification that lazy data has been deleted
  412. // -----------------------------------------------------------------------
  413. void ElementImpl::reinitElementImpl() {
  414.     delete gEmptyString;
  415.     gEmptyString = 0;
  416. }