Skip to main content

llimphi_compositor/
layout_builder.rs

1//! **LayoutBuilder** — el 4º seam de PARIDAD-FLUTTER: construir un subárbol
2//! sensible al **tamaño del slot** del nodo (no de la ventana — para eso
3//! alcanza `on_resize` + el Model). Flutter `LayoutBuilder`.
4//!
5//! El modelo de Llimphi corre `view → mount → compute → paint`: el `View` se
6//! arma ANTES de conocer el layout, así que "construir distinto según el espacio
7//! disponible" exige diferir. La solución, sin tocar `mount`/`paint`, es una
8//! **resolución en dos pasadas** orquestada por el runtime:
9//!
10//! 1. Montar el árbol tal cual ([`crate::View::layout_builder`] queda como
11//!    **hoja** — no tiene `children` estáticos) y computar el layout. Ahora cada
12//!    builder tiene su rect resuelto por su `Style`/contexto flex.
13//! 2. [`collect_builder_constraints`] lee esos rects (en pre-orden), se pide un
14//!    `view()` fresco y [`expand_layout_builders`] invoca cada closure con sus
15//!    [`crate::Constraints`] para producir el subárbol real. Ese árbol expandido
16//!    se monta y pinta normalmente.
17//!
18//! [`has_layout_builder`] hace que todo esto sea **coste cero** cuando ningún
19//! nodo usa el builder (el caso de la abrumadora mayoría de frames): es un
20//! simple walk que corta el camino de dos pasadas.
21//!
22//! **Correspondencia de orden.** `collect_builder_constraints` recorre
23//! `Mounted::nodes` (pre-orden, padre antes que hijos — el orden en que `mount`
24//! los pushea) filtrando `is_layout_builder`; `expand_layout_builders` recorre
25//! el `View` fresco en el MISMO pre-orden asignando un índice por builder. Como
26//! ambos árboles salen del mismo `view(model)` determinista, el i-ésimo builder
27//! de uno corresponde al i-ésimo del otro — por eso alcanza con un `Vec`
28//! ordenado, sin keys.
29//!
30//! **Límite v1**: sin anidamiento. Un builder cuyo subárbol producido contiene
31//! otro `layout_builder` no resuelve el interno (no existía en la pasada 1):
32//! queda como hoja. El anidamiento requeriría iterar la resolución; se difiere.
33
34use crate::{Constraints, ComputedLayout, Mounted, View};
35
36/// `true` si `view` o algún descendiente declara un [`crate::View::layout_builder`].
37/// El runtime lo usa para decidir si vale la pena la resolución en dos pasadas;
38/// cuando es `false` (lo normal) el camino diferido se evita por completo.
39pub fn has_layout_builder<Msg>(view: &View<Msg>) -> bool {
40    view.layout_builder.is_some() || view.children.iter().any(has_layout_builder)
41}
42
43/// Lee las [`Constraints`] (tamaño del slot) de cada nodo `is_layout_builder`
44/// del árbol montado, en pre-orden. El runtime las pasa a
45/// [`expand_layout_builders`]. Un nodo sin rect computado (fuera del layout)
46/// cae a `0×0`.
47pub fn collect_builder_constraints<Msg>(
48    mounted: &Mounted<Msg>,
49    computed: &ComputedLayout,
50) -> Vec<Constraints> {
51    mounted
52        .nodes
53        .iter()
54        .filter(|n| n.is_layout_builder)
55        .map(|n| {
56            computed
57                .get(n.id)
58                .map(|r| Constraints { max_width: r.w, max_height: r.h })
59                .unwrap_or(Constraints { max_width: 0.0, max_height: 0.0 })
60        })
61        .collect()
62}
63
64/// Expande los `layout_builder` de `view` (pre-orden) usando `cons` — una
65/// [`Constraints`] por builder, en el orden que produjo
66/// [`collect_builder_constraints`]. Cada builder se reemplaza por un nodo
67/// contenedor (su mismo `Style`) cuyo único hijo es lo que devolvió la closure
68/// invocada con sus constraints. Builders sin constraint correspondiente (más
69/// builders que `cons`, p. ej. uno anidado recién producido) caen a `0×0` y se
70/// resuelven igual, pero su tamaño será nulo (límite v1: sin anidamiento).
71/// Consume `view`.
72pub fn expand_layout_builders<Msg>(view: View<Msg>, cons: &[Constraints]) -> View<Msg> {
73    let mut idx = 0;
74    expand_rec(view, cons, &mut idx)
75}
76
77fn expand_rec<Msg>(mut view: View<Msg>, cons: &[Constraints], idx: &mut usize) -> View<Msg> {
78    if let Some(builder) = view.layout_builder.take() {
79        let c = cons
80            .get(*idx)
81            .copied()
82            .unwrap_or(Constraints { max_width: 0.0, max_height: 0.0 });
83        *idx += 1;
84        // El builder posee los hijos: descartamos cualquier `children` estático
85        // y ponemos lo que produjo la closure. NO recursamos en el resultado
86        // (v1 sin anidamiento — un builder interno queda como hoja al montarse).
87        let child = builder(c);
88        view.children = vec![child];
89        view
90    } else {
91        let children = std::mem::take(&mut view.children);
92        view.children = children
93            .into_iter()
94            .map(|c| expand_rec(c, cons, idx))
95            .collect();
96        view
97    }
98}
99
100#[cfg(test)]
101mod tests {
102    use super::*;
103    use crate::{mount, Constraints};
104    use llimphi_layout::taffy::prelude::*;
105    use llimphi_layout::{LayoutTree, Style};
106
107    /// Árbol sin builders → `has_layout_builder` falso y expand es no-op.
108    #[test]
109    fn sin_builder_es_noop() {
110        let v = View::<()>::new(Style::default())
111            .children(vec![View::<()>::new(Style::default())]);
112        assert!(!has_layout_builder(&v));
113        let v = expand_layout_builders(v, &[]);
114        assert_eq!(v.children.len(), 1);
115    }
116
117    #[test]
118    fn detecta_builder_anidado_en_hijos() {
119        let v = View::<()>::new(Style::default()).children(vec![
120            View::<()>::new(Style::default()),
121            View::<()>::new(Style::default()).layout_builder(|_c| View::<()>::new(Style::default())),
122        ]);
123        assert!(has_layout_builder(&v));
124    }
125
126    /// El builder recibe las constraints y produce su subárbol; el nodo deja de
127    /// ser builder y queda como contenedor con el hijo producido.
128    #[test]
129    fn expand_invoca_closure_con_constraints() {
130        // Dos columnas a percent(0.5) del root 400px → cada slot = 200px. La de
131        // la izquierda es un builder que mete 1 hijo si es angosta (<300) o 2 si
132        // es ancha. A 200px mete 1.
133        let build_col = |c: Constraints| {
134            let n = if c.max_width < 300.0 { 1 } else { 2 };
135            View::<()>::new(Style::default())
136                .children((0..n).map(|_| View::<()>::new(Style::default())).collect())
137        };
138        let root = View::<()>::new(Style {
139            size: Size { width: length(400.0), height: length(100.0) },
140            flex_direction: FlexDirection::Row,
141            ..Default::default()
142        })
143        .children(vec![
144            View::<()>::new(Style {
145                size: Size { width: percent(0.5), height: percent(1.0) },
146                ..Default::default()
147            })
148            .layout_builder(build_col),
149            View::<()>::new(Style {
150                size: Size { width: percent(0.5), height: percent(1.0) },
151                ..Default::default()
152            }),
153        ]);
154
155        // Pasada 1: montar (builder como hoja) y computar.
156        let mut l1 = LayoutTree::new();
157        let m1 = mount(&mut l1, root);
158        let c1 = l1.compute(m1.root, (400.0, 100.0)).expect("layout");
159        let cons = collect_builder_constraints(&m1, &c1);
160        assert_eq!(cons.len(), 1, "un solo builder");
161        assert!((cons[0].max_width - 200.0).abs() < 1.0, "slot 200px: {:?}", cons[0]);
162
163        // Pasada 2: árbol fresco (mismo Style) + expand.
164        let root2 = View::<()>::new(Style {
165            size: Size { width: length(400.0), height: length(100.0) },
166            flex_direction: FlexDirection::Row,
167            ..Default::default()
168        })
169        .children(vec![
170            View::<()>::new(Style {
171                size: Size { width: percent(0.5), height: percent(1.0) },
172                ..Default::default()
173            })
174            .layout_builder(build_col),
175            View::<()>::new(Style {
176                size: Size { width: percent(0.5), height: percent(1.0) },
177                ..Default::default()
178            }),
179        ]);
180        let expanded = expand_layout_builders(root2, &cons);
181        // El nodo builder (hijo 0 del root) ya no es builder y tiene 1 hijo
182        // producido (slot 200 < 300 → angosto → 1 columna).
183        let col_izq = &expanded.children[0];
184        assert!(col_izq.layout_builder.is_none(), "ya expandido");
185        assert_eq!(col_izq.children.len(), 1, "200px angosto → 1 hijo");
186    }
187
188    /// Con un slot ancho el mismo builder produce 2 hijos — verifica que la
189    /// rama de decisión depende de las constraints reales.
190    #[test]
191    fn slot_ancho_produce_mas_hijos() {
192        let build_col = |c: Constraints| {
193            let n = if c.max_width < 300.0 { 1 } else { 2 };
194            View::<()>::new(Style::default())
195                .children((0..n).map(|_| View::<()>::new(Style::default())).collect())
196        };
197        // Constraint inyectada directo: 500px → ancho. El builder devuelve UN
198        // contenedor (hijo único del nodo) con 2 columnas adentro.
199        let v = View::<()>::new(Style::default()).layout_builder(build_col);
200        let expanded = expand_layout_builders(v, &[Constraints { max_width: 500.0, max_height: 100.0 }]);
201        assert_eq!(expanded.children.len(), 1, "el builder produce 1 contenedor");
202        assert_eq!(expanded.children[0].children.len(), 2, "ancho → 2 columnas");
203    }
204
205    /// Pre-orden: dos builders hermanos reciben sus constraints en orden.
206    #[test]
207    fn dos_builders_reciben_constraints_en_preorden() {
208        let mk = |w: f32| {
209            move |_c: Constraints| {
210                View::<()>::new(Style {
211                    size: Size { width: length(w), height: length(10.0) },
212                    ..Default::default()
213                })
214            }
215        };
216        let root = View::<()>::new(Style::default()).children(vec![
217            View::<()>::new(Style::default()).layout_builder(mk(1.0)),
218            View::<()>::new(Style::default()).layout_builder(mk(2.0)),
219        ]);
220        let cons = vec![
221            Constraints { max_width: 111.0, max_height: 0.0 },
222            Constraints { max_width: 222.0, max_height: 0.0 },
223        ];
224        let expanded = expand_layout_builders(root, &cons);
225        // Ambos expandidos, en orden (verificamos vía el ancho del hijo producido
226        // que NO depende de la constraint acá — sólo confirmamos que se invocaron
227        // los dos y que ninguno quedó como builder).
228        assert!(expanded.children[0].layout_builder.is_none());
229        assert!(expanded.children[1].layout_builder.is_none());
230        assert_eq!(expanded.children[0].children.len(), 1);
231        assert_eq!(expanded.children[1].children.len(), 1);
232    }
233}