use alloc::sync::Arc; use bevy_asset::{AssetId, Assets}; use bevy_ecs::{ component::Component, entity::Entity, reflect::ReflectComponent, system::{ResMut, Resource}, }; use bevy_math::{UVec2, Vec2}; use bevy_reflect::{std_traits::ReflectDefault, Reflect}; use bevy_render::texture::Image; use bevy_sprite::TextureAtlasLayout; use bevy_utils::HashMap; use cosmic_text::{Attrs, Buffer, Family, Metrics, Shaping, Wrap}; use crate::{ error::TextError, ComputedTextBlock, Font, FontAtlasSets, FontSmoothing, JustifyText, LineBreak, PositionedGlyph, TextBounds, TextEntity, TextLayout, TextStyle, YAxisOrientation, }; /// A wrapper resource around a [`cosmic_text::FontSystem`] /// /// The font system is used to retrieve fonts and their information, including glyph outlines. /// /// This resource is updated by the [`TextPipeline`] resource. #[derive(Resource)] pub struct CosmicFontSystem(pub cosmic_text::FontSystem); impl Default for CosmicFontSystem { fn default() -> Self { let locale = sys_locale::get_locale().unwrap_or_else(|| String::from("en-US")); let db = cosmic_text::fontdb::Database::new(); // TODO: consider using `cosmic_text::FontSystem::new()` (load system fonts by default) Self(cosmic_text::FontSystem::new_with_locale_and_db(locale, db)) } } /// A wrapper resource around a [`cosmic_text::SwashCache`] /// /// The swash cache rasterizer is used to rasterize glyphs /// /// This resource is updated by the [`TextPipeline`] resource. #[derive(Resource)] pub struct SwashCache(pub cosmic_text::SwashCache); impl Default for SwashCache { fn default() -> Self { Self(cosmic_text::SwashCache::new()) } } /// Information about a font collected as part of preparing for text layout. #[derive(Clone)] struct FontFaceInfo { stretch: cosmic_text::fontdb::Stretch, style: cosmic_text::fontdb::Style, weight: cosmic_text::fontdb::Weight, family_name: Arc, } /// The `TextPipeline` is used to layout and render text blocks (see `Text`/[`Text2d`](crate::Text2d)). /// /// See the [crate-level documentation](crate) for more information. #[derive(Default, Resource)] pub struct TextPipeline { /// Identifies a font [`ID`](cosmic_text::fontdb::ID) by its [`Font`] [`Asset`](bevy_asset::Asset). map_handle_to_font_id: HashMap, (cosmic_text::fontdb::ID, Arc)>, /// Buffered vec for collecting spans. /// /// See [this dark magic](https://users.rust-lang.org/t/how-to-cache-a-vectors-capacity/94478/10). spans_buffer: Vec<(usize, &'static str, &'static TextStyle, FontFaceInfo)>, /// Buffered vec for collecting info for glyph assembly. glyph_info: Vec<(AssetId, FontSmoothing)>, } impl TextPipeline { /// Utilizes [`cosmic_text::Buffer`] to shape and layout text /// /// Negative or 0.0 font sizes will not be laid out. #[allow(clippy::too_many_arguments)] pub fn update_buffer<'a>( &mut self, fonts: &Assets, text_spans: impl Iterator, linebreak: LineBreak, justify: JustifyText, bounds: TextBounds, scale_factor: f64, computed: &mut ComputedTextBlock, font_system: &mut CosmicFontSystem, ) -> Result<(), TextError> { let font_system = &mut font_system.0; // Collect span information into a vec. This is necessary because font loading requires mut access // to FontSystem, which the cosmic-text Buffer also needs. let mut font_size: f32 = 0.; let mut spans: Vec<(usize, &str, &TextStyle, FontFaceInfo)> = core::mem::take(&mut self.spans_buffer) .into_iter() .map(|_| -> (usize, &str, &TextStyle, FontFaceInfo) { unreachable!() }) .collect(); computed.entities.clear(); for (span_index, (entity, depth, span, style)) in text_spans.enumerate() { // Return early if a font is not loaded yet. if !fonts.contains(style.font.id()) { spans.clear(); self.spans_buffer = spans .into_iter() .map( |_| -> (usize, &'static str, &'static TextStyle, FontFaceInfo) { unreachable!() }, ) .collect(); return Err(TextError::NoSuchFont); } // Save this span entity in the computed text block. computed.entities.push(TextEntity { entity, depth }); // Get max font size for use in cosmic Metrics. font_size = font_size.max(style.font_size); // Load Bevy fonts into cosmic-text's font system. let face_info = load_font_to_fontdb(style, font_system, &mut self.map_handle_to_font_id, fonts); // Save spans that aren't zero-sized. if scale_factor <= 0.0 || style.font_size <= 0.0 { continue; } spans.push((span_index, span, style, face_info)); } let line_height = font_size * 1.2; let mut metrics = Metrics::new(font_size, line_height).scale(scale_factor as f32); // Metrics of 0.0 cause `Buffer::set_metrics` to panic. We hack around this by 'falling // through' to call `Buffer::set_rich_text` with zero spans so any cached text will be cleared without // deallocating the buffer. metrics.font_size = metrics.font_size.max(0.000001); metrics.line_height = metrics.line_height.max(0.000001); // Map text sections to cosmic-text spans, and ignore sections with negative or zero fontsizes, // since they cannot be rendered by cosmic-text. // // The section index is stored in the metadata of the spans, and could be used // to look up the section the span came from and is not used internally // in cosmic-text. let spans_iter = spans.iter().map(|(span_index, span, style, font_info)| { ( *span, get_attrs(*span_index, style, font_info, scale_factor), ) }); // Update the buffer. let buffer = &mut computed.buffer; buffer.set_metrics(font_system, metrics); buffer.set_size(font_system, bounds.width, bounds.height); buffer.set_wrap( font_system, match linebreak { LineBreak::WordBoundary => Wrap::Word, LineBreak::AnyCharacter => Wrap::Glyph, LineBreak::WordOrCharacter => Wrap::WordOrGlyph, LineBreak::NoWrap => Wrap::None, }, ); buffer.set_rich_text(font_system, spans_iter, Attrs::new(), Shaping::Advanced); // PERF: https://github.com/pop-os/cosmic-text/issues/166: // Setting alignment afterwards appears to invalidate some layouting performed by `set_text` which is presumably not free? for buffer_line in buffer.lines.iter_mut() { buffer_line.set_align(Some(justify.into())); } buffer.shape_until_scroll(font_system, false); // Recover the spans buffer. spans.clear(); self.spans_buffer = spans .into_iter() .map(|_| -> (usize, &'static str, &'static TextStyle, FontFaceInfo) { unreachable!() }) .collect(); Ok(()) } /// Queues text for rendering /// /// Produces a [`TextLayoutInfo`], containing [`PositionedGlyph`]s /// which contain information for rendering the text. #[allow(clippy::too_many_arguments)] pub fn queue_text<'a>( &mut self, layout_info: &mut TextLayoutInfo, fonts: &Assets, text_spans: impl Iterator, scale_factor: f64, layout: &TextLayout, bounds: TextBounds, font_atlas_sets: &mut FontAtlasSets, texture_atlases: &mut Assets, textures: &mut Assets, y_axis_orientation: YAxisOrientation, computed: &mut ComputedTextBlock, font_system: &mut CosmicFontSystem, swash_cache: &mut SwashCache, ) -> Result<(), TextError> { layout_info.glyphs.clear(); layout_info.size = Default::default(); // Clear this here at the focal point of text rendering to ensure the field's lifecycle has strong boundaries. computed.needs_rerender = false; // Extract font ids from the iterator while traversing it. let mut glyph_info = core::mem::take(&mut self.glyph_info); glyph_info.clear(); let text_spans = text_spans.inspect(|(_, _, _, style)| { glyph_info.push((style.font.id(), style.font_smoothing)); }); let update_result = self.update_buffer( fonts, text_spans, layout.linebreak, layout.justify, bounds, scale_factor, computed, font_system, ); if let Err(err) = update_result { self.glyph_info = glyph_info; return Err(err); } let buffer = &mut computed.buffer; let box_size = buffer_dimensions(buffer); let result = buffer .layout_runs() .flat_map(|run| { run.glyphs .iter() .map(move |layout_glyph| (layout_glyph, run.line_y)) }) .try_for_each(|(layout_glyph, line_y)| { let mut temp_glyph; let span_index = layout_glyph.metadata; let font_id = glyph_info[span_index].0; let font_smoothing = glyph_info[span_index].1; let layout_glyph = if font_smoothing == FontSmoothing::None { // If font smoothing is disabled, round the glyph positions and sizes, // effectively discarding all subpixel layout. temp_glyph = layout_glyph.clone(); temp_glyph.x = temp_glyph.x.round(); temp_glyph.y = temp_glyph.y.round(); temp_glyph.w = temp_glyph.w.round(); temp_glyph.x_offset = temp_glyph.x_offset.round(); temp_glyph.y_offset = temp_glyph.y_offset.round(); temp_glyph.line_height_opt = temp_glyph.line_height_opt.map(f32::round); &temp_glyph } else { layout_glyph }; let font_atlas_set = font_atlas_sets.sets.entry(font_id).or_default(); let physical_glyph = layout_glyph.physical((0., 0.), 1.); let atlas_info = font_atlas_set .get_glyph_atlas_info(physical_glyph.cache_key, font_smoothing) .map(Ok) .unwrap_or_else(|| { font_atlas_set.add_glyph_to_atlas( texture_atlases, textures, &mut font_system.0, &mut swash_cache.0, layout_glyph, font_smoothing, ) })?; let texture_atlas = texture_atlases.get(&atlas_info.texture_atlas).unwrap(); let location = atlas_info.location; let glyph_rect = texture_atlas.textures[location.glyph_index]; let left = location.offset.x as f32; let top = location.offset.y as f32; let glyph_size = UVec2::new(glyph_rect.width(), glyph_rect.height()); // offset by half the size because the origin is center let x = glyph_size.x as f32 / 2.0 + left + physical_glyph.x as f32; let y = line_y.round() + physical_glyph.y as f32 - top + glyph_size.y as f32 / 2.0; let y = match y_axis_orientation { YAxisOrientation::TopToBottom => y, YAxisOrientation::BottomToTop => box_size.y - y, }; let position = Vec2::new(x, y); // TODO: recreate the byte index, that keeps track of where a cursor is, // when glyphs are not limited to single byte representation, relevant for #1319 let pos_glyph = PositionedGlyph::new(position, glyph_size.as_vec2(), atlas_info, span_index); layout_info.glyphs.push(pos_glyph); Ok(()) }); // Return the scratch vec. self.glyph_info = glyph_info; // Check result. result?; layout_info.size = box_size; Ok(()) } /// Queues text for measurement /// /// Produces a [`TextMeasureInfo`] which can be used by a layout system /// to measure the text area on demand. #[allow(clippy::too_many_arguments)] pub fn create_text_measure<'a>( &mut self, entity: Entity, fonts: &Assets, text_spans: impl Iterator, scale_factor: f64, layout: &TextLayout, computed: &mut ComputedTextBlock, font_system: &mut CosmicFontSystem, ) -> Result { const MIN_WIDTH_CONTENT_BOUNDS: TextBounds = TextBounds::new_horizontal(0.0); // Clear this here at the focal point of measured text rendering to ensure the field's lifecycle has // strong boundaries. computed.needs_rerender = false; self.update_buffer( fonts, text_spans, layout.linebreak, layout.justify, MIN_WIDTH_CONTENT_BOUNDS, scale_factor, computed, font_system, )?; let buffer = &mut computed.buffer; let min_width_content_size = buffer_dimensions(buffer); let max_width_content_size = { let font_system = &mut font_system.0; buffer.set_size(font_system, None, None); buffer_dimensions(buffer) }; Ok(TextMeasureInfo { min: min_width_content_size, max: max_width_content_size, entity, }) } /// Returns the [`cosmic_text::fontdb::ID`] for a given [`Font`] asset. pub fn get_font_id(&self, asset_id: AssetId) -> Option { self.map_handle_to_font_id .get(&asset_id) .cloned() .map(|(id, _)| id) } } /// Render information for a corresponding text block. /// /// Contains scaled glyphs and their size. Generated via [`TextPipeline::queue_text`] when an entity has /// [`TextLayout`] and [`ComputedTextBlock`] components. #[derive(Component, Clone, Default, Debug, Reflect)] #[reflect(Component, Default, Debug)] pub struct TextLayoutInfo { /// Scaled and positioned glyphs in screenspace pub glyphs: Vec, /// The glyphs resulting size pub size: Vec2, } /// Size information for a corresponding [`ComputedTextBlock`] component. /// /// Generated via [`TextPipeline::create_text_measure`]. #[derive(Debug)] pub struct TextMeasureInfo { /// Minimum size for a text area in pixels, to be used when laying out widgets with taffy pub min: Vec2, /// Maximum size for a text area in pixels, to be used when laying out widgets with taffy pub max: Vec2, /// The entity that is measured. pub entity: Entity, } impl TextMeasureInfo { /// Computes the size of the text area within the provided bounds. pub fn compute_size( &mut self, bounds: TextBounds, computed: &mut ComputedTextBlock, font_system: &mut cosmic_text::FontSystem, ) -> Vec2 { // Note that this arbitrarily adjusts the buffer layout. We assume the buffer is always 'refreshed' // whenever a canonical state is required. computed .buffer .set_size(font_system, bounds.width, bounds.height); buffer_dimensions(&computed.buffer) } } fn load_font_to_fontdb( style: &TextStyle, font_system: &mut cosmic_text::FontSystem, map_handle_to_font_id: &mut HashMap, (cosmic_text::fontdb::ID, Arc)>, fonts: &Assets, ) -> FontFaceInfo { let font_handle = style.font.clone(); let (face_id, family_name) = map_handle_to_font_id .entry(font_handle.id()) .or_insert_with(|| { let font = fonts.get(font_handle.id()).expect( "Tried getting a font that was not available, probably due to not being loaded yet", ); let data = Arc::clone(&font.data); let ids = font_system .db_mut() .load_font_source(cosmic_text::fontdb::Source::Binary(data)); // TODO: it is assumed this is the right font face let face_id = *ids.last().unwrap(); let face = font_system.db().face(face_id).unwrap(); let family_name = Arc::from(face.families[0].0.as_str()); (face_id, family_name) }); let face = font_system.db().face(*face_id).unwrap(); FontFaceInfo { stretch: face.stretch, style: face.style, weight: face.weight, family_name: family_name.clone(), } } /// Translates [`TextStyle`] to [`Attrs`]. fn get_attrs<'a>( span_index: usize, style: &TextStyle, face_info: &'a FontFaceInfo, scale_factor: f64, ) -> Attrs<'a> { let attrs = Attrs::new() .metadata(span_index) .family(Family::Name(&face_info.family_name)) .stretch(face_info.stretch) .style(face_info.style) .weight(face_info.weight) .metrics(Metrics::relative(style.font_size, 1.2).scale(scale_factor as f32)) .color(cosmic_text::Color(style.color.to_linear().as_u32())); attrs } /// Calculate the size of the text area for the given buffer. fn buffer_dimensions(buffer: &Buffer) -> Vec2 { let (width, height) = buffer .layout_runs() .map(|run| (run.line_w, run.line_height)) .reduce(|(w1, h1), (w2, h2)| (w1.max(w2), h1 + h2)) .unwrap_or((0.0, 0.0)); Vec2::new(width, height).ceil() } /// Discards stale data cached in `FontSystem`. pub(crate) fn trim_cosmic_cache(mut font_system: ResMut) { // A trim age of 2 was found to reduce frame time variance vs age of 1 when tested with dynamic text. // See https://github.com/bevyengine/bevy/pull/15037 // // We assume only text updated frequently benefits from the shape cache (e.g. animated text, or // text that is dynamically measured for UI). font_system.0.shape_run_cache.trim(2); }