/** * 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 #include #include // ### 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(); if (m_attrId == ATTR_ID) m_element->addId(DOMString(m_value).string()); 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(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; } qDebug("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() ) { //qDebug(" (%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 QString& id) { getDocument()->getElementByIdCache().remove(id, this); } void ElementImpl::addId(const QString& 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(QString::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 += ""; } else { result += " />"; } return result; } DOMString ElementImpl::toString() const { QString result = openTagStartToString().string(); //Accumulate in QString, since DOMString can't append well. if (hasChildNodes()) { result += ">"; for (NodeImpl *child = firstChild(); child != NULL; child = child->nextSibling()) { DOMString kid = child->toString(); result += QConstString(kid.unicode(), kid.length()).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 (const_cast(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(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(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; }