1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081 |
- /*
- * Copyright (c) 2018-2020, Andreas Kling <andreas@ladybird.org>
- * Copyright (c) 2021-2023, Sam Atkins <atkinssj@serenityos.org>
- *
- * SPDX-License-Identifier: BSD-2-Clause
- */
- #include "GeneratorUtil.h"
- #include <AK/CharacterTypes.h>
- #include <AK/GenericShorthands.h>
- #include <AK/SourceGenerator.h>
- #include <AK/StringBuilder.h>
- #include <LibCore/ArgsParser.h>
- #include <LibMain/Main.h>
- void replace_logical_aliases(JsonObject& properties);
- ErrorOr<void> generate_header_file(JsonObject& properties, Core::File& file);
- ErrorOr<void> generate_implementation_file(JsonObject& properties, Core::File& file);
- void generate_bounds_checking_function(JsonObject& properties, SourceGenerator& parent_generator, StringView css_type_name, StringView type_name, Optional<StringView> default_unit_name = {}, Optional<StringView> value_getter = {});
- bool is_animatable_property(JsonObject& properties, StringView property_name);
- static bool type_name_is_enum(StringView type_name)
- {
- return !AK::first_is_one_of(type_name,
- "angle"sv,
- "background-position"sv,
- "basic-shape"sv,
- "color"sv,
- "counter"sv,
- "custom-ident"sv,
- "easing-function"sv,
- "flex"sv,
- "frequency"sv,
- "image"sv,
- "integer"sv,
- "length"sv,
- "number"sv,
- "opentype-tag"sv,
- "paint"sv,
- "percentage"sv,
- "position"sv,
- "ratio"sv,
- "rect"sv,
- "resolution"sv,
- "string"sv,
- "time"sv,
- "url"sv);
- }
- static bool is_legacy_alias(JsonObject const& property)
- {
- return property.has_string("legacy-alias-for"sv);
- }
- ErrorOr<int> serenity_main(Main::Arguments arguments)
- {
- StringView generated_header_path;
- StringView generated_implementation_path;
- StringView properties_json_path;
- Core::ArgsParser args_parser;
- args_parser.add_option(generated_header_path, "Path to the PropertyID header file to generate", "generated-header-path", 'h', "generated-header-path");
- args_parser.add_option(generated_implementation_path, "Path to the PropertyID implementation file to generate", "generated-implementation-path", 'c', "generated-implementation-path");
- args_parser.add_option(properties_json_path, "Path to the JSON file to read from", "json-path", 'j', "json-path");
- args_parser.parse(arguments);
- auto json = TRY(read_entire_file_as_json(properties_json_path));
- VERIFY(json.is_object());
- auto properties = json.as_object();
- // Check we're in alphabetical order
- ByteString most_recent_name = "";
- properties.for_each_member([&](auto& name, auto&) {
- if (name < most_recent_name) {
- warnln("`{}` is in the wrong position in `{}`. Please keep this list alphabetical!", name, properties_json_path);
- VERIFY_NOT_REACHED();
- }
- most_recent_name = name;
- });
- replace_logical_aliases(properties);
- auto generated_header_file = TRY(Core::File::open(generated_header_path, Core::File::OpenMode::Write));
- auto generated_implementation_file = TRY(Core::File::open(generated_implementation_path, Core::File::OpenMode::Write));
- TRY(generate_header_file(properties, *generated_header_file));
- TRY(generate_implementation_file(properties, *generated_implementation_file));
- return 0;
- }
- void replace_logical_aliases(JsonObject& properties)
- {
- AK::HashMap<ByteString, ByteString> logical_aliases;
- properties.for_each_member([&](auto& name, auto& value) {
- VERIFY(value.is_object());
- auto const& value_as_object = value.as_object();
- auto const logical_alias_for = value_as_object.get_array("logical-alias-for"sv);
- if (logical_alias_for.has_value()) {
- auto const& aliased_properties = logical_alias_for.value();
- for (auto const& aliased_property : aliased_properties.values()) {
- logical_aliases.set(name, aliased_property.as_string());
- }
- }
- });
- for (auto& [name, alias] : logical_aliases) {
- auto const maybe_alias_object = properties.get_object(alias);
- if (!maybe_alias_object.has_value()) {
- dbgln("No property '{}' found for logical alias '{}'", alias, name);
- VERIFY_NOT_REACHED();
- }
- JsonObject alias_object = maybe_alias_object.value();
- // Copy over anything the logical property overrides
- properties.get_object(name).value().for_each_member([&](auto& key, auto& value) {
- alias_object.set(key, value);
- });
- properties.set(name, alias_object);
- }
- }
- ErrorOr<void> generate_header_file(JsonObject& properties, Core::File& file)
- {
- StringBuilder builder;
- SourceGenerator generator { builder };
- generator.append(R"~~~(
- #pragma once
- #include <AK/NonnullRefPtr.h>
- #include <AK/StringView.h>
- #include <AK/Traits.h>
- #include <LibJS/Forward.h>
- #include <LibWeb/Forward.h>
- namespace Web::CSS {
- enum class PropertyID {
- Invalid,
- Custom,
- All,
- )~~~");
- Vector<ByteString> inherited_shorthand_property_ids;
- Vector<ByteString> inherited_longhand_property_ids;
- Vector<ByteString> noninherited_shorthand_property_ids;
- Vector<ByteString> noninherited_longhand_property_ids;
- properties.for_each_member([&](auto& name, auto& value) {
- VERIFY(value.is_object());
- // Legacy aliases don't get a PropertyID
- if (is_legacy_alias(value.as_object()))
- return;
- bool inherited = value.as_object().get_bool("inherited"sv).value_or(false);
- if (value.as_object().has("longhands"sv)) {
- if (inherited)
- inherited_shorthand_property_ids.append(name);
- else
- noninherited_shorthand_property_ids.append(name);
- } else {
- if (inherited)
- inherited_longhand_property_ids.append(name);
- else
- noninherited_longhand_property_ids.append(name);
- }
- });
- // Section order:
- // 1. inherited shorthand properties
- // 2. noninherited shorthand properties
- // 3. inherited longhand properties
- // 4. noninherited longhand properties
- auto first_property_id = inherited_shorthand_property_ids.first();
- auto last_property_id = noninherited_longhand_property_ids.last();
- auto emit_properties = [&](auto& property_ids) {
- for (auto& name : property_ids) {
- auto member_generator = generator.fork();
- member_generator.set("name:titlecase", title_casify(name));
- member_generator.append(R"~~~(
- @name:titlecase@,
- )~~~");
- }
- };
- emit_properties(inherited_shorthand_property_ids);
- emit_properties(noninherited_shorthand_property_ids);
- emit_properties(inherited_longhand_property_ids);
- emit_properties(noninherited_longhand_property_ids);
- generator.set("first_property_id", title_casify(first_property_id));
- generator.set("last_property_id", title_casify(last_property_id));
- generator.set("first_longhand_property_id", title_casify(inherited_longhand_property_ids.first()));
- generator.set("last_longhand_property_id", title_casify(noninherited_longhand_property_ids.last()));
- generator.set("first_inherited_shorthand_property_id", title_casify(inherited_shorthand_property_ids.first()));
- generator.set("last_inherited_shorthand_property_id", title_casify(inherited_shorthand_property_ids.last()));
- generator.set("first_inherited_longhand_property_id", title_casify(inherited_longhand_property_ids.first()));
- generator.set("last_inherited_longhand_property_id", title_casify(inherited_longhand_property_ids.last()));
- generator.append(R"~~~(
- };
- enum class AnimationType {
- Discrete,
- ByComputedValue,
- RepeatableList,
- Custom,
- None,
- };
- AnimationType animation_type_from_longhand_property(PropertyID);
- bool is_animatable_property(PropertyID);
- Optional<PropertyID> property_id_from_camel_case_string(StringView);
- Optional<PropertyID> property_id_from_string(StringView);
- [[nodiscard]] FlyString const& string_from_property_id(PropertyID);
- [[nodiscard]] FlyString const& camel_case_string_from_property_id(PropertyID);
- bool is_inherited_property(PropertyID);
- NonnullRefPtr<CSSStyleValue> property_initial_value(Optional<JS::Realm&>, PropertyID);
- enum class ValueType {
- Angle,
- BackgroundPosition,
- BasicShape,
- Color,
- Counter,
- CustomIdent,
- EasingFunction,
- FilterValueList,
- Flex,
- Frequency,
- Image,
- Integer,
- Length,
- Number,
- OpenTypeTag,
- Paint,
- Percentage,
- Position,
- Ratio,
- Rect,
- Resolution,
- String,
- Time,
- Url,
- };
- bool property_accepts_type(PropertyID, ValueType);
- bool property_accepts_keyword(PropertyID, Keyword);
- Optional<ValueType> property_resolves_percentages_relative_to(PropertyID);
- // These perform range-checking, but are also safe to call with properties that don't accept that type. (They'll just return false.)
- bool property_accepts_angle(PropertyID, Angle const&);
- bool property_accepts_flex(PropertyID, Flex const&);
- bool property_accepts_frequency(PropertyID, Frequency const&);
- bool property_accepts_integer(PropertyID, i64 const&);
- bool property_accepts_length(PropertyID, Length const&);
- bool property_accepts_number(PropertyID, double const&);
- bool property_accepts_percentage(PropertyID, Percentage const&);
- bool property_accepts_resolution(PropertyID, Resolution const&);
- bool property_accepts_time(PropertyID, Time const&);
- bool property_is_shorthand(PropertyID);
- Vector<PropertyID> longhands_for_shorthand(PropertyID);
- size_t property_maximum_value_count(PropertyID);
- bool property_affects_layout(PropertyID);
- bool property_affects_stacking_context(PropertyID);
- constexpr PropertyID first_property_id = PropertyID::@first_property_id@;
- constexpr PropertyID last_property_id = PropertyID::@last_property_id@;
- constexpr PropertyID first_inherited_shorthand_property_id = PropertyID::@first_inherited_shorthand_property_id@;
- constexpr PropertyID last_inherited_shorthand_property_id = PropertyID::@last_inherited_shorthand_property_id@;
- constexpr PropertyID first_inherited_longhand_property_id = PropertyID::@first_inherited_longhand_property_id@;
- constexpr PropertyID last_inherited_longhand_property_id = PropertyID::@last_inherited_longhand_property_id@;
- constexpr PropertyID first_longhand_property_id = PropertyID::@first_longhand_property_id@;
- constexpr PropertyID last_longhand_property_id = PropertyID::@last_longhand_property_id@;
- enum class Quirk {
- // https://quirks.spec.whatwg.org/#the-hashless-hex-color-quirk
- HashlessHexColor,
- // https://quirks.spec.whatwg.org/#the-unitless-length-quirk
- UnitlessLength,
- };
- bool property_has_quirk(PropertyID, Quirk);
- } // namespace Web::CSS
- namespace AK {
- template<>
- struct Traits<Web::CSS::PropertyID> : public DefaultTraits<Web::CSS::PropertyID> {
- static unsigned hash(Web::CSS::PropertyID property_id) { return int_hash((unsigned)property_id); }
- };
- } // namespace AK
- )~~~");
- TRY(file.write_until_depleted(generator.as_string_view().bytes()));
- return {};
- }
- void generate_bounds_checking_function(JsonObject& properties, SourceGenerator& parent_generator, StringView css_type_name, StringView type_name, Optional<StringView> default_unit_name, Optional<StringView> value_getter)
- {
- auto generator = parent_generator.fork();
- generator.set("css_type_name", css_type_name);
- generator.set("type_name", type_name);
- generator.append(R"~~~(
- bool property_accepts_@css_type_name@(PropertyID property_id, [[maybe_unused]] @type_name@ const& value)
- {
- switch (property_id) {
- )~~~");
- properties.for_each_member([&](auto& name, JsonValue const& value) -> void {
- VERIFY(value.is_object());
- if (is_legacy_alias(value.as_object()))
- return;
- if (auto maybe_valid_types = value.as_object().get_array("valid-types"sv); maybe_valid_types.has_value() && !maybe_valid_types->is_empty()) {
- for (auto valid_type : maybe_valid_types->values()) {
- auto type_and_range = valid_type.as_string().split_view(' ');
- if (type_and_range.first() != css_type_name)
- continue;
- auto property_generator = generator.fork();
- property_generator.set("property_name:titlecase", title_casify(name));
- property_generator.append(R"~~~(
- case PropertyID::@property_name:titlecase@:
- return )~~~");
- if (type_and_range.size() > 1) {
- auto range = type_and_range[1];
- VERIFY(range.starts_with('[') && range.ends_with(']') && range.contains(','));
- auto comma_index = range.find(',').value();
- StringView min_value_string = range.substring_view(1, comma_index - 1);
- StringView max_value_string = range.substring_view(comma_index + 1, range.length() - comma_index - 2);
- // If the min/max value is infinite, we can just skip that side of the check.
- if (min_value_string == "-∞")
- min_value_string = {};
- if (max_value_string == "∞")
- max_value_string = {};
- if (min_value_string.is_empty() && max_value_string.is_empty()) {
- property_generator.appendln("true;");
- break;
- }
- auto output_check = [&](auto& value_string, StringView comparator) {
- if (value_getter.has_value()) {
- property_generator.set("value_number", value_string);
- property_generator.set("value_getter", value_getter.value());
- property_generator.set("comparator", comparator);
- property_generator.append("@value_getter@ @comparator@ @value_number@");
- return;
- }
- GenericLexer lexer { value_string };
- auto value_number = lexer.consume_until(is_ascii_alpha);
- auto value_unit = lexer.consume_while(is_ascii_alpha);
- if (value_unit.is_empty())
- value_unit = default_unit_name.value();
- VERIFY(lexer.is_eof());
- property_generator.set("value_number", value_number);
- property_generator.set("value_unit", title_casify(value_unit));
- property_generator.set("comparator", comparator);
- property_generator.append("value @comparator@ @type_name@(@value_number@, @type_name@::Type::@value_unit@)");
- };
- if (!min_value_string.is_empty())
- output_check(min_value_string, ">="sv);
- if (!min_value_string.is_empty() && !max_value_string.is_empty())
- property_generator.append(" && ");
- if (!max_value_string.is_empty())
- output_check(max_value_string, "<="sv);
- property_generator.appendln(";");
- } else {
- property_generator.appendln("true;");
- }
- break;
- }
- }
- });
- generator.append(R"~~~(
- default:
- return false;
- }
- }
- )~~~");
- }
- ErrorOr<void> generate_implementation_file(JsonObject& properties, Core::File& file)
- {
- StringBuilder builder;
- SourceGenerator generator { builder };
- generator.append(R"~~~(
- #include <AK/Assertions.h>
- #include <LibWeb/CSS/Enums.h>
- #include <LibWeb/CSS/Parser/Parser.h>
- #include <LibWeb/CSS/PropertyID.h>
- #include <LibWeb/CSS/PropertyName.h>
- #include <LibWeb/CSS/CSSStyleValue.h>
- #include <LibWeb/CSS/StyleValues/PercentageStyleValue.h>
- #include <LibWeb/CSS/StyleValues/TimeStyleValue.h>
- #include <LibWeb/Infra/Strings.h>
- namespace Web::CSS {
- Optional<PropertyID> property_id_from_camel_case_string(StringView string)
- {
- )~~~");
- properties.for_each_member([&](auto& name, auto& value) {
- VERIFY(value.is_object());
- auto member_generator = generator.fork();
- member_generator.set("name", name);
- member_generator.set("name:camelcase", camel_casify(name));
- if (auto legacy_alias_for = value.as_object().get_byte_string("legacy-alias-for"sv); legacy_alias_for.has_value()) {
- member_generator.set("name:titlecase", title_casify(legacy_alias_for.value()));
- } else {
- member_generator.set("name:titlecase", title_casify(name));
- }
- member_generator.append(R"~~~(
- if (string.equals_ignoring_ascii_case("@name:camelcase@"sv))
- return PropertyID::@name:titlecase@;
- )~~~");
- });
- generator.append(R"~~~(
- return {};
- }
- Optional<PropertyID> property_id_from_string(StringView string)
- {
- if (is_a_custom_property_name_string(string))
- return PropertyID::Custom;
- if (Infra::is_ascii_case_insensitive_match(string, "all"sv))
- return PropertyID::All;
- )~~~");
- properties.for_each_member([&](auto& name, auto& value) {
- VERIFY(value.is_object());
- auto member_generator = generator.fork();
- member_generator.set("name", name);
- if (auto legacy_alias_for = value.as_object().get_byte_string("legacy-alias-for"sv); legacy_alias_for.has_value()) {
- member_generator.set("name:titlecase", title_casify(legacy_alias_for.value()));
- } else {
- member_generator.set("name:titlecase", title_casify(name));
- }
- member_generator.append(R"~~~(
- if (Infra::is_ascii_case_insensitive_match(string, "@name@"sv))
- return PropertyID::@name:titlecase@;
- )~~~");
- });
- generator.append(R"~~~(
- return {};
- }
- FlyString const& string_from_property_id(PropertyID property_id) {
- switch (property_id) {
- )~~~");
- properties.for_each_member([&](auto& name, auto& value) {
- VERIFY(value.is_object());
- if (is_legacy_alias(value.as_object()))
- return;
- auto member_generator = generator.fork();
- member_generator.set("name", name);
- member_generator.set("name:titlecase", title_casify(name));
- member_generator.append(R"~~~(
- case PropertyID::@name:titlecase@: {
- static FlyString name = "@name@"_fly_string;
- return name;
- }
- )~~~");
- });
- generator.append(R"~~~(
- default: {
- static FlyString invalid_property_id_string = "(invalid CSS::PropertyID)"_fly_string;
- return invalid_property_id_string;
- }
- }
- }
- FlyString const& camel_case_string_from_property_id(PropertyID property_id) {
- switch (property_id) {
- )~~~");
- properties.for_each_member([&](auto& name, auto& value) {
- VERIFY(value.is_object());
- if (is_legacy_alias(value.as_object()))
- return;
- auto member_generator = generator.fork();
- member_generator.set("name", name);
- member_generator.set("name:titlecase", title_casify(name));
- member_generator.set("name:camelcase", camel_casify(name));
- member_generator.append(R"~~~(
- case PropertyID::@name:titlecase@: {
- static FlyString name = "@name:camelcase@"_fly_string;
- return name;
- }
- )~~~");
- });
- generator.append(R"~~~(
- default: {
- static FlyString invalid_property_id_string = "(invalid CSS::PropertyID)"_fly_string;
- return invalid_property_id_string;
- }
- }
- }
- AnimationType animation_type_from_longhand_property(PropertyID property_id)
- {
- switch (property_id) {
- )~~~");
- properties.for_each_member([&](auto& name, auto& value) {
- VERIFY(value.is_object());
- if (is_legacy_alias(value.as_object()))
- return;
- auto member_generator = generator.fork();
- member_generator.set("name:titlecase", title_casify(name));
- // Shorthand properties should have already been expanded before calling into this function
- if (value.as_object().has("longhands"sv)) {
- if (value.as_object().has("animation-type"sv)) {
- dbgln("Property '{}' with longhands cannot specify 'animation-type'", name);
- VERIFY_NOT_REACHED();
- }
- member_generator.append(R"~~~(
- case PropertyID::@name:titlecase@:
- VERIFY_NOT_REACHED();
- )~~~");
- return;
- }
- if (!value.as_object().has("animation-type"sv)) {
- dbgln("No animation-type specified for property '{}'", name);
- VERIFY_NOT_REACHED();
- }
- auto animation_type = value.as_object().get_byte_string("animation-type"sv).value();
- member_generator.set("value", title_casify(animation_type));
- member_generator.append(R"~~~(
- case PropertyID::@name:titlecase@:
- return AnimationType::@value@;
- )~~~");
- });
- generator.append(R"~~~(
- default:
- return AnimationType::None;
- }
- }
- bool is_animatable_property(PropertyID property_id)
- {
- switch (property_id) {
- )~~~");
- properties.for_each_member([&](auto& name, auto& value) {
- VERIFY(value.is_object());
- VERIFY(!name.is_empty() && !is_ascii_digit(name[0])); // Ensure `PropertyKey`s are not Numbers.
- if (is_legacy_alias(value.as_object()))
- return;
- if (is_animatable_property(properties, name)) {
- auto member_generator = generator.fork();
- member_generator.set("name:titlecase", title_casify(name));
- member_generator.append(R"~~~(
- case PropertyID::@name:titlecase@:
- )~~~");
- }
- });
- generator.append(R"~~~(
- return true;
- default:
- return false;
- }
- }
- bool is_inherited_property(PropertyID property_id)
- {
- if (property_id >= first_inherited_shorthand_property_id && property_id <= last_inherited_longhand_property_id)
- return true;
- if (property_id >= first_inherited_longhand_property_id && property_id <= last_inherited_longhand_property_id)
- return true;
- return false;
- }
- bool property_affects_layout(PropertyID property_id)
- {
- switch (property_id) {
- )~~~");
- properties.for_each_member([&](auto& name, auto& value) {
- VERIFY(value.is_object());
- if (is_legacy_alias(value.as_object()))
- return;
- bool affects_layout = true;
- if (value.as_object().has("affects-layout"sv))
- affects_layout = value.as_object().get_bool("affects-layout"sv).value_or(false);
- if (affects_layout) {
- auto member_generator = generator.fork();
- member_generator.set("name:titlecase", title_casify(name));
- member_generator.append(R"~~~(
- case PropertyID::@name:titlecase@:
- )~~~");
- }
- });
- generator.append(R"~~~(
- return true;
- default:
- return false;
- }
- }
- bool property_affects_stacking_context(PropertyID property_id)
- {
- switch (property_id) {
- )~~~");
- properties.for_each_member([&](auto& name, auto& value) {
- VERIFY(value.is_object());
- if (is_legacy_alias(value.as_object()))
- return;
- bool affects_stacking_context = false;
- if (value.as_object().has("affects-stacking-context"sv))
- affects_stacking_context = value.as_object().get_bool("affects-stacking-context"sv).value_or(false);
- if (affects_stacking_context) {
- auto member_generator = generator.fork();
- member_generator.set("name:titlecase", title_casify(name));
- member_generator.append(R"~~~(
- case PropertyID::@name:titlecase@:
- )~~~");
- }
- });
- generator.append(R"~~~(
- return true;
- default:
- return false;
- }
- }
- NonnullRefPtr<CSSStyleValue> property_initial_value(Optional<JS::Realm&> context_realm, PropertyID property_id)
- {
- static Array<RefPtr<CSSStyleValue>, to_underlying(last_property_id) + 1> initial_values;
- if (auto initial_value = initial_values[to_underlying(property_id)])
- return initial_value.release_nonnull();
- // We need a Realm to parse any new values.
- VERIFY(context_realm.has_value());
- // Lazily parse initial values as needed.
- // This ensures the shorthands will always be able to get the initial values of their longhands.
- // This also now allows a longhand have its own longhand (like background-position-x).
- Parser::ParsingContext parsing_context(context_realm.value());
- switch (property_id) {
- )~~~");
- auto output_initial_value_code = [&](auto& name, auto& object) {
- if (!object.has("initial"sv)) {
- dbgln("No initial value specified for property '{}'", name);
- VERIFY_NOT_REACHED();
- }
- auto initial_value = object.get_byte_string("initial"sv);
- VERIFY(initial_value.has_value());
- auto& initial_value_string = initial_value.value();
- auto member_generator = generator.fork();
- member_generator.set("name:titlecase", title_casify(name));
- member_generator.set("initial_value_string", initial_value_string);
- member_generator.append(
- R"~~~( case PropertyID::@name:titlecase@:
- {
- auto parsed_value = parse_css_value(parsing_context, "@initial_value_string@"sv, PropertyID::@name:titlecase@);
- VERIFY(!parsed_value.is_null());
- auto initial_value = parsed_value.release_nonnull();
- initial_values[to_underlying(PropertyID::@name:titlecase@)] = initial_value;
- return initial_value;
- }
- )~~~");
- };
- properties.for_each_member([&](auto& name, auto& value) {
- VERIFY(value.is_object());
- if (is_legacy_alias(value.as_object()))
- return;
- output_initial_value_code(name, value.as_object());
- });
- generator.append(
- R"~~~( default: VERIFY_NOT_REACHED();
- }
- VERIFY_NOT_REACHED();
- }
- bool property_has_quirk(PropertyID property_id, Quirk quirk)
- {
- switch (property_id) {
- )~~~");
- properties.for_each_member([&](auto& name, auto& value) {
- VERIFY(value.is_object());
- if (is_legacy_alias(value.as_object()))
- return;
- if (value.as_object().has("quirks"sv)) {
- auto quirks_value = value.as_object().get_array("quirks"sv);
- VERIFY(quirks_value.has_value());
- auto& quirks = quirks_value.value();
- if (!quirks.is_empty()) {
- auto property_generator = generator.fork();
- property_generator.set("name:titlecase", title_casify(name));
- property_generator.append(R"~~~(
- case PropertyID::@name:titlecase@: {
- switch (quirk) {
- )~~~");
- for (auto& quirk : quirks.values()) {
- VERIFY(quirk.is_string());
- auto quirk_generator = property_generator.fork();
- quirk_generator.set("quirk:titlecase", title_casify(quirk.as_string()));
- quirk_generator.append(R"~~~(
- case Quirk::@quirk:titlecase@:
- return true;
- )~~~");
- }
- property_generator.append(R"~~~(
- default:
- return false;
- }
- }
- )~~~");
- }
- }
- });
- generator.append(R"~~~(
- default:
- return false;
- }
- }
- bool property_accepts_type(PropertyID property_id, ValueType value_type)
- {
- switch (property_id) {
- )~~~");
- properties.for_each_member([&](auto& name, auto& value) {
- VERIFY(value.is_object());
- auto& object = value.as_object();
- if (is_legacy_alias(object))
- return;
- if (auto maybe_valid_types = object.get_array("valid-types"sv); maybe_valid_types.has_value() && !maybe_valid_types->is_empty()) {
- auto& valid_types = maybe_valid_types.value();
- auto property_generator = generator.fork();
- property_generator.set("name:titlecase", title_casify(name));
- property_generator.append(R"~~~(
- case PropertyID::@name:titlecase@: {
- switch (value_type) {
- )~~~");
- bool did_output_accepted_type = false;
- for (auto& type : valid_types.values()) {
- VERIFY(type.is_string());
- auto type_name = type.as_string().split_view(' ').first();
- if (type_name_is_enum(type_name))
- continue;
- if (type_name == "angle") {
- property_generator.appendln(" case ValueType::Angle:");
- } else if (type_name == "background-position") {
- property_generator.appendln(" case ValueType::BackgroundPosition:");
- } else if (type_name == "basic-shape") {
- property_generator.appendln(" case ValueType::BasicShape:");
- } else if (type_name == "color") {
- property_generator.appendln(" case ValueType::Color:");
- } else if (type_name == "counter") {
- property_generator.appendln(" case ValueType::Counter:");
- } else if (type_name == "custom-ident") {
- property_generator.appendln(" case ValueType::CustomIdent:");
- } else if (type_name == "easing-function") {
- property_generator.appendln(" case ValueType::EasingFunction:");
- } else if (type_name == "flex") {
- property_generator.appendln(" case ValueType::Flex:");
- } else if (type_name == "frequency") {
- property_generator.appendln(" case ValueType::Frequency:");
- } else if (type_name == "image") {
- property_generator.appendln(" case ValueType::Image:");
- } else if (type_name == "integer") {
- property_generator.appendln(" case ValueType::Integer:");
- } else if (type_name == "length") {
- property_generator.appendln(" case ValueType::Length:");
- } else if (type_name == "number") {
- property_generator.appendln(" case ValueType::Number:");
- } else if (type_name == "opentype-tag") {
- property_generator.appendln(" case ValueType::OpenTypeTag:");
- } else if (type_name == "paint") {
- property_generator.appendln(" case ValueType::Paint:");
- } else if (type_name == "percentage") {
- property_generator.appendln(" case ValueType::Percentage:");
- } else if (type_name == "position") {
- property_generator.appendln(" case ValueType::Position:");
- } else if (type_name == "ratio") {
- property_generator.appendln(" case ValueType::Ratio:");
- } else if (type_name == "rect") {
- property_generator.appendln(" case ValueType::Rect:");
- } else if (type_name == "resolution") {
- property_generator.appendln(" case ValueType::Resolution:");
- } else if (type_name == "string") {
- property_generator.appendln(" case ValueType::String:");
- } else if (type_name == "time") {
- property_generator.appendln(" case ValueType::Time:");
- } else if (type_name == "url") {
- property_generator.appendln(" case ValueType::Url:");
- } else {
- VERIFY_NOT_REACHED();
- }
- did_output_accepted_type = true;
- }
- if (did_output_accepted_type)
- property_generator.appendln(" return true;");
- property_generator.append(R"~~~(
- default:
- return false;
- }
- }
- )~~~");
- }
- });
- generator.append(R"~~~(
- default:
- return false;
- }
- }
- bool property_accepts_keyword(PropertyID property_id, Keyword keyword)
- {
- switch (property_id) {
- )~~~");
- properties.for_each_member([&](auto& name, auto& value) {
- VERIFY(value.is_object());
- auto& object = value.as_object();
- if (is_legacy_alias(object))
- return;
- auto property_generator = generator.fork();
- property_generator.set("name:titlecase", title_casify(name));
- property_generator.appendln(" case PropertyID::@name:titlecase@: {");
- if (auto maybe_valid_identifiers = object.get_array("valid-identifiers"sv); maybe_valid_identifiers.has_value() && !maybe_valid_identifiers->is_empty()) {
- property_generator.appendln(" switch (keyword) {");
- auto& valid_identifiers = maybe_valid_identifiers.value();
- for (auto& keyword : valid_identifiers.values()) {
- auto keyword_generator = generator.fork();
- keyword_generator.set("keyword:titlecase", title_casify(keyword.as_string()));
- keyword_generator.appendln(" case Keyword::@keyword:titlecase@:");
- }
- property_generator.append(R"~~~(
- return true;
- default:
- break;
- }
- )~~~");
- }
- if (auto maybe_valid_types = object.get_array("valid-types"sv); maybe_valid_types.has_value() && !maybe_valid_types->is_empty()) {
- auto& valid_types = maybe_valid_types.value();
- for (auto& valid_type : valid_types.values()) {
- auto type_name = valid_type.as_string().split_view(' ').first();
- if (!type_name_is_enum(type_name))
- continue;
- auto type_generator = generator.fork();
- type_generator.set("type_name:snakecase", snake_casify(type_name));
- type_generator.append(R"~~~(
- if (keyword_to_@type_name:snakecase@(keyword).has_value())
- return true;
- )~~~");
- }
- }
- property_generator.append(R"~~~(
- return false;
- }
- )~~~");
- });
- generator.append(R"~~~(
- default:
- return false;
- }
- }
- Optional<ValueType> property_resolves_percentages_relative_to(PropertyID property_id)
- {
- switch (property_id) {
- )~~~");
- properties.for_each_member([&](auto& name, auto& value) {
- VERIFY(value.is_object());
- if (is_legacy_alias(value.as_object()))
- return;
- if (auto resolved_type = value.as_object().get_byte_string("percentages-resolve-to"sv); resolved_type.has_value()) {
- auto property_generator = generator.fork();
- property_generator.set("name:titlecase", title_casify(name));
- property_generator.set("resolved_type:titlecase", title_casify(resolved_type.value()));
- property_generator.append(R"~~~(
- case PropertyID::@name:titlecase@:
- return ValueType::@resolved_type:titlecase@;
- )~~~");
- }
- });
- generator.append(R"~~~(
- default:
- return {};
- }
- }
- size_t property_maximum_value_count(PropertyID property_id)
- {
- switch (property_id) {
- )~~~");
- properties.for_each_member([&](auto& name, auto& value) {
- VERIFY(value.is_object());
- if (is_legacy_alias(value.as_object()))
- return;
- if (value.as_object().has("max-values"sv)) {
- JsonValue max_values = value.as_object().get("max-values"sv).release_value();
- VERIFY(max_values.is_integer<size_t>());
- auto property_generator = generator.fork();
- property_generator.set("name:titlecase", title_casify(name));
- property_generator.set("max_values", MUST(String::formatted("{}", max_values.as_integer<size_t>())));
- property_generator.append(R"~~~(
- case PropertyID::@name:titlecase@:
- return @max_values@;
- )~~~");
- }
- });
- generator.append(R"~~~(
- default:
- return 1;
- }
- })~~~");
- generate_bounds_checking_function(properties, generator, "angle"sv, "Angle"sv, "Deg"sv);
- generate_bounds_checking_function(properties, generator, "flex"sv, "Flex"sv, "Fr"sv);
- generate_bounds_checking_function(properties, generator, "frequency"sv, "Frequency"sv, "Hertz"sv);
- generate_bounds_checking_function(properties, generator, "integer"sv, "i64"sv, {}, "value"sv);
- generate_bounds_checking_function(properties, generator, "length"sv, "Length"sv, {}, "value.raw_value()"sv);
- generate_bounds_checking_function(properties, generator, "number"sv, "double"sv, {}, "value"sv);
- generate_bounds_checking_function(properties, generator, "percentage"sv, "Percentage"sv, {}, "value.value()"sv);
- generate_bounds_checking_function(properties, generator, "resolution"sv, "Resolution"sv, "Dpi"sv);
- generate_bounds_checking_function(properties, generator, "time"sv, "Time"sv, "S"sv);
- generator.append(R"~~~(
- bool property_is_shorthand(PropertyID property_id)
- {
- switch (property_id) {
- )~~~");
- properties.for_each_member([&](auto& name, auto& value) {
- if (is_legacy_alias(value.as_object()))
- return;
- if (value.as_object().has("longhands"sv)) {
- auto property_generator = generator.fork();
- property_generator.set("name:titlecase", title_casify(name));
- property_generator.append(R"~~~(
- case PropertyID::@name:titlecase@:
- )~~~");
- }
- });
- generator.append(R"~~~(
- return true;
- default:
- return false;
- }
- }
- )~~~");
- generator.append(R"~~~(
- Vector<PropertyID> longhands_for_shorthand(PropertyID property_id)
- {
- switch (property_id) {
- )~~~");
- properties.for_each_member([&](auto& name, auto& value) {
- if (is_legacy_alias(value.as_object()))
- return;
- if (value.as_object().has("longhands"sv)) {
- auto longhands = value.as_object().get("longhands"sv);
- VERIFY(longhands.has_value() && longhands->is_array());
- auto longhand_values = longhands->as_array();
- auto property_generator = generator.fork();
- property_generator.set("name:titlecase", title_casify(name));
- StringBuilder builder;
- bool first = true;
- longhand_values.for_each([&](auto& longhand) {
- if (first)
- first = false;
- else
- builder.append(", "sv);
- builder.appendff("PropertyID::{}", title_casify(longhand.as_string()));
- return IterationDecision::Continue;
- });
- property_generator.set("longhands", builder.to_byte_string());
- property_generator.append(R"~~~(
- case PropertyID::@name:titlecase@:
- return { @longhands@ };
- )~~~");
- }
- });
- generator.append(R"~~~(
- default:
- return { };
- }
- }
- )~~~");
- generator.append(R"~~~(
- } // namespace Web::CSS
- )~~~");
- TRY(file.write_until_depleted(generator.as_string_view().bytes()));
- return {};
- }
- bool is_animatable_property(JsonObject& properties, StringView property_name)
- {
- auto property = properties.get_object(property_name);
- VERIFY(property.has_value());
- if (auto animation_type = property.value().get_byte_string("animation-type"sv); animation_type.has_value()) {
- return animation_type != "none";
- }
- if (!property.value().has("longhands"sv)) {
- dbgln("Property '{}' must specify either 'animation-type' or 'longhands'"sv, property_name);
- VERIFY_NOT_REACHED();
- }
- auto longhands = property.value().get_array("longhands"sv);
- VERIFY(longhands.has_value());
- for (auto const& subproperty_name : longhands->values()) {
- VERIFY(subproperty_name.is_string());
- if (is_animatable_property(properties, subproperty_name.as_string()))
- return true;
- }
- return false;
- }
|