Use PassRef for constructing StyleRules.
[WebKit-https.git] / Source / WebCore / css / StyleRule.h
1 /*
2  * (C) 1999-2003 Lars Knoll (knoll@kde.org)
3  * (C) 2002-2003 Dirk Mueller (mueller@kde.org)
4  * Copyright (C) 2002, 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 #ifndef StyleRule_h
23 #define StyleRule_h
24
25 #include "CSSSelectorList.h"
26 #include "MediaList.h"
27 #include "StylePropertySet.h"
28 #include <wtf/RefPtr.h>
29
30 namespace WebCore {
31
32 class CSSRule;
33 class CSSStyleRule;
34 class CSSStyleSheet;
35 class MutableStylePropertySet;
36 class StylePropertySet;
37
38 class StyleRuleBase : public WTF::RefCountedBase {
39     WTF_MAKE_FAST_ALLOCATED;
40 public:
41     enum Type {
42         Unknown, // Not used.
43         Style,
44         Charset, // Not used. These are internally strings owned by the style sheet.
45         Import,
46         Media,
47         FontFace,
48         Page,
49         Keyframes,
50         Keyframe, // Not used. These are internally non-rule StyleKeyframe objects.
51 #if ENABLE(CSS3_CONDITIONAL_RULES)
52         Supports = 12,
53 #endif
54 #if ENABLE(CSS_DEVICE_ADAPTATION)
55         Viewport = 15,
56 #endif
57         Region = 16,
58 #if ENABLE(CSS_SHADERS)
59         Filter = 17,
60 #endif
61 #if ENABLE(SHADOW_DOM)
62         HostInternal = 18, // Spec says Host = 1001, but we can use only 5 bit for type().
63 #endif
64     };
65
66     Type type() const { return static_cast<Type>(m_type); }
67     
68     bool isCharsetRule() const { return type() == Charset; }
69     bool isFontFaceRule() const { return type() == FontFace; }
70     bool isKeyframesRule() const { return type() == Keyframes; }
71     bool isMediaRule() const { return type() == Media; }
72     bool isPageRule() const { return type() == Page; }
73     bool isStyleRule() const { return type() == Style; }
74     bool isRegionRule() const { return type() == Region; }
75 #if ENABLE(CSS3_CONDITIONAL_RULES)
76     bool isSupportsRule() const { return type() == Supports; }
77 #endif
78 #if ENABLE(CSS_DEVICE_ADAPTATION)
79     bool isViewportRule() const { return type() == Viewport; }
80 #endif
81     bool isImportRule() const { return type() == Import; }
82 #if ENABLE(SHADOW_DOM)
83     bool isHostRule() const { return type() == HostInternal; }
84 #endif
85 #if ENABLE(CSS_SHADERS)
86     bool isFilterRule() const { return type() == Filter; }
87 #endif
88
89     PassRef<StyleRuleBase> copy() const;
90
91     int sourceLine() const { return m_sourceLine; }
92
93     void deref()
94     {
95         if (derefBase())
96             destroy();
97     }
98
99     // FIXME: There shouldn't be any need for the null parent version.
100     PassRefPtr<CSSRule> createCSSOMWrapper(CSSStyleSheet* parentSheet = 0) const;
101     PassRefPtr<CSSRule> createCSSOMWrapper(CSSRule* parentRule) const;
102
103 protected:
104     StyleRuleBase(Type type, signed sourceLine = 0) : m_type(type), m_sourceLine(sourceLine) { }
105     StyleRuleBase(const StyleRuleBase& o) : WTF::RefCountedBase(), m_type(o.m_type), m_sourceLine(o.m_sourceLine) { }
106
107     ~StyleRuleBase() { }
108
109 private:
110     void destroy();
111     
112     PassRefPtr<CSSRule> createCSSOMWrapper(CSSStyleSheet* parentSheet, CSSRule* parentRule) const;
113
114     unsigned m_type : 5;
115     signed m_sourceLine : 27;
116 };
117
118 class StyleRule : public StyleRuleBase {
119     WTF_MAKE_FAST_ALLOCATED;
120 public:
121     static PassRef<StyleRule> create(int sourceLine, PassRef<StylePropertySet> properties)
122     {
123         return adoptRef(*new StyleRule(sourceLine, std::move(properties)));
124     }
125     
126     ~StyleRule();
127
128     const CSSSelectorList& selectorList() const { return m_selectorList; }
129     const StylePropertySet& properties() const { return m_properties.get(); }
130     MutableStylePropertySet& mutableProperties();
131     
132     void parserAdoptSelectorVector(Vector<OwnPtr<CSSParserSelector>>& selectors) { m_selectorList.adoptSelectorVector(selectors); }
133     void wrapperAdoptSelectorList(CSSSelectorList& selectors) { m_selectorList.adopt(selectors); }
134     void parserAdoptSelectorArray(CSSSelector* selectors) { m_selectorList.adoptSelectorArray(selectors); }
135
136     PassRef<StyleRule> copy() const { return adoptRef(*new StyleRule(*this)); }
137
138     Vector<RefPtr<StyleRule>> splitIntoMultipleRulesWithMaximumSelectorComponentCount(unsigned) const;
139
140     static unsigned averageSizeInBytes();
141
142 private:
143     StyleRule(int sourceLine, PassRef<StylePropertySet>);
144     StyleRule(const StyleRule&);
145
146     static PassRef<StyleRule> create(int sourceLine, const Vector<const CSSSelector*>&, PassRef<StylePropertySet>);
147
148     Ref<StylePropertySet> m_properties;
149     CSSSelectorList m_selectorList;
150 };
151
152 inline const StyleRule* toStyleRule(const StyleRuleBase* rule)
153 {
154     ASSERT_WITH_SECURITY_IMPLICATION(!rule || rule->isStyleRule());
155     return static_cast<const StyleRule*>(rule);
156 }
157
158 class StyleRuleFontFace : public StyleRuleBase {
159 public:
160     static PassRef<StyleRuleFontFace> create(PassRef<StylePropertySet> properties) { return adoptRef(*new StyleRuleFontFace(std::move(properties))); }
161     
162     ~StyleRuleFontFace();
163
164     const StylePropertySet& properties() const { return m_properties.get(); }
165     MutableStylePropertySet& mutableProperties();
166
167     PassRef<StyleRuleFontFace> copy() const { return adoptRef(*new StyleRuleFontFace(*this)); }
168
169
170 private:
171     StyleRuleFontFace(PassRef<StylePropertySet>);
172     StyleRuleFontFace(const StyleRuleFontFace&);
173
174     Ref<StylePropertySet> m_properties;
175 };
176
177 class StyleRulePage : public StyleRuleBase {
178 public:
179     static PassRef<StyleRulePage> create(PassRef<StylePropertySet> properties) { return adoptRef(*new StyleRulePage(std::move(properties))); }
180
181     ~StyleRulePage();
182
183     const CSSSelector* selector() const { return m_selectorList.first(); }    
184     const StylePropertySet& properties() const { return m_properties.get(); }
185     MutableStylePropertySet& mutableProperties();
186
187     void parserAdoptSelectorVector(Vector<OwnPtr<CSSParserSelector>>& selectors) { m_selectorList.adoptSelectorVector(selectors); }
188     void wrapperAdoptSelectorList(CSSSelectorList& selectors) { m_selectorList.adopt(selectors); }
189
190     PassRef<StyleRulePage> copy() const { return adoptRef(*new StyleRulePage(*this)); }
191
192 private:
193     StyleRulePage(PassRef<StylePropertySet>);
194     StyleRulePage(const StyleRulePage&);
195     
196     Ref<StylePropertySet> m_properties;
197     CSSSelectorList m_selectorList;
198 };
199
200 class StyleRuleGroup : public StyleRuleBase {
201 public:
202     const Vector<RefPtr<StyleRuleBase>>& childRules() const { return m_childRules; }
203     
204     void wrapperInsertRule(unsigned, PassRef<StyleRuleBase>);
205     void wrapperRemoveRule(unsigned);
206     
207 protected:
208     StyleRuleGroup(Type, Vector<RefPtr<StyleRuleBase>>& adoptRule);
209     StyleRuleGroup(const StyleRuleGroup&);
210     
211 private:
212     Vector<RefPtr<StyleRuleBase>> m_childRules;
213 };
214
215 class StyleRuleMedia : public StyleRuleGroup {
216 public:
217     static PassRef<StyleRuleMedia> create(PassRefPtr<MediaQuerySet> media, Vector<RefPtr<StyleRuleBase>>& adoptRules)
218     {
219         return adoptRef(*new StyleRuleMedia(media, adoptRules));
220     }
221
222     MediaQuerySet* mediaQueries() const { return m_mediaQueries.get(); }
223
224     PassRef<StyleRuleMedia> copy() const { return adoptRef(*new StyleRuleMedia(*this)); }
225
226 private:
227     StyleRuleMedia(PassRefPtr<MediaQuerySet>, Vector<RefPtr<StyleRuleBase>>& adoptRules);
228     StyleRuleMedia(const StyleRuleMedia&);
229
230     RefPtr<MediaQuerySet> m_mediaQueries;
231 };
232
233 #if ENABLE(CSS3_CONDITIONAL_RULES)
234 class StyleRuleSupports : public StyleRuleGroup {
235 public:
236     static PassRef<StyleRuleSupports> create(const String& conditionText, bool conditionIsSupported, Vector<RefPtr<StyleRuleBase>>& adoptRules)
237     {
238         return adoptRef(*new StyleRuleSupports(conditionText, conditionIsSupported, adoptRules));
239     }
240
241     String conditionText() const { return m_conditionText; }
242     bool conditionIsSupported() const { return m_conditionIsSupported; }
243     PassRef<StyleRuleSupports> copy() const { return adoptRef(*new StyleRuleSupports(*this)); }
244
245 private:
246     StyleRuleSupports(const String& conditionText, bool conditionIsSupported, Vector<RefPtr<StyleRuleBase>>& adoptRules);
247     StyleRuleSupports(const StyleRuleSupports&);
248
249     String m_conditionText;
250     bool m_conditionIsSupported;
251 };
252 #endif
253
254 class StyleRuleRegion : public StyleRuleGroup {
255 public:
256     static PassRef<StyleRuleRegion> create(Vector<OwnPtr<CSSParserSelector>>* selectors, Vector<RefPtr<StyleRuleBase>>& adoptRules)
257     {
258         return adoptRef(*new StyleRuleRegion(selectors, adoptRules));
259     }
260
261     const CSSSelectorList& selectorList() const { return m_selectorList; }
262
263     PassRef<StyleRuleRegion> copy() const { return adoptRef(*new StyleRuleRegion(*this)); }
264
265 private:
266     StyleRuleRegion(Vector<OwnPtr<CSSParserSelector>>*, Vector<RefPtr<StyleRuleBase>>& adoptRules);
267     StyleRuleRegion(const StyleRuleRegion&);
268     
269     CSSSelectorList m_selectorList;
270 };
271
272 #if ENABLE(SHADOW_DOM)
273 class StyleRuleHost : public StyleRuleGroup {
274 public:
275     static PassRef<StyleRuleHost> create(Vector<RefPtr<StyleRuleBase>>& adoptRules)
276     {
277         return adoptRef(*new StyleRuleHost(adoptRules));
278     }
279
280     PassRef<StyleRuleHost> copy() const { return adoptRef(*new StyleRuleHost(*this)); }
281
282 private:
283     StyleRuleHost(Vector<RefPtr<StyleRuleBase>>& adoptRules) : StyleRuleGroup(HostInternal, adoptRules) { }
284     StyleRuleHost(const StyleRuleHost& o) : StyleRuleGroup(o) { }
285 };
286 #endif
287
288 #if ENABLE(CSS_DEVICE_ADAPTATION)
289 class StyleRuleViewport : public StyleRuleBase {
290 public:
291     static PassRef<StyleRuleViewport> create(PassRef<StylePropertySet> properties) { return adoptRef(*new StyleRuleViewport(std::move(properties))); }
292
293     ~StyleRuleViewport();
294
295     const StylePropertySet& properties() const { return m_properties.get(); }
296     MutableStylePropertySet& mutableProperties();
297
298     PassRef<StyleRuleViewport> copy() const { return adoptRef(*new StyleRuleViewport(*this)); }
299
300 private:
301     StyleRuleViewport(PassRef<StylePropertySet>);
302     StyleRuleViewport(const StyleRuleViewport&);
303
304     Ref<StylePropertySet> m_properties;
305 };
306 #endif // ENABLE(CSS_DEVICE_ADAPTATION)
307
308 inline const StyleRuleMedia* toStyleRuleMedia(const StyleRuleGroup* rule)
309 {
310     ASSERT_WITH_SECURITY_IMPLICATION(!rule || rule->isMediaRule());
311     return static_cast<const StyleRuleMedia*>(rule);
312 }
313
314 #if ENABLE(CSS3_CONDITIONAL_RULES)
315 inline const StyleRuleSupports* toStyleRuleSupports(const StyleRuleGroup* rule)
316 {
317     ASSERT_WITH_SECURITY_IMPLICATION(!rule || rule->isSupportsRule());
318     return static_cast<const StyleRuleSupports*>(rule);
319 }
320 #endif
321
322 inline const StyleRuleRegion* toStyleRuleRegion(const StyleRuleGroup* rule)
323 {
324     ASSERT_WITH_SECURITY_IMPLICATION(!rule || rule->isRegionRule());
325     return static_cast<const StyleRuleRegion*>(rule);
326 }
327
328 #if ENABLE(CSS_SHADERS)
329 class StyleRuleFilter : public StyleRuleBase {
330 public:
331     static PassRef<StyleRuleFilter> create(const String& filterName, PassRef<StylePropertySet> properties)
332     {
333         return adoptRef(*new StyleRuleFilter(filterName, std::move(properties)));
334     }
335
336     ~StyleRuleFilter();
337
338     const String& filterName() const { return m_filterName; }
339
340     const StylePropertySet& properties() const { return m_properties.get(); }
341     MutableStylePropertySet& mutableProperties();
342
343     PassRef<StyleRuleFilter> copy() const { return adoptRef(*new StyleRuleFilter(*this)); }
344
345 private:
346     StyleRuleFilter(const String&, PassRef<StylePropertySet>);
347     StyleRuleFilter(const StyleRuleFilter&);
348
349     String m_filterName;
350     Ref<StylePropertySet> m_properties;
351 };
352 #endif // ENABLE(CSS_SHADERS)
353
354 } // namespace WebCore
355
356 #endif // StyleRule_h