Skip to content

Commit 94f8b02

Browse files
authored
improve fallback fonts alignment (#2724)
* use font metrics in layout * properly center scaled fonts * adjust docs * fix raised text * fix easymark viewer small text alignment caused by variable row heights
1 parent 089c7b4 commit 94f8b02

File tree

5 files changed

+173
-110
lines changed

5 files changed

+173
-110
lines changed

crates/egui_demo_lib/src/easy_mark/easy_mark_viewer.rs

+17-2
Original file line numberDiff line numberDiff line change
@@ -38,11 +38,26 @@ pub fn item_ui(ui: &mut Ui, item: easy_mark::Item<'_>) {
3838
}
3939

4040
easy_mark::Item::Text(style, text) => {
41-
ui.label(rich_text_from_style(text, &style));
41+
let label = rich_text_from_style(text, &style);
42+
if style.small && !style.raised {
43+
ui.with_layout(Layout::left_to_right(Align::BOTTOM), |ui| {
44+
ui.set_min_height(row_height);
45+
ui.label(label);
46+
});
47+
} else {
48+
ui.label(label);
49+
}
4250
}
4351
easy_mark::Item::Hyperlink(style, text, url) => {
4452
let label = rich_text_from_style(text, &style);
45-
ui.add(Hyperlink::from_label_and_url(label, url));
53+
if style.small && !style.raised {
54+
ui.with_layout(Layout::left_to_right(Align::BOTTOM), |ui| {
55+
ui.set_height(row_height);
56+
ui.add(Hyperlink::from_label_and_url(label, url));
57+
});
58+
} else {
59+
ui.add(Hyperlink::from_label_and_url(label, url));
60+
}
4661
}
4762

4863
easy_mark::Item::Separator => {

crates/epaint/src/text/font.rs

+101-75
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
use crate::{
22
mutex::{Mutex, RwLock},
3+
text::FontTweak,
34
TextureAtlas,
45
};
56
use emath::{vec2, Vec2};
@@ -42,6 +43,17 @@ pub struct GlyphInfo {
4243
/// Unit: points.
4344
pub advance_width: f32,
4445

46+
/// `ascent` value from the font metrics.
47+
/// this is the distance from the top to the baseline.
48+
///
49+
/// Unit: points.
50+
pub ascent: f32,
51+
52+
/// row height computed from the font metrics.
53+
///
54+
/// Unit: points.
55+
pub row_height: f32,
56+
4557
/// Texture coordinates.
4658
pub uv_rect: UvRect,
4759
}
@@ -52,6 +64,8 @@ impl Default for GlyphInfo {
5264
Self {
5365
id: ab_glyph::GlyphId(0),
5466
advance_width: 0.0,
67+
ascent: 0.0,
68+
row_height: 0.0,
5569
uv_rect: Default::default(),
5670
}
5771
}
@@ -69,6 +83,7 @@ pub struct FontImpl {
6983
height_in_points: f32,
7084
// move each character by this much (hack)
7185
y_offset: f32,
86+
ascent: f32,
7287
pixels_per_point: f32,
7388
glyph_info_cache: RwLock<ahash::HashMap<char, GlyphInfo>>, // TODO(emilk): standard Mutex
7489
atlas: Arc<Mutex<TextureAtlas>>,
@@ -80,20 +95,37 @@ impl FontImpl {
8095
pixels_per_point: f32,
8196
name: String,
8297
ab_glyph_font: ab_glyph::FontArc,
83-
scale_in_pixels: u32,
84-
y_offset_points: f32,
98+
scale_in_pixels: f32,
99+
tweak: FontTweak,
85100
) -> FontImpl {
86-
assert!(scale_in_pixels > 0);
101+
assert!(scale_in_pixels > 0.0);
87102
assert!(pixels_per_point > 0.0);
88103

89-
let height_in_points = scale_in_pixels as f32 / pixels_per_point;
104+
use ab_glyph::*;
105+
let scaled = ab_glyph_font.as_scaled(scale_in_pixels);
106+
let ascent = scaled.ascent() / pixels_per_point;
107+
let descent = scaled.descent() / pixels_per_point;
108+
let line_gap = scaled.line_gap() / pixels_per_point;
109+
110+
// Tweak the scale as the user desired
111+
let scale_in_pixels = scale_in_pixels * tweak.scale;
112+
113+
let baseline_offset = {
114+
let scale_in_points = scale_in_pixels / pixels_per_point;
115+
scale_in_points * tweak.baseline_offset_factor
116+
};
90117

91-
// TODO(emilk): use these font metrics?
92-
// use ab_glyph::ScaleFont as _;
93-
// let scaled = ab_glyph_font.as_scaled(scale_in_pixels as f32);
94-
// dbg!(scaled.ascent());
95-
// dbg!(scaled.descent());
96-
// dbg!(scaled.line_gap());
118+
let y_offset_points = {
119+
let scale_in_points = scale_in_pixels / pixels_per_point;
120+
scale_in_points * tweak.y_offset_factor
121+
} + tweak.y_offset;
122+
123+
// center scaled glyphs properly
124+
let y_offset_points = y_offset_points + (tweak.scale - 1.0) * 0.5 * (ascent + descent);
125+
126+
// Round to an even number of physical pixels to get even kerning.
127+
// See https://github.com/emilk/egui/issues/382
128+
let scale_in_pixels = scale_in_pixels.round() as u32;
97129

98130
// Round to closest pixel:
99131
let y_offset = (y_offset_points * pixels_per_point).round() / pixels_per_point;
@@ -102,8 +134,9 @@ impl FontImpl {
102134
name,
103135
ab_glyph_font,
104136
scale_in_pixels,
105-
height_in_points,
137+
height_in_points: ascent - descent + line_gap,
106138
y_offset,
139+
ascent: ascent + baseline_offset,
107140
pixels_per_point,
108141
glyph_info_cache: Default::default(),
109142
atlas,
@@ -194,15 +227,7 @@ impl FontImpl {
194227
if glyph_id.0 == 0 {
195228
None // unsupported character
196229
} else {
197-
let glyph_info = allocate_glyph(
198-
&mut self.atlas.lock(),
199-
&self.ab_glyph_font,
200-
glyph_id,
201-
self.scale_in_pixels as f32,
202-
self.y_offset,
203-
self.pixels_per_point,
204-
);
205-
230+
let glyph_info = self.allocate_glyph(glyph_id);
206231
self.glyph_info_cache.write().insert(c, glyph_info);
207232
Some(glyph_info)
208233
}
@@ -231,6 +256,62 @@ impl FontImpl {
231256
pub fn pixels_per_point(&self) -> f32 {
232257
self.pixels_per_point
233258
}
259+
260+
fn allocate_glyph(&self, glyph_id: ab_glyph::GlyphId) -> GlyphInfo {
261+
assert!(glyph_id.0 != 0);
262+
use ab_glyph::{Font as _, ScaleFont};
263+
264+
let glyph = glyph_id.with_scale_and_position(
265+
self.scale_in_pixels as f32,
266+
ab_glyph::Point { x: 0.0, y: 0.0 },
267+
);
268+
269+
let uv_rect = self.ab_glyph_font.outline_glyph(glyph).map(|glyph| {
270+
let bb = glyph.px_bounds();
271+
let glyph_width = bb.width() as usize;
272+
let glyph_height = bb.height() as usize;
273+
if glyph_width == 0 || glyph_height == 0 {
274+
UvRect::default()
275+
} else {
276+
let atlas = &mut self.atlas.lock();
277+
let (glyph_pos, image) = atlas.allocate((glyph_width, glyph_height));
278+
glyph.draw(|x, y, v| {
279+
if v > 0.0 {
280+
let px = glyph_pos.0 + x as usize;
281+
let py = glyph_pos.1 + y as usize;
282+
image[(px, py)] = v;
283+
}
284+
});
285+
286+
let offset_in_pixels = vec2(bb.min.x, bb.min.y);
287+
let offset = offset_in_pixels / self.pixels_per_point + self.y_offset * Vec2::Y;
288+
UvRect {
289+
offset,
290+
size: vec2(glyph_width as f32, glyph_height as f32) / self.pixels_per_point,
291+
min: [glyph_pos.0 as u16, glyph_pos.1 as u16],
292+
max: [
293+
(glyph_pos.0 + glyph_width) as u16,
294+
(glyph_pos.1 + glyph_height) as u16,
295+
],
296+
}
297+
}
298+
});
299+
let uv_rect = uv_rect.unwrap_or_default();
300+
301+
let advance_width_in_points = self
302+
.ab_glyph_font
303+
.as_scaled(self.scale_in_pixels as f32)
304+
.h_advance(glyph_id)
305+
/ self.pixels_per_point;
306+
307+
GlyphInfo {
308+
id: glyph_id,
309+
advance_width: advance_width_in_points,
310+
ascent: self.ascent,
311+
row_height: self.row_height(),
312+
uv_rect,
313+
}
314+
}
234315
}
235316

236317
type FontIndex = usize;
@@ -429,58 +510,3 @@ fn invisible_char(c: char) -> bool {
429510
| '\u{FEFF}' // ZERO WIDTH NO-BREAK SPACE
430511
)
431512
}
432-
433-
fn allocate_glyph(
434-
atlas: &mut TextureAtlas,
435-
font: &ab_glyph::FontArc,
436-
glyph_id: ab_glyph::GlyphId,
437-
scale_in_pixels: f32,
438-
y_offset: f32,
439-
pixels_per_point: f32,
440-
) -> GlyphInfo {
441-
assert!(glyph_id.0 != 0);
442-
use ab_glyph::{Font as _, ScaleFont};
443-
444-
let glyph =
445-
glyph_id.with_scale_and_position(scale_in_pixels, ab_glyph::Point { x: 0.0, y: 0.0 });
446-
447-
let uv_rect = font.outline_glyph(glyph).map(|glyph| {
448-
let bb = glyph.px_bounds();
449-
let glyph_width = bb.width() as usize;
450-
let glyph_height = bb.height() as usize;
451-
if glyph_width == 0 || glyph_height == 0 {
452-
UvRect::default()
453-
} else {
454-
let (glyph_pos, image) = atlas.allocate((glyph_width, glyph_height));
455-
glyph.draw(|x, y, v| {
456-
if v > 0.0 {
457-
let px = glyph_pos.0 + x as usize;
458-
let py = glyph_pos.1 + y as usize;
459-
image[(px, py)] = v;
460-
}
461-
});
462-
463-
let offset_in_pixels = vec2(bb.min.x, scale_in_pixels + bb.min.y);
464-
let offset = offset_in_pixels / pixels_per_point + y_offset * Vec2::Y;
465-
UvRect {
466-
offset,
467-
size: vec2(glyph_width as f32, glyph_height as f32) / pixels_per_point,
468-
min: [glyph_pos.0 as u16, glyph_pos.1 as u16],
469-
max: [
470-
(glyph_pos.0 + glyph_width) as u16,
471-
(glyph_pos.1 + glyph_height) as u16,
472-
],
473-
}
474-
}
475-
});
476-
let uv_rect = uv_rect.unwrap_or_default();
477-
478-
let advance_width_in_points =
479-
font.as_scaled(scale_in_pixels).h_advance(glyph_id) / pixels_per_point;
480-
481-
GlyphInfo {
482-
id: glyph_id,
483-
advance_width: advance_width_in_points,
484-
uv_rect,
485-
}
486-
}

crates/epaint/src/text/fonts.rs

+28-24
Original file line numberDiff line numberDiff line change
@@ -153,31 +153,42 @@ impl FontData {
153153
#[derive(Copy, Clone, Debug, PartialEq)]
154154
#[cfg_attr(feature = "serde", derive(serde::Deserialize, serde::Serialize))]
155155
pub struct FontTweak {
156-
/// Scale the font by this much.
156+
/// Scale the font's glyphs by this much.
157+
/// this is only a visual effect and does not affect the text layout.
157158
///
158159
/// Default: `1.0` (no scaling).
159160
pub scale: f32,
160161

161-
/// Shift font downwards by this fraction of the font size (in points).
162+
/// Shift font's glyphs downwards by this fraction of the font size (in points).
163+
/// this is only a visual effect and does not affect the text layout.
162164
///
163165
/// A positive value shifts the text downwards.
164166
/// A negative value shifts it upwards.
165167
///
166168
/// Example value: `-0.2`.
167169
pub y_offset_factor: f32,
168170

169-
/// Shift font downwards by this amount of logical points.
171+
/// Shift font's glyphs downwards by this amount of logical points.
172+
/// this is only a visual effect and does not affect the text layout.
170173
///
171174
/// Example value: `2.0`.
172175
pub y_offset: f32,
176+
177+
/// When using this font's metrics to layout a row,
178+
/// shift the entire row downwards by this fraction of the font size (in points).
179+
///
180+
/// A positive value shifts the text downwards.
181+
/// A negative value shifts it upwards.
182+
pub baseline_offset_factor: f32,
173183
}
174184

175185
impl Default for FontTweak {
176186
fn default() -> Self {
177187
Self {
178188
scale: 1.0,
179-
y_offset_factor: -0.2, // makes the default fonts look more centered in buttons and such
189+
y_offset_factor: 0.0,
180190
y_offset: 0.0,
191+
baseline_offset_factor: -0.0333, // makes the default fonts look more centered in buttons and such
181192
}
182193
}
183194
}
@@ -272,9 +283,8 @@ impl Default for FontDefinitions {
272283
"NotoEmoji-Regular".to_owned(),
273284
FontData::from_static(include_bytes!("../../fonts/NotoEmoji-Regular.ttf")).tweak(
274285
FontTweak {
275-
scale: 0.81, // make it smaller
276-
y_offset_factor: -0.2, // move it up
277-
y_offset: 0.0,
286+
scale: 0.81, // make it smaller
287+
..Default::default()
278288
},
279289
),
280290
);
@@ -284,9 +294,12 @@ impl Default for FontDefinitions {
284294
"emoji-icon-font".to_owned(),
285295
FontData::from_static(include_bytes!("../../fonts/emoji-icon-font.ttf")).tweak(
286296
FontTweak {
287-
scale: 0.88, // make it smaller
288-
y_offset_factor: 0.07, // move it down slightly
289-
y_offset: 0.0,
297+
scale: 0.88, // make it smaller
298+
299+
// probably not correct, but this does make texts look better (#2724 for details)
300+
y_offset_factor: 0.11, // move glyphs down to better align with common fonts
301+
baseline_offset_factor: -0.11, // ...now the entire row is a bit down so shift it back
302+
..Default::default()
290303
},
291304
),
292305
);
@@ -760,28 +773,19 @@ impl FontImplCache {
760773
let font_scaling = ab_glyph_font.height_unscaled() / units_per_em;
761774
let scale_in_pixels = scale_in_pixels * font_scaling;
762775

763-
// Tweak the scale as the user desired:
764-
let scale_in_pixels = scale_in_pixels * tweak.scale;
765-
766-
// Round to an even number of physical pixels to get even kerning.
767-
// See https://github.com/emilk/egui/issues/382
768-
let scale_in_pixels = scale_in_pixels.round() as u32;
769-
770-
let y_offset_points = {
771-
let scale_in_points = scale_in_pixels as f32 / self.pixels_per_point;
772-
scale_in_points * tweak.y_offset_factor
773-
} + tweak.y_offset;
774-
775776
self.cache
776-
.entry((scale_in_pixels, font_name.to_owned()))
777+
.entry((
778+
(scale_in_pixels * tweak.scale).round() as u32,
779+
font_name.to_owned(),
780+
))
777781
.or_insert_with(|| {
778782
Arc::new(FontImpl::new(
779783
self.atlas.clone(),
780784
self.pixels_per_point,
781785
font_name.to_owned(),
782786
ab_glyph_font,
783787
scale_in_pixels,
784-
y_offset_points,
788+
tweak,
785789
))
786790
})
787791
.clone()

0 commit comments

Comments
 (0)