use std::sync::Arc;
use fontdue::Font;
use crate::theme::FontStyle;
use crate::types::{ Color, Length, Rect };
use crate::render::Canvas;
use super::Element;
#[ cfg( test ) ]
mod tests;
#[ derive( Debug, Clone, Copy, PartialEq ) ]
pub enum TextAlign
{
Left,
Center,
Right,
}
pub struct Text
{
pub content: String,
pub size: Length,
pub color: Color,
pub align: TextAlign,
pub wrap: bool,
pub truncate: bool,
pub font: Option<( String, u16, FontStyle )>,
}
impl Text
{
pub fn new( content: impl Into<String> ) -> Self
{
Self
{
content: content.into(),
size: Length::px( 16.0 ),
color: Color::WHITE,
align: TextAlign::Left,
wrap: false,
truncate: true,
font: None,
}
}
#[ inline ]
fn resolved_size( &self, canvas: &Canvas ) -> f32
{
self.size.resolve( canvas.viewport_logical(), Length::EM_BASE_DEFAULT )
}
pub fn no_truncate( mut self ) -> Self
{
self.truncate = false;
self
}
pub fn font( mut self, family: impl Into<String>, weight: u16, style: FontStyle ) -> Self
{
self.font = Some( ( family.into(), weight, style ) );
self
}
pub fn weight( mut self, weight: u16 ) -> Self
{
self.font = Some( ( "sora".to_string(), weight, FontStyle::Normal ) );
self
}
pub fn size( mut self, s: impl Into<Length> ) -> Self
{
self.size = s.into();
self
}
pub fn color( mut self, c: Color ) -> Self
{
self.color = c;
self
}
pub fn align( mut self, a: TextAlign ) -> Self
{
self.align = a;
self
}
pub fn align_center( mut self ) -> Self
{
self.align = TextAlign::Center;
self
}
pub fn wrap( mut self, w: bool ) -> Self
{
self.wrap = w;
self
}
fn resolve_font( &self, canvas: &Canvas ) -> Option<Arc<Font>>
{
self.font.as_ref().map( |( family, weight, style )|
{
canvas.font_for( family, *weight, *style )
} )
}
fn measure( &self, text: &str, canvas: &Canvas, font: Option<&Arc<Font>> ) -> f32
{
let size = self.resolved_size( canvas );
match font
{
Some( f ) => canvas.measure_text_with_font( text, size, f ),
None => canvas.measure_text( text, size ),
}
}
fn measure_char( &self, ch: char, canvas: &Canvas, font: Option<&Arc<Font>> ) -> f32
{
let size = self.resolved_size( canvas );
match font
{
Some( f ) => f.metrics( ch, size * canvas.dpi_scale() ).advance_width,
None => canvas.font_metrics( ch, size ).advance_width,
}
}
fn paint( &self, canvas: &mut Canvas, text: &str, x: f32, y: f32, font: Option<&Arc<Font>> )
{
let size = self.resolved_size( canvas );
match font
{
Some( f ) => canvas.draw_text_with_font( text, x, y, size, self.color, f ),
None => canvas.draw_text( text, x, y, size, self.color ),
}
}
pub fn preferred_size( &self, max_width: f32, canvas: &Canvas ) -> ( f32, f32 )
{
let size = self.resolved_size( canvas );
let line_h = canvas.font_line_metrics( size )
.map( |m| m.new_line_size )
.unwrap_or( size );
let font = self.resolve_font( canvas );
if self.wrap
{
let lines = wrap_lines( &self.content, size, max_width, canvas, font.as_ref() );
let h = line_h * lines.len().max( 1 ) as f32;
( max_width, h )
}
else
{
let w = ( self.measure( &self.content, canvas, font.as_ref() ) + 8.0 ).min( max_width );
( w, line_h )
}
}
pub fn draw( &self, canvas: &mut Canvas, rect: Rect, _focused: bool )
{
let size = self.resolved_size( canvas );
let ascent = canvas.font_line_metrics( size )
.map( |m| m.ascent )
.unwrap_or( size * 0.8 );
let line_h = canvas.font_line_metrics( size )
.map( |m| m.new_line_size )
.unwrap_or( size );
let font = self.resolve_font( canvas );
if self.wrap
{
let lines = wrap_lines( &self.content, size, rect.width, canvas, font.as_ref() );
for ( i, line ) in lines.iter().enumerate()
{
let line_w = self.measure( line, canvas, font.as_ref() );
let slack = ( rect.width - line_w ).max( 0.0 );
let pad = 4.0_f32.min( slack );
let tx = match self.align
{
TextAlign::Left => rect.x + pad,
TextAlign::Center => rect.x + slack / 2.0,
TextAlign::Right => rect.x + rect.width - line_w - pad,
};
let ty = rect.y + ascent + line_h * i as f32;
self.paint( canvas, line, tx, ty, font.as_ref() );
}
return;
}
let text_w = self.measure( &self.content, canvas, font.as_ref() );
let display = if self.truncate && text_w > rect.width && rect.width > 0.0
{
let ellipsis = "...";
let ell_w = self.measure( ellipsis, canvas, font.as_ref() );
let budget = rect.width - ell_w;
if budget <= 0.0
{
ellipsis.to_string()
}
else
{
let mut accum = 0.0_f32;
let truncated: String = self.content.chars().take_while( |ch|
{
let cw = self.measure_char( *ch, canvas, font.as_ref() );
accum += cw;
accum <= budget
} ).collect();
format!( "{truncated}{ellipsis}" )
}
}
else
{
self.content.clone()
};
let disp_w = self.measure( &display, canvas, font.as_ref() );
let slack = ( rect.width - disp_w ).max( 0.0 );
let pad = 4.0_f32.min( slack );
let tx = match self.align
{
TextAlign::Left => rect.x + pad,
TextAlign::Center => rect.x + slack / 2.0,
TextAlign::Right => rect.x + rect.width - disp_w - pad,
};
let ty = rect.y + ascent;
self.paint( canvas, &display, tx, ty, font.as_ref() );
}
}
fn wrap_lines( text: &str, size: f32, max_width: f32, canvas: &Canvas, font: Option<&Arc<Font>> ) -> Vec<String>
{
if max_width <= 0.0 || text.is_empty()
{
return vec![ text.to_string() ];
}
let measure = |s: &str| -> f32
{
match font
{
Some( f ) => canvas.measure_text_with_font( s, size, f ),
None => canvas.measure_text( s, size ),
}
};
let space_w = measure( " " );
let mut lines = Vec::new();
let mut current = String::new();
let mut current_w = 0.0_f32;
for word in text.split_whitespace()
{
let word_w = measure( word );
if current.is_empty()
{
current.push_str( word );
current_w = word_w;
}
else if current_w + space_w + word_w <= max_width
{
current.push( ' ' );
current.push_str( word );
current_w += space_w + word_w;
}
else
{
lines.push( std::mem::take( &mut current ) );
current.push_str( word );
current_w = word_w;
}
}
if !current.is_empty() { lines.push( current ); }
if lines.is_empty() { lines.push( String::new() ); }
lines
}
impl<Msg: Clone + 'static> From<Text> for Element<Msg>
{
fn from( t: Text ) -> Self
{
Element::Text( t )
}
}