blob: 37b94cebaa553ee770f3f0e4cd6df14e8c750b40 [file] [log] [blame]
/*
* Copyright (C) 1999 Lars Knoll (knoll@kde.org)
* (C) 2004-2005 Allan Sandfeld Jensen (kde@carewolf.com)
* Copyright (C) 2006, 2007 Nicholas Shanks (webkit@nickshanks.com)
* Copyright (C) 2005-2018 Apple Inc. All rights reserved.
* Copyright (C) 2007 Alexey Proskuryakov <ap@webkit.org>
* Copyright (C) 2007, 2008 Eric Seidel <eric@webkit.org>
* 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 Google Inc. 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 "ElementRuleCollector.h"
#include "CSSKeyframeRule.h"
#include "CSSRuleList.h"
#include "CSSSelector.h"
#include "CSSValueKeywords.h"
#include "ContainerQueryEvaluator.h"
#include "ElementInlines.h"
#include "ElementRareData.h"
#include "HTMLElement.h"
#include "HTMLSlotElement.h"
#include "SVGElement.h"
#include "SelectorCheckerTestFunctions.h"
#include "SelectorCompiler.h"
#include "SelectorMatchingState.h"
#include "ShadowRoot.h"
#include "StyleProperties.h"
#include "StyleResolver.h"
#include "StyleScope.h"
#include "StyleScopeRuleSets.h"
#include "StyledElement.h"
#include "UserAgentStyle.h"
#include <wtf/SetForScope.h>
namespace WebCore {
namespace Style {
static const StyleProperties& leftToRightDeclaration()
{
static auto& declaration = [] () -> const StyleProperties& {
auto properties = MutableStyleProperties::create();
properties->setProperty(CSSPropertyDirection, CSSValueLtr);
return properties.leakRef();
}();
return declaration;
}
static const StyleProperties& rightToLeftDeclaration()
{
static auto& declaration = [] () -> const StyleProperties& {
auto properties = MutableStyleProperties::create();
properties->setProperty(CSSPropertyDirection, CSSValueRtl);
return properties.leakRef();
}();
return declaration;
}
class MatchRequest {
public:
MatchRequest(const RuleSet& ruleSet, ScopeOrdinal styleScopeOrdinal = ScopeOrdinal::Element)
: ruleSet(ruleSet)
, styleScopeOrdinal(styleScopeOrdinal)
{
}
const RuleSet& ruleSet;
ScopeOrdinal styleScopeOrdinal;
};
ElementRuleCollector::ElementRuleCollector(const Element& element, const ScopeRuleSets& ruleSets, SelectorMatchingState* selectorMatchingState)
: m_element(element)
, m_authorStyle(ruleSets.authorStyle())
, m_userStyle(ruleSets.userStyle())
, m_userAgentMediaQueryStyle(ruleSets.userAgentMediaQueryStyle())
, m_selectorMatchingState(selectorMatchingState)
{
ASSERT(!m_selectorMatchingState || m_selectorMatchingState->selectorFilter.parentStackIsConsistent(element.parentNode()));
}
ElementRuleCollector::ElementRuleCollector(const Element& element, const RuleSet& authorStyle, SelectorMatchingState* selectorMatchingState)
: m_element(element)
, m_authorStyle(authorStyle)
, m_selectorMatchingState(selectorMatchingState)
{
ASSERT(!m_selectorMatchingState || m_selectorMatchingState->selectorFilter.parentStackIsConsistent(element.parentNode()));
}
const MatchResult& ElementRuleCollector::matchResult() const
{
ASSERT(m_mode == SelectorChecker::Mode::ResolvingStyle);
return m_result;
}
const Vector<RefPtr<const StyleRule>>& ElementRuleCollector::matchedRuleList() const
{
ASSERT(m_mode == SelectorChecker::Mode::CollectingRules);
return m_matchedRuleList;
}
inline void ElementRuleCollector::addMatchedRule(const RuleData& ruleData, unsigned specificity, const MatchRequest& matchRequest)
{
auto cascadeLayerPriority = matchRequest.ruleSet.cascadeLayerPriorityFor(ruleData);
m_matchedRules.append({ &ruleData, specificity, matchRequest.styleScopeOrdinal, cascadeLayerPriority });
}
void ElementRuleCollector::clearMatchedRules()
{
m_matchedRules.clear();
m_matchedRuleTransferIndex = 0;
}
inline void ElementRuleCollector::addElementStyleProperties(const StyleProperties* propertySet, CascadeLayerPriority priority, bool isCacheable, FromStyleAttribute fromStyleAttribute)
{
if (!propertySet || propertySet->isEmpty())
return;
if (!isCacheable)
m_result.isCacheable = false;
auto matchedProperty = MatchedProperties { propertySet };
matchedProperty.cascadeLayerPriority = priority;
matchedProperty.fromStyleAttribute = fromStyleAttribute;
addMatchedProperties(WTFMove(matchedProperty), DeclarationOrigin::Author);
}
void ElementRuleCollector::collectMatchingRules(const MatchRequest& matchRequest)
{
ASSERT_WITH_MESSAGE(!(m_mode == SelectorChecker::Mode::CollectingRulesIgnoringVirtualPseudoElements && m_pseudoElementRequest.pseudoId != PseudoId::None), "When in StyleInvalidation or SharingRules, SelectorChecker does not try to match the pseudo ID. While ElementRuleCollector supports matching a particular pseudoId in this case, this would indicate a error at the call site since matching a particular element should be unnecessary.");
auto* shadowRoot = element().containingShadowRoot();
if (shadowRoot && shadowRoot->mode() == ShadowRootMode::UserAgent)
collectMatchingShadowPseudoElementRules(matchRequest);
// We need to collect the rules for id, class, tag, and everything else into a buffer and
// then sort the buffer.
auto& id = element().idForStyleResolution();
if (!id.isNull())
collectMatchingRulesForList(matchRequest.ruleSet.idRules(id), matchRequest);
if (element().hasClass()) {
for (size_t i = 0; i < element().classNames().size(); ++i)
collectMatchingRulesForList(matchRequest.ruleSet.classRules(element().classNames()[i]), matchRequest);
}
if (element().isLink())
collectMatchingRulesForList(matchRequest.ruleSet.linkPseudoClassRules(), matchRequest);
if (matchesFocusPseudoClass(element()))
collectMatchingRulesForList(matchRequest.ruleSet.focusPseudoClassRules(), matchRequest);
collectMatchingRulesForList(matchRequest.ruleSet.tagRules(element().localName(), element().isHTMLElement() && element().document().isHTMLDocument()), matchRequest);
collectMatchingRulesForList(matchRequest.ruleSet.universalRules(), matchRequest);
}
Vector<MatchedProperties>& ElementRuleCollector::declarationsForOrigin(MatchResult& matchResult, DeclarationOrigin declarationOrigin)
{
switch (declarationOrigin) {
case DeclarationOrigin::UserAgent: return matchResult.userAgentDeclarations;
case DeclarationOrigin::User: return matchResult.userDeclarations;
case DeclarationOrigin::Author: return matchResult.authorDeclarations;
}
ASSERT_NOT_REACHED();
return matchResult.authorDeclarations;
}
void ElementRuleCollector::sortAndTransferMatchedRules(DeclarationOrigin declarationOrigin)
{
if (m_matchedRules.isEmpty())
return;
sortMatchedRules();
transferMatchedRules(declarationOrigin);
}
void ElementRuleCollector::transferMatchedRules(DeclarationOrigin declarationOrigin, std::optional<ScopeOrdinal> fromScope)
{
if (m_mode != SelectorChecker::Mode::CollectingRules)
declarationsForOrigin(m_result, declarationOrigin).reserveCapacity(m_matchedRules.size());
for (; m_matchedRuleTransferIndex < m_matchedRules.size(); ++m_matchedRuleTransferIndex) {
auto& matchedRule = m_matchedRules[m_matchedRuleTransferIndex];
if (fromScope && matchedRule.styleScopeOrdinal < *fromScope)
break;
if (m_mode == SelectorChecker::Mode::CollectingRules) {
m_matchedRuleList.append(&matchedRule.ruleData->styleRule());
continue;
}
addMatchedProperties({
&matchedRule.ruleData->styleRule().properties(),
static_cast<uint8_t>(matchedRule.ruleData->linkMatchType()),
matchedRule.ruleData->propertyAllowlist(),
matchedRule.styleScopeOrdinal,
FromStyleAttribute::No,
matchedRule.cascadeLayerPriority
}, declarationOrigin);
}
}
void ElementRuleCollector::matchAuthorRules()
{
clearMatchedRules();
collectMatchingAuthorRules();
sortAndTransferMatchedRules(DeclarationOrigin::Author);
}
bool ElementRuleCollector::matchesAnyAuthorRules()
{
clearMatchedRules();
// FIXME: This should bail out on first match.
collectMatchingAuthorRules();
return !m_matchedRules.isEmpty();
}
void ElementRuleCollector::collectMatchingAuthorRules()
{
{
MatchRequest matchRequest(m_authorStyle);
collectMatchingRules(matchRequest);
}
auto* parent = element().parentElement();
if (parent && parent->shadowRoot())
matchSlottedPseudoElementRules();
if (element().shadowRoot())
matchHostPseudoClassRules();
if (element().isInShadowTree()) {
matchAuthorShadowPseudoElementRules();
matchPartPseudoElementRules();
}
}
void ElementRuleCollector::matchAuthorShadowPseudoElementRules()
{
ASSERT(element().isInShadowTree());
auto& shadowRoot = *element().containingShadowRoot();
if (shadowRoot.mode() != ShadowRootMode::UserAgent)
return;
// Look up shadow pseudo elements also from the host scope author style as they are web-exposed.
auto& hostAuthorRules = Scope::forNode(*shadowRoot.host()).resolver().ruleSets().authorStyle();
MatchRequest hostAuthorRequest { hostAuthorRules, ScopeOrdinal::ContainingHost };
collectMatchingShadowPseudoElementRules(hostAuthorRequest);
}
void ElementRuleCollector::matchHostPseudoClassRules()
{
ASSERT(element().shadowRoot());
auto& shadowAuthorStyle = element().shadowRoot()->styleScope().resolver().ruleSets().authorStyle();
auto& shadowHostRules = shadowAuthorStyle.hostPseudoClassRules();
if (shadowHostRules.isEmpty())
return;
MatchRequest hostMatchRequest { shadowAuthorStyle, ScopeOrdinal::Shadow };
collectMatchingRulesForList(&shadowHostRules, hostMatchRequest);
}
void ElementRuleCollector::matchSlottedPseudoElementRules()
{
auto* slot = element().assignedSlot();
auto styleScopeOrdinal = ScopeOrdinal::FirstSlot;
for (; slot; slot = slot->assignedSlot(), ++styleScopeOrdinal) {
auto& styleScope = Scope::forNode(*slot);
if (!styleScope.resolver().ruleSets().isAuthorStyleDefined())
continue;
auto& scopeAuthorRules = styleScope.resolver().ruleSets().authorStyle();
MatchRequest scopeMatchRequest(scopeAuthorRules, styleScopeOrdinal);
collectMatchingRulesForList(&scopeAuthorRules.slottedPseudoElementRules(), scopeMatchRequest);
if (styleScopeOrdinal == ScopeOrdinal::SlotLimit)
break;
}
}
void ElementRuleCollector::matchPartPseudoElementRules()
{
ASSERT(element().isInShadowTree());
bool isUAShadowPseudoElement = element().containingShadowRoot()->mode() == ShadowRootMode::UserAgent && !element().shadowPseudoId().isNull();
auto& partMatchingElement = isUAShadowPseudoElement ? *element().shadowHost() : element();
if (partMatchingElement.partNames().isEmpty() || !partMatchingElement.isInShadowTree())
return;
matchPartPseudoElementRulesForScope(partMatchingElement);
}
void ElementRuleCollector::matchPartPseudoElementRulesForScope(const Element& partMatchingElement)
{
auto* element = &partMatchingElement;
auto styleScopeOrdinal = ScopeOrdinal::Element;
for (; element; element = element->shadowHost(), --styleScopeOrdinal) {
auto& styleScope = Scope::forNode(const_cast<Element&>(*element));
if (!styleScope.resolver().ruleSets().isAuthorStyleDefined())
continue;
auto& hostAuthorRules = styleScope.resolver().ruleSets().authorStyle();
MatchRequest scopeMatchRequest(hostAuthorRules, styleScopeOrdinal);
collectMatchingRulesForList(&hostAuthorRules.partPseudoElementRules(), scopeMatchRequest);
// Element may only be exposed to styling from enclosing scopes via exportparts attributes.
if (element != &partMatchingElement && element->shadowRoot()->partMappings().isEmpty())
break;
if (styleScopeOrdinal == ScopeOrdinal::ContainingHostLimit)
break;
}
}
void ElementRuleCollector::collectMatchingShadowPseudoElementRules(const MatchRequest& matchRequest)
{
ASSERT(element().containingShadowRoot()->mode() == ShadowRootMode::UserAgent);
auto& rules = matchRequest.ruleSet;
#if ENABLE(VIDEO)
// FXIME: WebVTT should not be done by styling UA shadow trees like this.
if (element().isWebVTTElement())
collectMatchingRulesForList(&rules.cuePseudoRules(), matchRequest);
#endif
auto& pseudoId = element().shadowPseudoId();
if (!pseudoId.isEmpty())
collectMatchingRulesForList(rules.shadowPseudoElementRules(pseudoId), matchRequest);
}
void ElementRuleCollector::matchUserRules()
{
if (!m_userStyle)
return;
clearMatchedRules();
MatchRequest matchRequest(*m_userStyle);
collectMatchingRules(matchRequest);
sortAndTransferMatchedRules(DeclarationOrigin::User);
}
void ElementRuleCollector::matchUARules()
{
// First we match rules from the user agent sheet.
auto* userAgentStyleSheet = m_isPrintStyle
? UserAgentStyle::defaultPrintStyle : UserAgentStyle::defaultStyle;
matchUARules(*userAgentStyleSheet);
// In quirks mode, we match rules from the quirks user agent sheet.
if (element().document().inQuirksMode())
matchUARules(*UserAgentStyle::defaultQuirksStyle);
if (m_userAgentMediaQueryStyle)
matchUARules(*m_userAgentMediaQueryStyle);
}
void ElementRuleCollector::matchUARules(const RuleSet& rules)
{
clearMatchedRules();
collectMatchingRules(MatchRequest(rules));
sortAndTransferMatchedRules(DeclarationOrigin::UserAgent);
}
inline bool ElementRuleCollector::ruleMatches(const RuleData& ruleData, unsigned& specificity, ScopeOrdinal styleScopeOrdinal)
{
// We know a sufficiently simple single part selector matches simply because we found it from the rule hash when filtering the RuleSet.
// This is limited to HTML only so we don't need to check the namespace (because of tag name match).
auto matchBasedOnRuleHash = ruleData.matchBasedOnRuleHash();
if (matchBasedOnRuleHash != MatchBasedOnRuleHash::None && element().isHTMLElement()) {
ASSERT_WITH_MESSAGE(m_pseudoElementRequest.pseudoId == PseudoId::None, "If we match based on the rule hash while collecting for a particular pseudo element ID, we would add incorrect rules for that pseudo element ID. We should never end in ruleMatches() with a pseudo element if the ruleData cannot match any pseudo element.");
switch (matchBasedOnRuleHash) {
case MatchBasedOnRuleHash::None:
ASSERT_NOT_REACHED();
break;
case MatchBasedOnRuleHash::Universal:
specificity = 0;
break;
case MatchBasedOnRuleHash::ClassA:
specificity = static_cast<unsigned>(SelectorSpecificityIncrement::ClassA);
break;
case MatchBasedOnRuleHash::ClassB:
specificity = static_cast<unsigned>(SelectorSpecificityIncrement::ClassB);
break;
case MatchBasedOnRuleHash::ClassC:
specificity = static_cast<unsigned>(SelectorSpecificityIncrement::ClassC);
break;
}
return true;
}
#if ENABLE(CSS_SELECTOR_JIT)
auto& compiledSelector = ruleData.compiledSelector();
if (compiledSelector.status == SelectorCompilationStatus::NotCompiled)
SelectorCompiler::compileSelector(compiledSelector, ruleData.selector(), SelectorCompiler::SelectorContext::RuleCollector);
if (compiledSelector.status == SelectorCompilationStatus::SimpleSelectorChecker) {
compiledSelector.wasUsed();
#if !ASSERT_MSG_DISABLED
unsigned ignoreSpecificity;
ASSERT_WITH_MESSAGE(!SelectorCompiler::ruleCollectorSimpleSelectorChecker(compiledSelector, &element(), &ignoreSpecificity) || m_pseudoElementRequest.pseudoId == PseudoId::None, "When matching pseudo elements, we should never compile a selector checker without context unless it cannot match anything.");
#endif
bool selectorMatches = SelectorCompiler::ruleCollectorSimpleSelectorChecker(compiledSelector, &element(), &specificity);
if (selectorMatches && ruleData.containsUncommonAttributeSelector())
m_didMatchUncommonAttributeSelector = true;
return selectorMatches;
}
#endif // ENABLE(CSS_SELECTOR_JIT)
SelectorChecker::CheckingContext context(m_mode);
context.pseudoId = m_pseudoElementRequest.pseudoId;
context.scrollbarState = m_pseudoElementRequest.scrollbarState;
context.nameForHightlightPseudoElement = m_pseudoElementRequest.highlightName;
context.styleScopeOrdinal = styleScopeOrdinal;
context.selectorMatchingState = m_selectorMatchingState;
bool selectorMatches;
#if ENABLE(CSS_SELECTOR_JIT)
if (compiledSelector.status == SelectorCompilationStatus::SelectorCheckerWithCheckingContext) {
compiledSelector.wasUsed();
selectorMatches = SelectorCompiler::ruleCollectorSelectorCheckerWithCheckingContext(compiledSelector, &element(), &context, &specificity);
} else
#endif // ENABLE(CSS_SELECTOR_JIT)
{
auto* selector = ruleData.selector();
// Slow path.
SelectorChecker selectorChecker(element().document());
selectorMatches = selectorChecker.match(*selector, element(), context);
if (selectorMatches)
specificity = selector->computeSpecificity();
}
if (ruleData.containsUncommonAttributeSelector()) {
if (selectorMatches || context.pseudoIDSet)
m_didMatchUncommonAttributeSelector = true;
}
m_matchedPseudoElementIds.merge(context.pseudoIDSet);
m_styleRelations.appendVector(context.styleRelations);
return selectorMatches;
}
void ElementRuleCollector::collectMatchingRulesForList(const RuleSet::RuleDataVector* rules, const MatchRequest& matchRequest)
{
if (!rules)
return;
for (unsigned i = 0, size = rules->size(); i < size; ++i) {
const auto& ruleData = rules->data()[i];
if (UNLIKELY(!ruleData.isEnabled()))
continue;
if (!ruleData.canMatchPseudoElement() && m_pseudoElementRequest.pseudoId != PseudoId::None)
continue;
if (m_selectorMatchingState && m_selectorMatchingState->selectorFilter.fastRejectSelector(ruleData.descendantSelectorIdentifierHashes()))
continue;
if (matchRequest.ruleSet.hasContainerQueries() && !containerQueriesMatch(ruleData, matchRequest))
continue;
auto& rule = ruleData.styleRule();
// If the rule has no properties to apply, then ignore it in the non-debug mode.
// Note that if we get null back here, it means we have a rule with deferred properties,
// and that means we always have to consider it.
if (rule.properties().isEmpty() && !m_shouldIncludeEmptyRules)
continue;
unsigned specificity;
if (ruleMatches(ruleData, specificity, matchRequest.styleScopeOrdinal))
addMatchedRule(ruleData, specificity, matchRequest);
}
}
bool ElementRuleCollector::containerQueriesMatch(const RuleData& ruleData, const MatchRequest& matchRequest)
{
auto queries = matchRequest.ruleSet.containerQueriesFor(ruleData);
if (queries.isEmpty())
return true;
// Style bits indicating which pseudo-elements match are set during regular element matching. Container queries need to be evaluate in the right mode.
auto selectionMode = ruleData.canMatchPseudoElement() ? ContainerQueryEvaluator::SelectionMode::PseudoElement : ContainerQueryEvaluator::SelectionMode::Element;
// "Style rules defined on an element inside multiple nested container queries apply when all of the wrapping container queries are true for that element."
ContainerQueryEvaluator evaluator(element(), selectionMode, matchRequest.styleScopeOrdinal, m_selectorMatchingState);
for (auto* query : queries) {
if (!evaluator.evaluate(*query))
return false;
}
return true;
}
static inline bool compareRules(MatchedRule r1, MatchedRule r2)
{
// For normal properties the earlier scope wins. This may be reversed by !important which is handled when resolving cascade.
if (r1.styleScopeOrdinal != r2.styleScopeOrdinal)
return r1.styleScopeOrdinal > r2.styleScopeOrdinal;
if (r1.cascadeLayerPriority != r2.cascadeLayerPriority)
return r1.cascadeLayerPriority < r2.cascadeLayerPriority;
if (r1.specificity != r2.specificity)
return r1.specificity < r2.specificity;
return r1.ruleData->position() < r2.ruleData->position();
}
void ElementRuleCollector::sortMatchedRules()
{
std::sort(m_matchedRules.begin(), m_matchedRules.end(), compareRules);
}
void ElementRuleCollector::matchAllRules(bool matchAuthorAndUserStyles, bool includeSMILProperties)
{
matchUARules();
// Now we check user sheet rules.
if (matchAuthorAndUserStyles)
matchUserRules();
if (is<StyledElement>(element())) {
auto& styledElement = downcast<StyledElement>(element());
// https://html.spec.whatwg.org/#presentational-hints
addElementStyleProperties(styledElement.presentationalHintStyle(), RuleSet::cascadeLayerPriorityForPresentationalHints);
// Tables and table cells share an additional presentation style that must be applied
// after all attributes, since their style depends on the values of multiple attributes.
addElementStyleProperties(styledElement.additionalPresentationalHintStyle(), RuleSet::cascadeLayerPriorityForPresentationalHints);
if (is<HTMLElement>(styledElement)) {
bool isAuto;
auto textDirection = downcast<HTMLElement>(styledElement).directionalityIfhasDirAutoAttribute(isAuto);
auto& properties = textDirection == TextDirection::LTR ? leftToRightDeclaration() : rightToLeftDeclaration();
if (isAuto)
addMatchedProperties({ &properties }, DeclarationOrigin::Author);
}
}
if (matchAuthorAndUserStyles) {
clearMatchedRules();
collectMatchingAuthorRules();
sortMatchedRules();
transferMatchedRules(DeclarationOrigin::Author, ScopeOrdinal::Element);
// Inline style behaves as if it has higher specificity than any rule.
addElementInlineStyleProperties(includeSMILProperties);
// Rules from the host scopes override inline style.
transferMatchedRules(DeclarationOrigin::Author);
}
}
void ElementRuleCollector::addElementInlineStyleProperties(bool includeSMILProperties)
{
if (!is<StyledElement>(element()))
return;
if (auto* inlineStyle = downcast<StyledElement>(element()).inlineStyle()) {
// FIXME: Media control shadow trees seem to have problems with caching.
bool isInlineStyleCacheable = !inlineStyle->isMutable() && !element().isInShadowTree();
addElementStyleProperties(inlineStyle, RuleSet::cascadeLayerPriorityForUnlayered, isInlineStyleCacheable, FromStyleAttribute::Yes);
}
if (includeSMILProperties && is<SVGElement>(element()))
addElementStyleProperties(downcast<SVGElement>(element()).animatedSMILStyleProperties(), RuleSet::cascadeLayerPriorityForUnlayered, false /* isCacheable */);
}
bool ElementRuleCollector::hasAnyMatchingRules(const RuleSet& ruleSet)
{
clearMatchedRules();
m_mode = SelectorChecker::Mode::CollectingRulesIgnoringVirtualPseudoElements;
collectMatchingRules(MatchRequest(ruleSet));
return !m_matchedRules.isEmpty();
}
void ElementRuleCollector::addMatchedProperties(MatchedProperties&& matchedProperties, DeclarationOrigin declarationOrigin)
{
// FIXME: This should be moved to the matched properties cache code.
auto computeIsCacheable = [&] {
if (!m_result.isCacheable)
return false;
if (matchedProperties.styleScopeOrdinal != ScopeOrdinal::Element)
return false;
auto& properties = *matchedProperties.properties;
for (unsigned i = 0, count = properties.propertyCount(); i < count; ++i) {
// Currently the property cache only copy the non-inherited values and resolve
// the inherited ones.
// Here we define some exception were we have to resolve some properties that are not inherited
// by default. If those exceptions become too common on the web, it should be possible
// to build a list of exception to resolve instead of completely disabling the cache.
StyleProperties::PropertyReference current = properties.propertyAt(i);
if (current.isInherited())
continue;
// If the property value is explicitly inherited, we need to apply further non-inherited properties
// as they might override the value inherited here. For this reason we don't allow declarations with
// explicitly inherited properties to be cached.
const CSSValue& value = *current.value();
if (value.isInheritValue())
return false;
// The value currentColor has implicitely the same side effect. It depends on the value of color,
// which is an inherited value, making the non-inherited property implicitly inherited.
if (is<CSSPrimitiveValue>(value) && downcast<CSSPrimitiveValue>(value).valueID() == CSSValueCurrentcolor)
return false;
if (value.hasVariableReferences())
return false;
}
return true;
};
m_result.isCacheable = computeIsCacheable();
declarationsForOrigin(m_result, declarationOrigin).append(WTFMove(matchedProperties));
}
void ElementRuleCollector::addAuthorKeyframeRules(const StyleRuleKeyframe& keyframe)
{
ASSERT(m_result.authorDeclarations.isEmpty());
m_result.authorDeclarations.append({ &keyframe.properties(), SelectorChecker::MatchAll, propertyAllowlistForPseudoId(m_pseudoElementRequest.pseudoId) });
}
}
} // namespace WebCore