MediaQuery.cpp 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240
  1. /*
  2. * Copyright (c) 2021, Sam Atkins <atkinssj@serenityos.org>
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #include <LibWeb/CSS/MediaQuery.h>
  7. #include <LibWeb/DOM/Document.h>
  8. #include <LibWeb/DOM/Window.h>
  9. namespace Web::CSS {
  10. NonnullRefPtr<MediaQuery> MediaQuery::create_not_all()
  11. {
  12. auto media_query = new MediaQuery;
  13. media_query->m_negated = true;
  14. media_query->m_media_type = MediaType::All;
  15. return adopt_ref(*media_query);
  16. }
  17. String MediaQuery::MediaFeature::to_string() const
  18. {
  19. switch (type) {
  20. case Type::IsTrue:
  21. return name;
  22. case Type::ExactValue:
  23. return String::formatted("{}:{}", name, value->to_string());
  24. case Type::MinValue:
  25. return String::formatted("min-{}:{}", name, value->to_string());
  26. case Type::MaxValue:
  27. return String::formatted("max-{}:{}", name, value->to_string());
  28. }
  29. VERIFY_NOT_REACHED();
  30. }
  31. bool MediaQuery::MediaFeature::evaluate(DOM::Window const& window) const
  32. {
  33. auto queried_value = window.query_media_feature(name);
  34. if (!queried_value)
  35. return false;
  36. switch (type) {
  37. case Type::IsTrue:
  38. if (queried_value->has_number())
  39. return queried_value->to_number() != 0;
  40. if (queried_value->has_length())
  41. return queried_value->to_length().raw_value() != 0;
  42. if (queried_value->has_identifier())
  43. return queried_value->to_identifier() != ValueID::None;
  44. return false;
  45. case Type::ExactValue:
  46. return queried_value->equals(*value);
  47. case Type::MinValue:
  48. if (queried_value->has_number() && value->has_number())
  49. return queried_value->to_number() >= value->to_number();
  50. if (queried_value->has_length() && value->has_length()) {
  51. auto queried_length = queried_value->to_length();
  52. auto value_length = value->to_length();
  53. // FIXME: We should be checking that lengths are valid during parsing
  54. if (!value_length.is_absolute()) {
  55. dbgln("Media feature was given a non-absolute length, which is invalid! {}", value_length.to_string());
  56. return false;
  57. }
  58. return queried_length.absolute_length_to_px() >= value_length.absolute_length_to_px();
  59. }
  60. return false;
  61. case Type::MaxValue:
  62. if (queried_value->has_number() && value->has_number())
  63. return queried_value->to_number() <= value->to_number();
  64. if (queried_value->has_length() && value->has_length()) {
  65. auto queried_length = queried_value->to_length();
  66. auto value_length = value->to_length();
  67. // FIXME: We should be checking that lengths are valid during parsing
  68. if (!value_length.is_absolute()) {
  69. dbgln("Media feature was given a non-absolute length, which is invalid! {}", value_length.to_string());
  70. return false;
  71. }
  72. return queried_length.absolute_length_to_px() <= value_length.absolute_length_to_px();
  73. }
  74. return false;
  75. }
  76. VERIFY_NOT_REACHED();
  77. }
  78. String MediaQuery::MediaCondition::to_string() const
  79. {
  80. StringBuilder builder;
  81. builder.append('(');
  82. switch (type) {
  83. case Type::Single:
  84. builder.append(feature.to_string());
  85. break;
  86. case Type::Not:
  87. builder.append("not ");
  88. builder.append(conditions.first().to_string());
  89. break;
  90. case Type::And:
  91. builder.join(" and ", conditions);
  92. break;
  93. case Type::Or:
  94. builder.join(" or ", conditions);
  95. break;
  96. case Type::GeneralEnclosed:
  97. builder.append(general_enclosed->to_string());
  98. break;
  99. }
  100. builder.append(')');
  101. return builder.to_string();
  102. }
  103. MatchResult MediaQuery::MediaCondition::evaluate(DOM::Window const& window) const
  104. {
  105. switch (type) {
  106. case Type::Single:
  107. return as_match_result(feature.evaluate(window));
  108. case Type::Not:
  109. return negate(conditions.first().evaluate(window));
  110. case Type::And:
  111. return evaluate_and(conditions, [&](auto& child) { return child.evaluate(window); });
  112. case Type::Or:
  113. return evaluate_or(conditions, [&](auto& child) { return child.evaluate(window); });
  114. case Type::GeneralEnclosed:
  115. return general_enclosed->evaluate();
  116. }
  117. VERIFY_NOT_REACHED();
  118. }
  119. String MediaQuery::to_string() const
  120. {
  121. StringBuilder builder;
  122. if (m_negated)
  123. builder.append("not ");
  124. if (m_negated || m_media_type != MediaType::All || !m_media_condition) {
  125. switch (m_media_type) {
  126. case MediaType::All:
  127. builder.append("all");
  128. break;
  129. case MediaType::Aural:
  130. builder.append("aural");
  131. break;
  132. case MediaType::Braille:
  133. builder.append("braille");
  134. break;
  135. case MediaType::Embossed:
  136. builder.append("embossed");
  137. break;
  138. case MediaType::Handheld:
  139. builder.append("handheld");
  140. break;
  141. case MediaType::Print:
  142. builder.append("print");
  143. break;
  144. case MediaType::Projection:
  145. builder.append("projection");
  146. break;
  147. case MediaType::Screen:
  148. builder.append("screen");
  149. break;
  150. case MediaType::Speech:
  151. builder.append("speech");
  152. break;
  153. case MediaType::TTY:
  154. builder.append("tty");
  155. break;
  156. case MediaType::TV:
  157. builder.append("tv");
  158. break;
  159. }
  160. if (m_media_condition)
  161. builder.append(" and ");
  162. }
  163. if (m_media_condition) {
  164. builder.append(m_media_condition->to_string());
  165. }
  166. return builder.to_string();
  167. }
  168. bool MediaQuery::evaluate(DOM::Window const& window)
  169. {
  170. auto matches_media = [](MediaType media) -> MatchResult {
  171. switch (media) {
  172. case MediaType::All:
  173. return MatchResult::True;
  174. case MediaType::Print:
  175. // FIXME: Enable for printing, when we have printing!
  176. return MatchResult::False;
  177. case MediaType::Screen:
  178. // FIXME: Disable for printing, when we have printing!
  179. return MatchResult::True;
  180. // Deprecated, must never match:
  181. case MediaType::TTY:
  182. case MediaType::TV:
  183. case MediaType::Projection:
  184. case MediaType::Handheld:
  185. case MediaType::Braille:
  186. case MediaType::Embossed:
  187. case MediaType::Aural:
  188. case MediaType::Speech:
  189. return MatchResult::False;
  190. }
  191. VERIFY_NOT_REACHED();
  192. };
  193. MatchResult result = matches_media(m_media_type);
  194. if ((result == MatchResult::True) && m_media_condition)
  195. result = m_media_condition->evaluate(window);
  196. if (m_negated)
  197. result = negate(result);
  198. m_matches = result == MatchResult::True;
  199. return m_matches;
  200. }
  201. // https://www.w3.org/TR/cssom-1/#serialize-a-media-query-list
  202. String serialize_a_media_query_list(NonnullRefPtrVector<MediaQuery> const& media_queries)
  203. {
  204. // 1. If the media query list is empty, then return the empty string.
  205. if (media_queries.is_empty())
  206. return "";
  207. // 2. Serialize each media query in the list of media queries, in the same order as they
  208. // appear in the media query list, and then serialize the list.
  209. StringBuilder builder;
  210. builder.join(", ", media_queries);
  211. return builder.to_string();
  212. }
  213. }