Stop storing raw pointers to Documents
[WebKit-https.git] / Source / WebCore / css / CSSStyleRule.cpp
1 /*
2  * (C) 1999-2003 Lars Knoll (knoll@kde.org)
3  * (C) 2002-2003 Dirk Mueller (mueller@kde.org)
4  * Copyright (C) 2002, 2005, 2006, 2008, 2012, 2013 Apple Inc. All rights reserved.
5  *
6  * This library is free software; you can redistribute it and/or
7  * modify it under the terms of the GNU Library General Public
8  * License as published by the Free Software Foundation; either
9  * version 2 of the License, or (at your option) any later version.
10  *
11  * This library is distributed in the hope that it will be useful,
12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
14  * Library General Public License for more details.
15  *
16  * You should have received a copy of the GNU Library General Public License
17  * along with this library; see the file COPYING.LIB.  If not, write to
18  * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
19  * Boston, MA 02110-1301, USA.
20  */
21
22 #include "config.h"
23 #include "CSSStyleRule.h"
24
25 #include "CSSParser.h"
26 #include "CSSStyleSheet.h"
27 #include "PropertySetCSSStyleDeclaration.h"
28 #include "RuleSet.h"
29 #include "StyleProperties.h"
30 #include "StyleRule.h"
31 #include <wtf/NeverDestroyed.h>
32 #include <wtf/text/StringBuilder.h>
33
34 namespace WebCore {
35
36 typedef HashMap<const CSSStyleRule*, String> SelectorTextCache;
37 static SelectorTextCache& selectorTextCache()
38 {
39     static NeverDestroyed<SelectorTextCache> cache;
40     return cache;
41 }
42
43 CSSStyleRule::CSSStyleRule(StyleRule& styleRule, CSSStyleSheet* parent)
44     : CSSRule(parent)
45     , m_styleRule(styleRule)
46 {
47 }
48
49 CSSStyleRule::~CSSStyleRule()
50 {
51     if (m_propertiesCSSOMWrapper)
52         m_propertiesCSSOMWrapper->clearParentRule();
53
54     if (hasCachedSelectorText()) {
55         selectorTextCache().remove(this);
56         setHasCachedSelectorText(false);
57     }
58 }
59
60 CSSStyleDeclaration& CSSStyleRule::style()
61 {
62     if (!m_propertiesCSSOMWrapper)
63         m_propertiesCSSOMWrapper = StyleRuleCSSStyleDeclaration::create(m_styleRule->mutableProperties(), *this);
64     return *m_propertiesCSSOMWrapper;
65 }
66
67 String CSSStyleRule::generateSelectorText() const
68 {
69     return m_styleRule->selectorList().selectorsText();
70 }
71
72 String CSSStyleRule::selectorText() const
73 {
74     if (hasCachedSelectorText()) {
75         ASSERT(selectorTextCache().contains(this));
76         return selectorTextCache().get(this);
77     }
78
79     ASSERT(!selectorTextCache().contains(this));
80     String text = generateSelectorText();
81     selectorTextCache().set(this, text);
82     setHasCachedSelectorText(true);
83     return text;
84 }
85
86 void CSSStyleRule::setSelectorText(const String& selectorText)
87 {
88     // FIXME: getMatchedCSSRules can return CSSStyleRules that are missing parent stylesheet pointer while
89     // referencing StyleRules that are part of stylesheet. Disallow mutations in this case.
90     if (!parentStyleSheet())
91         return;
92
93     CSSParser p(parserContext());
94     CSSSelectorList selectorList;
95     p.parseSelector(selectorText, selectorList);
96     if (!selectorList.isValid())
97         return;
98
99     // NOTE: The selector list has to fit into RuleData. <http://webkit.org/b/118369>
100     if (selectorList.componentCount() > RuleData::maximumSelectorComponentCount)
101         return;
102
103     CSSStyleSheet::RuleMutationScope mutationScope(this);
104
105     m_styleRule->wrapperAdoptSelectorList(WTFMove(selectorList));
106
107     if (hasCachedSelectorText()) {
108         selectorTextCache().remove(this);
109         setHasCachedSelectorText(false);
110     }
111 }
112
113 String CSSStyleRule::cssText() const
114 {
115     StringBuilder result;
116     result.append(selectorText());
117     result.appendLiteral(" { ");
118     String decls = m_styleRule->properties().asText();
119     result.append(decls);
120     if (!decls.isEmpty())
121         result.append(' ');
122     result.append('}');
123     return result.toString();
124 }
125
126 void CSSStyleRule::reattach(StyleRuleBase& rule)
127 {
128     m_styleRule = downcast<StyleRule>(rule);
129     if (m_propertiesCSSOMWrapper)
130         m_propertiesCSSOMWrapper->reattach(m_styleRule->mutableProperties());
131 }
132
133 } // namespace WebCore