mirror of
https://github.com/LadybirdBrowser/ladybird.git
synced 2025-10-24 01:53:20 +00:00

Implements idea described in https://docs.google.com/document/d/1vEW86DaeVs4uQzNFI5R-_xS9TcS1Cs_EUsHRSgCHGu8 Invalidation sets are used to reduce the number of elements marked for style recalculation by collecting metadata from style rules about the dependencies between properties that could affect an element’s style. Currently, this optimization is only applied to style invalidation triggered by class list mutations on an element.
3113 lines
147 KiB
C++
3113 lines
147 KiB
C++
/*
|
||
* Copyright (c) 2018-2023, Andreas Kling <andreas@ladybird.org>
|
||
* Copyright (c) 2021, the SerenityOS developers.
|
||
* Copyright (c) 2021-2024, Sam Atkins <sam@ladybird.org>
|
||
* Copyright (c) 2024, Matthew Olsson <mattco@serenityos.org>
|
||
*
|
||
* SPDX-License-Identifier: BSD-2-Clause
|
||
*/
|
||
|
||
#include <AK/BinarySearch.h>
|
||
#include <AK/Debug.h>
|
||
#include <AK/Error.h>
|
||
#include <AK/Find.h>
|
||
#include <AK/Function.h>
|
||
#include <AK/HashMap.h>
|
||
#include <AK/Math.h>
|
||
#include <AK/NonnullRawPtr.h>
|
||
#include <AK/QuickSort.h>
|
||
#include <AK/TemporaryChange.h>
|
||
#include <LibGfx/Font/Font.h>
|
||
#include <LibGfx/Font/FontDatabase.h>
|
||
#include <LibGfx/Font/FontStyleMapping.h>
|
||
#include <LibGfx/Font/FontWeight.h>
|
||
#include <LibGfx/Font/ScaledFont.h>
|
||
#include <LibGfx/Font/Typeface.h>
|
||
#include <LibGfx/Font/WOFF/Loader.h>
|
||
#include <LibGfx/Font/WOFF2/Loader.h>
|
||
#include <LibWeb/Animations/AnimationEffect.h>
|
||
#include <LibWeb/Animations/DocumentTimeline.h>
|
||
#include <LibWeb/CSS/AnimationEvent.h>
|
||
#include <LibWeb/CSS/CSSAnimation.h>
|
||
#include <LibWeb/CSS/CSSFontFaceRule.h>
|
||
#include <LibWeb/CSS/CSSImportRule.h>
|
||
#include <LibWeb/CSS/CSSLayerBlockRule.h>
|
||
#include <LibWeb/CSS/CSSLayerStatementRule.h>
|
||
#include <LibWeb/CSS/CSSNestedDeclarations.h>
|
||
#include <LibWeb/CSS/CSSStyleRule.h>
|
||
#include <LibWeb/CSS/CSSTransition.h>
|
||
#include <LibWeb/CSS/Interpolation.h>
|
||
#include <LibWeb/CSS/InvalidationSet.h>
|
||
#include <LibWeb/CSS/Parser/Parser.h>
|
||
#include <LibWeb/CSS/SelectorEngine.h>
|
||
#include <LibWeb/CSS/StyleComputer.h>
|
||
#include <LibWeb/CSS/StyleSheet.h>
|
||
#include <LibWeb/CSS/StyleValues/AngleStyleValue.h>
|
||
#include <LibWeb/CSS/StyleValues/BorderRadiusStyleValue.h>
|
||
#include <LibWeb/CSS/StyleValues/CSSColorValue.h>
|
||
#include <LibWeb/CSS/StyleValues/CSSKeywordValue.h>
|
||
#include <LibWeb/CSS/StyleValues/CustomIdentStyleValue.h>
|
||
#include <LibWeb/CSS/StyleValues/DisplayStyleValue.h>
|
||
#include <LibWeb/CSS/StyleValues/EasingStyleValue.h>
|
||
#include <LibWeb/CSS/StyleValues/FilterValueListStyleValue.h>
|
||
#include <LibWeb/CSS/StyleValues/FrequencyStyleValue.h>
|
||
#include <LibWeb/CSS/StyleValues/GridTrackPlacementStyleValue.h>
|
||
#include <LibWeb/CSS/StyleValues/GridTrackSizeListStyleValue.h>
|
||
#include <LibWeb/CSS/StyleValues/IntegerStyleValue.h>
|
||
#include <LibWeb/CSS/StyleValues/LengthStyleValue.h>
|
||
#include <LibWeb/CSS/StyleValues/MathDepthStyleValue.h>
|
||
#include <LibWeb/CSS/StyleValues/NumberStyleValue.h>
|
||
#include <LibWeb/CSS/StyleValues/PercentageStyleValue.h>
|
||
#include <LibWeb/CSS/StyleValues/PositionStyleValue.h>
|
||
#include <LibWeb/CSS/StyleValues/RatioStyleValue.h>
|
||
#include <LibWeb/CSS/StyleValues/RectStyleValue.h>
|
||
#include <LibWeb/CSS/StyleValues/ShorthandStyleValue.h>
|
||
#include <LibWeb/CSS/StyleValues/StringStyleValue.h>
|
||
#include <LibWeb/CSS/StyleValues/StyleValueList.h>
|
||
#include <LibWeb/CSS/StyleValues/TimeStyleValue.h>
|
||
#include <LibWeb/CSS/StyleValues/TransformationStyleValue.h>
|
||
#include <LibWeb/CSS/StyleValues/TransitionStyleValue.h>
|
||
#include <LibWeb/CSS/StyleValues/UnresolvedStyleValue.h>
|
||
#include <LibWeb/DOM/Attr.h>
|
||
#include <LibWeb/DOM/Document.h>
|
||
#include <LibWeb/DOM/Element.h>
|
||
#include <LibWeb/DOM/ShadowRoot.h>
|
||
#include <LibWeb/HTML/HTMLBRElement.h>
|
||
#include <LibWeb/HTML/HTMLHtmlElement.h>
|
||
#include <LibWeb/HTML/Parser/HTMLParser.h>
|
||
#include <LibWeb/HTML/Scripting/TemporaryExecutionContext.h>
|
||
#include <LibWeb/HighResolutionTime/TimeOrigin.h>
|
||
#include <LibWeb/Layout/Node.h>
|
||
#include <LibWeb/MimeSniff/MimeType.h>
|
||
#include <LibWeb/MimeSniff/Resource.h>
|
||
#include <LibWeb/Namespace.h>
|
||
#include <LibWeb/Painting/PaintableBox.h>
|
||
#include <LibWeb/Platform/FontPlugin.h>
|
||
#include <LibWeb/ReferrerPolicy/AbstractOperations.h>
|
||
#include <math.h>
|
||
#include <stdio.h>
|
||
|
||
namespace Web::CSS {
|
||
|
||
struct FontFaceKey {
|
||
NonnullRawPtr<FlyString const> family_name;
|
||
int weight { 0 };
|
||
int slope { 0 };
|
||
};
|
||
|
||
}
|
||
|
||
namespace AK {
|
||
|
||
namespace Detail {
|
||
template<>
|
||
inline constexpr bool IsHashCompatible<Web::CSS::FontFaceKey, Web::CSS::OwnFontFaceKey> = true;
|
||
template<>
|
||
inline constexpr bool IsHashCompatible<Web::CSS::OwnFontFaceKey, Web::CSS::FontFaceKey> = true;
|
||
}
|
||
|
||
template<>
|
||
struct Traits<Web::CSS::FontFaceKey> : public DefaultTraits<Web::CSS::FontFaceKey> {
|
||
static unsigned hash(Web::CSS::FontFaceKey const& key) { return pair_int_hash(key.family_name->hash(), pair_int_hash(key.weight, key.slope)); }
|
||
};
|
||
|
||
template<>
|
||
struct Traits<Web::CSS::OwnFontFaceKey> : public DefaultTraits<Web::CSS::OwnFontFaceKey> {
|
||
static unsigned hash(Web::CSS::OwnFontFaceKey const& key) { return pair_int_hash(key.family_name.hash(), pair_int_hash(key.weight, key.slope)); }
|
||
};
|
||
|
||
}
|
||
|
||
namespace Web::CSS {
|
||
|
||
PropertyOwningCSSStyleDeclaration const& MatchingRule::declaration() const
|
||
{
|
||
if (rule->type() == CSSRule::Type::Style)
|
||
return static_cast<CSSStyleRule const&>(*rule).declaration();
|
||
if (rule->type() == CSSRule::Type::NestedDeclarations)
|
||
return static_cast<CSSNestedDeclarations const&>(*rule).declaration();
|
||
VERIFY_NOT_REACHED();
|
||
}
|
||
|
||
SelectorList const& MatchingRule::absolutized_selectors() const
|
||
{
|
||
if (rule->type() == CSSRule::Type::Style)
|
||
return static_cast<CSSStyleRule const&>(*rule).absolutized_selectors();
|
||
if (rule->type() == CSSRule::Type::NestedDeclarations)
|
||
return static_cast<CSSNestedDeclarations const&>(*rule).parent_style_rule().absolutized_selectors();
|
||
VERIFY_NOT_REACHED();
|
||
}
|
||
|
||
FlyString const& MatchingRule::qualified_layer_name() const
|
||
{
|
||
if (rule->type() == CSSRule::Type::Style)
|
||
return static_cast<CSSStyleRule const&>(*rule).qualified_layer_name();
|
||
if (rule->type() == CSSRule::Type::NestedDeclarations)
|
||
return static_cast<CSSNestedDeclarations const&>(*rule).parent_style_rule().qualified_layer_name();
|
||
VERIFY_NOT_REACHED();
|
||
}
|
||
|
||
OwnFontFaceKey::OwnFontFaceKey(FontFaceKey const& other)
|
||
: family_name(other.family_name)
|
||
, weight(other.weight)
|
||
, slope(other.slope)
|
||
{
|
||
}
|
||
|
||
OwnFontFaceKey::operator FontFaceKey() const
|
||
{
|
||
return FontFaceKey {
|
||
family_name,
|
||
weight,
|
||
slope
|
||
};
|
||
}
|
||
|
||
[[nodiscard]] bool OwnFontFaceKey::operator==(FontFaceKey const& other) const
|
||
{
|
||
return family_name == other.family_name
|
||
&& weight == other.weight
|
||
&& slope == other.slope;
|
||
}
|
||
|
||
static DOM::Element const* element_to_inherit_style_from(DOM::Element const*, Optional<CSS::Selector::PseudoElement::Type>);
|
||
|
||
StyleComputer::StyleComputer(DOM::Document& document)
|
||
: m_document(document)
|
||
, m_default_font_metrics(16, Platform::FontPlugin::the().default_font(16)->pixel_metrics())
|
||
, m_root_element_font_metrics(m_default_font_metrics)
|
||
{
|
||
m_qualified_layer_names_in_order.append({});
|
||
}
|
||
|
||
StyleComputer::~StyleComputer() = default;
|
||
|
||
FontLoader::FontLoader(StyleComputer& style_computer, FlyString family_name, Vector<Gfx::UnicodeRange> unicode_ranges, Vector<URL::URL> urls, Function<void(FontLoader const&)> on_load, Function<void()> on_fail)
|
||
: m_style_computer(style_computer)
|
||
, m_family_name(move(family_name))
|
||
, m_unicode_ranges(move(unicode_ranges))
|
||
, m_urls(move(urls))
|
||
, m_on_load(move(on_load))
|
||
, m_on_fail(move(on_fail))
|
||
{
|
||
}
|
||
|
||
FontLoader::~FontLoader() = default;
|
||
|
||
void FontLoader::resource_did_load()
|
||
{
|
||
resource_did_load_or_fail();
|
||
if (m_on_load)
|
||
m_on_load(*this);
|
||
}
|
||
|
||
void FontLoader::resource_did_fail()
|
||
{
|
||
resource_did_load_or_fail();
|
||
if (m_on_fail) {
|
||
m_on_fail();
|
||
}
|
||
}
|
||
|
||
void FontLoader::resource_did_load_or_fail()
|
||
{
|
||
// NOTE: Even if the resource "failed" to load, we still want to try to parse it as a font.
|
||
// This is necessary for https://wpt.live/ to work correctly, as it just drops the connection
|
||
// after sending a resource, which looks like an error, but is actually recoverable.
|
||
// FIXME: It would be nice to solve this in the network layer instead.
|
||
// It would also be nice to move font loading to using fetch primitives.
|
||
auto result = try_load_font();
|
||
if (result.is_error()) {
|
||
dbgln("Failed to parse font: {}", result.error());
|
||
start_loading_next_url();
|
||
return;
|
||
}
|
||
m_vector_font = result.release_value();
|
||
m_style_computer.did_load_font(m_family_name);
|
||
}
|
||
|
||
RefPtr<Gfx::Font> FontLoader::font_with_point_size(float point_size)
|
||
{
|
||
if (!m_vector_font) {
|
||
if (!resource())
|
||
start_loading_next_url();
|
||
return nullptr;
|
||
}
|
||
return m_vector_font->scaled_font(point_size);
|
||
}
|
||
|
||
void FontLoader::start_loading_next_url()
|
||
{
|
||
if (resource() && resource()->is_pending())
|
||
return;
|
||
if (m_urls.is_empty())
|
||
return;
|
||
LoadRequest request;
|
||
request.set_url(m_urls.take_first());
|
||
|
||
// HACK: We're crudely computing the referer value and shoving it into the
|
||
// request until fetch infrastructure is used here.
|
||
auto referrer_url = ReferrerPolicy::strip_url_for_use_as_referrer(m_style_computer.document().url());
|
||
if (referrer_url.has_value() && !request.headers().contains("Referer"))
|
||
request.set_header("Referer", referrer_url->serialize().to_byte_string());
|
||
|
||
set_resource(ResourceLoader::the().load_resource(Resource::Type::Generic, request));
|
||
}
|
||
|
||
ErrorOr<NonnullRefPtr<Gfx::Typeface>> FontLoader::try_load_font()
|
||
{
|
||
// FIXME: This could maybe use the format() provided in @font-face as well, since often the mime type is just application/octet-stream and we have to try every format
|
||
auto mime_type = MimeSniff::MimeType::parse(resource()->mime_type());
|
||
if (!mime_type.has_value() || !mime_type->is_font()) {
|
||
mime_type = MimeSniff::Resource::sniff(resource()->encoded_data(), Web::MimeSniff::SniffingConfiguration { .sniffing_context = Web::MimeSniff::SniffingContext::Font });
|
||
}
|
||
if (mime_type.has_value()) {
|
||
if (mime_type->essence() == "font/ttf"sv || mime_type->essence() == "application/x-font-ttf"sv || mime_type->essence() == "font/otf"sv) {
|
||
if (auto result = Gfx::Typeface::try_load_from_externally_owned_memory(resource()->encoded_data()); !result.is_error()) {
|
||
return result;
|
||
}
|
||
}
|
||
if (mime_type->essence() == "font/woff"sv || mime_type->essence() == "application/font-woff"sv) {
|
||
if (auto result = WOFF::try_load_from_externally_owned_memory(resource()->encoded_data()); !result.is_error()) {
|
||
return result;
|
||
}
|
||
}
|
||
if (mime_type->essence() == "font/woff2"sv || mime_type->essence() == "application/font-woff2"sv) {
|
||
if (auto result = WOFF2::try_load_from_externally_owned_memory(resource()->encoded_data()); !result.is_error()) {
|
||
return result;
|
||
}
|
||
}
|
||
}
|
||
|
||
return Error::from_string_literal("Automatic format detection failed");
|
||
}
|
||
|
||
struct StyleComputer::MatchingFontCandidate {
|
||
FontFaceKey key;
|
||
Variant<FontLoaderList*, Gfx::Typeface const*> loader_or_typeface;
|
||
|
||
[[nodiscard]] RefPtr<Gfx::FontCascadeList const> font_with_point_size(float point_size) const
|
||
{
|
||
RefPtr<Gfx::FontCascadeList> font_list = Gfx::FontCascadeList::create();
|
||
if (auto* loader_list = loader_or_typeface.get_pointer<FontLoaderList*>(); loader_list) {
|
||
for (auto const& loader : **loader_list) {
|
||
if (auto font = loader->font_with_point_size(point_size); font)
|
||
font_list->add(*font, loader->unicode_ranges());
|
||
}
|
||
return font_list;
|
||
}
|
||
|
||
font_list->add(loader_or_typeface.get<Gfx::Typeface const*>()->scaled_font(point_size));
|
||
return font_list;
|
||
}
|
||
};
|
||
|
||
static CSSStyleSheet& default_stylesheet(DOM::Document const& document)
|
||
{
|
||
static GC::Root<CSSStyleSheet> sheet;
|
||
if (!sheet.cell()) {
|
||
extern String default_stylesheet_source;
|
||
sheet = GC::make_root(parse_css_stylesheet(CSS::Parser::ParsingContext(document), default_stylesheet_source));
|
||
}
|
||
return *sheet;
|
||
}
|
||
|
||
static CSSStyleSheet& quirks_mode_stylesheet(DOM::Document const& document)
|
||
{
|
||
static GC::Root<CSSStyleSheet> sheet;
|
||
if (!sheet.cell()) {
|
||
extern String quirks_mode_stylesheet_source;
|
||
sheet = GC::make_root(parse_css_stylesheet(CSS::Parser::ParsingContext(document), quirks_mode_stylesheet_source));
|
||
}
|
||
return *sheet;
|
||
}
|
||
|
||
static CSSStyleSheet& mathml_stylesheet(DOM::Document const& document)
|
||
{
|
||
static GC::Root<CSSStyleSheet> sheet;
|
||
if (!sheet.cell()) {
|
||
extern String mathml_stylesheet_source;
|
||
sheet = GC::make_root(parse_css_stylesheet(CSS::Parser::ParsingContext(document), mathml_stylesheet_source));
|
||
}
|
||
return *sheet;
|
||
}
|
||
|
||
static CSSStyleSheet& svg_stylesheet(DOM::Document const& document)
|
||
{
|
||
static GC::Root<CSSStyleSheet> sheet;
|
||
if (!sheet.cell()) {
|
||
extern String svg_stylesheet_source;
|
||
sheet = GC::make_root(parse_css_stylesheet(CSS::Parser::ParsingContext(document), svg_stylesheet_source));
|
||
}
|
||
return *sheet;
|
||
}
|
||
|
||
Optional<String> StyleComputer::user_agent_style_sheet_source(StringView name)
|
||
{
|
||
extern String default_stylesheet_source;
|
||
extern String quirks_mode_stylesheet_source;
|
||
extern String mathml_stylesheet_source;
|
||
extern String svg_stylesheet_source;
|
||
|
||
if (name == "CSS/Default.css"sv)
|
||
return default_stylesheet_source;
|
||
if (name == "CSS/QuirksMode.css"sv)
|
||
return quirks_mode_stylesheet_source;
|
||
if (name == "MathML/Default.css"sv)
|
||
return mathml_stylesheet_source;
|
||
if (name == "SVG/Default.css"sv)
|
||
return svg_stylesheet_source;
|
||
return {};
|
||
}
|
||
|
||
template<typename Callback>
|
||
void StyleComputer::for_each_stylesheet(CascadeOrigin cascade_origin, Callback callback) const
|
||
{
|
||
if (cascade_origin == CascadeOrigin::UserAgent) {
|
||
callback(default_stylesheet(document()), {});
|
||
if (document().in_quirks_mode())
|
||
callback(quirks_mode_stylesheet(document()), {});
|
||
callback(mathml_stylesheet(document()), {});
|
||
callback(svg_stylesheet(document()), {});
|
||
}
|
||
if (cascade_origin == CascadeOrigin::User) {
|
||
if (m_user_style_sheet)
|
||
callback(*m_user_style_sheet, {});
|
||
}
|
||
if (cascade_origin == CascadeOrigin::Author) {
|
||
document().for_each_active_css_style_sheet([&](auto& sheet, auto shadow_root) {
|
||
callback(sheet, shadow_root);
|
||
});
|
||
}
|
||
}
|
||
|
||
StyleComputer::RuleCache const& StyleComputer::rule_cache_for_cascade_origin(CascadeOrigin cascade_origin) const
|
||
{
|
||
switch (cascade_origin) {
|
||
case CascadeOrigin::Author:
|
||
return *m_author_rule_cache;
|
||
case CascadeOrigin::User:
|
||
return *m_user_rule_cache;
|
||
case CascadeOrigin::UserAgent:
|
||
return *m_user_agent_rule_cache;
|
||
default:
|
||
TODO();
|
||
}
|
||
}
|
||
|
||
[[nodiscard]] static bool filter_namespace_rule(DOM::Element const& element, MatchingRule const& rule)
|
||
{
|
||
// FIXME: Filter out non-default namespace using prefixes
|
||
if (auto namespace_rule = rule.sheet->default_namespace_rule()) {
|
||
if (namespace_rule->namespace_uri() != element.namespace_uri())
|
||
return false;
|
||
}
|
||
return true;
|
||
}
|
||
|
||
[[nodiscard]] static bool filter_layer(FlyString const& qualified_layer_name, MatchingRule const& rule)
|
||
{
|
||
if (rule.rule && rule.qualified_layer_name() != qualified_layer_name)
|
||
return false;
|
||
return true;
|
||
}
|
||
|
||
bool StyleComputer::should_reject_with_ancestor_filter(Selector const& selector) const
|
||
{
|
||
for (u32 hash : selector.ancestor_hashes()) {
|
||
if (hash == 0)
|
||
break;
|
||
if (!m_ancestor_filter.may_contain(hash))
|
||
return true;
|
||
}
|
||
return false;
|
||
}
|
||
Vector<MatchingRule> const& StyleComputer::get_hover_rules() const
|
||
{
|
||
build_rule_cache_if_needed();
|
||
return m_hover_rules;
|
||
}
|
||
|
||
InvalidationSet StyleComputer::invalidation_set_for_properties(Vector<InvalidationSet::Property> const& properties) const
|
||
{
|
||
if (!m_style_invalidation_data)
|
||
return {};
|
||
auto const& descendant_invalidation_sets = m_style_invalidation_data->descendant_invalidation_sets;
|
||
InvalidationSet result;
|
||
for (auto const& property : properties) {
|
||
if (auto it = descendant_invalidation_sets.find(property); it != descendant_invalidation_sets.end())
|
||
result.include_all_from(it->value);
|
||
}
|
||
return result;
|
||
}
|
||
|
||
bool StyleComputer::invalidation_property_used_in_has_selector(InvalidationSet::Property const& property) const
|
||
{
|
||
if (!m_style_invalidation_data)
|
||
return true;
|
||
switch (property.type) {
|
||
case InvalidationSet::Property::Type::Id:
|
||
if (m_style_invalidation_data->ids_used_in_has_selectors.contains(property.name))
|
||
return true;
|
||
break;
|
||
case InvalidationSet::Property::Type::Class:
|
||
if (m_style_invalidation_data->class_names_used_in_has_selectors.contains(property.name))
|
||
return true;
|
||
break;
|
||
case InvalidationSet::Property::Type::Attribute:
|
||
if (m_style_invalidation_data->attribute_names_used_in_has_selectors.contains(property.name))
|
||
return true;
|
||
break;
|
||
case InvalidationSet::Property::Type::TagName:
|
||
if (m_style_invalidation_data->tag_names_used_in_has_selectors.contains(property.name))
|
||
return true;
|
||
break;
|
||
default:
|
||
break;
|
||
}
|
||
return false;
|
||
}
|
||
|
||
Vector<MatchingRule> StyleComputer::collect_matching_rules(DOM::Element const& element, CascadeOrigin cascade_origin, Optional<CSS::Selector::PseudoElement::Type> pseudo_element, bool& did_match_any_hover_rules, FlyString const& qualified_layer_name) const
|
||
{
|
||
auto const& root_node = element.root();
|
||
auto shadow_root = is<DOM::ShadowRoot>(root_node) ? static_cast<DOM::ShadowRoot const*>(&root_node) : nullptr;
|
||
|
||
GC::Ptr<DOM::Element const> shadow_host;
|
||
if (element.is_shadow_host())
|
||
shadow_host = element;
|
||
else if (shadow_root)
|
||
shadow_host = shadow_root->host();
|
||
|
||
auto const& rule_cache = rule_cache_for_cascade_origin(cascade_origin);
|
||
|
||
Vector<MatchingRule, 512> rules_to_run;
|
||
auto add_rules_to_run = [&](Vector<MatchingRule> const& rules) {
|
||
rules_to_run.grow_capacity(rules_to_run.size() + rules.size());
|
||
if (pseudo_element.has_value()) {
|
||
for (auto const& rule : rules) {
|
||
if (rule.contains_pseudo_element && filter_namespace_rule(element, rule) && filter_layer(qualified_layer_name, rule))
|
||
rules_to_run.unchecked_append(rule);
|
||
}
|
||
} else {
|
||
for (auto const& rule : rules) {
|
||
if (!rule.contains_pseudo_element && filter_namespace_rule(element, rule) && filter_layer(qualified_layer_name, rule))
|
||
rules_to_run.unchecked_append(rule);
|
||
}
|
||
}
|
||
};
|
||
|
||
for (auto const& class_name : element.class_names()) {
|
||
if (auto it = rule_cache.rules_by_class.find(class_name); it != rule_cache.rules_by_class.end())
|
||
add_rules_to_run(it->value);
|
||
}
|
||
if (auto id = element.id(); id.has_value()) {
|
||
if (auto it = rule_cache.rules_by_id.find(id.value()); it != rule_cache.rules_by_id.end())
|
||
add_rules_to_run(it->value);
|
||
}
|
||
if (auto it = rule_cache.rules_by_tag_name.find(element.local_name()); it != rule_cache.rules_by_tag_name.end())
|
||
add_rules_to_run(it->value);
|
||
if (pseudo_element.has_value()) {
|
||
if (CSS::Selector::PseudoElement::is_known_pseudo_element_type(pseudo_element.value())) {
|
||
add_rules_to_run(rule_cache.rules_by_pseudo_element.at(to_underlying(pseudo_element.value())));
|
||
} else {
|
||
// NOTE: We don't cache rules for unknown pseudo-elements. They can't match anything anyway.
|
||
}
|
||
}
|
||
|
||
if (element.is_document_element())
|
||
add_rules_to_run(rule_cache.root_rules);
|
||
|
||
element.for_each_attribute([&](auto& name, auto&) {
|
||
if (auto it = rule_cache.rules_by_attribute_name.find(name); it != rule_cache.rules_by_attribute_name.end()) {
|
||
add_rules_to_run(it->value);
|
||
}
|
||
});
|
||
|
||
add_rules_to_run(rule_cache.other_rules);
|
||
|
||
size_t maximum_match_count = 0;
|
||
|
||
for (auto& rule_to_run : rules_to_run) {
|
||
// FIXME: This needs to be revised when adding support for the ::shadow selector, as it needs to cross shadow boundaries.
|
||
auto rule_root = rule_to_run.shadow_root;
|
||
auto from_user_agent_or_user_stylesheet = rule_to_run.cascade_origin == CascadeOrigin::UserAgent || rule_to_run.cascade_origin == CascadeOrigin::User;
|
||
|
||
// NOTE: Inside shadow trees, we only match rules that are defined in the shadow tree's style sheets.
|
||
// The key exception is the shadow tree's *shadow host*, which needs to match :host rules from inside the shadow root.
|
||
// Also note that UA or User style sheets don't have a scope, so they are always relevant.
|
||
// FIXME: We should reorganize the data so that the document-level StyleComputer doesn't cache *all* rules,
|
||
// but instead we'd have some kind of "style scope" at the document level, and also one for each shadow root.
|
||
// Then we could only evaluate rules from the current style scope.
|
||
bool rule_is_relevant_for_current_scope = rule_root == shadow_root
|
||
|| (element.is_shadow_host() && rule_root == element.shadow_root())
|
||
|| from_user_agent_or_user_stylesheet;
|
||
|
||
if (!rule_is_relevant_for_current_scope) {
|
||
rule_to_run.skip = true;
|
||
continue;
|
||
}
|
||
|
||
auto const& selector = rule_to_run.absolutized_selectors()[rule_to_run.selector_index];
|
||
if (should_reject_with_ancestor_filter(*selector)) {
|
||
rule_to_run.skip = true;
|
||
continue;
|
||
}
|
||
|
||
++maximum_match_count;
|
||
}
|
||
|
||
if (maximum_match_count == 0)
|
||
return {};
|
||
|
||
Vector<MatchingRule> matching_rules;
|
||
matching_rules.ensure_capacity(maximum_match_count);
|
||
|
||
for (auto const& rule_to_run : rules_to_run) {
|
||
if (rule_to_run.skip)
|
||
continue;
|
||
|
||
// NOTE: When matching an element against a rule from outside the shadow root's style scope,
|
||
// we have to pass in null for the shadow host, otherwise combinator traversal will
|
||
// be confined to the element itself (since it refuses to cross the shadow boundary).
|
||
auto rule_root = rule_to_run.shadow_root;
|
||
auto shadow_host_to_use = shadow_host;
|
||
if (element.is_shadow_host() && rule_root != element.shadow_root())
|
||
shadow_host_to_use = nullptr;
|
||
|
||
auto const& selector = rule_to_run.absolutized_selectors()[rule_to_run.selector_index];
|
||
|
||
SelectorEngine::MatchContext context { .style_sheet_for_rule = *rule_to_run.sheet };
|
||
ScopeGuard guard = [&] {
|
||
if (context.did_match_any_hover_rules)
|
||
did_match_any_hover_rules = true;
|
||
};
|
||
if (rule_to_run.can_use_fast_matches) {
|
||
if (!SelectorEngine::fast_matches(selector, element, shadow_host_to_use, context))
|
||
continue;
|
||
} else {
|
||
if (!SelectorEngine::matches(selector, element, shadow_host_to_use, context, pseudo_element))
|
||
continue;
|
||
}
|
||
matching_rules.append(rule_to_run);
|
||
}
|
||
return matching_rules;
|
||
}
|
||
|
||
static void sort_matching_rules(Vector<MatchingRule>& matching_rules)
|
||
{
|
||
quick_sort(matching_rules, [&](MatchingRule& a, MatchingRule& b) {
|
||
auto const& a_selector = a.absolutized_selectors()[a.selector_index];
|
||
auto const& b_selector = b.absolutized_selectors()[b.selector_index];
|
||
auto a_specificity = a_selector->specificity();
|
||
auto b_specificity = b_selector->specificity();
|
||
if (a_specificity == b_specificity) {
|
||
if (a.style_sheet_index == b.style_sheet_index)
|
||
return a.rule_index < b.rule_index;
|
||
return a.style_sheet_index < b.style_sheet_index;
|
||
}
|
||
return a_specificity < b_specificity;
|
||
});
|
||
}
|
||
|
||
void StyleComputer::for_each_property_expanding_shorthands(PropertyID property_id, CSSStyleValue const& value, AllowUnresolved allow_unresolved, Function<void(PropertyID, CSSStyleValue const&)> const& set_longhand_property)
|
||
{
|
||
auto map_logical_property_to_real_property = [](PropertyID property_id) -> Optional<PropertyID> {
|
||
// FIXME: Honor writing-mode, direction and text-orientation.
|
||
switch (property_id) {
|
||
case PropertyID::MarginBlockStart:
|
||
return PropertyID::MarginTop;
|
||
case PropertyID::MarginBlockEnd:
|
||
return PropertyID::MarginBottom;
|
||
case PropertyID::MarginInlineStart:
|
||
return PropertyID::MarginLeft;
|
||
case PropertyID::MarginInlineEnd:
|
||
return PropertyID::MarginRight;
|
||
case PropertyID::PaddingBlockStart:
|
||
return PropertyID::PaddingTop;
|
||
case PropertyID::PaddingBlockEnd:
|
||
return PropertyID::PaddingBottom;
|
||
case PropertyID::PaddingInlineStart:
|
||
return PropertyID::PaddingLeft;
|
||
case PropertyID::PaddingInlineEnd:
|
||
return PropertyID::PaddingRight;
|
||
case PropertyID::InlineSize:
|
||
return PropertyID::Width;
|
||
case PropertyID::InsetBlockStart:
|
||
return PropertyID::Top;
|
||
case PropertyID::InsetBlockEnd:
|
||
return PropertyID::Bottom;
|
||
case PropertyID::InsetInlineStart:
|
||
return PropertyID::Left;
|
||
case PropertyID::InsetInlineEnd:
|
||
return PropertyID::Right;
|
||
default:
|
||
return {};
|
||
}
|
||
};
|
||
|
||
struct StartAndEndPropertyIDs {
|
||
PropertyID start;
|
||
PropertyID end;
|
||
};
|
||
auto map_logical_property_to_real_properties = [](PropertyID property_id) -> Optional<StartAndEndPropertyIDs> {
|
||
// FIXME: Honor writing-mode, direction and text-orientation.
|
||
switch (property_id) {
|
||
case PropertyID::MarginBlock:
|
||
return StartAndEndPropertyIDs { PropertyID::MarginTop, PropertyID::MarginBottom };
|
||
case PropertyID::MarginInline:
|
||
return StartAndEndPropertyIDs { PropertyID::MarginLeft, PropertyID::MarginRight };
|
||
case PropertyID::PaddingBlock:
|
||
return StartAndEndPropertyIDs { PropertyID::PaddingTop, PropertyID::PaddingBottom };
|
||
case PropertyID::PaddingInline:
|
||
return StartAndEndPropertyIDs { PropertyID::PaddingLeft, PropertyID::PaddingRight };
|
||
case PropertyID::InsetBlock:
|
||
return StartAndEndPropertyIDs { PropertyID::Top, PropertyID::Bottom };
|
||
case PropertyID::InsetInline:
|
||
return StartAndEndPropertyIDs { PropertyID::Left, PropertyID::Right };
|
||
default:
|
||
return {};
|
||
}
|
||
};
|
||
|
||
if (auto real_property_id = map_logical_property_to_real_property(property_id); real_property_id.has_value()) {
|
||
for_each_property_expanding_shorthands(real_property_id.value(), value, allow_unresolved, set_longhand_property);
|
||
return;
|
||
}
|
||
|
||
if (auto real_property_ids = map_logical_property_to_real_properties(property_id); real_property_ids.has_value()) {
|
||
if (value.is_value_list() && value.as_value_list().size() == 2) {
|
||
auto const& start = value.as_value_list().values()[0];
|
||
auto const& end = value.as_value_list().values()[1];
|
||
for_each_property_expanding_shorthands(real_property_ids->start, start, allow_unresolved, set_longhand_property);
|
||
for_each_property_expanding_shorthands(real_property_ids->end, end, allow_unresolved, set_longhand_property);
|
||
return;
|
||
}
|
||
for_each_property_expanding_shorthands(real_property_ids->start, value, allow_unresolved, set_longhand_property);
|
||
for_each_property_expanding_shorthands(real_property_ids->end, value, allow_unresolved, set_longhand_property);
|
||
return;
|
||
}
|
||
|
||
if (value.is_shorthand()) {
|
||
auto& shorthand_value = value.as_shorthand();
|
||
auto& properties = shorthand_value.sub_properties();
|
||
auto& values = shorthand_value.values();
|
||
for (size_t i = 0; i < properties.size(); ++i)
|
||
for_each_property_expanding_shorthands(properties[i], values[i], allow_unresolved, set_longhand_property);
|
||
return;
|
||
}
|
||
|
||
auto assign_edge_values = [&](PropertyID top_property, PropertyID right_property, PropertyID bottom_property, PropertyID left_property, auto const& values) {
|
||
if (values.size() == 4) {
|
||
set_longhand_property(top_property, values[0]);
|
||
set_longhand_property(right_property, values[1]);
|
||
set_longhand_property(bottom_property, values[2]);
|
||
set_longhand_property(left_property, values[3]);
|
||
} else if (values.size() == 3) {
|
||
set_longhand_property(top_property, values[0]);
|
||
set_longhand_property(right_property, values[1]);
|
||
set_longhand_property(bottom_property, values[2]);
|
||
set_longhand_property(left_property, values[1]);
|
||
} else if (values.size() == 2) {
|
||
set_longhand_property(top_property, values[0]);
|
||
set_longhand_property(right_property, values[1]);
|
||
set_longhand_property(bottom_property, values[0]);
|
||
set_longhand_property(left_property, values[1]);
|
||
} else if (values.size() == 1) {
|
||
set_longhand_property(top_property, values[0]);
|
||
set_longhand_property(right_property, values[0]);
|
||
set_longhand_property(bottom_property, values[0]);
|
||
set_longhand_property(left_property, values[0]);
|
||
}
|
||
};
|
||
|
||
if (property_id == CSS::PropertyID::Border) {
|
||
for_each_property_expanding_shorthands(CSS::PropertyID::BorderTop, value, allow_unresolved, set_longhand_property);
|
||
for_each_property_expanding_shorthands(CSS::PropertyID::BorderRight, value, allow_unresolved, set_longhand_property);
|
||
for_each_property_expanding_shorthands(CSS::PropertyID::BorderBottom, value, allow_unresolved, set_longhand_property);
|
||
for_each_property_expanding_shorthands(CSS::PropertyID::BorderLeft, value, allow_unresolved, set_longhand_property);
|
||
// FIXME: Also reset border-image, in line with the spec: https://www.w3.org/TR/css-backgrounds-3/#border-shorthands
|
||
return;
|
||
}
|
||
|
||
if (property_id == CSS::PropertyID::BorderStyle) {
|
||
if (value.is_value_list()) {
|
||
auto const& values_list = value.as_value_list();
|
||
assign_edge_values(PropertyID::BorderTopStyle, PropertyID::BorderRightStyle, PropertyID::BorderBottomStyle, PropertyID::BorderLeftStyle, values_list.values());
|
||
return;
|
||
}
|
||
|
||
set_longhand_property(CSS::PropertyID::BorderTopStyle, value);
|
||
set_longhand_property(CSS::PropertyID::BorderRightStyle, value);
|
||
set_longhand_property(CSS::PropertyID::BorderBottomStyle, value);
|
||
set_longhand_property(CSS::PropertyID::BorderLeftStyle, value);
|
||
return;
|
||
}
|
||
|
||
if (property_id == CSS::PropertyID::BorderWidth) {
|
||
if (value.is_value_list()) {
|
||
auto const& values_list = value.as_value_list();
|
||
assign_edge_values(PropertyID::BorderTopWidth, PropertyID::BorderRightWidth, PropertyID::BorderBottomWidth, PropertyID::BorderLeftWidth, values_list.values());
|
||
return;
|
||
}
|
||
|
||
set_longhand_property(CSS::PropertyID::BorderTopWidth, value);
|
||
set_longhand_property(CSS::PropertyID::BorderRightWidth, value);
|
||
set_longhand_property(CSS::PropertyID::BorderBottomWidth, value);
|
||
set_longhand_property(CSS::PropertyID::BorderLeftWidth, value);
|
||
return;
|
||
}
|
||
|
||
if (property_id == CSS::PropertyID::BorderColor) {
|
||
if (value.is_value_list()) {
|
||
auto const& values_list = value.as_value_list();
|
||
assign_edge_values(PropertyID::BorderTopColor, PropertyID::BorderRightColor, PropertyID::BorderBottomColor, PropertyID::BorderLeftColor, values_list.values());
|
||
return;
|
||
}
|
||
|
||
set_longhand_property(CSS::PropertyID::BorderTopColor, value);
|
||
set_longhand_property(CSS::PropertyID::BorderRightColor, value);
|
||
set_longhand_property(CSS::PropertyID::BorderBottomColor, value);
|
||
set_longhand_property(CSS::PropertyID::BorderLeftColor, value);
|
||
return;
|
||
}
|
||
|
||
if (property_id == CSS::PropertyID::BackgroundPosition) {
|
||
if (value.is_position()) {
|
||
auto const& position = value.as_position();
|
||
set_longhand_property(CSS::PropertyID::BackgroundPositionX, position.edge_x());
|
||
set_longhand_property(CSS::PropertyID::BackgroundPositionY, position.edge_y());
|
||
} else if (value.is_value_list()) {
|
||
// Expand background-position layer list into separate lists for x and y positions:
|
||
auto const& values_list = value.as_value_list();
|
||
StyleValueVector x_positions {};
|
||
StyleValueVector y_positions {};
|
||
x_positions.ensure_capacity(values_list.size());
|
||
y_positions.ensure_capacity(values_list.size());
|
||
for (auto& layer : values_list.values()) {
|
||
if (layer->is_position()) {
|
||
auto const& position = layer->as_position();
|
||
x_positions.unchecked_append(position.edge_x());
|
||
y_positions.unchecked_append(position.edge_y());
|
||
} else {
|
||
x_positions.unchecked_append(layer);
|
||
y_positions.unchecked_append(layer);
|
||
}
|
||
}
|
||
set_longhand_property(CSS::PropertyID::BackgroundPositionX, StyleValueList::create(move(x_positions), values_list.separator()));
|
||
set_longhand_property(CSS::PropertyID::BackgroundPositionY, StyleValueList::create(move(y_positions), values_list.separator()));
|
||
} else {
|
||
set_longhand_property(CSS::PropertyID::BackgroundPositionX, value);
|
||
set_longhand_property(CSS::PropertyID::BackgroundPositionY, value);
|
||
}
|
||
|
||
return;
|
||
}
|
||
|
||
if (property_id == CSS::PropertyID::Inset) {
|
||
if (value.is_value_list()) {
|
||
auto const& values_list = value.as_value_list();
|
||
assign_edge_values(PropertyID::Top, PropertyID::Right, PropertyID::Bottom, PropertyID::Left, values_list.values());
|
||
return;
|
||
}
|
||
|
||
set_longhand_property(CSS::PropertyID::Top, value);
|
||
set_longhand_property(CSS::PropertyID::Right, value);
|
||
set_longhand_property(CSS::PropertyID::Bottom, value);
|
||
set_longhand_property(CSS::PropertyID::Left, value);
|
||
return;
|
||
}
|
||
|
||
if (property_id == CSS::PropertyID::Margin) {
|
||
if (value.is_value_list()) {
|
||
auto const& values_list = value.as_value_list();
|
||
assign_edge_values(PropertyID::MarginTop, PropertyID::MarginRight, PropertyID::MarginBottom, PropertyID::MarginLeft, values_list.values());
|
||
return;
|
||
}
|
||
|
||
set_longhand_property(CSS::PropertyID::MarginTop, value);
|
||
set_longhand_property(CSS::PropertyID::MarginRight, value);
|
||
set_longhand_property(CSS::PropertyID::MarginBottom, value);
|
||
set_longhand_property(CSS::PropertyID::MarginLeft, value);
|
||
return;
|
||
}
|
||
|
||
if (property_id == CSS::PropertyID::Padding) {
|
||
if (value.is_value_list()) {
|
||
auto const& values_list = value.as_value_list();
|
||
assign_edge_values(PropertyID::PaddingTop, PropertyID::PaddingRight, PropertyID::PaddingBottom, PropertyID::PaddingLeft, values_list.values());
|
||
return;
|
||
}
|
||
|
||
set_longhand_property(CSS::PropertyID::PaddingTop, value);
|
||
set_longhand_property(CSS::PropertyID::PaddingRight, value);
|
||
set_longhand_property(CSS::PropertyID::PaddingBottom, value);
|
||
set_longhand_property(CSS::PropertyID::PaddingLeft, value);
|
||
return;
|
||
}
|
||
|
||
if (property_id == CSS::PropertyID::Gap) {
|
||
if (value.is_value_list()) {
|
||
auto const& values_list = value.as_value_list();
|
||
set_longhand_property(CSS::PropertyID::RowGap, values_list.values()[0]);
|
||
set_longhand_property(CSS::PropertyID::ColumnGap, values_list.values()[1]);
|
||
return;
|
||
}
|
||
set_longhand_property(CSS::PropertyID::RowGap, value);
|
||
set_longhand_property(CSS::PropertyID::ColumnGap, value);
|
||
return;
|
||
}
|
||
|
||
if (property_id == CSS::PropertyID::MaxInlineSize || property_id == CSS::PropertyID::MinInlineSize) {
|
||
// FIXME: Use writing-mode to determine if we should set width or height.
|
||
bool is_horizontal = true;
|
||
|
||
if (is_horizontal) {
|
||
if (property_id == CSS::PropertyID::MaxInlineSize) {
|
||
set_longhand_property(CSS::PropertyID::MaxWidth, value);
|
||
} else {
|
||
set_longhand_property(CSS::PropertyID::MinWidth, value);
|
||
}
|
||
} else {
|
||
if (property_id == CSS::PropertyID::MaxInlineSize) {
|
||
set_longhand_property(CSS::PropertyID::MaxHeight, value);
|
||
} else {
|
||
set_longhand_property(CSS::PropertyID::MinHeight, value);
|
||
}
|
||
}
|
||
return;
|
||
}
|
||
|
||
if (property_id == CSS::PropertyID::Transition) {
|
||
if (!value.is_transition()) {
|
||
// Handle `none` as a shorthand for `all 0s ease 0s`.
|
||
set_longhand_property(CSS::PropertyID::TransitionProperty, CSSKeywordValue::create(Keyword::All));
|
||
set_longhand_property(CSS::PropertyID::TransitionDuration, TimeStyleValue::create(CSS::Time::make_seconds(0)));
|
||
set_longhand_property(CSS::PropertyID::TransitionDelay, TimeStyleValue::create(CSS::Time::make_seconds(0)));
|
||
set_longhand_property(CSS::PropertyID::TransitionTimingFunction, CSSKeywordValue::create(Keyword::Ease));
|
||
return;
|
||
}
|
||
auto const& transitions = value.as_transition().transitions();
|
||
Array<Vector<ValueComparingNonnullRefPtr<CSSStyleValue const>>, 4> transition_values;
|
||
for (auto const& transition : transitions) {
|
||
transition_values[0].append(*transition.property_name);
|
||
transition_values[1].append(transition.duration.as_style_value());
|
||
transition_values[2].append(transition.delay.as_style_value());
|
||
if (transition.easing)
|
||
transition_values[3].append(*transition.easing);
|
||
}
|
||
|
||
set_longhand_property(CSS::PropertyID::TransitionProperty, StyleValueList::create(move(transition_values[0]), StyleValueList::Separator::Comma));
|
||
set_longhand_property(CSS::PropertyID::TransitionDuration, StyleValueList::create(move(transition_values[1]), StyleValueList::Separator::Comma));
|
||
set_longhand_property(CSS::PropertyID::TransitionDelay, StyleValueList::create(move(transition_values[2]), StyleValueList::Separator::Comma));
|
||
set_longhand_property(CSS::PropertyID::TransitionTimingFunction, StyleValueList::create(move(transition_values[3]), StyleValueList::Separator::Comma));
|
||
return;
|
||
}
|
||
|
||
if (property_id == CSS::PropertyID::Float) {
|
||
auto keyword = value.to_keyword();
|
||
|
||
// FIXME: Honor writing-mode, direction and text-orientation.
|
||
if (keyword == Keyword::InlineStart) {
|
||
set_longhand_property(CSS::PropertyID::Float, CSSKeywordValue::create(Keyword::Left));
|
||
return;
|
||
} else if (keyword == Keyword::InlineEnd) {
|
||
set_longhand_property(CSS::PropertyID::Float, CSSKeywordValue::create(Keyword::Right));
|
||
return;
|
||
}
|
||
}
|
||
|
||
if (property_is_shorthand(property_id)) {
|
||
// ShorthandStyleValue was handled already.
|
||
// That means if we got here, that `value` must be a CSS-wide keyword, which we should apply to our longhand properties.
|
||
// We don't directly call `set_longhand_property()` because the longhands might have longhands of their own.
|
||
// (eg `grid` -> `grid-template` -> `grid-template-areas` & `grid-template-rows` & `grid-template-columns`)
|
||
// Forget this requirement if we're ignoring unresolved values and the value is unresolved.
|
||
VERIFY(value.is_css_wide_keyword() || (allow_unresolved == AllowUnresolved::Yes && value.is_unresolved()));
|
||
for (auto longhand : longhands_for_shorthand(property_id))
|
||
for_each_property_expanding_shorthands(longhand, value, allow_unresolved, set_longhand_property);
|
||
return;
|
||
}
|
||
|
||
set_longhand_property(property_id, value);
|
||
}
|
||
|
||
void StyleComputer::set_property_expanding_shorthands(
|
||
CascadedProperties& cascaded_properties,
|
||
PropertyID property_id,
|
||
CSSStyleValue const& value,
|
||
GC::Ptr<CSSStyleDeclaration const> declaration,
|
||
CascadeOrigin cascade_origin,
|
||
Important important,
|
||
Optional<FlyString> layer_name)
|
||
{
|
||
for_each_property_expanding_shorthands(property_id, value, AllowUnresolved::No, [&](PropertyID longhand_id, CSSStyleValue const& longhand_value) {
|
||
if (longhand_value.is_revert()) {
|
||
cascaded_properties.revert_property(longhand_id, important, cascade_origin);
|
||
} else if (longhand_value.is_revert_layer()) {
|
||
cascaded_properties.revert_layer_property(longhand_id, important, layer_name);
|
||
} else {
|
||
cascaded_properties.set_property(longhand_id, longhand_value, important, cascade_origin, layer_name, declaration);
|
||
}
|
||
});
|
||
}
|
||
|
||
void StyleComputer::set_all_properties(
|
||
CascadedProperties& cascaded_properties,
|
||
DOM::Element& element,
|
||
Optional<Selector::PseudoElement::Type> pseudo_element,
|
||
CSSStyleValue const& value,
|
||
DOM::Document& document,
|
||
GC::Ptr<CSSStyleDeclaration const> declaration,
|
||
CascadeOrigin cascade_origin,
|
||
Important important,
|
||
Optional<FlyString> layer_name) const
|
||
{
|
||
for (auto i = to_underlying(CSS::first_longhand_property_id); i <= to_underlying(CSS::last_longhand_property_id); ++i) {
|
||
auto property_id = (CSS::PropertyID)i;
|
||
|
||
if (value.is_revert()) {
|
||
cascaded_properties.revert_property(property_id, important, cascade_origin);
|
||
continue;
|
||
}
|
||
|
||
if (value.is_revert_layer()) {
|
||
cascaded_properties.revert_layer_property(property_id, important, layer_name);
|
||
continue;
|
||
}
|
||
|
||
NonnullRefPtr<CSSStyleValue> property_value = value;
|
||
if (property_value->is_unresolved())
|
||
property_value = Parser::Parser::resolve_unresolved_style_value(Parser::ParsingContext { document }, element, pseudo_element, property_id, property_value->as_unresolved());
|
||
if (!property_value->is_unresolved())
|
||
set_property_expanding_shorthands(cascaded_properties, property_id, property_value, declaration, cascade_origin, important, layer_name);
|
||
|
||
set_property_expanding_shorthands(cascaded_properties, property_id, value, declaration, cascade_origin, important, layer_name);
|
||
}
|
||
}
|
||
|
||
void StyleComputer::cascade_declarations(
|
||
CascadedProperties& cascaded_properties,
|
||
DOM::Element& element,
|
||
Optional<CSS::Selector::PseudoElement::Type> pseudo_element,
|
||
Vector<MatchingRule> const& matching_rules,
|
||
CascadeOrigin cascade_origin,
|
||
Important important,
|
||
Optional<FlyString> layer_name) const
|
||
{
|
||
for (auto const& match : matching_rules) {
|
||
for (auto const& property : match.declaration().properties()) {
|
||
if (important != property.important)
|
||
continue;
|
||
|
||
if (property.property_id == CSS::PropertyID::All) {
|
||
set_all_properties(cascaded_properties, element, pseudo_element, property.value, m_document, &match.declaration(), cascade_origin, important, layer_name);
|
||
continue;
|
||
}
|
||
|
||
auto property_value = property.value;
|
||
if (property.value->is_unresolved())
|
||
property_value = Parser::Parser::resolve_unresolved_style_value(Parser::ParsingContext { document() }, element, pseudo_element, property.property_id, property.value->as_unresolved());
|
||
if (!property_value->is_unresolved())
|
||
set_property_expanding_shorthands(cascaded_properties, property.property_id, property_value, &match.declaration(), cascade_origin, important, layer_name);
|
||
}
|
||
}
|
||
|
||
if (cascade_origin == CascadeOrigin::Author && !pseudo_element.has_value()) {
|
||
if (auto const inline_style = element.inline_style()) {
|
||
for (auto const& property : inline_style->properties()) {
|
||
if (important != property.important)
|
||
continue;
|
||
|
||
if (property.property_id == CSS::PropertyID::All) {
|
||
set_all_properties(cascaded_properties, element, pseudo_element, property.value, m_document, inline_style, cascade_origin, important, layer_name);
|
||
continue;
|
||
}
|
||
|
||
auto property_value = property.value;
|
||
if (property.value->is_unresolved())
|
||
property_value = Parser::Parser::resolve_unresolved_style_value(Parser::ParsingContext { document() }, element, pseudo_element, property.property_id, property.value->as_unresolved());
|
||
if (!property_value->is_unresolved())
|
||
set_property_expanding_shorthands(cascaded_properties, property.property_id, property_value, inline_style, cascade_origin, important, layer_name);
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
static void cascade_custom_properties(DOM::Element& element, Optional<CSS::Selector::PseudoElement::Type> pseudo_element, Vector<MatchingRule> const& matching_rules, HashMap<FlyString, StyleProperty>& custom_properties)
|
||
{
|
||
size_t needed_capacity = 0;
|
||
for (auto const& matching_rule : matching_rules)
|
||
needed_capacity += matching_rule.declaration().custom_properties().size();
|
||
|
||
if (!pseudo_element.has_value()) {
|
||
if (auto const inline_style = element.inline_style())
|
||
needed_capacity += inline_style->custom_properties().size();
|
||
}
|
||
|
||
custom_properties.ensure_capacity(custom_properties.size() + needed_capacity);
|
||
|
||
for (auto const& matching_rule : matching_rules) {
|
||
for (auto const& it : matching_rule.declaration().custom_properties()) {
|
||
auto style_value = it.value.value;
|
||
if (style_value->is_revert_layer())
|
||
continue;
|
||
custom_properties.set(it.key, it.value);
|
||
}
|
||
}
|
||
|
||
if (!pseudo_element.has_value()) {
|
||
if (auto const inline_style = element.inline_style()) {
|
||
for (auto const& it : inline_style->custom_properties())
|
||
custom_properties.set(it.key, it.value);
|
||
}
|
||
}
|
||
}
|
||
|
||
void StyleComputer::collect_animation_into(DOM::Element& element, Optional<CSS::Selector::PseudoElement::Type> pseudo_element, GC::Ref<Animations::KeyframeEffect> effect, ComputedProperties& computed_properties, AnimationRefresh refresh) const
|
||
{
|
||
auto animation = effect->associated_animation();
|
||
if (!animation)
|
||
return;
|
||
|
||
auto output_progress = effect->transformed_progress();
|
||
if (!output_progress.has_value())
|
||
return;
|
||
|
||
if (!effect->key_frame_set())
|
||
return;
|
||
|
||
auto& keyframes = effect->key_frame_set()->keyframes_by_key;
|
||
if (keyframes.size() < 2) {
|
||
if constexpr (LIBWEB_CSS_ANIMATION_DEBUG) {
|
||
dbgln(" Did not find enough keyframes ({} keyframes)", keyframes.size());
|
||
for (auto it = keyframes.begin(); it != keyframes.end(); ++it)
|
||
dbgln(" - {}", it.key());
|
||
}
|
||
return;
|
||
}
|
||
|
||
auto key = static_cast<i64>(round(output_progress.value() * 100.0 * Animations::KeyframeEffect::AnimationKeyFrameKeyScaleFactor));
|
||
auto keyframe_start_it = [&] {
|
||
if (output_progress.value() <= 0) {
|
||
return keyframes.begin();
|
||
}
|
||
auto potential_match = keyframes.find_largest_not_above_iterator(key);
|
||
if (output_progress.value() >= 0) {
|
||
return --potential_match;
|
||
}
|
||
return potential_match;
|
||
}();
|
||
auto keyframe_start = static_cast<i64>(keyframe_start_it.key());
|
||
auto keyframe_values = *keyframe_start_it;
|
||
|
||
auto keyframe_end_it = ++keyframe_start_it;
|
||
VERIFY(!keyframe_end_it.is_end());
|
||
auto keyframe_end = static_cast<i64>(keyframe_end_it.key());
|
||
auto keyframe_end_values = *keyframe_end_it;
|
||
|
||
auto progress_in_keyframe
|
||
= static_cast<float>(key - keyframe_start) / static_cast<float>(keyframe_end - keyframe_start);
|
||
|
||
if constexpr (LIBWEB_CSS_ANIMATION_DEBUG) {
|
||
auto valid_properties = keyframe_values.properties.size();
|
||
dbgln("Animation {} contains {} properties to interpolate, progress = {}%", animation->id(), valid_properties, progress_in_keyframe * 100);
|
||
}
|
||
|
||
for (auto const& it : keyframe_values.properties) {
|
||
auto resolve_property = [&](auto& property) {
|
||
return property.visit(
|
||
[&](Animations::KeyframeEffect::KeyFrameSet::UseInitial) -> RefPtr<CSSStyleValue const> {
|
||
if (refresh == AnimationRefresh::Yes)
|
||
return {};
|
||
return computed_properties.property(it.key);
|
||
},
|
||
[&](RefPtr<CSSStyleValue const> value) -> RefPtr<CSSStyleValue const> {
|
||
if (value->is_revert() || value->is_revert_layer())
|
||
return computed_properties.property(it.key);
|
||
if (value->is_unresolved())
|
||
return Parser::Parser::resolve_unresolved_style_value(Parser::ParsingContext { element.document() }, element, pseudo_element, it.key, value->as_unresolved());
|
||
return value;
|
||
});
|
||
};
|
||
|
||
auto resolved_start_property = resolve_property(it.value);
|
||
|
||
auto const& end_property = keyframe_end_values.properties.get(it.key);
|
||
if (!end_property.has_value()) {
|
||
if (resolved_start_property) {
|
||
computed_properties.set_animated_property(it.key, *resolved_start_property);
|
||
dbgln_if(LIBWEB_CSS_ANIMATION_DEBUG, "No end property for property {}, using {}", string_from_property_id(it.key), resolved_start_property->to_string(CSSStyleValue::SerializationMode::Normal));
|
||
}
|
||
continue;
|
||
}
|
||
|
||
auto resolved_end_property = resolve_property(end_property.value());
|
||
|
||
if (resolved_end_property && !resolved_start_property)
|
||
resolved_start_property = property_initial_value(it.key);
|
||
|
||
if (!resolved_start_property || !resolved_end_property)
|
||
continue;
|
||
|
||
auto start = resolved_start_property.release_nonnull();
|
||
auto end = resolved_end_property.release_nonnull();
|
||
|
||
if (computed_properties.is_property_important(it.key)) {
|
||
continue;
|
||
}
|
||
|
||
if (auto next_value = interpolate_property(*effect->target(), it.key, *start, *end, progress_in_keyframe)) {
|
||
dbgln_if(LIBWEB_CSS_ANIMATION_DEBUG, "Interpolated value for property {} at {}: {} -> {} = {}", string_from_property_id(it.key), progress_in_keyframe, start->to_string(CSSStyleValue::SerializationMode::Normal), end->to_string(CSSStyleValue::SerializationMode::Normal), next_value->to_string(CSSStyleValue::SerializationMode::Normal));
|
||
computed_properties.set_animated_property(it.key, *next_value);
|
||
} else {
|
||
// If interpolate_property() fails, the element should not be rendered
|
||
dbgln_if(LIBWEB_CSS_ANIMATION_DEBUG, "Interpolated value for property {} at {}: {} -> {} is invalid", string_from_property_id(it.key), progress_in_keyframe, start->to_string(CSSStyleValue::SerializationMode::Normal), end->to_string(CSSStyleValue::SerializationMode::Normal));
|
||
computed_properties.set_animated_property(PropertyID::Visibility, CSSKeywordValue::create(Keyword::Hidden));
|
||
}
|
||
}
|
||
}
|
||
|
||
static void apply_animation_properties(DOM::Document& document, CascadedProperties& cascaded_properties, Animations::Animation& animation)
|
||
{
|
||
if (!animation.effect())
|
||
return;
|
||
|
||
auto& effect = verify_cast<Animations::KeyframeEffect>(*animation.effect());
|
||
|
||
Optional<CSS::Time> duration;
|
||
if (auto duration_value = cascaded_properties.property(PropertyID::AnimationDuration); duration_value) {
|
||
if (duration_value->is_time()) {
|
||
duration = duration_value->as_time().time();
|
||
} else if (duration_value->is_keyword() && duration_value->as_keyword().keyword() == Keyword::Auto) {
|
||
// We use empty optional to represent "auto".
|
||
duration = {};
|
||
}
|
||
}
|
||
|
||
CSS::Time delay { 0, CSS::Time::Type::S };
|
||
if (auto delay_value = cascaded_properties.property(PropertyID::AnimationDelay); delay_value && delay_value->is_time())
|
||
delay = delay_value->as_time().time();
|
||
|
||
double iteration_count = 1.0;
|
||
if (auto iteration_count_value = cascaded_properties.property(PropertyID::AnimationIterationCount); iteration_count_value) {
|
||
if (iteration_count_value->is_keyword() && iteration_count_value->to_keyword() == Keyword::Infinite)
|
||
iteration_count = HUGE_VAL;
|
||
else if (iteration_count_value->is_number())
|
||
iteration_count = iteration_count_value->as_number().number();
|
||
}
|
||
|
||
CSS::AnimationFillMode fill_mode { CSS::AnimationFillMode::None };
|
||
if (auto fill_mode_property = cascaded_properties.property(PropertyID::AnimationFillMode); fill_mode_property && fill_mode_property->is_keyword()) {
|
||
if (auto fill_mode_value = keyword_to_animation_fill_mode(fill_mode_property->to_keyword()); fill_mode_value.has_value())
|
||
fill_mode = *fill_mode_value;
|
||
}
|
||
|
||
CSS::AnimationDirection direction { CSS::AnimationDirection::Normal };
|
||
if (auto direction_property = cascaded_properties.property(PropertyID::AnimationDirection); direction_property && direction_property->is_keyword()) {
|
||
if (auto direction_value = keyword_to_animation_direction(direction_property->to_keyword()); direction_value.has_value())
|
||
direction = *direction_value;
|
||
}
|
||
|
||
CSS::AnimationPlayState play_state { CSS::AnimationPlayState::Running };
|
||
if (auto play_state_property = cascaded_properties.property(PropertyID::AnimationPlayState); play_state_property && play_state_property->is_keyword()) {
|
||
if (auto play_state_value = keyword_to_animation_play_state(play_state_property->to_keyword()); play_state_value.has_value())
|
||
play_state = *play_state_value;
|
||
}
|
||
|
||
CSS::EasingStyleValue::Function timing_function { CSS::EasingStyleValue::CubicBezier::ease() };
|
||
if (auto timing_property = cascaded_properties.property(PropertyID::AnimationTimingFunction); timing_property && timing_property->is_easing())
|
||
timing_function = timing_property->as_easing().function();
|
||
|
||
auto iteration_duration = duration.has_value()
|
||
? Variant<double, String> { duration.release_value().to_milliseconds() }
|
||
: "auto"_string;
|
||
effect.set_iteration_duration(iteration_duration);
|
||
effect.set_start_delay(delay.to_milliseconds());
|
||
effect.set_iteration_count(iteration_count);
|
||
effect.set_timing_function(move(timing_function));
|
||
effect.set_fill_mode(Animations::css_fill_mode_to_bindings_fill_mode(fill_mode));
|
||
effect.set_playback_direction(Animations::css_animation_direction_to_bindings_playback_direction(direction));
|
||
|
||
if (play_state != effect.last_css_animation_play_state()) {
|
||
if (play_state == CSS::AnimationPlayState::Running && animation.play_state() == Bindings::AnimationPlayState::Paused) {
|
||
HTML::TemporaryExecutionContext context(document.realm());
|
||
animation.play().release_value_but_fixme_should_propagate_errors();
|
||
} else if (play_state == CSS::AnimationPlayState::Paused && animation.play_state() != Bindings::AnimationPlayState::Paused) {
|
||
HTML::TemporaryExecutionContext context(document.realm());
|
||
animation.pause().release_value_but_fixme_should_propagate_errors();
|
||
}
|
||
|
||
effect.set_last_css_animation_play_state(play_state);
|
||
}
|
||
}
|
||
|
||
static void apply_dimension_attribute(CascadedProperties& cascaded_properties, DOM::Element const& element, FlyString const& attribute_name, CSS::PropertyID property_id)
|
||
{
|
||
auto attribute = element.attribute(attribute_name);
|
||
if (!attribute.has_value())
|
||
return;
|
||
|
||
auto parsed_value = HTML::parse_dimension_value(*attribute);
|
||
if (!parsed_value)
|
||
return;
|
||
|
||
cascaded_properties.set_property_from_presentational_hint(property_id, parsed_value.release_nonnull());
|
||
}
|
||
|
||
static void compute_transitioned_properties(ComputedProperties const& style, DOM::Element& element, Optional<Selector::PseudoElement::Type> pseudo_element)
|
||
{
|
||
// FIXME: Implement transitioning for pseudo-elements
|
||
(void)pseudo_element;
|
||
|
||
auto const source_declaration = style.transition_property_source();
|
||
if (!source_declaration)
|
||
return;
|
||
if (!element.computed_properties())
|
||
return;
|
||
if (source_declaration == element.cached_transition_property_source())
|
||
return;
|
||
// Reparse this transition property
|
||
element.clear_transitions();
|
||
element.set_cached_transition_property_source(*source_declaration);
|
||
|
||
auto const& transition_properties_value = style.property(PropertyID::TransitionProperty);
|
||
auto transition_properties = transition_properties_value.is_value_list()
|
||
? transition_properties_value.as_value_list().values()
|
||
: StyleValueVector { transition_properties_value };
|
||
|
||
Vector<Vector<PropertyID>> properties;
|
||
|
||
for (size_t i = 0; i < transition_properties.size(); i++) {
|
||
auto property_value = transition_properties[i];
|
||
Vector<PropertyID> properties_for_this_transition;
|
||
|
||
if (property_value->is_keyword()) {
|
||
auto keyword = property_value->as_keyword().keyword();
|
||
if (keyword == Keyword::None)
|
||
continue;
|
||
if (keyword == Keyword::All) {
|
||
for (auto prop = first_property_id; prop != last_property_id; prop = static_cast<PropertyID>(to_underlying(prop) + 1))
|
||
properties_for_this_transition.append(prop);
|
||
}
|
||
} else {
|
||
auto maybe_property = property_id_from_string(property_value->as_custom_ident().custom_ident());
|
||
if (!maybe_property.has_value())
|
||
continue;
|
||
|
||
auto transition_property = maybe_property.release_value();
|
||
if (property_is_shorthand(transition_property)) {
|
||
for (auto const& prop : longhands_for_shorthand(transition_property))
|
||
properties_for_this_transition.append(prop);
|
||
} else {
|
||
properties_for_this_transition.append(transition_property);
|
||
}
|
||
}
|
||
|
||
properties.append(move(properties_for_this_transition));
|
||
}
|
||
|
||
auto normalize_transition_length_list = [&properties, &style](PropertyID property, auto make_default_value) {
|
||
auto const* style_value = style.maybe_null_property(property);
|
||
StyleValueVector list;
|
||
|
||
if (!style_value || !style_value->is_value_list() || style_value->as_value_list().size() == 0) {
|
||
auto default_value = make_default_value();
|
||
for (size_t i = 0; i < properties.size(); i++)
|
||
list.append(default_value);
|
||
return list;
|
||
}
|
||
|
||
auto const& value_list = style_value->as_value_list();
|
||
for (size_t i = 0; i < properties.size(); i++)
|
||
list.append(value_list.value_at(i, true));
|
||
|
||
return list;
|
||
};
|
||
|
||
auto delays = normalize_transition_length_list(
|
||
PropertyID::TransitionDelay,
|
||
[] { return TimeStyleValue::create(Time::make_seconds(0.0)); });
|
||
auto durations = normalize_transition_length_list(
|
||
PropertyID::TransitionDuration,
|
||
[] { return TimeStyleValue::create(Time::make_seconds(0.0)); });
|
||
auto timing_functions = normalize_transition_length_list(
|
||
PropertyID::TransitionTimingFunction,
|
||
[] { return EasingStyleValue::create(EasingStyleValue::CubicBezier::ease()); });
|
||
|
||
element.add_transitioned_properties(move(properties), move(delays), move(durations), move(timing_functions));
|
||
}
|
||
|
||
// https://drafts.csswg.org/css-transitions/#starting
|
||
void StyleComputer::start_needed_transitions(ComputedProperties const& previous_style, ComputedProperties& new_style, DOM::Element& element, Optional<Selector::PseudoElement::Type> pseudo_element) const
|
||
{
|
||
// FIXME: Implement transitions for pseudo-elements
|
||
if (pseudo_element.has_value())
|
||
return;
|
||
|
||
// https://drafts.csswg.org/css-transitions/#transition-combined-duration
|
||
auto combined_duration = [](Animations::Animatable::TransitionAttributes const& transition_attributes) {
|
||
// Define the combined duration of the transition as the sum of max(matching transition duration, 0s) and the matching transition delay.
|
||
return max(transition_attributes.duration, 0) + transition_attributes.delay;
|
||
};
|
||
|
||
// For each element and property, the implementation must act as follows:
|
||
auto style_change_event_time = m_document->timeline()->current_time().value();
|
||
|
||
for (auto i = to_underlying(CSS::first_longhand_property_id); i <= to_underlying(CSS::last_longhand_property_id); ++i) {
|
||
auto property_id = static_cast<CSS::PropertyID>(i);
|
||
auto matching_transition_properties = element.property_transition_attributes(property_id);
|
||
auto const& before_change_value = previous_style.property(property_id, ComputedProperties::WithAnimationsApplied::No);
|
||
auto const& after_change_value = new_style.property(property_id, ComputedProperties::WithAnimationsApplied::No);
|
||
|
||
auto existing_transition = element.property_transition(property_id);
|
||
bool has_running_transition = existing_transition && !existing_transition->is_finished();
|
||
bool has_completed_transition = existing_transition && existing_transition->is_finished();
|
||
|
||
auto start_a_transition = [&](auto start_time, auto end_time, auto const& start_value, auto const& end_value, auto const& reversing_adjusted_start_value, auto reversing_shortening_factor) {
|
||
dbgln_if(CSS_TRANSITIONS_DEBUG, "Starting a transition of {} from {} to {}", string_from_property_id(property_id), start_value->to_string(), end_value->to_string());
|
||
|
||
auto transition = CSSTransition::start_a_transition(element, property_id, document().transition_generation(),
|
||
start_time, end_time, start_value, end_value, reversing_adjusted_start_value, reversing_shortening_factor);
|
||
// Immediately set the property's value to the transition's current value, to prevent single-frame jumps.
|
||
new_style.set_animated_property(property_id, transition->value_at_time(style_change_event_time));
|
||
};
|
||
|
||
// 1. If all of the following are true:
|
||
if (
|
||
// - the element does not have a running transition for the property,
|
||
(!has_running_transition) &&
|
||
// - the before-change style is different from the after-change style for that property, and the values for the property are transitionable,
|
||
(!before_change_value.equals(after_change_value) && property_values_are_transitionable(property_id, before_change_value, after_change_value)) &&
|
||
// - the element does not have a completed transition for the property
|
||
// or the end value of the completed transition is different from the after-change style for the property,
|
||
(!has_completed_transition || !existing_transition->transition_end_value()->equals(after_change_value)) &&
|
||
// - there is a matching transition-property value, and
|
||
(matching_transition_properties.has_value()) &&
|
||
// - the combined duration is greater than 0s,
|
||
(combined_duration(matching_transition_properties.value()) > 0)) {
|
||
|
||
dbgln_if(CSS_TRANSITIONS_DEBUG, "Transition step 1.");
|
||
|
||
// then implementations must remove the completed transition (if present) from the set of completed transitions
|
||
if (has_completed_transition)
|
||
element.remove_transition(property_id);
|
||
// and start a transition whose:
|
||
|
||
// - start time is the time of the style change event plus the matching transition delay,
|
||
auto start_time = style_change_event_time + matching_transition_properties->delay;
|
||
|
||
// - end time is the start time plus the matching transition duration,
|
||
auto end_time = start_time + matching_transition_properties->duration;
|
||
|
||
// - start value is the value of the transitioning property in the before-change style,
|
||
auto const& start_value = before_change_value;
|
||
|
||
// - end value is the value of the transitioning property in the after-change style,
|
||
auto const& end_value = after_change_value;
|
||
|
||
// - reversing-adjusted start value is the same as the start value, and
|
||
auto const& reversing_adjusted_start_value = start_value;
|
||
|
||
// - reversing shortening factor is 1.
|
||
double reversing_shortening_factor = 1;
|
||
|
||
start_a_transition(start_time, end_time, start_value, end_value, reversing_adjusted_start_value, reversing_shortening_factor);
|
||
}
|
||
|
||
// 2. Otherwise, if the element has a completed transition for the property
|
||
// and the end value of the completed transition is different from the after-change style for the property,
|
||
// then implementations must remove the completed transition from the set of completed transitions.
|
||
else if (has_completed_transition && !existing_transition->transition_end_value()->equals(after_change_value)) {
|
||
dbgln_if(CSS_TRANSITIONS_DEBUG, "Transition step 2.");
|
||
element.remove_transition(property_id);
|
||
}
|
||
|
||
// 3. If the element has a running transition or completed transition for the property,
|
||
// and there is not a matching transition-property value,
|
||
if (existing_transition && !matching_transition_properties.has_value()) {
|
||
// then implementations must cancel the running transition or remove the completed transition from the set of completed transitions.
|
||
dbgln_if(CSS_TRANSITIONS_DEBUG, "Transition step 3.");
|
||
if (has_running_transition)
|
||
existing_transition->cancel();
|
||
else
|
||
element.remove_transition(property_id);
|
||
}
|
||
|
||
// 4. If the element has a running transition for the property,
|
||
// there is a matching transition-property value,
|
||
// and the end value of the running transition is not equal to the value of the property in the after-change style, then:
|
||
if (has_running_transition && matching_transition_properties.has_value() && !existing_transition->transition_end_value()->equals(after_change_value)) {
|
||
dbgln_if(CSS_TRANSITIONS_DEBUG, "Transition step 4. existing end value = {}, after change value = {}", existing_transition->transition_end_value()->to_string(CSSStyleValue::SerializationMode::Normal), after_change_value.to_string(CSSStyleValue::SerializationMode::Normal));
|
||
// 1. If the current value of the property in the running transition is equal to the value of the property in the after-change style,
|
||
// or if these two values are not transitionable,
|
||
// then implementations must cancel the running transition.
|
||
auto current_value = existing_transition->value_at_time(style_change_event_time);
|
||
if (current_value->equals(after_change_value) || !property_values_are_transitionable(property_id, current_value, after_change_value)) {
|
||
dbgln_if(CSS_TRANSITIONS_DEBUG, "Transition step 4.1");
|
||
existing_transition->cancel();
|
||
}
|
||
|
||
// 2. Otherwise, if the combined duration is less than or equal to 0s,
|
||
// or if the current value of the property in the running transition is not transitionable with the value of the property in the after-change style,
|
||
// then implementations must cancel the running transition.
|
||
else if ((combined_duration(matching_transition_properties.value()) <= 0)
|
||
|| !property_values_are_transitionable(property_id, current_value, after_change_value)) {
|
||
dbgln_if(CSS_TRANSITIONS_DEBUG, "Transition step 4.2");
|
||
existing_transition->cancel();
|
||
}
|
||
|
||
// 3. Otherwise, if the reversing-adjusted start value of the running transition is the same as the value of the property in the after-change style
|
||
// (see the section on reversing of transitions for why these case exists),
|
||
else if (existing_transition->reversing_adjusted_start_value()->equals(after_change_value)) {
|
||
dbgln_if(CSS_TRANSITIONS_DEBUG, "Transition step 4.3");
|
||
// implementations must cancel the running transition and start a new transition whose:
|
||
existing_transition->cancel();
|
||
// AD-HOC: Remove the cancelled transition, otherwise it breaks the invariant that there is only one
|
||
// running or completed transition for a property at once.
|
||
element.remove_transition(property_id);
|
||
|
||
// - reversing-adjusted start value is the end value of the running transition,
|
||
auto reversing_adjusted_start_value = existing_transition->transition_end_value();
|
||
|
||
// - reversing shortening factor is the absolute value, clamped to the range [0, 1], of the sum of:
|
||
// 1. the output of the timing function of the old transition at the time of the style change event,
|
||
// times the reversing shortening factor of the old transition
|
||
auto term_1 = existing_transition->timing_function_output_at_time(style_change_event_time) * existing_transition->reversing_shortening_factor();
|
||
// 2. 1 minus the reversing shortening factor of the old transition.
|
||
auto term_2 = 1 - existing_transition->reversing_shortening_factor();
|
||
double reversing_shortening_factor = clamp(abs(term_1 + term_2), 0.0, 1.0);
|
||
|
||
// - start time is the time of the style change event plus:
|
||
// 1. if the matching transition delay is nonnegative, the matching transition delay, or
|
||
// 2. if the matching transition delay is negative, the product of the new transition’s reversing shortening factor and the matching transition delay,
|
||
auto start_time = style_change_event_time
|
||
+ (matching_transition_properties->delay >= 0
|
||
? (matching_transition_properties->delay)
|
||
: (reversing_shortening_factor * matching_transition_properties->delay));
|
||
|
||
// - end time is the start time plus the product of the matching transition duration and the new transition’s reversing shortening factor,
|
||
auto end_time = start_time + (matching_transition_properties->duration * reversing_shortening_factor);
|
||
|
||
// - start value is the current value of the property in the running transition,
|
||
auto const& start_value = current_value;
|
||
|
||
// - end value is the value of the property in the after-change style,
|
||
auto const& end_value = after_change_value;
|
||
|
||
start_a_transition(start_time, end_time, start_value, end_value, reversing_adjusted_start_value, reversing_shortening_factor);
|
||
}
|
||
|
||
// 4. Otherwise,
|
||
else {
|
||
dbgln_if(CSS_TRANSITIONS_DEBUG, "Transition step 4.4");
|
||
// implementations must cancel the running transition and start a new transition whose:
|
||
existing_transition->cancel();
|
||
// AD-HOC: Remove the cancelled transition, otherwise it breaks the invariant that there is only one
|
||
// running or completed transition for a property at once.
|
||
element.remove_transition(property_id);
|
||
|
||
// - start time is the time of the style change event plus the matching transition delay,
|
||
auto start_time = style_change_event_time + matching_transition_properties->delay;
|
||
|
||
// - end time is the start time plus the matching transition duration,
|
||
auto end_time = start_time + matching_transition_properties->duration;
|
||
|
||
// - start value is the current value of the property in the running transition,
|
||
auto const& start_value = current_value;
|
||
|
||
// - end value is the value of the property in the after-change style,
|
||
auto const& end_value = after_change_value;
|
||
|
||
// - reversing-adjusted start value is the same as the start value, and
|
||
auto const& reversing_adjusted_start_value = start_value;
|
||
|
||
// - reversing shortening factor is 1.
|
||
double reversing_shortening_factor = 1;
|
||
|
||
start_a_transition(start_time, end_time, start_value, end_value, reversing_adjusted_start_value, reversing_shortening_factor);
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
// https://www.w3.org/TR/css-cascade/#cascading
|
||
// https://drafts.csswg.org/css-cascade-5/#layering
|
||
GC::Ref<CascadedProperties> StyleComputer::compute_cascaded_values(DOM::Element& element, Optional<CSS::Selector::PseudoElement::Type> pseudo_element, bool& did_match_any_pseudo_element_rules, bool& did_match_any_hover_rules, ComputeStyleMode mode) const
|
||
{
|
||
auto cascaded_properties = m_document->heap().allocate<CascadedProperties>();
|
||
|
||
// First, we collect all the CSS rules whose selectors match `element`:
|
||
MatchingRuleSet matching_rule_set;
|
||
matching_rule_set.user_agent_rules = collect_matching_rules(element, CascadeOrigin::UserAgent, pseudo_element, did_match_any_hover_rules);
|
||
sort_matching_rules(matching_rule_set.user_agent_rules);
|
||
matching_rule_set.user_rules = collect_matching_rules(element, CascadeOrigin::User, pseudo_element, did_match_any_hover_rules);
|
||
sort_matching_rules(matching_rule_set.user_rules);
|
||
// @layer-ed author rules
|
||
for (auto const& layer_name : m_qualified_layer_names_in_order) {
|
||
auto layer_rules = collect_matching_rules(element, CascadeOrigin::Author, pseudo_element, did_match_any_hover_rules, layer_name);
|
||
sort_matching_rules(layer_rules);
|
||
matching_rule_set.author_rules.append({ layer_name, layer_rules });
|
||
}
|
||
// Un-@layer-ed author rules
|
||
auto unlayered_author_rules = collect_matching_rules(element, CascadeOrigin::Author, pseudo_element, did_match_any_hover_rules);
|
||
sort_matching_rules(unlayered_author_rules);
|
||
matching_rule_set.author_rules.append({ {}, unlayered_author_rules });
|
||
|
||
if (mode == ComputeStyleMode::CreatePseudoElementStyleIfNeeded) {
|
||
VERIFY(pseudo_element.has_value());
|
||
if (matching_rule_set.author_rules.is_empty() && matching_rule_set.user_rules.is_empty() && matching_rule_set.user_agent_rules.is_empty()) {
|
||
did_match_any_pseudo_element_rules = false;
|
||
return cascaded_properties;
|
||
}
|
||
did_match_any_pseudo_element_rules = true;
|
||
}
|
||
|
||
// Then we resolve all the CSS custom properties ("variables") for this element:
|
||
// FIXME: Also resolve !important custom properties, in a second cascade.
|
||
|
||
HashMap<FlyString, CSS::StyleProperty> custom_properties;
|
||
for (auto& layer : matching_rule_set.author_rules) {
|
||
cascade_custom_properties(element, pseudo_element, layer.rules, custom_properties);
|
||
}
|
||
element.set_custom_properties(pseudo_element, move(custom_properties));
|
||
|
||
// Then we apply the declarations from the matched rules in cascade order:
|
||
|
||
// Normal user agent declarations
|
||
cascade_declarations(*cascaded_properties, element, pseudo_element, matching_rule_set.user_agent_rules, CascadeOrigin::UserAgent, Important::No, {});
|
||
|
||
// Normal user declarations
|
||
cascade_declarations(*cascaded_properties, element, pseudo_element, matching_rule_set.user_rules, CascadeOrigin::User, Important::No, {});
|
||
|
||
// Author presentational hints
|
||
// The spec calls this a special "Author presentational hint origin":
|
||
// "For the purpose of cascading this author presentational hint origin is treated as an independent origin;
|
||
// however for the purpose of the revert keyword (but not for the revert-layer keyword) it is considered
|
||
// part of the author origin."
|
||
// https://drafts.csswg.org/css-cascade-5/#author-presentational-hint-origin
|
||
if (!pseudo_element.has_value()) {
|
||
element.apply_presentational_hints(cascaded_properties);
|
||
if (element.supports_dimension_attributes()) {
|
||
apply_dimension_attribute(cascaded_properties, element, HTML::AttributeNames::width, CSS::PropertyID::Width);
|
||
apply_dimension_attribute(cascaded_properties, element, HTML::AttributeNames::height, CSS::PropertyID::Height);
|
||
}
|
||
|
||
// SVG presentation attributes are parsed as CSS values, so we need to handle potential custom properties here.
|
||
if (element.is_svg_element()) {
|
||
cascaded_properties->resolve_unresolved_properties(element, pseudo_element);
|
||
}
|
||
}
|
||
|
||
// Normal author declarations, ordered by @layer, with un-@layer-ed rules last
|
||
for (auto const& layer : matching_rule_set.author_rules) {
|
||
cascade_declarations(cascaded_properties, element, pseudo_element, layer.rules, CascadeOrigin::Author, Important::No, layer.qualified_layer_name);
|
||
}
|
||
|
||
// Important author declarations, with un-@layer-ed rules first, followed by each @layer in reverse order.
|
||
for (auto const& layer : matching_rule_set.author_rules.in_reverse()) {
|
||
cascade_declarations(cascaded_properties, element, pseudo_element, layer.rules, CascadeOrigin::Author, Important::Yes, {});
|
||
}
|
||
|
||
// Important user declarations
|
||
cascade_declarations(cascaded_properties, element, pseudo_element, matching_rule_set.user_rules, CascadeOrigin::User, Important::Yes, {});
|
||
|
||
// Important user agent declarations
|
||
cascade_declarations(cascaded_properties, element, pseudo_element, matching_rule_set.user_agent_rules, CascadeOrigin::UserAgent, Important::Yes, {});
|
||
|
||
// Transition declarations [css-transitions-1]
|
||
// Note that we have to do these after finishing computing the style,
|
||
// so they're not done here, but as the final step in compute_style_impl()
|
||
|
||
return cascaded_properties;
|
||
}
|
||
|
||
DOM::Element const* element_to_inherit_style_from(DOM::Element const* element, Optional<CSS::Selector::PseudoElement::Type> pseudo_element)
|
||
{
|
||
// Pseudo-elements treat their originating element as their parent.
|
||
DOM::Element const* parent_element = nullptr;
|
||
if (pseudo_element.has_value()) {
|
||
parent_element = element;
|
||
} else if (element) {
|
||
parent_element = element->parent_or_shadow_host_element();
|
||
}
|
||
return parent_element;
|
||
}
|
||
|
||
NonnullRefPtr<CSSStyleValue const> StyleComputer::get_inherit_value(CSS::PropertyID property_id, DOM::Element const* element, Optional<CSS::Selector::PseudoElement::Type> pseudo_element)
|
||
{
|
||
auto* parent_element = element_to_inherit_style_from(element, pseudo_element);
|
||
|
||
if (!parent_element || !parent_element->computed_properties())
|
||
return property_initial_value(property_id);
|
||
return parent_element->computed_properties()->property(property_id);
|
||
}
|
||
|
||
void StyleComputer::compute_defaulted_property_value(ComputedProperties& style, DOM::Element const* element, CSS::PropertyID property_id, Optional<CSS::Selector::PseudoElement::Type> pseudo_element) const
|
||
{
|
||
// FIXME: If we don't know the correct initial value for a property, we fall back to `initial`.
|
||
|
||
auto& value_slot = style.m_property_values[to_underlying(property_id)];
|
||
if (!value_slot) {
|
||
if (is_inherited_property(property_id)) {
|
||
style.set_property(
|
||
property_id,
|
||
get_inherit_value(property_id, element, pseudo_element),
|
||
ComputedProperties::Inherited::Yes,
|
||
Important::No);
|
||
} else {
|
||
style.set_property(property_id, property_initial_value(property_id));
|
||
}
|
||
return;
|
||
}
|
||
|
||
if (value_slot->is_initial()) {
|
||
value_slot = property_initial_value(property_id);
|
||
return;
|
||
}
|
||
|
||
if (value_slot->is_inherit()) {
|
||
value_slot = get_inherit_value(property_id, element, pseudo_element);
|
||
style.set_property_inherited(property_id, ComputedProperties::Inherited::Yes);
|
||
return;
|
||
}
|
||
|
||
// https://www.w3.org/TR/css-cascade-4/#inherit-initial
|
||
// If the cascaded value of a property is the unset keyword,
|
||
if (value_slot->is_unset()) {
|
||
if (is_inherited_property(property_id)) {
|
||
// then if it is an inherited property, this is treated as inherit,
|
||
value_slot = get_inherit_value(property_id, element, pseudo_element);
|
||
style.set_property_inherited(property_id, ComputedProperties::Inherited::Yes);
|
||
} else {
|
||
// and if it is not, this is treated as initial.
|
||
value_slot = property_initial_value(property_id);
|
||
}
|
||
}
|
||
}
|
||
|
||
// https://www.w3.org/TR/css-cascade/#defaulting
|
||
void StyleComputer::compute_defaulted_values(ComputedProperties& style, DOM::Element const* element, Optional<CSS::Selector::PseudoElement::Type> pseudo_element) const
|
||
{
|
||
// Walk the list of all known CSS properties and:
|
||
// - Add them to `style` if they are missing.
|
||
// - Resolve `inherit` and `initial` as needed.
|
||
for (auto i = to_underlying(CSS::first_longhand_property_id); i <= to_underlying(CSS::last_longhand_property_id); ++i) {
|
||
auto property_id = (CSS::PropertyID)i;
|
||
compute_defaulted_property_value(style, element, property_id, pseudo_element);
|
||
}
|
||
|
||
// https://www.w3.org/TR/css-color-4/#resolving-other-colors
|
||
// In the color property, the used value of currentcolor is the inherited value.
|
||
auto const& color = style.property(CSS::PropertyID::Color);
|
||
if (color.to_keyword() == Keyword::Currentcolor) {
|
||
auto const& inherited_value = get_inherit_value(CSS::PropertyID::Color, element, pseudo_element);
|
||
style.set_property(CSS::PropertyID::Color, inherited_value);
|
||
}
|
||
}
|
||
|
||
Length::FontMetrics StyleComputer::calculate_root_element_font_metrics(ComputedProperties const& style) const
|
||
{
|
||
auto const& root_value = style.property(CSS::PropertyID::FontSize);
|
||
|
||
auto font_pixel_metrics = style.first_available_computed_font().pixel_metrics();
|
||
Length::FontMetrics font_metrics { m_default_font_metrics.font_size, font_pixel_metrics };
|
||
font_metrics.font_size = root_value.as_length().length().to_px(viewport_rect(), font_metrics, font_metrics);
|
||
font_metrics.line_height = style.compute_line_height(viewport_rect(), font_metrics, font_metrics);
|
||
|
||
return font_metrics;
|
||
}
|
||
|
||
RefPtr<Gfx::FontCascadeList const> StyleComputer::find_matching_font_weight_ascending(Vector<MatchingFontCandidate> const& candidates, int target_weight, float font_size_in_pt, bool inclusive)
|
||
{
|
||
using Fn = AK::Function<bool(MatchingFontCandidate const&)>;
|
||
auto pred = inclusive ? Fn([&](auto const& matching_font_candidate) { return matching_font_candidate.key.weight >= target_weight; })
|
||
: Fn([&](auto const& matching_font_candidate) { return matching_font_candidate.key.weight > target_weight; });
|
||
auto it = find_if(candidates.begin(), candidates.end(), pred);
|
||
for (; it != candidates.end(); ++it) {
|
||
if (auto found_font = it->font_with_point_size(font_size_in_pt))
|
||
return found_font;
|
||
}
|
||
return {};
|
||
}
|
||
|
||
RefPtr<Gfx::FontCascadeList const> StyleComputer::find_matching_font_weight_descending(Vector<MatchingFontCandidate> const& candidates, int target_weight, float font_size_in_pt, bool inclusive)
|
||
{
|
||
using Fn = AK::Function<bool(MatchingFontCandidate const&)>;
|
||
auto pred = inclusive ? Fn([&](auto const& matching_font_candidate) { return matching_font_candidate.key.weight <= target_weight; })
|
||
: Fn([&](auto const& matching_font_candidate) { return matching_font_candidate.key.weight < target_weight; });
|
||
auto it = find_if(candidates.rbegin(), candidates.rend(), pred);
|
||
for (; it != candidates.rend(); ++it) {
|
||
if (auto found_font = it->font_with_point_size(font_size_in_pt))
|
||
return found_font;
|
||
}
|
||
return {};
|
||
}
|
||
|
||
// Partial implementation of the font-matching algorithm: https://www.w3.org/TR/css-fonts-4/#font-matching-algorithm
|
||
// FIXME: This should be replaced by the full CSS font selection algorithm.
|
||
RefPtr<Gfx::FontCascadeList const> StyleComputer::font_matching_algorithm(FlyString const& family_name, int weight, int slope, float font_size_in_pt) const
|
||
{
|
||
// If a font family match occurs, the user agent assembles the set of font faces in that family and then
|
||
// narrows the set to a single face using other font properties in the order given below.
|
||
Vector<MatchingFontCandidate> matching_family_fonts;
|
||
for (auto const& font_key_and_loader : m_loaded_fonts) {
|
||
if (font_key_and_loader.key.family_name.equals_ignoring_ascii_case(family_name))
|
||
matching_family_fonts.empend(font_key_and_loader.key, const_cast<FontLoaderList*>(&font_key_and_loader.value));
|
||
}
|
||
Gfx::FontDatabase::the().for_each_typeface_with_family_name(family_name, [&](Gfx::Typeface const& typeface) {
|
||
matching_family_fonts.empend(
|
||
FontFaceKey {
|
||
.family_name = typeface.family(),
|
||
.weight = static_cast<int>(typeface.weight()),
|
||
.slope = typeface.slope(),
|
||
},
|
||
&typeface);
|
||
});
|
||
quick_sort(matching_family_fonts, [](auto const& a, auto const& b) {
|
||
return a.key.weight < b.key.weight;
|
||
});
|
||
// FIXME: 1. font-stretch is tried first.
|
||
// FIXME: 2. font-style is tried next.
|
||
// We don't have complete support of italic and oblique fonts, so matching on font-style can be simplified to:
|
||
// If a matching slope is found, all faces which don't have that matching slope are excluded from the matching set.
|
||
auto style_it = find_if(matching_family_fonts.begin(), matching_family_fonts.end(),
|
||
[&](auto const& matching_font_candidate) { return matching_font_candidate.key.slope == slope; });
|
||
if (style_it != matching_family_fonts.end()) {
|
||
matching_family_fonts.remove_all_matching([&](auto const& matching_font_candidate) {
|
||
return matching_font_candidate.key.slope != slope;
|
||
});
|
||
}
|
||
// 3. font-weight is matched next.
|
||
// If the desired weight is inclusively between 400 and 500, weights greater than or equal to the target weight
|
||
// are checked in ascending order until 500 is hit and checked, followed by weights less than the target weight
|
||
// in descending order, followed by weights greater than 500, until a match is found.
|
||
if (weight >= 400 && weight <= 500) {
|
||
auto it = find_if(matching_family_fonts.begin(), matching_family_fonts.end(),
|
||
[&](auto const& matching_font_candidate) { return matching_font_candidate.key.weight >= weight; });
|
||
for (; it != matching_family_fonts.end() && it->key.weight <= 500; ++it) {
|
||
if (auto found_font = it->font_with_point_size(font_size_in_pt))
|
||
return found_font;
|
||
}
|
||
if (auto found_font = find_matching_font_weight_descending(matching_family_fonts, weight, font_size_in_pt, false))
|
||
return found_font;
|
||
for (; it != matching_family_fonts.end(); ++it) {
|
||
if (auto found_font = it->font_with_point_size(font_size_in_pt))
|
||
return found_font;
|
||
}
|
||
}
|
||
// If the desired weight is less than 400, weights less than or equal to the desired weight are checked in descending order
|
||
// followed by weights above the desired weight in ascending order until a match is found.
|
||
if (weight < 400) {
|
||
if (auto found_font = find_matching_font_weight_descending(matching_family_fonts, weight, font_size_in_pt, true))
|
||
return found_font;
|
||
if (auto found_font = find_matching_font_weight_ascending(matching_family_fonts, weight, font_size_in_pt, false))
|
||
return found_font;
|
||
}
|
||
// If the desired weight is greater than 500, weights greater than or equal to the desired weight are checked in ascending order
|
||
// followed by weights below the desired weight in descending order until a match is found.
|
||
if (weight > 500) {
|
||
if (auto found_font = find_matching_font_weight_ascending(matching_family_fonts, weight, font_size_in_pt, true))
|
||
return found_font;
|
||
if (auto found_font = find_matching_font_weight_descending(matching_family_fonts, weight, font_size_in_pt, false))
|
||
return found_font;
|
||
}
|
||
return {};
|
||
}
|
||
|
||
CSSPixels StyleComputer::default_user_font_size()
|
||
{
|
||
// FIXME: This value should be configurable by the user.
|
||
return 16;
|
||
}
|
||
|
||
// https://w3c.github.io/csswg-drafts/css-fonts/#absolute-size-mapping
|
||
CSSPixelFraction StyleComputer::absolute_size_mapping(Keyword keyword)
|
||
{
|
||
switch (keyword) {
|
||
case Keyword::XxSmall:
|
||
return CSSPixels(3) / 5;
|
||
case Keyword::XSmall:
|
||
return CSSPixels(3) / 4;
|
||
case Keyword::Small:
|
||
return CSSPixels(8) / 9;
|
||
case Keyword::Medium:
|
||
return 1;
|
||
case Keyword::Large:
|
||
return CSSPixels(6) / 5;
|
||
case Keyword::XLarge:
|
||
return CSSPixels(3) / 2;
|
||
case Keyword::XxLarge:
|
||
return 2;
|
||
case Keyword::XxxLarge:
|
||
return 3;
|
||
case Keyword::Smaller:
|
||
return CSSPixels(4) / 5;
|
||
case Keyword::Larger:
|
||
return CSSPixels(5) / 4;
|
||
default:
|
||
return 1;
|
||
}
|
||
}
|
||
|
||
RefPtr<Gfx::FontCascadeList const> StyleComputer::compute_font_for_style_values(DOM::Element const* element, Optional<CSS::Selector::PseudoElement::Type> pseudo_element, CSSStyleValue const& font_family, CSSStyleValue const& font_size, CSSStyleValue const& font_style, CSSStyleValue const& font_weight, CSSStyleValue const& font_stretch, int math_depth) const
|
||
{
|
||
auto* parent_element = element_to_inherit_style_from(element, pseudo_element);
|
||
|
||
auto width = font_stretch.to_font_width();
|
||
auto weight = font_weight.to_font_weight();
|
||
|
||
auto font_size_in_px = default_user_font_size();
|
||
|
||
Gfx::FontPixelMetrics font_pixel_metrics;
|
||
if (parent_element && parent_element->computed_properties())
|
||
font_pixel_metrics = parent_element->computed_properties()->first_available_computed_font().pixel_metrics();
|
||
else
|
||
font_pixel_metrics = Platform::FontPlugin::the().default_font(font_size_in_px.to_float())->pixel_metrics();
|
||
auto parent_font_size = [&]() -> CSSPixels {
|
||
if (!parent_element || !parent_element->computed_properties())
|
||
return font_size_in_px;
|
||
auto const& value = parent_element->computed_properties()->property(CSS::PropertyID::FontSize);
|
||
if (value.is_length()) {
|
||
auto length = value.as_length().length();
|
||
if (length.is_absolute() || length.is_relative()) {
|
||
Length::FontMetrics font_metrics { font_size_in_px, font_pixel_metrics };
|
||
return length.to_px(viewport_rect(), font_metrics, m_root_element_font_metrics);
|
||
}
|
||
}
|
||
return font_size_in_px;
|
||
};
|
||
Length::FontMetrics font_metrics { parent_font_size(), font_pixel_metrics };
|
||
|
||
if (font_size.is_keyword()) {
|
||
auto const keyword = font_size.to_keyword();
|
||
|
||
if (keyword == Keyword::Math) {
|
||
auto math_scaling_factor = [&]() {
|
||
// https://w3c.github.io/mathml-core/#the-math-script-level-property
|
||
// If the specified value font-size is math then the computed value of font-size is obtained by multiplying
|
||
// the inherited value of font-size by a nonzero scale factor calculated by the following procedure:
|
||
// 1. Let A be the inherited math-depth value, B the computed math-depth value, C be 0.71 and S be 1.0
|
||
int inherited_math_depth = parent_element && parent_element->computed_properties()
|
||
? parent_element->computed_properties()->math_depth()
|
||
: InitialValues::math_depth();
|
||
int computed_math_depth = math_depth;
|
||
auto size_ratio = 0.71;
|
||
auto scale = 1.0;
|
||
// 2. If A = B then return S.
|
||
bool invert_scale_factor = false;
|
||
if (inherited_math_depth == computed_math_depth) {
|
||
return scale;
|
||
}
|
||
// If B < A, swap A and B and set InvertScaleFactor to true.
|
||
else if (computed_math_depth < inherited_math_depth) {
|
||
AK::swap(inherited_math_depth, computed_math_depth);
|
||
invert_scale_factor = true;
|
||
}
|
||
// Otherwise B > A and set InvertScaleFactor to false.
|
||
else {
|
||
invert_scale_factor = false;
|
||
}
|
||
// 3. Let E be B - A > 0.
|
||
double e = (computed_math_depth - inherited_math_depth) > 0;
|
||
// FIXME: 4. If the inherited first available font has an OpenType MATH table:
|
||
// - If A ≤ 0 and B ≥ 2 then multiply S by scriptScriptPercentScaleDown and decrement E by 2.
|
||
// - Otherwise if A = 1 then multiply S by scriptScriptPercentScaleDown / scriptPercentScaleDown and decrement E by 1.
|
||
// - Otherwise if B = 1 then multiply S by scriptPercentScaleDown and decrement E by 1.
|
||
// 5. Multiply S by C^E.
|
||
scale *= AK::pow(size_ratio, e);
|
||
// 6. Return S if InvertScaleFactor is false and 1/S otherwise.
|
||
if (!invert_scale_factor)
|
||
return scale;
|
||
return 1.0 / scale;
|
||
};
|
||
font_size_in_px = parent_font_size().scale_by(math_scaling_factor());
|
||
} else {
|
||
// https://w3c.github.io/csswg-drafts/css-fonts/#valdef-font-size-relative-size
|
||
// TODO: If the parent element has a keyword font size in the absolute size keyword mapping table,
|
||
// larger may compute the font size to the next entry in the table,
|
||
// and smaller may compute the font size to the previous entry in the table.
|
||
if (keyword == Keyword::Smaller || keyword == Keyword::Larger) {
|
||
if (parent_element && parent_element->computed_properties()) {
|
||
font_size_in_px = CSSPixels::nearest_value_for(parent_element->computed_properties()->first_available_computed_font().pixel_metrics().size);
|
||
}
|
||
}
|
||
font_size_in_px *= absolute_size_mapping(keyword);
|
||
}
|
||
} else {
|
||
Length::ResolutionContext const length_resolution_context {
|
||
.viewport_rect = viewport_rect(),
|
||
.font_metrics = font_metrics,
|
||
.root_font_metrics = m_root_element_font_metrics,
|
||
};
|
||
|
||
Optional<Length> maybe_length;
|
||
if (font_size.is_percentage()) {
|
||
// Percentages refer to parent element's font size
|
||
maybe_length = Length::make_px(CSSPixels::nearest_value_for(font_size.as_percentage().percentage().as_fraction() * parent_font_size().to_double()));
|
||
|
||
} else if (font_size.is_length()) {
|
||
maybe_length = font_size.as_length().length();
|
||
} else if (font_size.is_calculated()) {
|
||
if (font_size.as_calculated().contains_percentage()) {
|
||
maybe_length = font_size.as_calculated().resolve_length_percentage(length_resolution_context, Length::make_px(parent_font_size()));
|
||
} else {
|
||
maybe_length = font_size.as_calculated().resolve_length(length_resolution_context);
|
||
}
|
||
}
|
||
if (maybe_length.has_value()) {
|
||
font_size_in_px = maybe_length.value().to_px(length_resolution_context);
|
||
}
|
||
}
|
||
|
||
auto slope = font_style.to_font_slope();
|
||
|
||
// FIXME: Implement the full font-matching algorithm: https://www.w3.org/TR/css-fonts-4/#font-matching-algorithm
|
||
|
||
float const font_size_in_pt = font_size_in_px * 0.75f;
|
||
|
||
auto find_font = [&](FlyString const& family) -> RefPtr<Gfx::FontCascadeList const> {
|
||
FontFaceKey key {
|
||
.family_name = family,
|
||
.weight = weight,
|
||
.slope = slope,
|
||
};
|
||
auto result = Gfx::FontCascadeList::create();
|
||
if (auto it = m_loaded_fonts.find(key); it != m_loaded_fonts.end()) {
|
||
auto const& loaders = it->value;
|
||
for (auto const& loader : loaders) {
|
||
if (auto found_font = loader->font_with_point_size(font_size_in_pt))
|
||
result->add(*found_font, loader->unicode_ranges());
|
||
}
|
||
return result;
|
||
}
|
||
|
||
if (auto found_font = font_matching_algorithm(family, weight, slope, font_size_in_pt); found_font && !found_font->is_empty()) {
|
||
return found_font;
|
||
}
|
||
|
||
if (auto found_font = Gfx::FontDatabase::the().get(family, font_size_in_pt, weight, width, slope)) {
|
||
result->add(*found_font);
|
||
return result;
|
||
}
|
||
|
||
return {};
|
||
};
|
||
|
||
auto find_generic_font = [&](Keyword font_id) -> RefPtr<Gfx::FontCascadeList const> {
|
||
Platform::GenericFont generic_font {};
|
||
switch (font_id) {
|
||
case Keyword::Monospace:
|
||
case Keyword::UiMonospace:
|
||
generic_font = Platform::GenericFont::Monospace;
|
||
break;
|
||
case Keyword::Serif:
|
||
generic_font = Platform::GenericFont::Serif;
|
||
break;
|
||
case Keyword::Fantasy:
|
||
generic_font = Platform::GenericFont::Fantasy;
|
||
break;
|
||
case Keyword::SansSerif:
|
||
generic_font = Platform::GenericFont::SansSerif;
|
||
break;
|
||
case Keyword::Cursive:
|
||
generic_font = Platform::GenericFont::Cursive;
|
||
break;
|
||
case Keyword::UiSerif:
|
||
generic_font = Platform::GenericFont::UiSerif;
|
||
break;
|
||
case Keyword::UiSansSerif:
|
||
generic_font = Platform::GenericFont::UiSansSerif;
|
||
break;
|
||
case Keyword::UiRounded:
|
||
generic_font = Platform::GenericFont::UiRounded;
|
||
break;
|
||
default:
|
||
return {};
|
||
}
|
||
return find_font(Platform::FontPlugin::the().generic_font_name(generic_font));
|
||
};
|
||
|
||
auto font_list = Gfx::FontCascadeList::create();
|
||
if (font_family.is_value_list()) {
|
||
auto const& family_list = static_cast<StyleValueList const&>(font_family).values();
|
||
for (auto const& family : family_list) {
|
||
RefPtr<Gfx::FontCascadeList const> other_font_list;
|
||
if (family->is_keyword()) {
|
||
other_font_list = find_generic_font(family->to_keyword());
|
||
} else if (family->is_string()) {
|
||
other_font_list = find_font(family->as_string().string_value());
|
||
} else if (family->is_custom_ident()) {
|
||
other_font_list = find_font(family->as_custom_ident().custom_ident());
|
||
}
|
||
if (other_font_list)
|
||
font_list->extend(*other_font_list);
|
||
}
|
||
} else if (font_family.is_keyword()) {
|
||
if (auto other_font_list = find_generic_font(font_family.to_keyword()))
|
||
font_list->extend(*other_font_list);
|
||
} else if (font_family.is_string()) {
|
||
if (auto other_font_list = find_font(font_family.as_string().string_value()))
|
||
font_list->extend(*other_font_list);
|
||
} else if (font_family.is_custom_ident()) {
|
||
if (auto other_font_list = find_font(font_family.as_custom_ident().custom_ident()))
|
||
font_list->extend(*other_font_list);
|
||
}
|
||
|
||
auto default_font = Platform::FontPlugin::the().default_font(font_size_in_pt);
|
||
if (font_list->is_empty()) {
|
||
// This is needed to make sure we check default font before reaching to emojis.
|
||
font_list->add(*default_font);
|
||
}
|
||
|
||
if (auto emoji_font = Platform::FontPlugin::the().default_emoji_font(font_size_in_pt); emoji_font) {
|
||
font_list->add(*emoji_font);
|
||
}
|
||
|
||
// The default font is already included in the font list, but we explicitly set it
|
||
// as the last-resort font. This ensures that if none of the specified fonts contain
|
||
// the requested code point, there is still a font available to provide a fallback glyph.
|
||
font_list->set_last_resort_font(*default_font);
|
||
|
||
return font_list;
|
||
}
|
||
|
||
void StyleComputer::compute_font(ComputedProperties& style, DOM::Element const* element, Optional<CSS::Selector::PseudoElement::Type> pseudo_element) const
|
||
{
|
||
// To compute the font, first ensure that we've defaulted the relevant CSS font properties.
|
||
// FIXME: This should be more sophisticated.
|
||
compute_defaulted_property_value(style, element, CSS::PropertyID::FontFamily, pseudo_element);
|
||
compute_defaulted_property_value(style, element, CSS::PropertyID::FontSize, pseudo_element);
|
||
compute_defaulted_property_value(style, element, CSS::PropertyID::FontWidth, pseudo_element);
|
||
compute_defaulted_property_value(style, element, CSS::PropertyID::FontStyle, pseudo_element);
|
||
compute_defaulted_property_value(style, element, CSS::PropertyID::FontWeight, pseudo_element);
|
||
compute_defaulted_property_value(style, element, CSS::PropertyID::LineHeight, pseudo_element);
|
||
compute_defaulted_property_value(style, element, CSS::PropertyID::FontVariant, pseudo_element);
|
||
compute_defaulted_property_value(style, element, CSS::PropertyID::FontVariantAlternates, pseudo_element);
|
||
compute_defaulted_property_value(style, element, CSS::PropertyID::FontVariantCaps, pseudo_element);
|
||
compute_defaulted_property_value(style, element, CSS::PropertyID::FontVariantEmoji, pseudo_element);
|
||
compute_defaulted_property_value(style, element, CSS::PropertyID::FontVariantEastAsian, pseudo_element);
|
||
compute_defaulted_property_value(style, element, CSS::PropertyID::FontVariantLigatures, pseudo_element);
|
||
compute_defaulted_property_value(style, element, CSS::PropertyID::FontVariantNumeric, pseudo_element);
|
||
compute_defaulted_property_value(style, element, CSS::PropertyID::FontVariantPosition, pseudo_element);
|
||
|
||
auto const& font_family = style.property(CSS::PropertyID::FontFamily);
|
||
auto const& font_size = style.property(CSS::PropertyID::FontSize);
|
||
auto const& font_style = style.property(CSS::PropertyID::FontStyle);
|
||
auto const& font_weight = style.property(CSS::PropertyID::FontWeight);
|
||
auto const& font_width = style.property(CSS::PropertyID::FontWidth);
|
||
|
||
auto font_list = compute_font_for_style_values(element, pseudo_element, font_family, font_size, font_style, font_weight, font_width, style.math_depth());
|
||
VERIFY(font_list);
|
||
VERIFY(!font_list->is_empty());
|
||
|
||
RefPtr<Gfx::Font const> const found_font = font_list->first();
|
||
|
||
style.set_property(
|
||
CSS::PropertyID::FontSize,
|
||
LengthStyleValue::create(CSS::Length::make_px(CSSPixels::nearest_value_for(found_font->pixel_size()))),
|
||
style.is_property_inherited(CSS::PropertyID::FontSize) ? ComputedProperties::Inherited::Yes : ComputedProperties::Inherited::No);
|
||
style.set_property(
|
||
CSS::PropertyID::FontWeight,
|
||
NumberStyleValue::create(font_weight.to_font_weight()),
|
||
style.is_property_inherited(CSS::PropertyID::FontWeight) ? ComputedProperties::Inherited::Yes : ComputedProperties::Inherited::No);
|
||
|
||
style.set_computed_font_list(*font_list);
|
||
|
||
if (element && is<HTML::HTMLHtmlElement>(*element)) {
|
||
const_cast<StyleComputer&>(*this).m_root_element_font_metrics = calculate_root_element_font_metrics(style);
|
||
}
|
||
}
|
||
|
||
Gfx::Font const& StyleComputer::initial_font() const
|
||
{
|
||
// FIXME: This is not correct.
|
||
return ComputedProperties::font_fallback(false, false, 12);
|
||
}
|
||
|
||
void StyleComputer::absolutize_values(ComputedProperties& style) const
|
||
{
|
||
Length::FontMetrics font_metrics {
|
||
m_root_element_font_metrics.font_size,
|
||
style.first_available_computed_font().pixel_metrics()
|
||
};
|
||
|
||
auto font_size = style.property(CSS::PropertyID::FontSize).as_length().length().to_px(viewport_rect(), font_metrics, m_root_element_font_metrics);
|
||
font_metrics.font_size = font_size;
|
||
|
||
// NOTE: Percentage line-height values are relative to the font-size of the element.
|
||
// We have to resolve them right away, so that the *computed* line-height is ready for inheritance.
|
||
// We can't simply absolutize *all* percentage values against the font size,
|
||
// because most percentages are relative to containing block metrics.
|
||
auto& line_height_value_slot = style.m_property_values[to_underlying(CSS::PropertyID::LineHeight)];
|
||
if (line_height_value_slot && line_height_value_slot->is_percentage()) {
|
||
line_height_value_slot = LengthStyleValue::create(
|
||
Length::make_px(CSSPixels::nearest_value_for(font_size * static_cast<double>(line_height_value_slot->as_percentage().percentage().as_fraction()))));
|
||
}
|
||
|
||
auto line_height = style.compute_line_height(viewport_rect(), font_metrics, m_root_element_font_metrics);
|
||
font_metrics.line_height = line_height;
|
||
|
||
// NOTE: line-height might be using lh which should be resolved against the parent line height (like we did here already)
|
||
if (line_height_value_slot && line_height_value_slot->is_length())
|
||
line_height_value_slot = LengthStyleValue::create(Length::make_px(line_height));
|
||
|
||
for (size_t i = 0; i < style.m_property_values.size(); ++i) {
|
||
auto& value_slot = style.m_property_values[i];
|
||
if (!value_slot)
|
||
continue;
|
||
value_slot = value_slot->absolutized(viewport_rect(), font_metrics, m_root_element_font_metrics);
|
||
}
|
||
|
||
style.set_line_height({}, line_height);
|
||
}
|
||
|
||
void StyleComputer::resolve_effective_overflow_values(ComputedProperties& style) const
|
||
{
|
||
// https://www.w3.org/TR/css-overflow-3/#overflow-control
|
||
// The visible/clip values of overflow compute to auto/hidden (respectively) if one of overflow-x or
|
||
// overflow-y is neither visible nor clip.
|
||
auto overflow_x = keyword_to_overflow(style.property(PropertyID::OverflowX).to_keyword());
|
||
auto overflow_y = keyword_to_overflow(style.property(PropertyID::OverflowY).to_keyword());
|
||
auto overflow_x_is_visible_or_clip = overflow_x == Overflow::Visible || overflow_x == Overflow::Clip;
|
||
auto overflow_y_is_visible_or_clip = overflow_y == Overflow::Visible || overflow_y == Overflow::Clip;
|
||
if (!overflow_x_is_visible_or_clip || !overflow_y_is_visible_or_clip) {
|
||
if (overflow_x == CSS::Overflow::Visible)
|
||
style.set_property(CSS::PropertyID::OverflowX, CSSKeywordValue::create(Keyword::Auto));
|
||
if (overflow_x == CSS::Overflow::Clip)
|
||
style.set_property(CSS::PropertyID::OverflowX, CSSKeywordValue::create(Keyword::Hidden));
|
||
if (overflow_y == CSS::Overflow::Visible)
|
||
style.set_property(CSS::PropertyID::OverflowY, CSSKeywordValue::create(Keyword::Auto));
|
||
if (overflow_y == CSS::Overflow::Clip)
|
||
style.set_property(CSS::PropertyID::OverflowY, CSSKeywordValue::create(Keyword::Hidden));
|
||
}
|
||
}
|
||
|
||
enum class BoxTypeTransformation {
|
||
None,
|
||
Blockify,
|
||
Inlinify,
|
||
};
|
||
|
||
static BoxTypeTransformation required_box_type_transformation(ComputedProperties const& style, DOM::Element const& element, Optional<CSS::Selector::PseudoElement::Type> const& pseudo_element)
|
||
{
|
||
// NOTE: We never blockify <br> elements. They are always inline.
|
||
// There is currently no way to express in CSS how a <br> element really behaves.
|
||
// Spec issue: https://github.com/whatwg/html/issues/2291
|
||
if (is<HTML::HTMLBRElement>(element))
|
||
return BoxTypeTransformation::None;
|
||
|
||
// Absolute positioning or floating an element blockifies the box’s display type. [CSS2]
|
||
if (style.position() == CSS::Positioning::Absolute || style.position() == CSS::Positioning::Fixed || style.float_() != CSS::Float::None)
|
||
return BoxTypeTransformation::Blockify;
|
||
|
||
// FIXME: Containment in a ruby container inlinifies the box’s display type, as described in [CSS-RUBY-1].
|
||
|
||
// NOTE: If we're computing style for a pseudo-element, the effective parent will be the originating element itself, not its parent.
|
||
auto const* parent = pseudo_element.has_value() ? &element : element.parent_element();
|
||
|
||
// A parent with a grid or flex display value blockifies the box’s display type. [CSS-GRID-1] [CSS-FLEXBOX-1]
|
||
if (parent && parent->computed_properties()) {
|
||
auto const& parent_display = parent->computed_properties()->display();
|
||
if (parent_display.is_grid_inside() || parent_display.is_flex_inside())
|
||
return BoxTypeTransformation::Blockify;
|
||
}
|
||
|
||
return BoxTypeTransformation::None;
|
||
}
|
||
|
||
// https://drafts.csswg.org/css-display/#transformations
|
||
void StyleComputer::transform_box_type_if_needed(ComputedProperties& style, DOM::Element const& element, Optional<CSS::Selector::PseudoElement::Type> pseudo_element) const
|
||
{
|
||
// 2.7. Automatic Box Type Transformations
|
||
|
||
// Some layout effects require blockification or inlinification of the box type,
|
||
// which sets the box’s computed outer display type to block or inline (respectively).
|
||
// (This has no effect on display types that generate no box at all, such as none or contents.)
|
||
|
||
auto display = style.display();
|
||
|
||
if (display.is_none() || (display.is_contents() && !element.is_document_element()))
|
||
return;
|
||
|
||
// https://drafts.csswg.org/css-display/#root
|
||
// The root element’s display type is always blockified, and its principal box always establishes an independent formatting context.
|
||
if (element.is_document_element() && !display.is_block_outside()) {
|
||
style.set_property(CSS::PropertyID::Display, DisplayStyleValue::create(Display::from_short(CSS::Display::Short::Block)));
|
||
return;
|
||
}
|
||
|
||
auto new_display = display;
|
||
|
||
if (display.is_math_inside()) {
|
||
// https://w3c.github.io/mathml-core/#new-display-math-value
|
||
// For elements that are not MathML elements, if the specified value of display is inline math or block math
|
||
// then the computed value is block flow and inline flow respectively.
|
||
if (element.namespace_uri() != Namespace::MathML)
|
||
new_display = CSS::Display { display.outside(), CSS::DisplayInside::Flow };
|
||
// For the mtable element the computed value is block table and inline table respectively.
|
||
else if (element.tag_name().equals_ignoring_ascii_case("mtable"sv))
|
||
new_display = CSS::Display { display.outside(), CSS::DisplayInside::Table };
|
||
// For the mtr element, the computed value is table-row.
|
||
else if (element.tag_name().equals_ignoring_ascii_case("mtr"sv))
|
||
new_display = CSS::Display { CSS::DisplayInternal::TableRow };
|
||
// For the mtd element, the computed value is table-cell.
|
||
else if (element.tag_name().equals_ignoring_ascii_case("mtd"sv))
|
||
new_display = CSS::Display { CSS::DisplayInternal::TableCell };
|
||
}
|
||
|
||
switch (required_box_type_transformation(style, element, pseudo_element)) {
|
||
case BoxTypeTransformation::None:
|
||
break;
|
||
case BoxTypeTransformation::Blockify:
|
||
if (display.is_block_outside())
|
||
return;
|
||
// If a layout-internal box is blockified, its inner display type converts to flow so that it becomes a block container.
|
||
if (display.is_internal()) {
|
||
new_display = CSS::Display::from_short(CSS::Display::Short::Block);
|
||
} else {
|
||
VERIFY(display.is_outside_and_inside());
|
||
|
||
// For legacy reasons, if an inline block box (inline flow-root) is blockified, it becomes a block box (losing its flow-root nature).
|
||
// For consistency, a run-in flow-root box also blockifies to a block box.
|
||
if (display.is_inline_block()) {
|
||
new_display = CSS::Display { CSS::DisplayOutside::Block, CSS::DisplayInside::Flow, display.list_item() };
|
||
} else {
|
||
new_display = CSS::Display { CSS::DisplayOutside::Block, display.inside(), display.list_item() };
|
||
}
|
||
}
|
||
break;
|
||
case BoxTypeTransformation::Inlinify:
|
||
if (display.is_inline_outside()) {
|
||
// FIXME: If an inline box (inline flow) is inlinified, it recursively inlinifies all of its in-flow children,
|
||
// so that no block-level descendants break up the inline formatting context in which it participates.
|
||
if (display.is_flow_inside()) {
|
||
dbgln("FIXME: Inlinify inline box children recursively");
|
||
}
|
||
break;
|
||
}
|
||
if (display.is_internal()) {
|
||
// Inlinification has no effect on layout-internal boxes. (However, placement in such an inline context will typically cause them
|
||
// to be wrapped in an appropriately-typed anonymous inline-level box.)
|
||
} else {
|
||
VERIFY(display.is_outside_and_inside());
|
||
|
||
// If a block box (block flow) is inlinified, its inner display type is set to flow-root so that it remains a block container.
|
||
if (display.is_block_outside() && display.is_flow_inside()) {
|
||
new_display = CSS::Display { CSS::DisplayOutside::Inline, CSS::DisplayInside::FlowRoot, display.list_item() };
|
||
}
|
||
|
||
new_display = CSS::Display { CSS::DisplayOutside::Inline, display.inside(), display.list_item() };
|
||
}
|
||
break;
|
||
}
|
||
|
||
if (new_display != display)
|
||
style.set_property(CSS::PropertyID::Display, DisplayStyleValue::create(new_display));
|
||
}
|
||
|
||
GC::Ref<ComputedProperties> StyleComputer::create_document_style() const
|
||
{
|
||
auto style = document().heap().allocate<CSS::ComputedProperties>();
|
||
compute_math_depth(style, nullptr, {});
|
||
compute_font(style, nullptr, {});
|
||
compute_defaulted_values(style, nullptr, {});
|
||
absolutize_values(style);
|
||
style->set_property(CSS::PropertyID::Width, CSS::LengthStyleValue::create(CSS::Length::make_px(viewport_rect().width())));
|
||
style->set_property(CSS::PropertyID::Height, CSS::LengthStyleValue::create(CSS::Length::make_px(viewport_rect().height())));
|
||
style->set_property(CSS::PropertyID::Display, CSS::DisplayStyleValue::create(CSS::Display::from_short(CSS::Display::Short::Block)));
|
||
return style;
|
||
}
|
||
|
||
GC::Ref<ComputedProperties> StyleComputer::compute_style(DOM::Element& element, Optional<CSS::Selector::PseudoElement::Type> pseudo_element) const
|
||
{
|
||
return *compute_style_impl(element, move(pseudo_element), ComputeStyleMode::Normal);
|
||
}
|
||
|
||
GC::Ptr<ComputedProperties> StyleComputer::compute_pseudo_element_style_if_needed(DOM::Element& element, Optional<CSS::Selector::PseudoElement::Type> pseudo_element) const
|
||
{
|
||
return compute_style_impl(element, move(pseudo_element), ComputeStyleMode::CreatePseudoElementStyleIfNeeded);
|
||
}
|
||
|
||
GC::Ptr<ComputedProperties> StyleComputer::compute_style_impl(DOM::Element& element, Optional<CSS::Selector::PseudoElement::Type> pseudo_element, ComputeStyleMode mode) const
|
||
{
|
||
build_rule_cache_if_needed();
|
||
|
||
// Special path for elements that use pseudo element as style selector
|
||
if (element.use_pseudo_element().has_value()) {
|
||
auto& parent_element = verify_cast<HTML::HTMLElement>(*element.root().parent_or_shadow_host());
|
||
auto style = compute_style(parent_element, *element.use_pseudo_element());
|
||
|
||
// Merge back inline styles
|
||
if (auto inline_style = element.inline_style()) {
|
||
for (auto const& property : inline_style->properties())
|
||
style->set_property(property.property_id, property.value);
|
||
}
|
||
return style;
|
||
}
|
||
|
||
ScopeGuard guard { [&element]() { element.set_needs_style_update(false); } };
|
||
|
||
// 1. Perform the cascade. This produces the "specified style"
|
||
bool did_match_any_pseudo_element_rules = false;
|
||
bool did_match_any_hover_rules = false;
|
||
auto cascaded_properties = compute_cascaded_values(element, pseudo_element, did_match_any_pseudo_element_rules, did_match_any_hover_rules, mode);
|
||
|
||
element.set_cascaded_properties(pseudo_element, cascaded_properties);
|
||
|
||
if (mode == ComputeStyleMode::CreatePseudoElementStyleIfNeeded) {
|
||
// NOTE: If we're computing style for a pseudo-element, we look for a number of reasons to bail early.
|
||
|
||
// Bail if no pseudo-element rules matched.
|
||
if (!did_match_any_pseudo_element_rules)
|
||
return {};
|
||
|
||
// Bail if no pseudo-element would be generated due to...
|
||
// - content: none
|
||
// - content: normal (for ::before and ::after)
|
||
bool content_is_normal = false;
|
||
if (auto content_value = cascaded_properties->property(CSS::PropertyID::Content)) {
|
||
if (content_value->is_keyword()) {
|
||
auto content = content_value->as_keyword().keyword();
|
||
if (content == CSS::Keyword::None)
|
||
return {};
|
||
content_is_normal = content == CSS::Keyword::Normal;
|
||
} else {
|
||
content_is_normal = false;
|
||
}
|
||
} else {
|
||
// NOTE: `normal` is the initial value, so the absence of a value is treated as `normal`.
|
||
content_is_normal = true;
|
||
}
|
||
if (content_is_normal && first_is_one_of(*pseudo_element, CSS::Selector::PseudoElement::Type::Before, CSS::Selector::PseudoElement::Type::After)) {
|
||
return {};
|
||
}
|
||
}
|
||
|
||
auto computed_properties = compute_properties(element, pseudo_element, cascaded_properties);
|
||
if (did_match_any_hover_rules)
|
||
computed_properties->set_did_match_any_hover_rules();
|
||
return computed_properties;
|
||
}
|
||
|
||
GC::Ref<ComputedProperties> StyleComputer::compute_properties(DOM::Element& element, Optional<Selector::PseudoElement::Type> pseudo_element, CascadedProperties& cascaded_properties) const
|
||
{
|
||
auto computed_style = document().heap().allocate<CSS::ComputedProperties>();
|
||
|
||
for (auto i = to_underlying(first_longhand_property_id); i <= to_underlying(last_longhand_property_id); ++i) {
|
||
auto property_id = static_cast<CSS::PropertyID>(i);
|
||
auto value = cascaded_properties.property(property_id);
|
||
auto inherited = ComputedProperties::Inherited::No;
|
||
|
||
if ((!value && is_inherited_property(property_id))
|
||
|| (value && value->is_inherit())) {
|
||
if (auto inheritance_parent = element_to_inherit_style_from(&element, pseudo_element)) {
|
||
value = inheritance_parent->computed_properties()->property(property_id);
|
||
inherited = ComputedProperties::Inherited::Yes;
|
||
} else {
|
||
value = property_initial_value(property_id);
|
||
}
|
||
}
|
||
|
||
if (!value || value->is_initial())
|
||
value = property_initial_value(property_id);
|
||
|
||
if (value->is_unset()) {
|
||
if (is_inherited_property(property_id))
|
||
value = CSSKeywordValue::create(Keyword::Inherit);
|
||
else
|
||
value = CSSKeywordValue::create(Keyword::Initial);
|
||
}
|
||
|
||
computed_style->set_property(property_id, value.release_nonnull(), inherited);
|
||
|
||
if (property_id == PropertyID::AnimationName) {
|
||
computed_style->set_animation_name_source(cascaded_properties.property_source(property_id));
|
||
}
|
||
if (property_id == PropertyID::TransitionProperty) {
|
||
computed_style->set_transition_property_source(cascaded_properties.property_source(property_id));
|
||
}
|
||
}
|
||
|
||
// Animation declarations [css-animations-2]
|
||
auto animation_name = [&]() -> Optional<String> {
|
||
auto const animation_name = computed_style->maybe_null_property(PropertyID::AnimationName);
|
||
if (!animation_name)
|
||
return OptionalNone {};
|
||
if (animation_name->is_string())
|
||
return animation_name->as_string().string_value().to_string();
|
||
return animation_name->to_string(CSSStyleValue::SerializationMode::Normal);
|
||
}();
|
||
|
||
if (animation_name.has_value()) {
|
||
if (auto source_declaration = computed_style->animation_name_source()) {
|
||
auto& realm = element.realm();
|
||
|
||
if (source_declaration != element.cached_animation_name_source(pseudo_element)) {
|
||
// This animation name is new, so we need to create a new animation for it.
|
||
if (auto existing_animation = element.cached_animation_name_animation(pseudo_element))
|
||
existing_animation->cancel(Animations::Animation::ShouldInvalidate::No);
|
||
element.set_cached_animation_name_source(source_declaration, pseudo_element);
|
||
|
||
auto effect = Animations::KeyframeEffect::create(realm);
|
||
auto animation = CSSAnimation::create(realm);
|
||
animation->set_id(animation_name.release_value());
|
||
animation->set_timeline(m_document->timeline());
|
||
animation->set_owning_element(element);
|
||
animation->set_effect(effect);
|
||
apply_animation_properties(m_document, cascaded_properties, animation);
|
||
if (pseudo_element.has_value())
|
||
effect->set_pseudo_element(Selector::PseudoElement { pseudo_element.value() });
|
||
|
||
auto const& rule_cache = rule_cache_for_cascade_origin(CascadeOrigin::Author);
|
||
if (auto keyframe_set = rule_cache.rules_by_animation_keyframes.get(animation->id()); keyframe_set.has_value())
|
||
effect->set_key_frame_set(keyframe_set.value());
|
||
|
||
effect->set_target(&element);
|
||
element.set_cached_animation_name_animation(animation, pseudo_element);
|
||
|
||
HTML::TemporaryExecutionContext context(realm);
|
||
animation->play().release_value_but_fixme_should_propagate_errors();
|
||
} else {
|
||
// The animation hasn't changed, but some properties of the animation may have
|
||
if (auto animation = element.cached_animation_name_animation(pseudo_element); animation)
|
||
apply_animation_properties(m_document, cascaded_properties, *animation);
|
||
}
|
||
}
|
||
} else {
|
||
// If the element had an existing animation, cancel it
|
||
if (auto existing_animation = element.cached_animation_name_animation(pseudo_element)) {
|
||
existing_animation->cancel(Animations::Animation::ShouldInvalidate::No);
|
||
element.set_cached_animation_name_animation({}, pseudo_element);
|
||
element.set_cached_animation_name_source({}, pseudo_element);
|
||
}
|
||
}
|
||
|
||
auto animations = element.get_animations_internal(Animations::GetAnimationsOptions { .subtree = false });
|
||
if (animations.is_exception()) {
|
||
dbgln("Error getting animations for element {}", element.debug_description());
|
||
} else {
|
||
for (auto& animation : animations.value()) {
|
||
if (auto effect = animation->effect(); effect && effect->is_keyframe_effect()) {
|
||
auto& keyframe_effect = *static_cast<Animations::KeyframeEffect*>(effect.ptr());
|
||
if (keyframe_effect.pseudo_element_type() == pseudo_element)
|
||
collect_animation_into(element, pseudo_element, keyframe_effect, computed_style);
|
||
}
|
||
}
|
||
}
|
||
|
||
// 2. Compute the math-depth property, since that might affect the font-size
|
||
compute_math_depth(computed_style, &element, pseudo_element);
|
||
|
||
// 3. Compute the font, since that may be needed for font-relative CSS units
|
||
compute_font(computed_style, &element, pseudo_element);
|
||
|
||
// 4. Absolutize values, turning font/viewport relative lengths into absolute lengths
|
||
absolutize_values(computed_style);
|
||
|
||
// 5. Default the values, applying inheritance and 'initial' as needed
|
||
compute_defaulted_values(computed_style, &element, pseudo_element);
|
||
|
||
// 6. Run automatic box type transformations
|
||
transform_box_type_if_needed(computed_style, element, pseudo_element);
|
||
|
||
// 7. Resolve effective overflow values
|
||
resolve_effective_overflow_values(computed_style);
|
||
|
||
// 8. Let the element adjust computed style
|
||
element.adjust_computed_style(computed_style);
|
||
|
||
// 9. Transition declarations [css-transitions-1]
|
||
// Theoretically this should be part of the cascade, but it works with computed values, which we don't have until now.
|
||
compute_transitioned_properties(computed_style, element, pseudo_element);
|
||
if (auto previous_style = element.computed_properties(); previous_style) {
|
||
start_needed_transitions(*previous_style, computed_style, element, pseudo_element);
|
||
}
|
||
|
||
return computed_style;
|
||
}
|
||
|
||
void StyleComputer::build_rule_cache_if_needed() const
|
||
{
|
||
if (m_author_rule_cache && m_user_rule_cache && m_user_agent_rule_cache)
|
||
return;
|
||
const_cast<StyleComputer&>(*this).build_rule_cache();
|
||
}
|
||
|
||
struct SimplifiedSelectorForBucketing {
|
||
CSS::Selector::SimpleSelector::Type type;
|
||
FlyString name;
|
||
};
|
||
|
||
static Optional<SimplifiedSelectorForBucketing> is_roundabout_selector_bucketable_as_something_simpler(CSS::Selector::SimpleSelector const& simple_selector)
|
||
{
|
||
if (simple_selector.type != CSS::Selector::SimpleSelector::Type::PseudoClass)
|
||
return {};
|
||
|
||
if (simple_selector.pseudo_class().type != CSS::PseudoClass::Is
|
||
&& simple_selector.pseudo_class().type != CSS::PseudoClass::Where)
|
||
return {};
|
||
|
||
if (simple_selector.pseudo_class().argument_selector_list.size() != 1)
|
||
return {};
|
||
|
||
auto const& argument_selector = *simple_selector.pseudo_class().argument_selector_list.first();
|
||
|
||
auto const& compound_selector = argument_selector.compound_selectors().last();
|
||
if (compound_selector.simple_selectors.size() != 1)
|
||
return {};
|
||
|
||
auto const& inner_simple_selector = compound_selector.simple_selectors.first();
|
||
if (inner_simple_selector.type == CSS::Selector::SimpleSelector::Type::Class
|
||
|| inner_simple_selector.type == CSS::Selector::SimpleSelector::Type::Id) {
|
||
return SimplifiedSelectorForBucketing { inner_simple_selector.type, inner_simple_selector.name() };
|
||
}
|
||
|
||
if (inner_simple_selector.type == CSS::Selector::SimpleSelector::Type::TagName) {
|
||
return SimplifiedSelectorForBucketing { inner_simple_selector.type, inner_simple_selector.qualified_name().name.lowercase_name };
|
||
}
|
||
|
||
return {};
|
||
}
|
||
|
||
void StyleComputer::collect_selector_insights(Selector const& selector, SelectorInsights& insights)
|
||
{
|
||
for (auto const& compound_selector : selector.compound_selectors()) {
|
||
for (auto const& simple_selector : compound_selector.simple_selectors) {
|
||
if (simple_selector.type == Selector::SimpleSelector::Type::Attribute) {
|
||
insights.all_names_used_in_attribute_selectors.set(simple_selector.attribute().qualified_name.name.lowercase_name);
|
||
}
|
||
if (simple_selector.type == Selector::SimpleSelector::Type::PseudoClass) {
|
||
if (simple_selector.pseudo_class().type == PseudoClass::Has) {
|
||
insights.has_has_selectors = true;
|
||
}
|
||
if (simple_selector.pseudo_class().type == PseudoClass::Defined) {
|
||
insights.has_defined_selectors = true;
|
||
}
|
||
for (auto const& argument_selector : simple_selector.pseudo_class().argument_selector_list) {
|
||
collect_selector_insights(*argument_selector, insights);
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
NonnullOwnPtr<StyleComputer::RuleCache> StyleComputer::make_rule_cache_for_cascade_origin(CascadeOrigin cascade_origin, SelectorInsights& insights, Vector<MatchingRule>& hover_rules)
|
||
{
|
||
auto rule_cache = make<RuleCache>();
|
||
|
||
size_t num_class_rules = 0;
|
||
size_t num_id_rules = 0;
|
||
size_t num_tag_name_rules = 0;
|
||
size_t num_pseudo_element_rules = 0;
|
||
size_t num_root_rules = 0;
|
||
size_t num_attribute_rules = 0;
|
||
size_t num_hover_rules = 0;
|
||
|
||
Vector<MatchingRule> matching_rules;
|
||
size_t style_sheet_index = 0;
|
||
for_each_stylesheet(cascade_origin, [&](auto& sheet, GC::Ptr<DOM::ShadowRoot> shadow_root) {
|
||
size_t rule_index = 0;
|
||
sheet.for_each_effective_style_producing_rule([&](auto const& rule) {
|
||
size_t selector_index = 0;
|
||
SelectorList const& absolutized_selectors = [&]() {
|
||
if (rule.type() == CSSRule::Type::Style)
|
||
return static_cast<CSSStyleRule const&>(rule).absolutized_selectors();
|
||
if (rule.type() == CSSRule::Type::NestedDeclarations)
|
||
return static_cast<CSSNestedDeclarations const&>(rule).parent_style_rule().absolutized_selectors();
|
||
VERIFY_NOT_REACHED();
|
||
}();
|
||
|
||
for (auto const& selector : absolutized_selectors) {
|
||
m_style_invalidation_data->build_invalidation_sets_for_selector(selector);
|
||
}
|
||
|
||
for (CSS::Selector const& selector : absolutized_selectors) {
|
||
MatchingRule matching_rule {
|
||
shadow_root,
|
||
&rule,
|
||
sheet,
|
||
style_sheet_index,
|
||
rule_index,
|
||
selector_index,
|
||
selector.specificity(),
|
||
cascade_origin,
|
||
false,
|
||
SelectorEngine::can_use_fast_matches(selector),
|
||
false,
|
||
};
|
||
|
||
bool contains_root_pseudo_class = false;
|
||
Optional<CSS::Selector::PseudoElement::Type> pseudo_element;
|
||
|
||
collect_selector_insights(selector, insights);
|
||
|
||
for (auto const& simple_selector : selector.compound_selectors().last().simple_selectors) {
|
||
if (!matching_rule.contains_pseudo_element) {
|
||
if (simple_selector.type == CSS::Selector::SimpleSelector::Type::PseudoElement) {
|
||
matching_rule.contains_pseudo_element = true;
|
||
pseudo_element = simple_selector.pseudo_element().type();
|
||
++num_pseudo_element_rules;
|
||
}
|
||
}
|
||
if (!contains_root_pseudo_class) {
|
||
if (simple_selector.type == CSS::Selector::SimpleSelector::Type::PseudoClass
|
||
&& simple_selector.pseudo_class().type == CSS::PseudoClass::Root) {
|
||
contains_root_pseudo_class = true;
|
||
++num_root_rules;
|
||
}
|
||
}
|
||
|
||
if (!matching_rule.must_be_hovered) {
|
||
if (simple_selector.type == CSS::Selector::SimpleSelector::Type::PseudoClass && simple_selector.pseudo_class().type == CSS::PseudoClass::Hover) {
|
||
matching_rule.must_be_hovered = true;
|
||
++num_hover_rules;
|
||
}
|
||
if (simple_selector.type == CSS::Selector::SimpleSelector::Type::PseudoClass
|
||
&& (simple_selector.pseudo_class().type == CSS::PseudoClass::Is
|
||
|| simple_selector.pseudo_class().type == CSS::PseudoClass::Where)) {
|
||
auto const& argument_selectors = simple_selector.pseudo_class().argument_selector_list;
|
||
|
||
if (argument_selectors.size() == 1) {
|
||
auto const& simple_argument_selector = argument_selectors.first()->compound_selectors().last().simple_selectors.last();
|
||
if (simple_argument_selector.type == CSS::Selector::SimpleSelector::Type::PseudoClass
|
||
&& simple_argument_selector.pseudo_class().type == CSS::PseudoClass::Hover) {
|
||
matching_rule.must_be_hovered = true;
|
||
++num_hover_rules;
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
if (selector.contains_hover_pseudo_class()) {
|
||
hover_rules.append(matching_rule);
|
||
}
|
||
|
||
// NOTE: We traverse the simple selectors in reverse order to make sure that class/ID buckets are preferred over tag buckets
|
||
// in the common case of div.foo or div#foo selectors.
|
||
bool added_to_bucket = false;
|
||
|
||
auto add_to_id_bucket = [&](FlyString const& name) {
|
||
rule_cache->rules_by_id.ensure(name).append(move(matching_rule));
|
||
++num_id_rules;
|
||
added_to_bucket = true;
|
||
};
|
||
|
||
auto add_to_class_bucket = [&](FlyString const& name) {
|
||
rule_cache->rules_by_class.ensure(name).append(move(matching_rule));
|
||
++num_class_rules;
|
||
added_to_bucket = true;
|
||
};
|
||
|
||
auto add_to_tag_name_bucket = [&](FlyString const& name) {
|
||
rule_cache->rules_by_tag_name.ensure(name).append(move(matching_rule));
|
||
++num_tag_name_rules;
|
||
added_to_bucket = true;
|
||
};
|
||
|
||
for (auto const& simple_selector : selector.compound_selectors().last().simple_selectors.in_reverse()) {
|
||
if (simple_selector.type == CSS::Selector::SimpleSelector::Type::Id) {
|
||
add_to_id_bucket(simple_selector.name());
|
||
break;
|
||
}
|
||
if (simple_selector.type == CSS::Selector::SimpleSelector::Type::Class) {
|
||
add_to_class_bucket(simple_selector.name());
|
||
break;
|
||
}
|
||
if (simple_selector.type == CSS::Selector::SimpleSelector::Type::TagName) {
|
||
add_to_tag_name_bucket(simple_selector.qualified_name().name.lowercase_name);
|
||
break;
|
||
}
|
||
// NOTE: Selectors like `:is/where(.foo)` and `:is/where(.foo .bar)` are bucketed as class selectors for `foo` and `bar` respectively.
|
||
if (auto simplified = is_roundabout_selector_bucketable_as_something_simpler(simple_selector); simplified.has_value()) {
|
||
if (simplified->type == CSS::Selector::SimpleSelector::Type::TagName) {
|
||
add_to_tag_name_bucket(simplified->name);
|
||
break;
|
||
}
|
||
if (simplified->type == CSS::Selector::SimpleSelector::Type::Class) {
|
||
add_to_class_bucket(simplified->name);
|
||
break;
|
||
}
|
||
if (simplified->type == CSS::Selector::SimpleSelector::Type::Id) {
|
||
add_to_id_bucket(simplified->name);
|
||
break;
|
||
}
|
||
}
|
||
}
|
||
if (!added_to_bucket) {
|
||
if (matching_rule.contains_pseudo_element) {
|
||
if (CSS::Selector::PseudoElement::is_known_pseudo_element_type(pseudo_element.value())) {
|
||
rule_cache->rules_by_pseudo_element[to_underlying(pseudo_element.value())].append(move(matching_rule));
|
||
} else {
|
||
// NOTE: We don't cache rules for unknown pseudo-elements. They can't match anything anyway.
|
||
}
|
||
} else if (contains_root_pseudo_class) {
|
||
rule_cache->root_rules.append(move(matching_rule));
|
||
} else {
|
||
for (auto const& simple_selector : selector.compound_selectors().last().simple_selectors) {
|
||
if (simple_selector.type == CSS::Selector::SimpleSelector::Type::Attribute) {
|
||
rule_cache->rules_by_attribute_name.ensure(simple_selector.attribute().qualified_name.name.lowercase_name).append(move(matching_rule));
|
||
++num_attribute_rules;
|
||
added_to_bucket = true;
|
||
break;
|
||
}
|
||
}
|
||
if (!added_to_bucket) {
|
||
rule_cache->other_rules.append(move(matching_rule));
|
||
}
|
||
}
|
||
}
|
||
|
||
++selector_index;
|
||
}
|
||
++rule_index;
|
||
});
|
||
|
||
// Loosely based on https://drafts.csswg.org/css-animations-2/#keyframe-processing
|
||
sheet.for_each_effective_keyframes_at_rule([&](CSSKeyframesRule const& rule) {
|
||
auto keyframe_set = adopt_ref(*new Animations::KeyframeEffect::KeyFrameSet);
|
||
HashTable<PropertyID> animated_properties;
|
||
|
||
// Forwards pass, resolve all the user-specified keyframe properties.
|
||
for (auto const& keyframe_rule : *rule.css_rules()) {
|
||
auto const& keyframe = verify_cast<CSSKeyframeRule>(*keyframe_rule);
|
||
Animations::KeyframeEffect::KeyFrameSet::ResolvedKeyFrame resolved_keyframe;
|
||
|
||
auto key = static_cast<u64>(keyframe.key().value() * Animations::KeyframeEffect::AnimationKeyFrameKeyScaleFactor);
|
||
auto const& keyframe_style = *keyframe.style_as_property_owning_style_declaration();
|
||
for (auto const& it : keyframe_style.properties()) {
|
||
// Unresolved properties will be resolved in collect_animation_into()
|
||
for_each_property_expanding_shorthands(it.property_id, it.value, AllowUnresolved::Yes, [&](PropertyID shorthand_id, CSSStyleValue const& shorthand_value) {
|
||
animated_properties.set(shorthand_id);
|
||
resolved_keyframe.properties.set(shorthand_id, NonnullRefPtr<CSSStyleValue const> { shorthand_value });
|
||
});
|
||
}
|
||
|
||
keyframe_set->keyframes_by_key.insert(key, resolved_keyframe);
|
||
}
|
||
|
||
Animations::KeyframeEffect::generate_initial_and_final_frames(keyframe_set, animated_properties);
|
||
|
||
if constexpr (LIBWEB_CSS_DEBUG) {
|
||
dbgln("Resolved keyframe set '{}' into {} keyframes:", rule.name(), keyframe_set->keyframes_by_key.size());
|
||
for (auto it = keyframe_set->keyframes_by_key.begin(); it != keyframe_set->keyframes_by_key.end(); ++it)
|
||
dbgln(" - keyframe {}: {} properties", it.key(), it->properties.size());
|
||
}
|
||
|
||
rule_cache->rules_by_animation_keyframes.set(rule.name(), move(keyframe_set));
|
||
});
|
||
++style_sheet_index;
|
||
});
|
||
|
||
size_t total_rules = num_class_rules + num_id_rules + num_tag_name_rules + num_pseudo_element_rules + num_root_rules + num_attribute_rules + rule_cache->other_rules.size();
|
||
if constexpr (LIBWEB_CSS_DEBUG) {
|
||
dbgln("Built rule cache!");
|
||
dbgln(" ID: {}", num_id_rules);
|
||
dbgln(" Class: {}", num_class_rules);
|
||
dbgln(" TagName: {}", num_tag_name_rules);
|
||
dbgln("PseudoElement: {}", num_pseudo_element_rules);
|
||
dbgln(" Root: {}", num_root_rules);
|
||
dbgln(" Attribute: {}", num_attribute_rules);
|
||
dbgln(" Other: {}", rule_cache->other_rules.size());
|
||
dbgln(" Total: {}", total_rules);
|
||
}
|
||
return rule_cache;
|
||
}
|
||
|
||
struct LayerNode {
|
||
OrderedHashMap<FlyString, LayerNode> children {};
|
||
};
|
||
|
||
static void flatten_layer_names_tree(Vector<FlyString>& layer_names, StringView const& parent_qualified_name, FlyString const& name, LayerNode const& node)
|
||
{
|
||
FlyString qualified_name = parent_qualified_name.is_empty() ? name : MUST(String::formatted("{}.{}", parent_qualified_name, name));
|
||
|
||
for (auto const& item : node.children)
|
||
flatten_layer_names_tree(layer_names, qualified_name, item.key, item.value);
|
||
|
||
layer_names.append(qualified_name);
|
||
}
|
||
|
||
void StyleComputer::build_qualified_layer_names_cache()
|
||
{
|
||
LayerNode root;
|
||
|
||
auto insert_layer_name = [&](FlyString const& internal_qualified_name) {
|
||
auto* node = &root;
|
||
internal_qualified_name.bytes_as_string_view()
|
||
.for_each_split_view('.', SplitBehavior::Nothing, [&](StringView part) {
|
||
auto local_name = MUST(FlyString::from_utf8(part));
|
||
node = &node->children.ensure(local_name);
|
||
});
|
||
};
|
||
|
||
// Walk all style sheets, identifying when we first see a @layer name, and add its qualified name to the list.
|
||
// TODO: Separate the light and shadow-dom layers.
|
||
for_each_stylesheet(CascadeOrigin::Author, [&](auto& sheet, GC::Ptr<DOM::ShadowRoot>) {
|
||
// NOTE: Postorder so that a @layer block is iterated after its children,
|
||
// because we want those children to occur before it in the list.
|
||
sheet.for_each_effective_rule(TraversalOrder::Postorder, [&](auto& rule) {
|
||
switch (rule.type()) {
|
||
case CSSRule::Type::Import:
|
||
// TODO: Handle `layer(foo)` in import rules once we implement that.
|
||
break;
|
||
case CSSRule::Type::LayerBlock: {
|
||
auto& layer_block = static_cast<CSSLayerBlockRule const&>(rule);
|
||
insert_layer_name(layer_block.internal_qualified_name({}));
|
||
break;
|
||
}
|
||
case CSSRule::Type::LayerStatement: {
|
||
auto& layer_statement = static_cast<CSSLayerStatementRule const&>(rule);
|
||
auto qualified_names = layer_statement.internal_qualified_name_list({});
|
||
for (auto& name : qualified_names)
|
||
insert_layer_name(name);
|
||
break;
|
||
}
|
||
|
||
// Ignore everything else
|
||
case CSSRule::Type::Style:
|
||
case CSSRule::Type::Media:
|
||
case CSSRule::Type::FontFace:
|
||
case CSSRule::Type::Keyframes:
|
||
case CSSRule::Type::Keyframe:
|
||
case CSSRule::Type::Namespace:
|
||
case CSSRule::Type::NestedDeclarations:
|
||
case CSSRule::Type::Supports:
|
||
case CSSRule::Type::Property:
|
||
break;
|
||
}
|
||
});
|
||
});
|
||
|
||
// Now, produce a flat list of qualified names to use later
|
||
m_qualified_layer_names_in_order.clear();
|
||
flatten_layer_names_tree(m_qualified_layer_names_in_order, ""sv, {}, root);
|
||
}
|
||
|
||
void StyleComputer::build_rule_cache()
|
||
{
|
||
m_selector_insights = make<SelectorInsights>();
|
||
m_style_invalidation_data = make<StyleInvalidationData>();
|
||
|
||
if (auto user_style_source = document().page().user_style(); user_style_source.has_value()) {
|
||
m_user_style_sheet = GC::make_root(parse_css_stylesheet(CSS::Parser::ParsingContext(document()), user_style_source.value()));
|
||
}
|
||
|
||
build_qualified_layer_names_cache();
|
||
|
||
m_author_rule_cache = make_rule_cache_for_cascade_origin(CascadeOrigin::Author, *m_selector_insights, m_hover_rules);
|
||
m_user_rule_cache = make_rule_cache_for_cascade_origin(CascadeOrigin::User, *m_selector_insights, m_hover_rules);
|
||
m_user_agent_rule_cache = make_rule_cache_for_cascade_origin(CascadeOrigin::UserAgent, *m_selector_insights, m_hover_rules);
|
||
}
|
||
|
||
void StyleComputer::invalidate_rule_cache()
|
||
{
|
||
m_author_rule_cache = nullptr;
|
||
|
||
// NOTE: We could be smarter about keeping the user rule cache, and style sheet.
|
||
// Currently we are re-parsing the user style sheet every time we build the caches,
|
||
// as it may have changed.
|
||
m_user_rule_cache = nullptr;
|
||
m_user_style_sheet = nullptr;
|
||
|
||
// NOTE: It might not be necessary to throw away the UA rule cache.
|
||
// If we are sure that it's safe, we could keep it as an optimization.
|
||
m_user_agent_rule_cache = nullptr;
|
||
|
||
m_hover_rules.clear_with_capacity();
|
||
m_style_invalidation_data = nullptr;
|
||
}
|
||
|
||
void StyleComputer::did_load_font(FlyString const&)
|
||
{
|
||
document().invalidate_style(DOM::StyleInvalidationReason::CSSFontLoaded);
|
||
}
|
||
|
||
Optional<FontLoader&> StyleComputer::load_font_face(ParsedFontFace const& font_face, Function<void(FontLoader const&)> on_load, Function<void()> on_fail)
|
||
{
|
||
if (font_face.sources().is_empty()) {
|
||
if (on_fail)
|
||
on_fail();
|
||
return {};
|
||
}
|
||
|
||
FontFaceKey key {
|
||
.family_name = font_face.font_family(),
|
||
.weight = font_face.weight().value_or(0),
|
||
.slope = font_face.slope().value_or(0),
|
||
};
|
||
|
||
Vector<URL::URL> urls;
|
||
for (auto const& source : font_face.sources()) {
|
||
// FIXME: These should be loaded relative to the stylesheet URL instead of the document URL.
|
||
if (source.local_or_url.has<URL::URL>())
|
||
urls.append(m_document->encoding_parse_url(source.local_or_url.get<URL::URL>().to_string()));
|
||
// FIXME: Handle local()
|
||
}
|
||
|
||
if (urls.is_empty()) {
|
||
if (on_fail)
|
||
on_fail();
|
||
return {};
|
||
}
|
||
|
||
auto loader = make<FontLoader>(const_cast<StyleComputer&>(*this), font_face.font_family(), font_face.unicode_ranges(), move(urls), move(on_load), move(on_fail));
|
||
auto& loader_ref = *loader;
|
||
auto maybe_font_loaders_list = const_cast<StyleComputer&>(*this).m_loaded_fonts.get(key);
|
||
if (maybe_font_loaders_list.has_value()) {
|
||
maybe_font_loaders_list->append(move(loader));
|
||
} else {
|
||
FontLoaderList loaders;
|
||
loaders.append(move(loader));
|
||
const_cast<StyleComputer&>(*this).m_loaded_fonts.set(OwnFontFaceKey(key), move(loaders));
|
||
}
|
||
// Actual object owned by font loader list inside m_loaded_fonts, this isn't use-after-move/free
|
||
return loader_ref;
|
||
}
|
||
|
||
void StyleComputer::load_fonts_from_sheet(CSSStyleSheet& sheet)
|
||
{
|
||
for (auto const& rule : sheet.rules()) {
|
||
if (!is<CSSFontFaceRule>(*rule))
|
||
continue;
|
||
auto font_loader = load_font_face(static_cast<CSSFontFaceRule const&>(*rule).font_face());
|
||
if (font_loader.has_value()) {
|
||
sheet.add_associated_font_loader(font_loader.value());
|
||
}
|
||
}
|
||
}
|
||
|
||
void StyleComputer::unload_fonts_from_sheet(CSSStyleSheet& sheet)
|
||
{
|
||
for (auto& [_, font_loader_list] : m_loaded_fonts) {
|
||
font_loader_list.remove_all_matching([&](auto& font_loader) {
|
||
return sheet.has_associated_font_loader(*font_loader);
|
||
});
|
||
}
|
||
}
|
||
|
||
void StyleComputer::compute_math_depth(ComputedProperties& style, DOM::Element const* element, Optional<CSS::Selector::PseudoElement::Type> pseudo_element) const
|
||
{
|
||
// https://w3c.github.io/mathml-core/#propdef-math-depth
|
||
|
||
// First, ensure that the relevant CSS properties have been defaulted.
|
||
// FIXME: This should be more sophisticated.
|
||
compute_defaulted_property_value(style, element, CSS::PropertyID::MathDepth, pseudo_element);
|
||
compute_defaulted_property_value(style, element, CSS::PropertyID::MathStyle, pseudo_element);
|
||
|
||
auto inherited_math_depth = [&]() {
|
||
if (!element || !element->parent_element())
|
||
return InitialValues::math_depth();
|
||
return element->parent_element()->computed_properties()->math_depth();
|
||
};
|
||
|
||
auto const& value = style.property(CSS::PropertyID::MathDepth);
|
||
if (!value.is_math_depth()) {
|
||
style.set_math_depth(inherited_math_depth());
|
||
return;
|
||
}
|
||
auto const& math_depth = value.as_math_depth();
|
||
|
||
auto resolve_integer = [&](CSSStyleValue const& integer_value) {
|
||
if (integer_value.is_integer())
|
||
return integer_value.as_integer().integer();
|
||
if (integer_value.is_calculated())
|
||
return integer_value.as_calculated().resolve_integer().value();
|
||
VERIFY_NOT_REACHED();
|
||
};
|
||
|
||
// The computed value of the math-depth value is determined as follows:
|
||
// - If the specified value of math-depth is auto-add and the inherited value of math-style is compact
|
||
// then the computed value of math-depth of the element is its inherited value plus one.
|
||
if (math_depth.is_auto_add() && style.property(CSS::PropertyID::MathStyle).to_keyword() == Keyword::Compact) {
|
||
style.set_math_depth(inherited_math_depth() + 1);
|
||
return;
|
||
}
|
||
// - If the specified value of math-depth is of the form add(<integer>) then the computed value of
|
||
// math-depth of the element is its inherited value plus the specified integer.
|
||
if (math_depth.is_add()) {
|
||
style.set_math_depth(inherited_math_depth() + resolve_integer(*math_depth.integer_value()));
|
||
return;
|
||
}
|
||
// - If the specified value of math-depth is of the form <integer> then the computed value of math-depth
|
||
// of the element is the specified integer.
|
||
if (math_depth.is_integer()) {
|
||
style.set_math_depth(resolve_integer(*math_depth.integer_value()));
|
||
return;
|
||
}
|
||
// - Otherwise, the computed value of math-depth of the element is the inherited one.
|
||
style.set_math_depth(inherited_math_depth());
|
||
}
|
||
|
||
static void for_each_element_hash(DOM::Element const& element, auto callback)
|
||
{
|
||
callback(element.local_name().hash());
|
||
if (element.id().has_value())
|
||
callback(element.id().value().hash());
|
||
for (auto const& class_ : element.class_names())
|
||
callback(class_.hash());
|
||
element.for_each_attribute([&](auto& attribute) {
|
||
callback(attribute.local_name().hash());
|
||
});
|
||
}
|
||
|
||
void StyleComputer::reset_ancestor_filter()
|
||
{
|
||
m_ancestor_filter.clear();
|
||
}
|
||
|
||
void StyleComputer::push_ancestor(DOM::Element const& element)
|
||
{
|
||
for_each_element_hash(element, [&](u32 hash) {
|
||
m_ancestor_filter.increment(hash);
|
||
});
|
||
}
|
||
|
||
void StyleComputer::pop_ancestor(DOM::Element const& element)
|
||
{
|
||
for_each_element_hash(element, [&](u32 hash) {
|
||
m_ancestor_filter.decrement(hash);
|
||
});
|
||
}
|
||
|
||
size_t StyleComputer::number_of_css_font_faces_with_loading_in_progress() const
|
||
{
|
||
size_t count = 0;
|
||
for (auto const& [_, loaders] : m_loaded_fonts) {
|
||
for (auto const& loader : loaders) {
|
||
if (loader->is_loading())
|
||
++count;
|
||
}
|
||
}
|
||
return count;
|
||
}
|
||
|
||
bool StyleComputer::has_has_selectors() const
|
||
{
|
||
if (!document().is_active())
|
||
return false;
|
||
|
||
build_rule_cache_if_needed();
|
||
return m_selector_insights->has_has_selectors;
|
||
}
|
||
|
||
bool StyleComputer::has_defined_selectors() const
|
||
{
|
||
if (!document().is_active())
|
||
return false;
|
||
|
||
build_rule_cache_if_needed();
|
||
return m_selector_insights->has_defined_selectors;
|
||
}
|
||
|
||
bool StyleComputer::has_attribute_selector(FlyString const& attribute_name) const
|
||
{
|
||
if (!document().is_active())
|
||
return false;
|
||
|
||
build_rule_cache_if_needed();
|
||
return m_selector_insights->all_names_used_in_attribute_selectors.contains(attribute_name);
|
||
}
|
||
|
||
}
|