use bevy_ecs::prelude::Component; use bevy_ecs::reflect::ReflectComponent; use bevy_math::Vec2; use bevy_reflect::{FromReflect, Reflect, ReflectFromReflect}; use std::fmt::Formatter; pub use taffy::style::AvailableSpace; impl std::fmt::Debug for ContentSize { fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result { f.debug_struct("ContentSize").finish() } } /// A `Measure` is used to compute the size of a ui node /// when the size of that node is based on its content. pub trait Measure: Send + Sync + 'static { /// Calculate the size of the node given the constraints. fn measure( &self, width: Option, height: Option, available_width: AvailableSpace, available_height: AvailableSpace, ) -> Vec2; } /// A `FixedMeasure` is a `Measure` that ignores all constraints and /// always returns the same size. #[derive(Default, Clone)] pub struct FixedMeasure { size: Vec2, } impl Measure for FixedMeasure { fn measure( &self, _: Option, _: Option, _: AvailableSpace, _: AvailableSpace, ) -> Vec2 { self.size } } /// A node with a `ContentSize` component is a node where its size /// is based on its content. #[derive(Component, Reflect, FromReflect)] #[reflect(Component, FromReflect)] pub struct ContentSize { /// The `Measure` used to compute the intrinsic size #[reflect(ignore)] pub(crate) measure_func: Option, } impl ContentSize { /// Set a `Measure` for this function pub fn set(&mut self, measure: impl Measure) { let measure_func = move |size: taffy::prelude::Size>, available: taffy::prelude::Size| { let size = measure.measure(size.width, size.height, available.width, available.height); taffy::prelude::Size { width: size.x, height: size.y, } }; self.measure_func = Some(taffy::node::MeasureFunc::Boxed(Box::new(measure_func))); } } #[allow(clippy::derivable_impls)] impl Default for ContentSize { fn default() -> Self { Self { measure_func: Some(taffy::node::MeasureFunc::Raw(|_, _| { taffy::prelude::Size::ZERO })), } } }