/* 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 servo_arc::Arc; use style::shared_lock::SharedRwLock; use style::stylesheets::{CssRuleTypes, Stylesheet as StyleStyleSheet}; use crate::dom::bindings::codegen::Bindings::CSSStyleSheetBinding::CSSStyleSheetMethods; use crate::dom::bindings::error::{Error, ErrorResult, Fallible}; use crate::dom::bindings::reflector::{reflect_dom_object, DomGlobal}; 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, } impl CSSStyleSheet { fn new_inherited( owner: &Element, type_: DOMString, href: Option, title: Option, stylesheet: Arc, ) -> CSSStyleSheet { CSSStyleSheet { stylesheet: StyleSheet::new_inherited(type_, href, title), owner: MutNullableDom::new(Some(owner)), rulelist: MutNullableDom::new(None), style_stylesheet: stylesheet, origin_clean: Cell::new(true), } } #[cfg_attr(crown, allow(crown::unrooted_must_root))] pub(crate) fn new( window: &Window, owner: &Element, type_: DOMString, href: Option, title: Option, stylesheet: Arc, ) -> DomRoot { reflect_dom_object( Box::new(CSSStyleSheet::new_inherited( owner, type_, href, title, stylesheet, )), window, CanGc::note(), ) } fn rulelist(&self) -> DomRoot { self.rulelist.or_init(|| { let rules = self.style_stylesheet.contents.rules.clone(); CSSRuleList::new(self.global().as_window(), self, RulesSource::Rules(rules)) }) } 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) -> DomRoot { MediaList::new( self.global().as_window(), self, self.style_stylesheet().media.clone(), ) } } impl CSSStyleSheetMethods for CSSStyleSheet { // https://drafts.csswg.org/cssom/#dom-cssstylesheet-cssrules fn GetCssRules(&self) -> Fallible> { if !self.origin_clean.get() { return Err(Error::Security); } Ok(self.rulelist()) } // https://drafts.csswg.org/cssom/#dom-cssstylesheet-insertrule fn InsertRule(&self, rule: DOMString, index: u32) -> Fallible { if !self.origin_clean.get() { return Err(Error::Security); } self.rulelist() .insert_rule(&rule, index, CssRuleTypes::default(), None) } // https://drafts.csswg.org/cssom/#dom-cssstylesheet-deleterule fn DeleteRule(&self, index: u32) -> ErrorResult { if !self.origin_clean.get() { return Err(Error::Security); } self.rulelist().remove_rule(index) } }