GenerateCSSEasingFunctions.cpp 6.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214
  1. /*
  2. * Copyright (c) 2022-2023, Sam Atkins <atkinssj@serenityos.org>
  3. * Copyright (c) 2023, Ali Mohammad Pur <mpfard@serenityos.org>
  4. *
  5. * SPDX-License-Identifier: BSD-2-Clause
  6. */
  7. #include "GeneratorUtil.h"
  8. #include <AK/GenericLexer.h>
  9. #include <AK/SourceGenerator.h>
  10. #include <LibCore/ArgsParser.h>
  11. #include <LibMain/Main.h>
  12. ErrorOr<void> generate_header_file(JsonObject&, Core::File&);
  13. ErrorOr<void> generate_implementation_file(JsonObject&, Core::File&);
  14. ErrorOr<int> serenity_main(Main::Arguments arguments)
  15. {
  16. StringView generated_header_path;
  17. StringView generated_implementation_path;
  18. StringView functions_json_path;
  19. Core::ArgsParser args_parser;
  20. args_parser.add_option(generated_header_path, "Path to the EasingFunctions header file to generate", "generated-header-path", 'h', "generated-header-path");
  21. args_parser.add_option(generated_implementation_path, "Path to the EasingFunctions implementation file to generate", "generated-implementation-path", 'c', "generated-implementation-path");
  22. args_parser.add_option(functions_json_path, "Path to the JSON file to read from", "json-path", 'j', "json-path");
  23. args_parser.parse(arguments);
  24. auto json = TRY(read_entire_file_as_json(functions_json_path));
  25. VERIFY(json.is_object());
  26. auto easing_data = json.as_object();
  27. auto generated_header_file = TRY(Core::File::open(generated_header_path, Core::File::OpenMode::Write));
  28. auto generated_implementation_file = TRY(Core::File::open(generated_implementation_path, Core::File::OpenMode::Write));
  29. TRY(generate_header_file(easing_data, *generated_header_file));
  30. TRY(generate_implementation_file(easing_data, *generated_implementation_file));
  31. return 0;
  32. }
  33. ErrorOr<void> generate_header_file(JsonObject& easing_data, Core::File& file)
  34. {
  35. StringBuilder builder;
  36. SourceGenerator generator { builder };
  37. generator.append(R"~~~(
  38. #pragma once
  39. #include <AK/Optional.h>
  40. #include <AK/StringView.h>
  41. #include <AK/Vector.h>
  42. namespace Web::CSS {
  43. )~~~");
  44. generator.appendln("enum class EasingFunction {");
  45. easing_data.for_each_member([&](auto& name, auto&) {
  46. auto member_generator = generator.fork();
  47. member_generator.set("name:titlecase", title_casify(name));
  48. member_generator.appendln(" @name:titlecase@,");
  49. });
  50. generator.appendln("};");
  51. generator.appendln("Optional<EasingFunction> easing_function_from_string(StringView);");
  52. generator.appendln("StringView to_string(EasingFunction);");
  53. generator.append(R"~~~(
  54. enum class EasingFunctionParameterType {
  55. Integer,
  56. Number,
  57. NumberZeroToOne,
  58. StepPosition,
  59. };
  60. struct EasingFunctionParameter {
  61. EasingFunctionParameterType type;
  62. bool is_optional { false };
  63. };
  64. struct EasingFunctionMetadata {
  65. Vector<EasingFunctionParameter> parameters;
  66. };
  67. EasingFunctionMetadata easing_function_metadata(EasingFunction);
  68. )~~~");
  69. generator.appendln("\n}");
  70. TRY(file.write_until_depleted(generator.as_string_view().bytes()));
  71. return {};
  72. }
  73. ErrorOr<void> generate_implementation_file(JsonObject& easing_data, Core::File& file)
  74. {
  75. StringBuilder builder;
  76. SourceGenerator generator { builder };
  77. generator.append(R"~~~(
  78. #include <LibWeb/CSS/EasingFunctions.h>
  79. #include <AK/Assertions.h>
  80. namespace Web::CSS {
  81. )~~~");
  82. generator.append(R"~~~(
  83. Optional<EasingFunction> easing_function_from_string(StringView name)
  84. {
  85. )~~~");
  86. easing_data.for_each_member([&](auto& name, auto&) {
  87. auto member_generator = generator.fork();
  88. member_generator.set("name", name);
  89. member_generator.set("name:titlecase", title_casify(name));
  90. member_generator.append(R"~~~(
  91. if (name.equals_ignoring_ascii_case("@name@"sv))
  92. return EasingFunction::@name:titlecase@;
  93. )~~~");
  94. });
  95. generator.append(R"~~~(
  96. return {};
  97. }
  98. )~~~");
  99. generator.append(R"~~~(
  100. StringView to_string(EasingFunction easing_function)
  101. {
  102. switch (easing_function) {
  103. )~~~");
  104. easing_data.for_each_member([&](auto& name, auto&) {
  105. auto member_generator = generator.fork();
  106. member_generator.set("name", name);
  107. member_generator.set("name:titlecase", title_casify(name));
  108. member_generator.append(R"~~~(
  109. case EasingFunction::@name:titlecase@:
  110. return "@name@"sv;
  111. )~~~");
  112. });
  113. generator.append(R"~~~(
  114. default:
  115. VERIFY_NOT_REACHED();
  116. }
  117. }
  118. )~~~");
  119. generator.append(R"~~~(
  120. EasingFunctionMetadata easing_function_metadata(EasingFunction easing_function)
  121. {
  122. switch (easing_function) {
  123. )~~~");
  124. easing_data.for_each_member([&](auto& name, auto& value) {
  125. VERIFY(value.is_object());
  126. auto member_generator = generator.fork();
  127. member_generator.set("name:titlecase", title_casify(name));
  128. member_generator.append(R"~~~(
  129. case EasingFunction::@name:titlecase@:
  130. return EasingFunctionMetadata {
  131. .parameters = {)~~~");
  132. if (auto parameters = value.as_object().get_array("parameters"sv); parameters.has_value()) {
  133. bool first = true;
  134. // parameters: [ "<foo>", "<foo [0, 1]>" ]
  135. parameters.value().for_each([&](JsonValue const& value) {
  136. GenericLexer lexer { value.as_string() };
  137. VERIFY(lexer.consume_specific('<'));
  138. auto parameter_type_name = lexer.consume_until([](char ch) { return ch == ' ' || ch == '>'; });
  139. auto has_bounds = false;
  140. auto is_optional = false;
  141. if (lexer.consume_specific(" [")) {
  142. has_bounds = true;
  143. auto contents = lexer.consume_until(']');
  144. VERIFY(contents == "0, 1"sv);
  145. VERIFY(lexer.consume_specific(']'));
  146. }
  147. VERIFY(lexer.consume_specific('>'));
  148. if (lexer.consume_specific('?'))
  149. is_optional = true;
  150. StringView parameter_type = ""sv;
  151. if (parameter_type_name == "number"sv)
  152. parameter_type = has_bounds ? "NumberZeroToOne"sv : "Number"sv;
  153. else if (parameter_type_name == "integer"sv)
  154. parameter_type = "Integer"sv;
  155. else if (parameter_type_name == "step-position"sv)
  156. parameter_type = "StepPosition"sv;
  157. else
  158. VERIFY_NOT_REACHED();
  159. member_generator.append(first ? " "sv : ", "sv);
  160. first = false;
  161. member_generator.append(MUST(String::formatted(
  162. "{{ EasingFunctionParameterType::{}, {} }}",
  163. parameter_type,
  164. is_optional ? "true"sv : "false"sv)));
  165. });
  166. }
  167. member_generator.append(R"~~~( }
  168. };
  169. )~~~");
  170. });
  171. generator.append(R"~~~(
  172. default:
  173. VERIFY_NOT_REACHED();
  174. }
  175. }
  176. )~~~");
  177. generator.appendln("\n}");
  178. TRY(file.write_until_depleted(generator.as_string_view().bytes()));
  179. return {};
  180. }