mirror of
https://github.com/bevyengine/bevy
synced 2024-11-26 14:40:19 +00:00
MeasureFunc
improvements (#8402)
# Objective fixes #8516 * Give `CalculatedSize` a more specific and intuitive name. * `MeasureFunc`s should only be updated when their `CalculatedSize` is modified by the systems managing their content. For example, suppose that you have a UI displaying an image using an `ImageNode`. When the window is resized, the node's `MeasureFunc` will be updated even though the dimensions of the texture contained by the node are unchanged. * Fix the `CalculatedSize` API so that it no longer requires the extra boxing and the `dyn_clone` method. ## Solution * Rename `CalculatedSize` to `ContentSize` * Only update `MeasureFunc`s on `CalculatedSize` changes. * Remove the `dyn_clone` method from `Measure` and move the `Measure` from the `ContentSize` component rather than cloning it. * Change the measure_func field of `ContentSize` to type `Option<taffy::node::MeasureFunc>`. Add a `set` method that wraps the given measure appropriately. --- ## Changelog * Renamed `CalculatedSize` to `ContentSize`. * Replaced `upsert_leaf` with a function `update_measure` that only updates the node's `MeasureFunc`. * `MeasureFunc`s are only updated when the `ContentSize` changes and not when the layout changes. * Scale factor is no longer applied to the size values passed to the `MeasureFunc`. * Remove the `ContentSize` scaling in `text_system`. * The `dyn_clone` method has been removed from the `Measure` trait. * `Measure`s are moved from the `ContentSize` component instead of cloning them. * Added `set` method to `ContentSize` that replaces the `new` function. ## Migration Guide * `CalculatedSize` has been renamed to `ContentSize`. * The `upsert_leaf` function has been removed from `UiSurface` and replaced with `update_measure` which updates the `MeasureFunc` without node insertion. * The `dyn_clone` method has been removed from the `Measure` trait. * The new function of `CalculatedSize` has been replaced with the method `set`.
This commit is contained in:
parent
deba3806d6
commit
ba532e4a37
6 changed files with 79 additions and 115 deletions
|
@ -1,13 +1,14 @@
|
||||||
mod convert;
|
mod convert;
|
||||||
|
|
||||||
use crate::{CalculatedSize, Node, Style, UiScale};
|
use crate::{ContentSize, Node, Style, UiScale};
|
||||||
use bevy_ecs::{
|
use bevy_ecs::{
|
||||||
change_detection::DetectChanges,
|
change_detection::DetectChanges,
|
||||||
entity::Entity,
|
entity::Entity,
|
||||||
event::EventReader,
|
event::EventReader,
|
||||||
query::{Changed, Or, With, Without},
|
query::{Changed, With, Without},
|
||||||
removal_detection::RemovedComponents,
|
removal_detection::RemovedComponents,
|
||||||
system::{Query, Res, ResMut, Resource},
|
system::{Query, Res, ResMut, Resource},
|
||||||
|
world::Ref,
|
||||||
};
|
};
|
||||||
use bevy_hierarchy::{Children, Parent};
|
use bevy_hierarchy::{Children, Parent};
|
||||||
use bevy_log::warn;
|
use bevy_log::warn;
|
||||||
|
@ -16,11 +17,7 @@ use bevy_transform::components::Transform;
|
||||||
use bevy_utils::HashMap;
|
use bevy_utils::HashMap;
|
||||||
use bevy_window::{PrimaryWindow, Window, WindowResolution, WindowScaleFactorChanged};
|
use bevy_window::{PrimaryWindow, Window, WindowResolution, WindowScaleFactorChanged};
|
||||||
use std::fmt;
|
use std::fmt;
|
||||||
use taffy::{
|
use taffy::{prelude::Size, style_helpers::TaffyMaxContent, Taffy};
|
||||||
prelude::{AvailableSpace, Size},
|
|
||||||
style_helpers::TaffyMaxContent,
|
|
||||||
Taffy,
|
|
||||||
};
|
|
||||||
|
|
||||||
pub struct LayoutContext {
|
pub struct LayoutContext {
|
||||||
pub scale_factor: f64,
|
pub scale_factor: f64,
|
||||||
|
@ -75,6 +72,8 @@ impl Default for UiSurface {
|
||||||
}
|
}
|
||||||
|
|
||||||
impl UiSurface {
|
impl UiSurface {
|
||||||
|
/// Retrieves the taffy node corresponding to given entity exists, or inserts a new taffy node into the layout if no corresponding node exists.
|
||||||
|
/// Then convert the given `Style` and use it update the taffy node's style.
|
||||||
pub fn upsert_node(&mut self, entity: Entity, style: &Style, context: &LayoutContext) {
|
pub fn upsert_node(&mut self, entity: Entity, style: &Style, context: &LayoutContext) {
|
||||||
let mut added = false;
|
let mut added = false;
|
||||||
let taffy = &mut self.taffy;
|
let taffy = &mut self.taffy;
|
||||||
|
@ -90,43 +89,13 @@ impl UiSurface {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn upsert_leaf(
|
/// Update the `MeasureFunc` of the taffy node corresponding to the given [`Entity`].
|
||||||
&mut self,
|
pub fn update_measure(&mut self, entity: Entity, measure_func: taffy::node::MeasureFunc) {
|
||||||
entity: Entity,
|
let taffy_node = self.entity_to_taffy.get(&entity).unwrap();
|
||||||
style: &Style,
|
self.taffy.set_measure(*taffy_node, Some(measure_func)).ok();
|
||||||
calculated_size: &CalculatedSize,
|
|
||||||
context: &LayoutContext,
|
|
||||||
) {
|
|
||||||
let taffy = &mut self.taffy;
|
|
||||||
let taffy_style = convert::from_style(context, style);
|
|
||||||
let measure = calculated_size.measure.dyn_clone();
|
|
||||||
let measure_func = taffy::node::MeasureFunc::Boxed(Box::new(
|
|
||||||
move |constraints: Size<Option<f32>>, available: Size<AvailableSpace>| {
|
|
||||||
let size = measure.measure(
|
|
||||||
constraints.width,
|
|
||||||
constraints.height,
|
|
||||||
available.width,
|
|
||||||
available.height,
|
|
||||||
);
|
|
||||||
taffy::geometry::Size {
|
|
||||||
width: size.x,
|
|
||||||
height: size.y,
|
|
||||||
}
|
|
||||||
},
|
|
||||||
));
|
|
||||||
if let Some(taffy_node) = self.entity_to_taffy.get(&entity) {
|
|
||||||
self.taffy.set_style(*taffy_node, taffy_style).unwrap();
|
|
||||||
self.taffy
|
|
||||||
.set_measure(*taffy_node, Some(measure_func))
|
|
||||||
.unwrap();
|
|
||||||
} else {
|
|
||||||
let taffy_node = taffy
|
|
||||||
.new_leaf_with_measure(taffy_style, measure_func)
|
|
||||||
.unwrap();
|
|
||||||
self.entity_to_taffy.insert(entity, taffy_node);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Update the children of the taffy node corresponding to the given [`Entity`].
|
||||||
pub fn update_children(&mut self, entity: Entity, children: &Children) {
|
pub fn update_children(&mut self, entity: Entity, children: &Children) {
|
||||||
let mut taffy_children = Vec::with_capacity(children.len());
|
let mut taffy_children = Vec::with_capacity(children.len());
|
||||||
for child in children {
|
for child in children {
|
||||||
|
@ -160,6 +129,7 @@ without UI components as a child of an entity with UI components, results may be
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Retrieve or insert the root layout node and update its size to match the size of the window.
|
||||||
pub fn update_window(&mut self, window: Entity, window_resolution: &WindowResolution) {
|
pub fn update_window(&mut self, window: Entity, window_resolution: &WindowResolution) {
|
||||||
let taffy = &mut self.taffy;
|
let taffy = &mut self.taffy;
|
||||||
let node = self
|
let node = self
|
||||||
|
@ -185,6 +155,7 @@ without UI components as a child of an entity with UI components, results may be
|
||||||
.unwrap();
|
.unwrap();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Set the ui node entities without a [`Parent`] as children to the root node in the taffy layout.
|
||||||
pub fn set_window_children(
|
pub fn set_window_children(
|
||||||
&mut self,
|
&mut self,
|
||||||
parent_window: Entity,
|
parent_window: Entity,
|
||||||
|
@ -197,6 +168,7 @@ without UI components as a child of an entity with UI components, results may be
|
||||||
self.taffy.set_children(*taffy_node, &child_nodes).unwrap();
|
self.taffy.set_children(*taffy_node, &child_nodes).unwrap();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Compute the layout for each window entity's corresponding root node in the layout.
|
||||||
pub fn compute_window_layouts(&mut self) {
|
pub fn compute_window_layouts(&mut self) {
|
||||||
for window_node in self.window_nodes.values() {
|
for window_node in self.window_nodes.values() {
|
||||||
self.taffy
|
self.taffy
|
||||||
|
@ -214,6 +186,8 @@ without UI components as a child of an entity with UI components, results may be
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Get the layout geometry for the taffy node corresponding to the ui node [`Entity`].
|
||||||
|
/// Does not compute the layout geometry, `compute_window_layouts` should be run before using this function.
|
||||||
pub fn get_layout(&self, entity: Entity) -> Result<&taffy::layout::Layout, LayoutError> {
|
pub fn get_layout(&self, entity: Entity) -> Result<&taffy::layout::Layout, LayoutError> {
|
||||||
if let Some(taffy_node) = self.entity_to_taffy.get(&entity) {
|
if let Some(taffy_node) = self.entity_to_taffy.get(&entity) {
|
||||||
self.taffy
|
self.taffy
|
||||||
|
@ -235,6 +209,7 @@ pub enum LayoutError {
|
||||||
TaffyError(taffy::error::TaffyError),
|
TaffyError(taffy::error::TaffyError),
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Updates the UI's layout tree, computes the new layout geometry and then updates the sizes and transforms of all the UI nodes.
|
||||||
#[allow(clippy::too_many_arguments)]
|
#[allow(clippy::too_many_arguments)]
|
||||||
pub fn ui_layout_system(
|
pub fn ui_layout_system(
|
||||||
primary_window: Query<(Entity, &Window), With<PrimaryWindow>>,
|
primary_window: Query<(Entity, &Window), With<PrimaryWindow>>,
|
||||||
|
@ -244,18 +219,11 @@ pub fn ui_layout_system(
|
||||||
mut resize_events: EventReader<bevy_window::WindowResized>,
|
mut resize_events: EventReader<bevy_window::WindowResized>,
|
||||||
mut ui_surface: ResMut<UiSurface>,
|
mut ui_surface: ResMut<UiSurface>,
|
||||||
root_node_query: Query<Entity, (With<Node>, Without<Parent>)>,
|
root_node_query: Query<Entity, (With<Node>, Without<Parent>)>,
|
||||||
full_node_query: Query<(Entity, &Style, Option<&CalculatedSize>), With<Node>>,
|
style_query: Query<(Entity, Ref<Style>), With<Node>>,
|
||||||
changed_style_query: Query<
|
mut measure_query: Query<(Entity, &mut ContentSize)>,
|
||||||
(Entity, &Style),
|
|
||||||
(With<Node>, Without<CalculatedSize>, Changed<Style>),
|
|
||||||
>,
|
|
||||||
changed_size_query: Query<
|
|
||||||
(Entity, &Style, &CalculatedSize),
|
|
||||||
(With<Node>, Or<(Changed<CalculatedSize>, Changed<Style>)>),
|
|
||||||
>,
|
|
||||||
children_query: Query<(Entity, &Children), (With<Node>, Changed<Children>)>,
|
children_query: Query<(Entity, &Children), (With<Node>, Changed<Children>)>,
|
||||||
mut removed_children: RemovedComponents<Children>,
|
mut removed_children: RemovedComponents<Children>,
|
||||||
mut removed_calculated_sizes: RemovedComponents<CalculatedSize>,
|
mut removed_content_sizes: RemovedComponents<ContentSize>,
|
||||||
mut node_transform_query: Query<(Entity, &mut Node, &mut Transform, Option<&Parent>)>,
|
mut node_transform_query: Query<(Entity, &mut Node, &mut Transform, Option<&Parent>)>,
|
||||||
mut removed_nodes: RemovedComponents<Node>,
|
mut removed_nodes: RemovedComponents<Node>,
|
||||||
) {
|
) {
|
||||||
|
@ -285,35 +253,34 @@ pub fn ui_layout_system(
|
||||||
}
|
}
|
||||||
|
|
||||||
let scale_factor = logical_to_physical_factor * ui_scale.scale;
|
let scale_factor = logical_to_physical_factor * ui_scale.scale;
|
||||||
|
|
||||||
let layout_context = LayoutContext::new(scale_factor, physical_size);
|
let layout_context = LayoutContext::new(scale_factor, physical_size);
|
||||||
|
|
||||||
if !scale_factor_events.is_empty() || ui_scale.is_changed() || resized {
|
if !scale_factor_events.is_empty() || ui_scale.is_changed() || resized {
|
||||||
scale_factor_events.clear();
|
scale_factor_events.clear();
|
||||||
// update all nodes
|
// update all nodes
|
||||||
for (entity, style, calculated_size) in &full_node_query {
|
for (entity, style) in style_query.iter() {
|
||||||
if let Some(calculated_size) = calculated_size {
|
ui_surface.upsert_node(entity, &style, &layout_context);
|
||||||
ui_surface.upsert_leaf(entity, style, calculated_size, &layout_context);
|
|
||||||
} else {
|
|
||||||
ui_surface.upsert_node(entity, style, &layout_context);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
// update changed nodes without a calculated size
|
for (entity, style) in style_query.iter() {
|
||||||
for (entity, style) in changed_style_query.iter() {
|
if style.is_changed() {
|
||||||
ui_surface.upsert_node(entity, style, &layout_context);
|
ui_surface.upsert_node(entity, &style, &layout_context);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// update changed nodes with a calculated size
|
for (entity, mut content_size) in measure_query.iter_mut() {
|
||||||
for (entity, style, calculated_size) in changed_size_query.iter() {
|
if let Some(measure_func) = content_size.measure_func.take() {
|
||||||
ui_surface.upsert_leaf(entity, style, calculated_size, &layout_context);
|
ui_surface.update_measure(entity, measure_func);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// clean up removed nodes
|
// clean up removed nodes
|
||||||
ui_surface.remove_entities(removed_nodes.iter());
|
ui_surface.remove_entities(removed_nodes.iter());
|
||||||
|
|
||||||
// When a `CalculatedSize` component is removed from an entity, we need to remove the measure from the corresponding taffy node.
|
// When a `ContentSize` component is removed from an entity, we need to remove the measure from the corresponding taffy node.
|
||||||
for entity in removed_calculated_sizes.iter() {
|
for entity in removed_content_sizes.iter() {
|
||||||
ui_surface.try_remove_measure(entity);
|
ui_surface.try_remove_measure(entity);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -88,7 +88,7 @@ impl Plugin for UiPlugin {
|
||||||
.register_type::<AlignContent>()
|
.register_type::<AlignContent>()
|
||||||
.register_type::<AlignItems>()
|
.register_type::<AlignItems>()
|
||||||
.register_type::<AlignSelf>()
|
.register_type::<AlignSelf>()
|
||||||
.register_type::<CalculatedSize>()
|
.register_type::<ContentSize>()
|
||||||
.register_type::<Direction>()
|
.register_type::<Direction>()
|
||||||
.register_type::<Display>()
|
.register_type::<Display>()
|
||||||
.register_type::<FlexDirection>()
|
.register_type::<FlexDirection>()
|
||||||
|
@ -144,7 +144,7 @@ impl Plugin for UiPlugin {
|
||||||
#[cfg(feature = "bevy_text")]
|
#[cfg(feature = "bevy_text")]
|
||||||
app.add_plugin(accessibility::AccessibilityPlugin);
|
app.add_plugin(accessibility::AccessibilityPlugin);
|
||||||
app.add_systems(PostUpdate, {
|
app.add_systems(PostUpdate, {
|
||||||
let system = widget::update_image_calculated_size_system.before(UiSystem::Layout);
|
let system = widget::update_image_content_size_system.before(UiSystem::Layout);
|
||||||
// Potential conflicts: `Assets<Image>`
|
// Potential conflicts: `Assets<Image>`
|
||||||
// They run independently since `widget::image_node_system` will only ever observe
|
// They run independently since `widget::image_node_system` will only ever observe
|
||||||
// its own UiImage, and `widget::text_system` & `bevy_text::update_text2d_layout`
|
// its own UiImage, and `widget::text_system` & `bevy_text::update_text2d_layout`
|
||||||
|
|
|
@ -1,12 +1,13 @@
|
||||||
use bevy_ecs::prelude::Component;
|
use bevy_ecs::prelude::Component;
|
||||||
|
use bevy_ecs::reflect::ReflectComponent;
|
||||||
use bevy_math::Vec2;
|
use bevy_math::Vec2;
|
||||||
use bevy_reflect::Reflect;
|
use bevy_reflect::Reflect;
|
||||||
use std::fmt::Formatter;
|
use std::fmt::Formatter;
|
||||||
pub use taffy::style::AvailableSpace;
|
pub use taffy::style::AvailableSpace;
|
||||||
|
|
||||||
impl std::fmt::Debug for CalculatedSize {
|
impl std::fmt::Debug for ContentSize {
|
||||||
fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
|
fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
|
||||||
f.debug_struct("CalculatedSize").finish()
|
f.debug_struct("ContentSize").finish()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -21,9 +22,6 @@ pub trait Measure: Send + Sync + 'static {
|
||||||
available_width: AvailableSpace,
|
available_width: AvailableSpace,
|
||||||
available_height: AvailableSpace,
|
available_height: AvailableSpace,
|
||||||
) -> Vec2;
|
) -> Vec2;
|
||||||
|
|
||||||
/// Clone and box self.
|
|
||||||
fn dyn_clone(&self) -> Box<dyn Measure>;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// A `FixedMeasure` is a `Measure` that ignores all constraints and
|
/// A `FixedMeasure` is a `Measure` that ignores all constraints and
|
||||||
|
@ -43,35 +41,42 @@ impl Measure for FixedMeasure {
|
||||||
) -> Vec2 {
|
) -> Vec2 {
|
||||||
self.size
|
self.size
|
||||||
}
|
}
|
||||||
|
|
||||||
fn dyn_clone(&self) -> Box<dyn Measure> {
|
|
||||||
Box::new(self.clone())
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// A node with a `CalculatedSize` component is a node where its size
|
/// A node with a `ContentSize` component is a node where its size
|
||||||
/// is based on its content.
|
/// is based on its content.
|
||||||
#[derive(Component, Reflect)]
|
#[derive(Component, Reflect)]
|
||||||
pub struct CalculatedSize {
|
#[reflect(Component)]
|
||||||
|
pub struct ContentSize {
|
||||||
/// The `Measure` used to compute the intrinsic size
|
/// The `Measure` used to compute the intrinsic size
|
||||||
#[reflect(ignore)]
|
#[reflect(ignore)]
|
||||||
pub measure: Box<dyn Measure>,
|
pub(crate) measure_func: Option<taffy::node::MeasureFunc>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ContentSize {
|
||||||
|
/// Set a `Measure` for this function
|
||||||
|
pub fn set(&mut self, measure: impl Measure) {
|
||||||
|
let measure_func =
|
||||||
|
move |size: taffy::prelude::Size<Option<f32>>,
|
||||||
|
available: taffy::prelude::Size<AvailableSpace>| {
|
||||||
|
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)]
|
#[allow(clippy::derivable_impls)]
|
||||||
impl Default for CalculatedSize {
|
impl Default for ContentSize {
|
||||||
fn default() -> Self {
|
fn default() -> Self {
|
||||||
Self {
|
Self {
|
||||||
// Default `FixedMeasure` always returns zero size.
|
measure_func: Some(taffy::node::MeasureFunc::Raw(|_, _| {
|
||||||
measure: Box::<FixedMeasure>::default(),
|
taffy::prelude::Size::ZERO
|
||||||
}
|
})),
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Clone for CalculatedSize {
|
|
||||||
fn clone(&self) -> Self {
|
|
||||||
Self {
|
|
||||||
measure: self.measure.dyn_clone(),
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,7 +2,7 @@
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
widget::{Button, UiImageSize},
|
widget::{Button, UiImageSize},
|
||||||
BackgroundColor, CalculatedSize, FocusPolicy, Interaction, Node, Style, UiImage, ZIndex,
|
BackgroundColor, ContentSize, FocusPolicy, Interaction, Node, Style, UiImage, ZIndex,
|
||||||
};
|
};
|
||||||
use bevy_ecs::bundle::Bundle;
|
use bevy_ecs::bundle::Bundle;
|
||||||
use bevy_render::{
|
use bevy_render::{
|
||||||
|
@ -63,7 +63,7 @@ impl Default for NodeBundle {
|
||||||
}
|
}
|
||||||
|
|
||||||
/// A UI node that is an image
|
/// A UI node that is an image
|
||||||
#[derive(Bundle, Clone, Debug, Default)]
|
#[derive(Bundle, Debug, Default)]
|
||||||
pub struct ImageBundle {
|
pub struct ImageBundle {
|
||||||
/// Describes the logical size of the node
|
/// Describes the logical size of the node
|
||||||
///
|
///
|
||||||
|
@ -74,7 +74,7 @@ pub struct ImageBundle {
|
||||||
/// In some cases these styles also affect how the node drawn/painted.
|
/// In some cases these styles also affect how the node drawn/painted.
|
||||||
pub style: Style,
|
pub style: Style,
|
||||||
/// The calculated size based on the given image
|
/// The calculated size based on the given image
|
||||||
pub calculated_size: CalculatedSize,
|
pub calculated_size: ContentSize,
|
||||||
/// The background color, which serves as a "fill" for this node
|
/// The background color, which serves as a "fill" for this node
|
||||||
///
|
///
|
||||||
/// Combines with `UiImage` to tint the provided image.
|
/// Combines with `UiImage` to tint the provided image.
|
||||||
|
@ -107,7 +107,7 @@ pub struct ImageBundle {
|
||||||
|
|
||||||
#[cfg(feature = "bevy_text")]
|
#[cfg(feature = "bevy_text")]
|
||||||
/// A UI node that is text
|
/// A UI node that is text
|
||||||
#[derive(Bundle, Clone, Debug)]
|
#[derive(Bundle, Debug)]
|
||||||
pub struct TextBundle {
|
pub struct TextBundle {
|
||||||
/// Describes the logical size of the node
|
/// Describes the logical size of the node
|
||||||
pub node: Node,
|
pub node: Node,
|
||||||
|
@ -119,7 +119,7 @@ pub struct TextBundle {
|
||||||
/// Text layout information
|
/// Text layout information
|
||||||
pub text_layout_info: TextLayoutInfo,
|
pub text_layout_info: TextLayoutInfo,
|
||||||
/// The calculated size based on the given image
|
/// The calculated size based on the given image
|
||||||
pub calculated_size: CalculatedSize,
|
pub calculated_size: ContentSize,
|
||||||
/// Whether this node should block interaction with lower nodes
|
/// Whether this node should block interaction with lower nodes
|
||||||
pub focus_policy: FocusPolicy,
|
pub focus_policy: FocusPolicy,
|
||||||
/// The transform of the node
|
/// The transform of the node
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
use crate::{measurement::AvailableSpace, CalculatedSize, Measure, Node, UiImage};
|
use crate::{measurement::AvailableSpace, ContentSize, Measure, Node, UiImage};
|
||||||
use bevy_asset::Assets;
|
use bevy_asset::Assets;
|
||||||
#[cfg(feature = "bevy_text")]
|
#[cfg(feature = "bevy_text")]
|
||||||
use bevy_ecs::query::Without;
|
use bevy_ecs::query::Without;
|
||||||
|
@ -61,25 +61,21 @@ impl Measure for ImageMeasure {
|
||||||
}
|
}
|
||||||
size
|
size
|
||||||
}
|
}
|
||||||
|
|
||||||
fn dyn_clone(&self) -> Box<dyn Measure> {
|
|
||||||
Box::new(self.clone())
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Updates calculated size of the node based on the image provided
|
/// Updates content size of the node based on the image provided
|
||||||
pub fn update_image_calculated_size_system(
|
pub fn update_image_content_size_system(
|
||||||
textures: Res<Assets<Image>>,
|
textures: Res<Assets<Image>>,
|
||||||
#[cfg(feature = "bevy_text")] mut query: Query<
|
#[cfg(feature = "bevy_text")] mut query: Query<
|
||||||
(&mut CalculatedSize, &UiImage, &mut UiImageSize),
|
(&mut ContentSize, &UiImage, &mut UiImageSize),
|
||||||
(With<Node>, Without<Text>),
|
(With<Node>, Without<Text>),
|
||||||
>,
|
>,
|
||||||
#[cfg(not(feature = "bevy_text"))] mut query: Query<
|
#[cfg(not(feature = "bevy_text"))] mut query: Query<
|
||||||
(&mut CalculatedSize, &UiImage, &mut UiImageSize),
|
(&mut ContentSize, &UiImage, &mut UiImageSize),
|
||||||
With<Node>,
|
With<Node>,
|
||||||
>,
|
>,
|
||||||
) {
|
) {
|
||||||
for (mut calculated_size, image, mut image_size) in &mut query {
|
for (mut content_size, image, mut image_size) in &mut query {
|
||||||
if let Some(texture) = textures.get(&image.texture) {
|
if let Some(texture) = textures.get(&image.texture) {
|
||||||
let size = Vec2::new(
|
let size = Vec2::new(
|
||||||
texture.texture_descriptor.size.width as f32,
|
texture.texture_descriptor.size.width as f32,
|
||||||
|
@ -88,7 +84,7 @@ pub fn update_image_calculated_size_system(
|
||||||
// Update only if size has changed to avoid needless layout calculations
|
// Update only if size has changed to avoid needless layout calculations
|
||||||
if size != image_size.size {
|
if size != image_size.size {
|
||||||
image_size.size = size;
|
image_size.size = size;
|
||||||
calculated_size.measure = Box::new(ImageMeasure { size });
|
content_size.set(ImageMeasure { size });
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
use crate::{CalculatedSize, Measure, Node, UiScale};
|
use crate::{ContentSize, Measure, Node, UiScale};
|
||||||
use bevy_asset::Assets;
|
use bevy_asset::Assets;
|
||||||
use bevy_ecs::{
|
use bevy_ecs::{
|
||||||
entity::Entity,
|
entity::Entity,
|
||||||
|
@ -52,10 +52,6 @@ impl Measure for TextMeasure {
|
||||||
)
|
)
|
||||||
.ceil()
|
.ceil()
|
||||||
}
|
}
|
||||||
|
|
||||||
fn dyn_clone(&self) -> Box<dyn Measure> {
|
|
||||||
Box::new(self.clone())
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Creates a `Measure` for text nodes that allows the UI to determine the appropriate amount of space
|
/// Creates a `Measure` for text nodes that allows the UI to determine the appropriate amount of space
|
||||||
|
@ -70,7 +66,7 @@ pub fn measure_text_system(
|
||||||
mut text_queries: ParamSet<(
|
mut text_queries: ParamSet<(
|
||||||
Query<Entity, (Changed<Text>, With<Node>)>,
|
Query<Entity, (Changed<Text>, With<Node>)>,
|
||||||
Query<Entity, (With<Text>, With<Node>)>,
|
Query<Entity, (With<Text>, With<Node>)>,
|
||||||
Query<(&Text, &mut CalculatedSize)>,
|
Query<(&Text, &mut ContentSize)>,
|
||||||
)>,
|
)>,
|
||||||
) {
|
) {
|
||||||
let window_scale_factor = windows
|
let window_scale_factor = windows
|
||||||
|
@ -103,7 +99,7 @@ pub fn measure_text_system(
|
||||||
let mut new_queue = Vec::new();
|
let mut new_queue = Vec::new();
|
||||||
let mut query = text_queries.p2();
|
let mut query = text_queries.p2();
|
||||||
for entity in queued_text.drain(..) {
|
for entity in queued_text.drain(..) {
|
||||||
if let Ok((text, mut calculated_size)) = query.get_mut(entity) {
|
if let Ok((text, mut content_size)) = query.get_mut(entity) {
|
||||||
match text_pipeline.create_text_measure(
|
match text_pipeline.create_text_measure(
|
||||||
&fonts,
|
&fonts,
|
||||||
&text.sections,
|
&text.sections,
|
||||||
|
@ -112,7 +108,7 @@ pub fn measure_text_system(
|
||||||
text.linebreak_behavior,
|
text.linebreak_behavior,
|
||||||
) {
|
) {
|
||||||
Ok(measure) => {
|
Ok(measure) => {
|
||||||
calculated_size.measure = Box::new(TextMeasure { info: measure });
|
content_size.set(TextMeasure { info: measure });
|
||||||
}
|
}
|
||||||
Err(TextError::NoSuchFont) => {
|
Err(TextError::NoSuchFont) => {
|
||||||
new_queue.push(entity);
|
new_queue.push(entity);
|
||||||
|
|
Loading…
Reference in a new issue