GenerateCSSEasingFunctions.cpp 7.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219
  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. TRY(generator.try_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. TRY(generator.try_appendln("enum class EasingFunction {"));
  45. TRY(easing_data.try_for_each_member([&](auto& name, auto&) -> ErrorOr<void> {
  46. auto member_generator = TRY(generator.fork());
  47. member_generator.set("name:titlecase", TRY(title_casify(name)));
  48. TRY(member_generator.try_appendln(" @name:titlecase@,"));
  49. return {};
  50. }));
  51. TRY(generator.try_appendln("};"));
  52. TRY(generator.try_appendln("Optional<EasingFunction> easing_function_from_string(StringView);"));
  53. TRY(generator.try_appendln("StringView to_string(EasingFunction);"));
  54. TRY(generator.try_append(R"~~~(
  55. enum class EasingFunctionParameterType {
  56. Integer,
  57. Number,
  58. NumberZeroToOne,
  59. StepPosition,
  60. };
  61. struct EasingFunctionParameter {
  62. EasingFunctionParameterType type;
  63. bool is_optional { false };
  64. };
  65. struct EasingFunctionMetadata {
  66. Vector<EasingFunctionParameter> parameters;
  67. };
  68. EasingFunctionMetadata easing_function_metadata(EasingFunction);
  69. )~~~"));
  70. TRY(generator.try_appendln("\n}"));
  71. TRY(file.write_until_depleted(generator.as_string_view().bytes()));
  72. return {};
  73. }
  74. ErrorOr<void> generate_implementation_file(JsonObject& easing_data, Core::File& file)
  75. {
  76. StringBuilder builder;
  77. SourceGenerator generator { builder };
  78. TRY(generator.try_append(R"~~~(
  79. #include <LibWeb/CSS/EasingFunctions.h>
  80. #include <AK/Assertions.h>
  81. namespace Web::CSS {
  82. )~~~"));
  83. TRY(generator.try_append(R"~~~(
  84. Optional<EasingFunction> easing_function_from_string(StringView name)
  85. {
  86. )~~~"));
  87. TRY(easing_data.try_for_each_member([&](auto& name, auto&) -> ErrorOr<void> {
  88. auto member_generator = TRY(generator.fork());
  89. member_generator.set("name", TRY(String::from_deprecated_string(name)));
  90. member_generator.set("name:titlecase", TRY(title_casify(name)));
  91. TRY(member_generator.try_append(R"~~~(
  92. if (name.equals_ignoring_ascii_case("@name@"sv))
  93. return EasingFunction::@name:titlecase@;
  94. )~~~"));
  95. return {};
  96. }));
  97. TRY(generator.try_append(R"~~~(
  98. return {};
  99. }
  100. )~~~"));
  101. TRY(generator.try_append(R"~~~(
  102. StringView to_string(EasingFunction easing_function)
  103. {
  104. switch (easing_function) {
  105. )~~~"));
  106. TRY(easing_data.try_for_each_member([&](auto& name, auto&) -> ErrorOr<void> {
  107. auto member_generator = TRY(generator.fork());
  108. member_generator.set("name", TRY(String::from_deprecated_string(name)));
  109. member_generator.set("name:titlecase", TRY(title_casify(name)));
  110. TRY(member_generator.try_append(R"~~~(
  111. case EasingFunction::@name:titlecase@:
  112. return "@name@"sv;
  113. )~~~"));
  114. return {};
  115. }));
  116. TRY(generator.try_append(R"~~~(
  117. default:
  118. VERIFY_NOT_REACHED();
  119. }
  120. }
  121. )~~~"));
  122. TRY(generator.try_append(R"~~~(
  123. EasingFunctionMetadata easing_function_metadata(EasingFunction easing_function)
  124. {
  125. switch (easing_function) {
  126. )~~~"));
  127. TRY(easing_data.try_for_each_member([&](auto& name, auto& value) -> ErrorOr<void> {
  128. VERIFY(value.is_object());
  129. auto member_generator = TRY(generator.fork());
  130. member_generator.set("name:titlecase", TRY(title_casify(name)));
  131. TRY(member_generator.try_append(R"~~~(
  132. case EasingFunction::@name:titlecase@:
  133. return EasingFunctionMetadata {
  134. .parameters = {)~~~"));
  135. if (auto parameters = value.as_object().get_array("parameters"sv); parameters.has_value()) {
  136. bool first = true;
  137. // parameters: [ "<foo>", "<foo [0, 1]>" ]
  138. TRY(parameters.value().try_for_each([&](JsonValue const& value) -> ErrorOr<void> {
  139. GenericLexer lexer { value.as_string() };
  140. VERIFY(lexer.consume_specific('<'));
  141. auto parameter_type_name = lexer.consume_until([](char ch) { return ch == ' ' || ch == '>'; });
  142. auto has_bounds = false;
  143. auto is_optional = false;
  144. if (lexer.consume_specific(" [")) {
  145. has_bounds = true;
  146. auto contents = lexer.consume_until(']');
  147. VERIFY(contents == "0, 1"sv);
  148. VERIFY(lexer.consume_specific(']'));
  149. }
  150. VERIFY(lexer.consume_specific('>'));
  151. if (lexer.consume_specific('?'))
  152. is_optional = true;
  153. StringView parameter_type = ""sv;
  154. if (parameter_type_name == "number"sv)
  155. parameter_type = has_bounds ? "NumberZeroToOne"sv : "Number"sv;
  156. else if (parameter_type_name == "integer"sv)
  157. parameter_type = "Integer"sv;
  158. else if (parameter_type_name == "step-position"sv)
  159. parameter_type = "StepPosition"sv;
  160. else
  161. VERIFY_NOT_REACHED();
  162. TRY(member_generator.try_append(first ? " "sv : ", "sv));
  163. first = false;
  164. TRY(member_generator.try_append(TRY(String::formatted(
  165. "{{ EasingFunctionParameterType::{}, {} }}",
  166. parameter_type,
  167. is_optional ? "true"sv : "false"sv))));
  168. return {};
  169. }));
  170. }
  171. TRY(member_generator.try_append(R"~~~( }
  172. };
  173. )~~~"));
  174. return {};
  175. }));
  176. TRY(generator.try_append(R"~~~(
  177. default:
  178. VERIFY_NOT_REACHED();
  179. }
  180. }
  181. )~~~"));
  182. TRY(generator.try_appendln("\n}"));
  183. TRY(file.write_until_depleted(generator.as_string_view().bytes()));
  184. return {};
  185. }