style/
author_styles.rs

1/* This Source Code Form is subject to the terms of the Mozilla Public
2 * License, v. 2.0. If a copy of the MPL was not distributed with this
3 * file, You can obtain one at https://mozilla.org/MPL/2.0/. */
4
5//! A set of author stylesheets and their computed representation, such as the
6//! ones used for ShadowRoot.
7
8use crate::derives::*;
9use crate::dom::TElement;
10use crate::shared_lock::SharedRwLockReadGuard;
11use crate::stylesheet_set::AuthorStylesheetSet;
12use crate::stylesheets::StylesheetInDocument;
13use crate::stylist::CascadeData;
14use crate::stylist::Stylist;
15use servo_arc::Arc;
16use std::sync::LazyLock;
17
18/// A set of author stylesheets and their computed representation, such as the
19/// ones used for ShadowRoot.
20#[derive(MallocSizeOf)]
21pub struct GenericAuthorStyles<S>
22where
23    S: StylesheetInDocument + PartialEq + 'static,
24{
25    /// The sheet collection, which holds the sheet pointers, the invalidations,
26    /// and all that stuff.
27    pub stylesheets: AuthorStylesheetSet<S>,
28    /// The actual cascade data computed from the stylesheets.
29    #[ignore_malloc_size_of = "Measured as part of the stylist"]
30    pub data: Arc<CascadeData>,
31}
32
33pub use self::GenericAuthorStyles as AuthorStyles;
34
35static EMPTY_CASCADE_DATA: LazyLock<Arc<CascadeData>> =
36    LazyLock::new(|| Arc::new_leaked(CascadeData::new()));
37
38impl<S> GenericAuthorStyles<S>
39where
40    S: StylesheetInDocument + PartialEq + 'static,
41{
42    /// Create an empty AuthorStyles.
43    #[inline]
44    pub fn new() -> Self {
45        Self {
46            stylesheets: AuthorStylesheetSet::new(),
47            data: EMPTY_CASCADE_DATA.clone(),
48        }
49    }
50
51    /// Flush the pending sheet changes, updating `data` as appropriate.
52    ///
53    /// TODO(emilio): Need a host element and a snapshot map to do invalidation
54    /// properly.
55    #[inline]
56    pub fn flush<E>(&mut self, stylist: &mut Stylist, guard: &SharedRwLockReadGuard)
57    where
58        E: TElement,
59    {
60        let flusher = self
61            .stylesheets
62            .flush::<E>(/* host = */ None, /* snapshot_map = */ None);
63
64        let result = stylist.rebuild_author_data(&self.data, flusher.sheets, guard);
65        if let Ok(Some(new_data)) = result {
66            self.data = new_data;
67        }
68    }
69}