use anyhow::{Context as _, Ok, Result};
use cosmic_text::{
Attrs, AttrsList, Ellipsize, Family, Font as CosmicTextFont,
FontFeatures as CosmicFontFeatures, FontSystem, ShapeBuffer, ShapeLine,
};
use open_gpui::{
Bounds, DevicePixels, Font, FontFallbacks, FontFeatures, FontId, FontMetrics, FontRun, GlyphId,
LineLayout, Pixels, PlatformTextSystem, RenderGlyphParams, SUBPIXEL_VARIANTS_X,
SUBPIXEL_VARIANTS_Y, ShapedGlyph, ShapedRun, SharedString, Size, TextRenderingMode, point,
size,
};
use open_gpui_collections::HashMap;
use itertools::Itertools;
use parking_lot::RwLock;
use smallvec::SmallVec;
use std::{borrow::Cow, sync::Arc};
use swash::{
scale::{Render, ScaleContext, Source, StrikeWith},
zeno::{Format, Vector},
};
use unicode_segmentation::UnicodeSegmentation;
pub struct CosmicTextSystem(RwLock<CosmicTextSystemState>);
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
struct FontKey {
family: SharedString,
features: FontFeatures,
fallbacks: Option<FontFallbacks>,
}
impl FontKey {
fn new(family: SharedString, features: FontFeatures, fallbacks: Option<FontFallbacks>) -> Self {
Self {
family,
features,
fallbacks,
}
}
}
struct CosmicTextSystemState {
font_system: FontSystem,
scratch: ShapeBuffer,
swash_scale_context: ScaleContext,
loaded_fonts: Vec<LoadedFont>,
font_ids_by_family_cache: HashMap<FontKey, SmallVec<[FontId; 4]>>,
system_font_fallback: String,
}
struct LoadedFont {
font: Arc<CosmicTextFont>,
features: CosmicFontFeatures,
is_known_emoji_font: bool,
user_fallback_chain: Arc<[(FontId, SharedString)]>,
}
impl CosmicTextSystem {
pub fn new(system_font_fallback: &str) -> Self {
let font_system = FontSystem::new();
Self(RwLock::new(CosmicTextSystemState {
font_system,
scratch: ShapeBuffer::default(),
swash_scale_context: ScaleContext::new(),
loaded_fonts: Vec::new(),
font_ids_by_family_cache: HashMap::default(),
system_font_fallback: system_font_fallback.to_string(),
}))
}
pub fn new_without_system_fonts(system_font_fallback: &str) -> Self {
let font_system = FontSystem::new_with_locale_and_db(
"en-US".to_string(),
cosmic_text::fontdb::Database::new(),
);
Self(RwLock::new(CosmicTextSystemState {
font_system,
scratch: ShapeBuffer::default(),
swash_scale_context: ScaleContext::new(),
loaded_fonts: Vec::new(),
font_ids_by_family_cache: HashMap::default(),
system_font_fallback: system_font_fallback.to_string(),
}))
}
}
impl PlatformTextSystem for CosmicTextSystem {
fn add_fonts(&self, fonts: Vec<Cow<'static, [u8]>>) -> Result<()> {
self.0.write().add_fonts(fonts)
}
fn all_font_names(&self) -> Vec<String> {
let mut result = self
.0
.read()
.font_system
.db()
.faces()
.filter_map(|face| face.families.first().map(|family| family.0.clone()))
.collect_vec();
result.sort();
result.dedup();
result
}
fn font_id(&self, font: &Font) -> Result<FontId> {
let mut state = self.0.write();
let key = FontKey::new(
font.family.clone(),
font.features.clone(),
font.fallbacks.clone(),
);
let candidates = if let Some(font_ids) = state.font_ids_by_family_cache.get(&key) {
font_ids.as_slice()
} else {
let font_ids =
state.load_family(&font.family, &font.features, font.fallbacks.as_ref())?;
state.font_ids_by_family_cache.insert(key.clone(), font_ids);
state.font_ids_by_family_cache[&key].as_ref()
};
let ix = find_best_match(font, candidates, &state)?;
Ok(candidates[ix])
}
fn font_metrics(&self, font_id: FontId) -> FontMetrics {
let metrics = self
.0
.read()
.loaded_font(font_id)
.font
.as_swash()
.metrics(&[]);
FontMetrics {
units_per_em: metrics.units_per_em as u32,
ascent: metrics.ascent,
descent: -metrics.descent,
line_gap: metrics.leading,
underline_position: metrics.underline_offset,
underline_thickness: metrics.stroke_size,
cap_height: metrics.cap_height,
x_height: metrics.x_height,
bounding_box: Bounds {
origin: point(0.0, 0.0),
size: size(metrics.max_width, metrics.ascent + metrics.descent),
},
}
}
fn typographic_bounds(&self, font_id: FontId, glyph_id: GlyphId) -> Result<Bounds<f32>> {
let lock = self.0.read();
let glyph_metrics = lock.loaded_font(font_id).font.as_swash().glyph_metrics(&[]);
let glyph_id = glyph_id.0 as u16;
Ok(Bounds {
origin: point(0.0, 0.0),
size: size(
glyph_metrics.advance_width(glyph_id),
glyph_metrics.advance_height(glyph_id),
),
})
}
fn advance(&self, font_id: FontId, glyph_id: GlyphId) -> Result<Size<f32>> {
self.0.read().advance(font_id, glyph_id)
}
fn glyph_for_char(&self, font_id: FontId, ch: char) -> Option<GlyphId> {
self.0.read().glyph_for_char(font_id, ch)
}
fn glyph_raster_bounds(&self, params: &RenderGlyphParams) -> Result<Bounds<DevicePixels>> {
self.0.write().raster_bounds(params)
}
fn rasterize_glyph(
&self,
params: &RenderGlyphParams,
raster_bounds: Bounds<DevicePixels>,
) -> Result<(Size<DevicePixels>, Vec<u8>)> {
self.0.write().rasterize_glyph(params, raster_bounds)
}
fn layout_line(&self, text: &str, font_size: Pixels, runs: &[FontRun]) -> LineLayout {
self.0.write().layout_line(text, font_size, runs)
}
fn recommended_rendering_mode(
&self,
_font_id: FontId,
_font_size: Pixels,
) -> TextRenderingMode {
TextRenderingMode::Subpixel
}
}
impl CosmicTextSystemState {
fn loaded_font(&self, font_id: FontId) -> &LoadedFont {
&self.loaded_fonts[font_id.0]
}
#[profiling::function]
fn add_fonts(&mut self, fonts: Vec<Cow<'static, [u8]>>) -> Result<()> {
let db = self.font_system.db_mut();
for bytes in fonts {
match bytes {
Cow::Borrowed(embedded_font) => {
db.load_font_data(embedded_font.to_vec());
}
Cow::Owned(bytes) => {
db.load_font_data(bytes);
}
}
}
Ok(())
}
#[profiling::function]
fn load_family(
&mut self,
name: &str,
features: &FontFeatures,
fallbacks: Option<&FontFallbacks>,
) -> Result<SmallVec<[FontId; 4]>> {
let user_fallback_chain: Arc<[(FontId, SharedString)]> = match fallbacks {
Some(fallbacks) if !fallbacks.fallback_list().is_empty() => {
let mut chain: Vec<(FontId, SharedString)> = Vec::new();
for fallback_name in fallbacks.fallback_list() {
let fb_key = FontKey::new(
SharedString::from(fallback_name.clone()),
features.clone(),
None,
);
let fb_ids = if let Some(cached) = self.font_ids_by_family_cache.get(&fb_key) {
cached.clone()
} else {
let loaded = self.load_family(fallback_name, features, None)?;
self.font_ids_by_family_cache
.insert(fb_key.clone(), loaded.clone());
loaded
};
let Some(&fb_id) = fb_ids.first() else {
continue;
};
let db_id = self.loaded_fonts[fb_id.0].font.id();
if let Some(face) = self.font_system.db().face(db_id)
&& let Some(family) = face.families.first()
{
chain.push((fb_id, SharedString::from(family.0.clone())));
}
}
Arc::from(chain)
}
_ => Arc::from(Vec::new()),
};
let name = open_gpui::font_name_with_fallbacks(name, &self.system_font_fallback);
let families = self
.font_system
.db()
.faces()
.filter(|face| face.families.iter().any(|family| *name == family.0))
.map(|face| (face.id, face.post_script_name.clone()))
.collect::<SmallVec<[_; 4]>>();
let cosmic_features = cosmic_font_features(features)?;
let mut loaded_font_ids = SmallVec::new();
for (font_id, postscript_name) in families {
let font = self
.font_system
.get_font(font_id, cosmic_text::Weight::NORMAL)
.context("Could not load font")?;
let allowed_bad_font_names = [
"SegoeFluentIcons", "Segoe Fluent Icons",
];
if font.as_swash().charmap().map('m') == 0
&& !allowed_bad_font_names.contains(&postscript_name.as_str())
{
self.font_system.db_mut().remove_face(font.id());
continue;
};
let font_id = FontId(self.loaded_fonts.len());
loaded_font_ids.push(font_id);
self.loaded_fonts.push(LoadedFont {
font,
features: cosmic_features.clone(),
is_known_emoji_font: check_is_known_emoji_font(&postscript_name),
user_fallback_chain: Arc::clone(&user_fallback_chain),
});
}
Ok(loaded_font_ids)
}
fn advance(&self, font_id: FontId, glyph_id: GlyphId) -> Result<Size<f32>> {
let glyph_metrics = self.loaded_font(font_id).font.as_swash().glyph_metrics(&[]);
Ok(Size {
width: glyph_metrics.advance_width(glyph_id.0 as u16),
height: glyph_metrics.advance_height(glyph_id.0 as u16),
})
}
fn glyph_for_char(&self, font_id: FontId, ch: char) -> Option<GlyphId> {
let glyph_id = self.loaded_font(font_id).font.as_swash().charmap().map(ch);
if glyph_id == 0 {
None
} else {
Some(GlyphId(glyph_id.into()))
}
}
fn raster_bounds(&mut self, params: &RenderGlyphParams) -> Result<Bounds<DevicePixels>> {
let image = self.render_glyph_image(params)?;
Ok(Bounds {
origin: point(image.placement.left.into(), (-image.placement.top).into()),
size: size(image.placement.width.into(), image.placement.height.into()),
})
}
#[profiling::function]
fn rasterize_glyph(
&mut self,
params: &RenderGlyphParams,
glyph_bounds: Bounds<DevicePixels>,
) -> Result<(Size<DevicePixels>, Vec<u8>)> {
if glyph_bounds.size.width.0 == 0 || glyph_bounds.size.height.0 == 0 {
anyhow::bail!("glyph bounds are empty");
}
let mut image = self.render_glyph_image(params)?;
let bitmap_size = glyph_bounds.size;
match image.content {
swash::scale::image::Content::Color | swash::scale::image::Content::SubpixelMask => {
for pixel in image.data.chunks_exact_mut(4) {
pixel.swap(0, 2);
}
Ok((bitmap_size, image.data))
}
swash::scale::image::Content::Mask => {
if params.subpixel_rendering {
let expanded = image.data.iter().flat_map(|&a| [a, a, a, a]).collect();
Ok((bitmap_size, expanded))
} else {
Ok((bitmap_size, image.data))
}
}
}
}
fn render_glyph_image(
&mut self,
params: &RenderGlyphParams,
) -> Result<swash::scale::image::Image> {
let loaded_font = &self.loaded_fonts[params.font_id.0];
let font_ref = loaded_font.font.as_swash();
let pixel_size = f32::from(params.font_size);
let subpixel_offset = Vector::new(
params.subpixel_variant.x as f32 / SUBPIXEL_VARIANTS_X as f32 / params.scale_factor,
params.subpixel_variant.y as f32 / SUBPIXEL_VARIANTS_Y as f32 / params.scale_factor,
);
let mut scaler = self
.swash_scale_context
.builder(font_ref)
.size(pixel_size * params.scale_factor)
.hint(true)
.build();
let sources: &[Source] = if params.is_emoji {
&[
Source::ColorOutline(0),
Source::ColorBitmap(StrikeWith::BestFit),
Source::Outline,
]
} else {
&[Source::Bitmap(StrikeWith::ExactSize), Source::Outline]
};
let mut renderer = Render::new(sources);
if params.subpixel_rendering {
renderer
.format(Format::subpixel_bgra())
.offset(subpixel_offset);
} else {
renderer.format(Format::Alpha).offset(subpixel_offset);
}
let glyph_id: u16 = params.glyph_id.0.try_into()?;
renderer
.render(&mut scaler, glyph_id)
.with_context(|| format!("unable to render glyph via swash for {params:?}"))
}
fn font_id_for_cosmic_id(&mut self, id: cosmic_text::fontdb::ID) -> Result<FontId> {
if let Some(ix) = self
.loaded_fonts
.iter()
.position(|loaded_font| loaded_font.font.id() == id)
{
Ok(FontId(ix))
} else {
let font = self
.font_system
.get_font(id, cosmic_text::Weight::NORMAL)
.context("failed to get fallback font from cosmic-text font system")?;
let face = self
.font_system
.db()
.face(id)
.context("fallback font face not found in cosmic-text database")?;
let font_id = FontId(self.loaded_fonts.len());
self.loaded_fonts.push(LoadedFont {
font,
features: CosmicFontFeatures::new(),
is_known_emoji_font: check_is_known_emoji_font(&face.post_script_name),
user_fallback_chain: Arc::from(Vec::new()),
});
Ok(font_id)
}
}
#[profiling::function]
fn layout_line(&mut self, text: &str, font_size: Pixels, font_runs: &[FontRun]) -> LineLayout {
let mut attrs_list = AttrsList::new(&Attrs::new());
let mut offs = 0;
for run in font_runs {
let run_end = offs + run.len;
let loaded_font = self.loaded_font(run.font_id);
let Some(face) = self.font_system.db().face(loaded_font.font.id()) else {
log::warn!(
"font face not found in database for font_id {:?}",
run.font_id
);
offs = run_end;
continue;
};
let Some(first_family) = face.families.first() else {
log::warn!(
"font face has no family names for font_id {:?}",
run.font_id
);
offs = run_end;
continue;
};
let primary_family_name: SharedString = first_family.0.clone().into();
let primary_stretch = face.stretch;
let primary_style = face.style;
let primary_weight = face.weight;
let primary_features = loaded_font.features.clone();
let fallback_chain = Arc::clone(&loaded_font.user_fallback_chain);
let primary_attrs = Attrs::new()
.metadata(run.font_id.0)
.family(Family::Name(&primary_family_name))
.stretch(primary_stretch)
.style(primary_style)
.weight(primary_weight)
.font_features(primary_features.clone());
let fallback_attrs: SmallVec<[Attrs<'_>; 4]> = fallback_chain
.iter()
.map(|(fb_id, fb_name)| {
Attrs::new()
.metadata(fb_id.0)
.family(Family::Name(fb_name))
.stretch(primary_stretch)
.style(primary_style)
.weight(primary_weight)
.font_features(primary_features.clone())
})
.collect();
let spans = if fallback_chain.is_empty() {
let mut spans = SmallVec::<[RunSpan; 4]>::new();
spans.push(RunSpan {
start: offs,
end: run_end,
slot: None,
font_id: run.font_id,
});
spans
} else {
let loaded_fonts = &self.loaded_fonts;
let covers = |id: FontId, ch: char| charmap_covers(loaded_fonts, id, ch);
compute_run_spans(text, offs, run.len, run.font_id, &fallback_chain, &covers)
};
for span in spans {
let attrs = match span.slot {
None => &primary_attrs,
Some(ix) => &fallback_attrs[ix],
};
attrs_list.add_span(span.start..span.end, attrs);
}
offs = run_end;
}
let line = ShapeLine::new(
&mut self.font_system,
text,
&attrs_list,
cosmic_text::Shaping::Advanced,
4,
);
let mut layout_lines = Vec::with_capacity(1);
line.layout_to_buffer(
&mut self.scratch,
f32::from(font_size),
None, cosmic_text::Wrap::None,
Ellipsize::None,
None,
&mut layout_lines,
None,
cosmic_text::Hinting::Disabled,
);
let Some(layout) = layout_lines.first() else {
return LineLayout {
font_size,
width: Pixels::ZERO,
ascent: Pixels::ZERO,
descent: Pixels::ZERO,
runs: Vec::new(),
len: text.len(),
};
};
let mut runs: Vec<ShapedRun> = Vec::new();
for glyph in &layout.glyphs {
let mut font_id = FontId(glyph.metadata);
let mut loaded_font = self.loaded_font(font_id);
if loaded_font.font.id() != glyph.font_id {
match self.font_id_for_cosmic_id(glyph.font_id) {
std::result::Result::Ok(resolved_id) => {
font_id = resolved_id;
loaded_font = self.loaded_font(font_id);
}
Err(error) => {
log::warn!(
"failed to resolve cosmic font id {:?}: {error:#}",
glyph.font_id
);
continue;
}
}
}
let is_emoji = loaded_font.is_known_emoji_font;
if glyph.glyph_id == 3 && is_emoji {
continue;
}
let shaped_glyph = ShapedGlyph {
id: GlyphId(glyph.glyph_id as u32),
position: point(glyph.x.into(), glyph.y.into()),
index: glyph.start,
is_emoji,
};
if let Some(last_run) = runs
.last_mut()
.filter(|last_run| last_run.font_id == font_id)
{
last_run.glyphs.push(shaped_glyph);
} else {
runs.push(ShapedRun {
font_id,
glyphs: vec![shaped_glyph],
});
}
}
LineLayout {
font_size,
width: layout.w.into(),
ascent: layout.max_ascent.into(),
descent: layout.max_descent.into(),
runs,
len: text.len(),
}
}
}
#[cfg(feature = "font-kit")]
fn find_best_match(
font: &Font,
candidates: &[FontId],
state: &CosmicTextSystemState,
) -> Result<usize> {
let candidate_properties = candidates
.iter()
.map(|font_id| {
let database_id = state.loaded_font(*font_id).font.id();
let face_info = state
.font_system
.db()
.face(database_id)
.context("font face not found in database")?;
Ok(face_info_into_properties(face_info))
})
.collect::<Result<SmallVec<[_; 4]>>>()?;
let ix = find_best_font_kit_match(&candidate_properties, &font_into_properties(font))
.context("requested font family contains no font matching the other parameters")?;
Ok(ix)
}
#[cfg(not(feature = "font-kit"))]
fn find_best_match(
font: &Font,
candidates: &[FontId],
state: &CosmicTextSystemState,
) -> Result<usize> {
if candidates.is_empty() {
anyhow::bail!("requested font family contains no font matching the other parameters");
}
if candidates.len() == 1 {
return Ok(0);
}
let target_weight = font.weight.0;
let target_italic = matches!(
font.style,
open_gpui::FontStyle::Italic | open_gpui::FontStyle::Oblique
);
let mut best_index = 0;
let mut best_score = u32::MAX;
for (index, font_id) in candidates.iter().enumerate() {
let database_id = state.loaded_font(*font_id).font.id();
let face_info = state
.font_system
.db()
.face(database_id)
.context("font face not found in database")?;
let is_italic = matches!(
face_info.style,
cosmic_text::Style::Italic | cosmic_text::Style::Oblique
);
let style_penalty: u32 = if is_italic == target_italic { 0 } else { 1000 };
let weight_diff = (face_info.weight.0 as i32 - target_weight as i32).unsigned_abs();
let score = style_penalty + weight_diff;
if score < best_score {
best_score = score;
best_index = index;
}
}
Ok(best_index)
}
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
struct RunSpan {
start: usize,
end: usize,
slot: Option<usize>,
font_id: FontId,
}
fn compute_run_spans(
text: &str,
run_offset: usize,
run_len: usize,
primary: FontId,
fallback_chain: &[(FontId, SharedString)],
covers: &impl Fn(FontId, char) -> bool,
) -> SmallVec<[RunSpan; 4]> {
let mut spans = SmallVec::new();
let run_end = run_offset + run_len;
if run_end <= run_offset {
return spans;
}
if fallback_chain.is_empty() {
spans.push(RunSpan {
start: run_offset,
end: run_end,
slot: None,
font_id: primary,
});
return spans;
}
let run_text = &text[run_offset..run_end];
let mut span_start = run_offset;
let mut span_slot: Option<usize> = None;
let mut span_font_id = primary;
for (grapheme_idx, grapheme) in run_text.grapheme_indices(true) {
let abs = run_offset + grapheme_idx;
let ch = grapheme.chars().next().unwrap_or('\0');
let next_slot = pick_covering_slot(ch, span_slot, primary, fallback_chain, covers);
if next_slot == span_slot {
continue;
}
if abs > span_start {
spans.push(RunSpan {
start: span_start,
end: abs,
slot: span_slot,
font_id: span_font_id,
});
}
span_start = abs;
span_slot = next_slot;
span_font_id = slot_font_id(next_slot, primary, fallback_chain);
}
if span_start < run_end {
spans.push(RunSpan {
start: span_start,
end: run_end,
slot: span_slot,
font_id: span_font_id,
});
}
spans
}
fn slot_font_id(
slot: Option<usize>,
primary: FontId,
fallback_chain: &[(FontId, SharedString)],
) -> FontId {
match slot {
None => primary,
Some(ix) => fallback_chain[ix].0,
}
}
fn pick_covering_slot(
ch: char,
current: Option<usize>,
primary: FontId,
fallback_chain: &[(FontId, SharedString)],
covers: &impl Fn(FontId, char) -> bool,
) -> Option<usize> {
if (ch as u32) <= 0x7F {
return None;
}
if covers(primary, ch) {
return None;
}
let current_id = slot_font_id(current, primary, fallback_chain);
if covers(current_id, ch) {
return current;
}
for (ix, (fb_id, _)) in fallback_chain.iter().enumerate() {
if covers(*fb_id, ch) {
return Some(ix);
}
}
None
}
fn charmap_covers(loaded_fonts: &[LoadedFont], id: FontId, ch: char) -> bool {
loaded_fonts
.get(id.0)
.is_some_and(|loaded| loaded.font.as_swash().charmap().map(ch) != 0)
}
fn cosmic_font_features(features: &FontFeatures) -> Result<CosmicFontFeatures> {
let mut result = CosmicFontFeatures::new();
for feature in features.0.iter() {
let name_bytes: [u8; 4] = feature
.0
.as_bytes()
.try_into()
.context("Incorrect feature flag format")?;
let tag = cosmic_text::FeatureTag::new(&name_bytes);
result.set(tag, feature.1);
}
Ok(result)
}
#[cfg(feature = "font-kit")]
fn find_best_font_kit_match(
candidates: &[font_kit::properties::Properties],
query: &font_kit::properties::Properties,
) -> Option<usize> {
use font_kit::properties::{Stretch, Style, Weight};
let mut matching_set = (0..candidates.len()).collect::<SmallVec<[_; 8]>>();
if matching_set.is_empty() {
return None;
}
let matching_stretch = if matching_set
.iter()
.any(|&index| candidates[index].stretch == query.stretch)
{
query.stretch
} else if query.stretch <= Stretch::NORMAL {
matching_set
.iter()
.filter(|&index| candidates[*index].stretch < query.stretch)
.min_by(|&a, &b| {
stretch_distance(candidates[*a].stretch, query.stretch)
.total_cmp(&stretch_distance(candidates[*b].stretch, query.stretch))
})
.or_else(|| {
matching_set.iter().min_by(|&a, &b| {
stretch_distance(candidates[*a].stretch, query.stretch)
.total_cmp(&stretch_distance(candidates[*b].stretch, query.stretch))
})
})
.map(|index| candidates[*index].stretch)?
} else {
matching_set
.iter()
.filter(|&index| candidates[*index].stretch > query.stretch)
.min_by(|&a, &b| {
stretch_distance(candidates[*a].stretch, query.stretch)
.total_cmp(&stretch_distance(candidates[*b].stretch, query.stretch))
})
.or_else(|| {
matching_set.iter().min_by(|&a, &b| {
stretch_distance(candidates[*a].stretch, query.stretch)
.total_cmp(&stretch_distance(candidates[*b].stretch, query.stretch))
})
})
.map(|index| candidates[*index].stretch)?
};
matching_set.retain(|index| candidates[*index].stretch == matching_stretch);
let style_preference = match query.style {
Style::Italic => [Style::Italic, Style::Oblique, Style::Normal],
Style::Oblique => [Style::Oblique, Style::Italic, Style::Normal],
Style::Normal => [Style::Normal, Style::Oblique, Style::Italic],
};
let matching_style = *style_preference.iter().find(|&query_style| {
matching_set
.iter()
.any(|&index| candidates[index].style == *query_style)
})?;
matching_set.retain(|index| candidates[*index].style == matching_style);
let matching_weight = if matching_set
.iter()
.any(|&index| candidates[index].weight == query.weight)
{
query.weight
} else if query.weight >= Weight(400.0)
&& query.weight < Weight(450.0)
&& matching_set
.iter()
.any(|&index| candidates[index].weight == Weight(500.0))
{
Weight(500.0)
} else if query.weight >= Weight(450.0)
&& query.weight <= Weight(500.0)
&& matching_set
.iter()
.any(|&index| candidates[index].weight == Weight(400.0))
{
Weight(400.0)
} else if query.weight <= Weight(500.0) {
matching_set
.iter()
.filter(|&index| candidates[*index].weight <= query.weight)
.min_by(|&a, &b| {
weight_distance(candidates[*a].weight, query.weight)
.total_cmp(&weight_distance(candidates[*b].weight, query.weight))
})
.or_else(|| {
matching_set.iter().min_by(|&a, &b| {
weight_distance(candidates[*a].weight, query.weight)
.total_cmp(&weight_distance(candidates[*b].weight, query.weight))
})
})
.map(|index| candidates[*index].weight)?
} else {
matching_set
.iter()
.filter(|&index| candidates[*index].weight >= query.weight)
.min_by(|&a, &b| {
weight_distance(candidates[*a].weight, query.weight)
.total_cmp(&weight_distance(candidates[*b].weight, query.weight))
})
.or_else(|| {
matching_set.iter().min_by(|&a, &b| {
weight_distance(candidates[*a].weight, query.weight)
.total_cmp(&weight_distance(candidates[*b].weight, query.weight))
})
})
.map(|index| candidates[*index].weight)?
};
matching_set.retain(|index| candidates[*index].weight == matching_weight);
matching_set.into_iter().next()
}
#[cfg(feature = "font-kit")]
fn stretch_distance(a: font_kit::properties::Stretch, b: font_kit::properties::Stretch) -> f32 {
(a.0 - b.0).abs()
}
#[cfg(feature = "font-kit")]
fn weight_distance(a: font_kit::properties::Weight, b: font_kit::properties::Weight) -> f32 {
(a.0 - b.0).abs()
}
#[cfg(feature = "font-kit")]
fn font_into_properties(font: &open_gpui::Font) -> font_kit::properties::Properties {
font_kit::properties::Properties {
style: match font.style {
open_gpui::FontStyle::Normal => font_kit::properties::Style::Normal,
open_gpui::FontStyle::Italic => font_kit::properties::Style::Italic,
open_gpui::FontStyle::Oblique => font_kit::properties::Style::Oblique,
},
weight: font_kit::properties::Weight(font.weight.0),
stretch: Default::default(),
}
}
#[cfg(feature = "font-kit")]
fn face_info_into_properties(
face_info: &cosmic_text::fontdb::FaceInfo,
) -> font_kit::properties::Properties {
font_kit::properties::Properties {
style: match face_info.style {
cosmic_text::Style::Normal => font_kit::properties::Style::Normal,
cosmic_text::Style::Italic => font_kit::properties::Style::Italic,
cosmic_text::Style::Oblique => font_kit::properties::Style::Oblique,
},
weight: font_kit::properties::Weight(face_info.weight.0.into()),
stretch: match face_info.stretch {
cosmic_text::Stretch::Condensed => font_kit::properties::Stretch::CONDENSED,
cosmic_text::Stretch::Expanded => font_kit::properties::Stretch::EXPANDED,
cosmic_text::Stretch::ExtraCondensed => font_kit::properties::Stretch::EXTRA_CONDENSED,
cosmic_text::Stretch::ExtraExpanded => font_kit::properties::Stretch::EXTRA_EXPANDED,
cosmic_text::Stretch::Normal => font_kit::properties::Stretch::NORMAL,
cosmic_text::Stretch::SemiCondensed => font_kit::properties::Stretch::SEMI_CONDENSED,
cosmic_text::Stretch::SemiExpanded => font_kit::properties::Stretch::SEMI_EXPANDED,
cosmic_text::Stretch::UltraCondensed => font_kit::properties::Stretch::ULTRA_CONDENSED,
cosmic_text::Stretch::UltraExpanded => font_kit::properties::Stretch::ULTRA_EXPANDED,
},
}
}
fn check_is_known_emoji_font(postscript_name: &str) -> bool {
postscript_name == "NotoColorEmoji"
}
#[cfg(test)]
mod tests {
use super::*;
#[cfg(feature = "font-kit")]
fn props(
style: font_kit::properties::Style,
weight: f32,
stretch: font_kit::properties::Stretch,
) -> font_kit::properties::Properties {
font_kit::properties::Properties {
style,
weight: font_kit::properties::Weight(weight),
stretch,
}
}
fn fid(i: usize) -> FontId {
FontId(i)
}
fn chain(ids: &[usize]) -> SmallVec<[(FontId, SharedString); 4]> {
ids.iter()
.map(|&i| (fid(i), SharedString::from(format!("fb{i}"))))
.collect()
}
fn span(start: usize, end: usize, slot: Option<usize>, font_id: FontId) -> RunSpan {
RunSpan {
start,
end,
slot,
font_id,
}
}
#[cfg(feature = "font-kit")]
#[test]
fn font_kit_match_returns_none_for_empty_candidates() {
use font_kit::properties::{Stretch, Style};
assert_eq!(
find_best_font_kit_match(&[], &props(Style::Normal, 400.0, Stretch::NORMAL)),
None
);
}
#[cfg(feature = "font-kit")]
#[test]
fn font_kit_match_prefers_stretch_before_style_and_weight() {
use font_kit::properties::{Stretch, Style};
let candidates = [
props(Style::Italic, 400.0, Stretch::EXPANDED),
props(Style::Normal, 900.0, Stretch::CONDENSED),
props(Style::Oblique, 700.0, Stretch::NORMAL),
];
assert_eq!(
find_best_font_kit_match(
&candidates,
&props(Style::Normal, 400.0, Stretch::SEMI_CONDENSED)
),
Some(1)
);
}
#[cfg(feature = "font-kit")]
#[test]
fn font_kit_match_uses_css_style_preference_order() {
use font_kit::properties::{Stretch, Style};
let candidates = [
props(Style::Normal, 400.0, Stretch::NORMAL),
props(Style::Oblique, 400.0, Stretch::NORMAL),
props(Style::Italic, 400.0, Stretch::NORMAL),
];
assert_eq!(
find_best_font_kit_match(&candidates, &props(Style::Italic, 400.0, Stretch::NORMAL)),
Some(2)
);
assert_eq!(
find_best_font_kit_match(
&candidates[..2],
&props(Style::Italic, 400.0, Stretch::NORMAL)
),
Some(1)
);
}
#[cfg(feature = "font-kit")]
#[test]
fn font_kit_match_keeps_css_weight_edge_cases() {
use font_kit::properties::{Stretch, Style};
let candidates = [
props(Style::Normal, 300.0, Stretch::NORMAL),
props(Style::Normal, 400.0, Stretch::NORMAL),
props(Style::Normal, 500.0, Stretch::NORMAL),
];
assert_eq!(
find_best_font_kit_match(&candidates, &props(Style::Normal, 425.0, Stretch::NORMAL)),
Some(2)
);
assert_eq!(
find_best_font_kit_match(&candidates, &props(Style::Normal, 475.0, Stretch::NORMAL)),
Some(1)
);
}
#[test]
fn primary_wins_over_current_fallback_when_primary_covers() {
let primary = fid(0);
let fb = chain(&[1, 2]);
let covers = |id: FontId, _: char| id == fid(0) || id == fid(1);
assert_eq!(
pick_covering_slot('a', Some(0), primary, &fb, &covers),
None
);
}
#[test]
fn primary_preferred_over_fallback_when_both_cover() {
let primary = fid(0);
let fb = chain(&[1]);
let covers = |_: FontId, _: char| true;
assert_eq!(pick_covering_slot('a', None, primary, &fb, &covers), None);
}
#[test]
fn falls_through_chain_in_order() {
let primary = fid(0);
let fb = chain(&[1, 2, 3]);
let covers = |id: FontId, _: char| id == fid(2);
assert_eq!(
pick_covering_slot('字', None, primary, &fb, &covers),
Some(1)
);
}
#[test]
fn no_coverage_returns_primary() {
let primary = fid(0);
let fb = chain(&[1, 2]);
let covers = |_: FontId, _: char| false;
assert_eq!(
pick_covering_slot('\u{1F600}', Some(1), primary, &fb, &covers),
None
);
}
#[test]
fn empty_chain_always_returns_primary() {
let primary = fid(0);
let fb: SmallVec<[(FontId, SharedString); 4]> = SmallVec::new();
let covers = |_: FontId, _: char| false;
assert_eq!(pick_covering_slot('a', None, primary, &fb, &covers), None);
}
#[test]
fn slot_font_id_resolution() {
let primary = fid(7);
let fb = chain(&[10, 20]);
assert_eq!(slot_font_id(None, primary, &fb), fid(7));
assert_eq!(slot_font_id(Some(0), primary, &fb), fid(10));
assert_eq!(slot_font_id(Some(1), primary, &fb), fid(20));
}
#[test]
fn run_spans_with_no_chain_emit_one_primary_span() {
let primary = fid(0);
let fb: SmallVec<[(FontId, SharedString); 4]> = SmallVec::new();
let covers = |_: FontId, _: char| false;
let text = "hello";
let spans = compute_run_spans(text, 0, text.len(), primary, &fb, &covers);
assert_eq!(spans.as_slice(), &[span(0, text.len(), None, primary)]);
}
#[test]
fn run_spans_use_byte_offsets_for_multibyte_chars() {
let primary = fid(0);
let fb = chain(&[1]);
let covers = |id: FontId, ch: char| {
if id == primary {
ch.is_ascii()
} else {
!ch.is_ascii()
}
};
let text = "a字b";
let spans = compute_run_spans(text, 0, text.len(), primary, &fb, &covers);
assert_eq!(
spans.as_slice(),
&[
span(0, 1, None, primary),
span(1, 4, Some(0), fid(1)),
span(4, 5, None, primary),
]
);
}
#[test]
fn run_spans_respect_run_offset() {
let primary = fid(0);
let fb = chain(&[1]);
let covers = |id: FontId, ch: char| {
if id == primary {
ch.is_ascii()
} else {
!ch.is_ascii()
}
};
let text = "xx字y";
let run_offset = 2;
let run_len = text.len() - run_offset;
let spans = compute_run_spans(text, run_offset, run_len, primary, &fb, &covers);
assert_eq!(
spans.as_slice(),
&[span(2, 5, Some(0), fid(1)), span(5, 6, None, primary)]
);
}
#[test]
fn run_spans_keep_combining_marks_with_base_in_fallback() {
let primary = fid(0);
let fb = chain(&[1]);
let covers = |id: FontId, ch: char| {
if id == primary {
ch.is_ascii()
} else {
ch == '\u{0905}'
}
};
let text = "\u{0905}\u{0902}";
let spans = compute_run_spans(text, 0, text.len(), primary, &fb, &covers);
assert_eq!(spans.as_slice(), &[span(0, text.len(), Some(0), fid(1))]);
}
#[test]
fn run_spans_keep_zwj_inside_emoji_cluster() {
let primary = fid(0);
let fb = chain(&[1]);
let covers = |id: FontId, ch: char| id == fid(1) && ch != '\u{200D}';
let text = "\u{1F469}\u{200D}\u{1F467}";
let spans = compute_run_spans(text, 0, text.len(), primary, &fb, &covers);
assert_eq!(spans.as_slice(), &[span(0, text.len(), Some(0), fid(1))]);
}
#[test]
fn run_spans_collapse_adjacent_same_slot() {
let primary = fid(0);
let fb = chain(&[1]);
let covers = |id: FontId, ch: char| {
if id == primary {
ch.is_ascii()
} else {
!ch.is_ascii()
}
};
let text = "字字字";
let spans = compute_run_spans(text, 0, text.len(), primary, &fb, &covers);
assert_eq!(spans.as_slice(), &[span(0, text.len(), Some(0), fid(1))]);
}
#[test]
fn run_spans_empty_run_returns_no_spans() {
let primary = fid(0);
let fb = chain(&[1]);
let covers = |_: FontId, _: char| true;
let spans = compute_run_spans("anything", 3, 0, primary, &fb, &covers);
assert!(spans.is_empty());
}
}