Implement rotation for Text2d (#2084)
Fixes https://github.com/bevyengine/bevy/issues/2080  Co-authored-by: Nathan Stocks <cleancut@github.com> Co-authored-by: Denis Laprise <nside@users.noreply.github.com>
This commit is contained in:
parent
41d9122740
commit
7d0e98f34c
@ -1,4 +1,6 @@
|
|||||||
|
use crate::{PositionedGlyph, TextSection};
|
||||||
use bevy_math::{Mat4, Vec3};
|
use bevy_math::{Mat4, Vec3};
|
||||||
|
use bevy_render::pipeline::IndexFormat;
|
||||||
use bevy_render::{
|
use bevy_render::{
|
||||||
draw::{Draw, DrawContext, DrawError, Drawable},
|
draw::{Draw, DrawContext, DrawError, Drawable},
|
||||||
mesh,
|
mesh,
|
||||||
@ -8,19 +10,18 @@ use bevy_render::{
|
|||||||
renderer::{BindGroup, RenderResourceBindings, RenderResourceId},
|
renderer::{BindGroup, RenderResourceBindings, RenderResourceId},
|
||||||
};
|
};
|
||||||
use bevy_sprite::TextureAtlasSprite;
|
use bevy_sprite::TextureAtlasSprite;
|
||||||
|
use bevy_transform::prelude::GlobalTransform;
|
||||||
use bevy_utils::tracing::error;
|
use bevy_utils::tracing::error;
|
||||||
|
|
||||||
use crate::{PositionedGlyph, TextSection};
|
|
||||||
use bevy_render::pipeline::IndexFormat;
|
|
||||||
|
|
||||||
pub struct DrawableText<'a> {
|
pub struct DrawableText<'a> {
|
||||||
pub render_resource_bindings: &'a mut RenderResourceBindings,
|
pub render_resource_bindings: &'a mut RenderResourceBindings,
|
||||||
pub position: Vec3,
|
pub global_transform: GlobalTransform,
|
||||||
pub scale_factor: f32,
|
pub scale_factor: f32,
|
||||||
pub sections: &'a [TextSection],
|
pub sections: &'a [TextSection],
|
||||||
pub text_glyphs: &'a Vec<PositionedGlyph>,
|
pub text_glyphs: &'a Vec<PositionedGlyph>,
|
||||||
pub msaa: &'a Msaa,
|
pub msaa: &'a Msaa,
|
||||||
pub font_quad_vertex_layout: &'a VertexBufferLayout,
|
pub font_quad_vertex_layout: &'a VertexBufferLayout,
|
||||||
|
pub alignment_offset: Vec3,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<'a> Drawable for DrawableText<'a> {
|
impl<'a> Drawable for DrawableText<'a> {
|
||||||
@ -76,19 +77,12 @@ impl<'a> Drawable for DrawableText<'a> {
|
|||||||
flip_y: false,
|
flip_y: false,
|
||||||
};
|
};
|
||||||
|
|
||||||
// To get the rendering right for non-one scaling factors, we need
|
let transform = Mat4::from_rotation_translation(
|
||||||
// the sprite to be drawn in "physical" coordinates. This is because
|
self.global_transform.rotation,
|
||||||
// the shader uses the size of the sprite to control the size on
|
self.global_transform.translation,
|
||||||
// screen. To accomplish this we make the sprite transform
|
) * Mat4::from_scale(self.global_transform.scale / self.scale_factor)
|
||||||
// convert from physical coordinates to logical coordinates in
|
|
||||||
// addition to altering the origin. Since individual glyphs will
|
|
||||||
// already be in physical coordinates, we just need to convert the
|
|
||||||
// overall position to physical coordinates to get the sprites
|
|
||||||
// physical position.
|
|
||||||
|
|
||||||
let transform = Mat4::from_scale(Vec3::splat(1. / self.scale_factor))
|
|
||||||
* Mat4::from_translation(
|
* Mat4::from_translation(
|
||||||
self.position * self.scale_factor + tv.position.extend(0.),
|
self.alignment_offset * self.scale_factor + tv.position.extend(0.),
|
||||||
);
|
);
|
||||||
|
|
||||||
let transform_buffer = context.get_uniform_buffer(&transform).unwrap();
|
let transform_buffer = context.get_uniform_buffer(&transform).unwrap();
|
||||||
|
|||||||
@ -93,26 +93,25 @@ pub fn draw_text2d_system(
|
|||||||
let (width, height) = (calculated_size.size.width, calculated_size.size.height);
|
let (width, height) = (calculated_size.size.width, calculated_size.size.height);
|
||||||
|
|
||||||
if let Some(text_glyphs) = text_pipeline.get_glyphs(&entity) {
|
if let Some(text_glyphs) = text_pipeline.get_glyphs(&entity) {
|
||||||
let position = global_transform.translation
|
let alignment_offset = match text.alignment.vertical {
|
||||||
+ match text.alignment.vertical {
|
VerticalAlign::Top => Vec3::new(0.0, -height, 0.0),
|
||||||
VerticalAlign::Top => Vec3::ZERO,
|
|
||||||
VerticalAlign::Center => Vec3::new(0.0, -height * 0.5, 0.0),
|
VerticalAlign::Center => Vec3::new(0.0, -height * 0.5, 0.0),
|
||||||
VerticalAlign::Bottom => Vec3::new(0.0, -height, 0.0),
|
VerticalAlign::Bottom => Vec3::ZERO,
|
||||||
}
|
} + match text.alignment.horizontal {
|
||||||
+ match text.alignment.horizontal {
|
HorizontalAlign::Left => Vec3::ZERO,
|
||||||
HorizontalAlign::Left => Vec3::new(-width, 0.0, 0.0),
|
|
||||||
HorizontalAlign::Center => Vec3::new(-width * 0.5, 0.0, 0.0),
|
HorizontalAlign::Center => Vec3::new(-width * 0.5, 0.0, 0.0),
|
||||||
HorizontalAlign::Right => Vec3::ZERO,
|
HorizontalAlign::Right => Vec3::new(-width, 0.0, 0.0),
|
||||||
};
|
};
|
||||||
|
|
||||||
let mut drawable_text = DrawableText {
|
let mut drawable_text = DrawableText {
|
||||||
render_resource_bindings: &mut render_resource_bindings,
|
render_resource_bindings: &mut render_resource_bindings,
|
||||||
position,
|
global_transform: *global_transform,
|
||||||
|
scale_factor,
|
||||||
msaa: &msaa,
|
msaa: &msaa,
|
||||||
text_glyphs: &text_glyphs.glyphs,
|
text_glyphs: &text_glyphs.glyphs,
|
||||||
font_quad_vertex_layout: &font_quad_vertex_layout,
|
font_quad_vertex_layout: &font_quad_vertex_layout,
|
||||||
scale_factor,
|
|
||||||
sections: &text.sections,
|
sections: &text.sections,
|
||||||
|
alignment_offset,
|
||||||
};
|
};
|
||||||
|
|
||||||
drawable_text.draw(&mut draw, &mut context).unwrap();
|
drawable_text.draw(&mut draw, &mut context).unwrap();
|
||||||
|
|||||||
@ -167,16 +167,15 @@ pub fn draw_text_system(
|
|||||||
}
|
}
|
||||||
|
|
||||||
if let Some(text_glyphs) = text_pipeline.get_glyphs(&entity) {
|
if let Some(text_glyphs) = text_pipeline.get_glyphs(&entity) {
|
||||||
let position = global_transform.translation - (node.size / 2.0).extend(0.0);
|
|
||||||
|
|
||||||
let mut drawable_text = DrawableText {
|
let mut drawable_text = DrawableText {
|
||||||
render_resource_bindings: &mut render_resource_bindings,
|
render_resource_bindings: &mut render_resource_bindings,
|
||||||
position,
|
global_transform: *global_transform,
|
||||||
scale_factor: scale_factor as f32,
|
scale_factor: scale_factor as f32,
|
||||||
msaa: &msaa,
|
msaa: &msaa,
|
||||||
text_glyphs: &text_glyphs.glyphs,
|
text_glyphs: &text_glyphs.glyphs,
|
||||||
font_quad_vertex_layout: &vertex_buffer_layout,
|
font_quad_vertex_layout: &vertex_buffer_layout,
|
||||||
sections: &text.sections,
|
sections: &text.sections,
|
||||||
|
alignment_offset: (node.size / -2.0).extend(0.0),
|
||||||
};
|
};
|
||||||
|
|
||||||
drawable_text.draw(&mut draw, &mut context).unwrap();
|
drawable_text.draw(&mut draw, &mut context).unwrap();
|
||||||
|
|||||||
@ -4,36 +4,79 @@ fn main() {
|
|||||||
App::build()
|
App::build()
|
||||||
.add_plugins(DefaultPlugins)
|
.add_plugins(DefaultPlugins)
|
||||||
.add_startup_system(setup.system())
|
.add_startup_system(setup.system())
|
||||||
.add_system(animate.system())
|
.add_system(animate_translation.system())
|
||||||
|
.add_system(animate_rotation.system())
|
||||||
|
.add_system(animate_scale.system())
|
||||||
.run();
|
.run();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
struct AnimateTranslation;
|
||||||
|
struct AnimateRotation;
|
||||||
|
struct AnimateScale;
|
||||||
|
|
||||||
fn setup(mut commands: Commands, asset_server: Res<AssetServer>) {
|
fn setup(mut commands: Commands, asset_server: Res<AssetServer>) {
|
||||||
// 2d camera
|
let font = asset_server.load("fonts/FiraSans-Bold.ttf");
|
||||||
commands.spawn_bundle(OrthographicCameraBundle::new_2d());
|
let text_style = TextStyle {
|
||||||
commands.spawn_bundle(Text2dBundle {
|
font,
|
||||||
text: Text::with_section(
|
|
||||||
"This text is in the 2D scene.",
|
|
||||||
TextStyle {
|
|
||||||
font: asset_server.load("fonts/FiraSans-Bold.ttf"),
|
|
||||||
font_size: 60.0,
|
font_size: 60.0,
|
||||||
color: Color::WHITE,
|
color: Color::WHITE,
|
||||||
},
|
};
|
||||||
TextAlignment {
|
let text_alignment = TextAlignment {
|
||||||
vertical: VerticalAlign::Center,
|
vertical: VerticalAlign::Center,
|
||||||
horizontal: HorizontalAlign::Center,
|
horizontal: HorizontalAlign::Center,
|
||||||
},
|
};
|
||||||
),
|
// 2d camera
|
||||||
|
commands.spawn_bundle(OrthographicCameraBundle::new_2d());
|
||||||
|
// Demonstrate changing translation
|
||||||
|
commands
|
||||||
|
.spawn_bundle(Text2dBundle {
|
||||||
|
text: Text::with_section("translation", text_style.clone(), text_alignment),
|
||||||
..Default::default()
|
..Default::default()
|
||||||
});
|
})
|
||||||
|
.insert(AnimateTranslation);
|
||||||
|
// Demonstrate changing rotation
|
||||||
|
commands
|
||||||
|
.spawn_bundle(Text2dBundle {
|
||||||
|
text: Text::with_section("rotation", text_style.clone(), text_alignment),
|
||||||
|
..Default::default()
|
||||||
|
})
|
||||||
|
.insert(AnimateRotation);
|
||||||
|
// Demonstrate changing scale
|
||||||
|
commands
|
||||||
|
.spawn_bundle(Text2dBundle {
|
||||||
|
text: Text::with_section("scale", text_style, text_alignment),
|
||||||
|
..Default::default()
|
||||||
|
})
|
||||||
|
.insert(AnimateScale);
|
||||||
}
|
}
|
||||||
|
|
||||||
fn animate(time: Res<Time>, mut query: Query<&mut Transform, With<Text>>) {
|
fn animate_translation(
|
||||||
// `Transform.translation` will determine the location of the text.
|
time: Res<Time>,
|
||||||
// `Transform.scale` and `Transform.rotation` do not yet affect text (though you can set the
|
mut query: Query<&mut Transform, (With<Text>, With<AnimateTranslation>)>,
|
||||||
// size of the text via `Text.style.font_size`)
|
) {
|
||||||
for mut transform in query.iter_mut() {
|
for mut transform in query.iter_mut() {
|
||||||
transform.translation.x = 100.0 * time.seconds_since_startup().sin() as f32;
|
transform.translation.x = 100.0 * time.seconds_since_startup().sin() as f32 - 400.0;
|
||||||
transform.translation.y = 100.0 * time.seconds_since_startup().cos() as f32;
|
transform.translation.y = 100.0 * time.seconds_since_startup().cos() as f32;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn animate_rotation(
|
||||||
|
time: Res<Time>,
|
||||||
|
mut query: Query<&mut Transform, (With<Text>, With<AnimateRotation>)>,
|
||||||
|
) {
|
||||||
|
for mut transform in query.iter_mut() {
|
||||||
|
transform.rotation = Quat::from_rotation_z(time.seconds_since_startup().cos() as f32);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn animate_scale(
|
||||||
|
time: Res<Time>,
|
||||||
|
mut query: Query<&mut Transform, (With<Text>, With<AnimateScale>)>,
|
||||||
|
) {
|
||||||
|
// Consider changing font-size instead of scaling the transform. Scaling a Text2D will scale the
|
||||||
|
// rendered quad, resulting in a pixellated look.
|
||||||
|
for mut transform in query.iter_mut() {
|
||||||
|
transform.translation = Vec3::new(400.0, 0.0, 0.0);
|
||||||
|
transform.scale = Vec3::splat((time.seconds_since_startup().sin() as f32 + 1.1) * 2.0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
Loading…
Reference in New Issue
Block a user