/* 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 std::cell::Cell; use dom_struct::dom_struct; use js::rust::HandleObject; use servo_arc::Arc; use style::media_queries::MediaList as StyleMediaList; use style::shared_lock::SharedRwLock; use style::stylesheets::{ AllowImportRules, CssRuleTypes, Origin, Stylesheet as StyleStyleSheet, UrlExtraData, }; use crate::dom::bindings::codegen::Bindings::CSSStyleSheetBinding::{ CSSStyleSheetInit, CSSStyleSheetMethods, }; use crate::dom::bindings::codegen::Bindings::WindowBinding::WindowMethods; use crate::dom::bindings::codegen::GenericBindings::CSSRuleListBinding::CSSRuleList_Binding::CSSRuleListMethods; use crate::dom::bindings::codegen::UnionTypes::MediaListOrString; use crate::dom::bindings::error::{Error, ErrorResult, Fallible}; use crate::dom::bindings::reflector::{ DomGlobal, reflect_dom_object, reflect_dom_object_with_proto, }; use crate::dom::bindings::root::{DomRoot, MutNullableDom}; use crate::dom::bindings::str::DOMString; use crate::dom::cssrulelist::{CSSRuleList, RulesSource}; use crate::dom::element::Element; use crate::dom::medialist::MediaList; use crate::dom::node::NodeTraits; use crate::dom::stylesheet::StyleSheet; use crate::dom::window::Window; use crate::script_runtime::CanGc; #[dom_struct] pub(crate) struct CSSStyleSheet { stylesheet: StyleSheet, owner: MutNullableDom, rulelist: MutNullableDom, #[ignore_malloc_size_of = "Arc"] #[no_trace] style_stylesheet: Arc, origin_clean: Cell, is_constructed: bool, } impl CSSStyleSheet { fn new_inherited( owner: Option<&Element>, type_: DOMString, href: Option, title: Option, stylesheet: Arc, is_constructed: bool, ) -> CSSStyleSheet { CSSStyleSheet { stylesheet: StyleSheet::new_inherited(type_, href, title), owner: MutNullableDom::new(owner), rulelist: MutNullableDom::new(None), style_stylesheet: stylesheet, origin_clean: Cell::new(true), is_constructed, } } #[cfg_attr(crown, allow(crown::unrooted_must_root))] #[allow(clippy::too_many_arguments)] pub(crate) fn new( window: &Window, owner: Option<&Element>, type_: DOMString, href: Option, title: Option, stylesheet: Arc, is_constructed: bool, can_gc: CanGc, ) -> DomRoot { reflect_dom_object( Box::new(CSSStyleSheet::new_inherited( owner, type_, href, title, stylesheet, is_constructed, )), window, can_gc, ) } #[cfg_attr(crown, allow(crown::unrooted_must_root))] #[allow(clippy::too_many_arguments)] fn new_with_proto( window: &Window, proto: Option, owner: Option<&Element>, type_: DOMString, href: Option, title: Option, stylesheet: Arc, is_constructed: bool, can_gc: CanGc, ) -> DomRoot { reflect_dom_object_with_proto( Box::new(CSSStyleSheet::new_inherited( owner, type_, href, title, stylesheet, is_constructed, )), window, proto, can_gc, ) } fn rulelist(&self, can_gc: CanGc) -> DomRoot { self.rulelist.or_init(|| { let rules = self.style_stylesheet.contents.rules.clone(); CSSRuleList::new( self.global().as_window(), self, RulesSource::Rules(rules), can_gc, ) }) } pub(crate) fn disabled(&self) -> bool { self.style_stylesheet.disabled() } pub(crate) fn get_owner(&self) -> Option> { self.owner.get() } pub(crate) fn set_disabled(&self, disabled: bool) { if self.style_stylesheet.set_disabled(disabled) && self.get_owner().is_some() { self.get_owner() .unwrap() .stylesheet_list_owner() .invalidate_stylesheets(); } } pub(crate) fn set_owner(&self, value: Option<&Element>) { self.owner.set(value); } pub(crate) fn shared_lock(&self) -> &SharedRwLock { &self.style_stylesheet.shared_lock } pub(crate) fn style_stylesheet(&self) -> &StyleStyleSheet { &self.style_stylesheet } pub(crate) fn set_origin_clean(&self, origin_clean: bool) { self.origin_clean.set(origin_clean); } pub(crate) fn medialist(&self, can_gc: CanGc) -> DomRoot { MediaList::new( self.global().as_window(), self, self.style_stylesheet().media.clone(), can_gc, ) } /// #[inline] pub(crate) fn is_constructed(&self) -> bool { self.is_constructed } } impl CSSStyleSheetMethods for CSSStyleSheet { /// fn Constructor( window: &Window, proto: Option, can_gc: CanGc, options: &CSSStyleSheetInit, ) -> DomRoot { let doc = window.Document(); let shared_lock = doc.style_shared_lock().clone(); let media = Arc::new(shared_lock.wrap(match &options.media { Some(media) => match media { MediaListOrString::MediaList(media_list) => media_list.clone_media_list(), MediaListOrString::String(str) => MediaList::parse_media_list(str, window), }, None => StyleMediaList::empty(), })); let stylesheet = Arc::new(StyleStyleSheet::from_str( "", UrlExtraData(window.get_url().get_arc()), Origin::Author, media, shared_lock, None, window.css_error_reporter(), doc.quirks_mode(), AllowImportRules::No, )); if options.disabled { stylesheet.set_disabled(true); } Self::new_with_proto( window, proto, None, // owner "text/css".into(), None, // href None, // title stylesheet, true, // is_constructed can_gc, ) } /// fn GetCssRules(&self, can_gc: CanGc) -> Fallible> { if !self.origin_clean.get() { return Err(Error::Security); } Ok(self.rulelist(can_gc)) } /// fn InsertRule(&self, rule: DOMString, index: u32, can_gc: CanGc) -> Fallible { if !self.origin_clean.get() { return Err(Error::Security); } self.rulelist(can_gc) .insert_rule(&rule, index, CssRuleTypes::default(), None, can_gc) } /// fn DeleteRule(&self, index: u32, can_gc: CanGc) -> ErrorResult { if !self.origin_clean.get() { return Err(Error::Security); } self.rulelist(can_gc).remove_rule(index) } /// fn GetRules(&self, can_gc: CanGc) -> Fallible> { self.GetCssRules(can_gc) } /// fn RemoveRule(&self, index: u32, can_gc: CanGc) -> ErrorResult { self.DeleteRule(index, can_gc) } /// fn AddRule( &self, selector: DOMString, block: DOMString, optional_index: Option, can_gc: CanGc, ) -> Fallible { // > 1. Let *rule* be an empty string. // > 2. Append *selector* to *rule*. let mut rule = selector; // > 3. Append " { " to *rule*. // > 4. If *block* is not empty, append *block*, followed by a space, to *rule*. // > 5. Append "}" to *rule*. if block.is_empty() { rule.push_str(" { }"); } else { rule.push_str(" { "); rule.push_str(block.str()); rule.push_str(" }"); }; // > 6. Let *index* be *optionalIndex* if provided, or the number of CSS rules in the stylesheet otherwise. let index = optional_index.unwrap_or_else(|| self.rulelist(can_gc).Length()); // > 7. Call `insertRule()`, with *rule* and *index* as arguments. self.InsertRule(rule, index, can_gc)?; // > 8. Return -1. Ok(-1) } }