Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(668)

Unified Diff: Source/core/css/resolver/CascadedValues.cpp

Issue 350333003: Cascade declared property values instead of applying values on top of each other (Closed) Base URL: https://chromium.googlesource.com/chromium/blink.git@master
Patch Set: move functions around :| Created 6 years, 5 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
Index: Source/core/css/resolver/CascadedValues.cpp
diff --git a/Source/core/css/resolver/CascadedValues.cpp b/Source/core/css/resolver/CascadedValues.cpp
new file mode 100644
index 0000000000000000000000000000000000000000..78e36c0eddb51b5ab74487852a919447c11433b6
--- /dev/null
+++ b/Source/core/css/resolver/CascadedValues.cpp
@@ -0,0 +1,387 @@
+// Copyright 2014 The Chromium Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#include "config.h"
+#include "core/css/resolver/CascadedValues.h"
+
+#include "core/css/CSSValuePool.h"
+#include "core/css/StylePropertySet.h"
+#include "core/css/resolver/MatchResult.h"
+#include "core/css/resolver/StyleBuilder.h"
+#include "core/css/resolver/StyleResolverState.h"
+
+namespace blink {
+
+namespace {
+
+bool isValidCueStyleProperty(CSSPropertyID property)
esprehn 2014/07/21 18:50:34 static, get rid of the anonymous namespace.
Timothy Loh 2014/07/22 14:32:48 Done.
+{
+ switch (property) {
+ case CSSPropertyBackground:
+ case CSSPropertyBackgroundAttachment:
+ case CSSPropertyBackgroundClip:
+ case CSSPropertyBackgroundColor:
+ case CSSPropertyBackgroundImage:
+ case CSSPropertyBackgroundOrigin:
+ case CSSPropertyBackgroundPosition:
+ case CSSPropertyBackgroundPositionX:
+ case CSSPropertyBackgroundPositionY:
+ case CSSPropertyBackgroundRepeat:
+ case CSSPropertyBackgroundRepeatX:
+ case CSSPropertyBackgroundRepeatY:
+ case CSSPropertyBackgroundSize:
+ case CSSPropertyColor:
+ case CSSPropertyFont:
+ case CSSPropertyFontFamily:
+ case CSSPropertyFontSize:
+ case CSSPropertyFontStyle:
+ case CSSPropertyFontVariant:
+ case CSSPropertyFontWeight:
+ case CSSPropertyLineHeight:
+ case CSSPropertyOpacity:
+ case CSSPropertyOutline:
+ case CSSPropertyOutlineColor:
+ case CSSPropertyOutlineOffset:
+ case CSSPropertyOutlineStyle:
+ case CSSPropertyOutlineWidth:
+ case CSSPropertyVisibility:
+ case CSSPropertyWhiteSpace:
+ // FIXME: 'text-decoration' shorthand to be handled when available.
+ // See https://chromiumcodereview.appspot.com/19516002 for details.
+ case CSSPropertyTextDecoration:
+ case CSSPropertyTextShadow:
+ case CSSPropertyBorderStyle:
+ return true;
+ case CSSPropertyTextDecorationLine:
+ case CSSPropertyTextDecorationStyle:
+ case CSSPropertyTextDecorationColor:
+ return RuntimeEnabledFeatures::css3TextDecorationsEnabled();
+ default:
+ break;
+ }
+ return false;
+}
+
+bool isValidFirstLetterStyleProperty(CSSPropertyID property)
esprehn 2014/07/21 18:50:34 static
Timothy Loh 2014/07/22 14:32:48 Done.
+{
+ switch (property) {
+ // Valid ::first-letter properties listed in spec:
+ // http://www.w3.org/TR/css3-selectors/#application-in-css
+ case CSSPropertyBackgroundAttachment:
+ case CSSPropertyBackgroundBlendMode:
+ case CSSPropertyBackgroundClip:
+ case CSSPropertyBackgroundColor:
+ case CSSPropertyBackgroundImage:
+ case CSSPropertyBackgroundOrigin:
+ case CSSPropertyBackgroundPosition:
+ case CSSPropertyBackgroundPositionX:
+ case CSSPropertyBackgroundPositionY:
+ case CSSPropertyBackgroundRepeat:
+ case CSSPropertyBackgroundRepeatX:
+ case CSSPropertyBackgroundRepeatY:
+ case CSSPropertyBackgroundSize:
+ case CSSPropertyBorderBottomColor:
+ case CSSPropertyBorderBottomLeftRadius:
+ case CSSPropertyBorderBottomRightRadius:
+ case CSSPropertyBorderBottomStyle:
+ case CSSPropertyBorderBottomWidth:
+ case CSSPropertyBorderImageOutset:
+ case CSSPropertyBorderImageRepeat:
+ case CSSPropertyBorderImageSlice:
+ case CSSPropertyBorderImageSource:
+ case CSSPropertyBorderImageWidth:
+ case CSSPropertyBorderLeftColor:
+ case CSSPropertyBorderLeftStyle:
+ case CSSPropertyBorderLeftWidth:
+ case CSSPropertyBorderRightColor:
+ case CSSPropertyBorderRightStyle:
+ case CSSPropertyBorderRightWidth:
+ case CSSPropertyBorderTopColor:
+ case CSSPropertyBorderTopLeftRadius:
+ case CSSPropertyBorderTopRightRadius:
+ case CSSPropertyBorderTopStyle:
+ case CSSPropertyBorderTopWidth:
+ case CSSPropertyColor:
+ case CSSPropertyFloat:
+ case CSSPropertyFont:
+ case CSSPropertyFontFamily:
+ case CSSPropertyFontKerning:
+ case CSSPropertyFontSize:
+ case CSSPropertyFontStretch:
+ case CSSPropertyFontStyle:
+ case CSSPropertyFontVariant:
+ case CSSPropertyFontVariantLigatures:
+ case CSSPropertyFontWeight:
+ case CSSPropertyLetterSpacing:
+ case CSSPropertyLineHeight:
+ case CSSPropertyMarginBottom:
+ case CSSPropertyMarginLeft:
+ case CSSPropertyMarginRight:
+ case CSSPropertyMarginTop:
+ case CSSPropertyPaddingBottom:
+ case CSSPropertyPaddingLeft:
+ case CSSPropertyPaddingRight:
+ case CSSPropertyPaddingTop:
+ case CSSPropertyTextTransform:
+ case CSSPropertyVerticalAlign:
+ case CSSPropertyWebkitBackgroundClip:
+ case CSSPropertyWebkitBackgroundComposite:
+ case CSSPropertyWebkitBackgroundOrigin:
+ case CSSPropertyWebkitBackgroundSize:
+ case CSSPropertyWebkitBorderAfter:
+ case CSSPropertyWebkitBorderAfterColor:
+ case CSSPropertyWebkitBorderAfterStyle:
+ case CSSPropertyWebkitBorderAfterWidth:
+ case CSSPropertyWebkitBorderBefore:
+ case CSSPropertyWebkitBorderBeforeColor:
+ case CSSPropertyWebkitBorderBeforeStyle:
+ case CSSPropertyWebkitBorderBeforeWidth:
+ case CSSPropertyWebkitBorderEnd:
+ case CSSPropertyWebkitBorderEndColor:
+ case CSSPropertyWebkitBorderEndStyle:
+ case CSSPropertyWebkitBorderEndWidth:
+ case CSSPropertyWebkitBorderFit:
+ case CSSPropertyWebkitBorderHorizontalSpacing:
+ case CSSPropertyWebkitBorderImage:
+ case CSSPropertyWebkitBorderRadius:
+ case CSSPropertyWebkitBorderStart:
+ case CSSPropertyWebkitBorderStartColor:
+ case CSSPropertyWebkitBorderStartStyle:
+ case CSSPropertyWebkitBorderStartWidth:
+ case CSSPropertyWebkitBorderVerticalSpacing:
+ case CSSPropertyWebkitFontSmoothing:
+ case CSSPropertyWebkitMarginAfter:
+ case CSSPropertyWebkitMarginAfterCollapse:
+ case CSSPropertyWebkitMarginBefore:
+ case CSSPropertyWebkitMarginBeforeCollapse:
+ case CSSPropertyWebkitMarginBottomCollapse:
+ case CSSPropertyWebkitMarginCollapse:
+ case CSSPropertyWebkitMarginEnd:
+ case CSSPropertyWebkitMarginStart:
+ case CSSPropertyWebkitMarginTopCollapse:
+ case CSSPropertyWordSpacing:
+ return true;
+ case CSSPropertyTextDecorationColor:
+ case CSSPropertyTextDecorationLine:
+ case CSSPropertyTextDecorationStyle:
+ return RuntimeEnabledFeatures::css3TextDecorationsEnabled();
+
+ // text-shadow added in text decoration spec:
+ // http://www.w3.org/TR/css-text-decor-3/#text-shadow-property
+ case CSSPropertyTextShadow:
+ // box-shadox added in CSS3 backgrounds spec:
+ // http://www.w3.org/TR/css3-background/#placement
+ case CSSPropertyBoxShadow:
+ case CSSPropertyWebkitBoxShadow:
+ // Properties that we currently support outside of spec.
+ case CSSPropertyWebkitLineBoxContain:
+ case CSSPropertyVisibility:
+ return true;
+
+ default:
+ return false;
+ }
+}
+
+// We want properties in their prefixed and non-prefixed variants to cascade
+// together, so we map them to the same property before adding to the cascade.
+CSSPropertyID resolveProperty(CSSPropertyID property, TextDirection direction, WritingMode writingMode)
esprehn 2014/07/21 18:50:34 static
Timothy Loh 2014/07/22 14:32:48 Done.
+{
+ switch (property) {
+ case CSSPropertyWebkitBorderEndColor:
+ case CSSPropertyWebkitBorderBeforeStyle:
+ case CSSPropertyWebkitBorderEndStyle:
+ case CSSPropertyWebkitPaddingStart:
+ case CSSPropertyWebkitBorderStartWidth:
+ case CSSPropertyWebkitMaxLogicalWidth:
+ case CSSPropertyWebkitLogicalHeight:
+ case CSSPropertyWebkitMinLogicalWidth:
+ case CSSPropertyWebkitBorderBeforeWidth:
+ case CSSPropertyWebkitPaddingBefore:
+ case CSSPropertyWebkitBorderBeforeColor:
+ case CSSPropertyWebkitMarginEnd:
+ case CSSPropertyWebkitBorderAfterWidth:
+ case CSSPropertyWebkitMinLogicalHeight:
+ case CSSPropertyWebkitBorderEndWidth:
+ case CSSPropertyWebkitPaddingEnd:
+ case CSSPropertyWebkitLogicalWidth:
+ case CSSPropertyWebkitBorderAfterColor:
+ case CSSPropertyWebkitMaxLogicalHeight:
+ case CSSPropertyWebkitBorderStartColor:
+ case CSSPropertyWebkitBorderAfterStyle:
+ case CSSPropertyWebkitPaddingAfter:
+ case CSSPropertyWebkitBorderStartStyle:
+ case CSSPropertyWebkitMarginBefore:
+ case CSSPropertyWebkitMarginStart:
+ case CSSPropertyWebkitMarginAfter:
+ return CSSProperty::resolveDirectionAwareProperty(property, direction, writingMode);
+ case CSSPropertyWebkitBackfaceVisibility:
+ return CSSPropertyBackfaceVisibility;
+ case CSSPropertyWebkitBackgroundClip:
+ return CSSPropertyBackgroundClip;
+ case CSSPropertyWebkitBackgroundOrigin:
+ return CSSPropertyBackgroundOrigin;
+ case CSSPropertyWebkitBackgroundSize:
+ return CSSPropertyBackgroundSize;
+ case CSSPropertyWebkitBoxShadow:
+ return CSSPropertyBoxShadow;
+ case CSSPropertyWebkitTransform:
+ return CSSPropertyTransform;
+ case CSSPropertyWebkitTransformStyle:
+ return CSSPropertyTransformStyle;
+ case CSSPropertyWebkitPerspective:
+ return CSSPropertyPerspective;
+ case CSSPropertyWebkitPerspectiveOrigin:
+ return CSSPropertyPerspectiveOrigin;
+ // Used by editing when the css3 text decorations flag is off
+ case CSSPropertyTextDecoration:
+ ASSERT(!RuntimeEnabledFeatures::css3TextDecorationsEnabled());
+ return CSSPropertyTextDecorationLine;
+ // As per css3-text, word-wrap is an alias for overflow-wrap
+ case CSSPropertyWordWrap:
+ return CSSPropertyOverflowWrap;
+ default:
+ return property;
+ }
+}
+
+} // namespace
+
+CascadedValues::CascadedValues(StyleResolverState& state, const MatchResult& matchResult)
+ : state(state)
+{
+ memset(values, 0, sizeof(values));
+ memset(visitedLinkValues, 0, sizeof(visitedLinkValues));
+
+ addMatchResultInternal(matchResult, false, 0, matchResult.matchedProperties.size() - 1, CSSPropertyDirection, CSSPropertyWebkitWritingMode);
esprehn 2014/07/21 18:50:34 Can we get rid of this boolean argument?
Timothy Loh 2014/07/22 14:32:48 The MatchResult stores declarations in the order (
+ addMatchResultInternal(matchResult, true, matchResult.ranges.firstAuthorRule, matchResult.ranges.lastAuthorRule, CSSPropertyDirection, CSSPropertyWebkitWritingMode);
+ addMatchResultInternal(matchResult, true, matchResult.ranges.firstUARule, matchResult.ranges.lastUARule, CSSPropertyDirection, CSSPropertyWebkitWritingMode);
+ applyValues(CSSPropertyDirection, CSSPropertyWebkitWritingMode);
+ direction = state.style()->direction();
+ writingMode = state.style()->writingMode();
+}
+
+void CascadedValues::addMatchResult(const MatchResult& matchResult, bool isImportant, int firstIndex, int lastIndex)
+{
+ COMPILE_ASSERT(CSSPropertyColor == CSSPropertyWebkitWritingMode + 1, CSS_color_is_after_super_high_priority);
+ addMatchResultInternal(matchResult, isImportant, firstIndex, lastIndex, CSSPropertyColor, convertToCSSPropertyID(lastCSSProperty));
+}
+
+void CascadedValues::addMatchResultInternal(const MatchResult& matchResult, bool isImportant, int firstIndex, int lastIndex, CSSPropertyID firstId, CSSPropertyID lastId)
+{
+ if (firstIndex == -1)
+ return;
+
+ for (int i = firstIndex; i <= lastIndex; ++i) {
+ const MatchedProperties& matchedProperties = matchResult.matchedProperties[i];
+ SelectorChecker::LinkMatchMask linkMatchType = SelectorChecker::MatchLink;
+ if (state.style()->insideLink())
+ linkMatchType = static_cast<SelectorChecker::LinkMatchMask>(matchedProperties.m_types.linkMatchType);
+ addStylePropertySet(matchedProperties.properties.get(), matchResult.matchedRules[i], isImportant, firstId, lastId, static_cast<PropertyWhitelistType>(matchedProperties.m_types.whitelistType), linkMatchType);
+ }
+}
+
+void CascadedValues::addStylePropertySet(const StylePropertySet* properties, const StyleRule* rule, bool isImportant, CSSPropertyID first, CSSPropertyID last, PropertyWhitelistType propertyWhitelistType, SelectorChecker::LinkMatchMask linkMatchType)
+{
+ unsigned propertyCount = properties->propertyCount();
+ for (unsigned i = 0; i < propertyCount; ++i) {
+ StylePropertySet::PropertyReference current = properties->propertyAt(i);
+ if (isImportant != current.isImportant())
+ continue;
+
+ CSSPropertyID property = current.id();
+ if (property < first || property > last)
+ continue;
+
+ // FIXME: We should check the whitelist, otherwise we might run into
+ // problems with e.g. foo:first-letter { all: inherit; }
+ if (property == CSSPropertyAll) {
+ addAllProperty(current.value(), linkMatchType);
+ continue;
+ }
+
+ if (propertyWhitelistType == PropertyWhitelistCue && !isValidCueStyleProperty(property))
+ continue;
+ if (propertyWhitelistType == PropertyWhitelistFirstLetter && !isValidFirstLetterStyleProperty(property))
+ continue;
+ addValue(current.id(), current.value(), rule, linkMatchType);
+ }
+}
+
+void CascadedValues::addValue(CSSPropertyID property, CSSValue* value, const StyleRule* rule, SelectorChecker::LinkMatchMask linkMatchType)
+{
+ property = resolveProperty(property, direction, writingMode);
+
+ if (property == CSSPropertyInternalCallback) {
+ ASSERT(value->isPrimitiveValue() && toCSSPrimitiveValue(value)->getValueID() == CSSValueInternalPresence);
+ state.style()->addCallbackSelector(rule->selectorList().selectorsText());
+ return;
+ }
+
+ // Shorthands that are not expanded in the parser need special handling here.
+ // The shorthand property occurs earlier in CSSPropertyNames.in than its
+ // longhands so we can cascade properly. If a longhand value is explicitly
+ // set, with higher priority than the shorthand, then the value computed
+ // from the shorthand will be overriden.
+
+ // FIXME: font should always be expanded by the parser; crbug.com/353932
+ if (property == CSSPropertyFont) {
+ values[CSSPropertyLineHeight - firstCSSProperty] = nullptr;
esprehn 2014/07/21 18:50:34 Why are you always subtracting backwards? I think
Timothy Loh 2014/07/22 14:32:48 Done.
+ values[CSSPropertyFontStyle - firstCSSProperty] = nullptr;
+ values[CSSPropertyFontFamily - firstCSSProperty] = nullptr;
+ values[CSSPropertyFontVariant - firstCSSProperty] = nullptr;
+ values[CSSPropertyFontWeight - firstCSSProperty] = nullptr;
+ values[CSSPropertyFontStyle - firstCSSProperty] = nullptr;
+ values[CSSPropertyFontSize - firstCSSProperty] = nullptr;
+ }
+
+ // -webkit-border-image sometimes also sets border-*-width, but properly
+ // cascading this requires too much effort
+ if (property == CSSPropertyWebkitBorderImage) {
+ values[CSSPropertyBorderImageSource - firstCSSProperty] = nullptr;
+ values[CSSPropertyBorderImageSlice - firstCSSProperty] = nullptr;
+ values[CSSPropertyBorderImageWidth - firstCSSProperty] = nullptr;
+ values[CSSPropertyBorderImageOutset - firstCSSProperty] = nullptr;
+ values[CSSPropertyBorderImageRepeat - firstCSSProperty] = nullptr;
+ }
+
+ if (linkMatchType & SelectorChecker::MatchLink)
+ values[property - firstCSSProperty] = value;
+ if (linkMatchType & SelectorChecker::MatchVisited)
+ visitedLinkValues[property - firstCSSProperty] = value;
+}
+
+void CascadedValues::addAllProperty(CSSValue* value, SelectorChecker::LinkMatchMask linkMatchType)
+{
+ if (!value->isInitialValue() && !value->isInheritedValue()) {
+ ASSERT(value->isPrimitiveValue() && toCSSPrimitiveValue(value)->getValueID() == CSSValueUnset);
+ value = nullptr;
+ }
+
+ for (int i = firstCSSProperty; i <= lastCSSProperty; ++i) {
+ CSSPropertyID property = convertToCSSPropertyID(i);
+ if (CSSProperty::isAffectedByAllProperty(property))
+ addValue(property, value, nullptr, linkMatchType);
+ }
+}
+
+void CascadedValues::applyValues(CSSPropertyID first, CSSPropertyID last, bool inheritedOnly)
+{
+ for (int i = first - firstCSSProperty; i <= last - firstCSSProperty; i++) {
+ CSSPropertyID property = convertToCSSPropertyID(i + firstCSSProperty);
+ if (inheritedOnly && !CSSProperty::isInheritedProperty(property))
+ continue;
+ if (values[i])
+ StyleBuilder::applyProperty(property, state, values[i]);
+ if (visitedLinkValues[i]) {
+ state.setApplyPropertyToRegularStyle(false);
+ state.setApplyPropertyToVisitedLinkStyle(true);
+ StyleBuilder::applyProperty(property, state, visitedLinkValues[i]);
+ state.setApplyPropertyToRegularStyle(true);
+ state.setApplyPropertyToVisitedLinkStyle(false);
+ }
+ }
+}
+
+} // namespace blink

Powered by Google App Engine
This is Rietveld 408576698