/* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at https://mozilla.org/MPL/2.0/. */ use cssparser::{Parser, ParserInput}; use dom_struct::dom_struct; use servo_arc::Arc; use style::media_queries::{MediaList as StyleMediaList, MediaQuery}; use style::parser::ParserContext; use style::shared_lock::{Locked, SharedRwLock}; use style::stylesheets::{CssRuleType, Origin, UrlExtraData}; use style_traits::{ParseError, ParsingMode, ToCss}; use crate::dom::bindings::codegen::Bindings::MediaListBinding::MediaListMethods; use crate::dom::bindings::codegen::Bindings::WindowBinding::Window_Binding::WindowMethods; use crate::dom::bindings::reflector::{DomGlobal, Reflector, reflect_dom_object}; use crate::dom::bindings::root::{Dom, DomRoot}; use crate::dom::bindings::str::DOMString; use crate::dom::cssstylesheet::CSSStyleSheet; use crate::dom::window::Window; use crate::script_runtime::CanGc; #[dom_struct] pub(crate) struct MediaList { reflector_: Reflector, parent_stylesheet: Dom, #[ignore_malloc_size_of = "Arc"] #[no_trace] media_queries: Arc>, } impl MediaList { #[cfg_attr(crown, allow(crown::unrooted_must_root))] pub(crate) fn new_inherited( parent_stylesheet: &CSSStyleSheet, media_queries: Arc>, ) -> MediaList { MediaList { parent_stylesheet: Dom::from_ref(parent_stylesheet), reflector_: Reflector::new(), media_queries, } } #[cfg_attr(crown, allow(crown::unrooted_must_root))] pub(crate) fn new( window: &Window, parent_stylesheet: &CSSStyleSheet, media_queries: Arc>, can_gc: CanGc, ) -> DomRoot { reflect_dom_object( Box::new(MediaList::new_inherited(parent_stylesheet, media_queries)), window, can_gc, ) } fn shared_lock(&self) -> &SharedRwLock { &self.parent_stylesheet.style_stylesheet().shared_lock } /// pub(crate) fn parse_media_list(value: &str, window: &Window) -> StyleMediaList { if value.is_empty() { return StyleMediaList::empty(); } let mut input = ParserInput::new(value); let mut parser = Parser::new(&mut input); let url_data = UrlExtraData(window.get_url().get_arc()); let quirks_mode = window.Document().quirks_mode(); // FIXME(emilio): This looks somewhat fishy, since we use the context // only to parse the media query list, CssRuleType::Media doesn't make // much sense. let context = ParserContext::new( Origin::Author, &url_data, Some(CssRuleType::Media), ParsingMode::DEFAULT, quirks_mode, /* namespaces = */ Default::default(), window.css_error_reporter(), None, ); StyleMediaList::parse(&context, &mut parser) } /// pub(crate) fn parse_media_query<'i>( value: &'i str, window: &Window, ) -> Result> { let mut input = ParserInput::new(value); let mut parser = Parser::new(&mut input); let url_data = UrlExtraData(window.get_url().get_arc()); let quirks_mode = window.Document().quirks_mode(); let context = ParserContext::new( Origin::Author, &url_data, Some(CssRuleType::Media), ParsingMode::DEFAULT, quirks_mode, /* namespaces = */ Default::default(), window.css_error_reporter(), None, ); MediaQuery::parse(&context, &mut parser) } pub(crate) fn clone_media_list(&self) -> StyleMediaList { let guard = self.shared_lock().read(); self.media_queries.read_with(&guard).clone() } } impl MediaListMethods for MediaList { /// fn MediaText(&self) -> DOMString { let guard = self.shared_lock().read(); DOMString::from(self.media_queries.read_with(&guard).to_css_string()) } /// fn SetMediaText(&self, value: DOMString) { let global = self.global(); let mut guard = self.shared_lock().write(); let media_queries = self.media_queries.write_with(&mut guard); *media_queries = Self::parse_media_list(&value, global.as_window()); } // https://drafts.csswg.org/cssom/#dom-medialist-length fn Length(&self) -> u32 { let guard = self.shared_lock().read(); self.media_queries.read_with(&guard).media_queries.len() as u32 } /// fn Item(&self, index: u32) -> Option { let guard = self.shared_lock().read(); self.media_queries .read_with(&guard) .media_queries .get(index as usize) .map(|query| query.to_css_string().into()) } /// fn IndexedGetter(&self, index: u32) -> Option { self.Item(index) } /// fn AppendMedium(&self, medium: DOMString) { // Step 1 let global = self.global(); let m = Self::parse_media_query(&medium, global.as_window()); // Step 2 if m.is_err() { return; } // Step 3 let m_serialized = m.clone().unwrap().to_css_string(); let mut guard = self.shared_lock().write(); let mq = self.media_queries.write_with(&mut guard); let any = mq .media_queries .iter() .any(|q| m_serialized == q.to_css_string()); if any { return; } // Step 4 mq.media_queries.push(m.unwrap()); } /// fn DeleteMedium(&self, medium: DOMString) { // Step 1 let global = self.global(); let m = Self::parse_media_query(&medium, global.as_window()); // Step 2 if m.is_err() { return; } // Step 3 let m_serialized = m.unwrap().to_css_string(); let mut guard = self.shared_lock().write(); let media_list = self.media_queries.write_with(&mut guard); let new_vec = media_list .media_queries .drain(..) .filter(|q| m_serialized != q.to_css_string()) .collect(); media_list.media_queries = new_vec; } }