GenerateCSSPropertyID.cpp 28 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778
  1. /*
  2. * Copyright (c) 2018-2020, Andreas Kling <kling@serenityos.org>
  3. * Copyright (c) 2021-2023, Sam Atkins <atkinssj@serenityos.org>
  4. *
  5. * SPDX-License-Identifier: BSD-2-Clause
  6. */
  7. #include "GeneratorUtil.h"
  8. #include <AK/CharacterTypes.h>
  9. #include <AK/GenericShorthands.h>
  10. #include <AK/SourceGenerator.h>
  11. #include <AK/StringBuilder.h>
  12. #include <LibCore/ArgsParser.h>
  13. #include <LibMain/Main.h>
  14. ErrorOr<void> replace_logical_aliases(JsonObject& properties);
  15. ErrorOr<void> generate_header_file(JsonObject& properties, Core::File& file);
  16. ErrorOr<void> generate_implementation_file(JsonObject& properties, Core::File& file);
  17. void generate_bounds_checking_function(JsonObject& properties, SourceGenerator& parent_generator, StringView css_type_name, StringView type_name, Optional<StringView> default_unit_name = {}, Optional<StringView> value_getter = {});
  18. static bool type_name_is_enum(StringView type_name)
  19. {
  20. 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);
  21. }
  22. ErrorOr<int> serenity_main(Main::Arguments arguments)
  23. {
  24. StringView generated_header_path;
  25. StringView generated_implementation_path;
  26. StringView properties_json_path;
  27. Core::ArgsParser args_parser;
  28. args_parser.add_option(generated_header_path, "Path to the PropertyID header file to generate", "generated-header-path", 'h', "generated-header-path");
  29. args_parser.add_option(generated_implementation_path, "Path to the PropertyID implementation file to generate", "generated-implementation-path", 'c', "generated-implementation-path");
  30. args_parser.add_option(properties_json_path, "Path to the JSON file to read from", "json-path", 'j', "json-path");
  31. args_parser.parse(arguments);
  32. auto json = TRY(read_entire_file_as_json(properties_json_path));
  33. VERIFY(json.is_object());
  34. auto properties = json.as_object();
  35. TRY(replace_logical_aliases(properties));
  36. auto generated_header_file = TRY(Core::File::open(generated_header_path, Core::File::OpenMode::Write));
  37. auto generated_implementation_file = TRY(Core::File::open(generated_implementation_path, Core::File::OpenMode::Write));
  38. TRY(generate_header_file(properties, *generated_header_file));
  39. TRY(generate_implementation_file(properties, *generated_implementation_file));
  40. return 0;
  41. }
  42. ErrorOr<void> replace_logical_aliases(JsonObject& properties)
  43. {
  44. AK::HashMap<DeprecatedString, DeprecatedString> logical_aliases;
  45. properties.for_each_member([&](auto& name, auto& value) {
  46. VERIFY(value.is_object());
  47. const auto& value_as_object = value.as_object();
  48. const auto logical_alias_for = value_as_object.get_deprecated_string("logical-alias-for"sv);
  49. if (logical_alias_for.has_value()) {
  50. logical_aliases.set(name, logical_alias_for.value());
  51. }
  52. });
  53. for (auto& [name, alias] : logical_aliases) {
  54. auto const maybe_alias_object = properties.get_object(alias);
  55. if (!maybe_alias_object.has_value()) {
  56. dbgln("No property '{}' found for logical alias '{}'", alias, name);
  57. VERIFY_NOT_REACHED();
  58. }
  59. JsonObject alias_object = maybe_alias_object.value();
  60. // Copy over anything the logical property overrides
  61. properties.get_object(name).value().for_each_member([&](auto& key, auto& value) {
  62. alias_object.set(key, value);
  63. });
  64. properties.set(name, alias_object);
  65. }
  66. return {};
  67. }
  68. ErrorOr<void> generate_header_file(JsonObject& properties, Core::File& file)
  69. {
  70. StringBuilder builder;
  71. SourceGenerator generator { builder };
  72. generator.append(R"~~~(
  73. #pragma once
  74. #include <AK/NonnullRefPtr.h>
  75. #include <AK/StringView.h>
  76. #include <AK/Traits.h>
  77. #include <LibJS/Forward.h>
  78. #include <LibWeb/Forward.h>
  79. namespace Web::CSS {
  80. enum class PropertyID {
  81. Invalid,
  82. Custom,
  83. )~~~");
  84. Vector<DeprecatedString> shorthand_property_ids;
  85. Vector<DeprecatedString> longhand_property_ids;
  86. properties.for_each_member([&](auto& name, auto& value) {
  87. VERIFY(value.is_object());
  88. if (value.as_object().has("longhands"sv))
  89. shorthand_property_ids.append(name);
  90. else
  91. longhand_property_ids.append(name);
  92. });
  93. auto first_property_id = shorthand_property_ids.first();
  94. auto last_property_id = longhand_property_ids.last();
  95. for (auto& name : shorthand_property_ids) {
  96. auto member_generator = generator.fork();
  97. member_generator.set("name:titlecase", title_casify(name));
  98. member_generator.append(R"~~~(
  99. @name:titlecase@,
  100. )~~~");
  101. }
  102. for (auto& name : longhand_property_ids) {
  103. auto member_generator = generator.fork();
  104. member_generator.set("name:titlecase", title_casify(name));
  105. member_generator.append(R"~~~(
  106. @name:titlecase@,
  107. )~~~");
  108. }
  109. generator.set("first_property_id", title_casify(first_property_id));
  110. generator.set("last_property_id", title_casify(last_property_id));
  111. generator.set("first_shorthand_property_id", title_casify(shorthand_property_ids.first()));
  112. generator.set("last_shorthand_property_id", title_casify(shorthand_property_ids.last()));
  113. generator.set("first_longhand_property_id", title_casify(longhand_property_ids.first()));
  114. generator.set("last_longhand_property_id", title_casify(longhand_property_ids.last()));
  115. generator.append(R"~~~(
  116. };
  117. Optional<PropertyID> property_id_from_camel_case_string(StringView);
  118. Optional<PropertyID> property_id_from_string(StringView);
  119. StringView string_from_property_id(PropertyID);
  120. bool is_inherited_property(PropertyID);
  121. ErrorOr<NonnullRefPtr<StyleValue>> property_initial_value(JS::Realm&, PropertyID);
  122. enum class ValueType {
  123. Angle,
  124. Color,
  125. CustomIdent,
  126. FilterValueList,
  127. Frequency,
  128. Image,
  129. Integer,
  130. Length,
  131. Number,
  132. Paint,
  133. Percentage,
  134. Position,
  135. Rect,
  136. Resolution,
  137. String,
  138. Time,
  139. Url,
  140. };
  141. bool property_accepts_type(PropertyID, ValueType);
  142. bool property_accepts_identifier(PropertyID, ValueID);
  143. // These perform range-checking, but are also safe to call with properties that don't accept that type. (They'll just return false.)
  144. bool property_accepts_angle(PropertyID, Angle const&);
  145. bool property_accepts_frequency(PropertyID, Frequency const&);
  146. bool property_accepts_integer(PropertyID, i64 const&);
  147. bool property_accepts_length(PropertyID, Length const&);
  148. bool property_accepts_number(PropertyID, double const&);
  149. bool property_accepts_percentage(PropertyID, Percentage const&);
  150. bool property_accepts_resolution(PropertyID, Resolution const&);
  151. bool property_accepts_time(PropertyID, Time const&);
  152. Vector<PropertyID> longhands_for_shorthand(PropertyID);
  153. size_t property_maximum_value_count(PropertyID);
  154. bool property_affects_layout(PropertyID);
  155. bool property_affects_stacking_context(PropertyID);
  156. constexpr PropertyID first_property_id = PropertyID::@first_property_id@;
  157. constexpr PropertyID last_property_id = PropertyID::@last_property_id@;
  158. constexpr PropertyID first_shorthand_property_id = PropertyID::@first_shorthand_property_id@;
  159. constexpr PropertyID last_shorthand_property_id = PropertyID::@last_shorthand_property_id@;
  160. constexpr PropertyID first_longhand_property_id = PropertyID::@first_longhand_property_id@;
  161. constexpr PropertyID last_longhand_property_id = PropertyID::@last_longhand_property_id@;
  162. enum class Quirk {
  163. // https://quirks.spec.whatwg.org/#the-hashless-hex-color-quirk
  164. HashlessHexColor,
  165. // https://quirks.spec.whatwg.org/#the-unitless-length-quirk
  166. UnitlessLength,
  167. };
  168. bool property_has_quirk(PropertyID, Quirk);
  169. } // namespace Web::CSS
  170. namespace AK {
  171. template<>
  172. struct Traits<Web::CSS::PropertyID> : public GenericTraits<Web::CSS::PropertyID> {
  173. static unsigned hash(Web::CSS::PropertyID property_id) { return int_hash((unsigned)property_id); }
  174. };
  175. } // namespace AK
  176. )~~~");
  177. TRY(file.write_until_depleted(generator.as_string_view().bytes()));
  178. return {};
  179. }
  180. void generate_bounds_checking_function(JsonObject& properties, SourceGenerator& parent_generator, StringView css_type_name, StringView type_name, Optional<StringView> default_unit_name, Optional<StringView> value_getter)
  181. {
  182. auto generator = parent_generator.fork();
  183. generator.set("css_type_name", css_type_name);
  184. generator.set("type_name", type_name);
  185. generator.append(R"~~~(
  186. bool property_accepts_@css_type_name@(PropertyID property_id, [[maybe_unused]] @type_name@ const& value)
  187. {
  188. switch (property_id) {
  189. )~~~");
  190. properties.for_each_member([&](auto& name, JsonValue const& value) {
  191. VERIFY(value.is_object());
  192. if (auto maybe_valid_types = value.as_object().get_array("valid-types"sv); maybe_valid_types.has_value() && !maybe_valid_types->is_empty()) {
  193. for (auto valid_type : maybe_valid_types->values()) {
  194. auto type_and_range = valid_type.as_string().split_view(' ');
  195. if (type_and_range.first() != css_type_name)
  196. continue;
  197. auto property_generator = generator.fork();
  198. property_generator.set("property_name:titlecase", title_casify(name));
  199. property_generator.append(R"~~~(
  200. case PropertyID::@property_name:titlecase@:
  201. return )~~~");
  202. if (type_and_range.size() > 1) {
  203. auto range = type_and_range[1];
  204. VERIFY(range.starts_with('[') && range.ends_with(']') && range.contains(','));
  205. auto comma_index = range.find(',').value();
  206. StringView min_value_string = range.substring_view(1, comma_index - 1);
  207. StringView max_value_string = range.substring_view(comma_index + 1, range.length() - comma_index - 2);
  208. // If the min/max value is infinite, we can just skip that side of the check.
  209. if (min_value_string == "-∞")
  210. min_value_string = {};
  211. if (max_value_string == "∞")
  212. max_value_string = {};
  213. if (min_value_string.is_empty() && max_value_string.is_empty()) {
  214. property_generator.appendln("true;");
  215. break;
  216. }
  217. auto output_check = [&](auto& value_string, StringView comparator) {
  218. if (value_getter.has_value()) {
  219. property_generator.set("value_number", value_string);
  220. property_generator.set("value_getter", value_getter.value());
  221. property_generator.set("comparator", comparator);
  222. property_generator.append("@value_getter@ @comparator@ @value_number@");
  223. return;
  224. }
  225. GenericLexer lexer { value_string };
  226. auto value_number = lexer.consume_until(is_ascii_alpha);
  227. auto value_unit = lexer.consume_while(is_ascii_alpha);
  228. if (value_unit.is_empty())
  229. value_unit = default_unit_name.value();
  230. VERIFY(lexer.is_eof());
  231. property_generator.set("value_number", value_number);
  232. property_generator.set("value_unit", title_casify(value_unit));
  233. property_generator.set("comparator", comparator);
  234. property_generator.append("value @comparator@ @type_name@(@value_number@, @type_name@::Type::@value_unit@)");
  235. };
  236. if (!min_value_string.is_empty())
  237. output_check(min_value_string, ">="sv);
  238. if (!min_value_string.is_empty() && !max_value_string.is_empty())
  239. property_generator.append(" && ");
  240. if (!max_value_string.is_empty())
  241. output_check(max_value_string, "<="sv);
  242. property_generator.appendln(";");
  243. } else {
  244. property_generator.appendln("true;");
  245. }
  246. break;
  247. }
  248. }
  249. });
  250. generator.append(R"~~~(
  251. default:
  252. return false;
  253. }
  254. }
  255. )~~~");
  256. }
  257. ErrorOr<void> generate_implementation_file(JsonObject& properties, Core::File& file)
  258. {
  259. StringBuilder builder;
  260. SourceGenerator generator { builder };
  261. generator.append(R"~~~(
  262. #include <AK/Assertions.h>
  263. #include <LibWeb/CSS/Enums.h>
  264. #include <LibWeb/CSS/Parser/Parser.h>
  265. #include <LibWeb/CSS/PropertyID.h>
  266. #include <LibWeb/CSS/StyleValue.h>
  267. #include <LibWeb/CSS/StyleValues/PercentageStyleValue.h>
  268. #include <LibWeb/CSS/StyleValues/TimeStyleValue.h>
  269. #include <LibWeb/Infra/Strings.h>
  270. namespace Web::CSS {
  271. Optional<PropertyID> property_id_from_camel_case_string(StringView string)
  272. {
  273. )~~~");
  274. properties.for_each_member([&](auto& name, auto& value) {
  275. VERIFY(value.is_object());
  276. auto member_generator = generator.fork();
  277. member_generator.set("name", name);
  278. member_generator.set("name:titlecase", title_casify(name));
  279. member_generator.set("name:camelcase", camel_casify(name));
  280. member_generator.append(R"~~~(
  281. if (string.equals_ignoring_ascii_case("@name:camelcase@"sv))
  282. return PropertyID::@name:titlecase@;
  283. )~~~");
  284. });
  285. generator.append(R"~~~(
  286. return {};
  287. }
  288. Optional<PropertyID> property_id_from_string(StringView string)
  289. {
  290. )~~~");
  291. properties.for_each_member([&](auto& name, auto& value) {
  292. VERIFY(value.is_object());
  293. auto member_generator = generator.fork();
  294. member_generator.set("name", name);
  295. member_generator.set("name:titlecase", title_casify(name));
  296. member_generator.append(R"~~~(
  297. if (Infra::is_ascii_case_insensitive_match(string, "@name@"sv))
  298. return PropertyID::@name:titlecase@;
  299. )~~~");
  300. });
  301. generator.append(R"~~~(
  302. return {};
  303. }
  304. StringView string_from_property_id(PropertyID property_id) {
  305. switch (property_id) {
  306. )~~~");
  307. properties.for_each_member([&](auto& name, auto& value) {
  308. VERIFY(value.is_object());
  309. auto member_generator = generator.fork();
  310. member_generator.set("name", name);
  311. member_generator.set("name:titlecase", title_casify(name));
  312. member_generator.append(R"~~~(
  313. case PropertyID::@name:titlecase@:
  314. return "@name@"sv;
  315. )~~~");
  316. });
  317. generator.append(R"~~~(
  318. default:
  319. return "(invalid CSS::PropertyID)"sv;
  320. }
  321. }
  322. bool is_inherited_property(PropertyID property_id)
  323. {
  324. switch (property_id) {
  325. )~~~");
  326. properties.for_each_member([&](auto& name, auto& value) {
  327. VERIFY(value.is_object());
  328. bool inherited = false;
  329. if (value.as_object().has("inherited"sv)) {
  330. auto inherited_value = value.as_object().get_bool("inherited"sv);
  331. VERIFY(inherited_value.has_value());
  332. inherited = inherited_value.value();
  333. }
  334. if (inherited) {
  335. auto member_generator = generator.fork();
  336. member_generator.set("name:titlecase", title_casify(name));
  337. member_generator.append(R"~~~(
  338. case PropertyID::@name:titlecase@:
  339. return true;
  340. )~~~");
  341. }
  342. });
  343. generator.append(R"~~~(
  344. default:
  345. return false;
  346. }
  347. }
  348. bool property_affects_layout(PropertyID property_id)
  349. {
  350. switch (property_id) {
  351. )~~~");
  352. properties.for_each_member([&](auto& name, auto& value) {
  353. VERIFY(value.is_object());
  354. bool affects_layout = true;
  355. if (value.as_object().has("affects-layout"sv))
  356. affects_layout = value.as_object().get_bool("affects-layout"sv).value_or(false);
  357. if (affects_layout) {
  358. auto member_generator = generator.fork();
  359. member_generator.set("name:titlecase", title_casify(name));
  360. member_generator.append(R"~~~(
  361. case PropertyID::@name:titlecase@:
  362. )~~~");
  363. }
  364. });
  365. generator.append(R"~~~(
  366. return true;
  367. default:
  368. return false;
  369. }
  370. }
  371. bool property_affects_stacking_context(PropertyID property_id)
  372. {
  373. switch (property_id) {
  374. )~~~");
  375. properties.for_each_member([&](auto& name, auto& value) {
  376. VERIFY(value.is_object());
  377. bool affects_stacking_context = false;
  378. if (value.as_object().has("affects-stacking-context"sv))
  379. affects_stacking_context = value.as_object().get_bool("affects-stacking-context"sv).value_or(false);
  380. if (affects_stacking_context) {
  381. auto member_generator = generator.fork();
  382. member_generator.set("name:titlecase", title_casify(name));
  383. member_generator.append(R"~~~(
  384. case PropertyID::@name:titlecase@:
  385. )~~~");
  386. }
  387. });
  388. generator.append(R"~~~(
  389. return true;
  390. default:
  391. return false;
  392. }
  393. }
  394. ErrorOr<NonnullRefPtr<StyleValue>> property_initial_value(JS::Realm& context_realm, PropertyID property_id)
  395. {
  396. static Array<RefPtr<StyleValue>, to_underlying(last_property_id) + 1> initial_values;
  397. if (auto initial_value = initial_values[to_underlying(property_id)])
  398. return initial_value.release_nonnull();
  399. // Lazily parse initial values as needed.
  400. // This ensures the shorthands will always be able to get the initial values of their longhands.
  401. // This also now allows a longhand have its own longhand (like background-position-x).
  402. Parser::ParsingContext parsing_context(context_realm);
  403. switch (property_id) {
  404. )~~~");
  405. auto output_initial_value_code = [&](auto& name, auto& object) {
  406. if (!object.has("initial"sv)) {
  407. dbgln("No initial value specified for property '{}'", name);
  408. VERIFY_NOT_REACHED();
  409. }
  410. auto initial_value = object.get_deprecated_string("initial"sv);
  411. VERIFY(initial_value.has_value());
  412. auto& initial_value_string = initial_value.value();
  413. auto member_generator = generator.fork();
  414. member_generator.set("name:titlecase", title_casify(name));
  415. member_generator.set("initial_value_string", initial_value_string);
  416. member_generator.append(
  417. R"~~~( case PropertyID::@name:titlecase@:
  418. {
  419. auto parsed_value = TRY(parse_css_value(parsing_context, "@initial_value_string@"sv, PropertyID::@name:titlecase@));
  420. VERIFY(!parsed_value.is_null());
  421. auto initial_value = parsed_value.release_nonnull();
  422. initial_values[to_underlying(PropertyID::@name:titlecase@)] = initial_value;
  423. return initial_value;
  424. }
  425. )~~~");
  426. };
  427. properties.for_each_member([&](auto& name, auto& value) {
  428. VERIFY(value.is_object());
  429. output_initial_value_code(name, value.as_object());
  430. });
  431. generator.append(
  432. R"~~~( default: VERIFY_NOT_REACHED();
  433. }
  434. VERIFY_NOT_REACHED();
  435. }
  436. bool property_has_quirk(PropertyID property_id, Quirk quirk)
  437. {
  438. switch (property_id) {
  439. )~~~");
  440. properties.for_each_member([&](auto& name, auto& value) {
  441. VERIFY(value.is_object());
  442. if (value.as_object().has("quirks"sv)) {
  443. auto quirks_value = value.as_object().get_array("quirks"sv);
  444. VERIFY(quirks_value.has_value());
  445. auto& quirks = quirks_value.value();
  446. if (!quirks.is_empty()) {
  447. auto property_generator = generator.fork();
  448. property_generator.set("name:titlecase", title_casify(name));
  449. property_generator.append(R"~~~(
  450. case PropertyID::@name:titlecase@: {
  451. switch (quirk) {
  452. )~~~");
  453. for (auto& quirk : quirks.values()) {
  454. VERIFY(quirk.is_string());
  455. auto quirk_generator = property_generator.fork();
  456. quirk_generator.set("quirk:titlecase", title_casify(quirk.as_string()));
  457. quirk_generator.append(R"~~~(
  458. case Quirk::@quirk:titlecase@:
  459. return true;
  460. )~~~");
  461. }
  462. property_generator.append(R"~~~(
  463. default:
  464. return false;
  465. }
  466. }
  467. )~~~");
  468. }
  469. }
  470. });
  471. generator.append(R"~~~(
  472. default:
  473. return false;
  474. }
  475. }
  476. bool property_accepts_type(PropertyID property_id, ValueType value_type)
  477. {
  478. switch (property_id) {
  479. )~~~");
  480. properties.for_each_member([&](auto& name, auto& value) {
  481. VERIFY(value.is_object());
  482. auto& object = value.as_object();
  483. if (auto maybe_valid_types = object.get_array("valid-types"sv); maybe_valid_types.has_value() && !maybe_valid_types->is_empty()) {
  484. auto& valid_types = maybe_valid_types.value();
  485. auto property_generator = generator.fork();
  486. property_generator.set("name:titlecase", title_casify(name));
  487. property_generator.append(R"~~~(
  488. case PropertyID::@name:titlecase@: {
  489. switch (value_type) {
  490. )~~~");
  491. bool did_output_accepted_type = false;
  492. for (auto& type : valid_types.values()) {
  493. VERIFY(type.is_string());
  494. auto type_name = type.as_string().split_view(' ').first();
  495. if (type_name_is_enum(type_name))
  496. continue;
  497. if (type_name == "angle") {
  498. property_generator.appendln(" case ValueType::Angle:");
  499. } else if (type_name == "color") {
  500. property_generator.appendln(" case ValueType::Color:");
  501. } else if (type_name == "custom-ident") {
  502. property_generator.appendln(" case ValueType::CustomIdent:");
  503. } else if (type_name == "frequency") {
  504. property_generator.appendln(" case ValueType::Frequency:");
  505. } else if (type_name == "image") {
  506. property_generator.appendln(" case ValueType::Image:");
  507. } else if (type_name == "integer") {
  508. property_generator.appendln(" case ValueType::Integer:");
  509. } else if (type_name == "length") {
  510. property_generator.appendln(" case ValueType::Length:");
  511. } else if (type_name == "number") {
  512. property_generator.appendln(" case ValueType::Number:");
  513. } else if (type_name == "percentage") {
  514. property_generator.appendln(" case ValueType::Percentage:");
  515. } else if (type_name == "rect") {
  516. property_generator.appendln(" case ValueType::Rect:");
  517. } else if (type_name == "resolution") {
  518. property_generator.appendln(" case ValueType::Resolution:");
  519. } else if (type_name == "string") {
  520. property_generator.appendln(" case ValueType::String:");
  521. } else if (type_name == "time") {
  522. property_generator.appendln(" case ValueType::Time:");
  523. } else if (type_name == "url") {
  524. property_generator.appendln(" case ValueType::Url:");
  525. } else {
  526. VERIFY_NOT_REACHED();
  527. }
  528. did_output_accepted_type = true;
  529. }
  530. if (did_output_accepted_type)
  531. property_generator.appendln(" return true;");
  532. property_generator.append(R"~~~(
  533. default:
  534. return false;
  535. }
  536. }
  537. )~~~");
  538. }
  539. });
  540. generator.append(R"~~~(
  541. default:
  542. return false;
  543. }
  544. }
  545. bool property_accepts_identifier(PropertyID property_id, ValueID identifier)
  546. {
  547. switch (property_id) {
  548. )~~~");
  549. properties.for_each_member([&](auto& name, auto& value) {
  550. VERIFY(value.is_object());
  551. auto& object = value.as_object();
  552. auto property_generator = generator.fork();
  553. property_generator.set("name:titlecase", title_casify(name));
  554. property_generator.appendln(" case PropertyID::@name:titlecase@: {");
  555. if (auto maybe_valid_identifiers = object.get_array("valid-identifiers"sv); maybe_valid_identifiers.has_value() && !maybe_valid_identifiers->is_empty()) {
  556. property_generator.appendln(" switch (identifier) {");
  557. auto& valid_identifiers = maybe_valid_identifiers.value();
  558. for (auto& identifier : valid_identifiers.values()) {
  559. auto identifier_generator = generator.fork();
  560. identifier_generator.set("identifier:titlecase", title_casify(identifier.as_string()));
  561. identifier_generator.appendln(" case ValueID::@identifier:titlecase@:");
  562. }
  563. property_generator.append(R"~~~(
  564. return true;
  565. default:
  566. break;
  567. }
  568. )~~~");
  569. }
  570. if (auto maybe_valid_types = object.get_array("valid-types"sv); maybe_valid_types.has_value() && !maybe_valid_types->is_empty()) {
  571. auto& valid_types = maybe_valid_types.value();
  572. for (auto& valid_type : valid_types.values()) {
  573. auto type_name = valid_type.as_string().split_view(' ').first();
  574. if (!type_name_is_enum(type_name))
  575. continue;
  576. auto type_generator = generator.fork();
  577. type_generator.set("type_name:snakecase", snake_casify(type_name));
  578. type_generator.append(R"~~~(
  579. if (value_id_to_@type_name:snakecase@(identifier).has_value())
  580. return true;
  581. )~~~");
  582. }
  583. }
  584. property_generator.append(R"~~~(
  585. return false;
  586. }
  587. )~~~");
  588. });
  589. generator.append(R"~~~(
  590. default:
  591. return false;
  592. }
  593. }
  594. size_t property_maximum_value_count(PropertyID property_id)
  595. {
  596. switch (property_id) {
  597. )~~~");
  598. properties.for_each_member([&](auto& name, auto& value) {
  599. VERIFY(value.is_object());
  600. if (value.as_object().has("max-values"sv)) {
  601. auto max_values = value.as_object().get("max-values"sv);
  602. VERIFY(max_values.has_value() && max_values->is_number() && !max_values->is_double());
  603. auto property_generator = generator.fork();
  604. property_generator.set("name:titlecase", title_casify(name));
  605. property_generator.set("max_values", max_values->to_deprecated_string());
  606. property_generator.append(R"~~~(
  607. case PropertyID::@name:titlecase@:
  608. return @max_values@;
  609. )~~~");
  610. }
  611. });
  612. generator.append(R"~~~(
  613. default:
  614. return 1;
  615. }
  616. })~~~");
  617. generate_bounds_checking_function(properties, generator, "angle"sv, "Angle"sv, "Deg"sv);
  618. generate_bounds_checking_function(properties, generator, "frequency"sv, "Frequency"sv, "Hertz"sv);
  619. generate_bounds_checking_function(properties, generator, "integer"sv, "i64"sv, {}, "value"sv);
  620. generate_bounds_checking_function(properties, generator, "length"sv, "Length"sv, {}, "value.raw_value()"sv);
  621. generate_bounds_checking_function(properties, generator, "number"sv, "double"sv, {}, "value"sv);
  622. generate_bounds_checking_function(properties, generator, "percentage"sv, "Percentage"sv, {}, "value.value()"sv);
  623. generate_bounds_checking_function(properties, generator, "resolution"sv, "Resolution"sv, "Dpi"sv);
  624. generate_bounds_checking_function(properties, generator, "time"sv, "Time"sv, "S"sv);
  625. generator.append(R"~~~(
  626. Vector<PropertyID> longhands_for_shorthand(PropertyID property_id)
  627. {
  628. switch (property_id) {
  629. )~~~");
  630. properties.for_each_member([&](auto& name, auto& value) {
  631. if (value.as_object().has("longhands"sv)) {
  632. auto longhands = value.as_object().get("longhands"sv);
  633. VERIFY(longhands.has_value() && longhands->is_array());
  634. auto longhand_values = longhands->as_array();
  635. auto property_generator = generator.fork();
  636. property_generator.set("name:titlecase", title_casify(name));
  637. StringBuilder builder;
  638. bool first = true;
  639. longhand_values.for_each([&](auto& longhand) {
  640. if (first)
  641. first = false;
  642. else
  643. builder.append(", "sv);
  644. builder.appendff("PropertyID::{}", title_casify(longhand.to_deprecated_string()));
  645. return IterationDecision::Continue;
  646. });
  647. property_generator.set("longhands", builder.to_deprecated_string());
  648. property_generator.append(R"~~~(
  649. case PropertyID::@name:titlecase@:
  650. return { @longhands@ };
  651. )~~~");
  652. }
  653. });
  654. generator.append(R"~~~(
  655. default:
  656. return { };
  657. }
  658. }
  659. )~~~");
  660. generator.append(R"~~~(
  661. } // namespace Web::CSS
  662. )~~~");
  663. TRY(file.write_until_depleted(generator.as_string_view().bytes()));
  664. return {};
  665. }