| /* |
| * Copyright (C) 1999 Lars Knoll ([email protected]) |
| * Copyright (C) 2004-2005 Allan Sandfeld Jensen ([email protected]) |
| * Copyright (C) 2006, 2007 Nicholas Shanks ([email protected]) |
| * Copyright (C) 2005-2019 Apple Inc. All rights reserved. |
| * Copyright (C) 2007 Alexey Proskuryakov <[email protected]> |
| * Copyright (C) 2007, 2008 Eric Seidel <[email protected]> |
| * Copyright (C) 2008, 2009 Torch Mobile Inc. All rights reserved. (http://www.torchmobile.com/) |
| * Copyright (c) 2011, Code Aurora Forum. All rights reserved. |
| * Copyright (C) Research In Motion Limited 2011. All rights reserved. |
| * Copyright (C) 2012, 2013 Google Inc. All rights reserved. |
| * Copyright (C) 2014 Igalia S.L. |
| * |
| * 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 "StyleResolver.h" |
| |
| #include "BlendingKeyframes.h" |
| #include "CSSCustomPropertyValue.h" |
| #include "CSSFontSelector.h" |
| #include "CSSKeyframeRule.h" |
| #include "CSSKeyframesRule.h" |
| #include "CSSPrimitiveValueMappings.h" |
| #include "CSSPropertyNames.h" |
| #include "CSSSelector.h" |
| #include "CSSStyleRule.h" |
| #include "CSSStyleSheet.h" |
| #include "CSSViewTransitionRule.h" |
| #include "CompositeOperation.h" |
| #include "CustomFunctionRegistry.h" |
| #include "DocumentInlines.h" |
| #include "DocumentResourceLoader.h" |
| #include "DocumentView.h" |
| #include "ElementRuleCollector.h" |
| #include "FrameSelection.h" |
| #include "InspectorInstrumentation.h" |
| #include "LocalFrame.h" |
| #include "LocalFrameView.h" |
| #include "Logging.h" |
| #include "MatchResultCache.h" |
| #include "MediaList.h" |
| #include "MutableCSSSelector.h" |
| #include "NodeRenderStyle.h" |
| #include "PageRuleCollector.h" |
| #include "RenderScrollbar.h" |
| #include "RenderStyleConstants.h" |
| #include "RenderStyle+GettersInlines.h" |
| #include "RenderStyle+SettersInlines.h" |
| #include "RenderView.h" |
| #include "ResolvedStyle.h" |
| #include "RuleSet.h" |
| #include "RuleSetBuilder.h" |
| #include "SVGDocumentExtensions.h" |
| #include "SVGElement.h" |
| #include "SVGElementTypeHelpers.h" |
| #include "SVGFontFaceElement.h" |
| #include "SVGSVGElement.h" |
| #include "Settings.h" |
| #include "ShadowRoot.h" |
| #include "SharedStringHash.h" |
| #include "StyleAdjuster.h" |
| #include "StyleBuilder.h" |
| #include "StyleEasingFunction.h" |
| #include "StyleFontSizeFunctions.h" |
| #include "StyleProperties.h" |
| #include "StylePropertyShorthand.h" |
| #include "StyleResolveForDocument.h" |
| #include "StyleRule.h" |
| #include "StyleSheetContents.h" |
| #include "StyleSingleAnimationRangeName.h" |
| #include "TimingFunction.h" |
| #include "UserAgentParts.h" |
| #include "UserAgentStyle.h" |
| #include "VisibilityAdjustment.h" |
| #include "VisitedLinkState.h" |
| #include "WebAnimationTypes.h" |
| #include "WebKitFontFamilyNames.h" |
| #include <wtf/HashFunctions.h> |
| #include <wtf/HashTraits.h> |
| #include <wtf/Seconds.h> |
| #include <wtf/StdLibExtras.h> |
| #include <wtf/TZoneMallocInlines.h> |
| #include <wtf/Vector.h> |
| #include <wtf/text/AtomStringHash.h> |
| |
| namespace WTF { |
| |
| struct StyleRuleKeyframeKeyHash { |
| static unsigned hash(const WebCore::StyleRuleKeyframe::Key& p) { return pairIntHash(p.rangeName, p.offset); } |
| static bool equal(const WebCore::StyleRuleKeyframe::Key& a, const WebCore::StyleRuleKeyframe::Key& b) { return a == b; } |
| static const bool safeToCompareToEmptyOrDeleted = true; |
| }; |
| template<> struct HashTraits<WebCore::StyleRuleKeyframe::Key> : GenericHashTraits<WebCore::StyleRuleKeyframe::Key> { |
| static WebCore::StyleRuleKeyframe::Key emptyValue() { return { WebCore::CSSValueDefault, 0 }; } |
| static bool isEmptyValue(const WebCore::StyleRuleKeyframe::Key& value) { return value.rangeName == WebCore::CSSValueDefault; } |
| |
| static void constructDeletedValue(WebCore::StyleRuleKeyframe::Key& slot) { slot.rangeName = WebCore::CSSValueNone; } |
| static bool isDeletedValue(const WebCore::StyleRuleKeyframe::Key& slot) { return slot.rangeName == WebCore::CSSValueNone; } |
| }; |
| template<> struct DefaultHash<WebCore::StyleRuleKeyframe::Key> : StyleRuleKeyframeKeyHash { }; |
| |
| } |
| |
| namespace WebCore { |
| namespace Style { |
| |
| using namespace HTMLNames; |
| |
| WTF_MAKE_TZONE_ALLOCATED_IMPL(Resolver); |
| |
| class Resolver::State { |
| public: |
| State() = default; |
| State(const Element& element, const RenderStyle* parentStyle, const RenderStyle* documentElementStyle, TreeResolutionState* treeResolutionState) |
| : m_element(&element) |
| , m_parentStyle(parentStyle) |
| , m_treeResolutionState(treeResolutionState) |
| { |
| ASSERT(element.isConnected()); |
| |
| auto& document = element.document(); |
| auto* documentElement = document.documentElement(); |
| if (!documentElement || documentElement == &element) |
| m_rootElementStyle = document.initialContainingBlockStyle(); |
| else if (documentElementStyle) |
| m_rootElementStyle = documentElementStyle; |
| else if (auto* documentElementRenderStyle = documentElement->renderStyle()) |
| m_rootElementStyle = documentElementRenderStyle; |
| else |
| m_rootElementStyle = document.initialContainingBlockStyle(); |
| } |
| |
| const Element* element() const { return m_element; } |
| |
| void setStyle(std::unique_ptr<RenderStyle> style) { m_style = WTF::move(style); } |
| RenderStyle* style() const { return m_style.get(); } |
| std::unique_ptr<RenderStyle> takeStyle() { return WTF::move(m_style); } |
| |
| void setParentStyle(std::unique_ptr<RenderStyle> parentStyle) |
| { |
| m_ownedParentStyle = WTF::move(parentStyle); |
| m_parentStyle = m_ownedParentStyle.get(); |
| } |
| const RenderStyle* parentStyle() const { return m_parentStyle; } |
| const RenderStyle* rootElementStyle() const { return m_rootElementStyle; } |
| |
| CheckedPtr<TreeResolutionState> treeResolutionState() { return m_treeResolutionState; } |
| |
| private: |
| const Element* m_element { }; |
| std::unique_ptr<RenderStyle> m_style; |
| const RenderStyle* m_parentStyle { }; |
| std::unique_ptr<const RenderStyle> m_ownedParentStyle; |
| const RenderStyle* m_rootElementStyle { }; |
| |
| CheckedPtr<TreeResolutionState> m_treeResolutionState; |
| }; |
| |
| Ref<Resolver> Resolver::create(Document& document, ScopeType scopeType) |
| { |
| return adoptRef(*new Resolver(document, scopeType)); |
| } |
| |
| Resolver::Resolver(Document& document, ScopeType scopeType) |
| : m_document(document) |
| , m_scopeType(scopeType) |
| , m_ruleSets(*this) |
| , m_matchedDeclarationsCache(*this) |
| , m_matchAuthorAndUserStyles(settings().authorAndUserStylesEnabled()) |
| { |
| initialize(); |
| } |
| |
| void Resolver::initialize() |
| { |
| UserAgentStyle::initDefaultStyleSheet(); |
| |
| // construct document root element default style. this is needed |
| // to evaluate media queries that contain relative constraints, like "screen and (max-width: 10em)" |
| // This is here instead of constructor, because when constructor is run, |
| // document doesn't have documentElement |
| // NOTE: this assumes that element that gets passed to styleForElement -call |
| // is always from the document that owns the style selector |
| CheckedPtr view = document().view(); |
| if (view) |
| m_mediaQueryEvaluator = MQ::MediaQueryEvaluator { view->mediaType() }; |
| else |
| m_mediaQueryEvaluator = MQ::MediaQueryEvaluator { }; |
| |
| if (RefPtr documentElement = document().documentElement()) { |
| m_rootDefaultStyle = styleForElement(*documentElement, { document().initialContainingBlockStyle() }, RuleMatchingBehavior::MatchOnlyUserAgentRules).style; |
| // Turn off assertion against font lookups during style resolver initialization. We may need root style font for media queries. |
| document().fontSelector().incrementIsComputingRootStyleFont(); |
| m_rootDefaultStyle->fontCascade().update(&document().fontSelector()); |
| m_rootDefaultStyle->fontCascade().primaryFont(); |
| document().protectedFontSelector()->decrementIsComputingRootStyleFont(); |
| } |
| |
| if (m_rootDefaultStyle && view) |
| m_mediaQueryEvaluator = MQ::MediaQueryEvaluator { view->mediaType(), document(), m_rootDefaultStyle.get() }; |
| |
| m_ruleSets.resetAuthorStyle(); |
| m_ruleSets.resetUserAgentMediaQueryStyle(); |
| } |
| |
| Resolver::~Resolver() = default; |
| |
| Document& Resolver::document() |
| { |
| return *m_document; |
| } |
| |
| const Document& Resolver::document() const |
| { |
| return *m_document; |
| } |
| |
| const Settings& Resolver::settings() const |
| { |
| return document().settings(); |
| } |
| |
| void Resolver::addCurrentSVGFontFaceRules() |
| { |
| if (document().svgExtensionsIfExists()) { |
| auto& svgFontFaceElements = document().svgExtensionsIfExists()->svgFontFaceElements(); |
| for (Ref svgFontFaceElement : svgFontFaceElements) |
| document().fontSelector().addFontFaceRule(svgFontFaceElement->fontFaceRule(), svgFontFaceElement->isInUserAgentShadowTree()); |
| } |
| } |
| |
| void Resolver::appendAuthorStyleSheets(std::span<const Ref<CSSStyleSheet>> styleSheets) |
| { |
| m_ruleSets.appendAuthorStyleSheets(styleSheets, &m_mediaQueryEvaluator, m_inspectorCSSOMWrappers); |
| |
| if (auto renderView = document().renderView()) |
| renderView->style().fontCascade().update(&document().fontSelector()); |
| } |
| |
| KeyframesRuleMap& Resolver::userAgentKeyframes() |
| { |
| static NeverDestroyed<KeyframesRuleMap> keyframes; |
| return keyframes; |
| } |
| |
| void Resolver::addUserAgentKeyframeStyle(Ref<StyleRuleKeyframes>&& rule) |
| { |
| const auto& animationName = rule->name(); |
| userAgentKeyframes().set(animationName, WTF::move(rule)); |
| } |
| |
| // This is a simplified style setting function for keyframe styles |
| void Resolver::addKeyframeStyle(Ref<StyleRuleKeyframes>&& rule) |
| { |
| const auto& animationName = rule->name(); |
| m_keyframesRuleMap.set(animationName, WTF::move(rule)); |
| document().keyframesRuleDidChange(animationName); |
| } |
| |
| auto Resolver::initializeStateAndStyle(const Element& element, const ResolutionContext& context, std::unique_ptr<RenderStyle>&& initialStyle) -> State |
| { |
| auto state = State { element, context.parentStyle, context.documentElementStyle, context.treeResolutionState.get() }; |
| |
| if (initialStyle) |
| state.setStyle(WTF::move(initialStyle)); |
| else if (state.parentStyle()) { |
| state.setStyle(RenderStyle::createPtrWithRegisteredInitialValues(document().customPropertyRegistry())); |
| if (&element == document().documentElement() && !context.isSVGUseTreeRoot) { |
| // Initial values for custom properties are inserted to the document element style. Don't overwrite them. |
| state.style()->inheritIgnoringCustomPropertiesFrom(*state.parentStyle()); |
| } else |
| state.style()->inheritFrom(*state.parentStyle()); |
| } else { |
| state.setStyle(defaultStyleForElement(&element)); |
| state.setParentStyle(RenderStyle::clonePtr(*state.style())); |
| } |
| |
| // BuilderState::useSVGZoomRulesForLength equivalent |
| if (is<SVGElement>(element) && !(is<SVGSVGElement>(element) && element.parentNode())) |
| state.style()->setUseSVGZoomRulesForLength(true); |
| |
| if (element.isLink()) { |
| auto& style = *state.style(); |
| style.setIsLink(true); |
| InsideLink linkState = document().visitedLinkState().determineLinkState(element); |
| if (linkState != InsideLink::NotInside) { |
| bool forceVisited = InspectorInstrumentation::forcePseudoState(element, CSSSelector::PseudoClass::Visited); |
| if (forceVisited) |
| linkState = InsideLink::InsideVisited; |
| } |
| style.setInsideLink(linkState); |
| } |
| |
| return state; |
| } |
| |
| BuilderContext Resolver::builderContext(State& state) const |
| { |
| return { |
| document(), |
| state.parentStyle(), |
| state.rootElementStyle(), |
| state.element(), |
| state.treeResolutionState() |
| }; |
| } |
| |
| UnadjustedStyle Resolver::unadjustedStyleForElement(Element& element, const ResolutionContext& context, RuleMatchingBehavior matchingBehavior) |
| { |
| auto state = initializeStateAndStyle(element, context); |
| auto& style = *state.style(); |
| |
| UserAgentStyle::ensureDefaultStyleSheetsForElement(element); |
| |
| ElementRuleCollector collector(element, m_ruleSets, context.selectorMatchingState); |
| collector.setMedium(m_mediaQueryEvaluator); |
| |
| if (matchingBehavior == RuleMatchingBehavior::MatchOnlyUserAgentRules) |
| collector.matchUARules(); |
| else |
| collector.matchAllRules(m_matchAuthorAndUserStyles, matchingBehavior != RuleMatchingBehavior::MatchAllRulesExcludingSMIL); |
| |
| if (collector.matchedPseudoElements()) |
| style.setHasPseudoStyles(collector.matchedPseudoElements()); |
| |
| auto elementStyleRelations = commitRelationsToRenderStyle(style, element, collector.styleRelations()); |
| |
| applyMatchedProperties(state, collector.matchResult(), PropertyCascade::normalProperties()); |
| |
| return { |
| .style = state.takeStyle(), |
| .relations = WTF::move(elementStyleRelations), |
| .matchResult = collector.releaseMatchResult() |
| }; |
| } |
| |
| ResolvedStyle Resolver::styleForElement(Element& element, const ResolutionContext& context, RuleMatchingBehavior matchingBehavior) |
| { |
| auto unadjustedStyle = unadjustedStyleForElement(element, context, matchingBehavior); |
| auto& parentStyle = context.parentStyle ? *context.parentStyle : RenderStyle::defaultStyleSingleton(); |
| |
| auto style = WTF::move(unadjustedStyle.style); |
| |
| Adjuster adjuster(document(), parentStyle, context.parentBoxStyle, &element); |
| adjuster.adjust(*style); |
| |
| return { |
| .style = WTF::move(style), |
| .relations = WTF::move(unadjustedStyle.relations), |
| .matchResult = WTF::move(unadjustedStyle.matchResult) |
| }; |
| } |
| |
| UnadjustedStyle Resolver::unadjustedStyleForCachedMatchResult(Element& element, const ResolutionContext& context, CachedMatchResult&& cachedResult) |
| { |
| auto& unadjustedStyle = cachedResult.unadjustedStyle; |
| |
| if (cachedResult.changedProperties.isEmpty()) { |
| // The cached result can be used as-is. |
| return WTF::move(unadjustedStyle); |
| } |
| |
| bool applyPartially = !cachedResult.changedProperties.ids.isEmpty(); |
| |
| auto state = initializeStateAndStyle(element, context, applyPartially ? WTF::move(unadjustedStyle.style) : nullptr); |
| if (!applyPartially) { |
| state.style()->copyPseudoElementBitsFrom(*unadjustedStyle.style); |
| copyRelations(*state.style(), *unadjustedStyle.style); |
| } |
| |
| applyMatchedProperties(state, *unadjustedStyle.matchResult, WTF::move(cachedResult.changedProperties)); |
| |
| return { |
| .style = state.takeStyle(), |
| .relations = WTF::move(unadjustedStyle.relations), |
| .matchResult = unadjustedStyle.matchResult |
| }; |
| } |
| |
| std::unique_ptr<RenderStyle> Resolver::styleForKeyframe(Element& element, const RenderStyle& elementStyle, const ResolutionContext& context, const StyleRuleKeyframe& keyframe, BlendingKeyframe& blendingKeyframe) const |
| { |
| // Add all the animating properties to the keyframe. |
| bool hasRevert = false; |
| for (auto propertyReference : keyframe.properties()) { |
| auto unresolvedProperty = propertyReference.id(); |
| // The animation-composition and animation-timing-function within keyframes are special |
| // because they are not animated; they just describe the composite operation and timing |
| // function between this keyframe and the next. |
| if (CSSProperty::isDirectionAwareProperty(unresolvedProperty)) |
| blendingKeyframe.setContainsDirectionAwareProperty(true); |
| if (RefPtr value = propertyReference.value()) { |
| auto resolvedProperty = CSSProperty::resolveDirectionAwareProperty(unresolvedProperty, elementStyle.writingMode()); |
| if (resolvedProperty != CSSPropertyAnimationTimingFunction && resolvedProperty != CSSPropertyAnimationComposition) { |
| if (RefPtr customValue = dynamicDowncast<CSSCustomPropertyValue>(*value)) |
| blendingKeyframe.addProperty(customValue->name()); |
| else |
| blendingKeyframe.addProperty(resolvedProperty); |
| } |
| if (isValueID(*value, CSSValueRevert)) |
| hasRevert = true; |
| } |
| } |
| |
| auto state = State(element, nullptr, context.documentElementStyle, context.treeResolutionState.get()); |
| |
| state.setStyle(RenderStyle::clonePtr(elementStyle)); |
| state.setParentStyle(RenderStyle::clonePtr(context.parentStyle ? *context.parentStyle : elementStyle)); |
| |
| ElementRuleCollector collector(element, m_ruleSets, context.selectorMatchingState); |
| |
| auto pseudoElementIdentifier = elementStyle.pseudoElementIdentifier(); |
| if (pseudoElementIdentifier) |
| collector.setPseudoElementRequest(*pseudoElementIdentifier); |
| |
| if (hasRevert) { |
| // In the animation origin, 'revert' rolls back the cascaded value to the user level. |
| // Therefore, we need to collect UA and user rules. |
| collector.setMedium(m_mediaQueryEvaluator); |
| collector.matchUARules(); |
| collector.matchUserRules(); |
| } |
| collector.addAuthorKeyframeRules(keyframe); |
| Builder builder(*state.style(), builderContext(state), collector.matchResult()); |
| builder.state().setIsBuildingKeyframeStyle(); |
| builder.applyAllProperties(); |
| |
| Adjuster adjuster(document(), *state.parentStyle(), nullptr, !pseudoElementIdentifier ? &element : nullptr); |
| adjuster.adjust(*state.style()); |
| |
| return state.takeStyle(); |
| } |
| |
| bool Resolver::isAnimationNameValid(const String& name) const |
| { |
| return m_keyframesRuleMap.find(AtomString(name)) != m_keyframesRuleMap.end() |
| || userAgentKeyframes().find(AtomString(name)) != userAgentKeyframes().end(); |
| } |
| |
| Vector<Ref<StyleRuleKeyframe>> Resolver::keyframeRulesForName(const AtomString& animationName, const TimingFunction* defaultTimingFunction) const |
| { |
| if (animationName.isEmpty()) |
| return { }; |
| |
| m_keyframesRuleMap.checkConsistency(); |
| |
| // Check author map first then check user-agent map. |
| auto it = m_keyframesRuleMap.find(animationName); |
| if (it == m_keyframesRuleMap.end()) { |
| it = userAgentKeyframes().find(animationName); |
| if (it == userAgentKeyframes().end()) |
| return { }; |
| } |
| |
| auto compositeOperationForKeyframe = [](Ref<StyleRuleKeyframe> keyframe) -> CompositeOperation { |
| if (auto compositeOperationCSSValue = keyframe->properties().getPropertyCSSValue(CSSPropertyAnimationComposition)) { |
| if (auto compositeOperation = toCompositeOperation(*compositeOperationCSSValue)) |
| return *compositeOperation; |
| } |
| return Animation::initialCompositeOperation(); |
| }; |
| |
| auto timingFunctionForKeyframe = [&](Ref<StyleRuleKeyframe> keyframe) -> RefPtr<const TimingFunction> { |
| if (auto timingFunctionCSSValue = keyframe->properties().getPropertyCSSValue(CSSPropertyAnimationTimingFunction)) { |
| if (auto timingFunction = createTimingFunctionDeprecated(*timingFunctionCSSValue)) |
| return timingFunction; |
| } |
| if (defaultTimingFunction) |
| return defaultTimingFunction; |
| return &CubicBezierTimingFunction::defaultTimingFunction(); |
| }; |
| |
| HashSet<RefPtr<const TimingFunction>> timingFunctions; |
| auto uniqueTimingFunctionForKeyframe = [&](Ref<StyleRuleKeyframe> keyframe) -> RefPtr<const TimingFunction> { |
| auto timingFunction = timingFunctionForKeyframe(keyframe); |
| for (auto existingTimingFunction : timingFunctions) { |
| if (arePointingToEqualData(timingFunction, existingTimingFunction)) |
| return existingTimingFunction; |
| } |
| timingFunctions.add(timingFunction); |
| return timingFunction; |
| }; |
| |
| Ref keyframesRule = it->value; |
| auto* keyframes = &keyframesRule->keyframes(); |
| |
| using KeyframeUniqueKey = std::tuple<StyleRuleKeyframe::Key, RefPtr<const TimingFunction>, CompositeOperation>; |
| auto hasDuplicateKeys = [&]() -> bool { |
| HashSet<KeyframeUniqueKey> uniqueKeyframeKeys; |
| for (auto& keyframe : *keyframes) { |
| auto compositeOperation = compositeOperationForKeyframe(keyframe); |
| auto timingFunction = uniqueTimingFunctionForKeyframe(keyframe); |
| for (auto key : keyframe->keys()) { |
| if (!uniqueKeyframeKeys.add({ key, timingFunction, compositeOperation })) |
| return true; |
| } |
| } |
| return false; |
| }(); |
| |
| if (!hasDuplicateKeys) |
| return *keyframes; |
| |
| // Merge keyframes with a similar offset and timing function ensuring that merged keyframes |
| // move to the end of the list if the offset is a timeline range. |
| Vector<Ref<StyleRuleKeyframe>> deduplicatedKeyframes; |
| HashMap<KeyframeUniqueKey, Ref<StyleRuleKeyframe>> keyframesMap; |
| for (auto& originalKeyframe : *keyframes) { |
| auto compositeOperation = compositeOperationForKeyframe(originalKeyframe); |
| auto timingFunction = uniqueTimingFunctionForKeyframe(originalKeyframe); |
| for (auto key : originalKeyframe->keys()) { |
| KeyframeUniqueKey uniqueKey { key, timingFunction, compositeOperation }; |
| if (RefPtr existingStyleRuleKeyframe = keyframesMap.get(uniqueKey)) { |
| existingStyleRuleKeyframe->mutableProperties().mergeAndOverrideOnConflict(originalKeyframe->properties()); |
| if (existingStyleRuleKeyframe->keys()[0].rangeName == CSSValueNormal) |
| continue; |
| deduplicatedKeyframes.removeFirstMatching([&](const auto& styleRuleKeyframe) { |
| return styleRuleKeyframe.ptr() == existingStyleRuleKeyframe; |
| }); |
| deduplicatedKeyframes.append(*existingStyleRuleKeyframe); |
| } else { |
| auto styleRuleKeyframe = StyleRuleKeyframe::create(MutableStyleProperties::create()); |
| styleRuleKeyframe->setKey(key); |
| styleRuleKeyframe->mutableProperties().mergeAndOverrideOnConflict(originalKeyframe->properties()); |
| keyframesMap.set(uniqueKey, styleRuleKeyframe); |
| deduplicatedKeyframes.append(styleRuleKeyframe); |
| } |
| } |
| } |
| |
| return deduplicatedKeyframes; |
| } |
| |
| bool Resolver::keyframeStylesForAnimation(Element& element, const RenderStyle& elementStyle, const ResolutionContext& context, BlendingKeyframes& list, const TimingFunction* defaultTimingFunction) const |
| { |
| list.clear(); |
| |
| auto keyframeRules = keyframeRulesForName(list.keyframesName(), defaultTimingFunction); |
| if (keyframeRules.isEmpty()) |
| return false; |
| |
| // Construct and populate the style for each keyframe. |
| for (auto& keyframeRule : keyframeRules) { |
| // Add this keyframe style to all the indicated key times |
| for (auto& key : keyframeRule->keys()) { |
| BlendingKeyframe blendingKeyframe({ Style::convertCSSValueIDToSingleAnimationRangeName(key.rangeName), key.offset }, { nullptr }); |
| blendingKeyframe.setStyle(styleForKeyframe(element, elementStyle, context, keyframeRule.get(), blendingKeyframe)); |
| if (auto timingFunctionCSSValue = keyframeRule->properties().getPropertyCSSValue(CSSPropertyAnimationTimingFunction)) |
| blendingKeyframe.setTimingFunction(createTimingFunctionDeprecated(*timingFunctionCSSValue)); |
| if (auto compositeOperationCSSValue = keyframeRule->properties().getPropertyCSSValue(CSSPropertyAnimationComposition)) { |
| if (auto compositeOperation = toCompositeOperation(*compositeOperationCSSValue)) |
| blendingKeyframe.setCompositeOperation(*compositeOperation); |
| } |
| list.insert(WTF::move(blendingKeyframe)); |
| list.updatePropertiesMetadata(keyframeRule->properties()); |
| } |
| } |
| |
| return true; |
| } |
| |
| std::optional<ResolvedStyle> Resolver::styleForPseudoElement(Element& element, const PseudoElementRequest& pseudoElementRequest, const ResolutionContext& context) |
| { |
| auto state = State(element, context.parentStyle, context.documentElementStyle, context.treeResolutionState.get()); |
| |
| if (state.parentStyle()) { |
| state.setStyle(RenderStyle::createPtrWithRegisteredInitialValues(document().customPropertyRegistry())); |
| state.style()->inheritFrom(*state.parentStyle()); |
| } else { |
| state.setStyle(defaultStyleForElement(&element)); |
| state.setParentStyle(RenderStyle::clonePtr(*state.style())); |
| } |
| |
| ElementRuleCollector collector(element, m_ruleSets, context.selectorMatchingState); |
| collector.setPseudoElementRequest(pseudoElementRequest); |
| collector.setMedium(m_mediaQueryEvaluator); |
| collector.matchUARules(); |
| |
| if (m_matchAuthorAndUserStyles) { |
| collector.matchUserRules(); |
| collector.matchAuthorRules(); |
| } |
| |
| ASSERT(!collector.matchedPseudoElements()); |
| |
| if (collector.matchResult().isEmpty()) |
| return { }; |
| |
| state.style()->setPseudoElementIdentifier(pseudoElementRequest.identifier()); |
| |
| applyMatchedProperties(state, collector.matchResult(), PropertyCascade::normalProperties()); |
| |
| Adjuster adjuster(document(), *state.parentStyle(), context.parentBoxStyle, nullptr); |
| adjuster.adjust(*state.style()); |
| |
| Adjuster::adjustVisibilityForPseudoElement(*state.style(), element); |
| |
| return ResolvedStyle { state.takeStyle(), nullptr, collector.releaseMatchResult() }; |
| } |
| |
| std::unique_ptr<RenderStyle> Resolver::styleForPage(int pageIndex) |
| { |
| RefPtr documentElement = document().documentElement(); |
| if (!documentElement || !documentElement->renderStyle()) |
| return RenderStyle::createPtr(); |
| |
| auto state = State(*documentElement, document().initialContainingBlockStyle(), nullptr, nullptr); |
| |
| state.setStyle(RenderStyle::createPtr()); |
| state.style()->inheritFrom(*state.rootElementStyle()); |
| |
| PageRuleCollector collector(m_ruleSets, documentElement->renderStyle()->writingMode()); |
| collector.matchAllPageRules(pageIndex); |
| |
| auto& result = collector.matchResult(); |
| |
| Builder builder(*state.style(), builderContext(state), result); |
| builder.applyAllProperties(); |
| |
| // Now return the style. |
| return state.takeStyle(); |
| } |
| |
| std::unique_ptr<RenderStyle> Resolver::defaultStyleForElement(const Element* element) |
| { |
| auto style = RenderStyle::createPtrWithRegisteredInitialValues(document().customPropertyRegistry()); |
| |
| FontCascadeDescription fontDescription; |
| fontDescription.setOneFamily(standardFamily); |
| fontDescription.setKeywordSizeFromIdentifier(CSSValueMedium); |
| |
| auto size = fontSizeForKeyword(CSSValueMedium, false, document()); |
| fontDescription.setSpecifiedSize(size); |
| auto computedFontSize = computedFontSizeFromSpecifiedSize(size, fontDescription.isAbsoluteSize(), is<SVGElement>(element), style->computedStyle(), document()); |
| fontDescription.setComputedSize(computedFontSize.size, computedFontSize.usedZoomFactor); |
| |
| fontDescription.setShouldAllowUserInstalledFonts(settings().shouldAllowUserInstalledFonts() ? AllowUserInstalledFonts::Yes : AllowUserInstalledFonts::No); |
| style->setFontDescription(WTF::move(fontDescription)); |
| |
| return style; |
| } |
| |
| Vector<Ref<const StyleRule>> Resolver::styleRulesForElement(const Element* element, unsigned rulesToInclude) |
| { |
| return pseudoStyleRulesForElement(element, { }, rulesToInclude); |
| } |
| |
| Vector<Ref<const StyleRule>> Resolver::pseudoStyleRulesForElement(const Element* element, const std::optional<Style::PseudoElementRequest>& pseudoElementIdentifier, unsigned rulesToInclude) |
| { |
| if (!element) |
| return { }; |
| |
| auto state = State(*element, nullptr, nullptr, nullptr); |
| |
| ElementRuleCollector collector(*element, m_ruleSets, nullptr, SelectorChecker::Mode::CollectingRules); |
| if (pseudoElementIdentifier) |
| collector.setPseudoElementRequest(*pseudoElementIdentifier); |
| collector.setMedium(m_mediaQueryEvaluator); |
| collector.setIncludeEmptyRules(rulesToInclude & EmptyCSSRules); |
| |
| if (rulesToInclude & UAAndUserCSSRules) { |
| // First we match rules from the user agent sheet. |
| collector.matchUARules(); |
| |
| // Now we check user sheet rules. |
| if (m_matchAuthorAndUserStyles) |
| collector.matchUserRules(); |
| } |
| |
| if (m_matchAuthorAndUserStyles && (rulesToInclude & AuthorCSSRules)) |
| collector.matchAuthorRules(); |
| |
| return collector.matchedRuleList(); |
| } |
| |
| void Resolver::invalidateMatchedDeclarationsCache() |
| { |
| m_matchedDeclarationsCache.invalidate(); |
| } |
| |
| void Resolver::clearCachedDeclarationsAffectedByViewportUnits() |
| { |
| m_matchedDeclarationsCache.clearEntriesAffectedByViewportUnits(); |
| } |
| |
| void Resolver::applyMatchedProperties(State& state, const MatchResult& matchResult, PropertyCascade::IncludedProperties&& includedProperties) |
| { |
| auto& style = *state.style(); |
| auto& parentStyle = *state.parentStyle(); |
| Ref element = *state.element(); |
| |
| unsigned cacheHash = MatchedDeclarationsCache::computeHash(matchResult, parentStyle.inheritedCustomProperties()); |
| |
| auto cacheResult = m_matchedDeclarationsCache.find(cacheHash, matchResult, parentStyle.inheritedCustomProperties(), parentStyle); |
| |
| auto hasUsableEntry = cacheResult && MatchedDeclarationsCache::isCacheable(element.get(), style, parentStyle); |
| if (hasUsableEntry) { |
| auto& cacheEntry = cacheResult->entry; |
| bool inheritedEqual = cacheResult->inheritedEqual; |
| // We can build up the style by copying non-inherited properties from an earlier style object built using the same exact |
| // style declarations. We then only need to apply the inherited properties, if any, as their values can depend on the |
| // element context. This is fast and saves memory by reusing the style data structures. |
| style.copyNonInheritedFrom(*cacheEntry.renderStyle); |
| |
| bool hasExplicitlyInherited = cacheEntry.renderStyle->hasExplicitlyInheritedProperties(); |
| bool explicitlyInheritedEqual = !hasExplicitlyInherited || parentStyle.nonInheritedEqual(*cacheEntry.parentRenderStyle); |
| |
| if (inheritedEqual) { |
| InsideLink linkStatus = style.insideLink(); |
| // If the cache item parent style has identical inherited properties to the current parent style then the |
| // resulting style will be identical too. We copy the inherited properties over from the cache and are done. |
| style.inheritFrom(*cacheEntry.renderStyle); |
| |
| // Link status is treated like an inherited property. We need to explicitly restore it. |
| style.setInsideLink(linkStatus); |
| } |
| |
| includedProperties = { }; |
| |
| if (!inheritedEqual) |
| includedProperties.types.add(PropertyCascade::PropertyType::Inherited); |
| if (!explicitlyInheritedEqual) |
| includedProperties.types.add(PropertyCascade::PropertyType::ExplicitlyInherited); |
| if (!matchResult.nonCacheablePropertyIds.isEmpty()) |
| includedProperties.types.add(PropertyCascade::PropertyType::NonCacheable); |
| |
| if (includedProperties.isEmpty()) |
| return; |
| } |
| |
| Builder builder(style, builderContext(state), matchResult, WTF::move(includedProperties)); |
| |
| // Top priority properties may affect resolution of high priority ones. |
| builder.applyTopPriorityProperties(); |
| |
| // High priority properties may affect resolution of other properties (they are mostly font related). |
| builder.applyHighPriorityProperties(); |
| |
| if (cacheResult && !cacheResult->entry.isUsableAfterHighPriorityProperties(style)) { |
| // High-priority properties may affect resolution of other properties. Kick out the existing cache entry and try again. |
| m_matchedDeclarationsCache.remove(cacheHash); |
| applyMatchedProperties(state, matchResult, PropertyCascade::normalProperties()); |
| return; |
| } |
| |
| builder.applyNonHighPriorityProperties(); |
| builder.adjustAfterApplying(); |
| |
| setGlobalStateAfterApplyingProperties(builder.state()); |
| |
| if (((cacheHash && !cacheResult) || (cacheResult && !cacheResult->inheritedEqual)) && MatchedDeclarationsCache::isCacheable(element, style, parentStyle)) |
| m_matchedDeclarationsCache.add(style, parentStyle, cacheHash, matchResult); |
| } |
| |
| void Resolver::setGlobalStateAfterApplyingProperties(const BuilderState& builderState) |
| { |
| // FIXME: This stuff should be somewhere else. |
| for (auto& contentAttribute : builderState.registeredContentAttributes()) |
| ruleSets().mutableFeatures().registerContentAttribute(contentAttribute); |
| if (builderState.style().usesViewportUnits()) |
| document().setHasStyleWithViewportUnits(); |
| } |
| |
| bool Resolver::hasSelectorForAttribute(const Element& element, const AtomString& attributeName) const |
| { |
| ASSERT(!attributeName.isEmpty()); |
| if (element.isHTMLElement() && element.document().isHTMLDocument()) |
| return m_ruleSets.features().attributeLowercaseLocalNamesInRules.contains(attributeName); |
| return m_ruleSets.features().attributeLocalNamesInRules.contains(attributeName); |
| } |
| |
| bool Resolver::hasSelectorForId(const AtomString& idValue) const |
| { |
| ASSERT(!idValue.isEmpty()); |
| return m_ruleSets.features().idsInRules.contains(idValue); |
| } |
| |
| bool Resolver::hasViewportDependentMediaQueries() const |
| { |
| return m_ruleSets.hasViewportDependentMediaQueries(); |
| } |
| |
| std::optional<DynamicMediaQueryEvaluationChanges> Resolver::evaluateDynamicMediaQueries() |
| { |
| return m_ruleSets.evaluateDynamicMediaQueryRules(m_mediaQueryEvaluator); |
| } |
| |
| |
| static CSSSelectorList viewTransitionSelector(CSSSelector::PseudoElement element, const AtomString& name) |
| { |
| MutableCSSSelectorList selectorList; |
| |
| auto rootSelector = makeUnique<MutableCSSSelector>(); |
| rootSelector->setMatch(CSSSelector::Match::PseudoClass); |
| rootSelector->setPseudoClass(CSSSelector::PseudoClass::Root); |
| selectorList.append(WTF::move(rootSelector)); |
| |
| auto groupSelector = makeUnique<MutableCSSSelector>(); |
| groupSelector->setMatch(CSSSelector::Match::PseudoElement); |
| groupSelector->setPseudoElement(element); |
| |
| AtomString selectorName; |
| switch (element) { |
| case CSSSelector::PseudoElement::ViewTransitionGroup: |
| selectorName = "view-transition-group"_s; |
| break; |
| case CSSSelector::PseudoElement::ViewTransitionImagePair: |
| selectorName = "view-transition-image-pair"_s; |
| break; |
| case CSSSelector::PseudoElement::ViewTransitionNew: |
| selectorName = "view-transition-new"_s; |
| break; |
| case CSSSelector::PseudoElement::ViewTransitionOld: |
| selectorName = "view-transition-old"_s; |
| break; |
| default: |
| ASSERT_NOT_REACHED(); |
| break; |
| } |
| |
| groupSelector->setValue(selectorName); |
| groupSelector->setStringList({ { name } }); |
| |
| selectorList.first()->prependInComplexSelector(CSSSelector::Relation::Subselector, WTF::move(groupSelector)); |
| |
| return CSSSelectorList(WTF::move(selectorList)); |
| } |
| |
| RefPtr<StyleRuleViewTransition> Resolver::viewTransitionRule() const |
| { |
| return m_ruleSets.viewTransitionRule(); |
| } |
| |
| void Resolver::setViewTransitionStyles(CSSSelector::PseudoElement element, const AtomString& name, Ref<MutableStyleProperties> properties) |
| { |
| if (!m_document) |
| return; |
| |
| auto styleRule = StyleRule::create(WTF::move(properties), true, viewTransitionSelector(element, name)); |
| |
| RefPtr viewTransitionsStyle = m_ruleSets.dynamicViewTransitionsStyle(); |
| RuleSetBuilder builder(*viewTransitionsStyle, mediaQueryEvaluator(), this); |
| builder.addStyleRule(styleRule); |
| } |
| |
| const CustomFunctionRegistry* Resolver::customFunctionRegistry() const |
| { |
| return m_customFunctionRegistry.get(); |
| } |
| |
| CustomFunctionRegistry& Resolver::ensureCustomFunctionRegistry() |
| { |
| if (!m_customFunctionRegistry) |
| m_customFunctionRegistry = makeUnique<CustomFunctionRegistry>(); |
| return *m_customFunctionRegistry; |
| } |
| |
| } // namespace Style |
| } // namespace WebCore |