/* 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 dom::bindings::cell::DomRefCell; use dom::bindings::codegen::Bindings::EventListenerBinding::EventListener; use dom::bindings::codegen::Bindings::EventTargetBinding::AddEventListenerOptions; use dom::bindings::codegen::Bindings::EventTargetBinding::EventListenerOptions; use dom::bindings::codegen::Bindings::MediaQueryListBinding::{self, MediaQueryListMethods}; use dom::bindings::inheritance::Castable; use dom::bindings::reflector::DomObject; use dom::bindings::reflector::reflect_dom_object; use dom::bindings::root::{Dom, DomRoot}; use dom::bindings::str::DOMString; use dom::bindings::trace::JSTraceable; use dom::bindings::weakref::{WeakRef, WeakRefVec}; use dom::document::Document; use dom::event::Event; use dom::eventtarget::EventTarget; use dom::mediaquerylistevent::MediaQueryListEvent; use dom_struct::dom_struct; use js::jsapi::JSTracer; use std::cell::Cell; use std::rc::Rc; use style::media_queries::MediaList; use style_traits::ToCss; pub enum MediaQueryListMatchState { Same(bool), Changed(bool), } #[dom_struct] pub struct MediaQueryList { eventtarget: EventTarget, document: Dom, media_query_list: MediaList, last_match_state: Cell> } impl MediaQueryList { fn new_inherited(document: &Document, media_query_list: MediaList) -> MediaQueryList { MediaQueryList { eventtarget: EventTarget::new_inherited(), document: Dom::from_ref(document), media_query_list: media_query_list, last_match_state: Cell::new(None), } } pub fn new(document: &Document, media_query_list: MediaList) -> DomRoot { reflect_dom_object(Box::new(MediaQueryList::new_inherited(document, media_query_list)), document.window(), MediaQueryListBinding::Wrap) } } impl MediaQueryList { fn evaluate_changes(&self) -> MediaQueryListMatchState { let matches = self.evaluate(); let result = if let Some(old_matches) = self.last_match_state.get() { if old_matches == matches { MediaQueryListMatchState::Same(matches) } else { MediaQueryListMatchState::Changed(matches) } } else { MediaQueryListMatchState::Changed(matches) }; self.last_match_state.set(Some(matches)); result } pub fn evaluate(&self) -> bool { self.document.device().map_or(false, |device| { self.media_query_list.evaluate(&device, self.document.quirks_mode()) }) } } impl MediaQueryListMethods for MediaQueryList { // https://drafts.csswg.org/cssom-view/#dom-mediaquerylist-media fn Media(&self) -> DOMString { let mut s = String::new(); self.media_query_list.to_css(&mut s).unwrap(); DOMString::from_string(s) } // https://drafts.csswg.org/cssom-view/#dom-mediaquerylist-matches fn Matches(&self) -> bool { match self.last_match_state.get() { None => self.evaluate(), Some(state) => state, } } // https://drafts.csswg.org/cssom-view/#dom-mediaquerylist-addlistener fn AddListener(&self, listener: Option>) { self.upcast::().add_event_listener( DOMString::from_string("change".to_owned()), listener, AddEventListenerOptions { parent: EventListenerOptions { capture: false } }, ); } // https://drafts.csswg.org/cssom-view/#dom-mediaquerylist-removelistener fn RemoveListener(&self, listener: Option>) { self.upcast::().remove_event_listener( DOMString::from_string("change".to_owned()), listener, EventListenerOptions { capture: false }, ); } // https://drafts.csswg.org/cssom-view/#dom-mediaquerylist-onchange event_handler!(change, GetOnchange, SetOnchange); } #[derive(HeapSizeOf)] pub struct WeakMediaQueryListVec { cell: DomRefCell>, } impl WeakMediaQueryListVec { /// Create a new vector of weak references to MediaQueryList pub fn new() -> Self { WeakMediaQueryListVec { cell: DomRefCell::new(WeakRefVec::new()) } } pub fn push(&self, mql: &MediaQueryList) { self.cell.borrow_mut().push(WeakRef::new(mql)); } /// Evaluate media query lists and report changes /// https://drafts.csswg.org/cssom-view/#evaluate-media-queries-and-report-changes pub fn evaluate_and_report_changes(&self) { rooted_vec!(let mut mql_list); self.cell.borrow_mut().update(|mql| { let mql = mql.root().unwrap(); if let MediaQueryListMatchState::Changed(_) = mql.evaluate_changes() { // Recording list of changed Media Queries mql_list.push(Dom::from_ref(&*mql)); } }); // Sending change events for all changed Media Queries for mql in mql_list.iter() { let event = MediaQueryListEvent::new(&mql.global(), atom!("change"), false, false, mql.Media(), mql.Matches()); event.upcast::().fire(mql.upcast::()); } } } #[allow(unsafe_code)] unsafe impl JSTraceable for WeakMediaQueryListVec { unsafe fn trace(&self, _: *mut JSTracer) { self.cell.borrow_mut().retain_alive() } }