/* * Copyright (C) 2011 Andreas Kling (kling@webkit.org) * * 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 COMPUTER, 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 COMPUTER, 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 "CSSAspectRatioValue.h" #include "CSSBorderImageSliceValue.h" #include "CSSCalculationValue.h" #include "CSSCanvasValue.h" #include "CSSCrossfadeValue.h" #include "CSSCursorImageValue.h" #include "CSSFontFaceSrcValue.h" #include "CSSFunctionValue.h" #include "CSSGradientValue.h" #include "CSSImageGeneratorValue.h" #include "CSSImageSetValue.h" #include "CSSImageValue.h" #include "CSSInheritedValue.h" #include "CSSInitialValue.h" #include "CSSLineBoxContainValue.h" #include "CSSPrimitiveValue.h" #include "CSSReflectValue.h" #include "CSSTimingFunctionValue.h" #include "CSSUnicodeRangeValue.h" #include "CSSValueList.h" #if ENABLE(CSS_VARIABLES) #include "CSSVariableValue.h" #endif #include "FontValue.h" #include "FontFeatureValue.h" #include "ShadowValue.h" #include "SVGColor.h" #include "SVGPaint.h" #include "WebKitCSSArrayFunctionValue.h" #include "WebKitCSSFilterValue.h" #include "WebKitCSSMatFunctionValue.h" #include "WebKitCSSMixFunctionValue.h" #include "WebKitCSSShaderValue.h" #include "WebKitCSSTransformValue.h" #if ENABLE(SVG) #include "WebKitCSSSVGDocumentValue.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 PassRefPtr 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 && static_cast(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; return CSS_CUSTOM; } void CSSValue::addSubresourceStyleURLs(ListHashSet& urls, const StyleSheetContents* styleSheet) const { // This should get called for internal instances only. ASSERT(!isCSSOMSafe()); if (isPrimitiveValue()) static_cast(this)->addSubresourceStyleURLs(urls, styleSheet); else if (isValueList()) static_cast(this)->addSubresourceStyleURLs(urls, styleSheet); else if (classType() == FontFaceSrcClass) static_cast(this)->addSubresourceStyleURLs(urls, styleSheet); else if (classType() == ReflectClass) static_cast(this)->addSubresourceStyleURLs(urls, styleSheet); } bool CSSValue::hasFailedOrCanceledSubresources() const { // This should get called for internal instances only. ASSERT(!isCSSOMSafe()); if (isValueList()) return static_cast(this)->hasFailedOrCanceledSubresources(); if (classType() == FontFaceSrcClass) return static_cast(this)->hasFailedOrCanceledSubresources(); if (classType() == ImageClass) return static_cast(this)->hasFailedOrCanceledSubresources(); if (classType() == CrossfadeClass) return static_cast(this)->hasFailedOrCanceledSubresources(); #if ENABLE(CSS_IMAGE_SET) if (classType() == ImageSetClass) return static_cast(this)->hasFailedOrCanceledSubresources(); #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 CursorImageClass: 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 PrimitiveClass: return compareCSSValues(*this, other); case ReflectClass: return compareCSSValues(*this, other); case ShadowClass: return compareCSSValues(*this, other); case LinearTimingFunctionClass: 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 #if ENABLE(CSS_FILTERS) case WebKitCSSFilterClass: return compareCSSValues(*this, other); #if ENABLE(CSS_SHADERS) case WebKitCSSArrayFunctionValueClass: return compareCSSValues(*this, other); case WebKitCSSMatFunctionValueClass: return compareCSSValues(*this, other); case WebKitCSSMixFunctionValueClass: return compareCSSValues(*this, other); case WebKitCSSShaderClass: return compareCSSValues(*this, other); #endif #endif #if ENABLE(CSS_VARIABLES) case VariableClass: return compareCSSValues(*this, other); #endif #if ENABLE(SVG) case SVGColorClass: return compareCSSValues(*this, other); case SVGPaintClass: return compareCSSValues(*this, other); case WebKitCSSSVGDocumentClass: return compareCSSValues(*this, other); #endif default: ASSERT_NOT_REACHED(); return false; } } else if (m_classType == ValueListClass && other.m_classType != ValueListClass) return static_cast(this)->equals(other); else if (m_classType != ValueListClass && other.m_classType == ValueListClass) 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 static_cast(this)->customCssText(); case BorderImageSliceClass: return static_cast(this)->customCssText(); case CanvasClass: return static_cast(this)->customCssText(); case CursorImageClass: return static_cast(this)->customCssText(); case FontClass: return static_cast(this)->customCssText(); case FontFaceSrcClass: return static_cast(this)->customCssText(); case FontFeatureClass: return static_cast(this)->customCssText(); case FunctionClass: return static_cast(this)->customCssText(); case LinearGradientClass: return static_cast(this)->customCssText(); case RadialGradientClass: return static_cast(this)->customCssText(); case CrossfadeClass: return static_cast(this)->customCssText(); case ImageClass: return static_cast(this)->customCssText(); case InheritedClass: return static_cast(this)->customCssText(); case InitialClass: return static_cast(this)->customCssText(); case PrimitiveClass: return static_cast(this)->customCssText(); case ReflectClass: return static_cast(this)->customCssText(); case ShadowClass: return static_cast(this)->customCssText(); case LinearTimingFunctionClass: return static_cast(this)->customCssText(); case CubicBezierTimingFunctionClass: return static_cast(this)->customCssText(); case StepsTimingFunctionClass: return static_cast(this)->customCssText(); case UnicodeRangeClass: return static_cast(this)->customCssText(); case ValueListClass: return static_cast(this)->customCssText(); case WebKitCSSTransformClass: return static_cast(this)->customCssText(); case LineBoxContainClass: return static_cast(this)->customCssText(); case CalculationClass: return static_cast(this)->customCssText(); #if ENABLE(CSS_IMAGE_SET) case ImageSetClass: return static_cast(this)->customCssText(); #endif #if ENABLE(CSS_FILTERS) case WebKitCSSFilterClass: return static_cast(this)->customCssText(); #if ENABLE(CSS_SHADERS) case WebKitCSSArrayFunctionValueClass: return static_cast(this)->customCssText(); case WebKitCSSMatFunctionValueClass: return static_cast(this)->customCssText(); case WebKitCSSMixFunctionValueClass: return static_cast(this)->customCssText(); case WebKitCSSShaderClass: return static_cast(this)->customCssText(); #endif #endif #if ENABLE(CSS_VARIABLES) case VariableClass: return static_cast(this)->value(); #endif #if ENABLE(SVG) case SVGColorClass: return static_cast(this)->customCssText(); case SVGPaintClass: return static_cast(this)->customCssText(); case WebKitCSSSVGDocumentClass: return static_cast(this)->customCssText(); #endif } ASSERT_NOT_REACHED(); return String(); } #if ENABLE(CSS_VARIABLES) String CSSValue::serializeResolvingVariables(const HashMap& variables) const { switch (classType()) { case PrimitiveClass: return static_cast(this)->customSerializeResolvingVariables(variables); case ReflectClass: return static_cast(this)->customSerializeResolvingVariables(variables); case ValueListClass: return static_cast(this)->customSerializeResolvingVariables(variables); case WebKitCSSTransformClass: return static_cast(this)->customSerializeResolvingVariables(variables); default: return cssText(); } } #endif void CSSValue::destroy() { if (m_isTextClone) { ASSERT(isCSSOMSafe()); delete static_cast(this); return; } ASSERT(!isCSSOMSafe() || isSubtypeExposedToCSSOM()); switch (classType()) { case AspectRatioClass: delete static_cast(this); return; case BorderImageSliceClass: delete static_cast(this); return; case CanvasClass: delete static_cast(this); return; case CursorImageClass: delete static_cast(this); return; case FontClass: delete static_cast(this); return; case FontFaceSrcClass: delete static_cast(this); return; case FontFeatureClass: delete static_cast(this); return; case FunctionClass: delete static_cast(this); return; case LinearGradientClass: delete static_cast(this); return; case RadialGradientClass: delete static_cast(this); return; case CrossfadeClass: delete static_cast(this); return; case ImageClass: delete static_cast(this); return; case InheritedClass: delete static_cast(this); return; case InitialClass: delete static_cast(this); return; case PrimitiveClass: delete static_cast(this); return; case ReflectClass: delete static_cast(this); return; case ShadowClass: delete static_cast(this); return; case LinearTimingFunctionClass: delete static_cast(this); return; case CubicBezierTimingFunctionClass: delete static_cast(this); return; case StepsTimingFunctionClass: delete static_cast(this); return; case UnicodeRangeClass: delete static_cast(this); return; case ValueListClass: delete static_cast(this); return; case WebKitCSSTransformClass: delete static_cast(this); return; case LineBoxContainClass: delete static_cast(this); return; case CalculationClass: delete static_cast(this); return; #if ENABLE(CSS_IMAGE_SET) case ImageSetClass: delete static_cast(this); return; #endif #if ENABLE(CSS_FILTERS) case WebKitCSSFilterClass: delete static_cast(this); return; #if ENABLE(CSS_SHADERS) case WebKitCSSArrayFunctionValueClass: delete static_cast(this); return; case WebKitCSSMatFunctionValueClass: delete static_cast(this); return; case WebKitCSSMixFunctionValueClass: delete static_cast(this); return; case WebKitCSSShaderClass: delete static_cast(this); return; #endif #endif #if ENABLE(CSS_VARIABLES) case VariableClass: delete static_cast(this); return; #endif #if ENABLE(SVG) case SVGColorClass: delete static_cast(this); return; case SVGPaintClass: delete static_cast(this); return; case WebKitCSSSVGDocumentClass: delete static_cast(this); return; #endif } ASSERT_NOT_REACHED(); } PassRefPtr CSSValue::cloneForCSSOM() const { switch (classType()) { case PrimitiveClass: return static_cast(this)->cloneForCSSOM(); case ValueListClass: return static_cast(this)->cloneForCSSOM(); case ImageClass: case CursorImageClass: return static_cast(this)->cloneForCSSOM(); #if ENABLE(CSS_FILTERS) case WebKitCSSFilterClass: return static_cast(this)->cloneForCSSOM(); #if ENABLE(CSS_SHADERS) case WebKitCSSArrayFunctionValueClass: return static_cast(this)->cloneForCSSOM(); case WebKitCSSMatFunctionValueClass: return static_cast(this)->cloneForCSSOM(); case WebKitCSSMixFunctionValueClass: return static_cast(this)->cloneForCSSOM(); #endif #endif case WebKitCSSTransformClass: return static_cast(this)->cloneForCSSOM(); #if ENABLE(CSS_IMAGE_SET) case ImageSetClass: return static_cast(this)->cloneForCSSOM(); #endif #if ENABLE(SVG) case SVGColorClass: return static_cast(this)->cloneForCSSOM(); case SVGPaintClass: return static_cast(this)->cloneForCSSOM(); #endif default: ASSERT(!isSubtypeExposedToCSSOM()); return TextCloneCSSValue::create(classType(), cssText()); } } }