1use std::fmt;
2use std::rc::Rc;
3
4use cranpose_foundation::{ModifierNodeChain, NodeCapabilities, PointerEvent};
5use cranpose_ui_graphics::{ColorFilter, GraphicsLayer, RenderEffect};
6
7use super::{ModifierChainHandle, Point};
8use crate::draw::DrawCommand;
9use crate::modifier::Modifier;
10use crate::modifier_nodes::{
11 BackgroundNode, ClipToBoundsNode, CornerShapeNode, DrawCommandNode, GraphicsLayerNode,
12 PaddingNode,
13};
14use crate::text::{TextLayoutOptions, TextStyle};
15use crate::text_field_modifier_node::TextFieldModifierNode;
16use crate::text_modifier_node::{TextModifierNode, TextPreparedLayoutHandle};
17use cranpose_ui_graphics::EdgeInsets;
18
19#[derive(Default)]
21pub struct ModifierNodeSlices {
22 draw_commands: Vec<DrawCommand>,
23 pointer_inputs: Vec<Rc<dyn Fn(PointerEvent)>>,
24 click_handlers: Vec<Rc<dyn Fn(Point)>>,
25 clip_to_bounds: bool,
26 text_content: Option<Rc<crate::text::AnnotatedString>>,
27 text_style: Option<TextStyle>,
28 text_layout_options: Option<TextLayoutOptions>,
29 prepared_text_layout: Option<TextPreparedLayoutHandle>,
30 graphics_layer: Option<GraphicsLayer>,
31 graphics_layer_resolver: Option<Rc<dyn Fn() -> GraphicsLayer>>,
32 chain_guard: Option<Rc<ChainGuard>>,
33}
34
35struct ChainGuard {
36 _handle: ModifierChainHandle,
37}
38
39impl Clone for ModifierNodeSlices {
40 fn clone(&self) -> Self {
41 Self {
42 draw_commands: self.draw_commands.clone(),
43 pointer_inputs: self.pointer_inputs.clone(),
44 click_handlers: self.click_handlers.clone(),
45 clip_to_bounds: self.clip_to_bounds,
46 text_content: self.text_content.clone(),
47 text_style: self.text_style.clone(),
48 text_layout_options: self.text_layout_options,
49 prepared_text_layout: self.prepared_text_layout.clone(),
50 graphics_layer: self.graphics_layer.clone(),
51 graphics_layer_resolver: self.graphics_layer_resolver.clone(),
52 chain_guard: self.chain_guard.clone(),
53 }
54 }
55}
56
57fn merge_graphics_layers(base: GraphicsLayer, overlay: GraphicsLayer) -> GraphicsLayer {
72 GraphicsLayer {
73 alpha: (base.alpha * overlay.alpha).clamp(0.0, 1.0),
74 scale: base.scale * overlay.scale,
75 scale_x: base.scale_x * overlay.scale_x,
76 scale_y: base.scale_y * overlay.scale_y,
77 rotation_x: base.rotation_x + overlay.rotation_x,
78 rotation_y: base.rotation_y + overlay.rotation_y,
79 rotation_z: base.rotation_z + overlay.rotation_z,
80 camera_distance: overlay.camera_distance,
81 transform_origin: overlay.transform_origin,
82 translation_x: base.translation_x + overlay.translation_x,
83 translation_y: base.translation_y + overlay.translation_y,
84 shadow_elevation: overlay.shadow_elevation,
85 ambient_shadow_color: overlay.ambient_shadow_color,
86 spot_shadow_color: overlay.spot_shadow_color,
87 shape: overlay.shape,
88 clip: base.clip || overlay.clip,
89 compositing_strategy: overlay.compositing_strategy,
90 blend_mode: overlay.blend_mode,
91 color_filter: compose_color_filters(base.color_filter, overlay.color_filter),
92 render_effect: compose_render_effects(base.render_effect, overlay.render_effect),
95 backdrop_effect: overlay.backdrop_effect.or(base.backdrop_effect),
98 }
99}
100
101fn compose_render_effects(
102 outer: Option<RenderEffect>,
103 inner: Option<RenderEffect>,
104) -> Option<RenderEffect> {
105 match (outer, inner) {
106 (None, None) => None,
107 (Some(effect), None) | (None, Some(effect)) => Some(effect),
108 (Some(outer_effect), Some(inner_effect)) => Some(inner_effect.then(outer_effect)),
109 }
110}
111
112fn compose_color_filters(
113 base: Option<ColorFilter>,
114 overlay: Option<ColorFilter>,
115) -> Option<ColorFilter> {
116 match (base, overlay) {
117 (None, None) => None,
118 (Some(filter), None) | (None, Some(filter)) => Some(filter),
119 (Some(filter), Some(next)) => Some(filter.compose(next)),
120 }
121}
122
123impl ModifierNodeSlices {
124 pub fn draw_commands(&self) -> &[DrawCommand] {
125 &self.draw_commands
126 }
127
128 pub fn pointer_inputs(&self) -> &[Rc<dyn Fn(PointerEvent)>] {
129 &self.pointer_inputs
130 }
131
132 pub fn click_handlers(&self) -> &[Rc<dyn Fn(Point)>] {
133 &self.click_handlers
134 }
135
136 pub fn clip_to_bounds(&self) -> bool {
137 self.clip_to_bounds
138 }
139
140 pub fn text_content(&self) -> Option<&str> {
141 self.text_content.as_ref().map(|a| a.text.as_str())
142 }
143
144 pub fn annotated_text(&self) -> Option<&crate::text::AnnotatedString> {
145 self.text_content.as_deref()
146 }
147
148 pub fn annotated_string(&self) -> Option<crate::text::AnnotatedString> {
149 self.annotated_text().cloned()
150 }
151
152 pub fn text_style(&self) -> Option<&TextStyle> {
153 self.text_style.as_ref()
154 }
155
156 pub fn text_layout_options(&self) -> Option<TextLayoutOptions> {
157 self.text_layout_options
158 }
159
160 pub fn prepare_text_layout(
161 &self,
162 max_width: Option<f32>,
163 ) -> Option<crate::text::PreparedTextLayout> {
164 if let Some(handle) = &self.prepared_text_layout {
165 return Some(handle.prepare(max_width));
166 }
167
168 let text = self.annotated_text()?;
169 let style = self.text_style.clone().unwrap_or_default();
170 Some(crate::text::prepare_text_layout(
171 text,
172 &style,
173 self.text_layout_options.unwrap_or_default(),
174 max_width,
175 ))
176 }
177
178 pub fn graphics_layer(&self) -> Option<GraphicsLayer> {
179 if let Some(resolve) = &self.graphics_layer_resolver {
180 Some(resolve())
181 } else {
182 self.graphics_layer.clone()
183 }
184 }
185
186 fn push_graphics_layer(
187 &mut self,
188 layer: GraphicsLayer,
189 resolver: Option<Rc<dyn Fn() -> GraphicsLayer>>,
190 ) {
191 let existing_snapshot = self.graphics_layer();
192 let next_snapshot = existing_snapshot
193 .as_ref()
194 .map(|current| merge_graphics_layers(current.clone(), layer.clone()))
195 .unwrap_or_else(|| layer.clone());
196 let existing_resolver = self.graphics_layer_resolver.clone();
197
198 self.graphics_layer = Some(next_snapshot);
199 self.graphics_layer_resolver = match (existing_resolver, resolver) {
200 (None, None) => None,
201 (Some(current_resolver), None) => {
202 let layer = layer.clone();
203 Some(Rc::new(move || {
204 merge_graphics_layers(current_resolver(), layer.clone())
205 }))
206 }
207 (None, Some(next_resolver)) => {
208 let base = existing_snapshot.unwrap_or_default();
209 Some(Rc::new(move || {
210 merge_graphics_layers(base.clone(), next_resolver())
211 }))
212 }
213 (Some(current_resolver), Some(next_resolver)) => Some(Rc::new(move || {
214 merge_graphics_layers(current_resolver(), next_resolver())
215 })),
216 };
217 }
218
219 pub fn with_chain_guard(mut self, handle: ModifierChainHandle) -> Self {
220 self.chain_guard = Some(Rc::new(ChainGuard { _handle: handle }));
221 self
222 }
223
224 pub fn clear(&mut self) {
226 self.draw_commands.clear();
227 self.pointer_inputs.clear();
228 self.click_handlers.clear();
229 self.clip_to_bounds = false;
230 self.text_content = None;
231 self.text_style = None;
232 self.text_layout_options = None;
233 self.prepared_text_layout = None;
234 self.graphics_layer = None;
235 self.graphics_layer_resolver = None;
236 self.chain_guard = None;
237 }
238}
239
240impl fmt::Debug for ModifierNodeSlices {
241 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
242 f.debug_struct("ModifierNodeSlices")
243 .field("draw_commands", &self.draw_commands.len())
244 .field("pointer_inputs", &self.pointer_inputs.len())
245 .field("click_handlers", &self.click_handlers.len())
246 .field("clip_to_bounds", &self.clip_to_bounds)
247 .field("text_content", &self.text_content)
248 .field("text_style", &self.text_style)
249 .field("text_layout_options", &self.text_layout_options)
250 .field("prepared_text_layout", &self.prepared_text_layout.is_some())
251 .field("graphics_layer", &self.graphics_layer)
252 .field(
253 "graphics_layer_resolver",
254 &self.graphics_layer_resolver.is_some(),
255 )
256 .finish()
257 }
258}
259
260pub fn collect_modifier_slices(chain: &ModifierNodeChain) -> ModifierNodeSlices {
262 let mut slices = ModifierNodeSlices::default();
263 collect_modifier_slices_into(chain, &mut slices);
264 slices
265}
266
267pub fn collect_modifier_slices_into(chain: &ModifierNodeChain, slices: &mut ModifierNodeSlices) {
272 slices.clear();
273
274 let caps = chain.capabilities();
275 let has_pointer = caps.intersects(NodeCapabilities::POINTER_INPUT);
276 let has_draw = caps.intersects(NodeCapabilities::DRAW);
277 let has_layout = caps.intersects(NodeCapabilities::LAYOUT);
278
279 if !has_pointer && !has_draw && !has_layout {
280 return;
281 }
282
283 let mut background_color = None;
284 let mut background_insert_index = None::<usize>;
285 let mut corner_shape = None;
286 let mut padding = EdgeInsets::default();
287
288 for node_ref in chain.head_to_tail() {
289 let node_caps = node_ref.kind_set();
290
291 node_ref.with_node(|node| {
292 let any = node.as_any();
293
294 if has_pointer && node_caps.intersects(NodeCapabilities::POINTER_INPUT) {
296 if let Some(handler) = node
297 .as_pointer_input_node()
298 .and_then(|n| n.pointer_input_handler())
299 {
300 slices.pointer_inputs.push(handler);
301 }
302 }
303
304 if has_draw && node_caps.intersects(NodeCapabilities::DRAW) {
306 if let Some(bg_node) = any.downcast_ref::<BackgroundNode>() {
307 background_color = Some(bg_node.color());
308 background_insert_index = Some(slices.draw_commands.len());
309 if bg_node.shape().is_some() {
310 corner_shape = bg_node.shape();
311 }
312 }
313
314 if let Some(shape_node) = any.downcast_ref::<CornerShapeNode>() {
315 corner_shape = Some(shape_node.shape());
316 }
317
318 if let Some(commands) = any.downcast_ref::<DrawCommandNode>() {
319 slices
320 .draw_commands
321 .extend(commands.commands().iter().cloned());
322 }
323
324 if let Some(draw_node) = node.as_draw_node() {
325 if let Some(closure) = draw_node.create_draw_closure() {
326 slices.draw_commands.push(DrawCommand::Overlay(closure));
327 } else {
328 use cranpose_ui_graphics::{DrawScope as _, DrawScopeDefault};
329 let mut scope = DrawScopeDefault::new(crate::modifier::Size {
330 width: 0.0,
331 height: 0.0,
332 });
333 draw_node.draw(&mut scope);
334 let primitives = scope.into_primitives();
335 if !primitives.is_empty() {
336 let draw_cmd =
337 Rc::new(move |_size: crate::modifier::Size| primitives.clone());
338 slices.draw_commands.push(DrawCommand::Overlay(draw_cmd));
339 }
340 }
341 }
342
343 if let Some(layer_node) = any.downcast_ref::<GraphicsLayerNode>() {
344 slices.push_graphics_layer(layer_node.layer(), layer_node.layer_resolver());
345 }
346
347 if any.is::<ClipToBoundsNode>() {
348 slices.clip_to_bounds = true;
349 }
350 }
351
352 if has_layout && node_caps.intersects(NodeCapabilities::LAYOUT) {
354 if let Some(padding_node) = any.downcast_ref::<PaddingNode>() {
355 let p = padding_node.padding();
356 padding.left += p.left;
357 padding.top += p.top;
358 padding.right += p.right;
359 padding.bottom += p.bottom;
360 }
361
362 if let Some(text_node) = any.downcast_ref::<TextModifierNode>() {
363 slices.text_content = Some(text_node.annotated_text());
364 slices.text_style = Some(text_node.style().clone());
365 slices.text_layout_options = Some(text_node.options());
366 slices.prepared_text_layout = Some(text_node.prepared_layout_handle());
367 }
368
369 if let Some(text_field_node) = any.downcast_ref::<TextFieldModifierNode>() {
370 let text = text_field_node.text();
371 slices.text_content = Some(Rc::new(crate::text::AnnotatedString::from(text)));
372 slices.text_style = Some(text_field_node.style().clone());
373 slices.text_layout_options = Some(TextLayoutOptions::default());
374 slices.prepared_text_layout = None;
375
376 text_field_node.set_content_offset(padding.left);
377 text_field_node.set_content_y_offset(padding.top);
378 }
379 }
380 });
381 }
382
383 if let Some(color) = background_color {
385 let draw_cmd = Rc::new(move |size: crate::modifier::Size| {
386 use crate::modifier::{Brush, Rect};
387 use cranpose_ui_graphics::DrawPrimitive;
388
389 let brush = Brush::solid(color);
390 let rect = Rect {
391 x: 0.0,
392 y: 0.0,
393 width: size.width,
394 height: size.height,
395 };
396
397 if let Some(shape) = corner_shape {
398 let radii = shape.resolve(size.width, size.height);
399 vec![DrawPrimitive::RoundRect { rect, brush, radii }]
400 } else {
401 vec![DrawPrimitive::Rect { rect, brush }]
402 }
403 });
404
405 let insert_index = background_insert_index
406 .unwrap_or(0)
407 .min(slices.draw_commands.len());
408 slices
409 .draw_commands
410 .insert(insert_index, DrawCommand::Behind(draw_cmd));
411 }
412}
413
414pub fn collect_slices_from_modifier(modifier: &Modifier) -> ModifierNodeSlices {
416 let mut handle = ModifierChainHandle::new();
417 let _ = handle.update(modifier);
418 collect_modifier_slices(handle.chain()).with_chain_guard(handle)
419}