Selector.h 8.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246
  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. #include <LibWeb/CSS/PseudoClass.h>
  13. namespace Web::CSS {
  14. using SelectorList = Vector<NonnullRefPtr<class Selector>>;
  15. // This is a <complex-selector> in the spec. https://www.w3.org/TR/selectors-4/#complex
  16. class Selector : public RefCounted<Selector> {
  17. public:
  18. enum class PseudoElement {
  19. Before,
  20. After,
  21. FirstLine,
  22. FirstLetter,
  23. Marker,
  24. ProgressValue,
  25. ProgressBar,
  26. Placeholder,
  27. Selection,
  28. // Keep this last.
  29. PseudoElementCount,
  30. };
  31. struct SimpleSelector {
  32. enum class Type {
  33. Universal,
  34. TagName,
  35. Id,
  36. Class,
  37. Attribute,
  38. PseudoClass,
  39. PseudoElement,
  40. };
  41. struct ANPlusBPattern {
  42. int step_size { 0 }; // "A"
  43. int offset = { 0 }; // "B"
  44. // https://www.w3.org/TR/css-syntax-3/#serializing-anb
  45. ErrorOr<String> serialize() const
  46. {
  47. // 1. If A is zero, return the serialization of B.
  48. if (step_size == 0) {
  49. return String::formatted("{}", offset);
  50. }
  51. // 2. Otherwise, let result initially be an empty string.
  52. StringBuilder result;
  53. // 3.
  54. // - A is 1: Append "n" to result.
  55. if (step_size == 1)
  56. TRY(result.try_append('n'));
  57. // - A is -1: Append "-n" to result.
  58. else if (step_size == -1)
  59. TRY(result.try_append("-n"sv));
  60. // - A is non-zero: Serialize A and append it to result, then append "n" to result.
  61. else if (step_size != 0)
  62. TRY(result.try_appendff("{}n", step_size));
  63. // 4.
  64. // - B is greater than zero: Append "+" to result, then append the serialization of B to result.
  65. if (offset > 0)
  66. TRY(result.try_appendff("+{}", offset));
  67. // - B is less than zero: Append the serialization of B to result.
  68. if (offset < 0)
  69. TRY(result.try_appendff("{}", offset));
  70. // 5. Return result.
  71. return result.to_string();
  72. }
  73. };
  74. struct PseudoClassSelector {
  75. PseudoClass type;
  76. // FIXME: We don't need this field on every single SimpleSelector, but it's also annoying to malloc it somewhere.
  77. // Only used when "pseudo_class" is "NthChild" or "NthLastChild".
  78. ANPlusBPattern nth_child_pattern {};
  79. SelectorList argument_selector_list {};
  80. // Used for :lang(en-gb,dk)
  81. Vector<FlyString> languages {};
  82. };
  83. struct Name {
  84. Name(FlyString n)
  85. : name(move(n))
  86. , lowercase_name(name.to_string().to_lowercase().release_value_but_fixme_should_propagate_errors())
  87. {
  88. }
  89. FlyString name;
  90. FlyString lowercase_name;
  91. };
  92. // Equivalent to `<wq-name>`
  93. // https://www.w3.org/TR/selectors-4/#typedef-wq-name
  94. struct QualifiedName {
  95. enum class NamespaceType {
  96. Default, // `E`
  97. None, // `|E`
  98. Any, // `*|E`
  99. Named, // `ns|E`
  100. };
  101. NamespaceType namespace_type { NamespaceType::Default };
  102. FlyString namespace_ {};
  103. Name name;
  104. };
  105. struct Attribute {
  106. enum class MatchType {
  107. HasAttribute,
  108. ExactValueMatch,
  109. ContainsWord, // [att~=val]
  110. ContainsString, // [att*=val]
  111. StartsWithSegment, // [att|=val]
  112. StartsWithString, // [att^=val]
  113. EndsWithString, // [att$=val]
  114. };
  115. enum class CaseType {
  116. DefaultMatch,
  117. CaseSensitiveMatch,
  118. CaseInsensitiveMatch,
  119. };
  120. MatchType match_type;
  121. QualifiedName qualified_name;
  122. String value {};
  123. CaseType case_type;
  124. };
  125. Type type;
  126. Variant<Empty, Attribute, PseudoClassSelector, PseudoElement, Name, QualifiedName> value {};
  127. Attribute const& attribute() const { return value.get<Attribute>(); }
  128. Attribute& attribute() { return value.get<Attribute>(); }
  129. PseudoClassSelector const& pseudo_class() const { return value.get<PseudoClassSelector>(); }
  130. PseudoClassSelector& pseudo_class() { return value.get<PseudoClassSelector>(); }
  131. PseudoElement const& pseudo_element() const { return value.get<PseudoElement>(); }
  132. PseudoElement& pseudo_element() { return value.get<PseudoElement>(); }
  133. FlyString const& name() const { return value.get<Name>().name; }
  134. FlyString& name() { return value.get<Name>().name; }
  135. FlyString const& lowercase_name() const { return value.get<Name>().lowercase_name; }
  136. FlyString& lowercase_name() { return value.get<Name>().lowercase_name; }
  137. QualifiedName const& qualified_name() const { return value.get<QualifiedName>(); }
  138. QualifiedName& qualified_name() { return value.get<QualifiedName>(); }
  139. ErrorOr<String> serialize() const;
  140. };
  141. enum class Combinator {
  142. None,
  143. ImmediateChild, // >
  144. Descendant, // <whitespace>
  145. NextSibling, // +
  146. SubsequentSibling, // ~
  147. Column, // ||
  148. };
  149. struct CompoundSelector {
  150. // Spec-wise, the <combinator> is not part of a <compound-selector>,
  151. // but it is more understandable to put them together.
  152. Combinator combinator { Combinator::None };
  153. Vector<SimpleSelector> simple_selectors;
  154. };
  155. static NonnullRefPtr<Selector> create(Vector<CompoundSelector>&& compound_selectors)
  156. {
  157. return adopt_ref(*new Selector(move(compound_selectors)));
  158. }
  159. ~Selector() = default;
  160. Vector<CompoundSelector> const& compound_selectors() const { return m_compound_selectors; }
  161. Optional<PseudoElement> pseudo_element() const { return m_pseudo_element; }
  162. u32 specificity() const;
  163. ErrorOr<String> serialize() const;
  164. private:
  165. explicit Selector(Vector<CompoundSelector>&&);
  166. Vector<CompoundSelector> m_compound_selectors;
  167. mutable Optional<u32> m_specificity;
  168. Optional<Selector::PseudoElement> m_pseudo_element;
  169. };
  170. constexpr StringView pseudo_element_name(Selector::PseudoElement pseudo_element)
  171. {
  172. switch (pseudo_element) {
  173. case Selector::PseudoElement::Before:
  174. return "before"sv;
  175. case Selector::PseudoElement::After:
  176. return "after"sv;
  177. case Selector::PseudoElement::FirstLine:
  178. return "first-line"sv;
  179. case Selector::PseudoElement::FirstLetter:
  180. return "first-letter"sv;
  181. case Selector::PseudoElement::Marker:
  182. return "marker"sv;
  183. case Selector::PseudoElement::ProgressBar:
  184. return "-webkit-progress-bar"sv;
  185. case Selector::PseudoElement::ProgressValue:
  186. return "-webkit-progress-value"sv;
  187. case Selector::PseudoElement::Placeholder:
  188. return "placeholder"sv;
  189. case Selector::PseudoElement::Selection:
  190. return "selection"sv;
  191. case Selector::PseudoElement::PseudoElementCount:
  192. break;
  193. }
  194. VERIFY_NOT_REACHED();
  195. }
  196. Optional<Selector::PseudoElement> pseudo_element_from_string(StringView);
  197. ErrorOr<String> serialize_a_group_of_selectors(Vector<NonnullRefPtr<Selector>> const& selectors);
  198. }
  199. namespace AK {
  200. template<>
  201. struct Formatter<Web::CSS::Selector> : Formatter<StringView> {
  202. ErrorOr<void> format(FormatBuilder& builder, Web::CSS::Selector const& selector)
  203. {
  204. return Formatter<StringView>::format(builder, TRY(selector.serialize()));
  205. }
  206. };
  207. }