use bevy_asset::Handle; use bevy_ecs::{prelude::Component, reflect::ReflectComponent}; use bevy_reflect::{prelude::*, FromReflect}; use bevy_render::color::Color; use bevy_utils::default; use serde::{Deserialize, Serialize}; use crate::{Font, DEFAULT_FONT_HANDLE}; #[derive(Component, Debug, Clone, Reflect)] #[reflect(Component, Default)] pub struct Text { pub sections: Vec, /// The text's internal alignment. /// Should not affect its position within a container. pub alignment: TextAlignment, /// How the text should linebreak when running out of the bounds determined by max_size pub linebreak_behavior: BreakLineOn, } impl Default for Text { fn default() -> Self { Self { sections: Default::default(), alignment: TextAlignment::Left, linebreak_behavior: BreakLineOn::WordBoundary, } } } impl Text { /// Constructs a [`Text`] with a single section. /// /// ``` /// # use bevy_asset::Handle; /// # use bevy_render::color::Color; /// # use bevy_text::{Font, Text, TextStyle, TextAlignment}; /// # /// # let font_handle: Handle = Default::default(); /// # /// // Basic usage. /// let hello_world = Text::from_section( /// // Accepts a String or any type that converts into a String, such as &str. /// "hello world!", /// TextStyle { /// font: font_handle.clone(), /// font_size: 60.0, /// color: Color::WHITE, /// }, /// ); /// /// let hello_bevy = Text::from_section( /// "hello bevy!", /// TextStyle { /// font: font_handle, /// font_size: 60.0, /// color: Color::WHITE, /// }, /// ) // You can still add an alignment. /// .with_alignment(TextAlignment::Center); /// ``` pub fn from_section(value: impl Into, style: TextStyle) -> Self { Self { sections: vec![TextSection::new(value, style)], ..default() } } /// Constructs a [`Text`] from a list of sections. /// /// ``` /// # use bevy_asset::Handle; /// # use bevy_render::color::Color; /// # use bevy_text::{Font, Text, TextStyle, TextSection}; /// # /// # let font_handle: Handle = Default::default(); /// # /// let hello_world = Text::from_sections([ /// TextSection::new( /// "Hello, ", /// TextStyle { /// font: font_handle.clone(), /// font_size: 60.0, /// color: Color::BLUE, /// }, /// ), /// TextSection::new( /// "World!", /// TextStyle { /// font: font_handle, /// font_size: 60.0, /// color: Color::RED, /// }, /// ), /// ]); /// ``` pub fn from_sections(sections: impl IntoIterator) -> Self { Self { sections: sections.into_iter().collect(), ..default() } } /// Returns this [`Text`] with a new [`TextAlignment`]. pub const fn with_alignment(mut self, alignment: TextAlignment) -> Self { self.alignment = alignment; self } /// Returns this [`Text`] with soft wrapping disabled. /// Hard wrapping, where text contains an explicit linebreak such as the escape sequence `\n`, will still occur. pub const fn with_no_wrap(mut self) -> Self { self.linebreak_behavior = BreakLineOn::NoWrap; self } } #[derive(Debug, Default, Clone, FromReflect, Reflect)] pub struct TextSection { pub value: String, pub style: TextStyle, } impl TextSection { /// Create a new [`TextSection`]. pub fn new(value: impl Into, style: TextStyle) -> Self { Self { value: value.into(), style, } } /// Create an empty [`TextSection`] from a style. Useful when the value will be set dynamically. pub const fn from_style(style: TextStyle) -> Self { Self { value: String::new(), style, } } } /// Describes horizontal alignment preference for positioning & bounds. #[derive(Debug, Clone, Copy, PartialEq, Eq, Hash, Reflect, Serialize, Deserialize)] #[reflect(Serialize, Deserialize)] pub enum TextAlignment { /// Leftmost character is immediately to the right of the render position.
/// Bounds start from the render position and advance rightwards. Left, /// Leftmost & rightmost characters are equidistant to the render position.
/// Bounds start from the render position and advance equally left & right. Center, /// Rightmost character is immediately to the left of the render position.
/// Bounds start from the render position and advance leftwards. Right, } impl From for glyph_brush_layout::HorizontalAlign { fn from(val: TextAlignment) -> Self { match val { TextAlignment::Left => glyph_brush_layout::HorizontalAlign::Left, TextAlignment::Center => glyph_brush_layout::HorizontalAlign::Center, TextAlignment::Right => glyph_brush_layout::HorizontalAlign::Right, } } } #[derive(Clone, Debug, Reflect, FromReflect)] pub struct TextStyle { pub font: Handle, pub font_size: f32, pub color: Color, } impl Default for TextStyle { fn default() -> Self { Self { font: DEFAULT_FONT_HANDLE.typed(), font_size: 12.0, color: Color::WHITE, } } } /// Determines how lines will be broken when preventing text from running out of bounds. #[derive(Debug, Clone, Copy, PartialEq, Eq, Hash, Reflect, Serialize, Deserialize)] #[reflect(Serialize, Deserialize)] pub enum BreakLineOn { /// Uses the [Unicode Line Breaking Algorithm](https://www.unicode.org/reports/tr14/). /// Lines will be broken up at the nearest suitable word boundary, usually a space. /// This behavior suits most cases, as it keeps words intact across linebreaks. WordBoundary, /// Lines will be broken without discrimination on any character that would leave bounds. /// This is closer to the behavior one might expect from text in a terminal. /// However it may lead to words being broken up across linebreaks. AnyCharacter, /// No soft wrapping, where text is automatically broken up into separate lines when it overflows a boundary, will ever occur. /// Hard wrapping, where text contains an explicit linebreak such as the escape sequence `\n`, is still enabled. NoWrap, } impl From for glyph_brush_layout::BuiltInLineBreaker { fn from(val: BreakLineOn) -> Self { match val { // If `NoWrap` is set the choice of `BuiltInLineBreaker` doesn't matter as the text is given unbounded width and soft wrapping will never occur. // But `NoWrap` does not disable hard breaks where a [`Text`] contains a newline character. BreakLineOn::WordBoundary | BreakLineOn::NoWrap => { glyph_brush_layout::BuiltInLineBreaker::UnicodeLineBreaker } BreakLineOn::AnyCharacter => glyph_brush_layout::BuiltInLineBreaker::AnyCharLineBreaker, } } }