/* * Copyright (c) 2022, Sam Atkins * * SPDX-License-Identifier: BSD-2-Clause */ #include "GeneratorUtil.h" #include #include #include #include ErrorOr generate_header_file(JsonObject& enums_data, Core::Stream::File& file); ErrorOr generate_implementation_file(JsonObject& enums_data, Core::Stream::File& file); ErrorOr serenity_main(Main::Arguments arguments) { StringView generated_header_path; StringView generated_implementation_path; StringView identifiers_json_path; Core::ArgsParser args_parser; args_parser.add_option(generated_header_path, "Path to the Enums header file to generate", "generated-header-path", 'h', "generated-header-path"); args_parser.add_option(generated_implementation_path, "Path to the Enums implementation file to generate", "generated-implementation-path", 'c', "generated-implementation-path"); args_parser.add_option(identifiers_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(identifiers_json_path)); VERIFY(json.is_object()); auto enums_data = json.as_object(); auto generated_header_file = TRY(Core::Stream::File::open(generated_header_path, Core::Stream::OpenMode::Write)); auto generated_implementation_file = TRY(Core::Stream::File::open(generated_implementation_path, Core::Stream::OpenMode::Write)); TRY(generate_header_file(enums_data, *generated_header_file)); TRY(generate_implementation_file(enums_data, *generated_implementation_file)); return 0; } ErrorOr generate_header_file(JsonObject& enums_data, Core::Stream::File& file) { StringBuilder builder; SourceGenerator generator { builder }; generator.append(R"~~~( #pragma once #include namespace Web::CSS { enum class ValueID; )~~~"); enums_data.for_each_member([&](auto& name, auto& value) { VERIFY(value.is_array()); auto& members = value.as_array(); auto enum_generator = generator.fork(); enum_generator.set("name:titlecase", title_casify(name)); enum_generator.set("name:snakecase", snake_casify(name)); // Find the smallest possible type to use. auto member_max_value = members.size() - 1; if (NumericLimits::max() >= member_max_value) { enum_generator.set("enum_type", "u8"); } else if (NumericLimits::max() >= member_max_value) { enum_generator.set("enum_type", "u16"); } else if (NumericLimits::max() >= member_max_value) { enum_generator.set("enum_type", "u32"); } else { enum_generator.set("enum_type", "u64"); } enum_generator.appendln("enum class @name:titlecase@ : @enum_type@ {"); for (auto& member : members.values()) { auto member_name = member.to_string(); // Don't include aliases in the enum. if (member_name.contains('=')) continue; auto member_generator = enum_generator.fork(); member_generator.set("member:titlecase", title_casify(member_name)); member_generator.appendln(" @member:titlecase@,"); } enum_generator.appendln("};"); enum_generator.appendln("Optional<@name:titlecase@> value_id_to_@name:snakecase@(ValueID);"); enum_generator.appendln("ValueID to_value_id(@name:titlecase@);"); enum_generator.appendln("StringView to_string(@name:titlecase@);"); enum_generator.append("\n"); }); generator.appendln("}"); TRY(file.write(generator.as_string_view().bytes())); return {}; } ErrorOr generate_implementation_file(JsonObject& enums_data, Core::Stream::File& file) { StringBuilder builder; SourceGenerator generator { builder }; generator.append(R"~~~( #include #include namespace Web::CSS { )~~~"); enums_data.for_each_member([&](auto& name, auto& value) { VERIFY(value.is_array()); auto& members = value.as_array(); auto enum_generator = generator.fork(); enum_generator.set("name:titlecase", title_casify(name)); enum_generator.set("name:snakecase", snake_casify(name)); enum_generator.append(R"~~~( Optional<@name:titlecase@> value_id_to_@name:snakecase@(ValueID value_id) { switch (value_id) {)~~~"); for (auto& member : members.values()) { auto member_generator = enum_generator.fork(); auto member_name = member.to_string(); if (member_name.contains('=')) { auto parts = member_name.split_view('='); member_generator.set("valueid:titlecase", title_casify(parts[0])); member_generator.set("member:titlecase", title_casify(parts[1])); } else { member_generator.set("valueid:titlecase", title_casify(member_name)); member_generator.set("member:titlecase", title_casify(member_name)); } member_generator.append(R"~~~( case ValueID::@valueid:titlecase@: return @name:titlecase@::@member:titlecase@;)~~~"); } enum_generator.append(R"~~~( default: return {}; } } )~~~"); enum_generator.append(R"~~~( ValueID to_value_id(@name:titlecase@ @name:snakecase@_value) { switch (@name:snakecase@_value) {)~~~"); for (auto& member : members.values()) { auto member_generator = enum_generator.fork(); auto member_name = member.to_string(); if (member_name.contains('=')) continue; member_generator.set("member:titlecase", title_casify(member_name)); member_generator.append(R"~~~( case @name:titlecase@::@member:titlecase@: return ValueID::@member:titlecase@;)~~~"); } enum_generator.append(R"~~~( default: VERIFY_NOT_REACHED(); } } )~~~"); enum_generator.append(R"~~~( StringView to_string(@name:titlecase@ value) { switch (value) {)~~~"); for (auto& member : members.values()) { auto member_generator = enum_generator.fork(); auto member_name = member.to_string(); if (member_name.contains('=')) continue; member_generator.set("member:css", member_name); member_generator.set("member:titlecase", title_casify(member_name)); member_generator.append(R"~~~( case @name:titlecase@::@member:titlecase@: return "@member:css@"sv;)~~~"); } enum_generator.append(R"~~~( default: VERIFY_NOT_REACHED(); } } )~~~"); }); generator.appendln("}"); TRY(file.write(generator.as_string_view().bytes())); return {}; }