diff options
-rw-r--r-- | components/script/dom/cssmediarule.rs | 20 | ||||
-rw-r--r-- | components/script/dom/medialist.rs | 122 | ||||
-rw-r--r-- | components/script/dom/mod.rs | 1 | ||||
-rw-r--r-- | components/script/dom/webidls/CSSMediaRule.webidl | 2 | ||||
-rw-r--r-- | components/script/dom/webidls/MediaList.webidl | 13 | ||||
-rw-r--r-- | components/style/media_queries.rs | 4 | ||||
-rw-r--r-- | tests/wpt/metadata-css/cssom-1_dev/html/interfaces.htm.ini | 27 | ||||
-rw-r--r-- | tests/wpt/metadata/MANIFEST.json | 6 | ||||
-rw-r--r-- | tests/wpt/mozilla/tests/mozilla/interfaces.html | 1 | ||||
-rw-r--r-- | tests/wpt/web-platform-tests/cssom/MediaList.html | 43 |
10 files changed, 207 insertions, 32 deletions
diff --git a/components/script/dom/cssmediarule.rs b/components/script/dom/cssmediarule.rs index e42c1bc0021..7f4f43f3d5b 100644 --- a/components/script/dom/cssmediarule.rs +++ b/components/script/dom/cssmediarule.rs @@ -3,12 +3,14 @@ * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ use dom::bindings::codegen::Bindings::CSSMediaRuleBinding; -use dom::bindings::js::Root; -use dom::bindings::reflector::reflect_dom_object; +use dom::bindings::codegen::Bindings::CSSMediaRuleBinding::CSSMediaRuleMethods; +use dom::bindings::js::{JS, MutNullableHeap, Root}; +use dom::bindings::reflector::{Reflectable, reflect_dom_object}; use dom::bindings::str::DOMString; use dom::cssgroupingrule::CSSGroupingRule; use dom::cssrule::SpecificCSSRule; use dom::cssstylesheet::CSSStyleSheet; +use dom::medialist::MediaList; use dom::window::Window; use parking_lot::RwLock; use std::sync::Arc; @@ -20,6 +22,7 @@ pub struct CSSMediaRule { cssrule: CSSGroupingRule, #[ignore_heap_size_of = "Arc"] mediarule: Arc<RwLock<MediaRule>>, + medialist: MutNullableHeap<JS<MediaList>>, } impl CSSMediaRule { @@ -29,6 +32,7 @@ impl CSSMediaRule { CSSMediaRule { cssrule: CSSGroupingRule::new_inherited(parent_stylesheet, list), mediarule: mediarule, + medialist: MutNullableHeap::new(None), } } @@ -39,6 +43,11 @@ impl CSSMediaRule { window, CSSMediaRuleBinding::Wrap) } + + fn medialist(&self) -> Root<MediaList> { + self.medialist.or_init(|| MediaList::new(self.global().as_window(), + self.mediarule.read().media_queries.clone())) + } } impl SpecificCSSRule for CSSMediaRule { @@ -51,3 +60,10 @@ impl SpecificCSSRule for CSSMediaRule { self.mediarule.read().to_css_string().into() } } + +impl CSSMediaRuleMethods for CSSMediaRule { + // https://drafts.csswg.org/cssom/#dom-cssgroupingrule-media + fn Media(&self) -> Root<MediaList> { + self.medialist() + } +} diff --git a/components/script/dom/medialist.rs b/components/script/dom/medialist.rs new file mode 100644 index 00000000000..b6e5fd9c1ee --- /dev/null +++ b/components/script/dom/medialist.rs @@ -0,0 +1,122 @@ +/* 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 http://mozilla.org/MPL/2.0/. */ + +use core::default::Default; +use cssparser::Parser; +use dom::bindings::codegen::Bindings::MediaListBinding; +use dom::bindings::codegen::Bindings::MediaListBinding::MediaListMethods; +use dom::bindings::js::Root; +use dom::bindings::reflector::{Reflector, reflect_dom_object}; +use dom::bindings::str::DOMString; +use dom::window::Window; +use parking_lot::RwLock; +use std::sync::Arc; +use style::media_queries::{MediaQuery, parse_media_query_list}; +use style::media_queries::MediaList as StyleMediaList; +use style_traits::ToCss; + +#[dom_struct] +pub struct MediaList { + reflector_: Reflector, + #[ignore_heap_size_of = "Arc"] + media_queries: Arc<RwLock<StyleMediaList>>, +} + +impl MediaList { + #[allow(unrooted_must_root)] + pub fn new_inherited(media_queries: Arc<RwLock<StyleMediaList>>) -> MediaList { + MediaList { + reflector_: Reflector::new(), + media_queries: media_queries, + } + } + + #[allow(unrooted_must_root)] + pub fn new(window: &Window, media_queries: Arc<RwLock<StyleMediaList>>) + -> Root<MediaList> { + reflect_dom_object(box MediaList::new_inherited(media_queries), + window, + MediaListBinding::Wrap) + } + +} + +impl MediaListMethods for MediaList { + // https://drafts.csswg.org/cssom/#dom-medialist-mediatext + fn MediaText(&self) -> DOMString { + DOMString::from(self.media_queries.read().to_css_string()) + } + + // https://drafts.csswg.org/cssom/#dom-medialist-mediatext + fn SetMediaText(&self, value: DOMString) { + let mut media_queries = self.media_queries.write(); + // Step 2 + if value.is_empty() { + // Step 1 + *media_queries = StyleMediaList::default(); + return; + } + // Step 3 + let mut parser = Parser::new(&value); + *media_queries = parse_media_query_list(&mut parser); + } + + // https://drafts.csswg.org/cssom/#dom-medialist-length + fn Length(&self) -> u32 { + self.media_queries.read().media_queries.len() as u32 + } + + // https://drafts.csswg.org/cssom/#dom-medialist-item + fn Item(&self, index: u32) -> Option<DOMString> { + self.media_queries.read().media_queries.get(index as usize) + .and_then(|query| { + let mut s = String::new(); + query.to_css(&mut s).unwrap(); + Some(DOMString::from_string(s)) + }) + } + + // https://drafts.csswg.org/cssom/#dom-medialist-item + fn IndexedGetter(&self, index: u32) -> Option<DOMString> { + self.Item(index) + } + + // https://drafts.csswg.org/cssom/#dom-medialist-appendmedium + fn AppendMedium(&self, medium: DOMString) { + // Step 1 + let mut parser = Parser::new(&medium); + let m = MediaQuery::parse(&mut parser); + // Step 2 + if let Err(_) = m { + return; + } + // Step 3 + let m_serialized = m.clone().unwrap().to_css_string(); + let any = self.media_queries.read().media_queries.iter() + .any(|q| m_serialized == q.to_css_string()); + if any { + return; + } + // Step 4 + self.media_queries.write().media_queries.push(m.unwrap()); + } + + // https://drafts.csswg.org/cssom/#dom-medialist-deletemedium + fn DeleteMedium(&self, medium: DOMString) { + // Step 1 + let mut parser = Parser::new(&medium); + let m = MediaQuery::parse(&mut parser); + // Step 2 + if let Err(_) = m { + return; + } + // Step 3 + let m_serialized = m.unwrap().to_css_string(); + let mut media_list = self.media_queries.write(); + let new_vec = media_list.media_queries.drain(..) + .filter(|q| m_serialized != q.to_css_string()) + .collect(); + media_list.media_queries = new_vec; + } +} diff --git a/components/script/dom/mod.rs b/components/script/dom/mod.rs index 23209d89e07..9a0190cb57e 100644 --- a/components/script/dom/mod.rs +++ b/components/script/dom/mod.rs @@ -368,6 +368,7 @@ pub mod imagedata; pub mod keyboardevent; pub mod location; pub mod mediaerror; +pub mod medialist; pub mod mediaquerylist; pub mod messageevent; pub mod mimetype; diff --git a/components/script/dom/webidls/CSSMediaRule.webidl b/components/script/dom/webidls/CSSMediaRule.webidl index e2b89dee723..9ed133fb065 100644 --- a/components/script/dom/webidls/CSSMediaRule.webidl +++ b/components/script/dom/webidls/CSSMediaRule.webidl @@ -5,5 +5,5 @@ // https://drafts.csswg.org/cssom/#the-cssmediarule-interface [Exposed=Window] interface CSSMediaRule : CSSGroupingRule { - // [SameObject, PutForwards=mediaText] readonly attribute MediaList media; + [SameObject, PutForwards=mediaText] readonly attribute MediaList media; }; diff --git a/components/script/dom/webidls/MediaList.webidl b/components/script/dom/webidls/MediaList.webidl new file mode 100644 index 00000000000..f397badfd52 --- /dev/null +++ b/components/script/dom/webidls/MediaList.webidl @@ -0,0 +1,13 @@ +/* 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 http://mozilla.org/MPL/2.0/. */ + +// https://drafts.csswg.org/cssom/#the-medialist-interface +// [LegacyArrayClass] +interface MediaList { + [TreatNullAs=EmptyString] /* stringifier */ attribute DOMString mediaText; + readonly attribute unsigned long length; + getter DOMString? item(unsigned long index); + void appendMedium(DOMString medium); + void deleteMedium(DOMString medium); +}; diff --git a/components/style/media_queries.rs b/components/style/media_queries.rs index 522a252a05b..5ac70856068 100644 --- a/components/style/media_queries.rs +++ b/components/style/media_queries.rs @@ -85,7 +85,7 @@ pub enum Qualifier { Not, } -#[derive(Debug, PartialEq)] +#[derive(Clone, Debug, PartialEq)] #[cfg_attr(feature = "servo", derive(HeapSizeOf))] pub struct MediaQuery { pub qualifier: Option<Qualifier>, @@ -206,7 +206,7 @@ impl Expression { } impl MediaQuery { - fn parse(input: &mut Parser) -> Result<MediaQuery, ()> { + pub fn parse(input: &mut Parser) -> Result<MediaQuery, ()> { let mut expressions = vec![]; let qualifier = if input.try(|input| input.expect_ident_matching("only")).is_ok() { diff --git a/tests/wpt/metadata-css/cssom-1_dev/html/interfaces.htm.ini b/tests/wpt/metadata-css/cssom-1_dev/html/interfaces.htm.ini index ae5a08878e2..7747a24e315 100644 --- a/tests/wpt/metadata-css/cssom-1_dev/html/interfaces.htm.ini +++ b/tests/wpt/metadata-css/cssom-1_dev/html/interfaces.htm.ini @@ -75,33 +75,9 @@ [SVGElement interface: attribute style] expected: FAIL - [MediaList interface: existence and properties of interface object] - expected: FAIL - - [MediaList interface object length] - expected: FAIL - [MediaList interface: existence and properties of interface prototype object] expected: FAIL - [MediaList interface: existence and properties of interface prototype object's "constructor" property] - expected: FAIL - - [MediaList interface: attribute mediaText] - expected: FAIL - - [MediaList interface: attribute length] - expected: FAIL - - [MediaList interface: operation item(unsigned long)] - expected: FAIL - - [MediaList interface: operation appendMedium(DOMString)] - expected: FAIL - - [MediaList interface: operation deleteMedium(DOMString)] - expected: FAIL - [StyleSheet interface: attribute type] expected: FAIL @@ -177,9 +153,6 @@ [CSSImportRule interface: attribute styleSheet] expected: FAIL - [CSSMediaRule interface: attribute media] - expected: FAIL - [CSSPageRule interface: existence and properties of interface object] expected: FAIL diff --git a/tests/wpt/metadata/MANIFEST.json b/tests/wpt/metadata/MANIFEST.json index 908885a0b9d..2176f9ef317 100644 --- a/tests/wpt/metadata/MANIFEST.json +++ b/tests/wpt/metadata/MANIFEST.json @@ -39677,6 +39677,12 @@ "deleted_reftests": {}, "items": { "testharness": { + "cssom/MediaList.html": [ + { + "path": "cssom/MediaList.html", + "url": "/cssom/MediaList.html" + } + ], "cssom/shorthand-serialization.html": [ { "path": "cssom/shorthand-serialization.html", diff --git a/tests/wpt/mozilla/tests/mozilla/interfaces.html b/tests/wpt/mozilla/tests/mozilla/interfaces.html index 1e09f096618..a658a8eb346 100644 --- a/tests/wpt/mozilla/tests/mozilla/interfaces.html +++ b/tests/wpt/mozilla/tests/mozilla/interfaces.html @@ -140,6 +140,7 @@ test_interfaces([ "KeyboardEvent", "Location", "MediaError", + "MediaList", "MediaQueryList", "MessageEvent", "MimeType", diff --git a/tests/wpt/web-platform-tests/cssom/MediaList.html b/tests/wpt/web-platform-tests/cssom/MediaList.html new file mode 100644 index 00000000000..b39fb20478e --- /dev/null +++ b/tests/wpt/web-platform-tests/cssom/MediaList.html @@ -0,0 +1,43 @@ +<!doctype html> +<html> +<head> + <meta charset="utf-8"> + <title>CSSOM - MediaList interface</title> + <script src="/resources/testharness.js"></script> + <script src="/resources/testharnessreport.js"></script> + <style> + @media screen and (min-width: 480px), print, projection {} + </style> + + <script> + test(function () { + var media = document.styleSheets[0].cssRules[0].media; + assert_equals(media.length, 3, "MediaList length attribute"); + assert_equals(media.mediaText, "screen and (min-width: 480px), print, projection", "MediaList mediaText attribute"); + assert_equals(media[0], "screen and (min-width: 480px)", "MediaList indexed getter"); + assert_equals(media[1], "print", "MediaList indexed getter"); + assert_equals(media[2], "projection", "MediaList indexed getter"); + assert_equals(media[3], undefined, "MediaList indexed getter with out of range"); + assert_equals(media.item(0), "screen and (min-width: 480px)", "MediaList item method"); + assert_equals(media.item(3), null, "MediaList item method"); + + media.deleteMedium("print"); + assert_equals(media.length, 2, "MediaList length attribute after delete method"); + assert_equals(media.mediaText, "screen and (min-width: 480px), projection", "MediaList mediaText attribute after delete method"); + assert_equals(media[1], "projection", "MediaList indexed getter after delete method"); + assert_equals(media[2], undefined, "MediaList indexed getter with out of range after delete method"); + assert_equals(media.item(1), "projection", "MediaList indexed getter after delete method"); + assert_equals(media.item(2), null, "MediaList item method after delete method"); + + media.appendMedium("speech"); + assert_equals(media.length, 3, "MediaList length attribute after append method"); + assert_equals(media.mediaText, "screen and (min-width: 480px), projection, speech", "MediaList mediaText attribute after append method"); + assert_equals(media[1], "projection", "MediaList indexed getter after append method"); + assert_equals(media[2], "speech", "MediaList indexed getter after append method"); + assert_equals(media[3], undefined, "MediaList indexed getter with out of range after append method"); + assert_equals(media.item(2), "speech", "MediaList item method after append method"); + assert_equals(media.item(3), null, "MediaList item method after append method"); + }); + </script> +</head> +</html> |