| 
									
										
										
										
											2021-09-27 17:12:33 +01:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2023-08-22 12:40:18 +01:00
										 |  |  |  * Copyright (c) 2021-2023, Sam Atkins <atkinssj@serenityos.org> | 
					
						
							| 
									
										
										
										
											2021-09-27 17:12:33 +01:00
										 |  |  |  * | 
					
						
							|  |  |  |  * SPDX-License-Identifier: BSD-2-Clause | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <LibWeb/CSS/MediaQuery.h>
 | 
					
						
							| 
									
										
										
										
											2021-12-29 17:52:14 +00:00
										 |  |  | #include <LibWeb/CSS/Serialize.h>
 | 
					
						
							| 
									
										
										
										
											2023-05-08 06:37:18 +02:00
										 |  |  | #include <LibWeb/CSS/StyleComputer.h>
 | 
					
						
							| 
									
										
										
										
											2021-10-03 19:39:48 +01:00
										 |  |  | #include <LibWeb/DOM/Document.h>
 | 
					
						
							| 
									
										
										
										
											2022-03-07 23:08:26 +01:00
										 |  |  | #include <LibWeb/HTML/Window.h>
 | 
					
						
							| 
									
										
										
										
											2023-09-19 19:16:50 +02:00
										 |  |  | #include <LibWeb/Page/Page.h>
 | 
					
						
							| 
									
										
										
										
											2021-09-27 17:12:33 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace Web::CSS { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | NonnullRefPtr<MediaQuery> MediaQuery::create_not_all() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     auto media_query = new MediaQuery; | 
					
						
							|  |  |  |     media_query->m_negated = true; | 
					
						
							|  |  |  |     media_query->m_media_type = MediaType::All; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return adopt_ref(*media_query); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-22 12:40:18 +01:00
										 |  |  | String MediaFeatureValue::to_string() const | 
					
						
							| 
									
										
										
										
											2021-12-29 17:52:14 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     return m_value.visit( | 
					
						
							| 
									
										
										
										
											2024-08-14 14:06:03 +01:00
										 |  |  |         [](Keyword const& ident) { return MUST(String::from_utf8(string_from_keyword(ident))); }, | 
					
						
							| 
									
										
										
										
											2023-01-06 19:02:26 +01:00
										 |  |  |         [](Length const& length) { return length.to_string(); }, | 
					
						
							|  |  |  |         [](Ratio const& ratio) { return ratio.to_string(); }, | 
					
						
							|  |  |  |         [](Resolution const& resolution) { return resolution.to_string(); }, | 
					
						
							| 
									
										
										
										
											2024-10-14 10:05:01 +02:00
										 |  |  |         [](float number) { return String::number(number); }); | 
					
						
							| 
									
										
										
										
											2021-12-29 17:52:14 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool MediaFeatureValue::is_same_type(MediaFeatureValue const& other) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return m_value.visit( | 
					
						
							| 
									
										
										
										
											2024-08-14 14:06:03 +01:00
										 |  |  |         [&](Keyword const&) { return other.is_ident(); }, | 
					
						
							| 
									
										
										
										
											2022-01-13 17:31:00 +03:30
										 |  |  |         [&](Length const&) { return other.is_length(); }, | 
					
						
							| 
									
										
										
										
											2022-03-06 17:50:56 +00:00
										 |  |  |         [&](Ratio const&) { return other.is_ratio(); }, | 
					
						
							| 
									
										
										
										
											2022-02-22 14:09:19 +00:00
										 |  |  |         [&](Resolution const&) { return other.is_resolution(); }, | 
					
						
							| 
									
										
										
										
											2022-03-21 17:22:05 +00:00
										 |  |  |         [&](float) { return other.is_number(); }); | 
					
						
							| 
									
										
										
										
											2021-12-29 17:52:14 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-22 12:40:18 +01:00
										 |  |  | String MediaFeature::to_string() const | 
					
						
							| 
									
										
										
										
											2021-09-27 17:12:33 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2022-01-01 16:24:43 +00:00
										 |  |  |     auto comparison_string = [](Comparison comparison) -> StringView { | 
					
						
							|  |  |  |         switch (comparison) { | 
					
						
							|  |  |  |         case Comparison::Equal: | 
					
						
							|  |  |  |             return "="sv; | 
					
						
							|  |  |  |         case Comparison::LessThan: | 
					
						
							|  |  |  |             return "<"sv; | 
					
						
							|  |  |  |         case Comparison::LessThanOrEqual: | 
					
						
							|  |  |  |             return "<="sv; | 
					
						
							|  |  |  |         case Comparison::GreaterThan: | 
					
						
							|  |  |  |             return ">"sv; | 
					
						
							|  |  |  |         case Comparison::GreaterThanOrEqual: | 
					
						
							|  |  |  |             return ">="sv; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         VERIFY_NOT_REACHED(); | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-29 21:17:44 +00:00
										 |  |  |     switch (m_type) { | 
					
						
							| 
									
										
										
										
											2021-09-27 17:12:33 +01:00
										 |  |  |     case Type::IsTrue: | 
					
						
							| 
									
										
										
										
											2023-08-22 12:40:18 +01:00
										 |  |  |         return MUST(String::from_utf8(string_from_media_feature_id(m_id))); | 
					
						
							| 
									
										
										
										
											2021-09-27 17:12:33 +01:00
										 |  |  |     case Type::ExactValue: | 
					
						
							| 
									
										
										
										
											2024-04-17 07:48:30 +01:00
										 |  |  |         return MUST(String::formatted("{}: {}", string_from_media_feature_id(m_id), m_value->to_string())); | 
					
						
							| 
									
										
										
										
											2021-09-27 17:12:33 +01:00
										 |  |  |     case Type::MinValue: | 
					
						
							| 
									
										
										
										
											2024-04-17 07:48:30 +01:00
										 |  |  |         return MUST(String::formatted("min-{}: {}", string_from_media_feature_id(m_id), m_value->to_string())); | 
					
						
							| 
									
										
										
										
											2021-09-27 17:12:33 +01:00
										 |  |  |     case Type::MaxValue: | 
					
						
							| 
									
										
										
										
											2024-04-17 07:48:30 +01:00
										 |  |  |         return MUST(String::formatted("max-{}: {}", string_from_media_feature_id(m_id), m_value->to_string())); | 
					
						
							| 
									
										
										
										
											2022-01-01 16:24:43 +00:00
										 |  |  |     case Type::Range: | 
					
						
							|  |  |  |         if (!m_range->right_comparison.has_value()) | 
					
						
							| 
									
										
										
										
											2023-08-22 12:40:18 +01:00
										 |  |  |             return MUST(String::formatted("{} {} {}", m_range->left_value.to_string(), comparison_string(m_range->left_comparison), string_from_media_feature_id(m_id))); | 
					
						
							| 
									
										
										
										
											2022-01-01 16:24:43 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-22 12:40:18 +01:00
										 |  |  |         return MUST(String::formatted("{} {} {} {} {}", m_range->left_value.to_string(), comparison_string(m_range->left_comparison), string_from_media_feature_id(m_id), comparison_string(*m_range->right_comparison), m_range->right_value->to_string())); | 
					
						
							| 
									
										
										
										
											2021-09-27 17:12:33 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     VERIFY_NOT_REACHED(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-07 23:08:26 +01:00
										 |  |  | bool MediaFeature::evaluate(HTML::Window const& window) const | 
					
						
							| 
									
										
										
										
											2021-10-03 19:39:48 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2022-03-08 16:51:33 +00:00
										 |  |  |     auto maybe_queried_value = window.query_media_feature(m_id); | 
					
						
							| 
									
										
										
										
											2021-12-29 17:52:14 +00:00
										 |  |  |     if (!maybe_queried_value.has_value()) | 
					
						
							| 
									
										
										
										
											2021-10-03 19:39:48 +01:00
										 |  |  |         return false; | 
					
						
							| 
									
										
										
										
											2021-12-29 17:52:14 +00:00
										 |  |  |     auto queried_value = maybe_queried_value.release_value(); | 
					
						
							| 
									
										
										
										
											2021-10-03 19:39:48 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-29 21:17:44 +00:00
										 |  |  |     switch (m_type) { | 
					
						
							| 
									
										
										
										
											2021-10-03 19:39:48 +01:00
										 |  |  |     case Type::IsTrue: | 
					
						
							| 
									
										
										
										
											2021-12-29 17:52:14 +00:00
										 |  |  |         if (queried_value.is_number()) | 
					
						
							|  |  |  |             return queried_value.number() != 0; | 
					
						
							|  |  |  |         if (queried_value.is_length()) | 
					
						
							|  |  |  |             return queried_value.length().raw_value() != 0; | 
					
						
							| 
									
										
										
										
											2022-03-06 17:50:56 +00:00
										 |  |  |         // FIXME: I couldn't figure out from the spec how ratios should be evaluated in a boolean context.
 | 
					
						
							|  |  |  |         if (queried_value.is_ratio()) | 
					
						
							|  |  |  |             return !queried_value.ratio().is_degenerate(); | 
					
						
							| 
									
										
										
										
											2022-02-22 14:09:19 +00:00
										 |  |  |         if (queried_value.is_resolution()) | 
					
						
							|  |  |  |             return queried_value.resolution().to_dots_per_pixel() != 0; | 
					
						
							| 
									
										
										
										
											2022-03-16 17:24:14 +00:00
										 |  |  |         if (queried_value.is_ident()) { | 
					
						
							|  |  |  |             // NOTE: It is not technically correct to always treat `no-preference` as false, but every
 | 
					
						
							|  |  |  |             //       media-feature that accepts it as a value treats it as false, so good enough. :^)
 | 
					
						
							| 
									
										
										
										
											2024-08-14 14:06:03 +01:00
										 |  |  |             //       If other features gain this property for other keywords in the future, we can
 | 
					
						
							| 
									
										
										
										
											2022-03-16 17:24:14 +00:00
										 |  |  |             //       add more robust handling for them then.
 | 
					
						
							| 
									
										
										
										
											2024-08-14 14:06:03 +01:00
										 |  |  |             return queried_value.ident() != Keyword::None | 
					
						
							|  |  |  |                 && queried_value.ident() != Keyword::NoPreference; | 
					
						
							| 
									
										
										
										
											2022-03-16 17:24:14 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-10-03 19:39:48 +01:00
										 |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case Type::ExactValue: | 
					
						
							| 
									
										
										
										
											2022-02-15 16:12:22 +00:00
										 |  |  |         return compare(window, *m_value, Comparison::Equal, queried_value); | 
					
						
							| 
									
										
										
										
											2021-10-03 19:39:48 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     case Type::MinValue: | 
					
						
							| 
									
										
										
										
											2022-02-15 16:12:22 +00:00
										 |  |  |         return compare(window, queried_value, Comparison::GreaterThanOrEqual, *m_value); | 
					
						
							| 
									
										
										
										
											2022-01-01 16:24:43 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     case Type::MaxValue: | 
					
						
							| 
									
										
										
										
											2022-02-15 16:12:22 +00:00
										 |  |  |         return compare(window, queried_value, Comparison::LessThanOrEqual, *m_value); | 
					
						
							| 
									
										
										
										
											2021-12-29 17:52:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-01 16:24:43 +00:00
										 |  |  |     case Type::Range: | 
					
						
							| 
									
										
										
										
											2022-02-15 16:12:22 +00:00
										 |  |  |         if (!compare(window, m_range->left_value, m_range->left_comparison, queried_value)) | 
					
						
							| 
									
										
										
										
											2022-01-01 16:24:43 +00:00
										 |  |  |             return false; | 
					
						
							| 
									
										
										
										
											2021-12-29 17:52:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-01 16:24:43 +00:00
										 |  |  |         if (m_range->right_comparison.has_value()) | 
					
						
							| 
									
										
										
										
											2022-02-15 16:12:22 +00:00
										 |  |  |             if (!compare(window, queried_value, *m_range->right_comparison, *m_range->right_value)) | 
					
						
							| 
									
										
										
										
											2021-10-03 19:39:48 +01:00
										 |  |  |                 return false; | 
					
						
							| 
									
										
										
										
											2021-12-29 17:52:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-01 16:24:43 +00:00
										 |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     VERIFY_NOT_REACHED(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-07 23:08:26 +01:00
										 |  |  | bool MediaFeature::compare(HTML::Window const& window, MediaFeatureValue left, Comparison comparison, MediaFeatureValue right) | 
					
						
							| 
									
										
										
										
											2022-01-01 16:24:43 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     if (!left.is_same_type(right)) | 
					
						
							| 
									
										
										
										
											2021-10-03 19:39:48 +01:00
										 |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-01 16:24:43 +00:00
										 |  |  |     if (left.is_ident()) { | 
					
						
							|  |  |  |         if (comparison == Comparison::Equal) | 
					
						
							| 
									
										
										
										
											2022-03-08 17:57:37 +00:00
										 |  |  |             return left.ident() == right.ident(); | 
					
						
							| 
									
										
										
										
											2022-01-01 16:24:43 +00:00
										 |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-12-29 17:52:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-01 16:24:43 +00:00
										 |  |  |     if (left.is_number()) { | 
					
						
							|  |  |  |         switch (comparison) { | 
					
						
							|  |  |  |         case Comparison::Equal: | 
					
						
							|  |  |  |             return left.number() == right.number(); | 
					
						
							|  |  |  |         case Comparison::LessThan: | 
					
						
							|  |  |  |             return left.number() < right.number(); | 
					
						
							|  |  |  |         case Comparison::LessThanOrEqual: | 
					
						
							|  |  |  |             return left.number() <= right.number(); | 
					
						
							|  |  |  |         case Comparison::GreaterThan: | 
					
						
							|  |  |  |             return left.number() > right.number(); | 
					
						
							|  |  |  |         case Comparison::GreaterThanOrEqual: | 
					
						
							|  |  |  |             return left.number() >= right.number(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         VERIFY_NOT_REACHED(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-12-29 17:52:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-01 16:24:43 +00:00
										 |  |  |     if (left.is_length()) { | 
					
						
							| 
									
										
										
										
											2022-11-08 17:29:52 +00:00
										 |  |  |         CSSPixels left_px; | 
					
						
							|  |  |  |         CSSPixels right_px; | 
					
						
							| 
									
										
										
										
											2022-02-15 16:12:22 +00:00
										 |  |  |         // Save ourselves some work if neither side is a relative length.
 | 
					
						
							|  |  |  |         if (left.length().is_absolute() && right.length().is_absolute()) { | 
					
						
							|  |  |  |             left_px = left.length().absolute_length_to_px(); | 
					
						
							|  |  |  |             right_px = right.length().absolute_length_to_px(); | 
					
						
							|  |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2023-12-15 20:33:16 +01:00
										 |  |  |             auto viewport_rect = window.page().web_exposed_screen_area(); | 
					
						
							| 
									
										
										
										
											2022-02-15 16:12:22 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-11 12:53:32 +00:00
										 |  |  |             auto const& initial_font = window.associated_document().style_computer().initial_font(); | 
					
						
							| 
									
										
										
										
											2022-03-28 12:03:44 +02:00
										 |  |  |             Gfx::FontPixelMetrics const& initial_font_metrics = initial_font.pixel_metrics(); | 
					
						
							| 
									
										
										
										
											2024-06-04 15:56:01 +02:00
										 |  |  |             Length::FontMetrics font_metrics { CSSPixels { initial_font.point_size() }, initial_font_metrics }; | 
					
						
							| 
									
										
										
										
											2022-03-11 12:53:32 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-28 16:29:12 +01:00
										 |  |  |             left_px = left.length().to_px(viewport_rect, font_metrics, font_metrics); | 
					
						
							|  |  |  |             right_px = right.length().to_px(viewport_rect, font_metrics, font_metrics); | 
					
						
							| 
									
										
										
										
											2022-02-15 16:12:22 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-01-01 16:24:43 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         switch (comparison) { | 
					
						
							|  |  |  |         case Comparison::Equal: | 
					
						
							|  |  |  |             return left_px == right_px; | 
					
						
							|  |  |  |         case Comparison::LessThan: | 
					
						
							|  |  |  |             return left_px < right_px; | 
					
						
							|  |  |  |         case Comparison::LessThanOrEqual: | 
					
						
							|  |  |  |             return left_px <= right_px; | 
					
						
							|  |  |  |         case Comparison::GreaterThan: | 
					
						
							|  |  |  |             return left_px > right_px; | 
					
						
							|  |  |  |         case Comparison::GreaterThanOrEqual: | 
					
						
							|  |  |  |             return left_px >= right_px; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         VERIFY_NOT_REACHED(); | 
					
						
							| 
									
										
										
										
											2022-03-06 17:50:56 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (left.is_ratio()) { | 
					
						
							|  |  |  |         auto left_decimal = left.ratio().value(); | 
					
						
							|  |  |  |         auto right_decimal = right.ratio().value(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         switch (comparison) { | 
					
						
							|  |  |  |         case Comparison::Equal: | 
					
						
							|  |  |  |             return left_decimal == right_decimal; | 
					
						
							|  |  |  |         case Comparison::LessThan: | 
					
						
							|  |  |  |             return left_decimal < right_decimal; | 
					
						
							|  |  |  |         case Comparison::LessThanOrEqual: | 
					
						
							|  |  |  |             return left_decimal <= right_decimal; | 
					
						
							|  |  |  |         case Comparison::GreaterThan: | 
					
						
							|  |  |  |             return left_decimal > right_decimal; | 
					
						
							|  |  |  |         case Comparison::GreaterThanOrEqual: | 
					
						
							|  |  |  |             return left_decimal >= right_decimal; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         VERIFY_NOT_REACHED(); | 
					
						
							| 
									
										
										
										
											2021-10-03 19:39:48 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-22 14:09:19 +00:00
										 |  |  |     if (left.is_resolution()) { | 
					
						
							|  |  |  |         auto left_dppx = left.resolution().to_dots_per_pixel(); | 
					
						
							|  |  |  |         auto right_dppx = right.resolution().to_dots_per_pixel(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         switch (comparison) { | 
					
						
							|  |  |  |         case Comparison::Equal: | 
					
						
							|  |  |  |             return left_dppx == right_dppx; | 
					
						
							|  |  |  |         case Comparison::LessThan: | 
					
						
							|  |  |  |             return left_dppx < right_dppx; | 
					
						
							|  |  |  |         case Comparison::LessThanOrEqual: | 
					
						
							|  |  |  |             return left_dppx <= right_dppx; | 
					
						
							|  |  |  |         case Comparison::GreaterThan: | 
					
						
							|  |  |  |             return left_dppx > right_dppx; | 
					
						
							|  |  |  |         case Comparison::GreaterThanOrEqual: | 
					
						
							|  |  |  |             return left_dppx >= right_dppx; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         VERIFY_NOT_REACHED(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-03 19:39:48 +01:00
										 |  |  |     VERIFY_NOT_REACHED(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-30 14:25:30 +00:00
										 |  |  | NonnullOwnPtr<MediaCondition> MediaCondition::from_general_enclosed(GeneralEnclosed&& general_enclosed) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     auto result = new MediaCondition; | 
					
						
							|  |  |  |     result->type = Type::GeneralEnclosed; | 
					
						
							|  |  |  |     result->general_enclosed = move(general_enclosed); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return adopt_own(*result); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | NonnullOwnPtr<MediaCondition> MediaCondition::from_feature(MediaFeature&& feature) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     auto result = new MediaCondition; | 
					
						
							|  |  |  |     result->type = Type::Single; | 
					
						
							|  |  |  |     result->feature = move(feature); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return adopt_own(*result); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | NonnullOwnPtr<MediaCondition> MediaCondition::from_not(NonnullOwnPtr<MediaCondition>&& condition) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     auto result = new MediaCondition; | 
					
						
							|  |  |  |     result->type = Type::Not; | 
					
						
							|  |  |  |     result->conditions.append(move(condition)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return adopt_own(*result); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-06 17:16:25 +01:00
										 |  |  | NonnullOwnPtr<MediaCondition> MediaCondition::from_and_list(Vector<NonnullOwnPtr<MediaCondition>>&& conditions) | 
					
						
							| 
									
										
										
										
											2021-12-30 14:25:30 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     auto result = new MediaCondition; | 
					
						
							|  |  |  |     result->type = Type::And; | 
					
						
							|  |  |  |     result->conditions = move(conditions); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return adopt_own(*result); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-06 17:16:25 +01:00
										 |  |  | NonnullOwnPtr<MediaCondition> MediaCondition::from_or_list(Vector<NonnullOwnPtr<MediaCondition>>&& conditions) | 
					
						
							| 
									
										
										
										
											2021-12-30 14:25:30 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     auto result = new MediaCondition; | 
					
						
							|  |  |  |     result->type = Type::Or; | 
					
						
							|  |  |  |     result->conditions = move(conditions); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return adopt_own(*result); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-22 12:40:18 +01:00
										 |  |  | String MediaCondition::to_string() const | 
					
						
							| 
									
										
										
										
											2021-09-27 17:12:33 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     StringBuilder builder; | 
					
						
							|  |  |  |     builder.append('('); | 
					
						
							|  |  |  |     switch (type) { | 
					
						
							|  |  |  |     case Type::Single: | 
					
						
							| 
									
										
										
										
											2023-08-22 12:40:18 +01:00
										 |  |  |         builder.append(feature->to_string()); | 
					
						
							| 
									
										
										
										
											2021-09-27 17:12:33 +01:00
										 |  |  |         break; | 
					
						
							|  |  |  |     case Type::Not: | 
					
						
							| 
									
										
										
										
											2022-07-11 17:32:29 +00:00
										 |  |  |         builder.append("not "sv); | 
					
						
							| 
									
										
										
										
											2023-08-22 12:40:18 +01:00
										 |  |  |         builder.append(conditions.first()->to_string()); | 
					
						
							| 
									
										
										
										
											2021-09-27 17:12:33 +01:00
										 |  |  |         break; | 
					
						
							|  |  |  |     case Type::And: | 
					
						
							| 
									
										
										
										
											2022-07-11 17:32:29 +00:00
										 |  |  |         builder.join(" and "sv, conditions); | 
					
						
							| 
									
										
										
										
											2021-09-27 17:12:33 +01:00
										 |  |  |         break; | 
					
						
							|  |  |  |     case Type::Or: | 
					
						
							| 
									
										
										
										
											2022-07-11 17:32:29 +00:00
										 |  |  |         builder.join(" or "sv, conditions); | 
					
						
							| 
									
										
										
										
											2021-09-27 17:12:33 +01:00
										 |  |  |         break; | 
					
						
							| 
									
										
										
										
											2021-11-24 15:11:59 +00:00
										 |  |  |     case Type::GeneralEnclosed: | 
					
						
							|  |  |  |         builder.append(general_enclosed->to_string()); | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2021-09-27 17:12:33 +01:00
										 |  |  |     } | 
					
						
							|  |  |  |     builder.append(')'); | 
					
						
							| 
									
										
										
										
											2023-08-22 12:40:18 +01:00
										 |  |  |     return MUST(builder.to_string()); | 
					
						
							| 
									
										
										
										
											2021-09-27 17:12:33 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-07 23:08:26 +01:00
										 |  |  | MatchResult MediaCondition::evaluate(HTML::Window const& window) const | 
					
						
							| 
									
										
										
										
											2021-10-03 19:39:48 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     switch (type) { | 
					
						
							|  |  |  |     case Type::Single: | 
					
						
							| 
									
										
										
										
											2021-12-29 21:17:44 +00:00
										 |  |  |         return as_match_result(feature->evaluate(window)); | 
					
						
							| 
									
										
										
										
											2021-10-03 19:39:48 +01:00
										 |  |  |     case Type::Not: | 
					
						
							| 
									
										
										
										
											2023-03-06 17:16:25 +01:00
										 |  |  |         return negate(conditions.first()->evaluate(window)); | 
					
						
							| 
									
										
										
										
											2021-10-03 19:39:48 +01:00
										 |  |  |     case Type::And: | 
					
						
							| 
									
										
										
										
											2023-03-06 17:16:25 +01:00
										 |  |  |         return evaluate_and(conditions, [&](auto& child) { return child->evaluate(window); }); | 
					
						
							| 
									
										
										
										
											2021-10-03 19:39:48 +01:00
										 |  |  |     case Type::Or: | 
					
						
							| 
									
										
										
										
											2023-03-06 17:16:25 +01:00
										 |  |  |         return evaluate_or(conditions, [&](auto& child) { return child->evaluate(window); }); | 
					
						
							| 
									
										
										
										
											2021-11-24 15:11:59 +00:00
										 |  |  |     case Type::GeneralEnclosed: | 
					
						
							|  |  |  |         return general_enclosed->evaluate(); | 
					
						
							| 
									
										
										
										
											2021-10-03 19:39:48 +01:00
										 |  |  |     } | 
					
						
							|  |  |  |     VERIFY_NOT_REACHED(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-22 12:40:18 +01:00
										 |  |  | String MediaQuery::to_string() const | 
					
						
							| 
									
										
										
										
											2021-09-27 17:12:33 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     StringBuilder builder; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (m_negated) | 
					
						
							| 
									
										
										
										
											2022-07-11 17:32:29 +00:00
										 |  |  |         builder.append("not "sv); | 
					
						
							| 
									
										
										
										
											2021-09-27 17:12:33 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (m_negated || m_media_type != MediaType::All || !m_media_condition) { | 
					
						
							| 
									
										
										
										
											2022-04-27 14:58:25 +01:00
										 |  |  |         builder.append(CSS::to_string(m_media_type)); | 
					
						
							| 
									
										
										
										
											2021-09-27 17:12:33 +01:00
										 |  |  |         if (m_media_condition) | 
					
						
							| 
									
										
										
										
											2022-07-11 17:32:29 +00:00
										 |  |  |             builder.append(" and "sv); | 
					
						
							| 
									
										
										
										
											2021-09-27 17:12:33 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (m_media_condition) { | 
					
						
							| 
									
										
										
										
											2023-08-22 12:40:18 +01:00
										 |  |  |         builder.append(m_media_condition->to_string()); | 
					
						
							| 
									
										
										
										
											2021-09-27 17:12:33 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-22 12:40:18 +01:00
										 |  |  |     return MUST(builder.to_string()); | 
					
						
							| 
									
										
										
										
											2021-09-27 17:12:33 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-07 23:08:26 +01:00
										 |  |  | bool MediaQuery::evaluate(HTML::Window const& window) | 
					
						
							| 
									
										
										
										
											2021-10-03 19:39:48 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2021-11-24 14:59:31 +00:00
										 |  |  |     auto matches_media = [](MediaType media) -> MatchResult { | 
					
						
							| 
									
										
										
										
											2021-10-03 19:39:48 +01:00
										 |  |  |         switch (media) { | 
					
						
							|  |  |  |         case MediaType::All: | 
					
						
							| 
									
										
										
										
											2021-11-24 14:59:31 +00:00
										 |  |  |             return MatchResult::True; | 
					
						
							| 
									
										
										
										
											2021-10-03 19:39:48 +01:00
										 |  |  |         case MediaType::Print: | 
					
						
							|  |  |  |             // FIXME: Enable for printing, when we have printing!
 | 
					
						
							| 
									
										
										
										
											2021-11-24 14:59:31 +00:00
										 |  |  |             return MatchResult::False; | 
					
						
							| 
									
										
										
										
											2021-10-03 19:39:48 +01:00
										 |  |  |         case MediaType::Screen: | 
					
						
							|  |  |  |             // FIXME: Disable for printing, when we have printing!
 | 
					
						
							| 
									
										
										
										
											2021-11-24 14:59:31 +00:00
										 |  |  |             return MatchResult::True; | 
					
						
							| 
									
										
										
										
											2022-11-03 08:58:27 +03:00
										 |  |  |         case MediaType::Unknown: | 
					
						
							|  |  |  |             return MatchResult::False; | 
					
						
							| 
									
										
										
										
											2021-10-03 19:39:48 +01:00
										 |  |  |         // Deprecated, must never match:
 | 
					
						
							|  |  |  |         case MediaType::TTY: | 
					
						
							|  |  |  |         case MediaType::TV: | 
					
						
							|  |  |  |         case MediaType::Projection: | 
					
						
							|  |  |  |         case MediaType::Handheld: | 
					
						
							|  |  |  |         case MediaType::Braille: | 
					
						
							|  |  |  |         case MediaType::Embossed: | 
					
						
							|  |  |  |         case MediaType::Aural: | 
					
						
							|  |  |  |         case MediaType::Speech: | 
					
						
							| 
									
										
										
										
											2021-11-24 14:59:31 +00:00
										 |  |  |             return MatchResult::False; | 
					
						
							| 
									
										
										
										
											2021-10-03 19:39:48 +01:00
										 |  |  |         } | 
					
						
							|  |  |  |         VERIFY_NOT_REACHED(); | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-24 14:59:31 +00:00
										 |  |  |     MatchResult result = matches_media(m_media_type); | 
					
						
							| 
									
										
										
										
											2021-10-03 19:39:48 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-24 14:59:31 +00:00
										 |  |  |     if ((result == MatchResult::True) && m_media_condition) | 
					
						
							| 
									
										
										
										
											2021-10-03 19:39:48 +01:00
										 |  |  |         result = m_media_condition->evaluate(window); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-24 14:59:31 +00:00
										 |  |  |     if (m_negated) | 
					
						
							|  |  |  |         result = negate(result); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     m_matches = result == MatchResult::True; | 
					
						
							| 
									
										
										
										
											2021-10-03 19:39:48 +01:00
										 |  |  |     return m_matches; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-15 16:40:44 +01:00
										 |  |  | // https://www.w3.org/TR/cssom-1/#serialize-a-media-query-list
 | 
					
						
							| 
									
										
										
										
											2023-08-22 12:40:18 +01:00
										 |  |  | String serialize_a_media_query_list(Vector<NonnullRefPtr<MediaQuery>> const& media_queries) | 
					
						
							| 
									
										
										
										
											2021-10-15 16:40:44 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     // 1. If the media query list is empty, then return the empty string.
 | 
					
						
							|  |  |  |     if (media_queries.is_empty()) | 
					
						
							| 
									
										
										
										
											2023-02-14 20:19:58 +00:00
										 |  |  |         return String {}; | 
					
						
							| 
									
										
										
										
											2021-10-15 16:40:44 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // 2. Serialize each media query in the list of media queries, in the same order as they
 | 
					
						
							|  |  |  |     // appear in the media query list, and then serialize the list.
 | 
					
						
							| 
									
										
										
										
											2023-08-22 12:40:18 +01:00
										 |  |  |     return MUST(String::join(", "sv, media_queries)); | 
					
						
							| 
									
										
										
										
											2021-10-15 16:40:44 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-03 08:58:27 +03:00
										 |  |  | MediaQuery::MediaType media_type_from_string(StringView name) | 
					
						
							| 
									
										
										
										
											2022-04-27 14:58:25 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2023-03-10 08:48:54 +01:00
										 |  |  |     if (name.equals_ignoring_ascii_case("all"sv)) | 
					
						
							| 
									
										
										
										
											2022-04-27 14:58:25 +01:00
										 |  |  |         return MediaQuery::MediaType::All; | 
					
						
							| 
									
										
										
										
											2023-03-10 08:48:54 +01:00
										 |  |  |     if (name.equals_ignoring_ascii_case("aural"sv)) | 
					
						
							| 
									
										
										
										
											2022-04-27 14:58:25 +01:00
										 |  |  |         return MediaQuery::MediaType::Aural; | 
					
						
							| 
									
										
										
										
											2023-03-10 08:48:54 +01:00
										 |  |  |     if (name.equals_ignoring_ascii_case("braille"sv)) | 
					
						
							| 
									
										
										
										
											2022-04-27 14:58:25 +01:00
										 |  |  |         return MediaQuery::MediaType::Braille; | 
					
						
							| 
									
										
										
										
											2023-03-10 08:48:54 +01:00
										 |  |  |     if (name.equals_ignoring_ascii_case("embossed"sv)) | 
					
						
							| 
									
										
										
										
											2022-04-27 14:58:25 +01:00
										 |  |  |         return MediaQuery::MediaType::Embossed; | 
					
						
							| 
									
										
										
										
											2023-03-10 08:48:54 +01:00
										 |  |  |     if (name.equals_ignoring_ascii_case("handheld"sv)) | 
					
						
							| 
									
										
										
										
											2022-04-27 14:58:25 +01:00
										 |  |  |         return MediaQuery::MediaType::Handheld; | 
					
						
							| 
									
										
										
										
											2023-03-10 08:48:54 +01:00
										 |  |  |     if (name.equals_ignoring_ascii_case("print"sv)) | 
					
						
							| 
									
										
										
										
											2022-04-27 14:58:25 +01:00
										 |  |  |         return MediaQuery::MediaType::Print; | 
					
						
							| 
									
										
										
										
											2023-03-10 08:48:54 +01:00
										 |  |  |     if (name.equals_ignoring_ascii_case("projection"sv)) | 
					
						
							| 
									
										
										
										
											2022-04-27 14:58:25 +01:00
										 |  |  |         return MediaQuery::MediaType::Projection; | 
					
						
							| 
									
										
										
										
											2023-03-10 08:48:54 +01:00
										 |  |  |     if (name.equals_ignoring_ascii_case("screen"sv)) | 
					
						
							| 
									
										
										
										
											2022-04-27 14:58:25 +01:00
										 |  |  |         return MediaQuery::MediaType::Screen; | 
					
						
							| 
									
										
										
										
											2023-03-10 08:48:54 +01:00
										 |  |  |     if (name.equals_ignoring_ascii_case("speech"sv)) | 
					
						
							| 
									
										
										
										
											2022-04-27 14:58:25 +01:00
										 |  |  |         return MediaQuery::MediaType::Speech; | 
					
						
							| 
									
										
										
										
											2023-03-10 08:48:54 +01:00
										 |  |  |     if (name.equals_ignoring_ascii_case("tty"sv)) | 
					
						
							| 
									
										
										
										
											2022-04-27 14:58:25 +01:00
										 |  |  |         return MediaQuery::MediaType::TTY; | 
					
						
							| 
									
										
										
										
											2023-03-10 08:48:54 +01:00
										 |  |  |     if (name.equals_ignoring_ascii_case("tv"sv)) | 
					
						
							| 
									
										
										
										
											2022-04-27 14:58:25 +01:00
										 |  |  |         return MediaQuery::MediaType::TV; | 
					
						
							| 
									
										
										
										
											2022-11-03 08:58:27 +03:00
										 |  |  |     return MediaQuery::MediaType::Unknown; | 
					
						
							| 
									
										
										
										
											2022-04-27 14:58:25 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | StringView to_string(MediaQuery::MediaType media_type) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     switch (media_type) { | 
					
						
							|  |  |  |     case MediaQuery::MediaType::All: | 
					
						
							|  |  |  |         return "all"sv; | 
					
						
							|  |  |  |     case MediaQuery::MediaType::Aural: | 
					
						
							|  |  |  |         return "aural"sv; | 
					
						
							|  |  |  |     case MediaQuery::MediaType::Braille: | 
					
						
							|  |  |  |         return "braille"sv; | 
					
						
							|  |  |  |     case MediaQuery::MediaType::Embossed: | 
					
						
							|  |  |  |         return "embossed"sv; | 
					
						
							|  |  |  |     case MediaQuery::MediaType::Handheld: | 
					
						
							|  |  |  |         return "handheld"sv; | 
					
						
							|  |  |  |     case MediaQuery::MediaType::Print: | 
					
						
							|  |  |  |         return "print"sv; | 
					
						
							|  |  |  |     case MediaQuery::MediaType::Projection: | 
					
						
							|  |  |  |         return "projection"sv; | 
					
						
							|  |  |  |     case MediaQuery::MediaType::Screen: | 
					
						
							|  |  |  |         return "screen"sv; | 
					
						
							|  |  |  |     case MediaQuery::MediaType::Speech: | 
					
						
							|  |  |  |         return "speech"sv; | 
					
						
							|  |  |  |     case MediaQuery::MediaType::TTY: | 
					
						
							|  |  |  |         return "tty"sv; | 
					
						
							|  |  |  |     case MediaQuery::MediaType::TV: | 
					
						
							|  |  |  |         return "tv"sv; | 
					
						
							| 
									
										
										
										
											2022-11-03 08:58:27 +03:00
										 |  |  |     case MediaQuery::MediaType::Unknown: | 
					
						
							|  |  |  |         return "unknown"sv; | 
					
						
							| 
									
										
										
										
											2022-04-27 14:58:25 +01:00
										 |  |  |     } | 
					
						
							|  |  |  |     VERIFY_NOT_REACHED(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-27 17:12:33 +01:00
										 |  |  | } |