MediaQuery.cpp 7.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244
  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. }
  97. builder.append(')');
  98. return builder.to_string();
  99. }
  100. bool MediaQuery::MediaCondition::evaluate(DOM::Window const& window) const
  101. {
  102. switch (type) {
  103. case Type::Single:
  104. return feature.evaluate(window);
  105. case Type::Not:
  106. return !conditions.first().evaluate(window);
  107. case Type::And:
  108. for (auto& child : conditions) {
  109. if (!child.evaluate(window))
  110. return false;
  111. }
  112. return true;
  113. case Type::Or:
  114. for (auto& child : conditions) {
  115. if (child.evaluate(window))
  116. return true;
  117. }
  118. return false;
  119. }
  120. VERIFY_NOT_REACHED();
  121. }
  122. String MediaQuery::to_string() const
  123. {
  124. StringBuilder builder;
  125. if (m_negated)
  126. builder.append("not ");
  127. if (m_negated || m_media_type != MediaType::All || !m_media_condition) {
  128. switch (m_media_type) {
  129. case MediaType::All:
  130. builder.append("all");
  131. break;
  132. case MediaType::Aural:
  133. builder.append("aural");
  134. break;
  135. case MediaType::Braille:
  136. builder.append("braille");
  137. break;
  138. case MediaType::Embossed:
  139. builder.append("embossed");
  140. break;
  141. case MediaType::Handheld:
  142. builder.append("handheld");
  143. break;
  144. case MediaType::Print:
  145. builder.append("print");
  146. break;
  147. case MediaType::Projection:
  148. builder.append("projection");
  149. break;
  150. case MediaType::Screen:
  151. builder.append("screen");
  152. break;
  153. case MediaType::Speech:
  154. builder.append("speech");
  155. break;
  156. case MediaType::TTY:
  157. builder.append("tty");
  158. break;
  159. case MediaType::TV:
  160. builder.append("tv");
  161. break;
  162. }
  163. if (m_media_condition)
  164. builder.append(" and ");
  165. }
  166. if (m_media_condition) {
  167. builder.append(m_media_condition->to_string());
  168. }
  169. return builder.to_string();
  170. }
  171. bool MediaQuery::evaluate(DOM::Window const& window)
  172. {
  173. auto matches_media = [](MediaType media) -> bool {
  174. switch (media) {
  175. case MediaType::All:
  176. return true;
  177. case MediaType::Print:
  178. // FIXME: Enable for printing, when we have printing!
  179. return false;
  180. case MediaType::Screen:
  181. // FIXME: Disable for printing, when we have printing!
  182. return true;
  183. // Deprecated, must never match:
  184. case MediaType::TTY:
  185. case MediaType::TV:
  186. case MediaType::Projection:
  187. case MediaType::Handheld:
  188. case MediaType::Braille:
  189. case MediaType::Embossed:
  190. case MediaType::Aural:
  191. case MediaType::Speech:
  192. return false;
  193. }
  194. VERIFY_NOT_REACHED();
  195. };
  196. bool result = matches_media(m_media_type);
  197. if (result && m_media_condition)
  198. result = m_media_condition->evaluate(window);
  199. m_matches = m_negated ? !result : result;
  200. return m_matches;
  201. }
  202. // https://www.w3.org/TR/cssom-1/#serialize-a-media-query-list
  203. String serialize_a_media_query_list(NonnullRefPtrVector<MediaQuery> const& media_queries)
  204. {
  205. // 1. If the media query list is empty, then return the empty string.
  206. if (media_queries.is_empty())
  207. return "";
  208. // 2. Serialize each media query in the list of media queries, in the same order as they
  209. // appear in the media query list, and then serialize the list.
  210. StringBuilder builder;
  211. builder.join(", ", media_queries);
  212. return builder.to_string();
  213. }
  214. }