diff options
Diffstat (limited to 'khtml/xml/dom_docimpl.cpp')
-rw-r--r-- | khtml/xml/dom_docimpl.cpp | 2892 |
1 files changed, 0 insertions, 2892 deletions
diff --git a/khtml/xml/dom_docimpl.cpp b/khtml/xml/dom_docimpl.cpp deleted file mode 100644 index ff206a1cc..000000000 --- a/khtml/xml/dom_docimpl.cpp +++ /dev/null @@ -1,2892 +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) 2002-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 "xml/dom_textimpl.h" -#include "xml/dom_xmlimpl.h" -#include "xml/dom2_rangeimpl.h" -#include "xml/dom2_eventsimpl.h" -#include "xml/xml_tokenizer.h" -#include "html/htmltokenizer.h" -#include "xml/dom_restyler.h" - -#include "css/csshelper.h" -#include "css/cssstyleselector.h" -#include "css/css_stylesheetimpl.h" -#include "misc/htmlhashes.h" -#include "misc/helper.h" -#include "misc/seed.h" -#include "misc/loader.h" -#include "ecma/kjs_proxy.h" -#include "ecma/kjs_binding.h" - -#include <tqptrstack.h> -#include <tqpaintdevicemetrics.h> -#include <kdebug.h> -#include <klocale.h> -#include <kstaticdeleter.h> - -#include "rendering/counter_tree.h" -#include "rendering/render_canvas.h" -#include "rendering/render_replaced.h" -#include "rendering/render_arena.h" -#include "rendering/render_layer.h" -#include "rendering/render_frames.h" -#include "rendering/render_image.h" - -#include "khtmlview.h" -#include "khtml_part.h" - -#include <kglobalsettings.h> -#include <kstringhandler.h> -#include <krfcdate.h> -#include "khtml_settings.h" -#include "khtmlpart_p.h" - -#include "html/html_baseimpl.h" -#include "html/html_blockimpl.h" -#include "html/html_documentimpl.h" -#include "html/html_formimpl.h" -#include "html/html_headimpl.h" -#include "html/html_imageimpl.h" -#include "html/html_listimpl.h" -#include "html/html_miscimpl.h" -#include "html/html_tableimpl.h" -#include "html/html_objectimpl.h" - -#include <kapplication.h> -#include <kio/job.h> - -#include <stdlib.h> -#include "dom_docimpl.h" - -using namespace DOM; -using namespace khtml; - -// ------------------------------------------------------------------------ - -DOMImplementationImpl *DOMImplementationImpl::m_instance = 0; - -DOMImplementationImpl::DOMImplementationImpl() -{ -} - -DOMImplementationImpl::~DOMImplementationImpl() -{ -} - -bool DOMImplementationImpl::hasFeature ( const DOMString &feature, const DOMString &version ) -{ - // ### update when we (fully) support the relevant features - TQString lower = feature.string().lower(); - if ((lower == "html" || lower == "xml") && - (version.isEmpty() || version == "1.0" || version == "2.0" || version == "null")) - return true; - - // ## Do we support Core Level 3 ? - if ((lower == "core" ) && - (version.isEmpty() || version == "2.0" || version == "null")) - return true; - - if ((lower == "events" || lower == "uievents" || - lower == "mouseevents" || lower == "mutationevents" || - lower == "htmlevents" || lower == "textevents" ) && - (version.isEmpty() || version == "2.0" || version == "3.0" || version == "null")) - return true; - return false; -} - -DocumentTypeImpl *DOMImplementationImpl::createDocumentType( const DOMString &qualifiedName, const DOMString &publicId, - const DOMString &systemId, int &exceptioncode ) -{ - // Not mentioned in spec: throw NAMESPACE_ERR if no qualifiedName supplied - if (qualifiedName.isNull()) { - exceptioncode = DOMException::NAMESPACE_ERR; - return 0; - } - - // INVALID_CHARACTER_ERR: Raised if the specified qualified name contains an illegal character. - if (!Element::khtmlValidQualifiedName(qualifiedName)) { - exceptioncode = DOMException::INVALID_CHARACTER_ERR; - return 0; - } - - // NAMESPACE_ERR: Raised if the qualifiedName is malformed. - // Added special case for the empty string, which seems to be a common pre-DOM2 misuse - if (!qualifiedName.isEmpty() && Element::khtmlMalformedQualifiedName(qualifiedName)) { - exceptioncode = DOMException::NAMESPACE_ERR; - return 0; - } - - return new DocumentTypeImpl(this,0,qualifiedName,publicId,systemId); -} - -DOMImplementationImpl* DOMImplementationImpl::getInterface(const DOMString& /*feature*/) const -{ - // ### - return 0; -} - -DocumentImpl *DOMImplementationImpl::createDocument( const DOMString &namespaceURI, const DOMString &qualifiedName, - const DocumentType &doctype, int &exceptioncode ) -{ - exceptioncode = 0; - - if (!checkQualifiedName(qualifiedName, namespaceURI, 0, true/*nameCanBeNull*/, - true /*nameCanBeEmpty, see #61650*/, &exceptioncode) ) - return 0; - - DocumentTypeImpl *dtype = static_cast<DocumentTypeImpl*>(doctype.handle()); - // WRONG_DOCUMENT_ERR: Raised if doctype has already been used with a different document or was - // created from a different implementation. - if (dtype && (dtype->getDocument() || dtype->implementation() != this)) { - exceptioncode = DOMException::WRONG_DOCUMENT_ERR; - return 0; - } - - // ### this is completely broken.. without a view it will not work (Dirk) - DocumentImpl *doc = new DocumentImpl(this, 0); - - // now get the interesting parts of the doctype - // ### create new one if not there (currently always there) - if (doc->doctype() && dtype) - doc->doctype()->copyFrom(*dtype); - - // the document must be created empty if all parameters are null - // (or empty for qName/nsURI as a tolerance) - see DOM 3 Core. - if (dtype || !qualifiedName.isEmpty() || !namespaceURI.isEmpty()) { - ElementImpl *element = doc->createElementNS(namespaceURI,qualifiedName); - doc->appendChild(element,exceptioncode); - if (exceptioncode) { - delete element; - delete doc; - return 0; - } - } - return doc; -} - -CSSStyleSheetImpl *DOMImplementationImpl::createCSSStyleSheet(DOMStringImpl* /*title*/, DOMStringImpl *media, - int &/*exceptioncode*/) -{ - // ### TODO : title should be set, and media could have wrong syntax, in which case we should - // generate an exception. - CSSStyleSheetImpl *parent = 0L; - CSSStyleSheetImpl *sheet = new CSSStyleSheetImpl(parent, DOMString()); - sheet->setMedia(new MediaListImpl(sheet, media)); - return sheet; -} - -DocumentImpl *DOMImplementationImpl::createDocument( KHTMLView *v ) -{ - return new DocumentImpl(this, v); -} - -HTMLDocumentImpl *DOMImplementationImpl::createHTMLDocument( KHTMLView *v ) -{ - return new HTMLDocumentImpl(this, v); -} - -DOMImplementationImpl *DOMImplementationImpl::instance() -{ - if (!m_instance) { - m_instance = new DOMImplementationImpl(); - m_instance->ref(); - } - - return m_instance; -} - -// ------------------------------------------------------------------------ - - -ElementMappingCache::ElementMappingCache():m_dict(257) -{ - m_dict.setAutoDelete(true); -} - -void ElementMappingCache::add(const TQString& id, ElementImpl* nd) -{ - if (id.isEmpty()) return; - - ItemInfo* info = m_dict.find(id); - if (info) - { - info->ref++; - info->nd = 0; //Now ambigous - } - else - { - ItemInfo* info = new ItemInfo(); - info->ref = 1; - info->nd = nd; - m_dict.insert(id, info); - } -} - -void ElementMappingCache::set(const TQString& id, ElementImpl* nd) -{ - if (id.isEmpty()) return; - - ItemInfo* info = m_dict.find(id); - info->nd = nd; -} - -void ElementMappingCache::remove(const TQString& id, ElementImpl* nd) -{ - if (id.isEmpty()) return; - - ItemInfo* info = m_dict.find(id); - info->ref--; - if (info->ref == 0) - { - m_dict.take(id); - delete info; - } - else - { - if (info->nd == nd) - info->nd = 0; - } -} - -bool ElementMappingCache::contains(const TQString& id) -{ - if (id.isEmpty()) return false; - return m_dict.find(id); -} - -ElementMappingCache::ItemInfo* ElementMappingCache::get(const TQString& id) -{ - if (id.isEmpty()) return 0; - return m_dict.find(id); -} - -static KStaticDeleter< TQPtrList<DocumentImpl> > s_changedDocumentsDeleter; -TQPtrList<DocumentImpl> * DocumentImpl::changedDocuments; - -// KHTMLView might be 0 -DocumentImpl::DocumentImpl(DOMImplementationImpl *_implementation, KHTMLView *v) - : NodeBaseImpl( 0 ), m_domtree_version(0), m_counterDict(257), - m_imageLoadEventTimer(0) -{ - m_document.resetSkippingRef(this); //Make getDocument return us.. - m_selfOnlyRefCount = 0; - - m_paintDeviceMetrics = 0; - m_paintDevice = 0; - m_decoderMibEnum = 0; - m_textColor = Qt::black; - - m_view = v; - m_renderArena.reset(); - - KHTMLFactory::ref(); - - if ( v ) { - m_docLoader = new DocLoader(v->part(), this ); - setPaintDevice( TQT_TQPAINTDEVICE(m_view) ); - } - else - m_docLoader = new DocLoader( 0, this ); - - visuallyOrdered = false; - m_bParsing = false; - m_docChanged = false; - m_elemSheet = 0; - m_tokenizer = 0; - - // ### this should be created during parsing a <!DOCTYPE> - // not during construction. Not sure who added that and why (Dirk) - m_doctype = new DocumentTypeImpl(_implementation, getDocument(), - DOMString() /* qualifiedName */, - DOMString() /* publicId */, - DOMString() /* systemId */); - m_doctype->ref(); - - m_implementation = _implementation; - m_implementation->ref(); - pMode = Strict; - hMode = XHtml; - m_textColor = "#000000"; - m_attrMap = new IdNameMapping(ATTR_LAST_ATTR+1); - m_elementMap = new IdNameMapping(ID_LAST_TAG+1); - m_namespaceMap = new IdNameMapping(1); - TQString xhtml(XHTML_NAMESPACE); - m_namespaceMap->names.insert(emptyNamespace, new DOMStringImpl("")); - m_namespaceMap->names.insert(xhtmlNamespace, new DOMStringImpl(xhtml.unicode(), xhtml.length())); - m_namespaceMap->names[emptyNamespace]->ref(); - m_namespaceMap->names[xhtmlNamespace]->ref(); - m_namespaceMap->count+=2; - m_focusNode = 0; - m_hoverNode = 0; - m_activeNode = 0; - m_defaultView = new AbstractViewImpl(this); - m_defaultView->ref(); - m_listenerTypes = 0; - m_styleSheets = new StyleSheetListImpl; - m_styleSheets->ref(); - m_addedStyleSheets = 0; - m_inDocument = true; - m_styleSelectorDirty = false; - m_styleSelector = 0; - m_counterDict.setAutoDelete(true); - - m_inStyleRecalc = false; - m_pendingStylesheets = 0; - m_ignorePendingStylesheets = false; - m_async = true; - m_hadLoadError = false; - m_docLoading = false; - m_inSyncLoad = false; - m_loadingXMLDoc = 0; - m_cssTarget = 0; - m_dynamicDomRestyler = new khtml::DynamicDomRestyler(); -} - -void DocumentImpl::removedLastRef() -{ - if (m_selfOnlyRefCount) { - /* In this case, the only references to us are from children, - so we have a cycle. We'll try to break it by disconnecting the - children from us; this sucks/is wrong, but it's pretty much - the best we can do without tracing. - - Of course, if dumping the children causes the refcount from them to - drop to 0 we can get killed right here, so better hold - a temporary reference, too - */ - DocPtr<DocumentImpl> guard(this); - - // we must make sure not to be retaining any of our children through - // these extra pointers or we will create a reference cycle - if (m_doctype) { - m_doctype->deref(); - m_doctype = 0; - } - - if (m_cssTarget) { - m_cssTarget->deref(); - m_cssTarget = 0; - } - - if (m_focusNode) { - m_focusNode->deref(); - m_focusNode = 0; - } - - if (m_hoverNode) { - m_hoverNode->deref(); - m_hoverNode = 0; - } - - if (m_activeNode) { - m_activeNode->deref(); - m_activeNode = 0; - } - - removeChildren(); - - delete m_tokenizer; - m_tokenizer = 0; - } else { - delete this; - } -} - -DocumentImpl::~DocumentImpl() -{ - //Important: if you need to remove stuff here, - //you may also have to fix removedLastRef() above - M.O. - assert( !m_render ); - - TQIntDictIterator<NodeListImpl::Cache> it(m_nodeListCache); - for (; it.current(); ++it) - it.current()->deref(); - - if (m_loadingXMLDoc) - m_loadingXMLDoc->deref(this); - if (changedDocuments && m_docChanged) - changedDocuments->remove(this); - delete m_tokenizer; - m_document.resetSkippingRef(0); - delete m_styleSelector; - delete m_docLoader; - if (m_elemSheet ) m_elemSheet->deref(); - if (m_doctype) - m_doctype->deref(); - m_implementation->deref(); - delete m_paintDeviceMetrics; - delete m_elementMap; - delete m_attrMap; - delete m_namespaceMap; - delete m_dynamicDomRestyler; - m_defaultView->deref(); - m_styleSheets->deref(); - if (m_addedStyleSheets) - m_addedStyleSheets->deref(); - if (m_cssTarget) - m_cssTarget->deref(); - if (m_focusNode) - m_focusNode->deref(); - if ( m_hoverNode ) - m_hoverNode->deref(); - if (m_activeNode) - m_activeNode->deref(); - - m_renderArena.reset(); - - KHTMLFactory::deref(); -} - - -DocumentTypeImpl *DocumentImpl::doctype() const -{ - return m_doctype; -} - -DOMImplementationImpl *DocumentImpl::implementation() const -{ - return m_implementation; -} - -ElementImpl *DocumentImpl::documentElement() const -{ - NodeImpl *n = firstChild(); - while (n && n->nodeType() != Node::ELEMENT_NODE) - n = n->nextSibling(); - return static_cast<ElementImpl*>(n); -} - -ElementImpl *DocumentImpl::createElement( const DOMString &name, int* pExceptioncode ) -{ - Id id = getId( NodeImpl::ElementId, name.implementation(), - false /* allocate */, false /*HTMLDocumentImpl::createElement looked for HTML elements already*/, - pExceptioncode); - if ( pExceptioncode && *pExceptioncode ) - return 0; - - XMLElementImpl* e = new XMLElementImpl( getDocument(), id ); - e->setHTMLCompat( htmlMode() != XHtml ); // Not a real HTML element, but inside an html-compat doc all tags are uppercase. - return e; -} - -AttrImpl *DocumentImpl::createAttribute( const DOMString &tagName, int* pExceptioncode ) -{ - Id id = getId( NodeImpl::AttributeId, tagName.implementation(), - false /* allocate */, isHTMLDocument(), pExceptioncode); - if ( pExceptioncode && *pExceptioncode ) - return 0; - AttrImpl* attr = new AttrImpl( 0, getDocument(), id, DOMString("").implementation()); - attr->setHTMLCompat( htmlMode() != XHtml ); - return attr; -} - -DocumentFragmentImpl *DocumentImpl::createDocumentFragment( ) -{ - return new DocumentFragmentImpl( docPtr() ); -} - -CommentImpl *DocumentImpl::createComment ( DOMStringImpl* data ) -{ - return new CommentImpl( docPtr(), data ); -} - -CDATASectionImpl *DocumentImpl::createCDATASection ( DOMStringImpl* data ) -{ - return new CDATASectionImpl( docPtr(), data ); -} - -ProcessingInstructionImpl *DocumentImpl::createProcessingInstruction ( const DOMString &target, DOMStringImpl* data ) -{ - return new ProcessingInstructionImpl( docPtr(),target,data); -} - -EntityReferenceImpl *DocumentImpl::createEntityReference ( const DOMString &name ) -{ - return new EntityReferenceImpl(docPtr(), name.implementation()); -} - -NodeImpl *DocumentImpl::importNode(NodeImpl *importedNode, bool deep, int &exceptioncode) -{ - NodeImpl *result = 0; - - // Not mentioned in spec: throw NOT_FOUND_ERR if evt is null - if (!importedNode) { - exceptioncode = DOMException::NOT_FOUND_ERR; - return 0; - } - - if(importedNode->nodeType() == Node::ELEMENT_NODE) - { - // Why not use cloneNode? - ElementImpl *otherElem = static_cast<ElementImpl*>(importedNode); - NamedAttrMapImpl *otherMap = static_cast<ElementImpl *>(importedNode)->attributes(true); - - ElementImpl *tempElementImpl; - if (!importedNode->localName().isNull()) - tempElementImpl = createElementNS(otherElem->namespaceURI(),otherElem->nodeName()); - else - tempElementImpl = createElement(otherElem->nodeName()); - result = tempElementImpl; - - - if(otherMap) { - for(unsigned long i = 0; i < otherMap->length(); i++) - { - AttrImpl *otherAttr = otherMap->attrAt(i)->createAttr(otherElem,otherElem->docPtr()); - - if (!otherAttr->localName().isNull()) { - // attr was created via createElementNS() - tempElementImpl->setAttributeNS(otherAttr->namespaceURI(), - otherAttr->name(), - otherAttr->nodeValue(), - exceptioncode); - } - else { - // attr was created via createElement() - tempElementImpl->setAttribute(otherAttr->id(), - otherAttr->nodeValue(), - otherAttr->name(), - exceptioncode); - } - - if(exceptioncode != 0) - break; // ### properly cleanup here - } - } - } - else if(importedNode->nodeType() == Node::TEXT_NODE) - { - result = createTextNode(static_cast<TextImpl*>(importedNode)->string()); - deep = false; - } - else if(importedNode->nodeType() == Node::CDATA_SECTION_NODE) - { - result = createCDATASection(static_cast<CDATASectionImpl*>(importedNode)->string()); - deep = false; - } - else if(importedNode->nodeType() == Node::ENTITY_REFERENCE_NODE) - result = createEntityReference(importedNode->nodeName()); - else if(importedNode->nodeType() == Node::PROCESSING_INSTRUCTION_NODE) - { - result = createProcessingInstruction(importedNode->nodeName(), importedNode->nodeValue().implementation()); - deep = false; - } - else if(importedNode->nodeType() == Node::COMMENT_NODE) - { - result = createComment(static_cast<CommentImpl*>(importedNode)->string()); - deep = false; - } - else if (importedNode->nodeType() == Node::DOCUMENT_FRAGMENT_NODE) - result = createDocumentFragment(); - else - exceptioncode = DOMException::NOT_SUPPORTED_ERR; - - //### FIXME: This should handle Attributes, and a few other things - - if(deep && result) - { - for(Node n = importedNode->firstChild(); !n.isNull(); n = n.nextSibling()) - result->appendChild(importNode(n.handle(), true, exceptioncode), exceptioncode); - } - - return result; -} - -ElementImpl *DocumentImpl::createElementNS( const DOMString &_namespaceURI, const DOMString &_qualifiedName, int* pExceptioncode ) -{ - ElementImpl *e = 0; - int colonPos = -2; - // check NAMESPACE_ERR/INVALID_CHARACTER_ERR - if (pExceptioncode && !checkQualifiedName(_qualifiedName, _namespaceURI, &colonPos, - false/*nameCanBeNull*/, false/*nameCanBeEmpty*/, - pExceptioncode)) - return 0; - DOMString prefix, localName; - splitPrefixLocalName(_qualifiedName.implementation(), prefix, localName, colonPos); - - if ((isHTMLDocument() && _namespaceURI.isNull()) || - (strcasecmp(_namespaceURI, XHTML_NAMESPACE) == 0 && localName == localName.lower())) { - e = createHTMLElement(localName); - if (e) { - int _exceptioncode = 0; - if (!prefix.isNull()) - e->setPrefix(prefix, _exceptioncode); - if ( _exceptioncode ) { - if ( pExceptioncode ) *pExceptioncode = _exceptioncode; - delete e; - return 0; - } - e->setHTMLCompat( _namespaceURI.isNull() && htmlMode() != XHtml ); - } - } - if (!e) { - Id id = getId(NodeImpl::ElementId, _namespaceURI.implementation(), prefix.implementation(), - localName.implementation(), false, false /*HTML already looked up*/); - e = new XMLElementImpl( getDocument(), id, prefix.implementation() ); - } - - return e; -} - -AttrImpl *DocumentImpl::createAttributeNS( const DOMString &_namespaceURI, - const DOMString &_qualifiedName, int* pExceptioncode) -{ - int colonPos = -2; - // check NAMESPACE_ERR/INVALID_CHARACTER_ERR - if (pExceptioncode && !checkQualifiedName(_qualifiedName, _namespaceURI, &colonPos, - false/*nameCanBeNull*/, false/*nameCanBeEmpty*/, - pExceptioncode)) - return 0; - DOMString prefix, localName; - splitPrefixLocalName(_qualifiedName.implementation(), prefix, localName, colonPos); - Id id = getId(NodeImpl::AttributeId, _namespaceURI.implementation(), prefix.implementation(), - localName.implementation(), false, true /*lookupHTML*/); - AttrImpl* attr = new AttrImpl(0, getDocument(), id, DOMString("").implementation(), - prefix.implementation()); - attr->setHTMLCompat( _namespaceURI.isNull() && htmlMode() != XHtml ); - return attr; -} - -ElementImpl *DocumentImpl::getElementById( const DOMString &elementId ) const -{ - TQString stringKey = elementId.string(); - - ElementMappingCache::ItemInfo* info = m_getElementByIdCache.get(stringKey); - - if (!info) - return 0; - - //See if cache has an unambiguous answer. - if (info->nd) - return info->nd; - - //Now we actually have to walk. - TQPtrStack<NodeImpl> nodeStack; - NodeImpl *current = _first; - - while(1) - { - if(!current) - { - if(nodeStack.isEmpty()) break; - current = nodeStack.pop(); - current = current->nextSibling(); - } - else - { - if(current->isElementNode()) - { - ElementImpl *e = static_cast<ElementImpl *>(current); - if(e->getAttribute(ATTR_ID) == elementId) { - info->nd = e; - return e; - } - } - - NodeImpl *child = current->firstChild(); - if(child) - { - nodeStack.push(current); - current = child; - } - else - { - current = current->nextSibling(); - } - } - } - - assert(0); //If there is no item with such an ID, we should never get here - - //kdDebug() << "WARNING: *DocumentImpl::getElementById not found " << elementId.string() << endl; - - return 0; -} - -void DocumentImpl::setTitle(const DOMString& _title) -{ - if (_title == m_title && !m_title.isNull()) return; - - m_title = _title; - - TQString titleStr = m_title.string(); - for (unsigned int i = 0; i < titleStr.length(); ++i) - if (titleStr[i] < ' ') - titleStr[i] = ' '; - titleStr = titleStr.simplifyWhiteSpace(); - titleStr.compose(); - if ( view() && !view()->part()->parentPart() ) { - if (titleStr.isNull() || titleStr.isEmpty()) { - // empty title... set window caption as the URL - KURL url = m_url; - url.setRef(TQString::null); - url.setQuery(TQString::null); - titleStr = url.prettyURL(); - } - - emit view()->part()->setWindowCaption( KStringHandler::csqueeze( titleStr, 128 ) ); - } -} - -DOMString DocumentImpl::nodeName() const -{ - return "#document"; -} - -unsigned short DocumentImpl::nodeType() const -{ - return Node::DOCUMENT_NODE; -} - -DOMStringImpl* DocumentImpl::textContent() const -{ - return 0; -} - -void DocumentImpl::setTextContent( const DOMString&, int& ) -{} - -ElementImpl *DocumentImpl::createHTMLElement( const DOMString &name ) -{ - uint id = khtml::getTagID( name.string().lower().latin1(), name.string().length() ); -// id = makeId(xhtmlNamespace, id); - - ElementImpl *n = 0; - switch(id) - { - case ID_HTML: - n = new HTMLHtmlElementImpl(docPtr()); - break; - case ID_HEAD: - n = new HTMLHeadElementImpl(docPtr()); - break; - case ID_BODY: - n = new HTMLBodyElementImpl(docPtr()); - break; - -// head elements - case ID_BASE: - n = new HTMLBaseElementImpl(docPtr()); - break; - case ID_LINK: - n = new HTMLLinkElementImpl(docPtr()); - break; - case ID_META: - n = new HTMLMetaElementImpl(docPtr()); - break; - case ID_STYLE: - n = new HTMLStyleElementImpl(docPtr()); - break; - case ID_TITLE: - n = new HTMLTitleElementImpl(docPtr()); - break; - -// frames - case ID_FRAME: - n = new HTMLFrameElementImpl(docPtr()); - break; - case ID_FRAMESET: - n = new HTMLFrameSetElementImpl(docPtr()); - break; - case ID_IFRAME: - n = new HTMLIFrameElementImpl(docPtr()); - break; - -// form elements -// ### FIXME: we need a way to set form dependency after we have made the form elements - case ID_FORM: - n = new HTMLFormElementImpl(docPtr(), false); - break; - case ID_BUTTON: - n = new HTMLButtonElementImpl(docPtr()); - break; - case ID_FIELDSET: - n = new HTMLFieldSetElementImpl(docPtr()); - break; - case ID_INPUT: - n = new HTMLInputElementImpl(docPtr()); - break; - case ID_ISINDEX: - n = new HTMLIsIndexElementImpl(docPtr()); - break; - case ID_LABEL: - n = new HTMLLabelElementImpl(docPtr()); - break; - case ID_LEGEND: - n = new HTMLLegendElementImpl(docPtr()); - break; - case ID_OPTGROUP: - n = new HTMLOptGroupElementImpl(docPtr()); - break; - case ID_OPTION: - n = new HTMLOptionElementImpl(docPtr()); - break; - case ID_SELECT: - n = new HTMLSelectElementImpl(docPtr()); - break; - case ID_TEXTAREA: - n = new HTMLTextAreaElementImpl(docPtr()); - break; - -// lists - case ID_DL: - n = new HTMLDListElementImpl(docPtr()); - break; - case ID_DD: - n = new HTMLGenericElementImpl(docPtr(), id); - break; - case ID_DT: - n = new HTMLGenericElementImpl(docPtr(), id); - break; - case ID_UL: - n = new HTMLUListElementImpl(docPtr()); - break; - case ID_OL: - n = new HTMLOListElementImpl(docPtr()); - break; - case ID_DIR: - n = new HTMLDirectoryElementImpl(docPtr()); - break; - case ID_MENU: - n = new HTMLMenuElementImpl(docPtr()); - break; - case ID_LI: - n = new HTMLLIElementImpl(docPtr()); - break; - -// formatting elements (block) - case ID_DIV: - case ID_P: - n = new HTMLDivElementImpl( docPtr(), id ); - break; - case ID_BLOCKQUOTE: - case ID_H1: - case ID_H2: - case ID_H3: - case ID_H4: - case ID_H5: - case ID_H6: - n = new HTMLGenericElementImpl(docPtr(), id); - break; - case ID_HR: - n = new HTMLHRElementImpl(docPtr()); - break; - case ID_PLAINTEXT: - case ID_XMP: - case ID_PRE: - n = new HTMLPreElementImpl(docPtr(), id); - break; - -// font stuff - case ID_BASEFONT: - n = new HTMLBaseFontElementImpl(docPtr()); - break; - case ID_FONT: - n = new HTMLFontElementImpl(docPtr()); - break; - -// ins/del - case ID_DEL: - case ID_INS: - n = new HTMLGenericElementImpl(docPtr(), id); - break; - -// anchor - case ID_A: - n = new HTMLAnchorElementImpl(docPtr()); - break; - -// images - case ID_IMG: - n = new HTMLImageElementImpl(docPtr()); - break; - case ID_MAP: - n = new HTMLMapElementImpl(docPtr()); - /*n = map;*/ - break; - case ID_AREA: - n = new HTMLAreaElementImpl(docPtr()); - break; - -// objects, applets and scripts - case ID_APPLET: - n = new HTMLAppletElementImpl(docPtr()); - break; - case ID_OBJECT: - n = new HTMLObjectElementImpl(docPtr()); - break; - case ID_EMBED: - n = new HTMLEmbedElementImpl(docPtr()); - break; - case ID_PARAM: - n = new HTMLParamElementImpl(docPtr()); - break; - case ID_SCRIPT: - n = new HTMLScriptElementImpl(docPtr()); - break; - -// tables - case ID_TABLE: - n = new HTMLTableElementImpl(docPtr()); - break; - case ID_CAPTION: - n = new HTMLTableCaptionElementImpl(docPtr()); - break; - case ID_COLGROUP: - case ID_COL: - n = new HTMLTableColElementImpl(docPtr(), id); - break; - case ID_TR: - n = new HTMLTableRowElementImpl(docPtr()); - break; - case ID_TD: - case ID_TH: - n = new HTMLTableCellElementImpl(docPtr(), id); - break; - case ID_THEAD: - case ID_TBODY: - case ID_TFOOT: - n = new HTMLTableSectionElementImpl(docPtr(), id, false); - break; - -// inline elements - case ID_BR: - n = new HTMLBRElementImpl(docPtr()); - break; - case ID_Q: - n = new HTMLGenericElementImpl(docPtr(), id); - break; - -// elements with no special representation in the DOM - -// block: - case ID_ADDRESS: - case ID_CENTER: - n = new HTMLGenericElementImpl(docPtr(), id); - break; -// inline - // %fontstyle - case ID_TT: - case ID_U: - case ID_B: - case ID_I: - case ID_S: - case ID_STRIKE: - case ID_BIG: - case ID_SMALL: - - // %phrase - case ID_EM: - case ID_STRONG: - case ID_DFN: - case ID_CODE: - case ID_SAMP: - case ID_KBD: - case ID_VAR: - case ID_CITE: - case ID_ABBR: - case ID_ACRONYM: - - // %special - case ID_SUB: - case ID_SUP: - case ID_SPAN: - case ID_NOBR: - case ID_WBR: - case ID_BDO: - case ID_NOFRAMES: - n = new HTMLGenericElementImpl(docPtr(), id); - break; - - case ID_MARQUEE: - n = new HTMLMarqueeElementImpl(docPtr()); - break; -// text - case ID_TEXT: - kdDebug( 6020 ) << "Use document->createTextNode()" << endl; - break; - - default: - break; - } - return n; -} - -TQString DocumentImpl::nextState() -{ - TQString state; - if (!m_state.isEmpty()) - { - state = m_state.first(); - m_state.remove(m_state.begin()); - } - return state; -} - -TQStringList DocumentImpl::docState() -{ - TQStringList s; - for (TQPtrListIterator<NodeImpl> it(m_maintainsState); it.current(); ++it) - s.append(it.current()->state()); - - return s; -} - -bool DocumentImpl::unsubmittedFormChanges() -{ - for (TQPtrListIterator<NodeImpl> it(m_maintainsState); it.current(); ++it) - if (it.current()->state().right(1)=="M") - return true; - - return false; -} - -RangeImpl *DocumentImpl::createRange() -{ - return new RangeImpl( docPtr() ); -} - -NodeIteratorImpl *DocumentImpl::createNodeIterator(NodeImpl *root, unsigned long whatToShow, - NodeFilter &filter, bool entityReferenceExpansion, - int &exceptioncode) -{ - if (!root) { - exceptioncode = DOMException::NOT_SUPPORTED_ERR; - return 0; - } - - return new NodeIteratorImpl(root,whatToShow,filter,entityReferenceExpansion); -} - -TreeWalkerImpl *DocumentImpl::createTreeWalker(NodeImpl *root, unsigned long whatToShow, NodeFilterImpl *filter, - bool entityReferenceExpansion, int &exceptioncode) -{ - if (!root) { - exceptioncode = DOMException::NOT_SUPPORTED_ERR; - return 0; - } - - return new TreeWalkerImpl( root, whatToShow, filter, entityReferenceExpansion ); -} - -void DocumentImpl::setDocumentChanged(bool b) -{ - if (!changedDocuments) - changedDocuments = s_changedDocumentsDeleter.setObject( changedDocuments, new TQPtrList<DocumentImpl>() ); - - if (b && !m_docChanged) - changedDocuments->append(this); - else if (!b && m_docChanged) - changedDocuments->remove(this); - m_docChanged = b; -} - -void DocumentImpl::recalcStyle( StyleChange change ) -{ -// tqDebug("recalcStyle(%p)", this); -// TQTime qt; -// qt.start(); - if (m_inStyleRecalc) - return; // Guard against re-entrancy. -dwh - - m_inStyleRecalc = true; - - if( !m_render ) goto bail_out; - - if ( change == Force ) { - RenderStyle* oldStyle = m_render->style(); - if ( oldStyle ) oldStyle->ref(); - RenderStyle* _style = new RenderStyle(); - _style->setDisplay(BLOCK); - _style->setVisuallyOrdered( visuallyOrdered ); - // ### make the font stuff _really_ work!!!! - - khtml::FontDef fontDef; - TQFont f = TDEGlobalSettings::generalFont(); - fontDef.family = f.family(); - fontDef.italic = f.italic(); - fontDef.weight = f.weight(); - if (m_view) { - const KHTMLSettings *settings = m_view->part()->settings(); - TQString stdfont = settings->stdFontName(); - if ( !stdfont.isEmpty() ) - fontDef.family = stdfont; - - fontDef.size = m_styleSelector->fontSizes()[3]; - } - - //kdDebug() << "DocumentImpl::attach: setting to charset " << settings->charset() << endl; - _style->setFontDef(fontDef); - _style->htmlFont().update( paintDeviceMetrics() ); - if ( inCompatMode() ) - _style->setHtmlHacks(true); // enable html specific rendering tricks - - StyleChange ch = diff( _style, oldStyle ); - if(m_render && ch != NoChange) - m_render->setStyle(_style); - else - delete _style; - if ( change != Force ) - change = ch; - - if (oldStyle) - oldStyle->deref(); - } - - NodeImpl *n; - for (n = _first; n; n = n->nextSibling()) - if ( change>= Inherit || n->hasChangedChild() || n->changed() ) - n->recalcStyle( change ); - //kdDebug( 6020 ) << "TIME: recalcStyle() dt=" << qt.elapsed() << endl; - - if (changed() && m_view) - m_view->layout(); - -bail_out: - setChanged( false ); - setHasChangedChild( false ); - setDocumentChanged( false ); - - m_inStyleRecalc = false; -} - -void DocumentImpl::updateRendering() -{ - if (!hasChangedChild()) return; - -// TQTime time; -// time.start(); -// kdDebug() << "UPDATERENDERING: "<<endl; - - StyleChange change = NoChange; -#if 0 - if ( m_styleSelectorDirty ) { - recalcStyleSelector(); - change = Force; - } -#endif - recalcStyle( change ); - -// kdDebug() << "UPDATERENDERING time used="<<time.elapsed()<<endl; -} - -void DocumentImpl::updateDocumentsRendering() -{ - if (!changedDocuments) - return; - - while ( !changedDocuments->isEmpty() ) { - changedDocuments->first(); - DocumentImpl* it = changedDocuments->take(); - if (it->isDocumentChanged()) - it->updateRendering(); - } -} - -void DocumentImpl::updateLayout() -{ - bool oldIgnore = m_ignorePendingStylesheets; - - if (!haveStylesheetsLoaded()) { - m_ignorePendingStylesheets = true; - updateStyleSelector(); - } - - updateRendering(); - - // Only do a layout if changes have occurred that make it necessary. - if (m_view && renderer() && renderer()->needsLayout()) - m_view->layout(); - - m_ignorePendingStylesheets = oldIgnore; -} - -void DocumentImpl::attach() -{ - assert(!attached()); - - if ( m_view ) - setPaintDevice( TQT_TQPAINTDEVICE(m_view) ); - - if (!m_renderArena) - m_renderArena.reset(new RenderArena()); - - // Create the rendering tree - assert(!m_styleSelector); - m_styleSelector = new CSSStyleSelector( this, m_usersheet, m_styleSheets, m_url, - !inCompatMode() ); - m_render = new (m_renderArena.get()) RenderCanvas(this, m_view); - m_styleSelector->computeFontSizes(paintDeviceMetrics(), m_view ? m_view->part()->zoomFactor() : 100); - recalcStyle( Force ); - - RenderObject* render = m_render; - m_render = 0; - - NodeBaseImpl::attach(); - m_render = render; -} - -void DocumentImpl::detach() -{ - RenderObject* render = m_render; - - // indicate destruction mode, i.e. attached() but m_render == 0 - m_render = 0; - - delete m_tokenizer; - m_tokenizer = 0; - - // Empty out these lists as a performance optimization - m_imageLoadEventDispatchSoonList.clear(); - m_imageLoadEventDispatchingList.clear(); - NodeBaseImpl::detach(); - - if ( render ) - render->detach(); - - m_view = 0; - - m_renderArena.reset(); -} - -void DocumentImpl::setVisuallyOrdered() -{ - visuallyOrdered = true; - if (m_render) - m_render->style()->setVisuallyOrdered(true); -} - -void DocumentImpl::setSelection(NodeImpl* s, int sp, NodeImpl* e, int ep) -{ - if ( m_render ) - static_cast<RenderCanvas*>(m_render)->setSelection(s->renderer(),sp,e->renderer(),ep); -} - -void DocumentImpl::clearSelection() -{ - if ( m_render ) - static_cast<RenderCanvas*>(m_render)->clearSelection(); -} - -khtml::Tokenizer *DocumentImpl::createTokenizer() -{ - return new khtml::XMLTokenizer(docPtr(),m_view); -} - -void DocumentImpl::setPaintDevice( TQPaintDevice *dev ) -{ - if (m_paintDevice != dev) { - m_paintDevice = dev; - delete m_paintDeviceMetrics; - m_paintDeviceMetrics = new TQPaintDeviceMetrics( dev ); - } -} - -void DocumentImpl::open( bool clearEventListeners ) -{ - if (parsing()) return; - - if (m_tokenizer) - close(); - - delete m_tokenizer; - m_tokenizer = 0; - - KHTMLView* view = m_view; - bool was_attached = attached(); - if ( was_attached ) - detach(); - - removeChildren(); - delete m_styleSelector; - m_styleSelector = 0; - m_view = view; - if ( was_attached ) - attach(); - - if (clearEventListeners) - m_windowEventListeners.clear(); - - m_tokenizer = createTokenizer(); - m_decoderMibEnum = 0; - connect(m_tokenizer,TQT_SIGNAL(finishedParsing()),this,TQT_SIGNAL(finishedParsing())); - m_tokenizer->begin(); -} - -HTMLElementImpl* DocumentImpl::body() -{ - NodeImpl *de = documentElement(); - if (!de) - return 0; - - // try to prefer a FRAMESET element over BODY - NodeImpl* body = 0; - for (NodeImpl* i = de->firstChild(); i; i = i->nextSibling()) { - if (i->id() == ID_FRAMESET) - return static_cast<HTMLElementImpl*>(i); - - if (i->id() == ID_BODY) - body = i; - } - return static_cast<HTMLElementImpl *>(body); -} - -void DocumentImpl::close( ) -{ - if (parsing() || !m_tokenizer) return; - - if ( m_render ) - m_render->close(); - - // on an explicit document.close(), the tokenizer might still be waiting on scripts, - // and in that case we don't want to destroy it because that will prevent the - // scripts from getting processed. - if (m_tokenizer && !m_tokenizer->isWaitingForScripts() && !m_tokenizer->isExecutingScript()) { - delete m_tokenizer; - m_tokenizer = 0; - } - - if (m_view) - m_view->part()->checkEmitLoadEvent(); -} - -void DocumentImpl::write( const DOMString &text ) -{ - write(text.string()); -} - -void DocumentImpl::write( const TQString &text ) -{ - if (!m_tokenizer) { - open(); - if (m_view) - m_view->part()->resetFromScript(); - m_tokenizer->setAutoClose(); - write(TQString::fromLatin1("<html>")); - } - m_tokenizer->write(text, false); -} - -void DocumentImpl::writeln( const DOMString &text ) -{ - write(text); - write(DOMString("\n")); -} - -void DocumentImpl::finishParsing ( ) -{ - if(m_tokenizer) - m_tokenizer->finish(); -} - -void DocumentImpl::setUserStyleSheet( const TQString& sheet ) -{ - if ( m_usersheet != sheet ) { - m_usersheet = sheet; - updateStyleSelector(); - } -} - -CSSStyleSheetImpl* DocumentImpl::elementSheet() -{ - if (!m_elemSheet) { - m_elemSheet = new CSSStyleSheetImpl(this, baseURL().url() ); - m_elemSheet->ref(); - } - return m_elemSheet; -} - -void DocumentImpl::determineParseMode( const TQString &/*str*/ ) -{ - // For XML documents, use strict parse mode - pMode = Strict; - hMode = XHtml; - kdDebug(6020) << " using strict parseMode" << endl; -} - -NodeImpl *DocumentImpl::nextFocusNode(NodeImpl *fromNode) -{ - unsigned short fromTabIndex; - - if (!fromNode) { - // No starting node supplied; begin with the top of the document - NodeImpl *n; - - int lowestTabIndex = 65535; - for (n = this; n != 0; n = n->traverseNextNode()) { - if (n->isTabFocusable()) { - if ((n->tabIndex() > 0) && (n->tabIndex() < lowestTabIndex)) - lowestTabIndex = n->tabIndex(); - } - } - - if (lowestTabIndex == 65535) - lowestTabIndex = 0; - - // Go to the first node in the document that has the desired tab index - for (n = this; n != 0; n = n->traverseNextNode()) { - if (n->isTabFocusable() && (n->tabIndex() == lowestTabIndex)) - return n; - } - - return 0; - } - else { - fromTabIndex = fromNode->tabIndex(); - } - - if (fromTabIndex == 0) { - // Just need to find the next selectable node after fromNode (in document order) that doesn't have a tab index - NodeImpl *n = fromNode->traverseNextNode(); - while (n && !(n->isTabFocusable() && n->tabIndex() == 0)) - n = n->traverseNextNode(); - return n; - } - else { - // Find the lowest tab index out of all the nodes except fromNode, that is greater than or equal to fromNode's - // tab index. For nodes with the same tab index as fromNode, we are only interested in those that come after - // fromNode in document order. - // If we don't find a suitable tab index, the next focus node will be one with a tab index of 0. - unsigned short lowestSuitableTabIndex = 65535; - NodeImpl *n; - - bool reachedFromNode = false; - for (n = this; n != 0; n = n->traverseNextNode()) { - if (n->isTabFocusable() && - ((reachedFromNode && (n->tabIndex() >= fromTabIndex)) || - (!reachedFromNode && (n->tabIndex() > fromTabIndex))) && - (n->tabIndex() < lowestSuitableTabIndex) && - (n != fromNode)) { - - // We found a selectable node with a tab index at least as high as fromNode's. Keep searching though, - // as there may be another node which has a lower tab index but is still suitable for use. - lowestSuitableTabIndex = n->tabIndex(); - } - - if (n == fromNode) - reachedFromNode = true; - } - - if (lowestSuitableTabIndex == 65535) { - // No next node with a tab index -> just take first node with tab index of 0 - NodeImpl *n = this; - while (n && !(n->isTabFocusable() && n->tabIndex() == 0)) - n = n->traverseNextNode(); - return n; - } - - // Search forwards from fromNode - for (n = fromNode->traverseNextNode(); n != 0; n = n->traverseNextNode()) { - if (n->isTabFocusable() && (n->tabIndex() == lowestSuitableTabIndex)) - return n; - } - - // The next node isn't after fromNode, start from the beginning of the document - for (n = this; n != fromNode; n = n->traverseNextNode()) { - if (n->isTabFocusable() && (n->tabIndex() == lowestSuitableTabIndex)) - return n; - } - - assert(false); // should never get here - return 0; - } -} - -NodeImpl *DocumentImpl::previousFocusNode(NodeImpl *fromNode) -{ - NodeImpl *lastNode = this; - while (lastNode->lastChild()) - lastNode = lastNode->lastChild(); - - if (!fromNode) { - // No starting node supplied; begin with the very last node in the document - NodeImpl *n; - - int highestTabIndex = 0; - for (n = lastNode; n != 0; n = n->traversePreviousNode()) { - if (n->isTabFocusable()) { - if (n->tabIndex() == 0) - return n; - else if (n->tabIndex() > highestTabIndex) - highestTabIndex = n->tabIndex(); - } - } - - // No node with a tab index of 0; just go to the last node with the highest tab index - for (n = lastNode; n != 0; n = n->traversePreviousNode()) { - if (n->isTabFocusable() && (n->tabIndex() == highestTabIndex)) - return n; - } - - return 0; - } - else { - unsigned short fromTabIndex = fromNode->tabIndex(); - - if (fromTabIndex == 0) { - // Find the previous selectable node before fromNode (in document order) that doesn't have a tab index - NodeImpl *n = fromNode->traversePreviousNode(); - while (n && !(n->isTabFocusable() && n->tabIndex() == 0)) - n = n->traversePreviousNode(); - if (n) - return n; - - // No previous nodes with a 0 tab index, go to the last node in the document that has the highest tab index - int highestTabIndex = 0; - for (n = this; n != 0; n = n->traverseNextNode()) { - if (n->isTabFocusable() && (n->tabIndex() > highestTabIndex)) - highestTabIndex = n->tabIndex(); - } - - if (highestTabIndex == 0) - return 0; - - for (n = lastNode; n != 0; n = n->traversePreviousNode()) { - if (n->isTabFocusable() && (n->tabIndex() == highestTabIndex)) - return n; - } - - assert(false); // should never get here - return 0; - } - else { - // Find the lowest tab index out of all the nodes except fromNode, that is less than or equal to fromNode's - // tab index. For nodes with the same tab index as fromNode, we are only interested in those before - // fromNode. - // If we don't find a suitable tab index, then there will be no previous focus node. - unsigned short highestSuitableTabIndex = 0; - NodeImpl *n; - - bool reachedFromNode = false; - for (n = this; n != 0; n = n->traverseNextNode()) { - if (n->isTabFocusable() && - ((!reachedFromNode && (n->tabIndex() <= fromTabIndex)) || - (reachedFromNode && (n->tabIndex() < fromTabIndex))) && - (n->tabIndex() > highestSuitableTabIndex) && - (n != fromNode)) { - - // We found a selectable node with a tab index no higher than fromNode's. Keep searching though, as - // there may be another node which has a higher tab index but is still suitable for use. - highestSuitableTabIndex = n->tabIndex(); - } - - if (n == fromNode) - reachedFromNode = true; - } - - if (highestSuitableTabIndex == 0) { - // No previous node with a tab index. Since the order specified by HTML is nodes with tab index > 0 - // first, this means that there is no previous node. - return 0; - } - - // Search backwards from fromNode - for (n = fromNode->traversePreviousNode(); n != 0; n = n->traversePreviousNode()) { - if (n->isTabFocusable() && (n->tabIndex() == highestSuitableTabIndex)) - return n; - } - // The previous node isn't before fromNode, start from the end of the document - for (n = lastNode; n != fromNode; n = n->traversePreviousNode()) { - if (n->isTabFocusable() && (n->tabIndex() == highestSuitableTabIndex)) - return n; - } - - assert(false); // should never get here - return 0; - } - } -} - -ElementImpl* DocumentImpl::findAccessKeyElement(TQChar c) -{ - c = c.upper(); - for( NodeImpl* n = this; - n != NULL; - n = n->traverseNextNode()) { - if( n->isElementNode()) { - ElementImpl* en = static_cast< ElementImpl* >( n ); - DOMString s = en->getAttribute( ATTR_ACCESSKEY ); - if( s.length() == 1 - && s[ 0 ].upper() == c ) - return en; - } - } - return NULL; -} - -int DocumentImpl::nodeAbsIndex(NodeImpl *node) -{ - assert(node->getDocument() == this); - - int absIndex = 0; - for (NodeImpl *n = node; n && n != this; n = n->traversePreviousNode()) - absIndex++; - return absIndex; -} - -NodeImpl *DocumentImpl::nodeWithAbsIndex(int absIndex) -{ - NodeImpl *n = this; - for (int i = 0; n && (i < absIndex); i++) { - n = n->traverseNextNode(); - } - return n; -} - -void DocumentImpl::processHttpEquiv(const DOMString &equiv, const DOMString &content) -{ - assert(!equiv.isNull() && !content.isNull()); - - KHTMLView *v = getDocument()->view(); - - if(strcasecmp(equiv, "refresh") == 0 && v && v->part()->metaRefreshEnabled()) - { - // get delay and url - TQString str = content.string().stripWhiteSpace(); - int pos = str.find(TQRegExp("[;,]")); - if ( pos == -1 ) - pos = str.find(TQRegExp("[ \t]")); - - bool ok = false; - int delay = kMax( 0, content.implementation()->toInt(&ok) ); - if ( !ok && str.length() && str[0] == '.' ) - ok = true; - - if (pos == -1) // There can be no url (David) - { - if(ok) - v->part()->scheduleRedirection(delay, v->part()->url().url() ); - } else { - pos++; - while(pos < (int)str.length() && str[pos].isSpace()) pos++; - str = str.mid(pos); - if(str.find("url", 0, false ) == 0) str = str.mid(3); - str = str.stripWhiteSpace(); - if ( str.length() && str[0] == '=' ) str = str.mid( 1 ).stripWhiteSpace(); - while(str.length() && - (str[str.length()-1] == ';' || str[str.length()-1] == ',')) - str.setLength(str.length()-1); - str = parseURL( DOMString(str) ).string(); - TQString newURL = getDocument()->completeURL( str ); - if ( ok ) - v->part()->scheduleRedirection(delay, getDocument()->completeURL( str ), delay < 2 || newURL == URL().url()); - } - } - else if(strcasecmp(equiv, "expires") == 0) - { - bool relative = false; - TQString str = content.string().stripWhiteSpace(); - time_t expire_date = KRFCDate::parseDate(str); - if (!expire_date) - { - expire_date = str.toULong(); - relative = true; - } - if (!expire_date) - expire_date = 1; // expire now - if (m_docLoader) - m_docLoader->setExpireDate(expire_date, relative); - } - else if(v && (strcasecmp(equiv, "pragma") == 0 || strcasecmp(equiv, "cache-control") == 0)) - { - TQString str = content.string().lower().stripWhiteSpace(); - KURL url = v->part()->url(); - if ((str == "no-cache") && url.protocol().startsWith("http")) - { - TDEIO::http_update_cache(url, true, 0); - } - } - else if( (strcasecmp(equiv, "set-cookie") == 0)) - { - // ### make setCookie work on XML documents too; e.g. in case of <html:meta .....> - HTMLDocumentImpl *d = static_cast<HTMLDocumentImpl *>(this); - d->setCookie(content); - } - else if (strcasecmp(equiv, "default-style") == 0) { - // HTML 4.0 14.3.2 - // http://www.hixie.ch/tests/evil/css/import/main/preferred.html - m_preferredStylesheetSet = content; - updateStyleSelector(); - } - else if (strcasecmp(equiv, "content-language") == 0) { - m_contentLanguage = content.string(); - } -} - -bool DocumentImpl::prepareMouseEvent( bool readonly, int _x, int _y, MouseEvent *ev ) -{ - if ( m_render ) { - assert(m_render->isCanvas()); - RenderObject::NodeInfo renderInfo(readonly, ev->type == MousePress); - bool isInside = m_render->layer()->nodeAtPoint(renderInfo, _x, _y); - ev->innerNode = renderInfo.innerNode(); - ev->innerNonSharedNode = renderInfo.innerNonSharedNode(); - - if (renderInfo.URLElement()) { - assert(renderInfo.URLElement()->isElementNode()); - //tqDebug("urlnode: %s (%d)", getTagName(renderInfo.URLElement()->id()).string().latin1(), renderInfo.URLElement()->id()); - - ElementImpl* e = static_cast<ElementImpl*>(renderInfo.URLElement()); - DOMString href = khtml::parseURL(e->getAttribute(ATTR_HREF)); - DOMString target = e->getAttribute(ATTR_TARGET); - - if (!target.isNull() && !href.isNull()) { - ev->target = target; - ev->url = href; - } - else - ev->url = href; - } - - if (!readonly) - updateRendering(); - - return isInside; - } - - - return false; -} - - -// DOM Section 1.1.1 -bool DocumentImpl::childTypeAllowed( unsigned short type ) -{ - switch (type) { - case Node::ATTRIBUTE_NODE: - case Node::CDATA_SECTION_NODE: - case Node::DOCUMENT_FRAGMENT_NODE: - case Node::DOCUMENT_NODE: - case Node::ENTITY_NODE: - case Node::ENTITY_REFERENCE_NODE: - case Node::NOTATION_NODE: - case Node::TEXT_NODE: -// case Node::XPATH_NAMESPACE_NODE: - return false; - case Node::COMMENT_NODE: - case Node::PROCESSING_INSTRUCTION_NODE: - return true; - case Node::DOCUMENT_TYPE_NODE: - case Node::ELEMENT_NODE: - // Documents may contain no more than one of each of these. - // (One Element and one DocumentType.) - for (NodeImpl* c = firstChild(); c; c = c->nextSibling()) - if (c->nodeType() == type) - return false; - return true; - } - return false; -} - -NodeImpl *DocumentImpl::cloneNode ( bool /*deep*/ ) -{ - // Spec says cloning Document nodes is "implementation dependent" - // so we do not support it... - return 0; -} - - -typedef const char* (*NameLookupFunction)(unsigned short id); -typedef int (*IdLookupFunction)(const char *tagStr, int len); - -NodeImpl::Id DocumentImpl::getId( NodeImpl::IdType _type, DOMStringImpl* _nsURI, DOMStringImpl *_prefix, - DOMStringImpl *_name, bool readonly, bool /*lookupHTML*/, int *pExceptioncode) -{ - /*kdDebug() << "DocumentImpl::getId( type: " << _type << ", uri: " << DOMString(_nsURI).string() - << ", prefix: " << DOMString(_prefix).string() << ", name: " << DOMString(_name).string() - << ", readonly: " << readonly - << ", lookupHTML: " << lookupHTML - << ", exceptions: " << (pExceptioncode ? "yes" : "no") - << " )" << endl;*/ - - if(!_name) return 0; - IdNameMapping *map; - IdLookupFunction lookup; - - switch (_type) { - case NodeImpl::ElementId: - map = m_elementMap; - lookup = getTagID; - break; - case NodeImpl::AttributeId: - map = m_attrMap; - lookup = getAttrID; - break; - case NodeImpl::NamespaceId: - if( strcasecmp(_name, XHTML_NAMESPACE) == 0) - return xhtmlNamespace; - if( _name->l == 0) - return emptyNamespace; - // defaultNamespace handled by "if (!_name) return 0" - map = m_namespaceMap; - lookup = 0; - break; - default: - return 0; - } - // Names and attributes with "" - if (_name->l == 0) return 0; - - NodeImpl::Id id, nsid = 0; - TQConstString n(_name->s, _name->l); - bool cs = true; // case sensitive - if (_type != NodeImpl::NamespaceId) { - if (_nsURI) - nsid = getId( NodeImpl::NamespaceId, 0, 0, _nsURI, false, false, 0 ) << 16; - - // Each document maintains a mapping of tag name -> id for every tag name encountered - // in the document. - cs = (htmlMode() == XHtml) || (_nsURI && _type != NodeImpl::AttributeId); - - // First see if it's a HTML element name - // xhtml is lower case - case sensitive, easy to implement - if ( cs && (id = lookup(n.string().ascii(), _name->l)) ) { - map->addAlias(_prefix, _name, cs, id); - return nsid + id; - } - // compatibility: upper case - case insensitive - if ( !cs && (id = lookup(n.string().lower().ascii(), _name->l )) ) { - map->addAlias(_prefix, _name, cs, id); - return nsid + id; - } - } - - // Look in the names array for the name - // compatibility mode has to lookup upper case - TQString name = cs ? n.string() : n.string().upper(); - - if (!_nsURI) { - id = (NodeImpl::Id)(long) map->ids.find( name ); - if (!id && _type != NodeImpl::NamespaceId) { - id = (NodeImpl::Id)(long) map->ids.find( "aliases: " + name ); - } - } else { - id = (NodeImpl::Id)(long) map->ids.find( name ); - if (!readonly && id && _prefix && _prefix->l) { - // we were called in registration mode... check if the alias exists - TQConstString px( _prefix->s, _prefix->l ); - TQString qn("aliases: " + (cs ? px.string() : px.string().upper()) + ":" + name); - if (!map->ids.find( qn )) { - map->ids.insert( qn, (void*)id ); - } - } - } - - if (id) return nsid + id; - - // unknown - if (readonly) return 0; - - if ( pExceptioncode && _type != NodeImpl::NamespaceId && !Element::khtmlValidQualifiedName(_name)) { - *pExceptioncode = DOMException::INVALID_CHARACTER_ERR; - return 0; - } - - // Name not found, so let's add it - NodeImpl::Id cid = map->count++ + map->idStart; - map->names.insert( cid, _name ); - _name->ref(); - - map->ids.insert( name, (void*)cid ); - - // and register an alias if needed for DOM1 methods compatibility - map->addAlias(_prefix, _name, cs, cid); - - return nsid + cid; - } - -NodeImpl::Id DocumentImpl::getId( NodeImpl::IdType _type, DOMStringImpl *_nodeName, bool readonly, bool lookupHTML, int *pExceptioncode) -{ - return getId(_type, 0, 0, _nodeName, readonly, lookupHTML, pExceptioncode); -} - -DOMString DocumentImpl::getName( NodeImpl::IdType _type, NodeImpl::Id _id ) const -{ - IdNameMapping *map; - NameLookupFunction lookup; - bool hasNS = (namespacePart(_id) != defaultNamespace); - switch (_type) { - case NodeImpl::ElementId: - map = m_elementMap; - lookup = getTagName; - break; - case NodeImpl::AttributeId: - map = m_attrMap; - lookup = getAttrName; - break; - case NodeImpl::NamespaceId: - if( _id == xhtmlNamespace ) - return XHTML_NAMESPACE; - else - if( _id == emptyNamespace ) - return DOMString(""); - else - if ( _id == defaultNamespace ) - return DOMString(); - map = m_namespaceMap; - lookup = 0; - break; - default: - return DOMString();; - } - _id = localNamePart(_id) ; - if (_id >= map->idStart) { - return map->names[_id]; - } - else if (lookup) { - // ### put them in a cache - if (hasNS) - return DOMString(lookup(_id)).lower(); - else - return lookup(_id); - } else - return DOMString(); -} - -// This method is called whenever a top-level stylesheet has finished loading. -void DocumentImpl::styleSheetLoaded() -{ - // Make sure we knew this sheet was pending, and that our count isn't out of sync. - assert(m_pendingStylesheets > 0); - - m_pendingStylesheets--; - updateStyleSelector(); -} - -DOMString DocumentImpl::selectedStylesheetSet() const -{ - if (!view()) return DOMString(); - - return view()->part()->d->m_sheetUsed; -} - -void DocumentImpl::setSelectedStylesheetSet(const DOMString& s) -{ - // this code is evil - if (view() && view()->part()->d->m_sheetUsed != s.string()) { - view()->part()->d->m_sheetUsed = s.string(); - updateStyleSelector(); - } -} - -void DocumentImpl::addStyleSheet(StyleSheetImpl *sheet, int *exceptioncode) -{ - int excode = 0; - - if (!m_addedStyleSheets) { - m_addedStyleSheets = new StyleSheetListImpl; - m_addedStyleSheets->ref(); - } - - m_addedStyleSheets->add(sheet); - if (sheet->isCSSStyleSheet()) updateStyleSelector(); - - if (exceptioncode) *exceptioncode = excode; -} - -void DocumentImpl::removeStyleSheet(StyleSheetImpl *sheet, int *exceptioncode) -{ - int excode = 0; - bool removed = false; - bool is_css = sheet->isCSSStyleSheet(); - - if (m_addedStyleSheets) { - bool in_main_list = !sheet->hasOneRef(); - removed = m_addedStyleSheets->styleSheets.removeRef(sheet); - sheet->deref(); - - if (m_addedStyleSheets->styleSheets.count() == 0) { - bool reset = m_addedStyleSheets->hasOneRef(); - m_addedStyleSheets->deref(); - if (reset) m_addedStyleSheets = 0; - } - - // remove from main list, too - if (in_main_list) m_styleSheets->remove(sheet); - } - - if (removed) { - if (is_css) updateStyleSelector(); - } else - excode = DOMException::NOT_FOUND_ERR; - - if (exceptioncode) *exceptioncode = excode; -} - -void DocumentImpl::updateStyleSelector(bool shallow) -{ -// kdDebug() << "PENDING " << m_pendingStylesheets << endl; - - // Don't bother updating, since we haven't loaded all our style info yet. - if (m_pendingStylesheets > 0) - return; - - if (shallow) - rebuildStyleSelector(); - else - recalcStyleSelector(); - recalcStyle(Force); -#if 0 - - m_styleSelectorDirty = true; -#endif - if ( renderer() ) - renderer()->setNeedsLayoutAndMinMaxRecalc(); -} - -void DocumentImpl::recalcStyleSelector() -{ - if ( !m_render || !attached() ) return; - - assert(m_pendingStylesheets==0); - - TQPtrList<StyleSheetImpl> oldStyleSheets = m_styleSheets->styleSheets; - m_styleSheets->styleSheets.clear(); - TQString sheetUsed = view() ? view()->part()->d->m_sheetUsed.replace("&&", "&") : TQString(); - bool autoselect = sheetUsed.isEmpty(); - if (autoselect && !m_preferredStylesheetSet.isEmpty()) - sheetUsed = m_preferredStylesheetSet.string(); - NodeImpl *n; - for (int i=0 ; i<2 ; i++) { - m_availableSheets.clear(); - m_availableSheets << i18n("Basic Page Style"); - bool canResetSheet = false; - - for (n = this; n; n = n->traverseNextNode()) { - StyleSheetImpl *sheet = 0; - - if (n->nodeType() == Node::PROCESSING_INSTRUCTION_NODE) - { - // Processing instruction (XML documents only) - ProcessingInstructionImpl* pi = static_cast<ProcessingInstructionImpl*>(n); - sheet = pi->sheet(); - if (!sheet && !pi->localHref().isEmpty()) - { - // Processing instruction with reference to an element in this document - e.g. - // <?xml-stylesheet href="#mystyle">, with the element - // <foo id="mystyle">heading { color: red; }</foo> at some location in - // the document - ElementImpl* elem = getElementById(pi->localHref()); - if (elem) { - DOMString sheetText(""); - NodeImpl *c; - for (c = elem->firstChild(); c; c = c->nextSibling()) { - if (c->nodeType() == Node::TEXT_NODE || c->nodeType() == Node::CDATA_SECTION_NODE) - sheetText += c->nodeValue(); - } - - CSSStyleSheetImpl *cssSheet = new CSSStyleSheetImpl(this); - cssSheet->parseString(sheetText); - pi->setStyleSheet(cssSheet); - sheet = cssSheet; - } - } - - } - else if (n->isHTMLElement() && ( n->id() == ID_LINK || n->id() == ID_STYLE) ) { - TQString title; - if ( n->id() == ID_LINK ) { - HTMLLinkElementImpl* l = static_cast<HTMLLinkElementImpl*>(n); - if (l->isCSSStyleSheet()) { - sheet = l->sheet(); - - if (sheet || l->isLoading() || l->isAlternate() ) - title = l->getAttribute(ATTR_TITLE).string(); - - if ((autoselect || title != sheetUsed) && l->isDisabled()) { - sheet = 0; - } else if (!title.isEmpty() && !l->isAlternate() && sheetUsed.isEmpty()) { - sheetUsed = title; - l->setDisabled(false); - } - } - } - else { - // <STYLE> element - HTMLStyleElementImpl* s = static_cast<HTMLStyleElementImpl*>(n); - if (!s->isLoading()) { - sheet = s->sheet(); - if (sheet) title = s->getAttribute(ATTR_TITLE).string(); - } - if (!title.isEmpty() && sheetUsed.isEmpty()) - sheetUsed = title; - } - - if ( !title.isEmpty() ) { - if ( title != sheetUsed ) - sheet = 0; // don't use it - - title = title.replace('&', "&&"); - - if ( !m_availableSheets.contains( title ) ) - m_availableSheets.append( title ); - } - } - else if (n->isHTMLElement() && n->id() == ID_BODY) { - // <BODY> element (doesn't contain styles as such but vlink="..." and friends - // are treated as style declarations) - sheet = static_cast<HTMLBodyElementImpl*>(n)->sheet(); - } - - if (sheet) { - sheet->ref(); - m_styleSheets->styleSheets.append(sheet); - } - - // For HTML documents, stylesheets are not allowed within/after the <BODY> tag. So we - // can stop searching here. - if (isHTMLDocument() && n->id() == ID_BODY) { - canResetSheet = !canResetSheet; - break; - } - } - - // we're done if we don't select an alternative sheet - // or we found the sheet we selected - if (sheetUsed.isEmpty() || - (!canResetSheet && tokenizer()) || - m_availableSheets.contains(sheetUsed)) { - break; - } - - // the alternative sheet we used doesn't exist anymore - // so try from scratch again - if (view()) - view()->part()->d->m_sheetUsed = TQString::null; - if (!m_preferredStylesheetSet.isEmpty() && !(sheetUsed == m_preferredStylesheetSet)) - sheetUsed = m_preferredStylesheetSet.string(); - else - sheetUsed = TQString::null; - autoselect = true; - } - - // Include programmatically added style sheets - if (m_addedStyleSheets) { - TQPtrListIterator<StyleSheetImpl> it = m_addedStyleSheets->styleSheets; - for (; *it; ++it) { - if ((*it)->isCSSStyleSheet() && !(*it)->disabled()) - m_styleSheets->add(*it); - } - } - - // De-reference all the stylesheets in the old list - TQPtrListIterator<StyleSheetImpl> it(oldStyleSheets); - for (; it.current(); ++it) - it.current()->deref(); - - rebuildStyleSelector(); -} - -void DocumentImpl::rebuildStyleSelector() -{ - // Create a new style selector - delete m_styleSelector; - TQString usersheet = m_usersheet; - if ( m_view && m_view->mediaType() == "print" ) - usersheet += m_printSheet; - m_styleSelector = new CSSStyleSelector( this, usersheet, m_styleSheets, m_url, - !inCompatMode() ); - - m_styleSelectorDirty = false; -} - -void DocumentImpl::setHoverNode(NodeImpl *newHoverNode) -{ - NodeImpl* oldHoverNode = m_hoverNode; - if (newHoverNode ) newHoverNode->ref(); - m_hoverNode = newHoverNode; - if ( oldHoverNode ) oldHoverNode->deref(); -} - -void DocumentImpl::setActiveNode(NodeImpl* newActiveNode) -{ - NodeImpl* oldActiveNode = m_activeNode; - if (newActiveNode ) newActiveNode->ref(); - m_activeNode = newActiveNode; - if ( oldActiveNode ) oldActiveNode->deref(); -} - -void DocumentImpl::setFocusNode(NodeImpl *newFocusNode) -{ - // don't process focus changes while detaching - if( !m_render ) return; - - // We do want to blur if a widget is being detached, - // but we don't want to emit events since that - // triggers updateLayout() and may recurse detach() - bool widgetDetach = m_focusNode && m_focusNode != this && - m_focusNode->renderer() && !m_focusNode->renderer()->parent(); - - // Make sure newFocusNode is actually in this document - if (newFocusNode && (newFocusNode->getDocument() != this)) - return; - - if (m_focusNode != newFocusNode) { - NodeImpl *oldFocusNode = m_focusNode; - // Set focus on the new node - m_focusNode = newFocusNode; - // Remove focus from the existing focus node (if any) - if (oldFocusNode) { - if (oldFocusNode->active()) - oldFocusNode->setActive(false); - - oldFocusNode->setFocus(false); - - if (!widgetDetach) { - oldFocusNode->dispatchHTMLEvent(EventImpl::BLUR_EVENT,false,false); - oldFocusNode->dispatchUIEvent(EventImpl::DOMFOCUSOUT_EVENT); - } - if ((oldFocusNode == this) && oldFocusNode->hasOneRef()) { - oldFocusNode->deref(); // deletes this - return; - } - else { - oldFocusNode->deref(); - } - } - - if (m_focusNode) { - m_focusNode->ref(); - m_focusNode->dispatchHTMLEvent(EventImpl::FOCUS_EVENT,false,false); - if (m_focusNode != newFocusNode) return; - m_focusNode->dispatchUIEvent(EventImpl::DOMFOCUSIN_EVENT); - if (m_focusNode != newFocusNode) return; - m_focusNode->setFocus(); - if (m_focusNode != newFocusNode) return; - - // eww, I suck. set the qt focus correctly - // ### find a better place in the code for this - if (view()) { - if (!m_focusNode->renderer() || !m_focusNode->renderer()->isWidget()) - view()->setFocus(); - else if (static_cast<RenderWidget*>(m_focusNode->renderer())->widget()) - { - if (view()->isVisible()) - static_cast<RenderWidget*>(m_focusNode->renderer())->widget()->setFocus(); - } - } - } else { - //We're blurring. Better clear the Qt focus/give it to the view... - if (view()) - view()->setFocus(); - } - - if (!widgetDetach) - updateRendering(); - } -} - -void DocumentImpl::setCSSTarget(NodeImpl* n) -{ - if (n == m_cssTarget) - return; - - if (m_cssTarget) { - m_cssTarget->setChanged(); - m_cssTarget->deref(); - } - m_cssTarget = n; - if (n) { - n->setChanged(); - n->ref(); - } -} - -void DocumentImpl::attachNodeIterator(NodeIteratorImpl *ni) -{ - m_nodeIterators.append(ni); -} - -void DocumentImpl::detachNodeIterator(NodeIteratorImpl *ni) -{ - m_nodeIterators.remove(ni); -} - -void DocumentImpl::notifyBeforeNodeRemoval(NodeImpl *n) -{ - TQPtrListIterator<NodeIteratorImpl> it(m_nodeIterators); - for (; it.current(); ++it) - it.current()->notifyBeforeNodeRemoval(n); -} - -bool DocumentImpl::isURLAllowed(const TQString& url) const -{ - KHTMLPart *thisPart = part(); - - KURL newURL(completeURL(url)); - newURL.setRef(TQString::null); - - if (KHTMLFactory::defaultHTMLSettings()->isAdFiltered( newURL.url() )) - return false; - - // Prohibit non-file URLs if we are asked to. - if (!thisPart || thisPart->onlyLocalReferences() && newURL.protocol() != "file" && newURL.protocol() != "data") - return false; - - // do we allow this suburl ? - if ( !kapp || (newURL.protocol() != "javascript" && !kapp->authorizeURLAction("redirect", thisPart->url(), newURL)) ) - return false; - - // We allow one level of self-reference because some sites depend on that. - // But we don't allow more than one. - bool foundSelfReference = false; - for (KHTMLPart *part = thisPart; part; part = part->parentPart()) { - KURL partURL = part->url(); - partURL.setRef(TQString::null); - if (partURL == newURL) { - if (foundSelfReference) - return false; - foundSelfReference = true; - } - } - - return true; -} - -void DocumentImpl::setDesignMode(bool b) -{ - if (part()) - part()->setEditable(b); -} - -bool DocumentImpl::designMode() const -{ - return part() ? part()->isEditable() : false; -} - -EventImpl *DocumentImpl::createEvent(const DOMString &eventType, int &exceptioncode) -{ - if (eventType == "UIEvents" || eventType == "UIEvent") - return new UIEventImpl(); - else if (eventType == "MouseEvents" || eventType == "MouseEvent") - return new MouseEventImpl(); - else if (eventType == "TextEvent") - return new TextEventImpl(); - else if (eventType == "KeyboardEvent") - return new KeyboardEventImpl(); - else if (eventType == "MutationEvents" || eventType == "MutationEvent") - return new MutationEventImpl(); - else if (eventType == "HTMLEvents" || eventType == "Events" || - eventType == "HTMLEvent" || eventType == "Event") - return new EventImpl(); - else { - exceptioncode = DOMException::NOT_SUPPORTED_ERR; - return 0; - } -} - -CSSStyleDeclarationImpl *DocumentImpl::getOverrideStyle(ElementImpl* /*elt*/, DOMStringImpl* /*pseudoElt*/) -{ - return 0; // ### -} - -void DocumentImpl::abort() -{ - if (m_inSyncLoad) { - m_inSyncLoad = false; - kapp->exit_loop(); - } - - if (m_loadingXMLDoc) - m_loadingXMLDoc->deref(this); - m_loadingXMLDoc = 0; -} - -void DocumentImpl::load(const DOMString &uri) -{ - if (m_inSyncLoad) { - m_inSyncLoad = false; - kapp->exit_loop(); - } - - m_hadLoadError = false; - if (m_loadingXMLDoc) - m_loadingXMLDoc->deref(this); - - // Use the document loader to retrieve the XML file. We use CachedCSSStyleSheet because - // this is an easy way to retrieve an arbitrary text file... it is not specific to - // stylesheets. - - // ### Note: By loading the XML document this way we do not get the proper decoding - // of the data retrieved from the server based on the character set, as happens with - // HTML files. Need to look into a way of using the decoder in CachedCSSStyleSheet. - m_docLoading = true; - m_loadingXMLDoc = m_docLoader->requestStyleSheet(uri.string(),TQString(),"text/xml"); - - if (!m_loadingXMLDoc) { - m_docLoading = false; - return; - } - - m_loadingXMLDoc->ref(this); - - if (!m_async && m_docLoading) { - m_inSyncLoad = true; - kapp->enter_loop(); - } -} - -void DocumentImpl::loadXML(const DOMString &source) -{ - open(false); - write(source); - finishParsing(); - close(); - dispatchHTMLEvent(EventImpl::LOAD_EVENT,false,false); -} - -void DocumentImpl::setStyleSheet(const DOM::DOMString &url, const DOM::DOMString &sheet, const DOM::DOMString &charset) -{ - if (!m_hadLoadError) { - m_url = url.string(); - loadXML(sheet); - } - - m_docLoading = false; - if (m_inSyncLoad) { - m_inSyncLoad = false; - kapp->exit_loop(); - } - - assert(m_loadingXMLDoc != 0); - m_loadingXMLDoc->deref(this); - m_loadingXMLDoc = 0; -} - -void DocumentImpl::error(int err, const TQString &text) -{ - m_docLoading = false; - if (m_inSyncLoad) { - m_inSyncLoad = false; - kapp->exit_loop(); - } - - m_hadLoadError = true; - - int exceptioncode = 0; - EventImpl *evt = new EventImpl(EventImpl::ERROR_EVENT,false,false); - if (err != 0) - evt->setMessage(TDEIO::buildErrorString(err,text)); - else - evt->setMessage(text); - evt->ref(); - dispatchEvent(evt,exceptioncode,true); - evt->deref(); - - assert(m_loadingXMLDoc != 0); - m_loadingXMLDoc->deref(this); - m_loadingXMLDoc = 0; -} - -void DocumentImpl::defaultEventHandler(EventImpl *evt) -{ - // if any html event listeners are registered on the window, then dispatch them here - if (!m_windowEventListeners.listeners || evt->propagationStopped()) - return; - - TQValueList<RegisteredEventListener>::iterator it; - - //Grab a copy in case of clear - TQValueList<RegisteredEventListener> listeners = *m_windowEventListeners.listeners; - Event ev(evt); - for (it = listeners.begin(); it != listeners.end(); ++it) { - //Check to make sure it didn't get removed. KDE4: use Java-style iterators - if (!m_windowEventListeners.stillContainsListener(*it)) - continue; - - if ((*it).id == evt->id()) { - // currentTarget must be 0 in khtml for kjs_events to set "this" correctly. - // (this is how we identify events dispatched to the window, like window.onmousedown) - // ## currentTarget is unimplemented in IE, and is "window" in Mozilla (how? not a DOM node) - evt->setCurrentTarget(0); - (*it).listener->handleEvent(ev); - } - } -} - -void DocumentImpl::setHTMLWindowEventListener(int id, EventListener *listener) -{ - m_windowEventListeners.setHTMLEventListener(id, listener); -} - -EventListener *DocumentImpl::getHTMLWindowEventListener(int id) -{ - return m_windowEventListeners.getHTMLEventListener(id); -} - -void DocumentImpl::addWindowEventListener(int id, EventListener *listener, const bool useCapture) -{ - m_windowEventListeners.addEventListener(id, listener, useCapture); -} - -void DocumentImpl::removeWindowEventListener(int id, EventListener *listener, bool useCapture) -{ - m_windowEventListeners.removeEventListener(id, listener, useCapture); -} - -bool DocumentImpl::hasWindowEventListener(int id) -{ - return m_windowEventListeners.hasEventListener(id); -} - -EventListener *DocumentImpl::createHTMLEventListener(const TQString& code, const TQString& name, NodeImpl* node) -{ - return part() ? part()->createHTMLEventListener(code, name, node) : 0; -} - -void DocumentImpl::dispatchImageLoadEventSoon(HTMLImageElementImpl *image) -{ - m_imageLoadEventDispatchSoonList.append(image); - if (!m_imageLoadEventTimer) { - m_imageLoadEventTimer = startTimer(0); - } -} - -void DocumentImpl::removeImage(HTMLImageElementImpl *image) -{ - // Remove instances of this image from both lists. - // Use loops because we allow multiple instances to get into the lists. - while (m_imageLoadEventDispatchSoonList.removeRef(image)) { } - while (m_imageLoadEventDispatchingList.removeRef(image)) { } - if (m_imageLoadEventDispatchSoonList.isEmpty() && m_imageLoadEventTimer) { - killTimer(m_imageLoadEventTimer); - m_imageLoadEventTimer = 0; - } -} - -void DocumentImpl::dispatchImageLoadEventsNow() -{ - // need to avoid re-entering this function; if new dispatches are - // scheduled before the parent finishes processing the list, they - // will set a timer and eventually be processed - if (!m_imageLoadEventDispatchingList.isEmpty()) { - return; - } - - if (m_imageLoadEventTimer) { - killTimer(m_imageLoadEventTimer); - m_imageLoadEventTimer = 0; - } - - m_imageLoadEventDispatchingList = m_imageLoadEventDispatchSoonList; - m_imageLoadEventDispatchSoonList.clear(); - for (TQPtrListIterator<HTMLImageElementImpl> it(m_imageLoadEventDispatchingList); it.current(); ) { - HTMLImageElementImpl* image = it.current(); - // Must advance iterator *before* dispatching call. - // Otherwise, it might be advanced automatically if dispatching the call had a side effect - // of destroying the current HTMLImageElementImpl, and then we would advance past the *next* - // item, missing one altogether. - ++it; - image->dispatchLoadEvent(); - } - m_imageLoadEventDispatchingList.clear(); -} - -void DocumentImpl::timerEvent(TQTimerEvent *) -{ - dispatchImageLoadEventsNow(); -} - -void DocumentImpl::setDecoderCodec(const TQTextCodec *codec) -{ - m_decoderMibEnum = codec->mibEnum(); -} - -ElementImpl *DocumentImpl::ownerElement() const -{ - KHTMLPart *childPart = part(); - if (!childPart) - return 0; - ChildFrame *childFrame = childPart->d->m_frame; - if (!childFrame) - return 0; - RenderPart *renderPart = childFrame->m_frame; - if (!renderPart) - return 0; - return static_cast<ElementImpl *>(renderPart->element()); -} - -DOMString DocumentImpl::domain() const -{ - if ( m_domain.isEmpty() ) // not set yet (we set it on demand to save time and space) - m_domain = URL().host(); // Initially set to the host - return m_domain; -} - -void DocumentImpl::setDomain(const DOMString &newDomain) -{ - if ( m_domain.isEmpty() ) // not set yet (we set it on demand to save time and space) - m_domain = URL().host().lower(); // Initially set to the host - - if ( m_domain.isEmpty() /*&& part() && part()->openedByJS()*/ ) - m_domain = newDomain.lower(); - - // Both NS and IE specify that changing the domain is only allowed when - // the new domain is a suffix of the old domain. - int oldLength = m_domain.length(); - int newLength = newDomain.length(); - if ( newLength < oldLength ) // e.g. newDomain=kde.org (7) and m_domain=www.kde.org (11) - { - DOMString test = m_domain.copy(); - DOMString reference = newDomain.lower(); - if ( test[oldLength - newLength - 1] == '.' ) // Check that it's a subdomain, not e.g. "de.org" - { - test.remove( 0, oldLength - newLength ); // now test is "kde.org" from m_domain - if ( test == reference ) // and we check that it's the same thing as newDomain - m_domain = reference; - } - } -} - -DOMString DocumentImpl::toString() const -{ - DOMString result; - - for (NodeImpl *child = firstChild(); child != NULL; child = child->nextSibling()) { - result += child->toString(); - } - - return result; -} - - -KHTMLPart* DOM::DocumentImpl::part() const -{ - // ### TODO: make this independent from a KHTMLView one day. - return view() ? view()->part() : 0; -} - -NodeListImpl::Cache* DOM::DocumentImpl::acquireCachedNodeListInfo( - NodeListImpl::CacheFactory* factory, NodeImpl* base, int type) -{ - //### might want to flush the dict when the version number - //changes - NodeListImpl::CacheKey key(base, type); - - //Check to see if we have this sort of item cached. - NodeListImpl::Cache* cached = - (type == NodeListImpl::UNCACHEABLE) ? 0 : m_nodeListCache.find(key.hash()); - - if (cached) { - if (cached->key == key) { - cached->ref(); //Add the nodelist's reference - return cached; - } else { - //Conflict. Drop our reference to the old item. - cached->deref(); - } - } - - //Nothing to reuse, make a new item. - NodeListImpl::Cache* newInfo = factory(); - newInfo->key = key; - newInfo->clear(this); - newInfo->ref(); //Add the nodelist's reference - - if (type != NodeListImpl::UNCACHEABLE) { - newInfo->ref(); //Add the cache's reference - m_nodeListCache.replace(key.hash(), newInfo); - } - - return newInfo; -} - -void DOM::DocumentImpl::releaseCachedNodeListInfo(NodeListImpl::Cache* entry) -{ - entry->deref(); -} - -// ---------------------------------------------------------------------------- - -DocumentFragmentImpl::DocumentFragmentImpl(DocumentImpl *doc) : NodeBaseImpl(doc) -{ -} - -DocumentFragmentImpl::DocumentFragmentImpl(const DocumentFragmentImpl &other) - : NodeBaseImpl(other) -{ -} - -DOMString DocumentFragmentImpl::nodeName() const -{ - return "#document-fragment"; -} - -unsigned short DocumentFragmentImpl::nodeType() const -{ - return Node::DOCUMENT_FRAGMENT_NODE; -} - -// DOM Section 1.1.1 -bool DocumentFragmentImpl::childTypeAllowed( unsigned short type ) -{ - switch (type) { - case Node::ELEMENT_NODE: - case Node::PROCESSING_INSTRUCTION_NODE: - case Node::COMMENT_NODE: - case Node::TEXT_NODE: - case Node::CDATA_SECTION_NODE: - case Node::ENTITY_REFERENCE_NODE: - return true; - break; - default: - return false; - } -} - -DOMString DocumentFragmentImpl::toString() const -{ - DOMString result; - - for (NodeImpl *child = firstChild(); child != NULL; child = child->nextSibling()) { - result += child->toString(); - } - - return result; -} - -NodeImpl *DocumentFragmentImpl::cloneNode ( bool deep ) -{ - DocumentFragmentImpl *clone = new DocumentFragmentImpl( docPtr() ); - if (deep) - cloneChildNodes(clone); - return clone; -} - - -// ---------------------------------------------------------------------------- - -DocumentTypeImpl::DocumentTypeImpl(DOMImplementationImpl *implementation, DocumentImpl *doc, - const DOMString &qualifiedName, const DOMString &publicId, - const DOMString &systemId) - : NodeImpl(doc), m_implementation(implementation), - m_qualifiedName(qualifiedName), m_publicId(publicId), m_systemId(systemId) -{ - m_implementation->ref(); - - m_entities = 0; - m_notations = 0; - - // if doc is 0, it is not attached to a document and / or - // therefore does not provide entities or notations. (DOM Level 3) -} - -DocumentTypeImpl::~DocumentTypeImpl() -{ - m_implementation->deref(); - if (m_entities) - m_entities->deref(); - if (m_notations) - m_notations->deref(); -} - -void DocumentTypeImpl::copyFrom(const DocumentTypeImpl& other) -{ - m_qualifiedName = other.m_qualifiedName; - m_publicId = other.m_publicId; - m_systemId = other.m_systemId; - m_subset = other.m_subset; -} - -DOMString DocumentTypeImpl::toString() const -{ - DOMString result = "<!DOCTYPE"; - result += m_qualifiedName; - if (!m_publicId.isEmpty()) { - result += " PUBLIC \""; - result += m_publicId; - result += "\" \""; - result += m_systemId; - result += "\""; - } else if (!m_systemId.isEmpty()) { - result += " SYSTEM \""; - result += m_systemId; - result += "\""; - } - - if (!m_subset.isEmpty()) { - result += " ["; - result += m_subset; - result += "]"; - } - - result += ">"; - - return result; -} - -DOMString DocumentTypeImpl::nodeName() const -{ - return name(); -} - -unsigned short DocumentTypeImpl::nodeType() const -{ - return Node::DOCUMENT_TYPE_NODE; -} - -// DOM Section 1.1.1 -bool DocumentTypeImpl::childTypeAllowed( unsigned short /*type*/ ) -{ - return false; -} - -NodeImpl *DocumentTypeImpl::cloneNode ( bool /*deep*/ ) -{ - // Spec says cloning Document nodes is "implementation dependent" - // so we do not support it... - return 0; -} - -DOMStringImpl* DocumentTypeImpl::textContent() const -{ - return 0; -} - -void DocumentTypeImpl::setTextContent( const DOMString&, int& ) -{} - -NamedNodeMapImpl * DocumentTypeImpl::entities() const -{ - if ( !m_entities ) { - m_entities = new GenericRONamedNodeMapImpl( docPtr() ); - m_entities->ref(); - } - return m_entities; -} - -NamedNodeMapImpl * DocumentTypeImpl::notations() const -{ - if ( !m_notations ) { - m_notations = new GenericRONamedNodeMapImpl( docPtr() ); - m_notations->ref(); - } - return m_notations; -} - -#include "dom_docimpl.moc" |