GenerateCSSPropertyID.cpp 32 KB

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