1
1
use crate :: {
2
2
mutex:: { Mutex , RwLock } ,
3
+ text:: FontTweak ,
3
4
TextureAtlas ,
4
5
} ;
5
6
use emath:: { vec2, Vec2 } ;
@@ -42,6 +43,17 @@ pub struct GlyphInfo {
42
43
/// Unit: points.
43
44
pub advance_width : f32 ,
44
45
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
+
45
57
/// Texture coordinates.
46
58
pub uv_rect : UvRect ,
47
59
}
@@ -52,6 +64,8 @@ impl Default for GlyphInfo {
52
64
Self {
53
65
id : ab_glyph:: GlyphId ( 0 ) ,
54
66
advance_width : 0.0 ,
67
+ ascent : 0.0 ,
68
+ row_height : 0.0 ,
55
69
uv_rect : Default :: default ( ) ,
56
70
}
57
71
}
@@ -69,6 +83,7 @@ pub struct FontImpl {
69
83
height_in_points : f32 ,
70
84
// move each character by this much (hack)
71
85
y_offset : f32 ,
86
+ ascent : f32 ,
72
87
pixels_per_point : f32 ,
73
88
glyph_info_cache : RwLock < ahash:: HashMap < char , GlyphInfo > > , // TODO(emilk): standard Mutex
74
89
atlas : Arc < Mutex < TextureAtlas > > ,
@@ -80,20 +95,37 @@ impl FontImpl {
80
95
pixels_per_point : f32 ,
81
96
name : String ,
82
97
ab_glyph_font : ab_glyph:: FontArc ,
83
- scale_in_pixels : u32 ,
84
- y_offset_points : f32 ,
98
+ scale_in_pixels : f32 ,
99
+ tweak : FontTweak ,
85
100
) -> FontImpl {
86
- assert ! ( scale_in_pixels > 0 ) ;
101
+ assert ! ( scale_in_pixels > 0.0 ) ;
87
102
assert ! ( pixels_per_point > 0.0 ) ;
88
103
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
+ } ;
90
117
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 ;
97
129
98
130
// Round to closest pixel:
99
131
let y_offset = ( y_offset_points * pixels_per_point) . round ( ) / pixels_per_point;
@@ -102,8 +134,9 @@ impl FontImpl {
102
134
name,
103
135
ab_glyph_font,
104
136
scale_in_pixels,
105
- height_in_points,
137
+ height_in_points : ascent - descent + line_gap ,
106
138
y_offset,
139
+ ascent : ascent + baseline_offset,
107
140
pixels_per_point,
108
141
glyph_info_cache : Default :: default ( ) ,
109
142
atlas,
@@ -194,15 +227,7 @@ impl FontImpl {
194
227
if glyph_id. 0 == 0 {
195
228
None // unsupported character
196
229
} 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) ;
206
231
self . glyph_info_cache . write ( ) . insert ( c, glyph_info) ;
207
232
Some ( glyph_info)
208
233
}
@@ -231,6 +256,62 @@ impl FontImpl {
231
256
pub fn pixels_per_point ( & self ) -> f32 {
232
257
self . pixels_per_point
233
258
}
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
+ }
234
315
}
235
316
236
317
type FontIndex = usize ;
@@ -429,58 +510,3 @@ fn invisible_char(c: char) -> bool {
429
510
| '\u{FEFF}' // ZERO WIDTH NO-BREAK SPACE
430
511
)
431
512
}
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
- }
0 commit comments