blob: 0391d44fa6088aba33e5883f96d62a21762b9a6c [file] [log] [blame]
/*
* (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 "StylePropertySet.h"
#include <wtf/RefPtr.h>
namespace WebCore {
class CSSRule;
class CSSStyleRule;
class CSSStyleSheet;
class MutableStylePropertySet;
class StylePropertySet;
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.
#if ENABLE(CSS3_CONDITIONAL_RULES)
Supports = 12,
#endif
#if ENABLE(CSS_DEVICE_ADAPTATION)
Viewport = 15,
#endif
Region = 16,
#if ENABLE(CSS_SHADERS)
Filter = 17,
#endif
#if ENABLE(SHADOW_DOM)
HostInternal = 18, // Spec says Host = 1001, but we can use only 5 bit for type().
#endif
};
Type type() const { return static_cast<Type>(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; }
#if ENABLE(CSS3_CONDITIONAL_RULES)
bool isSupportsRule() const { return type() == Supports; }
#endif
#if ENABLE(CSS_DEVICE_ADAPTATION)
bool isViewportRule() const { return type() == Viewport; }
#endif
bool isImportRule() const { return type() == Import; }
#if ENABLE(SHADOW_DOM)
bool isHostRule() const { return type() == HostInternal; }
#endif
#if ENABLE(CSS_SHADERS)
bool isFilterRule() const { return type() == Filter; }
#endif
PassRef<StyleRuleBase> 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<CSSRule> createCSSOMWrapper(CSSStyleSheet* parentSheet = 0) const;
PassRefPtr<CSSRule> 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<CSSRule> createCSSOMWrapper(CSSStyleSheet* parentSheet, CSSRule* parentRule) const;
unsigned m_type : 5;
signed m_sourceLine : 27;
};
class StyleRule : public StyleRuleBase {
WTF_MAKE_FAST_ALLOCATED;
public:
static PassRef<StyleRule> create(int sourceLine, PassRef<StylePropertySet> properties)
{
return adoptRef(*new StyleRule(sourceLine, std::move(properties)));
}
~StyleRule();
const CSSSelectorList& selectorList() const { return m_selectorList; }
const StylePropertySet& properties() const { return m_properties.get(); }
MutableStylePropertySet& mutableProperties();
void parserAdoptSelectorVector(Vector<OwnPtr<CSSParserSelector>>& selectors) { m_selectorList.adoptSelectorVector(selectors); }
void wrapperAdoptSelectorList(CSSSelectorList& selectors) { m_selectorList.adopt(selectors); }
void parserAdoptSelectorArray(CSSSelector* selectors) { m_selectorList.adoptSelectorArray(selectors); }
PassRef<StyleRule> copy() const { return adoptRef(*new StyleRule(*this)); }
Vector<RefPtr<StyleRule>> splitIntoMultipleRulesWithMaximumSelectorComponentCount(unsigned) const;
static unsigned averageSizeInBytes();
private:
StyleRule(int sourceLine, PassRef<StylePropertySet>);
StyleRule(const StyleRule&);
static PassRef<StyleRule> create(int sourceLine, const Vector<const CSSSelector*>&, PassRef<StylePropertySet>);
Ref<StylePropertySet> m_properties;
CSSSelectorList m_selectorList;
};
inline const StyleRule* toStyleRule(const StyleRuleBase* rule)
{
ASSERT_WITH_SECURITY_IMPLICATION(!rule || rule->isStyleRule());
return static_cast<const StyleRule*>(rule);
}
class StyleRuleFontFace : public StyleRuleBase {
public:
static PassRef<StyleRuleFontFace> create(PassRef<StylePropertySet> properties) { return adoptRef(*new StyleRuleFontFace(std::move(properties))); }
~StyleRuleFontFace();
const StylePropertySet& properties() const { return m_properties.get(); }
MutableStylePropertySet& mutableProperties();
PassRef<StyleRuleFontFace> copy() const { return adoptRef(*new StyleRuleFontFace(*this)); }
private:
StyleRuleFontFace(PassRef<StylePropertySet>);
StyleRuleFontFace(const StyleRuleFontFace&);
Ref<StylePropertySet> m_properties;
};
class StyleRulePage : public StyleRuleBase {
public:
static PassRef<StyleRulePage> create(PassRef<StylePropertySet> properties) { return adoptRef(*new StyleRulePage(std::move(properties))); }
~StyleRulePage();
const CSSSelector* selector() const { return m_selectorList.first(); }
const StylePropertySet& properties() const { return m_properties.get(); }
MutableStylePropertySet& mutableProperties();
void parserAdoptSelectorVector(Vector<OwnPtr<CSSParserSelector>>& selectors) { m_selectorList.adoptSelectorVector(selectors); }
void wrapperAdoptSelectorList(CSSSelectorList& selectors) { m_selectorList.adopt(selectors); }
PassRef<StyleRulePage> copy() const { return adoptRef(*new StyleRulePage(*this)); }
private:
StyleRulePage(PassRef<StylePropertySet>);
StyleRulePage(const StyleRulePage&);
Ref<StylePropertySet> m_properties;
CSSSelectorList m_selectorList;
};
class StyleRuleGroup : public StyleRuleBase {
public:
const Vector<RefPtr<StyleRuleBase>>& childRules() const { return m_childRules; }
void wrapperInsertRule(unsigned, PassRef<StyleRuleBase>);
void wrapperRemoveRule(unsigned);
protected:
StyleRuleGroup(Type, Vector<RefPtr<StyleRuleBase>>& adoptRule);
StyleRuleGroup(const StyleRuleGroup&);
private:
Vector<RefPtr<StyleRuleBase>> m_childRules;
};
class StyleRuleMedia : public StyleRuleGroup {
public:
static PassRef<StyleRuleMedia> create(PassRefPtr<MediaQuerySet> media, Vector<RefPtr<StyleRuleBase>>& adoptRules)
{
return adoptRef(*new StyleRuleMedia(media, adoptRules));
}
MediaQuerySet* mediaQueries() const { return m_mediaQueries.get(); }
PassRef<StyleRuleMedia> copy() const { return adoptRef(*new StyleRuleMedia(*this)); }
private:
StyleRuleMedia(PassRefPtr<MediaQuerySet>, Vector<RefPtr<StyleRuleBase>>& adoptRules);
StyleRuleMedia(const StyleRuleMedia&);
RefPtr<MediaQuerySet> m_mediaQueries;
};
#if ENABLE(CSS3_CONDITIONAL_RULES)
class StyleRuleSupports : public StyleRuleGroup {
public:
static PassRef<StyleRuleSupports> create(const String& conditionText, bool conditionIsSupported, Vector<RefPtr<StyleRuleBase>>& adoptRules)
{
return adoptRef(*new StyleRuleSupports(conditionText, conditionIsSupported, adoptRules));
}
String conditionText() const { return m_conditionText; }
bool conditionIsSupported() const { return m_conditionIsSupported; }
PassRef<StyleRuleSupports> copy() const { return adoptRef(*new StyleRuleSupports(*this)); }
private:
StyleRuleSupports(const String& conditionText, bool conditionIsSupported, Vector<RefPtr<StyleRuleBase>>& adoptRules);
StyleRuleSupports(const StyleRuleSupports&);
String m_conditionText;
bool m_conditionIsSupported;
};
#endif
class StyleRuleRegion : public StyleRuleGroup {
public:
static PassRef<StyleRuleRegion> create(Vector<OwnPtr<CSSParserSelector>>* selectors, Vector<RefPtr<StyleRuleBase>>& adoptRules)
{
return adoptRef(*new StyleRuleRegion(selectors, adoptRules));
}
const CSSSelectorList& selectorList() const { return m_selectorList; }
PassRef<StyleRuleRegion> copy() const { return adoptRef(*new StyleRuleRegion(*this)); }
private:
StyleRuleRegion(Vector<OwnPtr<CSSParserSelector>>*, Vector<RefPtr<StyleRuleBase>>& adoptRules);
StyleRuleRegion(const StyleRuleRegion&);
CSSSelectorList m_selectorList;
};
#if ENABLE(SHADOW_DOM)
class StyleRuleHost : public StyleRuleGroup {
public:
static PassRef<StyleRuleHost> create(Vector<RefPtr<StyleRuleBase>>& adoptRules)
{
return adoptRef(*new StyleRuleHost(adoptRules));
}
PassRef<StyleRuleHost> copy() const { return adoptRef(*new StyleRuleHost(*this)); }
private:
StyleRuleHost(Vector<RefPtr<StyleRuleBase>>& adoptRules) : StyleRuleGroup(HostInternal, adoptRules) { }
StyleRuleHost(const StyleRuleHost& o) : StyleRuleGroup(o) { }
};
#endif
#if ENABLE(CSS_DEVICE_ADAPTATION)
class StyleRuleViewport : public StyleRuleBase {
public:
static PassRef<StyleRuleViewport> create(PassRef<StylePropertySet> properties) { return adoptRef(*new StyleRuleViewport(std::move(properties))); }
~StyleRuleViewport();
const StylePropertySet& properties() const { return m_properties.get(); }
MutableStylePropertySet& mutableProperties();
PassRef<StyleRuleViewport> copy() const { return adoptRef(*new StyleRuleViewport(*this)); }
private:
StyleRuleViewport(PassRef<StylePropertySet>);
StyleRuleViewport(const StyleRuleViewport&);
Ref<StylePropertySet> m_properties;
};
#endif // ENABLE(CSS_DEVICE_ADAPTATION)
inline const StyleRuleMedia* toStyleRuleMedia(const StyleRuleGroup* rule)
{
ASSERT_WITH_SECURITY_IMPLICATION(!rule || rule->isMediaRule());
return static_cast<const StyleRuleMedia*>(rule);
}
#if ENABLE(CSS3_CONDITIONAL_RULES)
inline const StyleRuleSupports* toStyleRuleSupports(const StyleRuleGroup* rule)
{
ASSERT_WITH_SECURITY_IMPLICATION(!rule || rule->isSupportsRule());
return static_cast<const StyleRuleSupports*>(rule);
}
#endif
inline const StyleRuleRegion* toStyleRuleRegion(const StyleRuleGroup* rule)
{
ASSERT_WITH_SECURITY_IMPLICATION(!rule || rule->isRegionRule());
return static_cast<const StyleRuleRegion*>(rule);
}
#if ENABLE(CSS_SHADERS)
class StyleRuleFilter : public StyleRuleBase {
public:
static PassRef<StyleRuleFilter> create(const String& filterName, PassRef<StylePropertySet> properties)
{
return adoptRef(*new StyleRuleFilter(filterName, std::move(properties)));
}
~StyleRuleFilter();
const String& filterName() const { return m_filterName; }
const StylePropertySet& properties() const { return m_properties.get(); }
MutableStylePropertySet& mutableProperties();
PassRef<StyleRuleFilter> copy() const { return adoptRef(*new StyleRuleFilter(*this)); }
private:
StyleRuleFilter(const String&, PassRef<StylePropertySet>);
StyleRuleFilter(const StyleRuleFilter&);
String m_filterName;
Ref<StylePropertySet> m_properties;
};
#endif // ENABLE(CSS_SHADERS)
} // namespace WebCore
#endif // StyleRule_h