GenerateCSSEnums.cpp 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166
  1. /*
  2. * Copyright (c) 2022, Sam Atkins <atkinssj@serenityos.org>
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #include "GeneratorUtil.h"
  7. #include <AK/SourceGenerator.h>
  8. #include <AK/StringBuilder.h>
  9. #include <LibCore/ArgsParser.h>
  10. #include <LibMain/Main.h>
  11. ErrorOr<void> generate_header_file(JsonObject& enums_data, Core::Stream::File& file);
  12. ErrorOr<void> generate_implementation_file(JsonObject& enums_data, Core::Stream::File& file);
  13. ErrorOr<int> serenity_main(Main::Arguments arguments)
  14. {
  15. StringView generated_header_path;
  16. StringView generated_implementation_path;
  17. StringView identifiers_json_path;
  18. Core::ArgsParser args_parser;
  19. args_parser.add_option(generated_header_path, "Path to the Enums header file to generate", "generated-header-path", 'h', "generated-header-path");
  20. args_parser.add_option(generated_implementation_path, "Path to the Enums implementation file to generate", "generated-implementation-path", 'c', "generated-implementation-path");
  21. args_parser.add_option(identifiers_json_path, "Path to the JSON file to read from", "json-path", 'j', "json-path");
  22. args_parser.parse(arguments);
  23. auto json = TRY(read_entire_file_as_json(identifiers_json_path));
  24. VERIFY(json.is_object());
  25. auto enums_data = json.as_object();
  26. auto generated_header_file = TRY(Core::Stream::File::open(generated_header_path, Core::Stream::OpenMode::Write));
  27. auto generated_implementation_file = TRY(Core::Stream::File::open(generated_implementation_path, Core::Stream::OpenMode::Write));
  28. TRY(generate_header_file(enums_data, *generated_header_file));
  29. TRY(generate_implementation_file(enums_data, *generated_implementation_file));
  30. return 0;
  31. }
  32. ErrorOr<void> generate_header_file(JsonObject& enums_data, Core::Stream::File& file)
  33. {
  34. StringBuilder builder;
  35. SourceGenerator generator { builder };
  36. generator.append(R"~~~(
  37. #pragma once
  38. #include <AK/Optional.h>
  39. namespace Web::CSS {
  40. enum class ValueID;
  41. )~~~");
  42. enums_data.for_each_member([&](auto& name, auto& value) {
  43. VERIFY(value.is_array());
  44. auto& members = value.as_array();
  45. auto enum_generator = generator.fork();
  46. enum_generator.set("name:titlecase", title_casify(name));
  47. enum_generator.set("name:snakecase", snake_casify(name));
  48. enum_generator.appendln("enum class @name:titlecase@ {");
  49. for (auto& member : members.values()) {
  50. auto member_name = member.to_string();
  51. // Don't include aliases in the enum.
  52. if (member_name.contains('='))
  53. continue;
  54. auto member_generator = enum_generator.fork();
  55. member_generator.set("member:titlecase", title_casify(member_name));
  56. member_generator.appendln(" @member:titlecase@,");
  57. }
  58. enum_generator.appendln("};");
  59. enum_generator.appendln("Optional<@name:titlecase@> value_id_to_@name:snakecase@(ValueID);");
  60. enum_generator.appendln("ValueID to_value_id(@name:titlecase@);");
  61. enum_generator.append("\n");
  62. });
  63. generator.appendln("}");
  64. TRY(file.write(generator.as_string_view().bytes()));
  65. return {};
  66. }
  67. ErrorOr<void> generate_implementation_file(JsonObject& enums_data, Core::Stream::File& file)
  68. {
  69. StringBuilder builder;
  70. SourceGenerator generator { builder };
  71. generator.append(R"~~~(
  72. #include <LibWeb/CSS/Enums.h>
  73. #include <LibWeb/CSS/ValueID.h>
  74. namespace Web::CSS {
  75. )~~~");
  76. enums_data.for_each_member([&](auto& name, auto& value) {
  77. VERIFY(value.is_array());
  78. auto& members = value.as_array();
  79. auto enum_generator = generator.fork();
  80. enum_generator.set("name:titlecase", title_casify(name));
  81. enum_generator.set("name:snakecase", snake_casify(name));
  82. enum_generator.append(R"~~~(
  83. Optional<@name:titlecase@> value_id_to_@name:snakecase@(ValueID value_id)
  84. {
  85. switch (value_id) {)~~~");
  86. for (auto& member : members.values()) {
  87. auto member_generator = enum_generator.fork();
  88. auto member_name = member.to_string();
  89. if (member_name.contains('=')) {
  90. auto parts = member_name.split_view('=');
  91. member_generator.set("valueid:titlecase", title_casify(parts[0]));
  92. member_generator.set("member:titlecase", title_casify(parts[1]));
  93. } else {
  94. member_generator.set("valueid:titlecase", title_casify(member_name));
  95. member_generator.set("member:titlecase", title_casify(member_name));
  96. }
  97. member_generator.append(R"~~~(
  98. case ValueID::@valueid:titlecase@:
  99. return @name:titlecase@::@member:titlecase@;)~~~");
  100. }
  101. enum_generator.append(R"~~~(
  102. default:
  103. return {};
  104. }
  105. }
  106. )~~~");
  107. enum_generator.append(R"~~~(
  108. ValueID to_value_id(@name:titlecase@ @name:snakecase@_value)
  109. {
  110. switch (@name:snakecase@_value) {)~~~");
  111. for (auto& member : members.values()) {
  112. auto member_generator = enum_generator.fork();
  113. auto member_name = member.to_string();
  114. if (member_name.contains('='))
  115. continue;
  116. member_generator.set("member:titlecase", title_casify(member_name));
  117. member_generator.append(R"~~~(
  118. case @name:titlecase@::@member:titlecase@:
  119. return ValueID::@member:titlecase@;)~~~");
  120. }
  121. enum_generator.append(R"~~~(
  122. default:
  123. VERIFY_NOT_REACHED();
  124. }
  125. }
  126. )~~~");
  127. });
  128. generator.appendln("}");
  129. TRY(file.write(generator.as_string_view().bytes()));
  130. return {};
  131. }