aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--components/script/dom/cssmediarule.rs20
-rw-r--r--components/script/dom/medialist.rs122
-rw-r--r--components/script/dom/mod.rs1
-rw-r--r--components/script/dom/webidls/CSSMediaRule.webidl2
-rw-r--r--components/script/dom/webidls/MediaList.webidl13
-rw-r--r--components/style/media_queries.rs4
-rw-r--r--tests/wpt/metadata-css/cssom-1_dev/html/interfaces.htm.ini27
-rw-r--r--tests/wpt/metadata/MANIFEST.json6
-rw-r--r--tests/wpt/mozilla/tests/mozilla/interfaces.html1
-rw-r--r--tests/wpt/web-platform-tests/cssom/MediaList.html43
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>