/* * Copyright (C) 2000 Peter Kelly (pmk@post.com) * Copyright (C) 2006, 2008, 2009 Apple Inc. All rights reserved. * * 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 "config.h" #include "core/dom/ProcessingInstruction.h" #include "FetchInitiatorTypeNames.h" #include "core/css/CSSStyleSheet.h" #include "core/css/MediaList.h" #include "core/css/StyleSheetContents.h" #include "core/dom/Document.h" #include "core/dom/StyleEngine.h" #include "core/fetch/CSSStyleSheetResource.h" #include "core/fetch/FetchRequest.h" #include "core/fetch/ResourceFetcher.h" #include "core/fetch/XSLStyleSheetResource.h" #include "core/xml/XSLStyleSheet.h" #include "core/xml/parser/XMLDocumentParser.h" // for parseAttributes() namespace WebCore { inline ProcessingInstruction::ProcessingInstruction(Document& document, const String& target, const String& data) : CharacterData(document, data, CreateOther) , m_target(target) , m_resource(0) , m_loading(false) , m_alternate(false) , m_createdByParser(false) , m_isCSS(false) , m_isXSL(false) { ScriptWrappable::init(this); } PassRefPtr ProcessingInstruction::create(Document& document, const String& target, const String& data) { return adoptRef(new ProcessingInstruction(document, target, data)); } ProcessingInstruction::~ProcessingInstruction() { if (m_sheet) m_sheet->clearOwnerNode(); if (m_resource) m_resource->removeClient(this); if (inDocument()) document().styleEngine()->removeStyleSheetCandidateNode(this); } String ProcessingInstruction::nodeName() const { return m_target; } Node::NodeType ProcessingInstruction::nodeType() const { return PROCESSING_INSTRUCTION_NODE; } PassRefPtr ProcessingInstruction::cloneNode(bool /*deep*/) { // FIXME: Is it a problem that this does not copy m_localHref? // What about other data members? return create(document(), m_target, m_data); } void ProcessingInstruction::checkStyleSheet() { if (m_target == "xml-stylesheet" && document().frame() && parentNode() == document()) { // see http://www.w3.org/TR/xml-stylesheet/ // ### support stylesheet included in a fragment of this (or another) document // ### make sure this gets called when adding from javascript bool attrsOk; const HashMap attrs = parseAttributes(m_data, attrsOk); if (!attrsOk) return; HashMap::const_iterator i = attrs.find("type"); String type; if (i != attrs.end()) type = i->value; m_isCSS = type.isEmpty() || type == "text/css"; m_isXSL = (type == "text/xml" || type == "text/xsl" || type == "application/xml" || type == "application/xhtml+xml" || type == "application/rss+xml" || type == "application/atom+xml"); if (!m_isCSS && !m_isXSL) return; String href = attrs.get("href"); String alternate = attrs.get("alternate"); m_alternate = alternate == "yes"; m_title = attrs.get("title"); m_media = attrs.get("media"); if (m_alternate && m_title.isEmpty()) return; if (href.length() > 1 && href[0] == '#') { m_localHref = href.substring(1); // We need to make a synthetic XSLStyleSheet that is embedded. It needs to be able // to kick off import/include loads that can hang off some parent sheet. if (m_isXSL) { KURL finalURL(ParsedURLString, m_localHref); m_sheet = XSLStyleSheet::createEmbedded(this, finalURL); m_loading = false; } } else { if (m_resource) { m_resource->removeClient(this); m_resource = 0; } String url = document().completeURL(href).string(); if (!dispatchBeforeLoadEvent(url)) return; m_loading = true; document().styleEngine()->addPendingSheet(); FetchRequest request(ResourceRequest(document().completeURL(href)), FetchInitiatorTypeNames::processinginstruction); if (m_isXSL) m_resource = document().fetcher()->fetchXSLStyleSheet(request); else { String charset = attrs.get("charset"); if (charset.isEmpty()) charset = document().charset(); request.setCharset(charset); m_resource = document().fetcher()->fetchCSSStyleSheet(request); } if (m_resource) m_resource->addClient(this); else { // The request may have been denied if (for example) the stylesheet is local and the document is remote. m_loading = false; document().styleEngine()->removePendingSheet(this); } } } } bool ProcessingInstruction::isLoading() const { if (m_loading) return true; if (!m_sheet) return false; return m_sheet->isLoading(); } bool ProcessingInstruction::sheetLoaded() { if (!isLoading()) { document().styleEngine()->removePendingSheet(this); return true; } return false; } void ProcessingInstruction::setCSSStyleSheet(const String& href, const KURL& baseURL, const String& charset, const CSSStyleSheetResource* sheet) { if (!inDocument()) { ASSERT(!m_sheet); return; } ASSERT(m_isCSS); CSSParserContext parserContext(document(), baseURL, charset); RefPtr newSheet = StyleSheetContents::create(href, parserContext); RefPtr cssSheet = CSSStyleSheet::create(newSheet, this); cssSheet->setDisabled(m_alternate); cssSheet->setTitle(m_title); cssSheet->setMediaQueries(MediaQuerySet::create(m_media)); m_sheet = cssSheet.release(); // We don't need the cross-origin security check here because we are // getting the sheet text in "strict" mode. This enforces a valid CSS MIME // type. parseStyleSheet(sheet->sheetText(true)); } void ProcessingInstruction::setXSLStyleSheet(const String& href, const KURL& baseURL, const String& sheet) { ASSERT(m_isXSL); m_sheet = XSLStyleSheet::create(this, href, baseURL); parseStyleSheet(sheet); } void ProcessingInstruction::parseStyleSheet(const String& sheet) { if (m_isCSS) toCSSStyleSheet(m_sheet.get())->contents()->parseString(sheet); else if (m_isXSL) toXSLStyleSheet(m_sheet.get())->parseString(sheet); if (m_resource) m_resource->removeClient(this); m_resource = 0; m_loading = false; if (m_isCSS) toCSSStyleSheet(m_sheet.get())->contents()->checkLoaded(); else if (m_isXSL) toXSLStyleSheet(m_sheet.get())->checkLoaded(); } void ProcessingInstruction::setCSSStyleSheet(PassRefPtr sheet) { ASSERT(!m_resource); ASSERT(!m_loading); m_sheet = sheet; sheet->setTitle(m_title); sheet->setDisabled(m_alternate); } void ProcessingInstruction::addSubresourceAttributeURLs(ListHashSet& urls) const { if (!sheet()) return; addSubresourceURL(urls, sheet()->baseURL()); } Node::InsertionNotificationRequest ProcessingInstruction::insertedInto(ContainerNode* insertionPoint) { CharacterData::insertedInto(insertionPoint); if (!insertionPoint->inDocument()) return InsertionDone; document().styleEngine()->addStyleSheetCandidateNode(this, m_createdByParser); checkStyleSheet(); return InsertionDone; } void ProcessingInstruction::removedFrom(ContainerNode* insertionPoint) { CharacterData::removedFrom(insertionPoint); if (!insertionPoint->inDocument()) return; document().styleEngine()->removeStyleSheetCandidateNode(this); RefPtr removedSheet = m_sheet; if (m_sheet) { ASSERT(m_sheet->ownerNode() == this); m_sheet->clearOwnerNode(); m_sheet = 0; } // If we're in document teardown, then we don't need to do any notification of our sheet's removal. if (document().isActive()) document().removedStyleSheet(removedSheet.get()); } void ProcessingInstruction::finishParsingChildren() { m_createdByParser = false; CharacterData::finishParsingChildren(); } } // namespace