bevy/crates/bevy_ui/src/node.rs

109 lines
2.9 KiB
Rust
Raw Normal View History

2020-04-06 21:20:53 +00:00
use super::{Anchors, Margins};
use bevy_math::{Mat4, Vec2, Vec3};
use bevy_render::renderer::RenderResources;
use bevy_transform::components::LocalTransform;
2020-01-13 00:51:21 +00:00
#[derive(Debug, Clone)]
enum MarginGrowDirection {
2020-01-13 00:51:21 +00:00
Negative,
Positive,
}
#[derive(Debug, Clone, Default, RenderResources)]
2020-01-13 00:51:21 +00:00
pub struct Node {
pub size: Vec2,
#[render_resources(ignore)]
2020-01-13 00:51:21 +00:00
pub position: Vec2,
#[render_resources(ignore)]
2020-01-13 00:51:21 +00:00
pub anchors: Anchors,
#[render_resources(ignore)]
2020-01-13 00:51:21 +00:00
pub margins: Margins,
}
impl Node {
pub fn new(anchors: Anchors, margins: Margins) -> Self {
2020-01-13 00:51:21 +00:00
Node {
anchors,
margins,
..Default::default()
2020-01-13 00:51:21 +00:00
}
}
pub fn positioned(position: Vec2, anchors: Anchors, margins: Margins) -> Self {
2020-01-13 00:51:21 +00:00
Node {
position,
anchors,
margins,
..Default::default()
2020-01-13 00:51:21 +00:00
}
}
pub fn update(
&mut self,
local_transform: &mut LocalTransform,
z_offset: f32,
parent_size: Vec2,
) {
2020-06-01 06:39:20 +00:00
let (quad_x, quad_width) = Self::compute_dimension_properties(
2020-01-13 00:51:21 +00:00
self.margins.left,
self.margins.right,
self.anchors.left,
self.anchors.right,
parent_size.x(),
2020-01-13 00:51:21 +00:00
);
2020-06-01 06:39:20 +00:00
let (quad_y, quad_height) = Self::compute_dimension_properties(
2020-01-13 00:51:21 +00:00
self.margins.bottom,
self.margins.top,
self.anchors.bottom,
self.anchors.top,
parent_size.y(),
2020-01-13 00:51:21 +00:00
);
self.size = Vec2::new(quad_width, quad_height);
local_transform.0 = Mat4::from_translation(
self.position.extend(z_offset) + Vec3::new(quad_x, quad_y, z_offset)
- (parent_size / 2.0).extend(0.0),
);
2020-01-13 00:51:21 +00:00
}
fn compute_dimension_properties(
margin0: f32,
margin1: f32,
anchor0: f32,
anchor1: f32,
length: f32,
) -> (f32, f32) {
let anchor_p0 = anchor0 * length;
let anchor_p1 = anchor1 * length;
let p0_grow_direction = if anchor_p0 <= 0.5 {
MarginGrowDirection::Positive
2020-01-13 00:51:21 +00:00
} else {
MarginGrowDirection::Negative
2020-01-13 00:51:21 +00:00
};
let p1_grow_direction = if anchor_p1 <= 0.5 {
MarginGrowDirection::Positive
2020-01-13 00:51:21 +00:00
} else {
MarginGrowDirection::Negative
2020-01-13 00:51:21 +00:00
};
let p0 = Self::compute_anchored_position(margin0, anchor_p0, p0_grow_direction);
let p1 = Self::compute_anchored_position(margin1, anchor_p1, p1_grow_direction);
2020-01-13 00:51:21 +00:00
let final_width = p1 - p0;
let p = (p0 + p1) / 2.0;
(p, final_width.abs())
2020-01-13 00:51:21 +00:00
}
fn compute_anchored_position(
2020-01-13 00:51:21 +00:00
margin: f32,
anchor_position: f32,
grow_direction: MarginGrowDirection,
2020-01-13 00:51:21 +00:00
) -> f32 {
match grow_direction {
MarginGrowDirection::Negative => anchor_position - margin,
MarginGrowDirection::Positive => anchor_position + margin,
2020-01-13 00:51:21 +00:00
}
}
}