summaryrefslogtreecommitdiffstats
path: root/khtml/xml/dom_nodeimpl.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'khtml/xml/dom_nodeimpl.cpp')
-rw-r--r--khtml/xml/dom_nodeimpl.cpp2068
1 files changed, 0 insertions, 2068 deletions
diff --git a/khtml/xml/dom_nodeimpl.cpp b/khtml/xml/dom_nodeimpl.cpp
deleted file mode 100644
index 5febe695e..000000000
--- a/khtml/xml/dom_nodeimpl.cpp
+++ /dev/null
@@ -1,2068 +0,0 @@
-/*
- * This file is part of the DOM implementation for KDE.
- *
- * Copyright (C) 1999 Lars Knoll ([email protected])
- * (C) 1999 Antti Koivisto ([email protected])
- * (C) 2001 Dirk Mueller ([email protected])
- * (C) 2003-2006 Apple Computer, Inc.
- * (C) 2006 Allan Sandfeld Jensen ([email protected])
- *
- * 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.
- */
-
-#include "dom/dom_exception.h"
-#include "misc/htmlattrs.h"
-#include "misc/htmltags.h"
-#include "xml/dom_elementimpl.h"
-#include "xml/dom_textimpl.h"
-#include "xml/dom2_eventsimpl.h"
-#include "xml/dom_docimpl.h"
-#include "xml/dom_nodeimpl.h"
-#include "xml/dom_restyler.h"
-
-#include <kglobal.h>
-#include <kdebug.h>
-
-#include "rendering/render_text.h"
-#include "rendering/render_flow.h"
-#include "rendering/render_line.h"
-
-#include "ecma/kjs_proxy.h"
-#include "khtmlview.h"
-#include "khtml_part.h"
-#include "dom_nodeimpl.h"
-
-// from khtml_caret_p.h
-namespace khtml {
-void /*KDE_NO_EXPORT*/ mapDOMPosToRenderPos(DOM::NodeImpl *node, long offset,
- khtml::RenderObject *&r, long &r_ofs, bool &outside, bool &outsideEnd);
-}
-
-using namespace DOM;
-using namespace khtml;
-
-NodeImpl::NodeImpl(DocumentImpl *doc)
- : m_document(doc),
- m_previous(0),
- m_next(0),
- m_render(0),
- m_tabIndex( 0 ),
- m_hasId( false ),
- m_attached(false),
- m_closed(false),
- m_changed( false ),
- m_hasChangedChild( false ),
- m_changedAscendentAttribute( false ),
- m_inDocument( false ),
- m_hasAnchor( false ),
- m_specified( false ),
- m_hovered( false ),
- m_focused( false ),
- m_active( false ),
- m_implicit( false ),
- m_htmlCompat( false ),
- m_hasClassList( false ),
- m_hasClass( false )
-{
-}
-
-NodeImpl::~NodeImpl()
-{
- if (m_render)
- detach();
- if (m_previous)
- m_previous->setNextSibling(0);
- if (m_next)
- m_next->setPreviousSibling(0);
-}
-
-DOMString NodeImpl::nodeValue() const
-{
- return DOMString();
-}
-
-void NodeImpl::setNodeValue( const DOMString &/*_nodeValue*/, int &/*exceptioncode*/ )
-{
- // by default nodeValue is null, so setting it has no effect
- // don't throw NO_MODIFICATION_ALLOWED_ERR from here, DOMTS-Core-Level1's hc_nodevalue03
- // (createEntityReference().setNodeValue())) says it would be wrong.
- // This must be done by subclasses instead.
-}
-
-DOMString NodeImpl::nodeName() const
-{
- return DOMString();
-}
-
-unsigned short NodeImpl::nodeType() const
-{
- return 0;
-}
-
-NodeListImpl *NodeImpl::childNodes()
-{
- return new ChildNodeListImpl(this);
-}
-
-NodeImpl *NodeImpl::firstChild() const
-{
- return 0;
-}
-
-NodeImpl *NodeImpl::lastChild() const
-{
- return 0;
-}
-
-NodeImpl *NodeImpl::insertBefore( NodeImpl *, NodeImpl *, int &exceptioncode )
-{
- exceptioncode = DOMException::HIERARCHY_REQUEST_ERR;
- return 0;
-}
-
-void NodeImpl::replaceChild( NodeImpl *, NodeImpl *, int &exceptioncode )
-{
- exceptioncode = DOMException::HIERARCHY_REQUEST_ERR;
-}
-
-void NodeImpl::removeChild( NodeImpl *, int &exceptioncode )
-{
- exceptioncode = DOMException::NOT_FOUND_ERR;
-}
-
-NodeImpl *NodeImpl::appendChild( NodeImpl *, int &exceptioncode )
-{
- exceptioncode = DOMException::HIERARCHY_REQUEST_ERR;
- return 0;
-}
-
-bool NodeImpl::hasChildNodes( ) const
-{
- return false;
-}
-
-void NodeImpl::normalize ()
-{
- // ### normalize attributes? (when we store attributes using child nodes)
- int exceptioncode = 0;
- NodeImpl *child = firstChild();
-
- // Recursively go through the subtree beneath us, normalizing all nodes. In the case
- // where there are two adjacent text nodes, they are merged together
- while (child) {
- NodeImpl *nextChild = child->nextSibling();
-
- if (nextChild && child->nodeType() == Node::TEXT_NODE && nextChild->nodeType() == Node::TEXT_NODE) {
- // Current child and the next one are both text nodes... merge them
- TextImpl *currentText = static_cast<TextImpl*>(child);
- TextImpl *nextText = static_cast<TextImpl*>(nextChild);
-
- currentText->appendData(nextText->data(),exceptioncode);
- if (exceptioncode)
- return;
-
- removeChild(nextChild,exceptioncode);
- if (exceptioncode)
- return;
- }
- else {
- child->normalize();
- child = nextChild;
- }
- }
-}
-
-DOMString NodeImpl::prefix() const
-{
- // For nodes other than elements and attributes, the prefix is always null
- return DOMString();
-}
-
-DOMString NodeImpl::namespaceURI() const
-{
- return DOMString();
-}
-
-void NodeImpl::setPrefix(const DOMString &/*_prefix*/, int &exceptioncode )
-{
- // The spec says that for nodes other than elements and attributes, prefix is always null.
- // It does not say what to do when the user tries to set the prefix on another type of
- // node, however mozilla throws a NAMESPACE_ERR exception
- exceptioncode = DOMException::NAMESPACE_ERR;
-}
-
-DOMString NodeImpl::localName() const
-{
- return DOMString();
-}
-
-void NodeImpl::setFirstChild(NodeImpl *)
-{
-}
-
-void NodeImpl::setLastChild(NodeImpl *)
-{
-}
-
-NodeImpl *NodeImpl::addChild(NodeImpl *)
-{
- return 0;
-}
-
-void NodeImpl::getCaret(int offset, bool override, int &_x, int &_y, int &width, int &height)
-{
- if (m_render) {
- RenderObject *r;
- long r_ofs;
- bool outside, outsideEnd;
-#if 0
-kdDebug(6200) << "getCaret: node " << this << " " << nodeName().string() << " offset: " << offset << endl;
-#endif
- mapDOMPosToRenderPos(this, offset, r, r_ofs, outside, outsideEnd);
-#if 0
-kdDebug(6200) << "getCaret: r " << r << " " << (r?r->renderName():TQString::null) << " r_ofs: " << r_ofs << " outside " << outside << " outsideEnd " << outsideEnd << endl;
-#endif
- if (r) {
- r->caretPos(r_ofs, override*RenderObject::CFOverride
- + outside*RenderObject::CFOutside
- + outsideEnd*RenderObject::CFOutsideEnd, _x, _y, width, height);
- } else
- _x = _y = height = -1, width = 1;
- } else _x = _y = height = -1, width = 1;
-}
-
-TQRect NodeImpl::getRect() const
-{
- int _x, _y;
- if(m_render && m_render->absolutePosition(_x, _y))
- return TQRect( _x + m_render->inlineXPos(), _y + m_render->inlineYPos(),
- m_render->width(), m_render->height() + renderer()->borderTopExtra() + renderer()->borderBottomExtra() );
-
- return TQRect();
-}
-
-void NodeImpl::setChanged(bool b)
-{
- if (b && !attached()) // changed compared to what?
- return;
-
- m_changed = b;
- if ( b ) {
- NodeImpl *p = parentNode();
- while ( p ) {
- p->setHasChangedChild( true );
- p = p->parentNode();
- }
- getDocument()->setDocumentChanged();
- }
-}
-
-bool NodeImpl::isInline() const
-{
- if (m_render) return m_render->style()->display() == khtml::INLINE;
- return !isElementNode();
-}
-
-
-unsigned long NodeImpl::nodeIndex() const
-{
- NodeImpl *_tempNode = previousSibling();
- unsigned long count=0;
- for( count=0; _tempNode; count++ )
- _tempNode = _tempNode->previousSibling();
- return count;
-}
-
-void NodeImpl::addEventListener(int id, EventListener *listener, const bool useCapture)
-{
- switch (id) {
- case EventImpl::DOMSUBTREEMODIFIED_EVENT:
- getDocument()->addListenerType(DocumentImpl::DOMSUBTREEMODIFIED_LISTENER);
- break;
- case EventImpl::DOMNODEINSERTED_EVENT:
- getDocument()->addListenerType(DocumentImpl::DOMNODEINSERTED_LISTENER);
- break;
- case EventImpl::DOMNODEREMOVED_EVENT:
- getDocument()->addListenerType(DocumentImpl::DOMNODEREMOVED_LISTENER);
- break;
- case EventImpl::DOMNODEREMOVEDFROMDOCUMENT_EVENT:
- getDocument()->addListenerType(DocumentImpl::DOMNODEREMOVEDFROMDOCUMENT_LISTENER);
- break;
- case EventImpl::DOMNODEINSERTEDINTODOCUMENT_EVENT:
- getDocument()->addListenerType(DocumentImpl::DOMNODEINSERTEDINTODOCUMENT_LISTENER);
- break;
- case EventImpl::DOMATTRMODIFIED_EVENT:
- getDocument()->addListenerType(DocumentImpl::DOMATTRMODIFIED_LISTENER);
- break;
- case EventImpl::DOMCHARACTERDATAMODIFIED_EVENT:
- getDocument()->addListenerType(DocumentImpl::DOMCHARACTERDATAMODIFIED_LISTENER);
- break;
- default:
- break;
- }
-
- m_regdListeners.addEventListener(id, listener, useCapture);
-}
-
-void NodeImpl::removeEventListener(int id, EventListener *listener, bool useCapture)
-{
- m_regdListeners.removeEventListener(id, listener, useCapture);
-}
-
-void NodeImpl::setHTMLEventListener(int id, EventListener *listener)
-{
- m_regdListeners.setHTMLEventListener(id, listener);
-}
-
-EventListener *NodeImpl::getHTMLEventListener(int id)
-{
- return m_regdListeners.getHTMLEventListener(id);
-}
-
-void NodeImpl::dispatchEvent(EventImpl *evt, int &exceptioncode, bool tempEvent)
-{
- evt->setTarget(this);
-
- // Since event handling code could cause this object to be deleted, grab a reference to the view now
- KHTMLView *view = getDocument()->view();
-
- dispatchGenericEvent( evt, exceptioncode );
-
- // If tempEvent is true, this means that the DOM implementation will not be storing a reference to the event, i.e.
- // there is no way to retrieve it from javascript if a script does not already have a reference to it in a variable.
- // So there is no need for the interpreter to keep the event in its cache
- if (tempEvent && view && view->part() && view->part()->jScript())
- view->part()->jScript()->finishedWithEvent(evt);
-}
-
-void NodeImpl::dispatchGenericEvent( EventImpl *evt, int &/*exceptioncode */)
-{
- // ### check that type specified
-
- // work out what nodes to send event to
- TQPtrList<NodeImpl> nodeChain;
- NodeImpl *n;
- for (n = this; n; n = n->parentNode()) {
- n->ref();
- nodeChain.prepend(n);
- }
-
- // trigger any capturing event handlers on our way down
- evt->setEventPhase(Event::CAPTURING_PHASE);
- TQPtrListIterator<NodeImpl> it(nodeChain);
- for (; it.current() && it.current() != this && !evt->propagationStopped(); ++it) {
- evt->setCurrentTarget(it.current());
- it.current()->handleLocalEvents(evt,true);
- }
-
- // dispatch to the actual target node
- it.toLast();
- NodeImpl* propagationSentinel = 0;
- if (!evt->propagationStopped()) {
- evt->setEventPhase(Event::AT_TARGET);
- evt->setCurrentTarget(it.current());
- it.current()->handleLocalEvents(evt, true);
- if (!evt->propagationStopped())
- it.current()->handleLocalEvents(evt,false);
- else
- propagationSentinel = it.current();
- }
- --it;
-
- if (evt->bubbles()) {
- evt->setEventPhase(Event::BUBBLING_PHASE);
- for (; it.current() && !evt->propagationStopped(); --it) {
- if (evt->propagationStopped()) propagationSentinel = it.current();
- evt->setCurrentTarget(it.current());
- it.current()->handleLocalEvents(evt,false);
- }
-
- // now we call all default event handlers (this is not part of DOM - it is internal to khtml)
- evt->setCurrentTarget(0);
- evt->setEventPhase(0); // I guess this is correct, the spec does not seem to say
- for (it.toLast(); it.current() && it.current() != propagationSentinel &&
- !evt->defaultPrevented() && !evt->defaultHandled(); --it)
- it.current()->defaultEventHandler(evt);
-
- if (evt->id() == EventImpl::CLICK_EVENT && !evt->defaultPrevented() &&
- static_cast<MouseEventImpl*>( evt )->button() == 0) // LMB click
- dispatchUIEvent(EventImpl::DOMACTIVATE_EVENT, static_cast<UIEventImpl*>(evt)->detail());
- }
-
- // copy this over into a local variable, as the following deref() calls might cause this to be deleted.
- DocumentImpl *doc = m_document.get();
- doc->ref();
-
- // deref all nodes in chain
- it.toFirst();
- for (; it.current(); ++it)
- it.current()->deref(); // this may delete us
-
- DocumentImpl::updateDocumentsRendering();
- doc->deref();
-}
-
-bool NodeImpl::dispatchHTMLEvent(int _id, bool canBubbleArg, bool cancelableArg)
-{
- int exceptioncode = 0;
- EventImpl* const evt = new EventImpl(static_cast<EventImpl::EventId>(_id),canBubbleArg,cancelableArg);
- evt->ref();
- dispatchEvent(evt,exceptioncode,true);
- bool ret = !evt->defaultPrevented();
- evt->deref();
- return ret;
-}
-
-void NodeImpl::dispatchWindowEvent(int _id, bool canBubbleArg, bool cancelableArg)
-{
- int exceptioncode = 0;
- EventImpl* const evt = new EventImpl(static_cast<EventImpl::EventId>(_id),canBubbleArg,cancelableArg);
- evt->setTarget( 0 );
- evt->ref();
- DocumentImpl *doc = getDocument();
- doc->ref();
- dispatchGenericEvent( evt, exceptioncode );
- if (!evt->defaultPrevented() && doc)
- doc->defaultEventHandler(evt);
-
- if (_id == EventImpl::LOAD_EVENT && !evt->propagationStopped() && doc) {
- // For onload events, send them to the enclosing frame only.
- // This is a DOM extension and is independent of bubbling/capturing rules of
- // the DOM. You send the event only to the enclosing frame. It does not
- // bubble through the parent document.
- DOM::ElementImpl* elt = doc->ownerElement();
- if (elt && (elt->getDocument()->domain().isNull() ||
- elt->getDocument()->domain() == doc->domain())) {
- // We also do a security check, since we don't want to allow the enclosing
- // iframe to see loads of child documents in other domains.
- evt->setCurrentTarget(elt);
-
- // Capturing first.
- elt->handleLocalEvents(evt,true);
-
- // Bubbling second.
- if (!evt->propagationStopped())
- elt->handleLocalEvents(evt,false);
- }
- }
-
- doc->deref();
- evt->deref();
-}
-
-void NodeImpl::dispatchMouseEvent(TQMouseEvent *_mouse, int overrideId, int overrideDetail)
-{
- bool cancelable = true;
- int detail = overrideDetail; // defaults to 0
- EventImpl::EventId evtId = EventImpl::UNKNOWN_EVENT;
- if (overrideId) {
- evtId = static_cast<EventImpl::EventId>(overrideId);
- }
- else {
- switch (_mouse->type()) {
- case TQEvent::MouseButtonPress:
- evtId = EventImpl::MOUSEDOWN_EVENT;
- break;
- case TQEvent::MouseButtonRelease:
- evtId = EventImpl::MOUSEUP_EVENT;
- break;
- case TQEvent::MouseButtonDblClick:
- evtId = EventImpl::CLICK_EVENT;
- detail = 1; // ### support for multiple double clicks
- break;
- case TQEvent::MouseMove:
- evtId = EventImpl::MOUSEMOVE_EVENT;
- cancelable = false;
- break;
- default:
- break;
- }
- }
- if (evtId == EventImpl::UNKNOWN_EVENT)
- return; // shouldn't happen
-
-
- int exceptioncode = 0;
- int pageX = _mouse->x();
- int pageY = _mouse->y();
- int clientX = pageX;
- int clientY = pageY;
- if ( getDocument()->view() )
- getDocument()->view()->viewportToContents( clientX, clientY, pageX, pageY );
-
- int screenX = _mouse->globalX();
- int screenY = _mouse->globalY();
-
- int button = -1;
- switch (_mouse->button()) {
- case Qt::LeftButton:
- button = 0;
- break;
- case Qt::MidButton:
- button = 1;
- break;
- case Qt::RightButton:
- button = 2;
- break;
- default:
- break;
- }
- bool ctrlKey = (_mouse->state() & TQt::ControlButton);
- bool altKey = (_mouse->state() & TQt::AltButton);
- bool shiftKey = (_mouse->state() & TQt::ShiftButton);
- bool metaKey = false; // ### qt support?
-
- EventImpl* const evt = new MouseEventImpl(evtId,true,cancelable,getDocument()->defaultView(),
- detail,screenX,screenY,clientX,clientY,pageX,pageY,ctrlKey,altKey,shiftKey,metaKey,
- button,0);
- evt->ref();
- dispatchEvent(evt,exceptioncode,true);
- evt->deref();
-}
-
-void NodeImpl::dispatchUIEvent(int _id, int detail)
-{
- assert (!( (_id != EventImpl::DOMFOCUSIN_EVENT &&
- _id != EventImpl::DOMFOCUSOUT_EVENT &&
- _id != EventImpl::DOMACTIVATE_EVENT)));
-
- bool cancelable = false;
- if (_id == EventImpl::DOMACTIVATE_EVENT)
- cancelable = true;
-
- int exceptioncode = 0;
- UIEventImpl* const evt = new UIEventImpl(static_cast<EventImpl::EventId>(_id),true,
- cancelable,getDocument()->defaultView(),detail);
- evt->ref();
- dispatchEvent(evt,exceptioncode,true);
- evt->deref();
-}
-
-void NodeImpl::dispatchSubtreeModifiedEvent()
-{
- childrenChanged();
- getDocument()->incDOMTreeVersion();
- if (!getDocument()->hasListenerType(DocumentImpl::DOMSUBTREEMODIFIED_LISTENER))
- return;
- int exceptioncode = 0;
- MutationEventImpl* const evt = new MutationEventImpl(EventImpl::DOMSUBTREEMODIFIED_EVENT,true,
- false,0,DOMString(),DOMString(),DOMString(),0);
- evt->ref();
- dispatchEvent(evt,exceptioncode,true);
- evt->deref();
-}
-
-bool NodeImpl::dispatchKeyEvent(TQKeyEvent *key, bool keypress)
-{
- int exceptioncode = 0;
- //kdDebug(6010) << "DOM::NodeImpl: dispatching keyboard event" << endl;
- EventImpl* keyEventImpl;
- if (keypress)
- keyEventImpl = new TextEventImpl(key, getDocument()->defaultView());
- else
- keyEventImpl = new KeyboardEventImpl(key, getDocument()->defaultView());
- keyEventImpl->ref();
- dispatchEvent(keyEventImpl,exceptioncode,true);
- bool r = keyEventImpl->defaultHandled() || keyEventImpl->defaultPrevented();
- keyEventImpl->deref();
- return r;
-}
-
-void NodeImpl::handleLocalEvents(EventImpl *evt, bool useCapture)
-{
- if (!m_regdListeners.listeners)
- return;
-
- Event ev = evt;
- // removeEventListener (e.g. called from a JS event listener) might
- // invalidate the item after the current iterator (which "it" is pointing to).
- // So we make a copy of the list.
- TQValueList<RegisteredEventListener> listeners = *m_regdListeners.listeners;
- TQValueList<RegisteredEventListener>::iterator it;
- for (it = listeners.begin(); it != listeners.end(); ++it) {
- //Check whether this got removed...KDE4: use Java-style iterators
- if (!m_regdListeners.stillContainsListener(*it))
- continue;
-
- RegisteredEventListener& current = (*it);
- if (current.id == evt->id() && current.useCapture == useCapture)
- current.listener->handleEvent(ev);
-
- // ECMA legacy hack
- if (current.useCapture == useCapture && evt->id() == EventImpl::CLICK_EVENT) {
- MouseEventImpl* me = static_cast<MouseEventImpl*>(evt);
- if (me->button() == 0) {
- // To find whether to call onclick or ondblclick, we can't
- // * use me->detail(), it's 2 when clicking twice w/o moving, even very slowly
- // * use me->qEvent(), it's not available when using initMouseEvent/dispatchEvent
- // So we currently store a bool in MouseEventImpl. If anyone needs to trigger
- // dblclicks from the DOM API, we'll need a timer here (well in the doc).
- if ( ( !me->isDoubleClick() && current.id == EventImpl::KHTML_ECMA_CLICK_EVENT) ||
- ( me->isDoubleClick() && current.id == EventImpl::KHTML_ECMA_DBLCLICK_EVENT) )
- current.listener->handleEvent(ev);
- }
- }
- }
-}
-
-void NodeImpl::defaultEventHandler(EventImpl *)
-{
-}
-
-unsigned long NodeImpl::childNodeCount()
-{
- return 0;
-}
-
-NodeImpl *NodeImpl::childNode(unsigned long /*index*/)
-{
- return 0;
-}
-
-NodeImpl *NodeImpl::traverseNextNode(NodeImpl *stayWithin) const
-{
- if (firstChild() || stayWithin == this)
- return firstChild();
- else if (nextSibling())
- return nextSibling();
- else {
- const NodeImpl *n = this;
- while (n && !n->nextSibling() && (!stayWithin || n->parentNode() != stayWithin))
- n = n->parentNode();
- if (n)
- return n->nextSibling();
- }
- return 0;
-}
-
-NodeImpl *NodeImpl::traversePreviousNode() const
-{
- if (previousSibling()) {
- NodeImpl *n = previousSibling();
- while (n->lastChild())
- n = n->lastChild();
- return n;
- }
- else if (parentNode()) {
- return parentNode();
- }
- else {
- return 0;
- }
-}
-
-void NodeImpl::checkSetPrefix(const DOMString &_prefix, int &exceptioncode)
-{
- // Perform error checking as required by spec for setting Node.prefix. Used by
- // ElementImpl::setPrefix() and AttrImpl::setPrefix()
-
- // INVALID_CHARACTER_ERR: Raised if the specified prefix contains an illegal character.
- if (!Element::khtmlValidPrefix(_prefix)) {
- exceptioncode = DOMException::INVALID_CHARACTER_ERR;
- return;
- }
-
- // NO_MODIFICATION_ALLOWED_ERR: Raised if this node is readonly.
- if (isReadOnly()) {
- exceptioncode = DOMException::NO_MODIFICATION_ALLOWED_ERR;
- return;
- }
-
- // NAMESPACE_ERR: - Raised if the specified prefix is malformed
- // - if the namespaceURI of this node is null,
- // - if the specified prefix is "xml" and the namespaceURI of this node is different from
- // "http://www.w3.org/XML/1998/namespace",
- // - if this node is an attribute and the specified prefix is "xmlns" and
- // the namespaceURI of this node is different from "http://www.w3.org/2000/xmlns/",
- // - or if this node is an attribute and the qualifiedName of this node is "xmlns" [Namespaces].
- if (Element::khtmlMalformedPrefix(_prefix) || (namespacePart(id()) == defaultNamespace && id() > ID_LAST_TAG) ||
- (_prefix == "xml" && namespaceURI() != "http://www.w3.org/XML/1998/namespace")) {
- exceptioncode = DOMException::NAMESPACE_ERR;
- return;
- }
-}
-
-void NodeImpl::checkAddChild(NodeImpl *newChild, int &exceptioncode)
-{
- // Perform error checking as required by spec for adding a new child. Used by
- // appendChild(), replaceChild() and insertBefore()
-
- // Not mentioned in spec: throw NOT_FOUND_ERR if newChild is null
- if (!newChild) {
- exceptioncode = DOMException::NOT_FOUND_ERR;
- return;
- }
-
- // NO_MODIFICATION_ALLOWED_ERR: Raised if this node is readonly
- if (isReadOnly()) {
- exceptioncode = DOMException::NO_MODIFICATION_ALLOWED_ERR;
- return;
- }
-
- // WRONG_DOCUMENT_ERR: Raised if newChild was created from a different document than the one that
- // created this node.
- // We assume that if newChild is a DocumentFragment, all children are created from the same document
- // as the fragment itself (otherwise they could not have been added as children)
- if (newChild->getDocument() != getDocument()) {
- exceptioncode = DOMException::WRONG_DOCUMENT_ERR;
- return;
- }
-
- // HIERARCHY_REQUEST_ERR: Raised if this node is of a type that does not allow children of the type of the
- // newChild node, or if the node to append is one of this node's ancestors.
-
- // check for ancestor/same node
- if (isAncestor(newChild)) {
- exceptioncode = DOMException::HIERARCHY_REQUEST_ERR;
- return;
- }
-
- // check node allowed
- if (newChild->nodeType() == Node::DOCUMENT_FRAGMENT_NODE) {
- // newChild is a DocumentFragment... check all its children instead of newChild itself
- NodeImpl *child;
- for (child = newChild->firstChild(); child; child = child->nextSibling()) {
- if (!childTypeAllowed(child->nodeType())) {
- exceptioncode = DOMException::HIERARCHY_REQUEST_ERR;
- return;
- }
- }
- }
- else {
- // newChild is not a DocumentFragment... check if it's allowed directly
- if(!childTypeAllowed(newChild->nodeType())) {
- exceptioncode = DOMException::HIERARCHY_REQUEST_ERR;
- return;
- }
- }
-}
-
-bool NodeImpl::isAncestor( NodeImpl *other )
-{
- // Return true if other is the same as this node or an ancestor of it, otherwise false
- NodeImpl *n;
- for (n = this; n; n = n->parentNode()) {
- if (n == other)
- return true;
- }
- return false;
-}
-
-bool NodeImpl::childAllowed( NodeImpl *newChild )
-{
- return childTypeAllowed(newChild->nodeType());
-}
-
-NodeImpl::StyleChange NodeImpl::diff( khtml::RenderStyle *s1, khtml::RenderStyle *s2 )
-{
- // This method won't work when a style contains noninherited properties with "inherit" value.
- StyleChange ch = NoInherit;
-
- EDisplay display1 = s1 ? s1->display() : NONE;
- EDisplay display2 = s2 ? s2->display() : NONE;
- EPosition position1 = s1 ? s1->position() : STATIC;
- EPosition position2 = s2 ? s2->position() : STATIC;
-
- if (display1 != display2 || position1 != position2)
- ch = Detach;
- else if ( !s1 || !s2 )
- ch = Inherit;
- else if ( *s1 == *s2 )
- ch = NoChange;
- else if (s1->useNormalContent() != s2->useNormalContent())
- ch = Detach; // when we add generated content all children must be detached
- else if ( s1->inheritedNotEqual( s2 ) )
- ch = Inherit;
-
- // Because the first-letter implementation is so f..ked up, the easiest way
- // to update first-letter is to remove the entire node and readd it.
- if (ch < Detach && pseudoDiff(s1, s2, khtml::RenderStyle::FIRST_LETTER))
- ch = Detach;
- // If the other pseudoStyles have changed, we want to return NoInherit
- if (ch == NoChange && pseudoDiff(s1, s2, khtml::RenderStyle::BEFORE))
- ch = NoInherit;
- if (ch == NoChange && pseudoDiff(s1, s2, khtml::RenderStyle::AFTER))
- ch = NoInherit;
- if (ch == NoChange && pseudoDiff(s1, s2, khtml::RenderStyle::MARKER))
- ch = NoInherit;
- if (ch == NoChange && pseudoDiff(s1, s2, khtml::RenderStyle::SELECTION))
- ch = NoInherit;
- if (ch == NoChange && pseudoDiff(s1, s2, khtml::RenderStyle::FIRST_LINE))
- ch = NoInherit;
-
- return ch;
-}
-
-bool NodeImpl::pseudoDiff( khtml::RenderStyle *s1, khtml::RenderStyle *s2, unsigned int pid)
-{
- khtml::RenderStyle *ps1 = s1 ? s1->getPseudoStyle((khtml::RenderStyle::PseudoId)pid) : 0;
- khtml::RenderStyle *ps2 = s2 ? s2->getPseudoStyle((khtml::RenderStyle::PseudoId)pid) : 0;
-
- if (ps1 == ps2)
- return false;
- else
- if (ps1 && ps2) {
- if (*ps1 == *ps2)
- return false;
- else
- return true;
- }
- else
- return true;
-}
-
-void NodeImpl::close()
-{
- if (m_render) m_render->close();
- m_closed = true;
-}
-
-void NodeImpl::attach()
-{
- assert(!attached());
- assert(!m_render || (m_render->style() && m_render->parent()));
- if (m_render) // set states to match node
- {
- if (closed()) m_render->close();
- if (hovered()) m_render->setMouseInside();
- }
- getDocument()->incDOMTreeVersion();
- m_attached = true;
-}
-
-void NodeImpl::detach()
-{
-// assert(m_attached);
-
- if ( m_render )
- m_render->detach();
-
- m_render = 0;
- getDocument()->incDOMTreeVersion();
- m_attached = false;
-}
-
-bool NodeImpl::maintainsState()
-{
- return false;
-}
-
-TQString NodeImpl::state()
-{
- return TQString::null;
-}
-
-void NodeImpl::restoreState(const TQString &/*state*/)
-{
-}
-
-void NodeImpl::insertedIntoDocument()
-{
- setInDocument(true);
-}
-
-void NodeImpl::removedFromDocument()
-{
- setInDocument(false);
-}
-
-void NodeImpl::childrenChanged()
-{
- if (parentNode())
- parentNode()->childrenChanged();
-}
-
-bool NodeImpl::isReadOnly()
-{
- // Entity & Entity Reference nodes and their descendants are read-only
- NodeImpl *n = this;
- while (n) {
- if (n->nodeType() == Node::ENTITY_NODE ||
- n->nodeType() == Node::ENTITY_REFERENCE_NODE)
- return true;
- n = n->parentNode();
- }
- return false;
-}
-
-RenderObject * NodeImpl::previousRenderer()
-{
- for (NodeImpl *n = previousSibling(); n; n = n->previousSibling()) {
- if (n->renderer())
- return n->renderer();
- }
- return 0;
-}
-
-RenderObject * NodeImpl::nextRenderer()
-{
- for (NodeImpl *n = nextSibling(); n; n = n->nextSibling()) {
- if (n->renderer())
- return n->renderer();
- }
- return 0;
-}
-
-void NodeImpl::createRendererIfNeeded()
-{
-#ifdef APPLE_CHANGES
- if (!getDocument()->shouldCreateRenderers())
- return;
-#endif
-
- assert(!m_render);
-
- NodeImpl *parent = parentNode();
- assert(parent);
-
- RenderObject *parentRenderer = parent->renderer();
- if (parentRenderer && parentRenderer->childAllowed()) {
- RenderStyle *style = styleForRenderer(parentRenderer);
- style->ref();
- if (rendererIsNeeded(style)) {
- m_render = createRenderer(getDocument()->renderArena(), style);
- m_render->setStyle(style);
- parentRenderer->addChild(m_render, nextRenderer());
- }
- style->deref();
- }
-}
-
-RenderStyle *NodeImpl::styleForRenderer(RenderObject *parent)
-{
- return parent->style();
-}
-
-bool NodeImpl::rendererIsNeeded(RenderStyle *style)
-{
- return (getDocument()->documentElement() == this) || (style->display() != NONE);
-}
-
-RenderObject *NodeImpl::createRenderer(RenderArena* /*arena*/, RenderStyle* /*style*/)
-{
- assert(false);
- return 0;
-}
-
-bool NodeImpl::contentEditable() const
-{
- RenderObject *r = renderer();
- if (!r || !r->style()) return false;
- return r->style()->userInput() == UI_ENABLED;
-}
-
-long NodeImpl::minOffset() const
-{
- // Arrgh! You'd think *every* offset starts at zero, but loo,
- // therefore we need this method
- return renderer() ? renderer()->minOffset() : 0;
-}
-
-long NodeImpl::maxOffset() const
-{
- return const_cast<NodeImpl *>(this)->childNodeCount();
-// return renderer() ? renderer()->maxOffset() : 1;
-}
-
-DOMStringImpl* NodeImpl::textContent() const
-{
- TQString out;
- for (NodeImpl *child = firstChild(); child != 0; child = child->nextSibling()) {
- short type = child->nodeType();
- if (type != Node::COMMENT_NODE && type != Node::PROCESSING_INSTRUCTION_NODE) {
- DOMStringImpl* kidText = child->textContent();
- if (kidText)
- out += TQConstString(kidText->s, kidText->l).string();
- delete kidText;
- }
- }
- return new DOMStringImpl(out.unicode(), out.length());
-}
-
-//-------------------------------------------------------------------------
-
-NodeBaseImpl::~NodeBaseImpl()
-{
- //kdDebug( 6020 ) << "NodeBaseImpl destructor" << endl;
- // we have to tell all children, that the parent has died...
- NodeImpl *n;
- NodeImpl *next;
-
- for( n = _first; n != 0; n = next ) {
- next = n->nextSibling();
- n->setPreviousSibling(0);
- n->setNextSibling(0);
- n->setParent(0);
- if ( !n->refCount() )
- delete n;
- }
-}
-
-
-NodeImpl *NodeBaseImpl::firstChild() const
-{
- return _first;
-}
-
-NodeImpl *NodeBaseImpl::lastChild() const
-{
- return _last;
-}
-
-NodeImpl *NodeBaseImpl::insertBefore ( NodeImpl *newChild, NodeImpl *refChild, int &exceptioncode )
-{
- exceptioncode = 0;
-
- // insertBefore(...,null) is equivalent to appendChild()
- if(!refChild)
- return appendChild(newChild, exceptioncode);
-
- // Make sure adding the new child is ok
- checkAddChild(newChild, exceptioncode);
- if (exceptioncode)
- return 0;
-
- // NOT_FOUND_ERR: Raised if refChild is not a child of this node
- if (refChild->parentNode() != this) {
- exceptioncode = DOMException::NOT_FOUND_ERR;
- return 0;
- }
-
- bool isFragment = newChild->nodeType() == Node::DOCUMENT_FRAGMENT_NODE;
-
- // If newChild is a DocumentFragment with no children.... there's nothing to do.
- // Just return the document fragment
- if (isFragment && !newChild->firstChild())
- return newChild;
-
- // Now actually add the child(ren)
- NodeImpl *nextChild;
- NodeImpl *child = isFragment ? newChild->firstChild() : newChild;
-
- NodeImpl *prev = refChild->previousSibling();
- if ( prev == newChild || refChild == newChild ) // nothing to do
- return newChild;
-
- while (child) {
- nextChild = isFragment ? child->nextSibling() : 0;
-
- // If child is already present in the tree, first remove it
- NodeImpl *newParent = child->parentNode();
-
- //...guard it in case we need to move it..
- SharedPtr<NodeImpl> guard(child);
-
- if(newParent)
- newParent->removeChild( child, exceptioncode );
- if ( exceptioncode )
- return 0;
-
- // Add child in the correct position
- if (prev)
- prev->setNextSibling(child);
- else
- _first = child;
- refChild->setPreviousSibling(child);
- child->setParent(this);
- child->setPreviousSibling(prev);
- child->setNextSibling(refChild);
-
- // Add child to the rendering tree
- // ### should we detach() it first if it's already attached?
- if (attached() && !child->attached())
- child->attach();
-
- // Dispatch the mutation events
- dispatchChildInsertedEvents(child,exceptioncode);
-
- prev = child;
- child = nextChild;
- }
-
- structureChanged();
-
- // ### set style in case it's attached
- dispatchSubtreeModifiedEvent();
- return newChild;
-}
-
-void NodeBaseImpl::replaceChild ( NodeImpl *newChild, NodeImpl *oldChild, int &exceptioncode )
-{
- exceptioncode = 0;
-
- if ( oldChild == newChild ) // nothing to do
- return;
-
- // Make sure adding the new child is ok
- checkAddChild(newChild, exceptioncode);
- if (exceptioncode)
- return;
-
- // NOT_FOUND_ERR: Raised if oldChild is not a child of this node.
- if (!oldChild || oldChild->parentNode() != this) {
- exceptioncode = DOMException::NOT_FOUND_ERR;
- return;
- }
-
- bool isFragment = newChild->nodeType() == Node::DOCUMENT_FRAGMENT_NODE;
- NodeImpl *nextChild;
- NodeImpl *child = isFragment ? newChild->firstChild() : newChild;
-
-
- // Remove the old child
- NodeImpl *prev = oldChild->previousSibling();
- NodeImpl *next = oldChild->nextSibling();
-
- removeChild(oldChild, exceptioncode);
- if (exceptioncode)
- return;
-
- // Add the new child(ren)
- while (child) {
- nextChild = isFragment ? child->nextSibling() : 0;
-
- // If child is already present in the tree, first remove it
- NodeImpl *newParent = child->parentNode();
- if ( child == next )
- next = child->nextSibling();
- if ( child == prev )
- prev = child->previousSibling();
- //...guard it in case we need to move it..
- SharedPtr<NodeImpl> guard(child);
- if(newParent)
- newParent->removeChild( child, exceptioncode );
- if (exceptioncode)
- return;
-
- // Add child in the correct position
- if (prev) prev->setNextSibling(child);
- if (next) next->setPreviousSibling(child);
- if(!prev) _first = child;
- if(!next) _last = child;
- child->setParent(this);
- child->setPreviousSibling(prev);
- child->setNextSibling(next);
-
- // Add child to the rendering tree
- // ### should we detach() it first if it's already attached?
- if (attached() && !child->attached())
- child->attach();
-
- // Dispatch the mutation events
- dispatchChildInsertedEvents(child,exceptioncode);
-
- prev = child;
- child = nextChild;
- }
-
- structureChanged();
-
- // ### set style in case it's attached
- dispatchSubtreeModifiedEvent();
- return;
-}
-
-void NodeBaseImpl::removeChild ( NodeImpl *oldChild, int &exceptioncode )
-{
- exceptioncode = 0;
-
- // NO_MODIFICATION_ALLOWED_ERR: Raised if this node is readonly.
- if (isReadOnly()) {
- exceptioncode = DOMException::NO_MODIFICATION_ALLOWED_ERR;
- return;
- }
-
- // NOT_FOUND_ERR: Raised if oldChild is not a child of this node.
- if (!oldChild || oldChild->parentNode() != this) {
- exceptioncode = DOMException::NOT_FOUND_ERR;
- return;
- }
-
- dispatchChildRemovalEvents(oldChild,exceptioncode);
- if (exceptioncode)
- return;
-
- SharedPtr<NodeImpl> memManage(oldChild); //Make sure to free if needed
-
- // Remove from rendering tree
- if (oldChild->attached())
- oldChild->detach();
-
- // Remove the child
- NodeImpl *prev, *next;
- prev = oldChild->previousSibling();
- next = oldChild->nextSibling();
-
- if(next) next->setPreviousSibling(prev);
- if(prev) prev->setNextSibling(next);
- if(_first == oldChild) _first = next;
- if(_last == oldChild) _last = prev;
-
- oldChild->setPreviousSibling(0);
- oldChild->setNextSibling(0);
- oldChild->setParent(0);
-
- structureChanged();
-
- // Dispatch post-removal mutation events
- dispatchSubtreeModifiedEvent();
-
- NodeImpl *p = this;
- while (p->parentNode())
- p = p->parentNode();
- if (p->nodeType() == Node::DOCUMENT_NODE) {
- for (NodeImpl *c = oldChild; c; c = c->traverseNextNode(oldChild))
- c->removedFromDocument();
- }
-}
-
-void NodeBaseImpl::removeChildren()
-{
- bool inDoc = inDocument();
- NodeImpl *n, *next;
- for( n = _first, _first = 0; n; n = next )
- {
- next = n->nextSibling();
- if (n->attached())
- n->detach();
- n->setPreviousSibling(0);
- n->setNextSibling(0);
- n->setParent(0);
-
- if ( inDoc )
- for ( NodeImpl* c = n; c; c = c->traverseNextNode( n ) )
- c->removedFromDocument();
-
- if( !n->refCount() )
- delete n;
- }
- _last = 0;
-}
-
-
-NodeImpl *NodeBaseImpl::appendChild ( NodeImpl *newChild, int &exceptioncode )
-{
- exceptioncode = 0;
-
- // Make sure adding the new child is ok
- checkAddChild(newChild, exceptioncode);
- if (exceptioncode)
- return 0;
-
- if ( newChild == _last ) // nothing to do
- return newChild;
-
- bool isFragment = newChild->nodeType() == Node::DOCUMENT_FRAGMENT_NODE;
-
- // If newChild is a DocumentFragment with no children.... there's nothing to do.
- // Just return the document fragment
- if (isFragment && !newChild->firstChild())
- return newChild;
-
- // Now actually add the child(ren)
- NodeImpl *nextChild;
- NodeImpl *child = isFragment ? newChild->firstChild() : newChild;
-
- while (child) {
- nextChild = isFragment ? child->nextSibling() : 0;
-
- // If child is already present in the tree, first remove it
- NodeImpl *oldParent = child->parentNode();
- SharedPtr<NodeImpl> guard(child); //Guard in case we move it
- if(oldParent) {
- oldParent->removeChild( child, exceptioncode );
- if (exceptioncode)
- return 0;
- }
-
- // Append child to the end of the list
- child->setParent(this);
-
- if(_last)
- {
- child->setPreviousSibling(_last);
- _last->setNextSibling(child);
- _last = child;
- }
- else
- {
- _first = _last = child;
- }
-
- // Add child to the rendering tree
- // ### should we detach() it first if it's already attached?
- if (attached() && !child->attached())
- child->attach();
-
- // Dispatch the mutation events
- dispatchChildInsertedEvents(child,exceptioncode);
-
- child = nextChild;
- }
-
- backwardsStructureChanged();
-
- // ### set style in case it's attached
- dispatchSubtreeModifiedEvent();
- return newChild;
-}
-
-bool NodeBaseImpl::hasChildNodes ( ) const
-{
- return _first != 0;
-}
-
-// not part of the DOM
-void NodeBaseImpl::setFirstChild(NodeImpl *child)
-{
- _first = child;
-}
-
-void NodeBaseImpl::setLastChild(NodeImpl *child)
-{
- _last = child;
-}
-
-// check for same source document:
-bool NodeBaseImpl::checkSameDocument( NodeImpl *newChild, int &exceptioncode )
-{
- exceptioncode = 0;
- DocumentImpl *ownerDocThis = getDocument();
- DocumentImpl *ownerDocNew = newChild->getDocument();
- if(ownerDocThis != ownerDocNew) {
- kdDebug(6010)<< "not same document, newChild = " << newChild << "document = " << getDocument() << endl;
- exceptioncode = DOMException::WRONG_DOCUMENT_ERR;
- return true;
- }
- return false;
-}
-
-// check for being child:
-bool NodeBaseImpl::checkIsChild( NodeImpl *oldChild, int &exceptioncode )
-{
- if(!oldChild || oldChild->parentNode() != this) {
- exceptioncode = DOMException::NOT_FOUND_ERR;
- return true;
- }
- return false;
-}
-
-NodeImpl *NodeBaseImpl::addChild(NodeImpl *newChild)
-{
- // do not add applyChanges here! This function is only used during parsing
-
- // short check for consistency with DTD
- if(getDocument()->isHTMLDocument() && !childAllowed(newChild))
- {
- //kdDebug( 6020 ) << "AddChild failed! id=" << id() << ", child->id=" << newChild->id() << endl;
- return 0;
- }
-
- // just add it...
- newChild->setParent(this);
-
- if(_last)
- {
- newChild->setPreviousSibling(_last);
- _last->setNextSibling(newChild);
- _last = newChild;
- }
- else
- {
- _first = _last = newChild;
- }
-
- if (inDocument())
- newChild->insertedIntoDocument();
- childrenChanged();
-
- if(newChild->nodeType() == Node::ELEMENT_NODE)
- return newChild;
- return this;
-}
-
-void NodeBaseImpl::attach()
-{
- NodeImpl *child = _first;
- while(child != 0)
- {
- child->attach();
- child = child->nextSibling();
- }
- NodeImpl::attach();
-}
-
-void NodeBaseImpl::detach()
-{
- NodeImpl *child = _first;
- while(child != 0)
- {
- NodeImpl* prev = child;
- child = child->nextSibling();
- prev->detach();
- }
- NodeImpl::detach();
-}
-
-void NodeBaseImpl::cloneChildNodes(NodeImpl *clone)
-{
- int exceptioncode = 0;
- NodeImpl *n;
- for(n = firstChild(); n && !exceptioncode; n = n->nextSibling())
- {
- clone->appendChild(n->cloneNode(true),exceptioncode);
- }
-}
-
-// I don't like this way of implementing the method, but I didn't find any
-// other way. Lars
-bool NodeBaseImpl::getUpperLeftCorner(int &xPos, int &yPos) const
-{
- if (!m_render)
- return false;
- RenderObject *o = m_render;
- xPos = yPos = 0;
- if ( !o->isInline() || o->isReplaced() ) {
- o->absolutePosition( xPos, yPos );
- return true;
- }
-
- // find the next text/image child, to get a position
- while(o) {
- if(o->firstChild())
- o = o->firstChild();
- else if(o->nextSibling())
- o = o->nextSibling();
- else {
- RenderObject *next = 0;
- while(!next) {
- o = o->parent();
- if(!o) return false;
- next = o->nextSibling();
- }
- o = next;
- }
- if((o->isText() && !o->isBR()) || o->isReplaced()) {
- o->container()->absolutePosition( xPos, yPos );
- if (o->isText()) {
- xPos += o->inlineXPos();
- yPos += o->inlineYPos();
- } else {
- xPos += o->xPos();
- yPos += o->yPos();
- }
- return true;
- }
- }
- return true;
-}
-
-bool NodeBaseImpl::getLowerRightCorner(int &xPos, int &yPos) const
-{
- if (!m_render)
- return false;
-
- RenderObject *o = m_render;
- xPos = yPos = 0;
- if (!o->isInline() || o->isReplaced())
- {
- o->absolutePosition( xPos, yPos );
- xPos += o->width();
- yPos += o->height() + o->borderTopExtra() + o->borderBottomExtra();
- return true;
- }
- // find the last text/image child, to get a position
- while(o) {
- if(o->lastChild())
- o = o->lastChild();
- else if(o->previousSibling())
- o = o->previousSibling();
- else {
- RenderObject *prev = 0;
- while(!prev) {
- o = o->parent();
- if(!o) return false;
- prev = o->previousSibling();
- }
- o = prev;
- }
- if((o->isText() && !o->isBR()) || o->isReplaced()) {
- o->container()->absolutePosition(xPos, yPos);
- if (o->isText()) {
- xPos += o->inlineXPos() + o->width();
- yPos += o->inlineYPos() + o->height();
- } else {
- xPos += o->xPos() + o->width();
- yPos += o->yPos() + o->height();
- }
- return true;
- }
- }
- return true;
-}
-
-void NodeBaseImpl::setFocus(bool received)
-{
- if (m_focused == received) return;
-
- NodeImpl::setFocus(received);
-
- // note that we need to recalc the style
- setChanged(); // *:focus is a default style, so we just assume personal dependency
- if (isElementNode()) {
- getDocument()->dynamicDomRestyler().restyleDepedent(static_cast<ElementImpl*>(this), OtherStateDependency);
- }
-}
-
-void NodeBaseImpl::setActive(bool down)
-{
- if (down == active()) return;
-
- NodeImpl::setActive(down);
-
- // note that we need to recalc the style
- if (isElementNode())
- getDocument()->dynamicDomRestyler().restyleDepedent(static_cast<ElementImpl*>(this), ActiveDependency);
-}
-
-void NodeBaseImpl::setHovered(bool hover)
-{
- if (hover == hovered()) return;
-
- NodeImpl::setHovered(hover);
-
- // note that we need to recalc the style
- if (isElementNode())
- getDocument()->dynamicDomRestyler().restyleDepedent(static_cast<ElementImpl*>(this), HoverDependency);
-}
-
-unsigned long NodeBaseImpl::childNodeCount()
-{
- unsigned long count = 0;
- NodeImpl *n;
- for (n = firstChild(); n; n = n->nextSibling())
- count++;
- return count;
-}
-
-NodeImpl *NodeBaseImpl::childNode(unsigned long index)
-{
- unsigned long i;
- NodeImpl *n = firstChild();
- for (i = 0; n && i < index; i++)
- n = n->nextSibling();
- return n;
-}
-
-void NodeBaseImpl::dispatchChildInsertedEvents( NodeImpl *child, int &exceptioncode )
-{
- if (getDocument()->hasListenerType(DocumentImpl::DOMNODEINSERTED_LISTENER)) {
- MutationEventImpl* const evt = new MutationEventImpl(EventImpl::DOMNODEINSERTED_EVENT,true,false,this,DOMString(),DOMString(),DOMString(),0);
- evt->ref();
- child->dispatchEvent(evt,exceptioncode,true);
- evt->deref();
- if (exceptioncode)
- return;
- }
-
- // dispatch the DOMNodeInsertedIntoDocument event to all descendants
- bool hasInsertedListeners = getDocument()->hasListenerType(DocumentImpl::DOMNODEINSERTEDINTODOCUMENT_LISTENER);
- NodeImpl *p = this;
- while (p->parentNode())
- p = p->parentNode();
- if (p->nodeType() == Node::DOCUMENT_NODE) {
- for (NodeImpl *c = child; c; c = c->traverseNextNode(child)) {
- c->insertedIntoDocument();
-
- if (hasInsertedListeners) {
- MutationEventImpl* const evt = new MutationEventImpl(EventImpl::DOMNODEINSERTEDINTODOCUMENT_EVENT,false,false,0,DOMString(),DOMString(),DOMString(),0);
- evt->ref();
- c->dispatchEvent(evt,exceptioncode,true);
- evt->deref();
- if (exceptioncode)
- return;
- }
- }
- }
-}
-
-void NodeBaseImpl::dispatchChildRemovalEvents( NodeImpl *child, int &exceptioncode )
-{
- // Dispatch pre-removal mutation events
- getDocument()->notifyBeforeNodeRemoval(child); // ### use events instead
- if (getDocument()->hasListenerType(DocumentImpl::DOMNODEREMOVED_LISTENER)) {
- MutationEventImpl* const evt = new MutationEventImpl(EventImpl::DOMNODEREMOVED_EVENT,true,false,this,DOMString(),DOMString(),DOMString(),0);
- evt->ref();
- child->dispatchEvent(evt,exceptioncode,true);
- evt->deref();
- if (exceptioncode)
- return;
- }
-
- bool hasRemovalListeners = getDocument()->hasListenerType(DocumentImpl::DOMNODEREMOVEDFROMDOCUMENT_LISTENER);
-
- // dispatch the DOMNodeRemovedFromDocument event to all descendants
- NodeImpl *p = this;
- while (p->parentNode())
- p = p->parentNode();
- if (p->nodeType() == Node::DOCUMENT_NODE) {
- for (NodeImpl *c = child; c; c = c->traverseNextNode(child)) {
- if (hasRemovalListeners) {
- MutationEventImpl* const evt = new MutationEventImpl(EventImpl::DOMNODEREMOVEDFROMDOCUMENT_EVENT,false,false,0,DOMString(),DOMString(),DOMString(),0);
- evt->ref();
- c->dispatchEvent(evt,exceptioncode,true);
- evt->deref();
- if (exceptioncode)
- return;
- }
- }
- }
-}
-
-void NodeBaseImpl::setTextContent( const DOMString &text, int& exceptioncode )
-{
- // NO_MODIFICATION_ALLOWED_ERR: Raised if this node is readonly.
- if (isReadOnly()) {
- exceptioncode = DOMException::NO_MODIFICATION_ALLOWED_ERR;
- return;
- }
-
- removeChildren();
-
- if ( !text.isEmpty() && !text.isNull() ) {
- TextImpl *t = new TextImpl( docPtr(), text.implementation() );
- appendChild( t, exceptioncode );
- }
-}
-
-// ---------------------------------------------------------------------------
-NodeImpl *NodeListImpl::item( unsigned long index ) const
-{
- unsigned long requestIndex = index;
-
- m_cache->updateNodeListInfo(m_refNode->getDocument());
-
- NodeImpl* n;
- bool usedCache = false;
- if (m_cache->current.node) {
- //Compute distance from the requested index to the cache node
- long cacheDist = QABS(long(index) - long(m_cache->position));
-
- if (cacheDist < (long)index) { //Closer to the cached position
- usedCache = true;
- if (index >= m_cache->position) { //Go ahead
- unsigned long relIndex = index - m_cache->position;
- n = recursiveItem(m_refNode, m_cache->current.node, relIndex);
- } else { //Go backwards
- unsigned long relIndex = m_cache->position - index;
- n = recursiveItemBack(m_refNode, m_cache->current.node, relIndex);
- }
- }
- }
-
- if (!usedCache)
- n = recursiveItem(m_refNode, m_refNode->firstChild(), index);
-
- //We always update the cache state, to make starting iteration
- //where it was left off easy.
- m_cache->current.node = n;
- m_cache->position = requestIndex;
- return n;
-}
-
-unsigned long NodeListImpl::length() const
-{
- m_cache->updateNodeListInfo(m_refNode->getDocument());
- if (!m_cache->hasLength) {
- m_cache->length = calcLength( m_refNode );
- m_cache->hasLength = true;
- }
- return m_cache->length;
-}
-
-unsigned long NodeListImpl::calcLength(NodeImpl *start) const
-{
- unsigned long len = 0;
- for(NodeImpl *n = start->firstChild(); n != 0; n = n->nextSibling()) {
- bool recurse = true;
- if (nodeMatches(n, recurse))
- len++;
- if (recurse)
- len+= NodeListImpl::calcLength(n);
- }
-
- return len;
-}
-
-NodeListImpl::NodeListImpl( NodeImpl *n, int type, CacheFactory* factory )
-{
- m_refNode = n;
- m_refNode->ref();
-
- m_cache = m_refNode->getDocument()->acquireCachedNodeListInfo(
- factory ? factory : Cache::make,
- n, type );
-}
-
-NodeListImpl::~NodeListImpl()
-{
- m_refNode->getDocument()->releaseCachedNodeListInfo(m_cache);
- m_refNode->deref();
-}
-
-
-/**
- Next item in the pre-order walk of tree from node, but not going outside
- absStart
-*/
-static NodeImpl* helperNext(NodeImpl* node, NodeImpl* absStart)
-{
- //Walk up until we wind a sibling to go to.
- while (!node->nextSibling() && node != absStart)
- node = node->parentNode();
-
- if (node != absStart)
- return node->nextSibling();
- else
- return 0;
-}
-
-NodeImpl *NodeListImpl::recursiveItem ( NodeImpl* absStart, NodeImpl *start, unsigned long &offset ) const
-{
- for(NodeImpl *n = start; n != 0; n = helperNext(n, absStart)) {
- bool recurse = true;
- if (nodeMatches(n, recurse))
- if (!offset--)
- return n;
-
- NodeImpl *depthSearch = recurse ? recursiveItem(n, n->firstChild(), offset) : 0;
- if (depthSearch)
- return depthSearch;
- }
-
- return 0; // no matching node in this subtree
-}
-
-
-NodeImpl *NodeListImpl::recursiveItemBack ( NodeImpl* absStart, NodeImpl *start, unsigned long &offset ) const
-{
- //### it might be cleaner/faster to split nodeMatches and recursion
- //filtering.
- bool dummy = true;
- NodeImpl* n = start;
-
- do {
- bool recurse = true;
-
- //Check whether the current node matches.
- if (nodeMatches(n, dummy))
- if (!offset--)
- return n;
-
- if (n->previousSibling()) {
- //Move to the last node of this whole subtree that we should recurse into
- n = n->previousSibling();
- recurse = true;
-
- while (n->lastChild()) {
- (void)nodeMatches(n, recurse);
- if (!recurse)
- break; //Don't go there
- n = n->lastChild();
- }
- } else {
- //We're done with this whole subtree, so move up
- n = n->parentNode();
- }
- }
- while (n && n != absStart);
-
- return 0;
-}
-
-
-NodeListImpl::Cache::~Cache()
-{}
-
-void NodeListImpl::Cache::clear(DocumentImpl* doc)
-{
- hasLength = false;
- current.node = 0;
- version = doc->domTreeVersion();
-}
-
-void NodeListImpl::Cache::updateNodeListInfo(DocumentImpl* doc)
-{
- //If version doesn't match, clear
- if (doc->domTreeVersion() != version)
- clear(doc);
-}
-
-ChildNodeListImpl::ChildNodeListImpl( NodeImpl *n ): NodeListImpl(n, CHILD_NODES)
-{}
-
-bool ChildNodeListImpl::nodeMatches( NodeImpl* /*testNode*/, bool& doRecurse ) const
-{
- doRecurse = false;
- return true;
-}
-
-TagNodeListImpl::TagNodeListImpl( NodeImpl *n, NodeImpl::Id id )
- : NodeListImpl(n, UNCACHEABLE),
- m_id(id),
- m_namespaceAware(false)
-{
- // An id of 0 here means "*" (match all nodes)
- m_matchAllNames = (id == 0);
- m_matchAllNamespaces = false;
-}
-
-TagNodeListImpl::TagNodeListImpl( NodeImpl *n, const DOMString &namespaceURI, const DOMString &localName )
- : NodeListImpl(n, UNCACHEABLE),
- m_id(0),
- m_namespaceURI(namespaceURI),
- m_localName(localName),
- m_namespaceAware(true)
-{
- m_matchAllNames = (localName == "*");
- m_matchAllNamespaces = (namespaceURI == "*");
-}
-
-
-bool TagNodeListImpl::nodeMatches( NodeImpl *testNode, bool& /*doRecurse*/ ) const
-{
- if ( testNode->nodeType() != Node::ELEMENT_NODE ) return false;
- if (m_namespaceAware)
- return (m_matchAllNamespaces || testNode->namespaceURI() == m_namespaceURI) &&
- (m_matchAllNames || testNode->localName() == m_localName);
- else {
- NodeImpl::Id testId = testNode->id();
- //we have to strip the namespaces if we compare in a namespace unaware fashion
- if ( !m_namespaceAware ) testId = localNamePart(testId);
- return (m_id == 0 || m_id == testId);
- }
-}
-
-NameNodeListImpl::NameNodeListImpl(NodeImpl *n, const DOMString &t )
- : NodeListImpl(n, UNCACHEABLE),
- nodeName(t)
-{}
-
-bool NameNodeListImpl::nodeMatches( NodeImpl *testNode, bool& /*doRecurse*/ ) const
-{
- if ( testNode->nodeType() != Node::ELEMENT_NODE ) return false;
- return static_cast<ElementImpl *>(testNode)->getAttribute(ATTR_NAME) == nodeName;
-}
-
-// ---------------------------------------------------------------------------
-
-NamedNodeMapImpl::NamedNodeMapImpl()
-{
-}
-
-NamedNodeMapImpl::~NamedNodeMapImpl()
-{
-}
-
-// ----------------------------------------------------------------------------
-
-GenericRONamedNodeMapImpl::GenericRONamedNodeMapImpl(DocumentImpl* doc)
- : NamedNodeMapImpl()
-{
- m_doc = doc;
- m_contents = new TQPtrList<NodeImpl>;
-}
-
-GenericRONamedNodeMapImpl::~GenericRONamedNodeMapImpl()
-{
- while (!m_contents->isEmpty())
- m_contents->take(0)->deref();
-
- delete m_contents;
-}
-
-NodeImpl *GenericRONamedNodeMapImpl::getNamedItem ( NodeImpl::Id id, bool /*nsAware*/, DOMStringImpl* /*qName*/ ) const
-{
- // ## do we need namespace support in this class?
- TQPtrListIterator<NodeImpl> it(*m_contents);
- for (; it.current(); ++it)
- if (it.current()->id() == id)
- return it.current();
- return 0;
-}
-
-Node GenericRONamedNodeMapImpl::setNamedItem ( NodeImpl* /*arg*/, bool /*nsAware*/, DOMStringImpl* /*qName*/, int &exceptioncode )
-{
- // can't modify this list through standard DOM functions
- // NO_MODIFICATION_ALLOWED_ERR: Raised if this map is readonly
- exceptioncode = DOMException::NO_MODIFICATION_ALLOWED_ERR;
- return 0;
-}
-
-Node GenericRONamedNodeMapImpl::removeNamedItem ( NodeImpl::Id /*id*/, bool /*nsAware*/, DOMStringImpl* /*qName*/, int &exceptioncode )
-{
- // can't modify this list through standard DOM functions
- // NO_MODIFICATION_ALLOWED_ERR: Raised if this map is readonly
- exceptioncode = DOMException::NO_MODIFICATION_ALLOWED_ERR;
- return 0;
-}
-
-NodeImpl *GenericRONamedNodeMapImpl::item ( unsigned long index ) const
-{
- if (index >= m_contents->count())
- return 0;
-
- return m_contents->at(index);
-}
-
-unsigned long GenericRONamedNodeMapImpl::length( ) const
-{
- return m_contents->count();
-}
-
-void GenericRONamedNodeMapImpl::addNode(NodeImpl *n)
-{
- // The spec says that in the case of duplicates we only keep the first one
- if (getNamedItem(n->id(), false, 0))
- return;
-
- n->ref();
- m_contents->append(n);
-}
-
-NodeImpl::Id GenericRONamedNodeMapImpl::mapId(DOMStringImpl* namespaceURI,
- DOMStringImpl* localName, bool readonly)
-{
- if (!m_doc)
- return 0;
-
- return m_doc->getId(NodeImpl::ElementId,
- namespaceURI, 0, localName, readonly,
- false /*don't lookupHTML*/);
-}
-
-// -----------------------------------------------------------------------------
-
-void RegisteredListenerList::addEventListener(int id, EventListener *listener, const bool useCapture)
-{
- RegisteredEventListener rl(static_cast<EventImpl::EventId>(id),listener,useCapture);
- if (!listeners)
- listeners = new TQValueList<RegisteredEventListener>;
-
- // if this id/listener/useCapture combination is already registered, do nothing.
- // the DOM2 spec says that "duplicate instances are discarded", and this keeps
- // the listener order intact.
- TQValueList<RegisteredEventListener>::iterator it;
- for (it = listeners->begin(); it != listeners->end(); ++it)
- if (*it == rl)
- return;
-
- listeners->append(rl);
-}
-
-void RegisteredListenerList::removeEventListener(int id, EventListener *listener, bool useCapture)
-{
- if (!listeners) // nothing to remove
- return;
-
- RegisteredEventListener rl(static_cast<EventImpl::EventId>(id),listener,useCapture);
-
- TQValueList<RegisteredEventListener>::iterator it;
- for (it = listeners->begin(); it != listeners->end(); ++it)
- if (*it == rl) {
- listeners->remove(it);
- return;
- }
-}
-
-bool RegisteredListenerList::isHTMLEventListener(EventListener* listener)
-{
- return (listener->eventListenerType() == "_khtml_HTMLEventListener");
-}
-
-void RegisteredListenerList::setHTMLEventListener(int id, EventListener *listener)
-{
- if (!listeners)
- listeners = new TQValueList<RegisteredEventListener>;
-
- TQValueList<RegisteredEventListener>::iterator it;
- if (!listener) {
- for (it = listeners->begin(); it != listeners->end(); ++it) {
- if ((*it).id == id && isHTMLEventListener((*it).listener)) {
- listeners->remove(it);
- break;
- }
- }
- return;
- }
-
- // if this event already has a registered handler, insert the new one in
- // place of the old one, to preserve the order.
- RegisteredEventListener rl(static_cast<EventImpl::EventId>(id),listener,false);
-
- int i;
- for (i = 0, it = listeners->begin(); it != listeners->end(); ++it, ++i)
- if ((*it).id == id && isHTMLEventListener((*it).listener)) {
- listeners->insert(it, rl);
- listeners->remove(it);
- return;
- }
-
- listeners->append(rl);
-}
-
-EventListener *RegisteredListenerList::getHTMLEventListener(int id)
-{
- if (!listeners)
- return 0;
-
- TQValueList<RegisteredEventListener>::iterator it;
- for (it = listeners->begin(); it != listeners->end(); ++it)
- if ((*it).id == id && isHTMLEventListener((*it).listener)) {
- return (*it).listener;
- }
- return 0;
-}
-
-bool RegisteredListenerList::hasEventListener(int id)
-{
- if (!listeners)
- return false;
-
- TQValueList<RegisteredEventListener>::iterator it;
- for (it = listeners->begin(); it != listeners->end(); ++it)
- if ((*it).id == id)
- return true;
-
- return false;
-}
-
-void RegisteredListenerList::clear()
-{
- delete listeners;
- listeners = 0;
-}
-
-bool RegisteredListenerList::stillContainsListener(const RegisteredEventListener& listener)
-{
- if (!listeners)
- return false;
- return listeners->find(listener) != listeners->end();
-}
-
-RegisteredListenerList::~RegisteredListenerList() {
- delete listeners; listeners = 0;
-}