diff options
Diffstat (limited to 'khtml/xml/dom_elementimpl.cpp')
-rw-r--r-- | khtml/xml/dom_elementimpl.cpp | 1301 |
1 files changed, 0 insertions, 1301 deletions
diff --git a/khtml/xml/dom_elementimpl.cpp b/khtml/xml/dom_elementimpl.cpp deleted file mode 100644 index 875c5f2e4..000000000 --- a/khtml/xml/dom_elementimpl.cpp +++ /dev/null @@ -1,1301 +0,0 @@ -/** - * This file is part of the DOM implementation for KDE. - * - * Copyright (C) 1999 Lars Knoll (knoll@kde.org) - * (C) 1999 Antti Koivisto (koivisto@kde.org) - * (C) 2001 Peter Kelly (pmk@post.com) - * (C) 2001 Dirk Mueller (mueller@kde.org) - * (C) 2006 Allan Sandfeld Jensen (kde@carewolf.com) - * - * This library is free software; you can redistribute it and/or - * modify it under the terms of the GNU Library General Public - * License as published by the Free Software Foundation; either - * version 2 of the License, or (at your option) any later version. - * - * This library is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU - * Library General Public License for more details. - * - * You should have received a copy of the GNU Library General Public License - * along with this library; see the file COPYING.LIB. If not, write to - * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, - * Boston, MA 02110-1301, USA. - */ - -//#define EVENT_DEBUG -#include "dom/dom_exception.h" -#include "dom/dom_node.h" -#include "dom/html_image.h" -#include "xml/dom_textimpl.h" -#include "xml/dom_docimpl.h" -#include "xml/dom2_eventsimpl.h" -#include "xml/dom_elementimpl.h" -#include "xml/dom_restyler.h" - -#include "html/dtd.h" -#include "html/htmlparser.h" -#include "html/html_imageimpl.h" - -#include "rendering/render_canvas.h" -#include "misc/htmlhashes.h" -#include "css/css_valueimpl.h" -#include "css/css_stylesheetimpl.h" -#include "css/cssstyleselector.h" -#include "css/cssvalues.h" -#include "css/cssproperties.h" -#include "xml/dom_xmlimpl.h" - -#include <tqtextstream.h> -#include <kdebug.h> -#include <stdlib.h> - -// ### support default attributes -// ### dispatch mutation events -// ### check for INVALID_CHARACTER_ERR where appropriate - -using namespace DOM; -using namespace khtml; - -AttrImpl::AttrImpl(ElementImpl* element, DocumentImpl* docPtr, NodeImpl::Id attrId, - DOMStringImpl *value, DOMStringImpl *prefix) - : NodeBaseImpl(docPtr), - m_element(element), - m_attrId(attrId) -{ - m_value = value; - m_value->ref(); - - m_prefix = prefix; - if (m_prefix) - m_prefix->ref(); - m_specified = true; // we don't yet support default attributes -} - -AttrImpl::~AttrImpl() -{ - m_value->deref(); - if (m_prefix) - m_prefix->deref(); -} - -DOMString AttrImpl::nodeName() const -{ - return name(); -} - -unsigned short AttrImpl::nodeType() const -{ - return Node::ATTRIBUTE_NODE; -} - -DOMString AttrImpl::prefix() const -{ - return m_prefix; -} - -void AttrImpl::setPrefix(const DOMString &_prefix, int &exceptioncode ) -{ - checkSetPrefix(_prefix, exceptioncode); - if (exceptioncode) - return; - - if (m_prefix == _prefix.implementation()) - return; - - if (m_prefix) - m_prefix->deref(); - m_prefix = _prefix.implementation(); - if (m_prefix) - m_prefix->ref(); -} - -DOMString AttrImpl::namespaceURI() const -{ - if (m_htmlCompat) - return DOMString(); - return getDocument()->getName(NamespaceId, m_attrId >> 16); -} - -DOMString AttrImpl::localName() const -{ - if (m_htmlCompat) - return DOMString(); - return getDocument()->getName(AttributeId, m_attrId); -} - -DOMString AttrImpl::nodeValue() const -{ - return m_value; -} - -DOMString AttrImpl::name() const -{ - DOMString n = getDocument()->getName(AttributeId, m_attrId); - - // compat mode always return attribute names in lowercase. - // that's not formally in the specification, but common - // practice - a w3c erratum to DOM L2 is pending. - if (m_htmlCompat) - n = n.lower(); - - if (m_prefix && m_prefix->l) - return DOMString(m_prefix) + ":" + n; - - return n; -} - -void AttrImpl::setValue( const DOMString &v, int &exceptioncode ) -{ - exceptioncode = 0; - - // ### according to the DOM docs, we should create an unparsed Text child - // node here - // do not interprete entities in the string, its literal! - - // NO_MODIFICATION_ALLOWED_ERR: Raised when the node is readonly - if (isReadOnly()) { - exceptioncode = DOMException::NO_MODIFICATION_ALLOWED_ERR; - return; - } - - // ### what to do on 0 ? - if (v.isNull()) { - exceptioncode = DOMException::DOMSTRING_SIZE_ERR; - return; - } - - if (m_value == v.implementation()) - return; - - if (m_element && m_attrId == ATTR_ID) - m_element->updateId(m_value, v.implementation()); - - m_value->deref(); - m_value = v.implementation(); - m_value->ref(); - - if (m_element) { - m_element->parseAttribute(m_attrId,m_value); - m_element->attributeChanged(m_attrId); - } -} - -void AttrImpl::setNodeValue( const DOMString &v, int &exceptioncode ) -{ - exceptioncode = 0; - // NO_MODIFICATION_ALLOWED_ERR: taken care of by setValue() - setValue(v, exceptioncode); -} - -NodeImpl *AttrImpl::cloneNode ( bool /*deep*/) -{ - AttrImpl* attr = new AttrImpl(0, docPtr(), m_attrId, m_value, m_prefix); - attr->setHTMLCompat(m_htmlCompat); - return attr; -} - -// DOM Section 1.1.1 -bool AttrImpl::childAllowed( NodeImpl *newChild ) -{ - if(!newChild) - return false; - - return childTypeAllowed(newChild->nodeType()); -} - -bool AttrImpl::childTypeAllowed( unsigned short type ) -{ - switch (type) { - case Node::TEXT_NODE: - case Node::ENTITY_REFERENCE_NODE: - return true; - break; - default: - return false; - } -} - -DOMString AttrImpl::toString() const -{ - DOMString result; - - result += nodeName(); - - // FIXME: substitute entities for any instances of " or ' -- - // maybe easier to just use text value and ignore existing - // entity refs? - - if ( firstChild() ) { - result += "=\""; - - for (NodeImpl *child = firstChild(); child != NULL; child = child->nextSibling()) { - result += child->toString(); - } - - result += "\""; - } else if ( !nodeValue().isEmpty() ){ - //remove the else once the AttributeImpl changes are merged - result += "=\""; - result += nodeValue(); - result += "\""; - } - - return result; -} - -void AttrImpl::setElement(ElementImpl *element) -{ - m_element = element; -} - -// Strictly speaking, these two methods should not be needed, but -// we can't fully deal with the mess that are DOM attributes right.. -DOMStringImpl* AttrImpl::textContent() const -{ - if (m_value) - return new DOMStringImpl(m_value->s, m_value->l); - else - return 0; -} - -void AttrImpl::setTextContent( const DOMString &text, int& exceptioncode ) -{ - setValue(text, exceptioncode); -} - -// ------------------------------------------------------------------------- - -void AttributeImpl::setValue(DOMStringImpl *value, ElementImpl *element) -{ - assert(value); - if (m_attrId) { - if (m_data.value == value) - return; - - if (element && m_attrId == ATTR_ID) - element->updateId(m_data.value, value); - - m_data.value->deref(); - m_data.value = value; - m_data.value->ref(); - - if (element) { - element->parseAttribute(this); - element->attributeChanged(m_attrId); - } - } - else { - int exceptioncode = 0; - m_data.attr->setValue(value,exceptioncode); - // AttrImpl::setValue() calls parseAttribute() - } -} - -AttrImpl *AttributeImpl::createAttr(ElementImpl *element, DocumentImpl *docPtr) -{ - if (m_attrId) { - AttrImpl *attr = new AttrImpl(element,docPtr,m_attrId,m_data.value); - if (!attr) return 0; - attr->setHTMLCompat( docPtr->htmlMode() != DocumentImpl::XHtml ); - m_data.value->deref(); - m_data.attr = attr; - m_data.attr->ref(); - m_attrId = 0; /* "has implementation" flag */ - } - - return m_data.attr; -} - -void AttributeImpl::free() -{ - if (m_attrId) { - m_data.value->deref(); - } - else { - m_data.attr->setElement(0); - m_data.attr->deref(); - } -} - -// ------------------------------------------------------------------------- - -ElementImpl::ElementImpl(DocumentImpl *doc) - : NodeBaseImpl(doc) -{ - namedAttrMap = 0; - m_styleDecls = 0; - m_prefix = 0; -} - -ElementImpl::~ElementImpl() -{ - if(namedAttrMap) { - namedAttrMap->detachFromElement(); - namedAttrMap->deref(); - } - - if (m_styleDecls) { - m_styleDecls->setNode(0); - m_styleDecls->setParent(0); - m_styleDecls->deref(); - } - - if (m_prefix) - m_prefix->deref(); -} - -unsigned short ElementImpl::nodeType() const -{ - return Node::ELEMENT_NODE; -} - -DOMStringImpl* ElementImpl::getAttributeImpl( NodeImpl::Id id, bool nsAware, DOMStringImpl* qName) const -{ - if (!namedAttrMap) - return 0; - - DOMStringImpl *value = namedAttrMap->getValue(id, nsAware, qName); - if (value) - return value; - - // then search in default attr in case it is not yet set - NamedAttrMapImpl* dm = defaultMap(); - value = dm ? dm->getValue(id, nsAware, qName) : 0; - if (value) - return value; - - return 0; -} - -DOMString ElementImpl::getAttribute( NodeImpl::Id id, bool nsAware, const DOMString& qName) const -{ - return DOMString(getAttributeImpl(id, nsAware, qName.implementation())); -} - -void ElementImpl::setAttribute(NodeImpl::Id id, const DOMString &value, const DOMString& qName, int &exceptioncode) -{ - // NO_MODIFICATION_ALLOWED_ERR: Raised when the node is readonly - if (isReadOnly()) { - exceptioncode = DOMException::NO_MODIFICATION_ALLOWED_ERR; - return; - } - attributes()->setValue(id, value.implementation(), (qName.isEmpty() ? 0: qName.implementation())); -} - -void ElementImpl::setAttributeNS( const DOMString &namespaceURI, const DOMString &qualifiedName, - const DOMString &value, int &exceptioncode ) -{ - // NO_MODIFICATION_ALLOWED_ERR: Raised when the node is readonly - if (isReadOnly()) { - exceptioncode = DOMException::NO_MODIFICATION_ALLOWED_ERR; - return; - } - int colonPos; - if (!DOM::checkQualifiedName(qualifiedName, namespaceURI, &colonPos, - false/*nameCanBeNull*/, false/*nameCanBeEmpty*/, - &exceptioncode)) - return; - DOMString prefix, localName; - splitPrefixLocalName(qualifiedName.implementation(), prefix, localName, colonPos); - NodeImpl::Id id = getDocument()->getId(AttributeId, namespaceURI.implementation(), - prefix.implementation(), localName.implementation(), false, true /*lookupHTML*/); - attributes()->setValue(id, value.implementation(), 0, prefix.implementation(), - true /*nsAware*/, !namespaceURI.isNull() /*hasNS*/); -} - -void ElementImpl::setAttribute(NodeImpl::Id id, const DOMString &value) -{ - int exceptioncode = 0; - setAttribute(id,value,DOMString(),exceptioncode); -} - -void ElementImpl::setAttributeMap( NamedAttrMapImpl* list ) -{ - // If setting the whole map changes the id attribute, we need to - // call updateId. - DOMStringImpl *oldId = namedAttrMap ? namedAttrMap->getValue(ATTR_ID) : 0; - DOMStringImpl *newId = list ? list->getValue(ATTR_ID) : 0; - - if (oldId || newId) { - updateId(oldId, newId); - } - - if (namedAttrMap) { - namedAttrMap->detachFromElement(); - namedAttrMap->deref(); - } - - namedAttrMap = list; - - if (namedAttrMap) { - namedAttrMap->ref(); - assert(namedAttrMap->m_element == 0); - namedAttrMap->setElement(this); - unsigned long len = namedAttrMap->length(); - for (unsigned long i = 0; i < len; i++) { - parseAttribute(&namedAttrMap->m_attrs[i]); - attributeChanged(namedAttrMap->m_attrs[i].id()); - } - } -} - -NodeImpl *ElementImpl::cloneNode(bool deep) -{ - ElementImpl *clone; - if ( !localName().isNull() ) - clone = getDocument()->createElementNS( namespaceURI(), nodeName() ); - else - clone = getDocument()->createElement( nodeName() ); - if (!clone) return 0; - finishCloneNode( clone, deep ); - return clone; -} - -void ElementImpl::finishCloneNode( ElementImpl* clone, bool deep ) -{ - // clone attributes - if (namedAttrMap) - clone->attributes()->copyAttributes(namedAttrMap); - - // clone individual style rules - if (m_styleDecls) - *(clone->styleRules()) = *m_styleDecls; - - if (deep) - cloneChildNodes(clone); -} - -DOMString ElementImpl::nodeName() const -{ - return tagName(); -} - -DOMString ElementImpl::namespaceURI() const -{ - if (m_htmlCompat) - return DOMString(); - return getDocument()->getName(NamespaceId, id() >> 16); -} - -DOMString ElementImpl::prefix() const -{ - return m_prefix; -} - -void ElementImpl::setPrefix( const DOMString &_prefix, int &exceptioncode ) -{ - checkSetPrefix(_prefix, exceptioncode); - if (exceptioncode) - return; - - if (m_prefix == _prefix.implementation()) - return; - - if (m_prefix) - m_prefix->deref(); - m_prefix = _prefix.implementation(); - if (m_prefix) - m_prefix->ref(); -} - -void ElementImpl::createAttributeMap() const -{ - namedAttrMap = new NamedAttrMapImpl(const_cast<ElementImpl*>(this)); - namedAttrMap->ref(); -} - -NamedAttrMapImpl* ElementImpl::defaultMap() const -{ - return 0; -} - -RenderStyle *ElementImpl::styleForRenderer(RenderObject * /*parentRenderer*/) -{ - return getDocument()->styleSelector()->styleForElement(this); -} - -RenderObject *ElementImpl::createRenderer(RenderArena *arena, RenderStyle *style) -{ - if (getDocument()->documentElement() == this && style->display() == NONE) { - // Ignore display: none on root elements. Force a display of block in that case. - RenderBlock* result = new (arena) RenderBlock(this); - if (result) result->setStyle(style); - return result; - } - return RenderObject::createObject(this, style); -} - -void ElementImpl::attach() -{ - assert(!attached()); - assert(!m_render); - assert(parentNode()); - -#if SPEED_DEBUG < 1 - createRendererIfNeeded(); -#endif - - NodeBaseImpl::attach(); -} - -void ElementImpl::close() -{ - NodeImpl::close(); - - // Trigger all the addChild changes as one large dynamic appendChildren change - if (attached()) - backwardsStructureChanged(); -} - -void ElementImpl::detach() -{ - getDocument()->dynamicDomRestyler().resetDependencies(this); - - NodeBaseImpl::detach(); -} - -void ElementImpl::structureChanged() -{ - NodeBaseImpl::structureChanged(); - - if (!getDocument()->renderer()) - return; // the document is about to be destroyed - - getDocument()->dynamicDomRestyler().restyleDepedent(this, StructuralDependency); - // In theory BackwardsStructurualDependencies are indifferent to prepend, - // but it's too rare to optimize. - getDocument()->dynamicDomRestyler().restyleDepedent(this, BackwardsStructuralDependency); -} - -void ElementImpl::backwardsStructureChanged() -{ - NodeBaseImpl::backwardsStructureChanged(); - - if (!getDocument()->renderer()) - return; // the document is about to be destroyed - - // Most selectors are not affected by append. Fire the few that are. - getDocument()->dynamicDomRestyler().restyleDepedent(this, BackwardsStructuralDependency); -} - -void ElementImpl::attributeChanged(NodeImpl::Id id) -{ - if (!getDocument()->renderer()) - return; // the document is about to be destroyed - -#if 0 // one-one dependencies for attributes disabled - getDocument()->dynamicDomRestyler().restyleDepedent(this, AttributeDependency); -#endif - if (getDocument()->dynamicDomRestyler().checkDependency(id, PersonalDependency)) - setChanged(true); - if (getDocument()->dynamicDomRestyler().checkDependency(id, AncestorDependency)) - setChangedAscendentAttribute(true); - if (getDocument()->dynamicDomRestyler().checkDependency(id, PredecessorDependency) && parent()) - // Any element that dependt on a predecessors attribute, also depend structurally on parent - parent()->structureChanged(); -} - -void ElementImpl::recalcStyle( StyleChange change ) -{ - // ### should go away and be done in renderobject - RenderStyle* _style = m_render ? m_render->style() : 0; - bool hasParentRenderer = parent() ? parent()->attached() : false; - -#if 0 - const char* debug; - switch(change) { - case NoChange: debug = "NoChange"; - break; - case NoInherit: debug= "NoInherit"; - break; - case Inherit: debug = "Inherit"; - break; - case Force: debug = "Force"; - break; - } - tqDebug("recalcStyle(%d: %s, changed: %d)[%p: %s]", change, debug, changed(), this, tagName().string().latin1()); -#endif - if ( hasParentRenderer && (change >= Inherit || changed()) ) { - RenderStyle *newStyle = getDocument()->styleSelector()->styleForElement(this); - newStyle->ref(); - StyleChange ch = diff( _style, newStyle ); - if (ch == Detach) { - if (attached()) detach(); - // ### Suboptimal. Style gets calculated again. - attach(); - // attach recalulates the style for all children. No need to do it twice. - setChanged( false ); - setHasChangedChild( false ); - newStyle->deref(); - return; - } - else if (ch != NoChange) { - if( m_render && newStyle ) { - m_render->setStyle(newStyle); - } - } - newStyle->deref(); - - if ( change != Force) - change = ch; - } - // If a changed attribute has ancestor dependencies, restyle all children - if (changedAscendentAttribute()) { - change = Force; - setChangedAscendentAttribute(false); - } - - NodeImpl *n; - for (n = _first; n; n = n->nextSibling()) { - if ( change >= Inherit || n->isTextNode() || - n->hasChangedChild() || n->changed() ) { - //tqDebug(" (%p) calling recalcStyle on child %p/%s, change=%d", this, n, n->isElementNode() ? ((ElementImpl *)n)->tagName().string().latin1() : n->isTextNode() ? "text" : "unknown", change ); - n->recalcStyle( change ); - } - } - - setChanged( false ); - setHasChangedChild( false ); -} - -bool ElementImpl::isFocusable() const -{ - // Only make editable elements selectable if its parent element - // is not editable. FIXME: this is not 100% right as non-editable elements - // within editable elements are focusable too. - return contentEditable() && !(parentNode() && parentNode()->contentEditable()); -} - -// DOM Section 1.1.1 -bool ElementImpl::childAllowed( NodeImpl *newChild ) -{ - if (!childTypeAllowed(newChild->nodeType())) - return false; - - // ### check xml element allowedness according to DTD - - // If either this node or the other node is an XML element node, allow regardless (we don't do DTD checks for XML - // yet) - if (isXMLElementNode() || newChild->isXMLElementNode()) - return true; - else - return checkChild(id(), newChild->id(), !getDocument()->inCompatMode()); -} - -bool ElementImpl::childTypeAllowed( unsigned short type ) -{ - switch (type) { - case Node::ELEMENT_NODE: - case Node::TEXT_NODE: - case Node::COMMENT_NODE: - case Node::PROCESSING_INSTRUCTION_NODE: - case Node::CDATA_SECTION_NODE: - case Node::ENTITY_REFERENCE_NODE: - return true; - break; - default: - return false; - } -} - -void ElementImpl::scrollIntoView(bool /*alignToTop*/) -{ - // ### - kdWarning() << "non-standard scrollIntoView() not implemented" << endl; -} - -void ElementImpl::createDecl( ) -{ - m_styleDecls = new CSSStyleDeclarationImpl(0); - m_styleDecls->ref(); - m_styleDecls->setParent(getDocument()->elementSheet()); - m_styleDecls->setNode(this); - m_styleDecls->setStrictParsing( !getDocument()->inCompatMode() ); -} - -void ElementImpl::dispatchAttrRemovalEvent(NodeImpl::Id /*id*/, DOMStringImpl * /*value*/) -{ - // ### enable this stuff again - if (!getDocument()->hasListenerType(DocumentImpl::DOMATTRMODIFIED_LISTENER)) - return; - //int exceptioncode = 0; - //dispatchEvent(new MutationEventImpl(EventImpl::DOMATTRMODIFIED_EVENT,true,false,attr,attr->value(), - //attr->value(), getDocument()->attrName(attr->id()),MutationEvent::REMOVAL),exceptioncode); -} - -void ElementImpl::dispatchAttrAdditionEvent(NodeImpl::Id /*id*/, DOMStringImpl * /*value*/) -{ - // ### enable this stuff again - if (!getDocument()->hasListenerType(DocumentImpl::DOMATTRMODIFIED_LISTENER)) - return; - //int exceptioncode = 0; - //dispatchEvent(new MutationEventImpl(EventImpl::DOMATTRMODIFIED_EVENT,true,false,attr,attr->value(), - //attr->value(),getDocument()->attrName(attr->id()),MutationEvent::ADDITION),exceptioncode); -} - -void ElementImpl::updateId(DOMStringImpl* oldId, DOMStringImpl* newId) -{ - if (!inDocument()) - return; - - if (oldId && oldId->l) - removeId(DOMString(oldId).string()); - - if (newId && newId->l) - addId(DOMString(newId).string()); -} - -void ElementImpl::removeId(const TQString& id) -{ - getDocument()->getElementByIdCache().remove(id, this); -} - -void ElementImpl::addId(const TQString& id) -{ - getDocument()->getElementByIdCache().add(id, this); -} - -void ElementImpl::insertedIntoDocument() -{ - // need to do superclass processing first so inDocument() is true - // by the time we reach updateId - NodeBaseImpl::insertedIntoDocument(); - - if (hasID()) { - DOMString id = getAttribute(ATTR_ID); - updateId(0, id.implementation()); - } -} - -void ElementImpl::removedFromDocument() -{ - if (hasID()) { - DOMString id = getAttribute(ATTR_ID); - updateId(id.implementation(), 0); - } - - NodeBaseImpl::removedFromDocument(); -} - -DOMString ElementImpl::openTagStartToString(bool expandurls) const -{ - DOMString result = DOMString("<") + tagName(); - - NamedAttrMapImpl *attrMap = attributes(true); - - if (attrMap) { - unsigned long numAttrs = attrMap->length(); - for (unsigned long i = 0; i < numAttrs; i++) { - result += " "; - - AttributeImpl *attribute = attrMap->attrAt(i); - AttrImpl *attr = attribute->attr(); - - if (attr) { - result += attr->toString(); - } else { - result += getDocument()->getName( NodeImpl::AttributeId, attribute->id()); - if (!attribute->value().isNull()) { - result += "=\""; - // FIXME: substitute entities for any instances of " or ' - // Expand out all urls, i.e. the src and href attributes - if(expandurls && ( attribute->id() == ATTR_SRC || attribute->id() == ATTR_HREF)) - if(getDocument()) { - //We need to sanitize the urls - strip out the passwords. - //FIXME: are src= and href= the only places that might have a password and need to be sanitized? - KURL safeURL(getDocument()->completeURL(attribute->value().string())); - safeURL.setPass(TQString::null); - result += safeURL.htmlURL(); - } - else { - kdWarning() << "getDocument() returned false"; - result += attribute->value(); - } - else - result += attribute->value(); - result += "\""; - } - } - } - } - - return result; -} -DOMString ElementImpl::selectionToString(NodeImpl *selectionStart, NodeImpl *selectionEnd, int startOffset, int endOffset, bool &found) const -{ - DOMString result = openTagStartToString(); - - if (hasChildNodes()) { - result += ">"; - - for (NodeImpl *child = firstChild(); child != NULL; child = child->nextSibling()) { - result += child->selectionToString(selectionStart, selectionEnd, startOffset, endOffset, found); // this might set found to true - if(child == selectionEnd) - found = true; - if(found) break; - } - - result += "</"; - result += tagName(); - result += ">"; - } else { - result += " />"; - } - - return result; -} - -DOMString ElementImpl::toString() const -{ - TQString result = openTagStartToString().string(); //Accumulate in TQString, since DOMString can't append well. - - if (hasChildNodes()) { - result += ">"; - - for (NodeImpl *child = firstChild(); child != NULL; child = child->nextSibling()) { - DOMString kid = child->toString(); - result += TQConstString(kid.unicode(), kid.length()).string(); - } - - result += "</"; - result += tagName().string(); - result += ">"; - } else if (result.length() == 1) { - // ensure we dont get results like < /> can happen when serialize document - result = ""; - } else { - result += " />"; - } - - return result; -} - -bool ElementImpl::contentEditable() const { -#if 0 - DOM::CSSPrimitiveValueImpl *val = static_cast<DOM::CSSPrimitiveValueImpl *> - (const_cast<ElementImpl *>(this)->styleRules() - ->getPropertyCSSValue(CSS_PROP__KONQ_USER_INPUT)); -// kdDebug() << "val" << val << endl; - return val ? val->getIdent() == CSS_VAL_ENABLED : false; -#endif - return NodeImpl::contentEditable(); -} - -void ElementImpl::setContentEditable(bool enabled) { - // FIXME: the approach is flawed, better use an enum instead of bool - int value; - if (enabled) - value = CSS_VAL_ENABLED; - else { - // Intelligently use "none" or "disabled", depending on the type of - // element - // FIXME: intelligence not impl'd yet - value = CSS_VAL_NONE; - - // FIXME: reset caret if it is in this node or a child - }/*end if*/ - // FIXME: use addCSSProperty when I get permission to move it here -// kdDebug(6000) << "CSS_PROP__KHTML_USER_INPUT: "<< value << endl; - styleRules()->setProperty(CSS_PROP__KHTML_USER_INPUT, value, false, true); - setChanged(); - -} - -// ------------------------------------------------------------------------- - -XMLElementImpl::XMLElementImpl(DocumentImpl *doc, NodeImpl::Id id) - : ElementImpl(doc) -{ - // Called from createElement(). In this case localName, prefix, and namespaceURI all need to be null. - m_id = id; -} - -XMLElementImpl::XMLElementImpl(DocumentImpl *doc, NodeImpl::Id id, DOMStringImpl *_prefix) - : ElementImpl(doc) -{ - // Called from createElementNS() - m_id = id; - - m_prefix = _prefix; - if (m_prefix) - m_prefix->ref(); -} - -XMLElementImpl::~XMLElementImpl() -{ -} - -DOMString XMLElementImpl::localName() const -{ - if ( m_htmlCompat ) - return DOMString(); // was created with non-namespace-aware createElement() - return getDocument()->getName(ElementId, m_id); -} - -DOMString XMLElementImpl::tagName() const -{ - DOMString tn = getDocument()->getName(ElementId, id()); - if (m_htmlCompat) - tn = tn.upper(); - - if (m_prefix) - return DOMString(m_prefix) + ":" + tn; - - return tn; -} - -NodeImpl *XMLElementImpl::cloneNode ( bool deep ) -{ - XMLElementImpl *clone = new XMLElementImpl(docPtr(), id(), m_prefix); - finishCloneNode( clone, deep ); - return clone; -} - -// ------------------------------------------------------------------------- - -NamedAttrMapImpl::NamedAttrMapImpl(ElementImpl *element) - : m_element(element), - m_attrs(0), - m_attrCount(0) -{ -} - -NamedAttrMapImpl::~NamedAttrMapImpl() -{ - for (unsigned long i = 0; i < m_attrCount; i++) - m_attrs[i].free(); - free(m_attrs); -} - -NodeImpl *NamedAttrMapImpl::getNamedItem ( NodeImpl::Id id, bool nsAware, DOMStringImpl* qName ) const -{ - if (!m_element) - return 0; - unsigned int mask = nsAware ? ~0L : NodeImpl_IdLocalMask; - id = (id & mask); - - for (unsigned long i = 0; i < m_attrCount; i++) { - if ((m_attrs[i].id() & mask) == id) { - // if we are called with a qualified name, filter out NS-aware elements with non-matching name. - if (qName && (namespacePart(m_attrs[i].id()) != defaultNamespace) && - strcasecmp(m_attrs[i].name(), DOMString(qName))) - continue; - return m_attrs[i].createAttr(m_element,m_element->docPtr()); - } - } - - return 0; -} - -Node NamedAttrMapImpl::removeNamedItem ( NodeImpl::Id id, bool nsAware, DOMStringImpl* qName, int &exceptioncode ) -{ - if (!m_element) { - exceptioncode = DOMException::NOT_FOUND_ERR; - return 0; - } - - // NO_MODIFICATION_ALLOWED_ERR: Raised when the node is readonly - if (isReadOnly()) { - exceptioncode = DOMException::NO_MODIFICATION_ALLOWED_ERR; - return 0; - } - unsigned int mask = nsAware ? ~0L : NodeImpl_IdLocalMask; - id = (id & mask); - - for (unsigned long i = 0; i < m_attrCount; i++) { - if ((m_attrs[i].id() & mask) == id) { - // if we are called with a qualified name, filter out NS-aware elements with non-matching name. - if (qName && (namespacePart(m_attrs[i].id()) != defaultNamespace) && - strcasecmp(m_attrs[i].name(), DOMString(qName))) - continue; - id = m_attrs[i].id(); - if (id == ATTR_ID) - m_element->updateId(m_attrs[i].val(), 0); - Node removed(m_attrs[i].createAttr(m_element,m_element->docPtr())); - m_attrs[i].free(); - memmove(m_attrs+i,m_attrs+i+1,(m_attrCount-i-1)*sizeof(AttributeImpl)); - m_attrCount--; - m_attrs = (AttributeImpl*)realloc(m_attrs,m_attrCount*sizeof(AttributeImpl)); - m_element->parseAttribute(id,0); - m_element->attributeChanged(id); - return removed; - } - } - - // NOT_FOUND_ERR: Raised if there is no node with the specified namespaceURI - // and localName in this map. - exceptioncode = DOMException::NOT_FOUND_ERR; - return 0; -} - -Node NamedAttrMapImpl::setNamedItem ( NodeImpl* arg, bool nsAware, DOMStringImpl* qName, int &exceptioncode ) -{ - if (!m_element) { - exceptioncode = DOMException::NOT_FOUND_ERR; - return 0; - } - - // NO_MODIFICATION_ALLOWED_ERR: Raised if this map is readonly. - if (isReadOnly()) { - exceptioncode = DOMException::NO_MODIFICATION_ALLOWED_ERR; - return 0; - } - - // WRONG_DOCUMENT_ERR: Raised if arg was created from a different document than the one that created this map. - if (arg->getDocument() != m_element->getDocument()) { - exceptioncode = DOMException::WRONG_DOCUMENT_ERR; - return 0; - } - - // HIERARCHY_REQUEST_ERR: Raised if an attempt is made to add a node doesn't belong in this NamedNodeMap - if (!arg->isAttributeNode()) { - exceptioncode = DOMException::HIERARCHY_REQUEST_ERR; - return 0; - } - AttrImpl *attr = static_cast<AttrImpl*>(arg); - - // INUSE_ATTRIBUTE_ERR: Raised if arg is an Attr that is already an attribute of another Element object. - // The DOM user must explicitly clone Attr nodes to re-use them in other elements. - if (attr->ownerElement() && attr->ownerElement() != m_element) { - exceptioncode = DOMException::INUSE_ATTRIBUTE_ERR; - return 0; - } - - if (attr->ownerElement() == m_element) { - // Already have this attribute. - // DOMTS core-1 test "hc_elementreplaceattributewithself" says we should return it. - return attr; - } - unsigned int mask = nsAware ? ~0L : NodeImpl_IdLocalMask; - NodeImpl::Id id = (attr->id() & mask); - - for (unsigned long i = 0; i < m_attrCount; i++) { - if ((m_attrs[i].id() & mask) == id) { - // if we are called with a qualified name, filter out NS-aware elements with non-matching name. - if (qName && (namespacePart(m_attrs[i].id()) != defaultNamespace) && - strcasecmp(m_attrs[i].name(), DOMString(qName))) - continue; - // Attribute exists; replace it - if (id == ATTR_ID) - m_element->updateId(m_attrs[i].val(), attr->val()); - - Node replaced = m_attrs[i].createAttr(m_element,m_element->docPtr()); - m_attrs[i].free(); - m_attrs[i].m_attrId = 0; /* "has implementation" flag */ - m_attrs[i].m_data.attr = attr; - m_attrs[i].m_data.attr->ref(); - attr->setElement(m_element); - m_element->parseAttribute(&m_attrs[i]); - m_element->attributeChanged(m_attrs[i].id()); - // ### dispatch mutation events - return replaced; - } - } - - // No existing attribute; add to list - m_attrCount++; - m_attrs = (AttributeImpl*)realloc(m_attrs,m_attrCount*sizeof(AttributeImpl)); - m_attrs[m_attrCount-1].m_attrId = 0; /* "has implementation" flag */ - m_attrs[m_attrCount-1].m_data.attr = attr; - m_attrs[m_attrCount-1].m_data.attr->ref(); - attr->setElement(m_element); - if (id == ATTR_ID) - m_element->updateId(0, attr->val()); - m_element->parseAttribute(&m_attrs[m_attrCount-1]); - m_element->attributeChanged(m_attrs[m_attrCount-1].id()); - // ### dispatch mutation events - - return 0; -} - -NodeImpl *NamedAttrMapImpl::item ( unsigned long index ) const -{ - if (!m_element) - return 0; - - if (index >= m_attrCount) - return 0; - else - return m_attrs[index].createAttr(m_element,m_element->docPtr()); -} - -unsigned long NamedAttrMapImpl::length( ) const -{ - if (!m_element) - return 0; - - return m_attrCount; -} - -NodeImpl::Id NamedAttrMapImpl::idAt(unsigned long index) const -{ - assert(index <= m_attrCount); - return m_attrs[index].id(); -} - -DOMStringImpl *NamedAttrMapImpl::valueAt(unsigned long index) const -{ - assert(index <= m_attrCount); - return m_attrs[index].val(); -} - -DOMStringImpl *NamedAttrMapImpl::getValue(NodeImpl::Id id, bool nsAware, DOMStringImpl* qName) const -{ - unsigned int mask = nsAware ? ~0L : NodeImpl_IdLocalMask; - id = (id & mask); - for (unsigned long i = 0; i < m_attrCount; i++) - if ((m_attrs[i].id() & mask) == id) { - // if we are called with a qualified name, filter out NS-aware elements with non-matching name. - if (qName && (namespacePart(m_attrs[i].id()) != defaultNamespace) && - strcasecmp(m_attrs[i].name(), qName)) - continue; - return m_attrs[i].val(); - } - return 0; -} - -void NamedAttrMapImpl::setValue(NodeImpl::Id id, DOMStringImpl *value, DOMStringImpl* qName, - DOMStringImpl *prefix, bool nsAware, bool hasNS) -{ - assert( !(qName && nsAware) ); - if (!id) return; - // Passing in a null value here causes the attribute to be removed. This is a khtml extension - // (the spec does not specify what to do in this situation). - int exceptioncode = 0; - if (!value) { - removeNamedItem(id, nsAware, qName, exceptioncode); - return; - } - unsigned int mask = nsAware ? ~0L : NodeImpl_IdLocalMask; - NodeImpl::Id mid = (id & mask); - - // Check for an existing attribute. - for (unsigned long i = 0; i < m_attrCount; i++) { - if ((m_attrs[i].id() & mask) == mid) { - // if we are called with a qualified name, filter out NS-aware elements with non-matching name. - if (qName && (namespacePart(m_attrs[i].id()) != defaultNamespace) && - strcasecmp(m_attrs[i].name(), DOMString(qName))) - continue; - if (prefix) - m_attrs[i].attr()->setPrefix(prefix,exceptioncode); - m_attrs[i].setValue(value,m_element); - // ### dispatch mutation events - return; - } - } - - // No existing matching attribute; add a new one - m_attrCount++; - m_attrs = (AttributeImpl*)realloc(m_attrs,m_attrCount*sizeof(AttributeImpl)); - if (!nsAware) { - // Called from setAttribute()... we only have a name - m_attrs[m_attrCount-1].m_attrId = id; - m_attrs[m_attrCount-1].m_data.value = value; - m_attrs[m_attrCount-1].m_data.value->ref(); - } - else { - // Called from setAttributeNS()... need to create a full AttrImpl here - if(!m_element) - return; - m_attrs[m_attrCount-1].m_data.attr = new AttrImpl(m_element,m_element->docPtr(), - id, - value, - prefix); - m_attrs[m_attrCount-1].m_attrId = 0; /* "has implementation" flag */ - m_attrs[m_attrCount-1].m_data.attr->ref(); - m_attrs[m_attrCount-1].m_data.attr->setHTMLCompat( !hasNS && - m_element->getDocument()->htmlMode() != DocumentImpl::XHtml ); - } - if (m_element) { - if (id == ATTR_ID) - m_element->updateId(0, value); - m_element->parseAttribute(&m_attrs[m_attrCount-1]); - m_element->attributeChanged(m_attrs[m_attrCount-1].id()); - } - // ### dispatch mutation events -} - -Attr NamedAttrMapImpl::removeAttr(AttrImpl *attr) -{ - for (unsigned long i = 0; i < m_attrCount; i++) { - if (m_attrs[i].attr() == attr) { - NodeImpl::Id id = m_attrs[i].id(); - if (id == ATTR_ID) - m_element->updateId(attr->val(), 0); - Node removed(m_attrs[i].createAttr(m_element,m_element->docPtr())); - m_attrs[i].free(); - memmove(m_attrs+i,m_attrs+i+1,(m_attrCount-i-1)*sizeof(AttributeImpl)); - m_attrCount--; - m_attrs = (AttributeImpl*)realloc(m_attrs,m_attrCount*sizeof(AttributeImpl)); - m_element->parseAttribute(id,0); - m_element->attributeChanged(id); - // ### dispatch mutation events - return removed; - } - } - - return 0; -} - -NodeImpl::Id NamedAttrMapImpl::mapId(DOMStringImpl* namespaceURI, - DOMStringImpl* localName, bool readonly) -{ - if (!m_element) - return 0; - - return m_element->getDocument()->getId(NodeImpl::AttributeId, namespaceURI, 0, localName, readonly, - true /*lookupHTML*/); -} - -void NamedAttrMapImpl::copyAttributes(NamedAttrMapImpl *other) -{ - assert(m_element); - unsigned long i; - for (i = 0; i < m_attrCount; i++) { - if (m_attrs[i].id() == ATTR_ID) - m_element->updateId(m_attrs[i].val(), 0); - m_attrs[i].free(); - } - m_attrCount = other->m_attrCount; - m_attrs = (AttributeImpl*)realloc(m_attrs,m_attrCount*sizeof(AttributeImpl)); - for (i = 0; i < m_attrCount; i++) { - m_attrs[i].m_attrId = other->m_attrs[i].m_attrId; - if (m_attrs[i].m_attrId) { - m_attrs[i].m_data.value = other->m_attrs[i].m_data.value; - m_attrs[i].m_data.value->ref(); - } - else { - m_attrs[i].m_data.attr = static_cast<AttrImpl*>(other->m_attrs[i].m_data.attr->cloneNode(true)); - m_attrs[i].m_data.attr->ref(); - m_attrs[i].m_data.attr->setElement(m_element); - } - if (m_attrs[i].id() == ATTR_ID) - m_element->updateId(0, m_attrs[i].val()); - m_element->parseAttribute(&m_attrs[i]); - m_element->attributeChanged(m_attrs[i].id()); - } -} - -void NamedAttrMapImpl::setElement(ElementImpl *element) -{ - assert(!m_element); - m_element = element; - - for (unsigned long i = 0; i < m_attrCount; i++) - if (m_attrs[i].attr()) - m_attrs[i].attr()->setElement(element); -} - -void NamedAttrMapImpl::detachFromElement() -{ - // This makes the map invalid; nothing can really be done with it now since it's not - // associated with an element. But we have to keep it around in memory in case there - // are still references to it. - m_element = 0; - for (unsigned long i = 0; i < m_attrCount; i++) - m_attrs[i].free(); - free(m_attrs); - m_attrs = 0; - m_attrCount = 0; -} |