/* 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::RefCell; use cssparser::{Parser, ParserInput}; use dom_struct::dom_struct; use js::context::JSContext; use servo_arc::Arc; use style::shared_lock::{Locked, SharedRwLockReadGuard, ToCssWithGuard}; use style::stylesheets::keyframes_rule::{Keyframe, KeyframeSelector, KeyframesRule}; use style::stylesheets::{CssRuleType, StylesheetInDocument}; use style::values::KeyframesName; use super::csskeyframerule::CSSKeyframeRule; use super::cssrule::{CSSRule, SpecificCSSRule}; use super::cssrulelist::{CSSRuleList, RulesSource}; use super::cssstylesheet::CSSStyleSheet; use crate::dom::bindings::codegen::Bindings::CSSKeyframesRuleBinding::CSSKeyframesRuleMethods; use crate::dom::bindings::error::ErrorResult; use crate::dom::bindings::inheritance::Castable; use crate::dom::bindings::reflector::{DomGlobal, reflect_dom_object_with_cx}; use crate::dom::bindings::root::{DomRoot, MutNullableDom}; use crate::dom::bindings::str::DOMString; use crate::dom::window::Window; #[dom_struct] pub(crate) struct CSSKeyframesRule { css_rule: CSSRule, #[ignore_malloc_size_of = "Stylo"] #[no_trace] keyframes_rule: RefCell>>, rule_list: MutNullableDom, } impl CSSKeyframesRule { fn new_inherited( parent_stylesheet: &CSSStyleSheet, keyframesrule: Arc>, ) -> CSSKeyframesRule { CSSKeyframesRule { css_rule: CSSRule::new_inherited(parent_stylesheet), keyframes_rule: RefCell::new(keyframesrule), rule_list: MutNullableDom::new(None), } } pub(crate) fn new( cx: &mut JSContext, window: &Window, parent_stylesheet: &CSSStyleSheet, keyframesrule: Arc>, ) -> DomRoot { reflect_dom_object_with_cx( Box::new(CSSKeyframesRule::new_inherited( parent_stylesheet, keyframesrule, )), window, cx, ) } fn rulelist(&self, cx: &mut JSContext) -> DomRoot { self.rule_list.or_init(|| { let parent_stylesheet = &self.upcast::().parent_stylesheet(); CSSRuleList::new( cx, self.global().as_window(), parent_stylesheet, RulesSource::Keyframes(self.keyframes_rule.borrow().clone()), ) }) } /// Given a keyframe selector, finds the index of the first corresponding rule if any fn find_rule(&self, selector: &DOMString) -> Option { let selector = selector.str(); let mut input = ParserInput::new(&selector); let mut input = Parser::new(&mut input); if let Ok(sel) = KeyframeSelector::parse(&mut input) { let guard = self.css_rule.shared_lock().read(); // This finds the *last* element matching a selector // because that's the rule that applies. Thus, rposition self.keyframes_rule .borrow() .read_with(&guard) .keyframes .iter() .rposition(|frame| frame.read_with(&guard).selector == sel) } else { None } } pub(crate) fn update_rule( &self, keyframesrule: Arc>, guard: &SharedRwLockReadGuard, ) { if let Some(rulelist) = self.rule_list.get() { rulelist.update_rules(RulesSource::Keyframes(keyframesrule.clone()), guard); } *self.keyframes_rule.borrow_mut() = keyframesrule; } } impl CSSKeyframesRuleMethods for CSSKeyframesRule { /// fn CssRules(&self, cx: &mut JSContext) -> DomRoot { self.rulelist(cx) } /// fn AppendRule(&self, cx: &mut JSContext, rule: DOMString) { let style_stylesheet = self.css_rule.parent_stylesheet().style_stylesheet(); let rule = rule.str(); let rule = { let guard = style_stylesheet.shared_lock.read(); Keyframe::parse( &rule, style_stylesheet.contents(&guard), &style_stylesheet.shared_lock, ) }; if let Ok(rule) = rule { self.css_rule.parent_stylesheet().will_modify(); let mut guard = self.css_rule.shared_lock().write(); self.keyframes_rule .borrow() .write_with(&mut guard) .keyframes .push(rule); self.rulelist(cx).append_lazy_dom_rule(); self.css_rule.parent_stylesheet().notify_invalidations(); } } /// fn DeleteRule(&self, cx: &mut JSContext, selector: DOMString) { if let Some(idx) = self.find_rule(&selector) { let _ = self.rulelist(cx).remove_rule(idx as u32); } } /// fn FindRule( &self, cx: &mut JSContext, selector: DOMString, ) -> Option> { self.find_rule(&selector) .and_then(|idx| self.rulelist(cx).item(cx, idx as u32)) .and_then(DomRoot::downcast) } /// fn Name(&self) -> DOMString { let guard = self.css_rule.shared_lock().read(); DOMString::from( &**self .keyframes_rule .borrow() .read_with(&guard) .name .as_atom(), ) } /// fn SetName(&self, value: DOMString) -> ErrorResult { // Spec deviation: https://github.com/w3c/csswg-drafts/issues/801 // Setting this property to a CSS-wide keyword or `none` does not throw, // it stores a value that serializes as a quoted string. self.css_rule.parent_stylesheet().will_modify(); let name = KeyframesName::from_ident(&value.str()); let mut guard = self.css_rule.shared_lock().write(); self.keyframes_rule.borrow().write_with(&mut guard).name = name; self.css_rule.parent_stylesheet().notify_invalidations(); Ok(()) } } impl SpecificCSSRule for CSSKeyframesRule { fn ty(&self) -> CssRuleType { CssRuleType::Keyframes } fn get_css(&self) -> DOMString { let guard = self.css_rule.shared_lock().read(); self.keyframes_rule .borrow() .read_with(&guard) .to_css_string(&guard) .into() } fn deparent_children(&self) { if let Some(list) = self.rule_list.get() { list.deparent_all() } } }