/* * (C) 1999-2003 Lars Knoll (knoll@kde.org) * (C) 2002-2003 Dirk Mueller (mueller@kde.org) * Copyright (C) 2002, 2006, 2008, 2012, 2013 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. */ #ifndef StyleRule_h #define StyleRule_h #include "CSSSelectorList.h" #include "MediaList.h" #include "StyleProperties.h" #include #include namespace WebCore { class CSSRule; class CSSStyleRule; class CSSStyleSheet; class MutableStyleProperties; class StyleProperties; class StyleRuleBase : public WTF::RefCountedBase { WTF_MAKE_FAST_ALLOCATED; public: enum Type { Unknown, // Not used. Style, Charset, // Not used. These are internally strings owned by the style sheet. Import, Media, FontFace, Page, Keyframes, Keyframe, // Not used. These are internally non-rule StyleKeyframe objects. Supports = 12, #if ENABLE(CSS_DEVICE_ADAPTATION) Viewport = 15, #endif Region = 16, }; Type type() const { return static_cast(m_type); } bool isCharsetRule() const { return type() == Charset; } bool isFontFaceRule() const { return type() == FontFace; } bool isKeyframesRule() const { return type() == Keyframes; } bool isMediaRule() const { return type() == Media; } bool isPageRule() const { return type() == Page; } bool isStyleRule() const { return type() == Style; } bool isRegionRule() const { return type() == Region; } bool isSupportsRule() const { return type() == Supports; } #if ENABLE(CSS_DEVICE_ADAPTATION) bool isViewportRule() const { return type() == Viewport; } #endif bool isImportRule() const { return type() == Import; } Ref copy() const; int sourceLine() const { return m_sourceLine; } void deref() { if (derefBase()) destroy(); } // FIXME: There shouldn't be any need for the null parent version. PassRefPtr createCSSOMWrapper(CSSStyleSheet* parentSheet = 0) const; PassRefPtr createCSSOMWrapper(CSSRule* parentRule) const; protected: StyleRuleBase(Type type, signed sourceLine = 0) : m_type(type), m_sourceLine(sourceLine) { } StyleRuleBase(const StyleRuleBase& o) : WTF::RefCountedBase(), m_type(o.m_type), m_sourceLine(o.m_sourceLine) { } ~StyleRuleBase() { } private: void destroy(); PassRefPtr createCSSOMWrapper(CSSStyleSheet* parentSheet, CSSRule* parentRule) const; unsigned m_type : 5; signed m_sourceLine : 27; }; class StyleRule : public StyleRuleBase { WTF_MAKE_FAST_ALLOCATED; public: static Ref create(int sourceLine, Ref&& properties) { return adoptRef(*new StyleRule(sourceLine, WTFMove(properties))); } ~StyleRule(); const CSSSelectorList& selectorList() const { return m_selectorList; } const StyleProperties& properties() const { return m_properties; } MutableStyleProperties& mutableProperties(); void parserAdoptSelectorVector(Vector>& selectors) { m_selectorList.adoptSelectorVector(selectors); } void wrapperAdoptSelectorList(CSSSelectorList& selectors) { m_selectorList = WTFMove(selectors); } void parserAdoptSelectorArray(CSSSelector* selectors) { m_selectorList.adoptSelectorArray(selectors); } Ref copy() const { return adoptRef(*new StyleRule(*this)); } Vector> splitIntoMultipleRulesWithMaximumSelectorComponentCount(unsigned) const; static unsigned averageSizeInBytes(); private: StyleRule(int sourceLine, Ref&&); StyleRule(const StyleRule&); static Ref create(int sourceLine, const Vector&, Ref&&); Ref m_properties; CSSSelectorList m_selectorList; }; class StyleRuleFontFace : public StyleRuleBase { public: static Ref create(Ref&& properties) { return adoptRef(*new StyleRuleFontFace(WTFMove(properties))); } ~StyleRuleFontFace(); const StyleProperties& properties() const { return m_properties; } MutableStyleProperties& mutableProperties(); Ref copy() const { return adoptRef(*new StyleRuleFontFace(*this)); } private: explicit StyleRuleFontFace(Ref&&); StyleRuleFontFace(const StyleRuleFontFace&); Ref m_properties; }; class StyleRulePage : public StyleRuleBase { public: static Ref create(Ref&& properties) { return adoptRef(*new StyleRulePage(WTFMove(properties))); } ~StyleRulePage(); const CSSSelector* selector() const { return m_selectorList.first(); } const StyleProperties& properties() const { return m_properties; } MutableStyleProperties& mutableProperties(); void parserAdoptSelectorVector(Vector>& selectors) { m_selectorList.adoptSelectorVector(selectors); } void wrapperAdoptSelectorList(CSSSelectorList& selectors) { m_selectorList = WTFMove(selectors); } Ref copy() const { return adoptRef(*new StyleRulePage(*this)); } private: explicit StyleRulePage(Ref&&); StyleRulePage(const StyleRulePage&); Ref m_properties; CSSSelectorList m_selectorList; }; class StyleRuleGroup : public StyleRuleBase { public: const Vector>& childRules() const { return m_childRules; } void wrapperInsertRule(unsigned, Ref&&); void wrapperRemoveRule(unsigned); protected: StyleRuleGroup(Type, Vector>& adoptRule); StyleRuleGroup(const StyleRuleGroup&); private: Vector> m_childRules; }; class StyleRuleMedia : public StyleRuleGroup { public: static Ref create(PassRefPtr media, Vector>& adoptRules) { return adoptRef(*new StyleRuleMedia(media, adoptRules)); } MediaQuerySet* mediaQueries() const { return m_mediaQueries.get(); } Ref copy() const { return adoptRef(*new StyleRuleMedia(*this)); } private: StyleRuleMedia(PassRefPtr, Vector>& adoptRules); StyleRuleMedia(const StyleRuleMedia&); RefPtr m_mediaQueries; }; class StyleRuleSupports : public StyleRuleGroup { public: static Ref create(const String& conditionText, bool conditionIsSupported, Vector>& adoptRules) { return adoptRef(*new StyleRuleSupports(conditionText, conditionIsSupported, adoptRules)); } String conditionText() const { return m_conditionText; } bool conditionIsSupported() const { return m_conditionIsSupported; } Ref copy() const { return adoptRef(*new StyleRuleSupports(*this)); } private: StyleRuleSupports(const String& conditionText, bool conditionIsSupported, Vector>& adoptRules); StyleRuleSupports(const StyleRuleSupports&); String m_conditionText; bool m_conditionIsSupported; }; class StyleRuleRegion : public StyleRuleGroup { public: static Ref create(Vector>* selectors, Vector>& adoptRules) { return adoptRef(*new StyleRuleRegion(selectors, adoptRules)); } const CSSSelectorList& selectorList() const { return m_selectorList; } Ref copy() const { return adoptRef(*new StyleRuleRegion(*this)); } private: StyleRuleRegion(Vector>*, Vector>& adoptRules); StyleRuleRegion(const StyleRuleRegion&); CSSSelectorList m_selectorList; }; #if ENABLE(CSS_DEVICE_ADAPTATION) class StyleRuleViewport : public StyleRuleBase { public: static Ref create(Ref&& properties) { return adoptRef(*new StyleRuleViewport(WTFMove(properties))); } ~StyleRuleViewport(); const StyleProperties& properties() const { return m_properties.get(); } MutableStyleProperties& mutableProperties(); Ref copy() const { return adoptRef(*new StyleRuleViewport(*this)); } private: explicit StyleRuleViewport(Ref&&); StyleRuleViewport(const StyleRuleViewport&); Ref m_properties; }; #endif // ENABLE(CSS_DEVICE_ADAPTATION) } // namespace WebCore SPECIALIZE_TYPE_TRAITS_BEGIN(WebCore::StyleRule) static bool isType(const WebCore::StyleRuleBase& rule) { return rule.isStyleRule(); } SPECIALIZE_TYPE_TRAITS_END() SPECIALIZE_TYPE_TRAITS_BEGIN(WebCore::StyleRuleFontFace) static bool isType(const WebCore::StyleRuleBase& rule) { return rule.isFontFaceRule(); } SPECIALIZE_TYPE_TRAITS_END() SPECIALIZE_TYPE_TRAITS_BEGIN(WebCore::StyleRuleMedia) static bool isType(const WebCore::StyleRuleBase& rule) { return rule.isMediaRule(); } SPECIALIZE_TYPE_TRAITS_END() SPECIALIZE_TYPE_TRAITS_BEGIN(WebCore::StyleRulePage) static bool isType(const WebCore::StyleRuleBase& rule) { return rule.isPageRule(); } SPECIALIZE_TYPE_TRAITS_END() SPECIALIZE_TYPE_TRAITS_BEGIN(WebCore::StyleRuleSupports) static bool isType(const WebCore::StyleRuleBase& rule) { return rule.isSupportsRule(); } SPECIALIZE_TYPE_TRAITS_END() SPECIALIZE_TYPE_TRAITS_BEGIN(WebCore::StyleRuleRegion) static bool isType(const WebCore::StyleRuleBase& rule) { return rule.isRegionRule(); } SPECIALIZE_TYPE_TRAITS_END() #if ENABLE(CSS_DEVICE_ADAPTATION) SPECIALIZE_TYPE_TRAITS_BEGIN(WebCore::StyleRuleViewport) static bool isType(const WebCore::StyleRuleBase& rule) { return rule.isViewportRule(); } SPECIALIZE_TYPE_TRAITS_END() #endif // ENABLE(CSS_DEVICE_ADAPTATION) #endif // StyleRule_h