void ElementImpl::setAttributeMap( NamedAttrMapImpl* list ) { if (inDocument()) getDocument()->incDOMTreeVersion(); // If setting the whole map changes the id attribute, we need to // call updateId. AttributeImpl *oldId = namedAttrMap ? namedAttrMap->getAttributeItem(ATTR_ID) : 0; AttributeImpl *newId = list ? list->getAttributeItem(ATTR_ID) : 0; if (oldId || newId) { updateId(oldId ? oldId->value() : nullAtom, newId ? newId->value() : nullAtom); } if(namedAttrMap) namedAttrMap->deref(); namedAttrMap = list; if(namedAttrMap) { namedAttrMap->ref(); namedAttrMap->element = this; unsigned int len = namedAttrMap->length(); for(unsigned int i = 0; i < len; i++) attributeChanged(namedAttrMap->attrs[i]); } }
NamedAttrMapImpl& NamedAttrMapImpl::operator=(const NamedAttrMapImpl& other) { // clone all attributes in the other map, but attach to our element if (!element) return *this; // If assigning the map changes the id attribute, we need to call // updateId. AttributeImpl *oldId = getAttributeItem(ATTR_ID); AttributeImpl *newId = other.getAttributeItem(ATTR_ID); if (oldId || newId) { element->updateId(oldId ? oldId->value() : nullAtom, newId ? newId->value() : nullAtom); } clearAttributes(); len = other.len; attrs = new AttributeImpl* [len]; // first initialize attrs vector, then call attributeChanged on it // this allows attributeChanged to use getAttribute for (uint i = 0; i < len; i++) { attrs[i] = other.attrs[i]->clone(); attrs[i]->ref(); } // FIXME: This is wasteful. The class list could be preserved on a copy, and we // wouldn't have to waste time reparsing the attribute. // The derived class, HTMLNamedAttrMapImpl, which manages a parsed class list for the CLASS attribute, // will update its member variable when parse attribute is called. for(uint i = 0; i < len; i++) element->attributeChanged(attrs[i], true); return *this; }
DOMString ElementImpl::openTagStartToString() const { DOMString result = DOMString("<") + tagName(); NamedAttrMapImpl *attrMap = attributes(true); if (attrMap) { unsigned long numAttrs = attrMap->length(); for (unsigned long i = 0; i < numAttrs; i++) { result += " "; AttributeImpl *attribute = attrMap->attributeItem(i); AttrImpl *attr = attribute->attrImpl(); if (attr) { result += attr->toString(); } else { result += getDocument()->attrName(attribute->id()); if (!attribute->value().isNull()) { result += "=\""; // FIXME: substitute entities for any instances of " or ' result += attribute->value(); result += "\""; } } } } return result; }
Node NamedAttrMapImpl::setNamedItem ( NodeImpl* arg, int &exceptioncode ) { if (!element) { exceptioncode = DOMException::NOT_FOUND_ERR; return 0; } // NO_MODIFICATION_ALLOWED_ERR: Raised if this map is readonly. if (isReadOnly()) { exceptioncode = DOMException::NO_MODIFICATION_ALLOWED_ERR; return 0; } // WRONG_DOCUMENT_ERR: Raised if arg was created from a different document than the one that created this map. if (arg->getDocument() != element->getDocument()) { exceptioncode = DOMException::WRONG_DOCUMENT_ERR; return 0; } // Not mentioned in spec: throw a HIERARCHY_REQUEST_ERROR if the user passes in a non-attribute node if (!arg->isAttributeNode()) { exceptioncode = DOMException::HIERARCHY_REQUEST_ERR; return 0; } AttrImpl *attr = static_cast<AttrImpl*>(arg); AttributeImpl* a = attr->attrImpl(); AttributeImpl* old = getAttributeItem(a->id()); if (old == a) return arg; // we know about it already // INUSE_ATTRIBUTE_ERR: Raised if arg is an Attr that is already an attribute of another Element object. // The DOM user must explicitly clone Attr nodes to re-use them in other elements. if (attr->ownerElement()) { exceptioncode = DOMException::INUSE_ATTRIBUTE_ERR; return 0; } if (a->id() == ATTR_ID) { element->updateId(old ? old->value() : nullAtom, a->value()); } // ### slightly inefficient - resizes attribute array twice. Node r; if (old) { if (!old->attrImpl()) old->allocateImpl(element); r = old->_impl; removeAttribute(a->id()); } addAttribute(a); return r; }
DOMString ElementImpl::openTagStartToString(bool expandurls) const { DOMString result = DOMString("<") + tagName(); NamedAttrMapImpl *attrMap = attributes(true); if(attrMap) { unsigned long numAttrs = attrMap->length(); for(unsigned long i = 0; i < numAttrs; i++) { result += " "; AttributeImpl *attribute = attrMap->attrAt(i); AttrImpl *attr = attribute->attr(); if(attr) { result += attr->toString(); } else { result += getDocument()->getName(NodeImpl::AttributeId, attribute->id()); if(!attribute->value().isNull()) { result += "=\""; // FIXME: substitute entities for any instances of " or ' // Expand out all urls, i.e. the src and href attributes if(expandurls && (attribute->id() == ATTR_SRC || attribute->id() == ATTR_HREF)) if(getDocument()) { // We need to sanitize the urls - strip out the passwords. // FIXME: are src= and href= the only places that might have a password and need to be sanitized? KURL safeURL(getDocument()->completeURL(attribute->value().string())); safeURL.setPass(QString::null); result += safeURL.htmlURL(); } else { kdWarning() << "getDocument() returned false"; result += attribute->value(); } else result += attribute->value(); result += "\""; } } } } return result; }
NodeImpl *KHTMLParser::handleIsindex( Token *t ) { NodeImpl *n; HTMLFormElementImpl *myform = form; if ( !myform ) { myform = new HTMLFormElementImpl(document); n = myform; } else n = new HTMLDivElementImpl( document ); NodeImpl *child = new HTMLHRElementImpl( document ); n->addChild( child ); AttributeImpl* a = t->attrs ? t->attrs->getAttributeItem(ATTR_PROMPT) : 0; #if APPLE_CHANGES && !KWIQ DOMString text = searchableIndexIntroduction(); #elif KWIQ // FIXME: KWIQ: Internationalization. Hardcoded string. DOMString text = "This is a searchable index. Enter search keywords: "; #else DOMString text = i18n("This is a searchable index. Enter search keywords: "); #endif if (a) text = DOMString(a->value()) + " "; child = new TextImpl(document, text); n->addChild( child ); child = new HTMLIsIndexElementImpl(document, myform); static_cast<ElementImpl *>(child)->setAttribute(ATTR_TYPE, "khtml_isindex"); n->addChild( child ); child = new HTMLHRElementImpl( document ); n->addChild( child ); return n; }
// The DOM2 spec doesn't say that removeAttribute[NS] throws NOT_FOUND_ERR // if the attribute is not found, but at this level we have to throw NOT_FOUND_ERR // because of removeNamedItem, removeNamedItemNS, and removeAttributeNode. Node NamedAttrMapImpl::removeNamedItem ( NodeImpl::Id id, int &exceptioncode ) { // ### should this really be raised when the attribute to remove isn't there at all? // NO_MODIFICATION_ALLOWED_ERR: Raised when the node is readonly if (isReadOnly()) { exceptioncode = DOMException::NO_MODIFICATION_ALLOWED_ERR; return Node(); } AttributeImpl* a = getAttributeItem(id); if (!a) { exceptioncode = DOMException::NOT_FOUND_ERR; return Node(); } if (!a->attrImpl()) a->allocateImpl(element); Node r(a->attrImpl()); if (id == ATTR_ID) { element->updateId(a->value(), nullAtom); } removeAttribute(id); return r; }
void ElementImpl::setAttribute(NodeImpl::Id id, DOMStringImpl* value, int &exceptioncode ) { if (inDocument()) getDocument()->incDOMTreeVersion(); // allocate attributemap if necessary AttributeImpl* old = attributes(false)->getAttributeItem(id); // NO_MODIFICATION_ALLOWED_ERR: Raised when the node is readonly if (namedAttrMap->isReadOnly()) { exceptioncode = DOMException::NO_MODIFICATION_ALLOWED_ERR; return; } if (id == ATTR_ID) { updateId(old ? old->value() : nullAtom, value); } if (old && !value) namedAttrMap->removeAttribute(id); else if (!old && value) namedAttrMap->addAttribute(createAttribute(id, value)); else if (old && value) { old->setValue(value); attributeChanged(old); } }
NodeImpl *KHTMLParser::handleIsindex( Token *t ) { NodeImpl *n; HTMLFormElementImpl *myform = form; if ( !myform ) { myform = new HTMLFormElementImpl(document, true); n = myform; } else n = new HTMLDivElementImpl( document, ID_DIV ); NodeImpl *child = new HTMLHRElementImpl( document ); n->addChild( child ); AttributeImpl* a = t->attrs ? t->attrs->getAttributeItem(ATTR_PROMPT) : 0; DOMString text = i18n("This is a searchable index. Enter search keywords: "); if (a) text = a->value(); child = new TextImpl(document, text.implementation()); n->addChild( child ); child = new HTMLIsIndexElementImpl(document, myform); static_cast<ElementImpl *>(child)->setAttribute(ATTR_TYPE, "khtml_isindex"); n->addChild( child ); child = new HTMLHRElementImpl( document ); n->addChild( child ); return n; }
const AtomicString& ElementImpl::getAttribute(NodeImpl::Id id) const { if (id == ATTR_STYLE) updateStyleAttributeIfNeeded(); if (namedAttrMap) { AttributeImpl* a = namedAttrMap->getAttributeItem(id); if (a) return a->value(); } return nullAtom; }
void ElementImpl::dump(QTextStream *stream, QString ind) const { updateStyleAttributeIfNeeded(); if (namedAttrMap) { for (uint i = 0; i < namedAttrMap->length(); i++) { AttributeImpl *attr = namedAttrMap->attributeItem(i); *stream << " " << DOMString(getDocument()->attrName(attr->id())).string().ascii() << "=\"" << DOMString(attr->value()).string().ascii() << "\""; } } NodeBaseImpl::dump(stream,ind); }
void ElementImpl::removedFromDocument() { if (hasID()) { NamedAttrMapImpl *attrs = attributes(true); if (attrs) { AttributeImpl *idAttr = attrs->getAttributeItem(ATTR_ID); if (idAttr && !idAttr->isNull()) { updateId(idAttr->value(), nullAtom); } } } NodeBaseImpl::removedFromDocument(); }
void ElementImpl::insertedIntoDocument() { // need to do superclass processing first so inDocument() is true // by the time we reach updateId NodeBaseImpl::insertedIntoDocument(); if (hasID()) { NamedAttrMapImpl *attrs = attributes(true); if (attrs) { AttributeImpl *idAttr = attrs->getAttributeItem(ATTR_ID); if (idAttr && !idAttr->isNull()) { updateId(nullAtom, idAttr->value()); } } } }
bool HTMLNamedAttrMapImpl::mapsEquivalent(const HTMLNamedAttrMapImpl* otherMap) const { // The # of decls must match. if (declCount() != otherMap->declCount()) return false; // The values for each decl must match. for (uint i = 0; i < length(); i++) { HTMLAttributeImpl* attr = attributeItem(i); if (attr->decl()) { AttributeImpl* otherAttr = otherMap->getAttributeItem(attr->id()); if (!otherAttr || (attr->value() != otherAttr->value())) return false; } } return true; }
void RenderPartObject::updateWidget() { QString url; QString serviceType; QStringList params; KHTMLPart *part = m_view->part(); setNeedsLayoutAndMinMaxRecalc(); if (element()->id() == ID_OBJECT) { HTMLObjectElementImpl *o = static_cast<HTMLObjectElementImpl *>(element()); // Check for a child EMBED tag. HTMLEmbedElementImpl *embed = 0; for (NodeImpl *child = o->firstChild(); child; child = child->nextSibling()) { if (child->id() == ID_EMBED) { embed = static_cast<HTMLEmbedElementImpl *>( child ); break; } } // Use the attributes from the EMBED tag instead of the OBJECT tag including WIDTH and HEIGHT. HTMLElementImpl *embedOrObject; if (embed) { embedOrObject = (HTMLElementImpl *)embed; DOMString attribute = embedOrObject->getAttribute(ATTR_WIDTH); if (!attribute.isEmpty()) { o->setAttribute(ATTR_WIDTH, attribute); } attribute = embedOrObject->getAttribute(ATTR_HEIGHT); if (!attribute.isEmpty()) { o->setAttribute(ATTR_HEIGHT, attribute); } url = embed->url; serviceType = embed->serviceType; } else { embedOrObject = (HTMLElementImpl *)o; } // If there was no URL or type defined in EMBED, try the OBJECT tag. if (url.isEmpty()) { url = o->url; } if (serviceType.isEmpty()) { serviceType = o->serviceType; } // Then try the PARAM tags for the URL and type attributes. NodeImpl *child = o->firstChild(); while (child && (url.isEmpty() || serviceType.isEmpty())) { if (child->id() == ID_PARAM) { HTMLParamElementImpl *p = static_cast<HTMLParamElementImpl *>( child ); QString name = p->name().lower(); if (url.isEmpty() && (QString::equals(name,"src") || QString::equals(name,"movie") || QString::equals(name,"code"))) { url = p->value(); } if (serviceType.isEmpty() && QString::equals(name,"type")) { serviceType = p->value(); } } child = child->nextSibling(); } // Lastly try to map a specific CLASSID to a type. if (serviceType.isEmpty() && !o->classId.isEmpty()) { if (o->classId.contains("D27CDB6E-AE6D-11cf-96B8-444553540000")) { // It is ActiveX, but the nsplugin system handling // should also work, that's why we don't override the // serviceType with application/x-activex-handler // but let the KTrader in khtmlpart::createPart() detect // the user's preference: launch with activex viewer or // with nspluginviewer (Niko) serviceType = "application/x-shockwave-flash"; } else if(o->classId.contains("CFCDAA03-8BE4-11cf-B84B-0020AFBBCCFA")) { serviceType = "audio/x-pn-realaudio-plugin"; } else if(o->classId.contains("02BF25D5-8C17-4B23-BC80-D3488ABDDC6B")) { serviceType = "video/quicktime"; } else if(o->classId.contains("166B1BCA-3F9C-11CF-8075-444553540000")) { serviceType = "application/x-director"; } else { // We have a clsid, means this is activex (Niko) serviceType = "application/x-activex-handler"; } // TODO: add more plugins here } // If no URL and type, abort. if (url.isEmpty() && serviceType.isEmpty()) { #ifdef DEBUG_LAYOUT kdDebug() << "RenderPartObject::close - empty url and serverType" << endl; #endif return; } // Turn the attributes of either the EMBED tag or OBJECT tag into an array. NamedAttrMapImpl* attributes = embedOrObject->attributes(); if (attributes) { for (unsigned long i = 0; i < attributes->length(); ++i) { AttributeImpl* it = attributes->attributeItem(i); params.append(o->getDocument()->attrName(it->id()).string() + "=\"" + it->value().string() + "\""); } } params.append( QString::fromLatin1("__KHTML__CLASSID=\"%1\"").arg( o->classId ) ); params.append( QString::fromLatin1("__KHTML__CODEBASE=\"%1\"").arg( o->getAttribute(ATTR_CODEBASE).string() ) ); part->requestObject( this, url, serviceType, params ); } else if ( element()->id() == ID_EMBED ) { HTMLEmbedElementImpl *o = static_cast<HTMLEmbedElementImpl *>(element()); url = o->url; serviceType = o->serviceType; if ( url.isEmpty() && serviceType.isEmpty() ) { #ifdef DEBUG_LAYOUT kdDebug() << "RenderPartObject::close - empty url and serverType" << endl; #endif return; } // add all attributes set on the embed object NamedAttrMapImpl* a = o->attributes(); if (a) { for (unsigned long i = 0; i < a->length(); ++i) { AttributeImpl* it = a->attributeItem(i); params.append(o->getDocument()->attrName(it->id()).string() + "=\"" + it->value().string() + "\""); } } part->requestObject( this, url, serviceType, params ); } else { assert(element()->id() == ID_IFRAME); HTMLIFrameElementImpl *o = static_cast<HTMLIFrameElementImpl *>(element()); url = o->url.string(); if (url.isEmpty()) url = "about:blank"; KHTMLView *v = static_cast<KHTMLView *>(m_view); bool requestSucceeded = v->part()->requestFrame( this, url, o->name.string(), QStringList(), true ); if (requestSucceeded && QString::equals(url,"about:blank")) { KHTMLPart *newPart = v->part()->findFrame( o->name.string() ); if (newPart && newPart->xmlDocImpl()) { newPart->xmlDocImpl()->setBaseURL( v->part()->baseURL().url() ); } } } }