/* * (C) 1999-2003 Lars Knoll (knoll@kde.org) * (C) 2002-2003 Dirk Mueller (mueller@kde.org) * Copyright (C) 2002, 2005, 2006, 2008, 2012 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. */ #include "config.h" #include "StyleRule.h" #include "CSSCharsetRule.h" #include "CSSFontFaceRule.h" #include "CSSHostRule.h" #include "CSSImportRule.h" #include "CSSMediaRule.h" #include "CSSPageRule.h" #include "CSSStyleRule.h" #include "CSSSupportsRule.h" #include "CSSUnknownRule.h" #include "StylePropertySet.h" #include "StyleRuleImport.h" #include "WebKitCSSFilterRule.h" #include "WebKitCSSKeyframeRule.h" #include "WebKitCSSKeyframesRule.h" #include "WebKitCSSRegionRule.h" #include "WebKitCSSViewportRule.h" namespace WebCore { struct SameSizeAsStyleRuleBase : public WTF::RefCountedBase { unsigned bitfields; }; COMPILE_ASSERT(sizeof(StyleRuleBase) == sizeof(SameSizeAsStyleRuleBase), StyleRuleBase_should_stay_small); PassRefPtr StyleRuleBase::createCSSOMWrapper(CSSStyleSheet* parentSheet) const { return createCSSOMWrapper(parentSheet, 0); } PassRefPtr StyleRuleBase::createCSSOMWrapper(CSSRule* parentRule) const { return createCSSOMWrapper(0, parentRule); } void StyleRuleBase::destroy() { switch (type()) { case Style: delete static_cast(this); return; case Page: delete static_cast(this); return; case FontFace: delete static_cast(this); return; case Media: delete static_cast(this); return; #if ENABLE(CSS3_CONDITIONAL_RULES) case Supports: delete static_cast(this); return; #endif #if ENABLE(CSS_REGIONS) case Region: delete static_cast(this); return; #endif case Import: delete static_cast(this); return; case Keyframes: delete static_cast(this); return; #if ENABLE(SHADOW_DOM) case HostInternal: delete static_cast(this); return; #endif #if ENABLE(CSS_DEVICE_ADAPTATION) case Viewport: delete static_cast(this); return; #endif #if ENABLE(CSS_SHADERS) case Filter: delete static_cast(this); return; #endif case Unknown: case Charset: case Keyframe: #if !ENABLE(CSS_REGIONS) case Region: #endif ASSERT_NOT_REACHED(); return; } ASSERT_NOT_REACHED(); } PassRefPtr StyleRuleBase::copy() const { switch (type()) { case Style: return static_cast(this)->copy(); case Page: return static_cast(this)->copy(); case FontFace: return static_cast(this)->copy(); case Media: return static_cast(this)->copy(); #if ENABLE(CSS3_CONDITIONAL_RULES) case Supports: return static_cast(this)->copy(); #endif #if ENABLE(CSS_REGIONS) case Region: return static_cast(this)->copy(); #endif case Import: // FIXME: Copy import rules. ASSERT_NOT_REACHED(); return 0; case Keyframes: return static_cast(this)->copy(); #if ENABLE(SHADOW_DOM) case HostInternal: return static_cast(this)->copy(); #endif #if ENABLE(CSS_DEVICE_ADAPTATION) case Viewport: return static_cast(this)->copy(); #endif #if ENABLE(CSS_SHADERS) case Filter: return static_cast(this)->copy(); #endif case Unknown: case Charset: case Keyframe: #if !ENABLE(CSS_REGIONS) case Region: #endif ASSERT_NOT_REACHED(); return 0; } ASSERT_NOT_REACHED(); return 0; } PassRefPtr StyleRuleBase::createCSSOMWrapper(CSSStyleSheet* parentSheet, CSSRule* parentRule) const { RefPtr rule; StyleRuleBase* self = const_cast(this); switch (type()) { case Style: rule = CSSStyleRule::create(static_cast(self), parentSheet); break; case Page: rule = CSSPageRule::create(static_cast(self), parentSheet); break; case FontFace: rule = CSSFontFaceRule::create(static_cast(self), parentSheet); break; case Media: rule = CSSMediaRule::create(static_cast(self), parentSheet); break; #if ENABLE(CSS3_CONDITIONAL_RULES) case Supports: rule = CSSSupportsRule::create(static_cast(self), parentSheet); break; #endif #if ENABLE(CSS_REGIONS) case Region: rule = WebKitCSSRegionRule::create(static_cast(self), parentSheet); break; #endif case Import: rule = CSSImportRule::create(static_cast(self), parentSheet); break; case Keyframes: rule = WebKitCSSKeyframesRule::create(static_cast(self), parentSheet); break; #if ENABLE(CSS_DEVICE_ADAPTATION) case Viewport: rule = WebKitCSSViewportRule::create(static_cast(self), parentSheet); break; #endif #if ENABLE(SHADOW_DOM) case HostInternal: rule = CSSHostRule::create(static_cast(self), parentSheet); break; #endif #if ENABLE(CSS_SHADERS) case Filter: rule = WebKitCSSFilterRule::create(static_cast(self), parentSheet); break; #endif case Unknown: case Charset: case Keyframe: #if !ENABLE(CSS_REGIONS) case Region: #endif ASSERT_NOT_REACHED(); return 0; } if (parentRule) rule->setParentRule(parentRule); return rule.release(); } unsigned StyleRule::averageSizeInBytes() { return sizeof(StyleRule) + sizeof(CSSSelector) + StylePropertySet::averageSizeInBytes(); } StyleRule::StyleRule(int sourceLine) : StyleRuleBase(Style, sourceLine) { } StyleRule::StyleRule(const StyleRule& o) : StyleRuleBase(o) , m_properties(o.m_properties->mutableCopy()) , m_selectorList(o.m_selectorList) { } StyleRule::~StyleRule() { } MutableStylePropertySet* StyleRule::mutableProperties() { if (!m_properties->isMutable()) m_properties = m_properties->mutableCopy(); return static_cast(m_properties.get()); } void StyleRule::setProperties(PassRefPtr properties) { m_properties = properties; } PassRefPtr StyleRule::create(int sourceLine, const Vector& selectors, PassRefPtr properties) { ASSERT_WITH_SECURITY_IMPLICATION(!selectors.isEmpty()); CSSSelector* selectorListArray = reinterpret_cast(fastMalloc(sizeof(CSSSelector) * selectors.size())); for (unsigned i = 0; i < selectors.size(); ++i) new (NotNull, &selectorListArray[i]) CSSSelector(*selectors.at(i)); selectorListArray[selectors.size() - 1].setLastInSelectorList(); RefPtr rule = StyleRule::create(sourceLine); rule->parserAdoptSelectorArray(selectorListArray); rule->setProperties(properties); return rule.release(); } Vector > StyleRule::splitIntoMultipleRulesWithMaximumSelectorComponentCount(unsigned maxCount) const { ASSERT(selectorList().componentCount() > maxCount); Vector > rules; Vector componentsSinceLastSplit; for (const CSSSelector* selector = selectorList().first(); selector; selector = CSSSelectorList::next(selector)) { Vector componentsInThisSelector; for (const CSSSelector* component = selector; component; component = component->tagHistory()) componentsInThisSelector.append(component); if (componentsInThisSelector.size() + componentsSinceLastSplit.size() > maxCount && !componentsSinceLastSplit.isEmpty()) { rules.append(create(sourceLine(), componentsSinceLastSplit, m_properties)); componentsSinceLastSplit.clear(); } componentsSinceLastSplit.appendVector(componentsInThisSelector); } if (!componentsSinceLastSplit.isEmpty()) rules.append(create(sourceLine(), componentsSinceLastSplit, m_properties)); return rules; } StyleRulePage::StyleRulePage() : StyleRuleBase(Page) { } StyleRulePage::StyleRulePage(const StyleRulePage& o) : StyleRuleBase(o) , m_properties(o.m_properties->mutableCopy()) , m_selectorList(o.m_selectorList) { } StyleRulePage::~StyleRulePage() { } MutableStylePropertySet* StyleRulePage::mutableProperties() { if (!m_properties->isMutable()) m_properties = m_properties->mutableCopy(); return static_cast(m_properties.get()); } void StyleRulePage::setProperties(PassRefPtr properties) { m_properties = properties; } StyleRuleFontFace::StyleRuleFontFace() : StyleRuleBase(FontFace, 0) { } StyleRuleFontFace::StyleRuleFontFace(const StyleRuleFontFace& o) : StyleRuleBase(o) , m_properties(o.m_properties->mutableCopy()) { } StyleRuleFontFace::~StyleRuleFontFace() { } MutableStylePropertySet* StyleRuleFontFace::mutableProperties() { if (!m_properties->isMutable()) m_properties = m_properties->mutableCopy(); return static_cast(m_properties.get()); } void StyleRuleFontFace::setProperties(PassRefPtr properties) { m_properties = properties; } StyleRuleGroup::StyleRuleGroup(Type type, Vector >& adoptRule) : StyleRuleBase(type, 0) { m_childRules.swap(adoptRule); } StyleRuleGroup::StyleRuleGroup(const StyleRuleGroup& o) : StyleRuleBase(o) , m_childRules(o.m_childRules.size()) { for (unsigned i = 0; i < m_childRules.size(); ++i) m_childRules[i] = o.m_childRules[i]->copy(); } void StyleRuleGroup::wrapperInsertRule(unsigned index, PassRefPtr rule) { m_childRules.insert(index, rule); } void StyleRuleGroup::wrapperRemoveRule(unsigned index) { m_childRules.remove(index); } StyleRuleMedia::StyleRuleMedia(PassRefPtr media, Vector >& adoptRules) : StyleRuleGroup(Media, adoptRules) , m_mediaQueries(media) { } StyleRuleMedia::StyleRuleMedia(const StyleRuleMedia& o) : StyleRuleGroup(o) { if (o.m_mediaQueries) m_mediaQueries = o.m_mediaQueries->copy(); } #if ENABLE(CSS3_CONDITIONAL_RULES) StyleRuleSupports::StyleRuleSupports(const String& conditionText, bool conditionIsSupported, Vector >& adoptRules) : StyleRuleGroup(Supports, adoptRules) , m_conditionText(conditionText) , m_conditionIsSupported(conditionIsSupported) { } StyleRuleSupports::StyleRuleSupports(const StyleRuleSupports& o) : StyleRuleGroup(o) , m_conditionText(o.m_conditionText) , m_conditionIsSupported(o.m_conditionIsSupported) { } #endif StyleRuleRegion::StyleRuleRegion(Vector >* selectors, Vector >& adoptRules) : StyleRuleGroup(Region, adoptRules) { m_selectorList.adoptSelectorVector(*selectors); } StyleRuleRegion::StyleRuleRegion(const StyleRuleRegion& o) : StyleRuleGroup(o) , m_selectorList(o.m_selectorList) { } #if ENABLE(CSS_DEVICE_ADAPTATION) StyleRuleViewport::StyleRuleViewport() : StyleRuleBase(Viewport, 0) { } StyleRuleViewport::StyleRuleViewport(const StyleRuleViewport& o) : StyleRuleBase(o) , m_properties(o.m_properties->mutableCopy()) { } StyleRuleViewport::~StyleRuleViewport() { } MutableStylePropertySet* StyleRuleViewport::mutableProperties() { if (!m_properties->isMutable()) m_properties = m_properties->mutableCopy(); return static_cast(m_properties.get()); } void StyleRuleViewport::setProperties(PassRefPtr properties) { m_properties = properties; } #endif // ENABLE(CSS_DEVICE_ADAPTATION) #if ENABLE(CSS_SHADERS) StyleRuleFilter::StyleRuleFilter(const String& filterName) : StyleRuleBase(Filter, 0) , m_filterName(filterName) { } StyleRuleFilter::StyleRuleFilter(const StyleRuleFilter& o) : StyleRuleBase(o) , m_filterName(o.m_filterName) , m_properties(o.m_properties->mutableCopy()) { } StyleRuleFilter::~StyleRuleFilter() { } MutableStylePropertySet* StyleRuleFilter::mutableProperties() { if (!m_properties->isMutable()) m_properties = m_properties->mutableCopy(); return static_cast(m_properties.get()); } void StyleRuleFilter::setProperties(PassRefPtr properties) { m_properties = properties; } #endif // ENABLE(CSS_SHADERS) } // namespace WebCore