/* 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 script_bindings::root::Dom; 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::cell::DomRefCell; 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, USVString}; use crate::dom::cssrulelist::{CSSRuleList, RulesSource}; use crate::dom::document::Document; use crate::dom::element::Element; use crate::dom::medialist::MediaList; use crate::dom::node::NodeTraits; use crate::dom::stylesheet::StyleSheet; use crate::dom::stylesheetlist::StyleSheetListOwner; use crate::dom::window::Window; use crate::script_runtime::CanGc; #[dom_struct] pub(crate) struct CSSStyleSheet { stylesheet: StyleSheet, /// owner_node: MutNullableDom, /// rulelist: MutNullableDom, /// The inner Stylo's [Stylesheet]. #[ignore_malloc_size_of = "Arc"] #[no_trace] style_stylesheet: Arc, /// origin_clean: Cell, /// In which [Document] that this stylesheet was constructed. /// /// constructor_document: Option>, /// Documents or shadow DOMs thats adopt this stylesheet, they will be notified whenever /// the stylesheet is modified. adopters: DomRefCell>, } impl CSSStyleSheet { fn new_inherited( owner: Option<&Element>, type_: DOMString, href: Option, title: Option, stylesheet: Arc, constructor_document: Option<&Document>, ) -> CSSStyleSheet { CSSStyleSheet { stylesheet: StyleSheet::new_inherited(type_, href, title), owner_node: MutNullableDom::new(owner), rulelist: MutNullableDom::new(None), style_stylesheet: stylesheet, origin_clean: Cell::new(true), constructor_document: constructor_document.map(Dom::from_ref), adopters: Default::default(), } } #[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, constructor_document: Option<&Document>, can_gc: CanGc, ) -> DomRoot { reflect_dom_object( Box::new(CSSStyleSheet::new_inherited( owner, type_, href, title, stylesheet, constructor_document, )), 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, constructor_document: Option<&Document>, can_gc: CanGc, ) -> DomRoot { reflect_dom_object_with_proto( Box::new(CSSStyleSheet::new_inherited( owner, type_, href, title, stylesheet, constructor_document, )), 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 owner_node(&self) -> Option> { self.owner_node.get() } pub(crate) fn set_disabled(&self, disabled: bool) { if self.style_stylesheet.set_disabled(disabled) { self.notify_invalidations(); } } pub(crate) fn set_owner_node(&self, value: Option<&Element>) { self.owner_node.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 style_stylesheet_arc(&self) -> &Arc { &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.constructor_document.is_some() } pub(crate) fn constructor_document_matches(&self, other_doc: &Document) -> bool { match &self.constructor_document { Some(doc) => *doc == other_doc, None => false, } } /// Add a [StyleSheetListOwner] as an adopter to be notified whenever this stylesheet is /// modified. #[cfg_attr(crown, allow(crown::unrooted_must_root))] pub(crate) fn add_adopter(&self, owner: StyleSheetListOwner) { debug_assert!(self.is_constructed()); self.adopters.borrow_mut().push(owner); } pub(crate) fn remove_adopter(&self, owner: &StyleSheetListOwner) { let adopters = &mut *self.adopters.borrow_mut(); if let Some(index) = adopters.iter().position(|o| o == owner) { adopters.swap_remove(index); } } /// Invalidate all stylesheet set this stylesheet is a part on. pub(crate) fn notify_invalidations(&self) { if let Some(owner) = self.owner_node() { owner.stylesheet_list_owner().invalidate_stylesheets(); } for adopter in self.adopters.borrow().iter() { adopter.invalidate_stylesheets(); } } } 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, Some(&window.Document()), // constructor_document 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) } /// fn ReplaceSync(&self, text: USVString) -> Result<(), Error> { // Step 1. If the constructed flag is not set throw a NotAllowedError if !self.is_constructed() { return Err(Error::NotAllowed); } // Step 2. Let rules be the result of running parse a stylesheet’s contents from text. let global = self.global(); let window = global.as_window(); StyleStyleSheet::update_from_str( &self.style_stylesheet, &text, UrlExtraData(window.get_url().get_arc()), None, window.css_error_reporter(), AllowImportRules::No, // Step 3.If rules contains one or more @import rules, remove those rules from rules. ); // Step 4. Set sheet’s CSS rules to rules. // We reset our rule list, which will be initialized properly // at the next getter access. self.rulelist.set(None); // Notify invalidation to update the styles immediately. self.notify_invalidations(); Ok(()) } }