/* * Copyright (C) 2011 Andreas Kling (kling@webkit.org) * Copyright (C) 2013 Adobe Systems Incorporated. All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. * */ #include "config.h" #include "CSSValue.h" #include "CSSAnimationTriggerScrollValue.h" #include "CSSAspectRatioValue.h" #include "CSSBorderImageSliceValue.h" #include "CSSCalculationValue.h" #include "CSSCanvasValue.h" #include "CSSContentDistributionValue.h" #include "CSSCrossfadeValue.h" #include "CSSCursorImageValue.h" #include "CSSCustomPropertyValue.h" #include "CSSFilterImageValue.h" #include "CSSFontFaceSrcValue.h" #include "CSSFontFeatureValue.h" #include "CSSFontValue.h" #include "CSSFunctionValue.h" #include "CSSGradientValue.h" #include "CSSImageSetValue.h" #include "CSSImageValue.h" #include "CSSInheritedValue.h" #include "CSSInitialValue.h" #include "CSSLineBoxContainValue.h" #include "CSSNamedImageValue.h" #include "CSSPrimitiveValue.h" #include "CSSProperty.h" #include "CSSReflectValue.h" #include "CSSShadowValue.h" #include "CSSTimingFunctionValue.h" #include "CSSUnicodeRangeValue.h" #include "CSSUnsetValue.h" #include "CSSValueList.h" #include "CSSVariableDependentValue.h" #include "CSSVariableValue.h" #include "SVGColor.h" #include "SVGPaint.h" #include "WebKitCSSFilterValue.h" #include "WebKitCSSTransformValue.h" #if ENABLE(CSS_GRID_LAYOUT) #include "CSSGridLineNamesValue.h" #include "CSSGridTemplateAreasValue.h" #endif namespace WebCore { struct SameSizeAsCSSValue : public RefCounted { uint32_t bitfields; }; COMPILE_ASSERT(sizeof(CSSValue) == sizeof(SameSizeAsCSSValue), CSS_value_should_stay_small); class TextCloneCSSValue : public CSSValue { public: static Ref create(ClassType classType, const String& text) { return adoptRef(*new TextCloneCSSValue(classType, text)); } String cssText() const { return m_cssText; } private: TextCloneCSSValue(ClassType classType, const String& text) : CSSValue(classType, /*isCSSOMSafe*/ true) , m_cssText(text) { m_isTextClone = true; } String m_cssText; }; bool CSSValue::isImplicitInitialValue() const { return m_classType == InitialClass && downcast(*this).isImplicit(); } CSSValue::Type CSSValue::cssValueType() const { if (isInheritedValue()) return CSS_INHERIT; if (isPrimitiveValue()) return CSS_PRIMITIVE_VALUE; if (isValueList()) return CSS_VALUE_LIST; if (isInitialValue()) return CSS_INITIAL; if (isUnsetValue()) return CSS_UNSET; if (isRevertValue()) return CSS_REVERT; return CSS_CUSTOM; } void CSSValue::addSubresourceStyleURLs(ListHashSet& urls, const StyleSheetContents* styleSheet) const { // This should get called for internal instances only. ASSERT(!isCSSOMSafe()); if (is(*this)) downcast(*this).addSubresourceStyleURLs(urls, styleSheet); else if (is(*this)) downcast(*this).addSubresourceStyleURLs(urls, styleSheet); else if (is(*this)) downcast(*this).addSubresourceStyleURLs(urls, styleSheet); else if (is(*this)) downcast(*this).addSubresourceStyleURLs(urls, styleSheet); } bool CSSValue::traverseSubresources(const std::function& handler) const { // This should get called for internal instances only. ASSERT(!isCSSOMSafe()); if (is(*this)) return downcast(*this).traverseSubresources(handler); if (is(*this)) return downcast(*this).traverseSubresources(handler); if (is(*this)) return downcast(*this).traverseSubresources(handler); if (is(*this)) return downcast(*this).traverseSubresources(handler); if (is(*this)) return downcast(*this).traverseSubresources(handler); #if ENABLE(CSS_IMAGE_SET) if (is(*this)) return downcast(*this).traverseSubresources(handler); #endif return false; } template inline static bool compareCSSValues(const CSSValue& first, const CSSValue& second) { return static_cast(first).equals(static_cast(second)); } bool CSSValue::equals(const CSSValue& other) const { if (m_isTextClone) { ASSERT(isCSSOMSafe()); return static_cast(this)->cssText() == other.cssText(); } if (m_classType == other.m_classType) { switch (m_classType) { case AspectRatioClass: return compareCSSValues(*this, other); case BorderImageSliceClass: return compareCSSValues(*this, other); case CanvasClass: return compareCSSValues(*this, other); case NamedImageClass: return compareCSSValues(*this, other); case CursorImageClass: return compareCSSValues(*this, other); case FilterImageClass: return compareCSSValues(*this, other); case FontClass: return compareCSSValues(*this, other); case FontFaceSrcClass: return compareCSSValues(*this, other); case FontFeatureClass: return compareCSSValues(*this, other); case FunctionClass: return compareCSSValues(*this, other); case LinearGradientClass: return compareCSSValues(*this, other); case RadialGradientClass: return compareCSSValues(*this, other); case CrossfadeClass: return compareCSSValues(*this, other); case ImageClass: return compareCSSValues(*this, other); case InheritedClass: return compareCSSValues(*this, other); case InitialClass: return compareCSSValues(*this, other); case UnsetClass: return compareCSSValues(*this, other); case RevertClass: return compareCSSValues(*this, other); #if ENABLE(CSS_GRID_LAYOUT) case GridLineNamesClass: return compareCSSValues(*this, other); case GridTemplateAreasClass: return compareCSSValues(*this, other); #endif case PrimitiveClass: return compareCSSValues(*this, other); case ReflectClass: return compareCSSValues(*this, other); case ShadowClass: return compareCSSValues(*this, other); case CubicBezierTimingFunctionClass: return compareCSSValues(*this, other); case StepsTimingFunctionClass: return compareCSSValues(*this, other); case UnicodeRangeClass: return compareCSSValues(*this, other); case ValueListClass: return compareCSSValues(*this, other); case WebKitCSSTransformClass: return compareCSSValues(*this, other); case LineBoxContainClass: return compareCSSValues(*this, other); case CalculationClass: return compareCSSValues(*this, other); #if ENABLE(CSS_IMAGE_SET) case ImageSetClass: return compareCSSValues(*this, other); #endif case WebKitCSSFilterClass: return compareCSSValues(*this, other); case SVGColorClass: return compareCSSValues(*this, other); case SVGPaintClass: return compareCSSValues(*this, other); #if ENABLE(CSS_ANIMATIONS_LEVEL_2) case AnimationTriggerScrollClass: return compareCSSValues(*this, other); #endif case CSSContentDistributionClass: return compareCSSValues(*this, other); case CustomPropertyClass: return compareCSSValues(*this, other); case VariableDependentClass: return compareCSSValues(*this, other); case VariableClass: return compareCSSValues(*this, other); default: ASSERT_NOT_REACHED(); return false; } } else if (is(*this) && !is(other)) return downcast(*this).equals(other); else if (!is(*this) && is(other)) return static_cast(other).equals(*this); return false; } String CSSValue::cssText() const { if (m_isTextClone) { ASSERT(isCSSOMSafe()); return static_cast(this)->cssText(); } ASSERT(!isCSSOMSafe() || isSubtypeExposedToCSSOM()); switch (classType()) { case AspectRatioClass: return downcast(*this).customCSSText(); case BorderImageSliceClass: return downcast(*this).customCSSText(); case CanvasClass: return downcast(*this).customCSSText(); case NamedImageClass: return downcast(*this).customCSSText(); case CursorImageClass: return downcast(*this).customCSSText(); case FilterImageClass: return downcast(*this).customCSSText(); case FontClass: return downcast(*this).customCSSText(); case FontFaceSrcClass: return downcast(*this).customCSSText(); case FontFeatureClass: return downcast(*this).customCSSText(); case FunctionClass: return downcast(*this).customCSSText(); case LinearGradientClass: return downcast(*this).customCSSText(); case RadialGradientClass: return downcast(*this).customCSSText(); case CrossfadeClass: return downcast(*this).customCSSText(); case ImageClass: return downcast(*this).customCSSText(); case InheritedClass: return downcast(*this).customCSSText(); case InitialClass: return downcast(*this).customCSSText(); case UnsetClass: return downcast(*this).customCSSText(); case RevertClass: return downcast(*this).customCSSText(); #if ENABLE(CSS_GRID_LAYOUT) case GridLineNamesClass: return downcast(*this).customCSSText(); case GridTemplateAreasClass: return downcast(*this).customCSSText(); #endif case PrimitiveClass: return downcast(*this).customCSSText(); case ReflectClass: return downcast(*this).customCSSText(); case ShadowClass: return downcast(*this).customCSSText(); case CubicBezierTimingFunctionClass: return downcast(*this).customCSSText(); case StepsTimingFunctionClass: return downcast(*this).customCSSText(); case UnicodeRangeClass: return downcast(*this).customCSSText(); case ValueListClass: return downcast(*this).customCSSText(); case WebKitCSSTransformClass: return downcast(*this).customCSSText(); case LineBoxContainClass: return downcast(*this).customCSSText(); case CalculationClass: return downcast(*this).customCSSText(); #if ENABLE(CSS_IMAGE_SET) case ImageSetClass: return downcast(*this).customCSSText(); #endif case WebKitCSSFilterClass: return downcast(*this).customCSSText(); case SVGColorClass: return downcast(*this).customCSSText(); case SVGPaintClass: return downcast(*this).customCSSText(); #if ENABLE(CSS_ANIMATIONS_LEVEL_2) case AnimationTriggerScrollClass: return downcast(*this).customCSSText(); #endif case CSSContentDistributionClass: return downcast(*this).customCSSText(); case CustomPropertyClass: return downcast(*this).customCSSText(); case VariableDependentClass: return downcast(*this).customCSSText(); case VariableClass: return downcast(*this).customCSSText(); } ASSERT_NOT_REACHED(); return String(); } void CSSValue::destroy() { if (m_isTextClone) { ASSERT(isCSSOMSafe()); delete static_cast(this); return; } ASSERT(!isCSSOMSafe() || isSubtypeExposedToCSSOM()); switch (classType()) { case AspectRatioClass: delete downcast(this); return; case BorderImageSliceClass: delete downcast(this); return; case CanvasClass: delete downcast(this); return; case NamedImageClass: delete downcast(this); return; case CursorImageClass: delete downcast(this); return; case FontClass: delete downcast(this); return; case FontFaceSrcClass: delete downcast(this); return; case FontFeatureClass: delete downcast(this); return; case FunctionClass: delete downcast(this); return; case LinearGradientClass: delete downcast(this); return; case RadialGradientClass: delete downcast(this); return; case CrossfadeClass: delete downcast(this); return; case ImageClass: delete downcast(this); return; case InheritedClass: delete downcast(this); return; case InitialClass: delete downcast(this); return; case UnsetClass: delete downcast(this); return; case RevertClass: delete downcast(this); return; #if ENABLE(CSS_GRID_LAYOUT) case GridLineNamesClass: delete downcast(this); return; case GridTemplateAreasClass: delete downcast(this); return; #endif case PrimitiveClass: delete downcast(this); return; case ReflectClass: delete downcast(this); return; case ShadowClass: delete downcast(this); return; case CubicBezierTimingFunctionClass: delete downcast(this); return; case StepsTimingFunctionClass: delete downcast(this); return; case UnicodeRangeClass: delete downcast(this); return; case ValueListClass: delete downcast(this); return; case WebKitCSSTransformClass: delete downcast(this); return; case LineBoxContainClass: delete downcast(this); return; case CalculationClass: delete downcast(this); return; #if ENABLE(CSS_IMAGE_SET) case ImageSetClass: delete downcast(this); return; #endif case FilterImageClass: delete downcast(this); return; case WebKitCSSFilterClass: delete downcast(this); return; case SVGColorClass: delete downcast(this); return; case SVGPaintClass: delete downcast(this); return; #if ENABLE(CSS_ANIMATIONS_LEVEL_2) case AnimationTriggerScrollClass: delete downcast(this); return; #endif case CSSContentDistributionClass: delete downcast(this); return; case CustomPropertyClass: delete downcast(this); return; case VariableDependentClass: delete downcast(this); return; case VariableClass: delete downcast(this); return; } ASSERT_NOT_REACHED(); } RefPtr CSSValue::cloneForCSSOM() const { switch (classType()) { case PrimitiveClass: return downcast(*this).cloneForCSSOM(); case ValueListClass: return downcast(*this).cloneForCSSOM(); case ImageClass: case CursorImageClass: return downcast(*this).cloneForCSSOM(); case WebKitCSSFilterClass: return downcast(*this).cloneForCSSOM(); case WebKitCSSTransformClass: return downcast(*this).cloneForCSSOM(); #if ENABLE(CSS_IMAGE_SET) case ImageSetClass: return downcast(*this).cloneForCSSOM(); #endif case SVGColorClass: return downcast(*this).cloneForCSSOM(); case SVGPaintClass: return downcast(*this).cloneForCSSOM(); default: ASSERT(!isSubtypeExposedToCSSOM()); return TextCloneCSSValue::create(classType(), cssText()); } } bool CSSValue::isInvalidCustomPropertyValue() const { return isCustomPropertyValue() && downcast(*this).isInvalid(); } bool CSSValue::treatAsInheritedValue(CSSPropertyID propertyID) const { return classType() == InheritedClass || (classType() == UnsetClass && CSSProperty::isInheritedProperty(propertyID)); } bool CSSValue::treatAsInitialValue(CSSPropertyID propertyID) const { return classType() == InitialClass || (classType() == UnsetClass && !CSSProperty::isInheritedProperty(propertyID)); } }