mirror of
https://github.com/flutter/flutter.git
synced 2026-02-20 02:29:02 +08:00
StyleElement existed because Blink supported <link> and XSL sheets, Sky only supports <style> so we can merge them together. R=eseidel@chromium.org Review URL: https://codereview.chromium.org/780083003
217 lines
6.1 KiB
C++
217 lines
6.1 KiB
C++
/*
|
|
* Copyright (C) 1999 Lars Knoll (knoll@kde.org)
|
|
* (C) 1999 Antti Koivisto (koivisto@kde.org)
|
|
* (C) 2001 Dirk Mueller (mueller@kde.org)
|
|
* Copyright (C) 2003, 2010 Apple Inc. All rights reserved.
|
|
* (C) 2007 Rob Buis (buis@kde.org)
|
|
*
|
|
* 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 "sky/engine/config.h"
|
|
#include "sky/engine/core/html/HTMLStyleElement.h"
|
|
|
|
#include "gen/sky/core/HTMLNames.h"
|
|
#include "sky/engine/core/css/MediaList.h"
|
|
#include "sky/engine/core/css/MediaQueryEvaluator.h"
|
|
#include "sky/engine/core/dom/Document.h"
|
|
#include "sky/engine/core/dom/Element.h"
|
|
#include "sky/engine/core/dom/StyleEngine.h"
|
|
#include "sky/engine/core/dom/shadow/ShadowRoot.h"
|
|
#include "sky/engine/core/frame/LocalFrame.h"
|
|
#include "sky/engine/platform/TraceEvent.h"
|
|
|
|
namespace blink {
|
|
|
|
inline HTMLStyleElement::HTMLStyleElement(Document& document, bool createdByParser)
|
|
: HTMLElement(HTMLNames::styleTag, document)
|
|
, m_createdByParser(createdByParser)
|
|
, m_loading(false)
|
|
, m_registeredAsCandidate(false)
|
|
, m_startPosition(TextPosition::belowRangePosition())
|
|
{
|
|
if (createdByParser)
|
|
m_startPosition = document.parserPosition();
|
|
}
|
|
|
|
HTMLStyleElement::~HTMLStyleElement()
|
|
{
|
|
clearDocumentData();
|
|
if (m_sheet)
|
|
clearSheet();
|
|
}
|
|
|
|
PassRefPtr<HTMLStyleElement> HTMLStyleElement::create(Document& document, bool createdByParser)
|
|
{
|
|
return adoptRef(new HTMLStyleElement(document, createdByParser));
|
|
}
|
|
|
|
void HTMLStyleElement::parseAttribute(const QualifiedName& name, const AtomicString& value)
|
|
{
|
|
if (name == HTMLNames::mediaAttr && inDocument() && document().isActive() && m_sheet) {
|
|
m_sheet->setMediaQueries(MediaQuerySet::create(value));
|
|
document().modifiedStyleSheet(m_sheet.get());
|
|
} else {
|
|
HTMLElement::parseAttribute(name, value);
|
|
}
|
|
}
|
|
|
|
void HTMLStyleElement::finishParsingChildren()
|
|
{
|
|
process();
|
|
m_createdByParser = false;
|
|
}
|
|
|
|
void HTMLStyleElement::insertedInto(ContainerNode* insertionPoint)
|
|
{
|
|
HTMLElement::insertedInto(insertionPoint);
|
|
|
|
if (!inDocument())
|
|
return;
|
|
|
|
processStyleSheet();
|
|
|
|
if (ShadowRoot* scope = containingShadowRoot())
|
|
scope->registerScopedHTMLStyleChild();
|
|
}
|
|
|
|
void HTMLStyleElement::removedFrom(ContainerNode* insertionPoint)
|
|
{
|
|
HTMLElement::removedFrom(insertionPoint);
|
|
|
|
if (!insertionPoint->inDocument())
|
|
return;
|
|
|
|
ShadowRoot* scopingNode = containingShadowRoot();
|
|
if (!scopingNode)
|
|
scopingNode = insertionPoint->containingShadowRoot();
|
|
|
|
if (scopingNode)
|
|
scopingNode->unregisterScopedHTMLStyleChild();
|
|
|
|
TreeScope* containingScope = containingShadowRoot();
|
|
TreeScope& scope = containingScope ? *containingScope : insertionPoint->treeScope();
|
|
|
|
if (m_registeredAsCandidate) {
|
|
document().styleEngine()->removeStyleSheetCandidateNode(this, scopingNode, scope);
|
|
m_registeredAsCandidate = false;
|
|
}
|
|
|
|
RefPtr<CSSStyleSheet> removedSheet = m_sheet.get();
|
|
|
|
if (m_sheet)
|
|
clearSheet();
|
|
if (removedSheet)
|
|
document().removedStyleSheet(removedSheet.get());
|
|
}
|
|
|
|
void HTMLStyleElement::childrenChanged(const ChildrenChange& change)
|
|
{
|
|
HTMLElement::childrenChanged(change);
|
|
|
|
if (m_createdByParser)
|
|
return;
|
|
process();
|
|
}
|
|
|
|
const AtomicString& HTMLStyleElement::media() const
|
|
{
|
|
return getAttribute(HTMLNames::mediaAttr);
|
|
}
|
|
|
|
const AtomicString& HTMLStyleElement::type() const
|
|
{
|
|
return getAttribute(HTMLNames::typeAttr);
|
|
}
|
|
|
|
ContainerNode* HTMLStyleElement::scopingNode()
|
|
{
|
|
if (!inDocument())
|
|
return 0;
|
|
|
|
if (isInShadowTree())
|
|
return containingShadowRoot();
|
|
|
|
return &document();
|
|
}
|
|
|
|
void HTMLStyleElement::process()
|
|
{
|
|
if (!inDocument())
|
|
return;
|
|
createSheet();
|
|
}
|
|
|
|
void HTMLStyleElement::clearSheet()
|
|
{
|
|
ASSERT(m_sheet);
|
|
m_sheet.release()->clearOwnerNode();
|
|
}
|
|
|
|
void HTMLStyleElement::createSheet()
|
|
{
|
|
ASSERT(inDocument());
|
|
|
|
if (m_sheet)
|
|
clearSheet();
|
|
|
|
const AtomicString& type = this->type();
|
|
if (type.isEmpty() || type == "text/css") {
|
|
RefPtr<MediaQuerySet> mediaQueries = MediaQuerySet::create(media());
|
|
|
|
MediaQueryEvaluator screenEval("screen", true);
|
|
MediaQueryEvaluator printEval("print", true);
|
|
if (screenEval.eval(mediaQueries.get()) || printEval.eval(mediaQueries.get())) {
|
|
m_loading = true;
|
|
const String& text = textFromChildren();
|
|
TextPosition startPosition = m_startPosition == TextPosition::belowRangePosition() ? TextPosition::minimumPosition() : m_startPosition;
|
|
m_sheet = document().styleEngine()->createSheet(this, text, startPosition, m_createdByParser);
|
|
m_sheet->setMediaQueries(mediaQueries.release());
|
|
m_loading = false;
|
|
}
|
|
}
|
|
|
|
document().styleResolverChanged();
|
|
}
|
|
|
|
void HTMLStyleElement::clearDocumentData()
|
|
{
|
|
if (m_sheet)
|
|
m_sheet->clearOwnerNode();
|
|
|
|
if (inDocument()) {
|
|
ContainerNode* scopingNode = this->scopingNode();
|
|
TreeScope& scope = scopingNode ? scopingNode->treeScope() : treeScope();
|
|
document().styleEngine()->removeStyleSheetCandidateNode(this, scopingNode, scope);
|
|
}
|
|
}
|
|
|
|
void HTMLStyleElement::processStyleSheet()
|
|
{
|
|
TRACE_EVENT0("blink", "StyleElement::processStyleSheet");
|
|
|
|
ASSERT(inDocument());
|
|
|
|
m_registeredAsCandidate = true;
|
|
document().styleEngine()->addStyleSheetCandidateNode(this, m_createdByParser);
|
|
if (m_createdByParser)
|
|
return;
|
|
|
|
process();
|
|
}
|
|
|
|
}
|