mirror of
https://github.com/bevyengine/bevy
synced 2024-11-22 12:43:34 +00:00
5986d5d309
# Replace ab_glyph with the more capable cosmic-text Fixes #7616. Cosmic-text is a more mature text-rendering library that handles scripts and ligatures better than ab_glyph, it can also handle system fonts which can be implemented in bevy in the future Rebase of https://github.com/bevyengine/bevy/pull/8808 ## Changelog Replaces text renderer ab_glyph with cosmic-text The definition of the font size has changed with the migration to cosmic text. The behavior is now consistent with other platforms (e.g. the web), where the font size in pixels measures the height of the font (the distance between the top of the highest ascender and the bottom of the lowest descender). Font sizes in your app need to be rescaled to approximately 1.2x smaller; for example, if you were using a font size of 60.0, you should now use a font size of 50.0. ## Migration guide - `Text2dBounds` has been replaced with `TextBounds`, and it now accepts `Option`s to the bounds, instead of using `f32::INFINITY` to inidicate lack of bounds - Textsizes should be changed, dividing the current size with 1.2 will result in the same size as before. - `TextSettings` struct is removed - Feature `subpixel_alignment` has been removed since cosmic-text already does this automatically - TextBundles and things rendering texts requires the `CosmicBuffer` Component on them as well ## Suggested followups: - TextPipeline: reconstruct byte indices for keeping track of eventual cursors in text input - TextPipeline: (future work) split text entities into section entities - TextPipeline: (future work) text editing - Support line height as an option. Unitless `1.2` is the default used in browsers (1.2x font size). - Support System Fonts and font families - Example showing of animated text styles. Eg. throbbing hyperlinks --------- Co-authored-by: tigregalis <anak.harimau@gmail.com> Co-authored-by: Nico Burns <nico@nicoburns.com> Co-authored-by: sam edelsten <samedelsten1@gmail.com> Co-authored-by: Dimchikkk <velo.app1@gmail.com> Co-authored-by: Alice Cecile <alice.i.cecile@gmail.com> Co-authored-by: Rob Parrett <robparrett@gmail.com>
144 lines
4.9 KiB
Rust
144 lines
4.9 KiB
Rust
//! This example illustrates how to create UI text and update it in a system.
|
|
//!
|
|
//! It displays the current FPS in the top left corner, as well as text that changes color
|
|
//! in the bottom right. For text within a scene, please see the text2d example.
|
|
|
|
use bevy::{
|
|
color::palettes::css::GOLD,
|
|
diagnostic::{DiagnosticsStore, FrameTimeDiagnosticsPlugin},
|
|
prelude::*,
|
|
};
|
|
|
|
fn main() {
|
|
App::new()
|
|
.add_plugins((DefaultPlugins, FrameTimeDiagnosticsPlugin))
|
|
.add_systems(Startup, setup)
|
|
.add_systems(Update, (text_update_system, text_color_system))
|
|
.run();
|
|
}
|
|
|
|
// A unit struct to help identify the FPS UI component, since there may be many Text components
|
|
#[derive(Component)]
|
|
struct FpsText;
|
|
|
|
// A unit struct to help identify the color-changing Text component
|
|
#[derive(Component)]
|
|
struct ColorText;
|
|
|
|
fn setup(mut commands: Commands, asset_server: Res<AssetServer>) {
|
|
// UI camera
|
|
commands.spawn(Camera2dBundle::default());
|
|
// Text with one section
|
|
commands.spawn((
|
|
// Create a TextBundle that has a Text with a single section.
|
|
TextBundle::from_section(
|
|
// Accepts a `String` or any type that converts into a `String`, such as `&str`
|
|
"hello\nbevy!",
|
|
TextStyle {
|
|
// This font is loaded and will be used instead of the default font.
|
|
font: asset_server.load("fonts/FiraSans-Bold.ttf"),
|
|
font_size: 80.0,
|
|
..default()
|
|
},
|
|
) // Set the justification of the Text
|
|
.with_text_justify(JustifyText::Center)
|
|
// Set the style of the TextBundle itself.
|
|
.with_style(Style {
|
|
position_type: PositionType::Absolute,
|
|
bottom: Val::Px(5.0),
|
|
right: Val::Px(5.0),
|
|
..default()
|
|
}),
|
|
ColorText,
|
|
));
|
|
|
|
// Text with multiple sections
|
|
commands.spawn((
|
|
// Create a TextBundle that has a Text with a list of sections.
|
|
TextBundle::from_sections([
|
|
TextSection::new(
|
|
"FPS: ",
|
|
TextStyle {
|
|
// This font is loaded and will be used instead of the default font.
|
|
font: asset_server.load("fonts/FiraSans-Bold.ttf"),
|
|
font_size: 50.0,
|
|
..default()
|
|
},
|
|
),
|
|
TextSection::from_style(if cfg!(feature = "default_font") {
|
|
TextStyle {
|
|
font_size: 40.0,
|
|
color: GOLD.into(),
|
|
// If no font is specified, the default font (a minimal subset of FiraMono) will be used.
|
|
..default()
|
|
}
|
|
} else {
|
|
// "default_font" feature is unavailable, load a font to use instead.
|
|
TextStyle {
|
|
font: asset_server.load("fonts/FiraMono-Medium.ttf"),
|
|
font_size: 40.0,
|
|
color: GOLD.into(),
|
|
}
|
|
}),
|
|
]),
|
|
FpsText,
|
|
));
|
|
|
|
#[cfg(feature = "default_font")]
|
|
commands.spawn(
|
|
// Here we are able to call the `From` method instead of creating a new `TextSection`.
|
|
// This will use the default font (a minimal subset of FiraMono) and apply the default styling.
|
|
TextBundle::from("From an &str into a TextBundle with the default font!").with_style(
|
|
Style {
|
|
position_type: PositionType::Absolute,
|
|
bottom: Val::Px(5.0),
|
|
left: Val::Px(15.0),
|
|
..default()
|
|
},
|
|
),
|
|
);
|
|
|
|
#[cfg(not(feature = "default_font"))]
|
|
commands.spawn(
|
|
TextBundle::from_section(
|
|
"Default font disabled",
|
|
TextStyle {
|
|
font: asset_server.load("fonts/FiraMono-Medium.ttf"),
|
|
..default()
|
|
},
|
|
)
|
|
.with_style(Style {
|
|
position_type: PositionType::Absolute,
|
|
bottom: Val::Px(5.0),
|
|
left: Val::Px(15.0),
|
|
..default()
|
|
}),
|
|
);
|
|
}
|
|
|
|
fn text_color_system(time: Res<Time>, mut query: Query<&mut Text, With<ColorText>>) {
|
|
for mut text in &mut query {
|
|
let seconds = time.elapsed_seconds();
|
|
|
|
// Update the color of the first and only section.
|
|
text.sections[0].style.color = Color::srgb(
|
|
(1.25 * seconds).sin() / 2.0 + 0.5,
|
|
(0.75 * seconds).sin() / 2.0 + 0.5,
|
|
(0.50 * seconds).sin() / 2.0 + 0.5,
|
|
);
|
|
}
|
|
}
|
|
|
|
fn text_update_system(
|
|
diagnostics: Res<DiagnosticsStore>,
|
|
mut query: Query<&mut Text, With<FpsText>>,
|
|
) {
|
|
for mut text in &mut query {
|
|
if let Some(fps) = diagnostics.get(&FrameTimeDiagnosticsPlugin::FPS) {
|
|
if let Some(value) = fps.smoothed() {
|
|
// Update the value of the second section
|
|
text.sections[1].value = format!("{value:.2}");
|
|
}
|
|
}
|
|
}
|
|
}
|