script/dom/css/
cssnesteddeclarations.rs1use std::cell::RefCell;
6
7use dom_struct::dom_struct;
8use js::context::JSContext;
9use servo_arc::Arc;
10use style::shared_lock::{Locked, SharedRwLockReadGuard, ToCssWithGuard};
11use style::stylesheets::{CssRuleType, NestedDeclarationsRule};
12
13use super::cssrule::{CSSRule, SpecificCSSRule};
14use super::cssstyledeclaration::{CSSModificationAccess, CSSStyleDeclaration, CSSStyleOwner};
15use super::cssstylesheet::CSSStyleSheet;
16use crate::dom::bindings::codegen::Bindings::CSSNestedDeclarationsBinding::CSSNestedDeclarationsMethods;
17use crate::dom::bindings::inheritance::Castable;
18use crate::dom::bindings::reflector::{DomGlobal, reflect_dom_object};
19use crate::dom::bindings::root::{Dom, DomRoot, MutNullableDom};
20use crate::dom::bindings::str::DOMString;
21use crate::dom::window::Window;
22use crate::script_runtime::CanGc;
23
24#[dom_struct]
25pub(crate) struct CSSNestedDeclarations {
26 css_rule: CSSRule,
27 #[ignore_malloc_size_of = "Stylo"]
28 #[no_trace]
29 nesteddeclarationsrule: RefCell<Arc<Locked<NestedDeclarationsRule>>>,
30 style_declaration: MutNullableDom<CSSStyleDeclaration>,
31}
32
33impl CSSNestedDeclarations {
34 pub(crate) fn new_inherited(
35 parent_stylesheet: &CSSStyleSheet,
36 nesteddeclarationsrule: Arc<Locked<NestedDeclarationsRule>>,
37 ) -> Self {
38 Self {
39 css_rule: CSSRule::new_inherited(parent_stylesheet),
40 nesteddeclarationsrule: RefCell::new(nesteddeclarationsrule),
41 style_declaration: Default::default(),
42 }
43 }
44
45 pub(crate) fn new(
46 window: &Window,
47 parent_stylesheet: &CSSStyleSheet,
48 nesteddeclarationsrule: Arc<Locked<NestedDeclarationsRule>>,
49 can_gc: CanGc,
50 ) -> DomRoot<Self> {
51 reflect_dom_object(
52 Box::new(Self::new_inherited(
53 parent_stylesheet,
54 nesteddeclarationsrule,
55 )),
56 window,
57 can_gc,
58 )
59 }
60
61 pub(crate) fn update_rule(
62 &self,
63 nesteddeclarationsrule: Arc<Locked<NestedDeclarationsRule>>,
64 guard: &SharedRwLockReadGuard,
65 ) {
66 if let Some(ref style_decl) = self.style_declaration.get() {
67 style_decl
68 .update_property_declaration_block(&nesteddeclarationsrule.read_with(guard).block);
69 }
70 *self.nesteddeclarationsrule.borrow_mut() = nesteddeclarationsrule;
71 }
72}
73
74impl SpecificCSSRule for CSSNestedDeclarations {
75 fn ty(&self) -> CssRuleType {
76 CssRuleType::NestedDeclarations
77 }
78
79 fn get_css(&self) -> DOMString {
80 let guard = self.css_rule.shared_lock().read();
81 self.nesteddeclarationsrule
82 .borrow()
83 .read_with(&guard)
84 .to_css_string(&guard)
85 .into()
86 }
87}
88
89impl CSSNestedDeclarationsMethods<crate::DomTypeHolder> for CSSNestedDeclarations {
90 fn Style(&self, cx: &mut JSContext) -> DomRoot<CSSStyleDeclaration> {
92 self.style_declaration.or_init(|| {
93 let guard = self.css_rule.shared_lock().read();
94 CSSStyleDeclaration::new(
95 self.global().as_window(),
96 CSSStyleOwner::CSSRule(
97 Dom::from_ref(self.upcast()),
98 RefCell::new(
99 self.nesteddeclarationsrule
100 .borrow()
101 .read_with(&guard)
102 .block
103 .clone(),
104 ),
105 ),
106 None,
107 CSSModificationAccess::ReadWrite,
108 CanGc::from_cx(cx),
109 )
110 })
111 }
112}