Skip to main content

freya_core/
tree.rs

1use std::{
2    any::Any,
3    borrow::Cow,
4    collections::{
5        VecDeque,
6        hash_map::Entry,
7    },
8    fmt::Debug,
9    rc::Rc,
10};
11
12use bitflags::bitflags;
13use freya_engine::prelude::{
14    FontCollection,
15    FontMgr,
16};
17use futures_channel::mpsc::UnboundedSender;
18use itertools::Itertools;
19use rustc_hash::{
20    FxHashMap,
21    FxHashSet,
22};
23use torin::{
24    prelude::{
25        Area,
26        LayoutMeasurer,
27        Size2D,
28    },
29    torin::{
30        DirtyReason,
31        Torin,
32    },
33};
34
35use crate::{
36    accessibility::groups::AccessibilityGroups,
37    data::{
38        AccessibilityState,
39        EffectState,
40        LayerState,
41        TextStyleState,
42    },
43    element::{
44        ElementExt,
45        LayoutContext,
46    },
47    elements::rect::RectElement,
48    events::{
49        data::{
50            EventType,
51            SizedEventData,
52        },
53        emittable::EmmitableEvent,
54        name::EventName,
55    },
56    extended_hashmap::ExtendedHashMap,
57    integration::{
58        AccessibilityDirtyNodes,
59        AccessibilityGenerator,
60        EventsChunk,
61    },
62    layers::Layers,
63    node_id::NodeId,
64    runner::{
65        MutationAdd,
66        MutationModified,
67        MutationMove,
68        MutationRemove,
69        Mutations,
70    },
71    text_cache::TextCache,
72    tree_layout_adapter::TreeAdapterFreya,
73};
74
75#[derive(Default)]
76pub struct Tree {
77    pub parents: FxHashMap<NodeId, NodeId>,
78    pub children: FxHashMap<NodeId, Vec<NodeId>>,
79    pub heights: FxHashMap<NodeId, u16>,
80
81    pub elements: FxHashMap<NodeId, Rc<dyn ElementExt>>,
82
83    // Event listeners
84    pub listeners: FxHashMap<EventName, Vec<NodeId>>,
85
86    // Derived states
87    pub layer_state: FxHashMap<NodeId, LayerState>,
88    pub accessibility_state: FxHashMap<NodeId, AccessibilityState>,
89    pub effect_state: FxHashMap<NodeId, EffectState>,
90    pub text_style_state: FxHashMap<NodeId, TextStyleState>,
91
92    // Other
93    pub layout: Torin<NodeId>,
94    pub layers: Layers,
95    pub text_cache: TextCache,
96
97    // Accessibility
98    pub accessibility_groups: AccessibilityGroups,
99    pub accessibility_diff: AccessibilityDirtyNodes,
100    pub accessibility_generator: AccessibilityGenerator,
101}
102
103impl Debug for Tree {
104    fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
105        f.debug_struct("Tree")
106            .field("children", &self.children.capacity())
107            .field("parents", &self.parents.capacity())
108            .field("elements", &self.elements.capacity())
109            .field("heights", &self.heights.capacity())
110            .field("listeners", &self.listeners.capacity())
111            .field("layer_state", &self.layer_state.capacity())
112            .field("layout_size", &self.layout.size())
113            .field("layers", &self.layers.capacity())
114            .field("effect_state", &self.effect_state.capacity())
115            .field("accessibility_state", &self.accessibility_state.capacity())
116            .field("text_style_state", &self.text_style_state.capacity())
117            .field("text_cache", &self.text_cache)
118            .finish()
119    }
120}
121
122impl Tree {
123    pub fn size(&self) -> usize {
124        self.elements.len()
125    }
126
127    pub fn traverse_depth(&self, mut then: impl FnMut(NodeId)) {
128        let mut buffer = vec![NodeId::ROOT];
129        while let Some(node_id) = buffer.pop() {
130            if let Some(children) = self.children.get(&node_id) {
131                buffer.extend(children.iter().rev());
132            }
133            then(node_id);
134        }
135    }
136
137    pub fn traverse_depth_cancel(&self, mut then: impl FnMut(NodeId) -> bool) {
138        let mut buffer = vec![NodeId::ROOT];
139        while let Some(node_id) = buffer.pop() {
140            if let Some(children) = self.children.get(&node_id) {
141                buffer.extend(children.iter().rev());
142            }
143            if then(node_id) {
144                break;
145            }
146        }
147    }
148
149    #[cfg_attr(feature = "hotpath", hotpath::measure)]
150    pub fn apply_mutations(&mut self, mutations: Mutations) -> MutationsApplyResult {
151        let mut needs_render = !mutations.removed.is_empty();
152        let mut needs_accessibility = !mutations.removed.is_empty();
153        let mut dirty = Vec::<(NodeId, DiffModifies)>::default();
154
155        #[cfg(debug_assertions)]
156        tracing::info!("{mutations:?}");
157
158        if let Entry::Vacant(e) = self.elements.entry(NodeId::ROOT) {
159            e.insert(Rc::new(RectElement::default()));
160            self.heights.insert(NodeId::ROOT, 0);
161            dirty.push((NodeId::ROOT, DiffModifies::all()));
162        }
163
164        hotpath::measure_block!("mutations run", {
165            for remove in mutations.removed.into_iter().sorted() {
166                let node_id = remove.node_id();
167                let mut buff = vec![remove];
168                let Some(parent_id) = self.parents.get(&node_id).copied() else {
169                    continue;
170                };
171                self.layout.invalidate(parent_id);
172                needs_render = true;
173
174                while let Some(remove) = buff.pop() {
175                    let node_id = remove.node_id();
176                    self.layout.raw_remove(node_id);
177
178                    let parent_id = self.parents.remove(&node_id).unwrap();
179
180                    // Remove element
181                    let old_element = self.elements.remove(&node_id).unwrap();
182
183                    if let Some(children) = self.children.get_mut(&parent_id) {
184                        match remove {
185                            MutationRemove::Element { index, .. } => {
186                                children.remove(index as usize);
187                            }
188                            MutationRemove::Scope { .. } => {
189                                children.retain(|id| *id != node_id);
190                            }
191                        }
192                    }
193
194                    // Remove its children too
195                    if let Some(children) = self.children.remove(&node_id) {
196                        buff.extend(children.into_iter().enumerate().map(|(i, e)| {
197                            MutationRemove::Element {
198                                id: e,
199                                index: i as u32,
200                            }
201                        }));
202                    }
203
204                    // Remove old events
205                    if let Some(events) = old_element.events_handlers() {
206                        for event in events.keys() {
207                            self.listeners
208                                .entry(*event)
209                                .or_default()
210                                .retain(|id| *id != node_id);
211                        }
212                    }
213
214                    // Remove from the layers
215                    let layer_state = self.layer_state.remove(&node_id).unwrap();
216                    layer_state.remove(node_id, &mut self.layers);
217
218                    // Remove from the accessibility
219                    let accessibility_state = self.accessibility_state.remove(&node_id).unwrap();
220                    accessibility_state.remove(
221                        node_id,
222                        parent_id,
223                        &mut self.accessibility_diff,
224                        &mut self.accessibility_groups,
225                    );
226
227                    // Remove from other states
228                    self.heights.remove(&node_id);
229                    self.effect_state.remove(&node_id);
230                    self.text_style_state.remove(&node_id);
231                    self.text_cache.remove(&node_id);
232                }
233            }
234
235            for MutationAdd {
236                node_id,
237                parent_id,
238                index,
239                element,
240            } in mutations
241                .added
242                .into_iter()
243                .sorted_by_key(|m| (m.parent_id, m.index))
244            {
245                let parent_height = *self.heights.entry(parent_id).or_default();
246
247                self.parents.insert(node_id, parent_id);
248                self.heights.insert(node_id, parent_height + 1);
249
250                let parent = self.children.entry(parent_id).or_default();
251
252                // TODO: Improve this
253                if parent.len() < index as usize + 1 {
254                    parent.resize(index as usize + 1, NodeId::PLACEHOLDER);
255
256                    parent[index as usize] = node_id;
257                } else if parent.get(index as usize) == Some(&NodeId::PLACEHOLDER) {
258                    parent[index as usize] = node_id;
259                } else {
260                    parent.insert(index as usize, node_id);
261                }
262
263                // Add events
264                if let Some(events) = element.events_handlers() {
265                    for event in events.keys() {
266                        self.listeners.entry(*event).or_default().push(node_id);
267                    }
268                }
269
270                self.elements.insert(node_id, element);
271                dirty.push((node_id, DiffModifies::all()));
272            }
273
274            for (parent_node_id, movements) in mutations.moved {
275                let parent = self.children.get_mut(&parent_node_id).unwrap();
276                for MutationMove { index: to, node_id } in
277                    movements.into_iter().sorted_by_key(|m| m.index)
278                {
279                    let from = parent.iter().position(|id| *id == node_id).unwrap();
280
281                    if from < to as usize {
282                        parent.insert(to as usize, node_id);
283                        parent.remove(from);
284                    } else {
285                        parent.remove(from);
286                        parent.insert(to as usize, node_id);
287                    }
288                }
289                let mut diff = DiffModifies::empty();
290                diff.insert(DiffModifies::REORDER_LAYOUT);
291                diff.insert(DiffModifies::ACCESSIBILITY);
292                diff.insert(DiffModifies::STYLE);
293                dirty.push((parent_node_id, diff));
294            }
295
296            for MutationModified {
297                node_id,
298                element,
299                flags,
300            } in mutations.modified
301            {
302                dirty.push((node_id, flags));
303
304                let old_element = self.elements.remove(&node_id).unwrap();
305
306                if flags.contains(DiffModifies::EVENT_HANDLERS) {
307                    // Remove old events
308                    if let Some(events) = old_element.events_handlers() {
309                        for event in events.keys() {
310                            self.listeners
311                                .entry(*event)
312                                .or_default()
313                                .retain(|id| *id != node_id);
314                        }
315                    }
316
317                    // Add new events
318                    if let Some(events) = element.events_handlers() {
319                        for event in events.keys() {
320                            self.listeners.entry(*event).or_default().push(node_id);
321                        }
322                    }
323                }
324
325                self.elements.insert(node_id, element);
326            }
327        });
328
329        // Run states cascades
330        let mut layer_cascades: Vec<NodeId> = Vec::new();
331        let mut effects_cascades: Vec<NodeId> = Vec::new();
332        let mut text_style_cascades: Vec<NodeId> = Vec::new();
333
334        assert_eq!(dirty.len(), FxHashSet::from_iter(&dirty).len());
335
336        hotpath::measure_block!("dirty run", {
337            for (node_id, flags) in dirty {
338                let element = self.elements.get(&node_id).unwrap();
339                let height_b = self.heights.get(&node_id).unwrap();
340
341                if flags.contains(DiffModifies::REORDER_LAYOUT) {
342                    self.layout
343                        .invalidate_with_reason(node_id, DirtyReason::Reorder);
344                }
345
346                if flags.contains(DiffModifies::INNER_LAYOUT) {
347                    self.layout
348                        .invalidate_with_reason(node_id, DirtyReason::InnerLayout);
349                }
350
351                if flags.contains(DiffModifies::LAYOUT) {
352                    self.layout.invalidate(node_id);
353                }
354
355                if !needs_render
356                    && (flags.intersects(
357                        DiffModifies::STYLE
358                            | DiffModifies::LAYER
359                            | DiffModifies::EFFECT
360                            | DiffModifies::TEXT_STYLE,
361                    ))
362                {
363                    needs_render = true;
364                }
365
366                if !needs_accessibility && (flags.intersects(DiffModifies::ACCESSIBILITY)) {
367                    needs_accessibility = true;
368                }
369
370                if flags.contains(DiffModifies::ACCESSIBILITY) {
371                    match self.accessibility_state.get_mut(&node_id) {
372                        Some(accessibility_state) => accessibility_state.update(
373                            node_id,
374                            element,
375                            &mut self.accessibility_diff,
376                            &mut self.accessibility_groups,
377                        ),
378                        None => {
379                            self.accessibility_state.insert(
380                                node_id,
381                                AccessibilityState::create(
382                                    node_id,
383                                    element,
384                                    &mut self.accessibility_diff,
385                                    &self.accessibility_generator,
386                                    &mut self.accessibility_groups,
387                                ),
388                            );
389                        }
390                    }
391                }
392
393                let handle_cascade = |cascades: &mut Vec<NodeId>| {
394                    // Skip scanning if we already know this node is the a root
395                    if cascades.iter_mut().any(|root| {
396                        let height_a = self.heights.get(root).unwrap();
397
398                        match height_a.cmp(height_b) {
399                            std::cmp::Ordering::Less => {
400                                self.balance_heights(&node_id, root) == Some(*root)
401                            }
402                            std::cmp::Ordering::Greater => {
403                                let balanced_root = self.balance_heights(root, &node_id);
404                                match balanced_root {
405                                    Some(r) if r == node_id => {
406                                        // If this node is ascendant than the
407                                        // current root we set it as the new root
408                                        *root = node_id;
409                                        true
410                                    }
411                                    _ => false,
412                                }
413                            }
414                            std::cmp::Ordering::Equal => false,
415                        }
416                    }) {
417                        return;
418                    }
419                    cascades.push(node_id);
420                };
421
422                if flags.intersects(DiffModifies::LAYER) {
423                    handle_cascade(&mut layer_cascades);
424                }
425                if flags.intersects(DiffModifies::EFFECT) {
426                    let element = self.elements.get(&node_id).unwrap();
427                    // Has data or the parent has state
428                    let run_cascade = element.effect().is_some()
429                        || self
430                            .parents
431                            .get(&node_id)
432                            .map(|parent| self.effect_state.contains_key(parent))
433                            .unwrap_or_default();
434                    if run_cascade {
435                        handle_cascade(&mut effects_cascades);
436                    }
437                }
438                if flags.intersects(DiffModifies::TEXT_STYLE) {
439                    handle_cascade(&mut text_style_cascades);
440                }
441            }
442        });
443
444        hotpath::measure_block!("layer cascade", {
445            // Run the layer state
446            for layer_root in layer_cascades {
447                let mut buffer = VecDeque::new();
448                buffer.push_front(&layer_root);
449
450                while let Some(node_id) = buffer.pop_front() {
451                    let element = self.elements.get(node_id).unwrap();
452                    if let Some(parent_node_id) = self.parents.get(node_id) {
453                        let entries = self
454                            .layer_state
455                            .get_disjoint_entries([node_id, parent_node_id], |_id| {
456                                LayerState::default()
457                            });
458                        if let Some([layer_state, parent_layer_state]) = entries {
459                            layer_state.update(
460                                parent_layer_state,
461                                *node_id,
462                                element,
463                                &mut self.layers,
464                            );
465                        }
466                    } else {
467                        assert_eq!(*node_id, NodeId::ROOT);
468                        self.layer_state.insert(
469                            NodeId::ROOT,
470                            LayerState::create_for_root(*node_id, &mut self.layers),
471                        );
472                    }
473                    if let Some(children) = self.children.get(node_id) {
474                        buffer.extend(children);
475                    }
476                }
477            }
478        });
479
480        hotpath::measure_block!("effect cascade", {
481            // Run the effect state
482            for effect_root in effects_cascades {
483                let mut buffer = VecDeque::new();
484                buffer.push_front(&effect_root);
485
486                while let Some(node_id) = buffer.pop_front() {
487                    let element = self.elements.get(node_id).unwrap();
488                    if let Some(parent_node_id) = self.parents.get(node_id) {
489                        let entries = self.effect_state.get_disjoint_two_entries(
490                            parent_node_id,
491                            node_id,
492                            |_id| EffectState::default(),
493                            |left, _id| left.clone(),
494                        );
495                        if let [Some(parent_effect_state), Some(effect_state)] = entries {
496                            let effect_data = element.effect();
497                            let layer = element.layer();
498                            effect_state.update(
499                                *parent_node_id,
500                                parent_effect_state,
501                                *node_id,
502                                effect_data,
503                                layer,
504                            );
505                        }
506                    } else {
507                        assert_eq!(*node_id, NodeId::ROOT);
508                    }
509                    if let Some(children) = self.children.get(node_id) {
510                        buffer.extend(children);
511                    }
512                }
513            }
514        });
515
516        hotpath::measure_block!("text style cascade", {
517            // Run the text style state
518            for text_style_root in text_style_cascades {
519                let mut buffer = VecDeque::new();
520                buffer.push_front(&text_style_root);
521
522                while let Some(node_id) = buffer.pop_front() {
523                    let element = self.elements.get(node_id).unwrap();
524                    if let Some(parent_node_id) = self.parents.get(node_id) {
525                        let entries = self
526                            .text_style_state
527                            .get_disjoint_entries([node_id, parent_node_id], |_id| {
528                                TextStyleState::default()
529                            });
530                        if let Some([text_style_state, parent_text_style_state]) = entries {
531                            text_style_state.update(
532                                *node_id,
533                                parent_text_style_state,
534                                element,
535                                &mut self.layout,
536                            );
537                        }
538                    } else {
539                        assert_eq!(*node_id, NodeId::ROOT);
540                        self.text_style_state
541                            .insert(NodeId::ROOT, TextStyleState::default());
542                    }
543                    if let Some(children) = self.children.get(node_id) {
544                        buffer.extend(children);
545                    }
546                }
547            }
548
549            #[cfg(all(debug_assertions, feature = "debug-integrity"))]
550            self.verify_tree_integrity();
551        });
552
553        MutationsApplyResult {
554            needs_render,
555            needs_accessibility,
556        }
557    }
558
559    /// Walk to the ancestor of `base` with the same height of `target`
560    fn balance_heights(&self, base: &NodeId, target: &NodeId) -> Option<NodeId> {
561        let target_height = self.heights.get(target)?;
562        let mut current = base;
563        loop {
564            if self.heights.get(current)? == target_height {
565                break;
566            }
567
568            let parent_current = self.parents.get(current);
569            if let Some(parent_current) = parent_current {
570                current = parent_current;
571            }
572        }
573        Some(*current)
574    }
575
576    pub fn measure_layout(
577        &mut self,
578        size: Size2D,
579        font_collection: &FontCollection,
580        font_manager: &FontMgr,
581        events_sender: &UnboundedSender<EventsChunk>,
582        scale_factor: f64,
583        fallback_fonts: &[Cow<'static, str>],
584    ) {
585        let mut tree_adapter = TreeAdapterFreya {
586            elements: &self.elements,
587            parents: &self.parents,
588            children: &self.children,
589            heights: &self.heights,
590            scale_factor,
591        };
592
593        let mut events = Vec::new();
594
595        let layout_adapter = LayoutMeasurerAdapter {
596            elements: &self.elements,
597            text_style_state: &self.text_style_state,
598            font_collection,
599            font_manager,
600            events: &mut events,
601            scale_factor,
602            fallback_fonts,
603            text_cache: &mut self.text_cache,
604        };
605
606        self.layout.find_best_root(&mut tree_adapter);
607        self.layout.measure(
608            NodeId::ROOT,
609            Area::from_size(size),
610            &mut Some(layout_adapter),
611            &mut tree_adapter,
612        );
613        events_sender
614            .unbounded_send(EventsChunk::Batch(events))
615            .unwrap();
616    }
617
618    pub fn print_ascii(&self, node_id: NodeId, prefix: String, last: bool) {
619        let height = self.heights.get(&node_id).unwrap();
620        let layer = self.layer_state.get(&node_id).unwrap();
621
622        // Print current node
623        println!(
624            "{}{}{:?} [{}] ({})",
625            prefix,
626            if last { "└── " } else { "├── " },
627            node_id,
628            height,
629            layer.layer
630        );
631
632        // Get children
633        if let Some(children) = self.children.get(&node_id) {
634            let len = children.len();
635            for (i, child) in children.iter().enumerate() {
636                let is_last = i == len - 1;
637                // Extend prefix
638                let new_prefix = format!("{}{}", prefix, if last { "    " } else { "│   " });
639                self.print_ascii(*child, new_prefix, is_last);
640            }
641        }
642    }
643
644    #[cfg(all(debug_assertions, feature = "debug-integrity"))]
645    #[cfg_attr(feature = "hotpath", hotpath::measure)]
646    pub fn verify_tree_integrity(&self) {
647        let mut visited = FxHashSet::default();
648        let size = self.elements.len();
649        let mut buffer = vec![NodeId::ROOT];
650        while let Some(node_id) = buffer.pop() {
651            if visited.contains(&node_id) {
652                continue;
653            }
654            visited.insert(node_id);
655            if let Some(parent) = self.parents.get(&node_id) {
656                buffer.push(*parent);
657            }
658            if let Some(children) = self.children.get(&node_id) {
659                buffer.extend(children);
660            }
661        }
662        assert_eq!(size, visited.len())
663    }
664}
665
666bitflags! {
667    #[derive(Debug, Clone, Copy, PartialEq, Eq, PartialOrd, Ord, Hash)]
668    pub struct DiffModifies: u32 {
669        const LAYOUT = 1 << 0;
670        const STYLE = 1 << 1;
671        const ACCESSIBILITY = 1 << 2;
672        const EVENT_HANDLERS = 1 << 3;
673        const LAYER = 1 << 4;
674        const TEXT_STYLE = 1 << 5;
675        const EFFECT = 1 << 6;
676        const INNER_LAYOUT = 1 << 7;
677        const REORDER_LAYOUT = 1 << 8;
678    }
679}
680
681pub struct MutationsApplyResult {
682    pub needs_render: bool,
683    pub needs_accessibility: bool,
684}
685
686pub struct LayoutMeasurerAdapter<'a> {
687    pub font_collection: &'a FontCollection,
688    pub font_manager: &'a FontMgr,
689    elements: &'a FxHashMap<NodeId, Rc<dyn ElementExt>>,
690    text_style_state: &'a FxHashMap<NodeId, TextStyleState>,
691    events: &'a mut Vec<EmmitableEvent>,
692    scale_factor: f64,
693    fallback_fonts: &'a [Cow<'static, str>],
694    text_cache: &'a mut TextCache,
695}
696
697impl LayoutMeasurer<NodeId> for LayoutMeasurerAdapter<'_> {
698    fn measure(
699        &mut self,
700        node_id: NodeId,
701        torin_node: &torin::node::Node,
702        area_size: &Size2D,
703    ) -> Option<(Size2D, Rc<dyn Any>)> {
704        self.elements.get(&node_id)?.measure(LayoutContext {
705            node_id,
706            torin_node,
707            area_size,
708            font_collection: self.font_collection,
709            font_manager: self.font_manager,
710            text_style_state: self.text_style_state.get(&node_id).unwrap(),
711            scale_factor: self.scale_factor,
712            fallback_fonts: self.fallback_fonts,
713            text_cache: self.text_cache,
714        })
715    }
716
717    fn should_hook_measurement(&mut self, node_id: NodeId) -> bool {
718        if let Some(element) = self.elements.get(&node_id) {
719            element.should_hook_measurement()
720        } else {
721            false
722        }
723    }
724
725    fn should_measure_inner_children(&mut self, node_id: NodeId) -> bool {
726        if let Some(element) = self.elements.get(&node_id) {
727            element.should_measure_inner_children()
728        } else {
729            false
730        }
731    }
732
733    fn notify_layout_references(
734        &mut self,
735        node_id: NodeId,
736        area: Area,
737        visible_area: Area,
738        inner_sizes: Size2D,
739    ) {
740        let mut data = SizedEventData::new(area, visible_area, inner_sizes);
741        data.div(self.scale_factor as f32);
742        self.events.push(EmmitableEvent {
743            node_id,
744            name: EventName::Sized,
745            data: EventType::Sized(data),
746            bubbles: false,
747            source_event: EventName::Sized,
748        });
749    }
750}