blob: a21c156f0b7db90a9148ea04ba4ed33b552e1e50 [file] [log] [blame]
/*
* Copyright (C) 1999 Lars Knoll ([email protected])
* Copyright (C) 2003-2011, 2014 Apple 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.
*
*/
#pragma once
#include "CSSSelectorList.h"
#include "CommonAtomStrings.h"
#include <wtf/Forward.h>
#include <wtf/HashMap.h>
#include <wtf/HashSet.h>
#include <wtf/text/AtomString.h>
#include <wtf/text/AtomStringHash.h>
namespace WebCore {
class StyleRule;
class StyleRuleScope;
namespace Style {
class RuleData;
// FIXME: Has* values should be separated so we could describe both the :has() argument and its position in the selector.
enum class MatchElement : uint8_t {
Subject,
Parent,
Ancestor,
DirectSibling,
IndirectSibling,
AnySibling,
ParentSibling,
AncestorSibling,
ParentAnySibling,
AncestorAnySibling,
HasChild,
HasDescendant,
HasSibling,
HasSiblingDescendant,
HasAnySibling,
HasChildParent,
HasChildAncestor,
HasNonSubject, // FIXME: This is a catch-all for the rest of cases where :has() is in a non-subject position.
HasScopeBreaking, // FIXME: This is a catch-all for cases where :has() contains a scope breaking sub-selector like, like :has(:is(.x .y)).
Host,
HostChild
};
constexpr unsigned matchElementCount = static_cast<unsigned>(MatchElement::HostChild) + 1;
enum class IsNegation : bool { No, Yes };
enum class CanBreakScope : bool { No, Yes }; // Are we inside a logical combination pseudo-class like :is() or :not(), which if we were inside a :has(), could break out of its scope?
enum class DoesBreakScope : bool { No, Yes }; // Did we find a logical combination pseudo-class like :is() or :not() with selector combinators that do break out of a :has() scope?
// For MSVC.
#pragma pack(push, 4)
struct RuleAndSelector {
RuleAndSelector(const RuleData&);
RefPtr<const StyleRule> styleRule;
uint16_t selectorIndex; // Keep in sync with RuleData's selectorIndex size.
uint16_t selectorListIndex; // Keep in sync with RuleData's selectorListIndex size.
};
struct RuleFeature : public RuleAndSelector {
RuleFeature(const RuleData&, MatchElement, IsNegation);
MatchElement matchElement;
IsNegation isNegation; // Whether the selector is in a (non-paired) :not() context.
};
static_assert(sizeof(RuleFeature) <= 16, "RuleFeature is a frequently allocated object. Keep it small.");
struct RuleFeatureWithInvalidationSelector : public RuleFeature {
RuleFeatureWithInvalidationSelector(const RuleData&, MatchElement, IsNegation, CSSSelectorList&& invalidationSelector);
CSSSelectorList invalidationSelector;
};
#pragma pack(pop)
using PseudoClassInvalidationKey = std::tuple<unsigned, uint8_t, AtomString>;
using RuleFeatureVector = Vector<RuleFeature>;
struct RuleFeatureSet {
void add(const RuleFeatureSet&);
void clear();
void shrinkToFit();
void collectFeatures(const RuleData&, const Vector<Ref<const StyleRuleScope>>& scopeRules = { });
void registerContentAttribute(const AtomString&);
bool usesHasPseudoClass() const;
bool usesMatchElement(MatchElement matchElement) const { return usedMatchElements[enumToUnderlyingType(matchElement)]; }
void setUsesMatchElement(MatchElement matchElement) { usedMatchElements[enumToUnderlyingType(matchElement)] = true; }
HashSet<AtomString> idsInRules;
HashSet<AtomString> idsMatchingAncestorsInRules;
HashSet<AtomString> attributeLowercaseLocalNamesInRules;
HashSet<AtomString> attributeLocalNamesInRules;
HashSet<AtomString> contentAttributeNamesInRules;
HashMap<AtomString, std::unique_ptr<RuleFeatureVector>> idRules;
HashMap<AtomString, std::unique_ptr<RuleFeatureVector>> classRules;
HashMap<AtomString, std::unique_ptr<Vector<RuleFeatureWithInvalidationSelector>>> attributeRules;
HashMap<PseudoClassInvalidationKey, std::unique_ptr<RuleFeatureVector>> pseudoClassRules;
HashMap<PseudoClassInvalidationKey, std::unique_ptr<Vector<RuleFeatureWithInvalidationSelector>>> hasPseudoClassRules;
Vector<RuleAndSelector> scopeBreakingHasPseudoClassRules;
HashSet<AtomString> classesAffectingHost;
HashSet<AtomString> attributesAffectingHost;
HashSet<CSSSelector::PseudoClass, IntHash<CSSSelector::PseudoClass>, WTF::StrongEnumHashTraits<CSSSelector::PseudoClass>> pseudoClassesAffectingHost;
HashSet<CSSSelector::PseudoClass, IntHash<CSSSelector::PseudoClass>, WTF::StrongEnumHashTraits<CSSSelector::PseudoClass>> pseudoClasses;
std::array<bool, matchElementCount> usedMatchElements { };
bool usesFirstLineRules { false };
bool usesFirstLetterRules { false };
bool hasStartingStyleRules { false };
private:
struct SelectorFeatures {
using InvalidationFeature = std::tuple<const CSSSelector*, MatchElement, IsNegation>;
using HasInvalidationFeature = std::tuple<const CSSSelector*, MatchElement, IsNegation, DoesBreakScope>;
Vector<InvalidationFeature> ids;
Vector<InvalidationFeature> classes;
Vector<InvalidationFeature> attributes;
Vector<InvalidationFeature> pseudoClasses;
Vector<HasInvalidationFeature> hasPseudoClasses;
};
DoesBreakScope recursivelyCollectFeaturesFromSelector(SelectorFeatures&, const CSSSelector&, MatchElement = MatchElement::Subject, IsNegation = IsNegation::No, CanBreakScope = CanBreakScope::No);
};
bool isHasPseudoClassMatchElement(MatchElement);
MatchElement computeHasPseudoClassMatchElement(const CSSSelector&);
enum class InvalidationKeyType : uint8_t { Universal = 1, Class, Id, Tag };
PseudoClassInvalidationKey makePseudoClassInvalidationKey(CSSSelector::PseudoClass, InvalidationKeyType, const AtomString& = starAtom());
inline bool isUniversalInvalidation(const PseudoClassInvalidationKey& key)
{
return static_cast<InvalidationKeyType>(std::get<1>(key)) == InvalidationKeyType::Universal;
}
inline bool RuleFeatureSet::usesHasPseudoClass() const
{
return usesMatchElement(MatchElement::HasChild)
|| usesMatchElement(MatchElement::HasDescendant)
|| usesMatchElement(MatchElement::HasSibling)
|| usesMatchElement(MatchElement::HasSiblingDescendant)
|| usesMatchElement(MatchElement::HasAnySibling)
|| usesMatchElement(MatchElement::HasNonSubject)
|| usesMatchElement(MatchElement::HasScopeBreaking);
}
} // namespace Style
} // namespace WebCore