/* * (C) 1999-2003 Lars Knoll (knoll@kde.org) * Copyright (C) 2004, 2005, 2006, 2007, 2008, 2009, 2010, 2012 Apple Inc. All rights reserved. * Copyright (C) 2011 Research In Motion Limited. 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 "StylePropertySet.h" #include "CSSParser.h" #include "CSSValueKeywords.h" #include "CSSValueList.h" #include "CSSValuePool.h" #include "Document.h" #include "PropertySetCSSStyleDeclaration.h" #include "StylePropertyShorthand.h" #include "StyleSheetContents.h" #include #include #ifndef NDEBUG #include #include #include #endif using namespace std; namespace WebCore { typedef HashMap > PropertySetCSSOMWrapperMap; static PropertySetCSSOMWrapperMap& propertySetCSSOMWrapperMap() { DEFINE_STATIC_LOCAL(PropertySetCSSOMWrapperMap, propertySetCSSOMWrapperMapInstance, ()); return propertySetCSSOMWrapperMapInstance; } StylePropertySet::StylePropertySet(CSSParserMode cssParserMode) : m_cssParserMode(cssParserMode) , m_ownsCSSOMWrapper(false) { } StylePropertySet::StylePropertySet(StylePropertyVector& properties, CSSParserMode cssParserMode) : m_cssParserMode(cssParserMode) , m_ownsCSSOMWrapper(false) { m_properties.swap(properties); // This shrinkToFit() will be a no-op in the typical case (no duplicate properties were eliminated after parsing.) m_properties.shrinkToFit(); } StylePropertySet::StylePropertySet(const StylePropertySet& o) : RefCounted() , m_properties(o.m_properties) , m_cssParserMode(o.m_cssParserMode) , m_ownsCSSOMWrapper(false) { } StylePropertySet::~StylePropertySet() { ASSERT(!m_ownsCSSOMWrapper || propertySetCSSOMWrapperMap().contains(this)); if (m_ownsCSSOMWrapper) propertySetCSSOMWrapperMap().remove(this); } void StylePropertySet::copyPropertiesFrom(const StylePropertySet& other) { m_properties = other.m_properties; } String StylePropertySet::getPropertyValue(CSSPropertyID propertyID) const { RefPtr value = getPropertyCSSValue(propertyID); if (value) return value->cssText(); // Shorthand and 4-values properties switch (propertyID) { case CSSPropertyBorderSpacing: return borderSpacingValue(borderSpacingShorthand()); case CSSPropertyBackgroundPosition: return getLayeredShorthandValue(backgroundPositionShorthand()); case CSSPropertyBackgroundRepeat: return getLayeredShorthandValue(backgroundRepeatShorthand()); case CSSPropertyBackground: return getLayeredShorthandValue(backgroundShorthand()); case CSSPropertyBorder: return borderPropertyValue(OmitUncommonValues); case CSSPropertyBorderTop: return getShorthandValue(borderTopShorthand()); case CSSPropertyBorderRight: return getShorthandValue(borderRightShorthand()); case CSSPropertyBorderBottom: return getShorthandValue(borderBottomShorthand()); case CSSPropertyBorderLeft: return getShorthandValue(borderLeftShorthand()); case CSSPropertyOutline: return getShorthandValue(outlineShorthand()); case CSSPropertyBorderColor: return get4Values(borderColorShorthand()); case CSSPropertyBorderWidth: return get4Values(borderWidthShorthand()); case CSSPropertyBorderStyle: return get4Values(borderStyleShorthand()); case CSSPropertyWebkitFlexFlow: return getShorthandValue(webkitFlexFlowShorthand()); case CSSPropertyFont: return fontValue(); case CSSPropertyMargin: return get4Values(marginShorthand()); case CSSPropertyOverflow: return getCommonValue(overflowShorthand()); case CSSPropertyPadding: return get4Values(paddingShorthand()); case CSSPropertyListStyle: return getShorthandValue(listStyleShorthand()); case CSSPropertyWebkitMaskPosition: return getLayeredShorthandValue(webkitMaskPositionShorthand()); case CSSPropertyWebkitMaskRepeat: return getLayeredShorthandValue(webkitMaskRepeatShorthand()); case CSSPropertyWebkitMask: return getLayeredShorthandValue(webkitMaskShorthand()); case CSSPropertyWebkitTransformOrigin: return getShorthandValue(webkitTransformOriginShorthand()); case CSSPropertyWebkitTransition: return getLayeredShorthandValue(webkitTransitionShorthand()); case CSSPropertyWebkitAnimation: return getLayeredShorthandValue(webkitAnimationShorthand()); #if ENABLE(CSS_EXCLUSIONS) case CSSPropertyWebkitWrap: return getShorthandValue(webkitWrapShorthand()); #endif #if ENABLE(SVG) case CSSPropertyMarker: { RefPtr value = getPropertyCSSValue(CSSPropertyMarkerStart); if (value) return value->cssText(); return String(); } #endif case CSSPropertyBorderRadius: return get4Values(borderRadiusShorthand()); default: return String(); } } String StylePropertySet::borderSpacingValue(const StylePropertyShorthand& shorthand) const { RefPtr horizontalValue = getPropertyCSSValue(shorthand.properties()[0]); RefPtr verticalValue = getPropertyCSSValue(shorthand.properties()[1]); // While standard border-spacing property does not allow specifying border-spacing-vertical without // specifying border-spacing-horizontal , // -webkit-border-spacing-vertical can be set without -webkit-border-spacing-horizontal. if (!horizontalValue || !verticalValue) return String(); String horizontalValueCSSText = horizontalValue->cssText(); String verticalValueCSSText = verticalValue->cssText(); if (horizontalValueCSSText == verticalValueCSSText) return horizontalValueCSSText; return horizontalValueCSSText + ' ' + verticalValueCSSText; } bool StylePropertySet::appendFontLonghandValueIfExplicit(CSSPropertyID propertyId, StringBuilder& result) const { const CSSProperty* property = findPropertyWithId(propertyId); if (!property) return false; // All longhands must have at least implicit values if "font" is specified. if (property->isImplicit()) return true; char prefix = '\0'; switch (propertyId) { case CSSPropertyFontStyle: break; // No prefix. case CSSPropertyFontFamily: case CSSPropertyFontVariant: case CSSPropertyFontWeight: prefix = ' '; break; case CSSPropertyLineHeight: prefix = '/'; break; default: ASSERT_NOT_REACHED(); } if (prefix && !result.isEmpty()) result.append(prefix); result.append(property->value()->cssText()); return true; } String StylePropertySet::fontValue() const { const CSSProperty* fontSizeProperty = findPropertyWithId(CSSPropertyFontSize); if (!fontSizeProperty || fontSizeProperty->isImplicit()) return emptyString(); StringBuilder result; bool success = true; success &= appendFontLonghandValueIfExplicit(CSSPropertyFontStyle, result); success &= appendFontLonghandValueIfExplicit(CSSPropertyFontVariant, result); success &= appendFontLonghandValueIfExplicit(CSSPropertyFontWeight, result); if (!result.isEmpty()) result.append(' '); result.append(fontSizeProperty->value()->cssText()); success &= appendFontLonghandValueIfExplicit(CSSPropertyLineHeight, result); success &= appendFontLonghandValueIfExplicit(CSSPropertyFontFamily, result); if (!success) { // An invalid "font" value has been built (should never happen, as at least implicit values // for mandatory longhands are always found in the style), report empty value instead. ASSERT_NOT_REACHED(); return emptyString(); } return result.toString(); } String StylePropertySet::get4Values(const StylePropertyShorthand& shorthand) const { // Assume the properties are in the usual order top, right, bottom, left. const CSSProperty* top = findPropertyWithId(shorthand.properties()[0]); const CSSProperty* right = findPropertyWithId(shorthand.properties()[1]); const CSSProperty* bottom = findPropertyWithId(shorthand.properties()[2]); const CSSProperty* left = findPropertyWithId(shorthand.properties()[3]); // All 4 properties must be specified. if (!top || !top->value() || !right || !right->value() || !bottom || !bottom->value() || !left || !left->value()) return String(); if (top->value()->isInitialValue() || right->value()->isInitialValue() || bottom->value()->isInitialValue() || left->value()->isInitialValue()) return String(); if (top->isImportant() != right->isImportant() || right->isImportant() != bottom->isImportant() || bottom->isImportant() != left->isImportant()) return String(); bool showLeft = right->value()->cssText() != left->value()->cssText(); bool showBottom = (top->value()->cssText() != bottom->value()->cssText()) || showLeft; bool showRight = (top->value()->cssText() != right->value()->cssText()) || showBottom; String res = top->value()->cssText(); if (showRight) res += " " + right->value()->cssText(); if (showBottom) res += " " + bottom->value()->cssText(); if (showLeft) res += " " + left->value()->cssText(); return res; } String StylePropertySet::getLayeredShorthandValue(const StylePropertyShorthand& shorthand) const { String res; const unsigned size = shorthand.length(); // Begin by collecting the properties into an array. Vector< RefPtr > values(size); size_t numLayers = 0; for (unsigned i = 0; i < size; ++i) { values[i] = getPropertyCSSValue(shorthand.properties()[i]); if (values[i]) { if (values[i]->isValueList()) { CSSValueList* valueList = static_cast(values[i].get()); numLayers = max(valueList->length(), numLayers); } else numLayers = max(1U, numLayers); } } // Now stitch the properties together. Implicit initial values are flagged as such and // can safely be omitted. for (size_t i = 0; i < numLayers; i++) { String layerRes; bool useRepeatXShorthand = false; bool useRepeatYShorthand = false; bool useSingleWordShorthand = false; bool foundBackgroundPositionYCSSProperty = false; for (unsigned j = 0; j < size; j++) { RefPtr value; if (values[j]) { if (values[j]->isValueList()) value = static_cast(values[j].get())->item(i); else { value = values[j]; // Color only belongs in the last layer. if (shorthand.properties()[j] == CSSPropertyBackgroundColor) { if (i != numLayers - 1) value = 0; } else if (i != 0) // Other singletons only belong in the first layer. value = 0; } } // We need to report background-repeat as it was written in the CSS. If the property is implicit, // then it was written with only one value. Here we figure out which value that was so we can // report back correctly. if (shorthand.properties()[j] == CSSPropertyBackgroundRepeatX && isPropertyImplicit(shorthand.properties()[j])) { // BUG 49055: make sure the value was not reset in the layer check just above. if (j < size - 1 && shorthand.properties()[j + 1] == CSSPropertyBackgroundRepeatY && value) { RefPtr yValue; RefPtr nextValue = values[j + 1]; if (nextValue->isValueList()) yValue = static_cast(nextValue.get())->itemWithoutBoundsCheck(i); else yValue = nextValue; int xId = static_cast(value.get())->getIdent(); int yId = static_cast(yValue.get())->getIdent(); if (xId != yId) { if (xId == CSSValueRepeat && yId == CSSValueNoRepeat) { useRepeatXShorthand = true; ++j; } else if (xId == CSSValueNoRepeat && yId == CSSValueRepeat) { useRepeatYShorthand = true; continue; } } else { useSingleWordShorthand = true; ++j; } } } if (value && !value->isImplicitInitialValue()) { if (!layerRes.isNull()) layerRes += " "; if (foundBackgroundPositionYCSSProperty && shorthand.properties()[j] == CSSPropertyBackgroundSize) layerRes += "/ "; if (!foundBackgroundPositionYCSSProperty && shorthand.properties()[j] == CSSPropertyBackgroundSize) continue; if (useRepeatXShorthand) { useRepeatXShorthand = false; layerRes += getValueName(CSSValueRepeatX); } else if (useRepeatYShorthand) { useRepeatYShorthand = false; layerRes += getValueName(CSSValueRepeatY); } else if (useSingleWordShorthand) { useSingleWordShorthand = false; layerRes += value->cssText(); } else layerRes += value->cssText(); if (shorthand.properties()[j] == CSSPropertyBackgroundPositionY) foundBackgroundPositionYCSSProperty = true; } } if (!layerRes.isNull()) { if (!res.isNull()) res += ", "; res += layerRes; } } return res; } String StylePropertySet::getShorthandValue(const StylePropertyShorthand& shorthand) const { String res; for (unsigned i = 0; i < shorthand.length(); ++i) { if (!isPropertyImplicit(shorthand.properties()[i])) { RefPtr value = getPropertyCSSValue(shorthand.properties()[i]); if (!value) return String(); if (value->isInitialValue()) continue; if (!res.isNull()) res += " "; res += value->cssText(); } } return res; } // only returns a non-null value if all properties have the same, non-null value String StylePropertySet::getCommonValue(const StylePropertyShorthand& shorthand) const { String res; bool lastPropertyWasImportant = false; for (unsigned i = 0; i < shorthand.length(); ++i) { RefPtr value = getPropertyCSSValue(shorthand.properties()[i]); // FIXME: CSSInitialValue::cssText should generate the right value. if (!value) return String(); String text = value->cssText(); if (text.isNull()) return String(); if (res.isNull()) res = text; else if (res != text) return String(); bool currentPropertyIsImportant = propertyIsImportant(shorthand.properties()[i]); if (i && lastPropertyWasImportant != currentPropertyIsImportant) return String(); lastPropertyWasImportant = currentPropertyIsImportant; } return res; } String StylePropertySet::borderPropertyValue(CommonValueMode valueMode) const { const StylePropertyShorthand properties[3] = { borderWidthShorthand(), borderStyleShorthand(), borderColorShorthand() }; StringBuilder result; for (size_t i = 0; i < WTF_ARRAY_LENGTH(properties); ++i) { String value = getCommonValue(properties[i]); if (value.isNull()) { if (valueMode == ReturnNullOnUncommonValues) return String(); ASSERT(valueMode == OmitUncommonValues); continue; } if (value == "initial") continue; if (!result.isEmpty()) result.append(' '); result.append(value); } return result.isEmpty() ? String() : result.toString(); } PassRefPtr StylePropertySet::getPropertyCSSValue(CSSPropertyID propertyID) const { const CSSProperty* property = findPropertyWithId(propertyID); return property ? property->value() : 0; } bool StylePropertySet::removeShorthandProperty(CSSPropertyID propertyID) { StylePropertyShorthand shorthand = shorthandForProperty(propertyID); if (!shorthand.length()) return false; return removePropertiesInSet(shorthand.properties(), shorthand.length()); } bool StylePropertySet::removeProperty(CSSPropertyID propertyID, String* returnText) { if (removeShorthandProperty(propertyID)) { // FIXME: Return an equivalent shorthand when possible. if (returnText) *returnText = ""; return true; } CSSProperty* foundProperty = findPropertyWithId(propertyID); if (!foundProperty) { if (returnText) *returnText = ""; return false; } if (returnText) *returnText = foundProperty->value()->cssText(); // A more efficient removal strategy would involve marking entries as empty // and sweeping them when the vector grows too big. m_properties.remove(foundProperty - m_properties.data()); return true; } bool StylePropertySet::propertyIsImportant(CSSPropertyID propertyID) const { const CSSProperty* property = findPropertyWithId(propertyID); if (property) return property->isImportant(); StylePropertyShorthand shorthand = shorthandForProperty(propertyID); if (!shorthand.length()) return false; for (unsigned i = 0; i < shorthand.length(); ++i) { if (!propertyIsImportant(shorthand.properties()[i])) return false; } return true; } CSSPropertyID StylePropertySet::getPropertyShorthand(CSSPropertyID propertyID) const { const CSSProperty* property = findPropertyWithId(propertyID); return property ? property->shorthandID() : CSSPropertyInvalid; } bool StylePropertySet::isPropertyImplicit(CSSPropertyID propertyID) const { const CSSProperty* property = findPropertyWithId(propertyID); return property ? property->isImplicit() : false; } bool StylePropertySet::setProperty(CSSPropertyID propertyID, const String& value, bool important, StyleSheetContents* contextStyleSheet) { // Setting the value to an empty string just removes the property in both IE and Gecko. // Setting it to null seems to produce less consistent results, but we treat it just the same. if (value.isEmpty()) { removeProperty(propertyID); return true; } // When replacing an existing property value, this moves the property to the end of the list. // Firefox preserves the position, and MSIE moves the property to the beginning. return CSSParser::parseValue(this, propertyID, value, important, cssParserMode(), contextStyleSheet); } void StylePropertySet::setProperty(CSSPropertyID propertyID, PassRefPtr prpValue, bool important) { StylePropertyShorthand shorthand = shorthandForProperty(propertyID); if (!shorthand.length()) { setProperty(CSSProperty(propertyID, prpValue, important)); return; } removePropertiesInSet(shorthand.properties(), shorthand.length()); RefPtr value = prpValue; for (unsigned i = 0; i < shorthand.length(); ++i) m_properties.append(CSSProperty(shorthand.properties()[i], value, important)); } void StylePropertySet::setProperty(const CSSProperty& property, CSSProperty* slot) { if (!removeShorthandProperty(property.id())) { CSSProperty* toReplace = slot ? slot : findPropertyWithId(property.id()); if (toReplace) { *toReplace = property; return; } } m_properties.append(property); } bool StylePropertySet::setProperty(CSSPropertyID propertyID, int identifier, bool important) { setProperty(CSSProperty(propertyID, cssValuePool().createIdentifierValue(identifier), important)); return true; } void StylePropertySet::parseDeclaration(const String& styleDeclaration, StyleSheetContents* contextStyleSheet) { m_properties.clear(); CSSParserContext context(cssParserMode()); if (contextStyleSheet) { context = contextStyleSheet->parserContext(); context.mode = cssParserMode(); } CSSParser parser(context); parser.parseDeclaration(this, styleDeclaration, 0, contextStyleSheet); } void StylePropertySet::addParsedProperties(const Vector& properties) { m_properties.reserveCapacity(m_properties.size() + properties.size()); for (unsigned i = 0; i < properties.size(); ++i) addParsedProperty(properties[i]); } void StylePropertySet::addParsedProperty(const CSSProperty& property) { // Only add properties that have no !important counterpart present if (!propertyIsImportant(property.id()) || property.isImportant()) setProperty(property); } String StylePropertySet::asText() const { StringBuilder result; const CSSProperty* positionXProp = 0; const CSSProperty* positionYProp = 0; const CSSProperty* repeatXProp = 0; const CSSProperty* repeatYProp = 0; BitArray shorthandPropertyUsed; BitArray shorthandPropertyAppeared; unsigned size = m_properties.size(); for (unsigned n = 0; n < size; ++n) { const CSSProperty& prop = m_properties[n]; CSSPropertyID propertyID = prop.id(); CSSPropertyID shorthandPropertyID = CSSPropertyInvalid; CSSPropertyID borderFallbackShorthandProperty = CSSPropertyInvalid; String value; switch (propertyID) { case CSSPropertyBackgroundPositionX: positionXProp = ∝ continue; case CSSPropertyBackgroundPositionY: positionYProp = ∝ continue; case CSSPropertyBackgroundRepeatX: repeatXProp = ∝ continue; case CSSPropertyBackgroundRepeatY: repeatYProp = ∝ continue; case CSSPropertyBorderTopWidth: case CSSPropertyBorderRightWidth: case CSSPropertyBorderBottomWidth: case CSSPropertyBorderLeftWidth: if (!borderFallbackShorthandProperty) borderFallbackShorthandProperty = CSSPropertyBorderWidth; case CSSPropertyBorderTopStyle: case CSSPropertyBorderRightStyle: case CSSPropertyBorderBottomStyle: case CSSPropertyBorderLeftStyle: if (!borderFallbackShorthandProperty) borderFallbackShorthandProperty = CSSPropertyBorderStyle; case CSSPropertyBorderTopColor: case CSSPropertyBorderRightColor: case CSSPropertyBorderBottomColor: case CSSPropertyBorderLeftColor: if (!borderFallbackShorthandProperty) borderFallbackShorthandProperty = CSSPropertyBorderColor; // FIXME: Deal with cases where only some of border-(top|right|bottom|left) are specified. if (!shorthandPropertyAppeared.get(CSSPropertyBorder - firstCSSProperty)) { value = borderPropertyValue(ReturnNullOnUncommonValues); if (value.isNull()) shorthandPropertyAppeared.set(CSSPropertyBorder - firstCSSProperty); else shorthandPropertyID = CSSPropertyBorder; } else if (shorthandPropertyUsed.get(CSSPropertyBorder - firstCSSProperty)) shorthandPropertyID = CSSPropertyBorder; if (!shorthandPropertyID) shorthandPropertyID = borderFallbackShorthandProperty; break; case CSSPropertyWebkitBorderHorizontalSpacing: case CSSPropertyWebkitBorderVerticalSpacing: shorthandPropertyID = CSSPropertyBorderSpacing; break; case CSSPropertyFontFamily: case CSSPropertyLineHeight: case CSSPropertyFontSize: case CSSPropertyFontStyle: case CSSPropertyFontVariant: case CSSPropertyFontWeight: // Don't use CSSPropertyFont because old UAs can't recognize them but are important for editing. break; case CSSPropertyListStyleType: case CSSPropertyListStylePosition: case CSSPropertyListStyleImage: shorthandPropertyID = CSSPropertyListStyle; break; case CSSPropertyMarginTop: case CSSPropertyMarginRight: case CSSPropertyMarginBottom: case CSSPropertyMarginLeft: shorthandPropertyID = CSSPropertyMargin; break; case CSSPropertyOutlineWidth: case CSSPropertyOutlineStyle: case CSSPropertyOutlineColor: shorthandPropertyID = CSSPropertyOutline; break; case CSSPropertyOverflowX: case CSSPropertyOverflowY: shorthandPropertyID = CSSPropertyOverflow; break; case CSSPropertyPaddingTop: case CSSPropertyPaddingRight: case CSSPropertyPaddingBottom: case CSSPropertyPaddingLeft: shorthandPropertyID = CSSPropertyPadding; break; case CSSPropertyWebkitAnimationName: case CSSPropertyWebkitAnimationDuration: case CSSPropertyWebkitAnimationTimingFunction: case CSSPropertyWebkitAnimationDelay: case CSSPropertyWebkitAnimationIterationCount: case CSSPropertyWebkitAnimationDirection: case CSSPropertyWebkitAnimationFillMode: shorthandPropertyID = CSSPropertyWebkitAnimation; break; case CSSPropertyWebkitFlexDirection: case CSSPropertyWebkitFlexWrap: shorthandPropertyID = CSSPropertyWebkitFlexFlow; break; case CSSPropertyWebkitMaskPositionX: case CSSPropertyWebkitMaskPositionY: case CSSPropertyWebkitMaskRepeatX: case CSSPropertyWebkitMaskRepeatY: case CSSPropertyWebkitMaskImage: case CSSPropertyWebkitMaskRepeat: case CSSPropertyWebkitMaskAttachment: case CSSPropertyWebkitMaskPosition: case CSSPropertyWebkitMaskClip: case CSSPropertyWebkitMaskOrigin: shorthandPropertyID = CSSPropertyWebkitMask; break; case CSSPropertyWebkitTransformOriginX: case CSSPropertyWebkitTransformOriginY: case CSSPropertyWebkitTransformOriginZ: shorthandPropertyID = CSSPropertyWebkitTransformOrigin; break; case CSSPropertyWebkitTransitionProperty: case CSSPropertyWebkitTransitionDuration: case CSSPropertyWebkitTransitionTimingFunction: case CSSPropertyWebkitTransitionDelay: shorthandPropertyID = CSSPropertyWebkitTransition; break; #if ENABLE(CSS_EXCLUSIONS) case CSSPropertyWebkitWrapFlow: case CSSPropertyWebkitWrapMargin: case CSSPropertyWebkitWrapPadding: shorthandPropertyID = CSSPropertyWebkitWrap; break; #endif default: break; } unsigned shortPropertyIndex = shorthandPropertyID - firstCSSProperty; if (shorthandPropertyID) { if (shorthandPropertyUsed.get(shortPropertyIndex)) continue; if (!shorthandPropertyAppeared.get(shortPropertyIndex) && value.isNull()) value = getPropertyValue(shorthandPropertyID); shorthandPropertyAppeared.set(shortPropertyIndex); } if (!value.isNull()) { propertyID = shorthandPropertyID; shorthandPropertyUsed.set(shortPropertyIndex); } else value = prop.value()->cssText(); if (value == "initial" && !CSSProperty::isInheritedProperty(propertyID)) continue; result.append(getPropertyName(propertyID)); result.append(": "); result.append(value); result.append(prop.isImportant() ? " !important" : ""); result.append("; "); } // FIXME: This is a not-so-nice way to turn x/y positions into single background-position in output. // It is required because background-position-x/y are non-standard properties and WebKit generated output // would not work in Firefox () // It would be a better solution if background-position was CSS_PAIR. if (positionXProp && positionYProp && positionXProp->isImportant() == positionYProp->isImportant()) { result.append("background-position: "); if (positionXProp->value()->isValueList() || positionYProp->value()->isValueList()) result.append(getLayeredShorthandValue(backgroundPositionShorthand())); else { result.append(positionXProp->value()->cssText()); result.append(" "); result.append(positionYProp->value()->cssText()); } if (positionXProp->isImportant()) result.append(" !important"); result.append("; "); } else { if (positionXProp) result.append(positionXProp->cssText()); if (positionYProp) result.append(positionYProp->cssText()); } // FIXME: We need to do the same for background-repeat. if (repeatXProp && repeatYProp && repeatXProp->isImportant() == repeatYProp->isImportant()) { result.append("background-repeat: "); if (repeatXProp->value()->isValueList() || repeatYProp->value()->isValueList()) result.append(getLayeredShorthandValue(backgroundRepeatShorthand())); else { result.append(repeatXProp->value()->cssText()); result.append(" "); result.append(repeatYProp->value()->cssText()); } if (repeatXProp->isImportant()) result.append(" !important"); result.append("; "); } else { if (repeatXProp) result.append(repeatXProp->cssText()); if (repeatYProp) result.append(repeatYProp->cssText()); } return result.toString(); } void StylePropertySet::merge(const StylePropertySet* other, bool argOverridesOnConflict) { unsigned size = other->m_properties.size(); for (unsigned n = 0; n < size; ++n) { const CSSProperty& toMerge = other->m_properties[n]; CSSProperty* old = findPropertyWithId(toMerge.id()); if (old) { if (!argOverridesOnConflict && old->value()) continue; setProperty(toMerge, old); } else m_properties.append(toMerge); } } void StylePropertySet::addSubresourceStyleURLs(ListHashSet& urls, StyleSheetContents* contextStyleSheet) { size_t size = m_properties.size(); for (size_t i = 0; i < size; ++i) m_properties[i].value()->addSubresourceStyleURLs(urls, contextStyleSheet); } // This is the list of properties we want to copy in the copyBlockProperties() function. // It is the list of CSS properties that apply specially to block-level elements. static const CSSPropertyID blockProperties[] = { CSSPropertyOrphans, CSSPropertyOverflow, // This can be also be applied to replaced elements CSSPropertyWebkitAspectRatio, CSSPropertyWebkitColumnCount, CSSPropertyWebkitColumnGap, CSSPropertyWebkitColumnRuleColor, CSSPropertyWebkitColumnRuleStyle, CSSPropertyWebkitColumnRuleWidth, CSSPropertyWebkitColumnBreakBefore, CSSPropertyWebkitColumnBreakAfter, CSSPropertyWebkitColumnBreakInside, CSSPropertyWebkitColumnWidth, CSSPropertyPageBreakAfter, CSSPropertyPageBreakBefore, CSSPropertyPageBreakInside, #if ENABLE(CSS_REGIONS) CSSPropertyWebkitRegionBreakAfter, CSSPropertyWebkitRegionBreakBefore, CSSPropertyWebkitRegionBreakInside, #endif CSSPropertyTextAlign, CSSPropertyTextIndent, CSSPropertyWidows }; const unsigned numBlockProperties = WTF_ARRAY_LENGTH(blockProperties); PassRefPtr StylePropertySet::copyBlockProperties() const { return copyPropertiesInSet(blockProperties, numBlockProperties); } void StylePropertySet::removeBlockProperties() { removePropertiesInSet(blockProperties, numBlockProperties); } bool StylePropertySet::removePropertiesInSet(const CSSPropertyID* set, unsigned length) { if (m_properties.isEmpty()) return false; // FIXME: This is always used with static sets and in that case constructing the hash repeatedly is pretty pointless. HashSet toRemove; for (unsigned i = 0; i < length; ++i) toRemove.add(set[i]); StylePropertyVector newProperties; newProperties.reserveInitialCapacity(m_properties.size()); unsigned size = m_properties.size(); for (unsigned n = 0; n < size; ++n) { const CSSProperty& property = m_properties[n]; // Not quite sure if the isImportant test is needed but it matches the existing behavior. if (!property.isImportant()) { if (toRemove.contains(property.id())) continue; } newProperties.append(property); } bool changed = newProperties.size() != m_properties.size(); m_properties = newProperties; return changed; } const CSSProperty* StylePropertySet::findPropertyWithId(CSSPropertyID propertyID) const { for (int n = m_properties.size() - 1 ; n >= 0; --n) { if (propertyID == m_properties[n].id()) return &m_properties[n]; } return 0; } CSSProperty* StylePropertySet::findPropertyWithId(CSSPropertyID propertyID) { for (int n = m_properties.size() - 1 ; n >= 0; --n) { if (propertyID == m_properties[n].id()) return &m_properties[n]; } return 0; } bool StylePropertySet::propertyMatches(const CSSProperty* property) const { RefPtr value = getPropertyCSSValue(property->id()); return value && value->cssText() == property->value()->cssText(); } void StylePropertySet::removeEquivalentProperties(const StylePropertySet* style) { Vector propertiesToRemove; size_t size = m_properties.size(); for (size_t i = 0; i < size; ++i) { const CSSProperty& property = m_properties[i]; if (style->propertyMatches(&property)) propertiesToRemove.append(property.id()); } // FIXME: This should use mass removal. for (unsigned i = 0; i < propertiesToRemove.size(); ++i) removeProperty(propertiesToRemove[i]); } void StylePropertySet::removeEquivalentProperties(const CSSStyleDeclaration* style) { Vector propertiesToRemove; size_t size = m_properties.size(); for (size_t i = 0; i < size; ++i) { const CSSProperty& property = m_properties[i]; if (style->cssPropertyMatches(&property)) propertiesToRemove.append(property.id()); } // FIXME: This should use mass removal. for (unsigned i = 0; i < propertiesToRemove.size(); ++i) removeProperty(propertiesToRemove[i]); } PassRefPtr StylePropertySet::copy() const { return adoptRef(new StylePropertySet(*this)); } PassRefPtr StylePropertySet::copyPropertiesInSet(const CSSPropertyID* set, unsigned length) const { StylePropertyVector list; list.reserveInitialCapacity(length); for (unsigned i = 0; i < length; ++i) { RefPtr value = getPropertyCSSValue(set[i]); if (value) list.append(CSSProperty(set[i], value.release(), false)); } return StylePropertySet::adopt(list); } CSSStyleDeclaration* StylePropertySet::ensureCSSStyleDeclaration() const { if (m_ownsCSSOMWrapper) { ASSERT(!static_cast(propertySetCSSOMWrapperMap().get(this))->parentRule()); ASSERT(!propertySetCSSOMWrapperMap().get(this)->parentElement()); return propertySetCSSOMWrapperMap().get(this); } m_ownsCSSOMWrapper = true; PropertySetCSSStyleDeclaration* cssomWrapper = new PropertySetCSSStyleDeclaration(const_cast(this)); propertySetCSSOMWrapperMap().add(this, adoptPtr(cssomWrapper)); return cssomWrapper; } CSSStyleDeclaration* StylePropertySet::ensureInlineCSSStyleDeclaration(const StyledElement* parentElement) const { if (m_ownsCSSOMWrapper) { ASSERT(propertySetCSSOMWrapperMap().get(this)->parentElement() == parentElement); return propertySetCSSOMWrapperMap().get(this); } m_ownsCSSOMWrapper = true; PropertySetCSSStyleDeclaration* cssomWrapper = new InlineCSSStyleDeclaration(const_cast(this), const_cast(parentElement)); propertySetCSSOMWrapperMap().add(this, adoptPtr(cssomWrapper)); return cssomWrapper; } void StylePropertySet::clearParentElement(StyledElement* element) { if (!m_ownsCSSOMWrapper) return; ASSERT_UNUSED(element, propertySetCSSOMWrapperMap().get(this)->parentElement() == element); propertySetCSSOMWrapperMap().get(this)->clearParentElement(); } unsigned StylePropertySet::averageSizeInBytes() { // Please update this if the storage scheme changes so that this longer reflects the actual size. return sizeof(StylePropertySet); } // See the function above if you need to update this. class SameSizeAsStylePropertySet : public RefCounted { StylePropertyVector properties; unsigned bitfield; }; COMPILE_ASSERT(sizeof(StylePropertySet) == sizeof(SameSizeAsStylePropertySet), style_property_set_should_stay_small); #ifndef NDEBUG void StylePropertySet::showStyle() { fprintf(stderr, "%s\n", asText().ascii().data()); } #endif } // namespace WebCore