//! Draw and interact with text. pub mod editor; pub mod highlighter; pub mod paragraph; pub use editor::Editor; pub use highlighter::Highlighter; pub use paragraph::Paragraph; use crate::alignment; use crate::{ Background, Border, Color, Padding, Pixels, Point, Rectangle, Size, }; use std::borrow::Cow; use std::hash::{Hash, Hasher}; /// A paragraph. #[derive(Debug, Clone)] pub struct Text { /// The content of the paragraph. pub content: Content, /// The bounds of the paragraph. pub bounds: Size, /// The size of the [`Text`] in logical pixels. pub size: Pixels, /// The line height of the [`Text`]. pub line_height: LineHeight, /// The font of the [`Text`]. pub font: Font, /// The horizontal alignment of the [`Text`]. pub horizontal_alignment: alignment::Horizontal, /// The vertical alignment of the [`Text`]. pub vertical_alignment: alignment::Vertical, /// The [`Shaping`] strategy of the [`Text`]. pub shaping: Shaping, /// The [`Wrapping`] strategy of the [`Text`]. pub wrapping: Wrapping, } /// The shaping strategy of some text. #[derive(Debug, Clone, Copy, PartialEq, Eq, Hash, Default)] pub enum Shaping { /// No shaping and no font fallback. /// /// This shaping strategy is very cheap, but it will not display complex /// scripts properly nor try to find missing glyphs in your system fonts. /// /// You should use this strategy when you have complete control of the text /// and the font you are displaying in your application. /// /// This is the default. #[default] Basic, /// Advanced text shaping and font fallback. /// /// You will need to enable this flag if the text contains a complex /// script, the font used needs it, and/or multiple fonts in your system /// may be needed to display all of the glyphs. /// /// Advanced shaping is expensive! You should only enable it when necessary. Advanced, } /// The wrapping strategy of some text. #[derive(Debug, Clone, Copy, PartialEq, Eq, Hash, Default)] pub enum Wrapping { /// No wrapping. None, /// Wraps at the word level. /// /// This is the default. #[default] Word, /// Wraps at the glyph level. Glyph, /// Wraps at the word level, or fallback to glyph level if a word can't fit on a line by itself. WordOrGlyph, } /// The height of a line of text in a paragraph. #[derive(Debug, Clone, Copy, PartialEq)] pub enum LineHeight { /// A factor of the size of the text. Relative(f32), /// An absolute height in logical pixels. Absolute(Pixels), } impl LineHeight { /// Returns the [`LineHeight`] in absolute logical pixels. pub fn to_absolute(self, text_size: Pixels) -> Pixels { match self { Self::Relative(factor) => Pixels(factor * text_size.0), Self::Absolute(pixels) => pixels, } } } impl Default for LineHeight { fn default() -> Self { Self::Relative(1.3) } } impl From for LineHeight { fn from(factor: f32) -> Self { Self::Relative(factor) } } impl From for LineHeight { fn from(pixels: Pixels) -> Self { Self::Absolute(pixels) } } impl Hash for LineHeight { fn hash(&self, state: &mut H) { match self { Self::Relative(factor) => { state.write_u8(0); factor.to_bits().hash(state); } Self::Absolute(pixels) => { state.write_u8(1); f32::from(*pixels).to_bits().hash(state); } } } } /// The result of hit testing on text. #[derive(Debug, Clone, Copy, PartialEq)] pub enum Hit { /// The point was within the bounds of the returned character index. CharOffset(usize), } impl Hit { /// Computes the cursor position of the [`Hit`] . pub fn cursor(self) -> usize { match self { Self::CharOffset(i) => i, } } } /// The difference detected in some text. /// /// You will obtain a [`Difference`] when you [`compare`] a [`Paragraph`] with some /// [`Text`]. /// /// [`compare`]: Paragraph::compare #[derive(Debug, Clone, Copy, PartialEq, Eq)] pub enum Difference { /// No difference. /// /// The text can be reused as it is! None, /// A bounds difference. /// /// This normally means a relayout is necessary, but the shape of the text can /// be reused. Bounds, /// A shape difference. /// /// The contents, alignment, sizes, fonts, or any other essential attributes /// of the shape of the text have changed. A complete reshape and relayout of /// the text is necessary. Shape, } /// A renderer capable of measuring and drawing [`Text`]. pub trait Renderer: crate::Renderer { /// The font type used. type Font: Copy + PartialEq; /// The [`Paragraph`] of this [`Renderer`]. type Paragraph: Paragraph + 'static; /// The [`Editor`] of this [`Renderer`]. type Editor: Editor + 'static; /// The icon font of the backend. const ICON_FONT: Self::Font; /// The `char` representing a ✔ icon in the [`ICON_FONT`]. /// /// [`ICON_FONT`]: Self::ICON_FONT const CHECKMARK_ICON: char; /// The `char` representing a ▼ icon in the built-in [`ICON_FONT`]. /// /// [`ICON_FONT`]: Self::ICON_FONT const ARROW_DOWN_ICON: char; /// Returns the default [`Self::Font`]. fn default_font(&self) -> Self::Font; /// Returns the default size of [`Text`]. fn default_size(&self) -> Pixels; /// Draws the given [`Paragraph`] at the given position and with the given /// [`Color`]. fn fill_paragraph( &mut self, text: &Self::Paragraph, position: Point, color: Color, clip_bounds: Rectangle, ); /// Draws the given [`Editor`] at the given position and with the given /// [`Color`]. fn fill_editor( &mut self, editor: &Self::Editor, position: Point, color: Color, clip_bounds: Rectangle, ); /// Draws the given [`Text`] at the given position and with the given /// [`Color`]. fn fill_text( &mut self, text: Text, position: Point, color: Color, clip_bounds: Rectangle, ); } /// A span of text. #[derive(Debug, Clone)] pub struct Span<'a, Link = (), Font = crate::Font> { /// The [`Fragment`] of text. pub text: Fragment<'a>, /// The size of the [`Span`] in [`Pixels`]. pub size: Option, /// The [`LineHeight`] of the [`Span`]. pub line_height: Option, /// The font of the [`Span`]. pub font: Option, /// The [`Color`] of the [`Span`]. pub color: Option, /// The link of the [`Span`]. pub link: Option, /// The [`Highlight`] of the [`Span`]. pub highlight: Option, /// The [`Padding`] of the [`Span`]. /// /// Currently, it only affects the bounds of the [`Highlight`]. pub padding: Padding, /// Whether the [`Span`] should be underlined or not. pub underline: bool, /// Whether the [`Span`] should be struck through or not. pub strikethrough: bool, } /// A text highlight. #[derive(Debug, Clone, PartialEq)] pub struct Highlight { /// The [`Background`] of the highlight. pub background: Background, /// The [`Border`] of the highlight. pub border: Border, } impl<'a, Link, Font> Span<'a, Link, Font> { /// Creates a new [`Span`] of text with the given text fragment. pub fn new(fragment: impl IntoFragment<'a>) -> Self { Self { text: fragment.into_fragment(), ..Self::default() } } /// Sets the size of the [`Span`]. pub fn size(mut self, size: impl Into) -> Self { self.size = Some(size.into()); self } /// Sets the [`LineHeight`] of the [`Span`]. pub fn line_height(mut self, line_height: impl Into) -> Self { self.line_height = Some(line_height.into()); self } /// Sets the font of the [`Span`]. pub fn font(mut self, font: impl Into) -> Self { self.font = Some(font.into()); self } /// Sets the font of the [`Span`], if any. pub fn font_maybe(mut self, font: Option>) -> Self { self.font = font.map(Into::into); self } /// Sets the [`Color`] of the [`Span`]. pub fn color(mut self, color: impl Into) -> Self { self.color = Some(color.into()); self } /// Sets the [`Color`] of the [`Span`], if any. pub fn color_maybe(mut self, color: Option>) -> Self { self.color = color.map(Into::into); self } /// Sets the link of the [`Span`]. pub fn link(mut self, link: impl Into) -> Self { self.link = Some(link.into()); self } /// Sets the link of the [`Span`], if any. pub fn link_maybe(mut self, link: Option>) -> Self { self.link = link.map(Into::into); self } /// Sets the [`Background`] of the [`Span`]. pub fn background(self, background: impl Into) -> Self { self.background_maybe(Some(background)) } /// Sets the [`Background`] of the [`Span`], if any. pub fn background_maybe( mut self, background: Option>, ) -> Self { let Some(background) = background else { return self; }; match &mut self.highlight { Some(highlight) => { highlight.background = background.into(); } None => { self.highlight = Some(Highlight { background: background.into(), border: Border::default(), }); } } self } /// Sets the [`Border`] of the [`Span`]. pub fn border(self, border: impl Into) -> Self { self.border_maybe(Some(border)) } /// Sets the [`Border`] of the [`Span`], if any. pub fn border_maybe(mut self, border: Option>) -> Self { let Some(border) = border else { return self; }; match &mut self.highlight { Some(highlight) => { highlight.border = border.into(); } None => { self.highlight = Some(Highlight { border: border.into(), background: Background::Color(Color::TRANSPARENT), }); } } self } /// Sets the [`Padding`] of the [`Span`]. /// /// It only affects the [`background`] and [`border`] of the /// [`Span`], currently. /// /// [`background`]: Self::background /// [`border`]: Self::border pub fn padding(mut self, padding: impl Into) -> Self { self.padding = padding.into(); self } /// Sets whether the [`Span`] should be underlined or not. pub fn underline(mut self, underline: bool) -> Self { self.underline = underline; self } /// Sets whether the [`Span`] should be struck through or not. pub fn strikethrough(mut self, strikethrough: bool) -> Self { self.strikethrough = strikethrough; self } /// Turns the [`Span`] into a static one. pub fn to_static(self) -> Span<'static, Link, Font> { Span { text: Cow::Owned(self.text.into_owned()), size: self.size, line_height: self.line_height, font: self.font, color: self.color, link: self.link, highlight: self.highlight, padding: self.padding, underline: self.underline, strikethrough: self.strikethrough, } } } impl Default for Span<'_, Link, Font> { fn default() -> Self { Self { text: Cow::default(), size: None, line_height: None, font: None, color: None, link: None, highlight: None, padding: Padding::default(), underline: false, strikethrough: false, } } } impl<'a, Link, Font> From<&'a str> for Span<'a, Link, Font> { fn from(value: &'a str) -> Self { Span::new(value) } } impl PartialEq for Span<'_, Link, Font> { fn eq(&self, other: &Self) -> bool { self.text == other.text && self.size == other.size && self.line_height == other.line_height && self.font == other.font && self.color == other.color } } /// A fragment of [`Text`]. /// /// This is just an alias to a string that may be either /// borrowed or owned. pub type Fragment<'a> = Cow<'a, str>; /// A trait for converting a value to some text [`Fragment`]. pub trait IntoFragment<'a> { /// Converts the value to some text [`Fragment`]. fn into_fragment(self) -> Fragment<'a>; } impl<'a> IntoFragment<'a> for Fragment<'a> { fn into_fragment(self) -> Fragment<'a> { self } } impl<'a> IntoFragment<'a> for &'a Fragment<'_> { fn into_fragment(self) -> Fragment<'a> { Fragment::Borrowed(self) } } impl<'a> IntoFragment<'a> for &'a str { fn into_fragment(self) -> Fragment<'a> { Fragment::Borrowed(self) } } impl<'a> IntoFragment<'a> for &'a String { fn into_fragment(self) -> Fragment<'a> { Fragment::Borrowed(self.as_str()) } } impl<'a> IntoFragment<'a> for String { fn into_fragment(self) -> Fragment<'a> { Fragment::Owned(self) } } macro_rules! into_fragment { ($type:ty) => { impl<'a> IntoFragment<'a> for $type { fn into_fragment(self) -> Fragment<'a> { Fragment::Owned(self.to_string()) } } impl<'a> IntoFragment<'a> for &$type { fn into_fragment(self) -> Fragment<'a> { Fragment::Owned(self.to_string()) } } }; } into_fragment!(char); into_fragment!(bool); into_fragment!(u8); into_fragment!(u16); into_fragment!(u32); into_fragment!(u64); into_fragment!(u128); into_fragment!(usize); into_fragment!(i8); into_fragment!(i16); into_fragment!(i32); into_fragment!(i64); into_fragment!(i128); into_fragment!(isize); into_fragment!(f32); into_fragment!(f64);