Selector.h 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316
  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. #pragma once
  8. #include <AK/FlyString.h>
  9. #include <AK/RefCounted.h>
  10. #include <AK/String.h>
  11. #include <AK/Vector.h>
  12. namespace Web::CSS {
  13. using SelectorList = Vector<NonnullRefPtr<class Selector>>;
  14. // This is a <complex-selector> in the spec. https://www.w3.org/TR/selectors-4/#complex
  15. class Selector : public RefCounted<Selector> {
  16. public:
  17. enum class PseudoElement {
  18. Before,
  19. After,
  20. FirstLine,
  21. FirstLetter,
  22. Marker,
  23. ProgressValue,
  24. ProgressBar,
  25. Placeholder,
  26. // Keep this last.
  27. PseudoElementCount,
  28. };
  29. struct SimpleSelector {
  30. enum class Type {
  31. Universal,
  32. TagName,
  33. Id,
  34. Class,
  35. Attribute,
  36. PseudoClass,
  37. PseudoElement,
  38. };
  39. struct ANPlusBPattern {
  40. int step_size { 0 }; // "A"
  41. int offset = { 0 }; // "B"
  42. // https://www.w3.org/TR/css-syntax-3/#serializing-anb
  43. ErrorOr<String> serialize() const
  44. {
  45. // 1. If A is zero, return the serialization of B.
  46. if (step_size == 0) {
  47. return String::formatted("{}", offset);
  48. }
  49. // 2. Otherwise, let result initially be an empty string.
  50. StringBuilder result;
  51. // 3.
  52. // - A is 1: Append "n" to result.
  53. if (step_size == 1)
  54. TRY(result.try_append('n'));
  55. // - A is -1: Append "-n" to result.
  56. else if (step_size == -1)
  57. TRY(result.try_append("-n"sv));
  58. // - A is non-zero: Serialize A and append it to result, then append "n" to result.
  59. else if (step_size != 0)
  60. TRY(result.try_appendff("{}n", step_size));
  61. // 4.
  62. // - B is greater than zero: Append "+" to result, then append the serialization of B to result.
  63. if (offset > 0)
  64. TRY(result.try_appendff("+{}", offset));
  65. // - B is less than zero: Append the serialization of B to result.
  66. if (offset < 0)
  67. TRY(result.try_appendff("{}", offset));
  68. // 5. Return result.
  69. return result.to_string();
  70. }
  71. };
  72. struct PseudoClass {
  73. enum class Type {
  74. Link,
  75. Visited,
  76. Hover,
  77. Focus,
  78. FocusWithin,
  79. FirstChild,
  80. LastChild,
  81. OnlyChild,
  82. NthChild,
  83. NthLastChild,
  84. Empty,
  85. Root,
  86. FirstOfType,
  87. LastOfType,
  88. OnlyOfType,
  89. NthOfType,
  90. NthLastOfType,
  91. Disabled,
  92. Enabled,
  93. Checked,
  94. Indeterminate,
  95. Is,
  96. Not,
  97. Where,
  98. Active,
  99. Lang,
  100. Scope,
  101. };
  102. Type type;
  103. // FIXME: We don't need this field on every single SimpleSelector, but it's also annoying to malloc it somewhere.
  104. // Only used when "pseudo_class" is "NthChild" or "NthLastChild".
  105. ANPlusBPattern nth_child_pattern {};
  106. SelectorList argument_selector_list {};
  107. // Used for :lang(en-gb,dk)
  108. Vector<FlyString> languages {};
  109. };
  110. struct Attribute {
  111. enum class MatchType {
  112. HasAttribute,
  113. ExactValueMatch,
  114. ContainsWord, // [att~=val]
  115. ContainsString, // [att*=val]
  116. StartsWithSegment, // [att|=val]
  117. StartsWithString, // [att^=val]
  118. EndsWithString, // [att$=val]
  119. };
  120. enum class CaseType {
  121. DefaultMatch,
  122. CaseSensitiveMatch,
  123. CaseInsensitiveMatch,
  124. };
  125. MatchType match_type;
  126. FlyString name {};
  127. String value {};
  128. CaseType case_type;
  129. };
  130. struct Name {
  131. Name(FlyString n)
  132. : name(move(n))
  133. , lowercase_name(name.to_string().to_lowercase().release_value_but_fixme_should_propagate_errors())
  134. {
  135. }
  136. FlyString name;
  137. FlyString lowercase_name;
  138. };
  139. Type type;
  140. Variant<Empty, Attribute, PseudoClass, PseudoElement, Name> value {};
  141. Attribute const& attribute() const { return value.get<Attribute>(); }
  142. Attribute& attribute() { return value.get<Attribute>(); }
  143. PseudoClass const& pseudo_class() const { return value.get<PseudoClass>(); }
  144. PseudoClass& pseudo_class() { return value.get<PseudoClass>(); }
  145. PseudoElement const& pseudo_element() const { return value.get<PseudoElement>(); }
  146. PseudoElement& pseudo_element() { return value.get<PseudoElement>(); }
  147. FlyString const& name() const { return value.get<Name>().name; }
  148. FlyString& name() { return value.get<Name>().name; }
  149. FlyString const& lowercase_name() const { return value.get<Name>().lowercase_name; }
  150. FlyString& lowercase_name() { return value.get<Name>().lowercase_name; }
  151. ErrorOr<String> serialize() const;
  152. };
  153. enum class Combinator {
  154. None,
  155. ImmediateChild, // >
  156. Descendant, // <whitespace>
  157. NextSibling, // +
  158. SubsequentSibling, // ~
  159. Column, // ||
  160. };
  161. struct CompoundSelector {
  162. // Spec-wise, the <combinator> is not part of a <compound-selector>,
  163. // but it is more understandable to put them together.
  164. Combinator combinator { Combinator::None };
  165. Vector<SimpleSelector> simple_selectors;
  166. };
  167. static NonnullRefPtr<Selector> create(Vector<CompoundSelector>&& compound_selectors)
  168. {
  169. return adopt_ref(*new Selector(move(compound_selectors)));
  170. }
  171. ~Selector() = default;
  172. Vector<CompoundSelector> const& compound_selectors() const { return m_compound_selectors; }
  173. Optional<PseudoElement> pseudo_element() const { return m_pseudo_element; }
  174. u32 specificity() const;
  175. ErrorOr<String> serialize() const;
  176. private:
  177. explicit Selector(Vector<CompoundSelector>&&);
  178. Vector<CompoundSelector> m_compound_selectors;
  179. mutable Optional<u32> m_specificity;
  180. Optional<Selector::PseudoElement> m_pseudo_element;
  181. };
  182. constexpr StringView pseudo_element_name(Selector::PseudoElement pseudo_element)
  183. {
  184. switch (pseudo_element) {
  185. case Selector::PseudoElement::Before:
  186. return "before"sv;
  187. case Selector::PseudoElement::After:
  188. return "after"sv;
  189. case Selector::PseudoElement::FirstLine:
  190. return "first-line"sv;
  191. case Selector::PseudoElement::FirstLetter:
  192. return "first-letter"sv;
  193. case Selector::PseudoElement::Marker:
  194. return "marker"sv;
  195. case Selector::PseudoElement::ProgressBar:
  196. return "-webkit-progress-bar"sv;
  197. case Selector::PseudoElement::ProgressValue:
  198. return "-webkit-progress-value"sv;
  199. case Selector::PseudoElement::Placeholder:
  200. return "placeholder"sv;
  201. case Selector::PseudoElement::PseudoElementCount:
  202. break;
  203. }
  204. VERIFY_NOT_REACHED();
  205. }
  206. Optional<Selector::PseudoElement> pseudo_element_from_string(StringView);
  207. constexpr StringView pseudo_class_name(Selector::SimpleSelector::PseudoClass::Type pseudo_class)
  208. {
  209. switch (pseudo_class) {
  210. case Selector::SimpleSelector::PseudoClass::Type::Link:
  211. return "link"sv;
  212. case Selector::SimpleSelector::PseudoClass::Type::Visited:
  213. return "visited"sv;
  214. case Selector::SimpleSelector::PseudoClass::Type::Hover:
  215. return "hover"sv;
  216. case Selector::SimpleSelector::PseudoClass::Type::Focus:
  217. return "focus"sv;
  218. case Selector::SimpleSelector::PseudoClass::Type::FocusWithin:
  219. return "focus-within"sv;
  220. case Selector::SimpleSelector::PseudoClass::Type::FirstChild:
  221. return "first-child"sv;
  222. case Selector::SimpleSelector::PseudoClass::Type::LastChild:
  223. return "last-child"sv;
  224. case Selector::SimpleSelector::PseudoClass::Type::OnlyChild:
  225. return "only-child"sv;
  226. case Selector::SimpleSelector::PseudoClass::Type::Empty:
  227. return "empty"sv;
  228. case Selector::SimpleSelector::PseudoClass::Type::Root:
  229. return "root"sv;
  230. case Selector::SimpleSelector::PseudoClass::Type::FirstOfType:
  231. return "first-of-type"sv;
  232. case Selector::SimpleSelector::PseudoClass::Type::LastOfType:
  233. return "last-of-type"sv;
  234. case Selector::SimpleSelector::PseudoClass::Type::OnlyOfType:
  235. return "only-of-type"sv;
  236. case Selector::SimpleSelector::PseudoClass::Type::NthOfType:
  237. return "nth-of-type"sv;
  238. case Selector::SimpleSelector::PseudoClass::Type::NthLastOfType:
  239. return "nth-last-of-type"sv;
  240. case Selector::SimpleSelector::PseudoClass::Type::Disabled:
  241. return "disabled"sv;
  242. case Selector::SimpleSelector::PseudoClass::Type::Enabled:
  243. return "enabled"sv;
  244. case Selector::SimpleSelector::PseudoClass::Type::Checked:
  245. return "checked"sv;
  246. case Selector::SimpleSelector::PseudoClass::Type::Indeterminate:
  247. return "indeterminate"sv;
  248. case Selector::SimpleSelector::PseudoClass::Type::Active:
  249. return "active"sv;
  250. case Selector::SimpleSelector::PseudoClass::Type::NthChild:
  251. return "nth-child"sv;
  252. case Selector::SimpleSelector::PseudoClass::Type::NthLastChild:
  253. return "nth-last-child"sv;
  254. case Selector::SimpleSelector::PseudoClass::Type::Is:
  255. return "is"sv;
  256. case Selector::SimpleSelector::PseudoClass::Type::Not:
  257. return "not"sv;
  258. case Selector::SimpleSelector::PseudoClass::Type::Where:
  259. return "where"sv;
  260. case Selector::SimpleSelector::PseudoClass::Type::Lang:
  261. return "lang"sv;
  262. case Selector::SimpleSelector::PseudoClass::Type::Scope:
  263. return "scope"sv;
  264. }
  265. VERIFY_NOT_REACHED();
  266. }
  267. ErrorOr<String> serialize_a_group_of_selectors(Vector<NonnullRefPtr<Selector>> const& selectors);
  268. }
  269. namespace AK {
  270. template<>
  271. struct Formatter<Web::CSS::Selector> : Formatter<StringView> {
  272. ErrorOr<void> format(FormatBuilder& builder, Web::CSS::Selector const& selector)
  273. {
  274. return Formatter<StringView>::format(builder, TRY(selector.serialize()));
  275. }
  276. };
  277. }