GenerateCSSPropertyID.cpp 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592
  1. /*
  2. * Copyright (c) 2018-2020, Andreas Kling <kling@serenityos.org>
  3. * Copyright (c) 2021-2022, Sam Atkins <atkinssj@serenityos.org>
  4. *
  5. * SPDX-License-Identifier: BSD-2-Clause
  6. */
  7. #include "GeneratorUtil.h"
  8. #include <AK/SourceGenerator.h>
  9. #include <AK/StringBuilder.h>
  10. #include <LibCore/ArgsParser.h>
  11. #include <LibMain/Main.h>
  12. ErrorOr<void> generate_header_file(JsonObject& properties, Core::Stream::File& file);
  13. ErrorOr<void> generate_implementation_file(JsonObject& properties, Core::Stream::File& file);
  14. ErrorOr<int> serenity_main(Main::Arguments arguments)
  15. {
  16. StringView generated_header_path;
  17. StringView generated_implementation_path;
  18. StringView properties_json_path;
  19. Core::ArgsParser args_parser;
  20. args_parser.add_option(generated_header_path, "Path to the PropertyID header file to generate", "generated-header-path", 'h', "generated-header-path");
  21. args_parser.add_option(generated_implementation_path, "Path to the PropertyID implementation file to generate", "generated-implementation-path", 'c', "generated-implementation-path");
  22. args_parser.add_option(properties_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(properties_json_path));
  25. VERIFY(json.is_object());
  26. auto properties = json.as_object();
  27. auto generated_header_file = TRY(Core::Stream::File::open(generated_header_path, Core::Stream::OpenMode::Write));
  28. auto generated_implementation_file = TRY(Core::Stream::File::open(generated_implementation_path, Core::Stream::OpenMode::Write));
  29. TRY(generate_header_file(properties, *generated_header_file));
  30. TRY(generate_implementation_file(properties, *generated_implementation_file));
  31. return 0;
  32. }
  33. ErrorOr<void> generate_header_file(JsonObject& properties, Core::Stream::File& file)
  34. {
  35. StringBuilder builder;
  36. SourceGenerator generator { builder };
  37. generator.append(R"~~~(
  38. #pragma once
  39. #include <AK/NonnullRefPtr.h>
  40. #include <AK/StringView.h>
  41. #include <AK/Traits.h>
  42. #include <LibWeb/Forward.h>
  43. namespace Web::CSS {
  44. enum class PropertyID {
  45. Invalid,
  46. Custom,
  47. )~~~");
  48. Vector<String> shorthand_property_ids;
  49. Vector<String> longhand_property_ids;
  50. properties.for_each_member([&](auto& name, auto& value) {
  51. VERIFY(value.is_object());
  52. if (value.as_object().has("longhands"))
  53. shorthand_property_ids.append(name);
  54. else
  55. longhand_property_ids.append(name);
  56. });
  57. auto first_property_id = shorthand_property_ids.first();
  58. auto last_property_id = longhand_property_ids.last();
  59. for (auto& name : shorthand_property_ids) {
  60. auto member_generator = generator.fork();
  61. member_generator.set("name:titlecase", title_casify(name));
  62. member_generator.append(R"~~~(
  63. @name:titlecase@,
  64. )~~~");
  65. }
  66. for (auto& name : longhand_property_ids) {
  67. auto member_generator = generator.fork();
  68. member_generator.set("name:titlecase", title_casify(name));
  69. member_generator.append(R"~~~(
  70. @name:titlecase@,
  71. )~~~");
  72. }
  73. generator.set("first_property_id", title_casify(first_property_id));
  74. generator.set("last_property_id", title_casify(last_property_id));
  75. generator.set("first_shorthand_property_id", title_casify(shorthand_property_ids.first()));
  76. generator.set("last_shorthand_property_id", title_casify(shorthand_property_ids.last()));
  77. generator.set("first_longhand_property_id", title_casify(longhand_property_ids.first()));
  78. generator.set("last_longhand_property_id", title_casify(longhand_property_ids.last()));
  79. generator.append(R"~~~(
  80. };
  81. PropertyID property_id_from_camel_case_string(StringView);
  82. PropertyID property_id_from_string(StringView);
  83. const char* string_from_property_id(PropertyID);
  84. bool is_inherited_property(PropertyID);
  85. NonnullRefPtr<StyleValue> property_initial_value(PropertyID);
  86. bool property_accepts_value(PropertyID, StyleValue&);
  87. size_t property_maximum_value_count(PropertyID);
  88. bool property_affects_layout(PropertyID);
  89. bool property_affects_stacking_context(PropertyID);
  90. constexpr PropertyID first_property_id = PropertyID::@first_property_id@;
  91. constexpr PropertyID last_property_id = PropertyID::@last_property_id@;
  92. constexpr PropertyID first_shorthand_property_id = PropertyID::@first_shorthand_property_id@;
  93. constexpr PropertyID last_shorthand_property_id = PropertyID::@last_shorthand_property_id@;
  94. constexpr PropertyID first_longhand_property_id = PropertyID::@first_longhand_property_id@;
  95. constexpr PropertyID last_longhand_property_id = PropertyID::@last_longhand_property_id@;
  96. enum class Quirk {
  97. // https://quirks.spec.whatwg.org/#the-hashless-hex-color-quirk
  98. HashlessHexColor,
  99. // https://quirks.spec.whatwg.org/#the-unitless-length-quirk
  100. UnitlessLength,
  101. };
  102. bool property_has_quirk(PropertyID, Quirk);
  103. } // namespace Web::CSS
  104. namespace AK {
  105. template<>
  106. struct Traits<Web::CSS::PropertyID> : public GenericTraits<Web::CSS::PropertyID> {
  107. static unsigned hash(Web::CSS::PropertyID property_id) { return int_hash((unsigned)property_id); }
  108. };
  109. } // namespace AK
  110. )~~~");
  111. TRY(file.write(generator.as_string_view().bytes()));
  112. return {};
  113. }
  114. ErrorOr<void> generate_implementation_file(JsonObject& properties, Core::Stream::File& file)
  115. {
  116. StringBuilder builder;
  117. SourceGenerator generator { builder };
  118. generator.append(R"~~~(
  119. #include <AK/Assertions.h>
  120. #include <LibWeb/CSS/Parser/Parser.h>
  121. #include <LibWeb/CSS/PropertyID.h>
  122. #include <LibWeb/CSS/StyleValue.h>
  123. namespace Web::CSS {
  124. PropertyID property_id_from_camel_case_string(StringView string)
  125. {
  126. )~~~");
  127. properties.for_each_member([&](auto& name, auto& value) {
  128. VERIFY(value.is_object());
  129. auto member_generator = generator.fork();
  130. member_generator.set("name", name);
  131. member_generator.set("name:titlecase", title_casify(name));
  132. member_generator.set("name:camelcase", camel_casify(name));
  133. member_generator.append(R"~~~(
  134. if (string.equals_ignoring_case("@name:camelcase@"sv))
  135. return PropertyID::@name:titlecase@;
  136. )~~~");
  137. });
  138. generator.append(R"~~~(
  139. return PropertyID::Invalid;
  140. }
  141. PropertyID property_id_from_string(StringView string)
  142. {
  143. )~~~");
  144. properties.for_each_member([&](auto& name, auto& value) {
  145. VERIFY(value.is_object());
  146. auto member_generator = generator.fork();
  147. member_generator.set("name", name);
  148. member_generator.set("name:titlecase", title_casify(name));
  149. member_generator.append(R"~~~(
  150. if (string.equals_ignoring_case("@name@"))
  151. return PropertyID::@name:titlecase@;
  152. )~~~");
  153. });
  154. generator.append(R"~~~(
  155. return PropertyID::Invalid;
  156. }
  157. const char* string_from_property_id(PropertyID property_id) {
  158. switch (property_id) {
  159. )~~~");
  160. properties.for_each_member([&](auto& name, auto& value) {
  161. VERIFY(value.is_object());
  162. auto member_generator = generator.fork();
  163. member_generator.set("name", name);
  164. member_generator.set("name:titlecase", title_casify(name));
  165. member_generator.append(R"~~~(
  166. case PropertyID::@name:titlecase@:
  167. return "@name@";
  168. )~~~");
  169. });
  170. generator.append(R"~~~(
  171. default:
  172. return "(invalid CSS::PropertyID)";
  173. }
  174. }
  175. bool is_inherited_property(PropertyID property_id)
  176. {
  177. switch (property_id) {
  178. )~~~");
  179. properties.for_each_member([&](auto& name, auto& value) {
  180. VERIFY(value.is_object());
  181. bool inherited = false;
  182. if (value.as_object().has("inherited")) {
  183. auto& inherited_value = value.as_object().get("inherited");
  184. VERIFY(inherited_value.is_bool());
  185. inherited = inherited_value.as_bool();
  186. }
  187. if (inherited) {
  188. auto member_generator = generator.fork();
  189. member_generator.set("name:titlecase", title_casify(name));
  190. member_generator.append(R"~~~(
  191. case PropertyID::@name:titlecase@:
  192. return true;
  193. )~~~");
  194. }
  195. });
  196. generator.append(R"~~~(
  197. default:
  198. return false;
  199. }
  200. }
  201. bool property_affects_layout(PropertyID property_id)
  202. {
  203. switch (property_id) {
  204. )~~~");
  205. properties.for_each_member([&](auto& name, auto& value) {
  206. VERIFY(value.is_object());
  207. bool affects_layout = true;
  208. if (value.as_object().has("affects-layout"))
  209. affects_layout = value.as_object().get("affects-layout").to_bool();
  210. if (affects_layout) {
  211. auto member_generator = generator.fork();
  212. member_generator.set("name:titlecase", title_casify(name));
  213. member_generator.append(R"~~~(
  214. case PropertyID::@name:titlecase@:
  215. )~~~");
  216. }
  217. });
  218. generator.append(R"~~~(
  219. return true;
  220. default:
  221. return false;
  222. }
  223. }
  224. bool property_affects_stacking_context(PropertyID property_id)
  225. {
  226. switch (property_id) {
  227. )~~~");
  228. properties.for_each_member([&](auto& name, auto& value) {
  229. VERIFY(value.is_object());
  230. bool affects_stacking_context = false;
  231. if (value.as_object().has("affects-stacking-context"))
  232. affects_stacking_context = value.as_object().get("affects-stacking-context").to_bool();
  233. if (affects_stacking_context) {
  234. auto member_generator = generator.fork();
  235. member_generator.set("name:titlecase", title_casify(name));
  236. member_generator.append(R"~~~(
  237. case PropertyID::@name:titlecase@:
  238. )~~~");
  239. }
  240. });
  241. generator.append(R"~~~(
  242. return true;
  243. default:
  244. return false;
  245. }
  246. }
  247. NonnullRefPtr<StyleValue> property_initial_value(PropertyID property_id)
  248. {
  249. static Array<RefPtr<StyleValue>, to_underlying(last_property_id) + 1> initial_values;
  250. static bool initialized = false;
  251. if (!initialized) {
  252. initialized = true;
  253. Parser::ParsingContext parsing_context;
  254. )~~~");
  255. // NOTE: Parsing a shorthand property requires that its longhands are already available here.
  256. // So, we do this in two passes: First longhands, then shorthands.
  257. // Probably we should build a dependency graph and then handle them in order, but this
  258. // works for now! :^)
  259. auto output_initial_value_code = [&](auto& name, auto& object) {
  260. if (!object.has("initial")) {
  261. dbgln("No initial value specified for property '{}'", name);
  262. VERIFY_NOT_REACHED();
  263. }
  264. auto& initial_value = object.get("initial");
  265. VERIFY(initial_value.is_string());
  266. auto initial_value_string = initial_value.as_string();
  267. auto member_generator = generator.fork();
  268. member_generator.set("name:titlecase", title_casify(name));
  269. member_generator.set("initial_value_string", initial_value_string);
  270. member_generator.append(R"~~~(
  271. {
  272. auto parsed_value = parse_css_value(parsing_context, "@initial_value_string@", PropertyID::@name:titlecase@);
  273. VERIFY(!parsed_value.is_null());
  274. initial_values[to_underlying(PropertyID::@name:titlecase@)] = parsed_value.release_nonnull();
  275. }
  276. )~~~");
  277. };
  278. properties.for_each_member([&](auto& name, auto& value) {
  279. VERIFY(value.is_object());
  280. if (value.as_object().has("longhands"))
  281. return;
  282. output_initial_value_code(name, value.as_object());
  283. });
  284. properties.for_each_member([&](auto& name, auto& value) {
  285. VERIFY(value.is_object());
  286. if (!value.as_object().has("longhands"))
  287. return;
  288. output_initial_value_code(name, value.as_object());
  289. });
  290. generator.append(R"~~~(
  291. }
  292. return *initial_values[to_underlying(property_id)];
  293. }
  294. bool property_has_quirk(PropertyID property_id, Quirk quirk)
  295. {
  296. switch (property_id) {
  297. )~~~");
  298. properties.for_each_member([&](auto& name, auto& value) {
  299. VERIFY(value.is_object());
  300. if (value.as_object().has("quirks")) {
  301. auto& quirks_value = value.as_object().get("quirks");
  302. VERIFY(quirks_value.is_array());
  303. auto& quirks = quirks_value.as_array();
  304. if (!quirks.is_empty()) {
  305. auto property_generator = generator.fork();
  306. property_generator.set("name:titlecase", title_casify(name));
  307. property_generator.append(R"~~~(
  308. case PropertyID::@name:titlecase@: {
  309. switch (quirk) {
  310. )~~~");
  311. for (auto& quirk : quirks.values()) {
  312. VERIFY(quirk.is_string());
  313. auto quirk_generator = property_generator.fork();
  314. quirk_generator.set("quirk:titlecase", title_casify(quirk.as_string()));
  315. quirk_generator.append(R"~~~(
  316. case Quirk::@quirk:titlecase@:
  317. return true;
  318. )~~~");
  319. }
  320. property_generator.append(R"~~~(
  321. default:
  322. return false;
  323. }
  324. }
  325. )~~~");
  326. }
  327. }
  328. });
  329. generator.append(R"~~~(
  330. default:
  331. return false;
  332. }
  333. }
  334. bool property_accepts_value(PropertyID property_id, StyleValue& style_value)
  335. {
  336. if (style_value.is_builtin())
  337. return true;
  338. switch (property_id) {
  339. )~~~");
  340. properties.for_each_member([&](auto& name, auto& value) {
  341. VERIFY(value.is_object());
  342. auto& object = value.as_object();
  343. bool has_valid_types = object.has("valid-types");
  344. auto has_valid_identifiers = object.has("valid-identifiers");
  345. if (has_valid_types || has_valid_identifiers) {
  346. auto property_generator = generator.fork();
  347. property_generator.set("name:titlecase", title_casify(name));
  348. property_generator.append(R"~~~(
  349. case PropertyID::@name:titlecase@: {
  350. )~~~");
  351. auto output_numeric_value_check = [](SourceGenerator& generator, StringView type_check_function, StringView value_getter, Span<StringView> resolved_type_names, StringView min_value, StringView max_value) {
  352. auto test_generator = generator.fork();
  353. test_generator.set("type_check_function", type_check_function);
  354. test_generator.set("value_getter", value_getter);
  355. test_generator.append(R"~~~(
  356. if ((style_value.@type_check_function@())~~~");
  357. if (!min_value.is_empty() && min_value != "-∞") {
  358. test_generator.set("minvalue", min_value);
  359. test_generator.append(" && (style_value.@value_getter@ >= @minvalue@)");
  360. }
  361. if (!max_value.is_empty() && max_value != "∞") {
  362. test_generator.set("maxvalue", max_value);
  363. test_generator.append(" && (style_value.@value_getter@ <= @maxvalue@)");
  364. }
  365. test_generator.append(")");
  366. if (!resolved_type_names.is_empty()) {
  367. test_generator.append(R"~~~(
  368. || (style_value.is_calculated() && ()~~~");
  369. bool first = true;
  370. for (auto& type_name : resolved_type_names) {
  371. test_generator.set("resolved_type_name", type_name);
  372. if (!first)
  373. test_generator.append(" || ");
  374. test_generator.append("style_value.as_calculated().resolved_type() == CalculatedStyleValue::ResolvedType::@resolved_type_name@");
  375. first = false;
  376. }
  377. test_generator.append("))");
  378. }
  379. test_generator.append(R"~~~() {
  380. return true;
  381. }
  382. )~~~");
  383. };
  384. if (has_valid_types) {
  385. auto valid_types_value = object.get("valid-types");
  386. VERIFY(valid_types_value.is_array());
  387. auto valid_types = valid_types_value.as_array();
  388. if (!valid_types.is_empty()) {
  389. for (auto& type : valid_types.values()) {
  390. VERIFY(type.is_string());
  391. auto type_parts = type.as_string().split_view(' ');
  392. auto type_name = type_parts.first();
  393. auto type_args = type_parts.size() > 1 ? type_parts[1] : ""sv;
  394. StringView min_value;
  395. StringView max_value;
  396. if (!type_args.is_empty()) {
  397. VERIFY(type_args.starts_with('[') && type_args.ends_with(']'));
  398. auto comma_index = type_args.find(',').value();
  399. min_value = type_args.substring_view(1, comma_index - 1);
  400. max_value = type_args.substring_view(comma_index + 1, type_args.length() - comma_index - 2);
  401. }
  402. if (type_name == "angle") {
  403. output_numeric_value_check(property_generator, "is_angle", "as_angle().angle().to_degrees()", Array { "Angle"sv }, min_value, max_value);
  404. } else if (type_name == "color") {
  405. property_generator.append(R"~~~(
  406. if (style_value.has_color())
  407. return true;
  408. )~~~");
  409. } else if (type_name == "frequency") {
  410. output_numeric_value_check(property_generator, "is_frequency", "as_frequency().frequency().to_hertz()", Array { "Frequency"sv }, min_value, max_value);
  411. } else if (type_name == "image") {
  412. property_generator.append(R"~~~(
  413. if (style_value.is_image())
  414. return true;
  415. )~~~");
  416. } else if (type_name == "integer") {
  417. output_numeric_value_check(property_generator, "has_integer", "to_integer()", Array { "Integer"sv }, min_value, max_value);
  418. } else if (type_name == "length") {
  419. output_numeric_value_check(property_generator, "has_length", "to_length().raw_value()", Array { "Length"sv }, min_value, max_value);
  420. } else if (type_name == "number") {
  421. output_numeric_value_check(property_generator, "has_number", "to_number()", Array { "Integer"sv, "Number"sv }, min_value, max_value);
  422. } else if (type_name == "percentage") {
  423. output_numeric_value_check(property_generator, "is_percentage", "as_percentage().percentage().value()", Array { "Percentage"sv }, min_value, max_value);
  424. } else if (type_name == "resolution") {
  425. output_numeric_value_check(property_generator, "is_resolution", "as_resolution().resolution().to_dots_per_pixel()", Array<StringView, 0> {}, min_value, max_value);
  426. } else if (type_name == "string") {
  427. property_generator.append(R"~~~(
  428. if (style_value.is_string())
  429. return true;
  430. )~~~");
  431. } else if (type_name == "time") {
  432. output_numeric_value_check(property_generator, "is_time", "as_time().time().to_seconds()", Array { "Time"sv }, min_value, max_value);
  433. } else if (type_name == "url") {
  434. // FIXME: Handle urls!
  435. } else {
  436. warnln("Unrecognized valid-type name: '{}'", type_name);
  437. VERIFY_NOT_REACHED();
  438. }
  439. }
  440. }
  441. }
  442. if (has_valid_identifiers) {
  443. auto valid_identifiers_value = object.get("valid-identifiers");
  444. VERIFY(valid_identifiers_value.is_array());
  445. auto valid_identifiers = valid_identifiers_value.as_array();
  446. if (!valid_identifiers.is_empty()) {
  447. property_generator.append(R"~~~(
  448. switch (style_value.to_identifier()) {
  449. )~~~");
  450. for (auto& identifier : valid_identifiers.values()) {
  451. VERIFY(identifier.is_string());
  452. auto identifier_generator = generator.fork();
  453. identifier_generator.set("identifier:titlecase", title_casify(identifier.as_string()));
  454. identifier_generator.append(R"~~~(
  455. case ValueID::@identifier:titlecase@:
  456. )~~~");
  457. }
  458. property_generator.append(R"~~~(
  459. return true;
  460. default:
  461. break;
  462. }
  463. )~~~");
  464. }
  465. }
  466. generator.append(R"~~~(
  467. return false;
  468. }
  469. )~~~");
  470. }
  471. });
  472. generator.append(R"~~~(
  473. default:
  474. return true;
  475. }
  476. }
  477. size_t property_maximum_value_count(PropertyID property_id)
  478. {
  479. switch (property_id) {
  480. )~~~");
  481. properties.for_each_member([&](auto& name, auto& value) {
  482. VERIFY(value.is_object());
  483. if (value.as_object().has("max-values")) {
  484. auto max_values = value.as_object().get("max-values");
  485. VERIFY(max_values.is_number() && !max_values.is_double());
  486. auto property_generator = generator.fork();
  487. property_generator.set("name:titlecase", title_casify(name));
  488. property_generator.set("max_values", max_values.to_string());
  489. property_generator.append(R"~~~(
  490. case PropertyID::@name:titlecase@:
  491. return @max_values@;
  492. )~~~");
  493. }
  494. });
  495. generator.append(R"~~~(
  496. default:
  497. return 1;
  498. }
  499. }
  500. } // namespace Web::CSS
  501. )~~~");
  502. TRY(file.write(generator.as_string_view().bytes()));
  503. return {};
  504. }