From fd48f170e6f93eb83e2b2225455e5551b1239367 Mon Sep 17 00:00:00 2001 From: Chris <1731074+ccbrown@users.noreply.github.com> Date: Fri, 27 Dec 2024 22:42:34 -0600 Subject: [PATCH] fix: eliminate undesired impact of transparent components on layout (#53) --- packages/iocraft/src/component.rs | 2 +- packages/iocraft/src/components/box.rs | 33 ++++++++++++++++++++++++++ packages/iocraft/src/render.rs | 30 ++++++++++++++++------- 3 files changed, 56 insertions(+), 9 deletions(-) diff --git a/packages/iocraft/src/component.rs b/packages/iocraft/src/component.rs index 89e9a0c..1c0fece 100644 --- a/packages/iocraft/src/component.rs +++ b/packages/iocraft/src/component.rs @@ -162,7 +162,7 @@ impl InstantiatedComponent { pub fn update( &mut self, context: &mut UpdateContext<'_>, - unattached_child_node_ids: Option<&mut Vec>, + unattached_child_node_ids: &mut Vec, component_context_stack: &mut ContextStack<'_>, props: AnyProps, ) { diff --git a/packages/iocraft/src/components/box.rs b/packages/iocraft/src/components/box.rs index 26a051f..0574771 100644 --- a/packages/iocraft/src/components/box.rs +++ b/packages/iocraft/src/components/box.rs @@ -346,6 +346,18 @@ mod tests { use crate::prelude::*; use indoc::indoc; + #[derive(Default, Props)] + pub struct MyTextProps { + pub content: String, + } + + #[component] + pub fn MyText<'a>(props: &MyTextProps) -> impl Into> { + element! { + Text(content: &props.content) + } + } + #[test] fn test_box() { assert_eq!(element!(Box).to_string(), ""); @@ -591,5 +603,26 @@ mod tests { └──────────────────┘ "}, ); + + // regression test for https://github.com/ccbrown/iocraft/issues/52 + assert_eq!( + element! { + Box(width: 20, border_style: BorderStyle::Single, row_gap: 1, flex_direction: FlexDirection::Column) { + Text(content: "foo") + MyText(content: "bar") + MyText(content: "baz") + } + } + .to_string(), + indoc! {" + ┌──────────────────┐ + │foo │ + │ │ + │bar │ + │ │ + │baz │ + └──────────────────┘ + "}, + ); } } diff --git a/packages/iocraft/src/render.rs b/packages/iocraft/src/render.rs index 8817d13..2fe1bcb 100644 --- a/packages/iocraft/src/render.rs +++ b/packages/iocraft/src/render.rs @@ -19,7 +19,7 @@ use futures::{ stream::{Stream, StreamExt}, }; use std::io; -use taffy::{AvailableSpace, Layout, NodeId, Point, Size, Style, TaffyTree}; +use taffy::{AvailableSpace, Display, Layout, NodeId, Point, Size, Style, TaffyTree}; pub(crate) struct UpdateContext<'a> { terminal: Option<&'a mut Terminal>, @@ -33,7 +33,7 @@ pub struct ComponentUpdater<'a, 'b: 'a, 'c: 'a> { node_id: NodeId, transparent_layout: bool, children: &'a mut Components, - unattached_child_node_ids: Option<&'a mut Vec>, + unattached_child_node_ids: &'a mut Vec, context: &'a mut UpdateContext<'b>, component_context_stack: &'a mut ContextStack<'c>, } @@ -42,7 +42,7 @@ impl<'a, 'b, 'c> ComponentUpdater<'a, 'b, 'c> { pub(crate) fn new( node_id: NodeId, children: &'a mut Components, - unattached_child_node_ids: Option<&'a mut Vec>, + unattached_child_node_ids: &'a mut Vec, context: &'a mut UpdateContext<'b>, component_context_stack: &'a mut ContextStack<'c>, ) -> Self { @@ -123,6 +123,18 @@ impl<'a, 'b, 'c> ComponentUpdater<'a, 'b, 'c> { /// children will effectively be direct descendants of the parent of the current component for /// layout purposes. pub fn set_transparent_layout(&mut self, transparent_layout: bool) { + if transparent_layout && !self.transparent_layout { + self.context + .layout_engine + .set_style( + self.node_id, + Style { + display: Display::None, + ..Default::default() + }, + ) + .expect("we should be able to set the style"); + } self.transparent_layout = transparent_layout; } @@ -142,9 +154,7 @@ impl<'a, 'b, 'c> ComponentUpdater<'a, 'b, 'c> { let mut direct_child_node_ids = Vec::new(); let child_node_ids = if self.transparent_layout { - self.unattached_child_node_ids - .as_deref_mut() - .unwrap_or(&mut direct_child_node_ids) + &mut self.unattached_child_node_ids } else { &mut direct_child_node_ids }; @@ -175,7 +185,7 @@ impl<'a, 'b, 'c> ComponentUpdater<'a, 'b, 'c> { }; component.update( self.context, - Some(child_node_ids), + child_node_ids, component_context_stack, child.props_mut(), ); @@ -315,6 +325,7 @@ impl<'a> Tree<'a> { max_width: Option, terminal: Option<&mut Terminal>, ) -> RenderOutput { + let mut wrapper_child_node_ids = vec![self.root_component.node_id()]; let did_clear_terminal_output = { let mut context = UpdateContext { terminal, @@ -324,12 +335,15 @@ impl<'a> Tree<'a> { let mut component_context_stack = ContextStack::root(&mut self.system_context); self.root_component.update( &mut context, - None, + &mut wrapper_child_node_ids, &mut component_context_stack, self.root_component_props.borrow(), ); context.did_clear_terminal_output }; + self.layout_engine + .set_children(self.wrapper_node_id, &wrapper_child_node_ids) + .expect("we should be able to set the children"); self.layout_engine .compute_layout_with_measure(