mirror of
https://github.com/LadybirdBrowser/ladybird.git
synced 2025-04-25 14:05:15 +00:00
592 lines
20 KiB
C++
592 lines
20 KiB
C++
/*
|
|
* Copyright (c) 2018-2020, Andreas Kling <kling@serenityos.org>
|
|
* Copyright (c) 2021-2023, Sam Atkins <atkinssj@serenityos.org>
|
|
*
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
*/
|
|
|
|
#include "GeneratorUtil.h"
|
|
#include <AK/GenericShorthands.h>
|
|
#include <AK/SourceGenerator.h>
|
|
#include <AK/StringBuilder.h>
|
|
#include <LibCore/ArgsParser.h>
|
|
#include <LibMain/Main.h>
|
|
|
|
ErrorOr<void> generate_header_file(JsonObject& properties, Core::File& file);
|
|
ErrorOr<void> generate_implementation_file(JsonObject& properties, Core::File& file);
|
|
|
|
static bool type_name_is_enum(StringView type_name)
|
|
{
|
|
return !AK::first_is_one_of(type_name, "angle"sv, "color"sv, "custom-ident"sv, "frequency"sv, "image"sv, "integer"sv, "length"sv, "number"sv, "percentage"sv, "rect"sv, "resolution"sv, "string"sv, "time"sv, "url"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();
|
|
|
|
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;
|
|
}
|
|
|
|
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,
|
|
)~~~");
|
|
|
|
Vector<DeprecatedString> shorthand_property_ids;
|
|
Vector<DeprecatedString> longhand_property_ids;
|
|
|
|
properties.for_each_member([&](auto& name, auto& value) {
|
|
VERIFY(value.is_object());
|
|
if (value.as_object().has("longhands"sv))
|
|
shorthand_property_ids.append(name);
|
|
else
|
|
longhand_property_ids.append(name);
|
|
});
|
|
|
|
auto first_property_id = shorthand_property_ids.first();
|
|
auto last_property_id = longhand_property_ids.last();
|
|
|
|
for (auto& name : shorthand_property_ids) {
|
|
auto member_generator = generator.fork();
|
|
member_generator.set("name:titlecase", title_casify(name));
|
|
|
|
member_generator.append(R"~~~(
|
|
@name:titlecase@,
|
|
)~~~");
|
|
}
|
|
|
|
for (auto& name : longhand_property_ids) {
|
|
auto member_generator = generator.fork();
|
|
member_generator.set("name:titlecase", title_casify(name));
|
|
|
|
member_generator.append(R"~~~(
|
|
@name:titlecase@,
|
|
)~~~");
|
|
}
|
|
|
|
generator.set("first_property_id", title_casify(first_property_id));
|
|
generator.set("last_property_id", title_casify(last_property_id));
|
|
|
|
generator.set("first_shorthand_property_id", title_casify(shorthand_property_ids.first()));
|
|
generator.set("last_shorthand_property_id", title_casify(shorthand_property_ids.last()));
|
|
|
|
generator.set("first_longhand_property_id", title_casify(longhand_property_ids.first()));
|
|
generator.set("last_longhand_property_id", title_casify(longhand_property_ids.last()));
|
|
|
|
generator.append(R"~~~(
|
|
};
|
|
|
|
Optional<PropertyID> property_id_from_camel_case_string(StringView);
|
|
Optional<PropertyID> property_id_from_string(StringView);
|
|
StringView string_from_property_id(PropertyID);
|
|
bool is_inherited_property(PropertyID);
|
|
ErrorOr<NonnullRefPtr<StyleValue>> property_initial_value(JS::Realm&, PropertyID);
|
|
|
|
enum class ValueType {
|
|
Angle,
|
|
Color,
|
|
CustomIdent,
|
|
FilterValueList,
|
|
Frequency,
|
|
Image,
|
|
Integer,
|
|
Length,
|
|
Number,
|
|
Paint,
|
|
Percentage,
|
|
Position,
|
|
Rect,
|
|
Resolution,
|
|
String,
|
|
Time,
|
|
Url,
|
|
};
|
|
bool property_accepts_type(PropertyID, ValueType);
|
|
bool property_accepts_identifier(PropertyID, ValueID);
|
|
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_shorthand_property_id = PropertyID::@first_shorthand_property_id@;
|
|
constexpr PropertyID last_shorthand_property_id = PropertyID::@last_shorthand_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 GenericTraits<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 {};
|
|
}
|
|
|
|
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/StyleValue.h>
|
|
#include <LibWeb/CSS/StyleValues/PercentageStyleValue.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:titlecase", title_casify(name));
|
|
member_generator.set("name:camelcase", camel_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)
|
|
{
|
|
)~~~");
|
|
|
|
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: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 {};
|
|
}
|
|
|
|
StringView string_from_property_id(PropertyID property_id) {
|
|
switch (property_id) {
|
|
)~~~");
|
|
|
|
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:titlecase", title_casify(name));
|
|
member_generator.append(R"~~~(
|
|
case PropertyID::@name:titlecase@:
|
|
return "@name@"sv;
|
|
)~~~");
|
|
});
|
|
|
|
generator.append(R"~~~(
|
|
default:
|
|
return "(invalid CSS::PropertyID)"sv;
|
|
}
|
|
}
|
|
|
|
bool is_inherited_property(PropertyID property_id)
|
|
{
|
|
switch (property_id) {
|
|
)~~~");
|
|
|
|
properties.for_each_member([&](auto& name, auto& value) {
|
|
VERIFY(value.is_object());
|
|
|
|
bool inherited = false;
|
|
if (value.as_object().has("inherited"sv)) {
|
|
auto inherited_value = value.as_object().get_bool("inherited"sv);
|
|
VERIFY(inherited_value.has_value());
|
|
inherited = inherited_value.value();
|
|
}
|
|
|
|
if (inherited) {
|
|
auto member_generator = generator.fork();
|
|
member_generator.set("name:titlecase", title_casify(name));
|
|
member_generator.append(R"~~~(
|
|
case PropertyID::@name:titlecase@:
|
|
return true;
|
|
)~~~");
|
|
}
|
|
});
|
|
|
|
generator.append(R"~~~(
|
|
default:
|
|
return false;
|
|
}
|
|
}
|
|
|
|
bool property_affects_layout(PropertyID property_id)
|
|
{
|
|
switch (property_id) {
|
|
)~~~");
|
|
|
|
properties.for_each_member([&](auto& name, auto& value) {
|
|
VERIFY(value.is_object());
|
|
|
|
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());
|
|
|
|
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;
|
|
}
|
|
}
|
|
|
|
ErrorOr<NonnullRefPtr<StyleValue>> property_initial_value(JS::Realm& context_realm, PropertyID property_id)
|
|
{
|
|
static Array<RefPtr<StyleValue>, to_underlying(last_property_id) + 1> initial_values;
|
|
if (auto initial_value = initial_values[to_underlying(property_id)])
|
|
return initial_value.release_nonnull();
|
|
|
|
// 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);
|
|
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_deprecated_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 = TRY(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());
|
|
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 (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 (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 == "color") {
|
|
property_generator.appendln(" case ValueType::Color:");
|
|
} else if (type_name == "custom-ident") {
|
|
property_generator.appendln(" case ValueType::CustomIdent:");
|
|
} 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 == "percentage") {
|
|
property_generator.appendln(" case ValueType::Percentage:");
|
|
} 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_identifier(PropertyID property_id, ValueID identifier)
|
|
{
|
|
switch (property_id) {
|
|
)~~~");
|
|
properties.for_each_member([&](auto& name, auto& value) {
|
|
VERIFY(value.is_object());
|
|
auto& object = value.as_object();
|
|
|
|
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 (identifier) {");
|
|
auto& valid_identifiers = maybe_valid_identifiers.value();
|
|
for (auto& identifier : valid_identifiers.values()) {
|
|
auto identifier_generator = generator.fork();
|
|
identifier_generator.set("identifier:titlecase", title_casify(identifier.as_string()));
|
|
identifier_generator.appendln(" case ValueID::@identifier: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 (value_id_to_@type_name:snakecase@(identifier).has_value())
|
|
return true;
|
|
)~~~");
|
|
}
|
|
}
|
|
property_generator.append(R"~~~(
|
|
return false;
|
|
}
|
|
)~~~");
|
|
});
|
|
generator.append(R"~~~(
|
|
default:
|
|
return false;
|
|
}
|
|
}
|
|
|
|
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 (value.as_object().has("max-values"sv)) {
|
|
auto max_values = value.as_object().get("max-values"sv);
|
|
VERIFY(max_values.has_value() && max_values->is_number() && !max_values->is_double());
|
|
auto property_generator = generator.fork();
|
|
property_generator.set("name:titlecase", title_casify(name));
|
|
property_generator.set("max_values", max_values->to_deprecated_string());
|
|
property_generator.append(R"~~~(
|
|
case PropertyID::@name:titlecase@:
|
|
return @max_values@;
|
|
)~~~");
|
|
}
|
|
});
|
|
|
|
generator.append(R"~~~(
|
|
default:
|
|
return 1;
|
|
}
|
|
}
|
|
|
|
} // namespace Web::CSS
|
|
|
|
)~~~");
|
|
|
|
TRY(file.write_until_depleted(generator.as_string_view().bytes()));
|
|
return {};
|
|
}
|