mirror of
https://github.com/emilk/egui.git
synced 2026-06-26 22:53:14 -04:00
Make font hinting target configurable via FontTweak (#8262)
Fixes #8079, where font hinting only sharpens the vertical axis (vertical stems stay blurry), and none of the skrifa knobs were reachable — only switching to `Target::Mono` helped, but that wasn't exposed. This makes the hinting target configurable instead of hardcoding `Target::Smooth { symmetric_rendering: true, preserve_linear_metrics: true }`. ### API - New `epaint::text::HintingTarget` mirroring `skrifa::outline::Target`: - `Mono` - `Smooth(SmoothHinting)` where `SmoothHinting { light, symmetric_rendering, preserve_linear_metrics }` - New field `FontTweak::hinting_target: HintingTarget`. - Each variant/field is documented with what it does and the egui-specific caveats (e.g. `symmetric_rendering` only affects interpreter-hinted fonts; egui positions glyphs from shaper advances so `preserve_linear_metrics` mostly affects sharpness, not layout). ### Render - `font.rs` converts `HintingTarget` → `skrifa::outline::Target` and threads it into the per-glyph `reconfigure` call; the hinting instance is also reconfigured when the target changes. ### UI - The font-tweak settings panel gets a `hinting_target` row: Smooth/Mono radios, `light` / `symmetric_rendering` / `preserve_linear_metrics` checkboxes, and a `Reset` button — all with tooltips. ### Behavior - `HintingTarget::default()` matches egui's previous hardcoded target, so **rendering is unchanged** unless you opt in. To fix the horizontal blur from #8079, uncheck `preserve_linear_metrics` (or pick `Mono`). Whether to flip the *default* is left as a follow-up. 🤖 Generated with [Claude Code](https://claude.com/claude-code) --------- Co-authored-by: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
This commit is contained in:
@@ -225,6 +225,7 @@ impl FontCell {
|
||||
glyph_id: GlyphId,
|
||||
bin: SubpixelBin,
|
||||
location: skrifa::instance::LocationRef<'_>,
|
||||
hinting_target: skrifa::outline::Target,
|
||||
) -> Option<GlyphAllocation> {
|
||||
debug_assert!(
|
||||
glyph_id != skrifa::GlyphId::NOTDEF,
|
||||
@@ -244,18 +245,10 @@ impl FontCell {
|
||||
let size = skrifa::instance::Size::new(metrics.scale);
|
||||
if hinting_instance.size() != size
|
||||
|| hinting_instance.location().coords() != location.coords()
|
||||
|| hinting_instance.target() != hinting_target
|
||||
{
|
||||
hinting_instance
|
||||
.reconfigure(
|
||||
&font_data.outline_glyphs,
|
||||
size,
|
||||
location,
|
||||
skrifa::outline::Target::Smooth {
|
||||
mode: skrifa::outline::SmoothMode::Normal,
|
||||
symmetric_rendering: true,
|
||||
preserve_linear_metrics: true,
|
||||
},
|
||||
)
|
||||
.reconfigure(&font_data.outline_glyphs, size, location, hinting_target)
|
||||
.ok()?;
|
||||
}
|
||||
let draw_settings = skrifa::outline::DrawSettings::hinted(hinting_instance, false);
|
||||
@@ -637,9 +630,17 @@ impl FontFace {
|
||||
|
||||
let cache_key = GlyphCacheKey::new(glyph_id, metrics, bin);
|
||||
|
||||
let hinting_target = self.tweak.hinting_target.into();
|
||||
let alloc = *self.glyph_alloc_cache.entry(cache_key).or_insert_with(|| {
|
||||
self.font
|
||||
.allocate_glyph_uncached(atlas, metrics, glyph_id, bin, (&metrics.location).into())
|
||||
.allocate_glyph_uncached(
|
||||
atlas,
|
||||
metrics,
|
||||
glyph_id,
|
||||
bin,
|
||||
(&metrics.location).into(),
|
||||
hinting_target,
|
||||
)
|
||||
.unwrap_or_default()
|
||||
});
|
||||
|
||||
|
||||
@@ -242,6 +242,11 @@ pub struct FontTweak {
|
||||
/// `None` means use the global setting in [`TextOptions::font_hinting`].
|
||||
pub hinting: Option<bool>,
|
||||
|
||||
/// How to grid-fit the glyph outlines when hinting is enabled.
|
||||
///
|
||||
/// Has no effect when hinting is disabled (see [`Self::hinting`]).
|
||||
pub hinting_target: HintingTarget,
|
||||
|
||||
/// Override the global sub-pixel binning setting for this specific font.
|
||||
///
|
||||
/// `None` means use the global setting in [`TextOptions::subpixel_binning`].
|
||||
@@ -271,6 +276,7 @@ impl Default for FontTweak {
|
||||
y_offset_factor: 0.0,
|
||||
y_offset: 0.0,
|
||||
hinting: None,
|
||||
hinting_target: HintingTarget::default(),
|
||||
subpixel_binning: None,
|
||||
coords: VariationCoords::default(),
|
||||
thin_space_width: 0.5,
|
||||
@@ -281,6 +287,111 @@ impl Default for FontTweak {
|
||||
|
||||
// ----------------------------------------------------------------------------
|
||||
|
||||
/// How to *hint* glyph outlines, i.e. how aggressively to nudge them onto the
|
||||
/// pixel grid before rasterizing. Mirrors [`skrifa::outline::Target`].
|
||||
///
|
||||
/// Hinting trades shape fidelity for sharpness: snapping stems to whole pixels
|
||||
/// makes text crisp at small sizes / low dpi, at the cost of slightly distorting
|
||||
/// the designer's outlines. At high dpi it matters little.
|
||||
///
|
||||
/// This only has an effect if the font is actually hinted — either it ships
|
||||
/// TrueType instructions, or it was auto-hinted (see [`FontTweak::hinting`]).
|
||||
///
|
||||
/// Used by [`FontTweak::hinting_target`].
|
||||
#[derive(Clone, Copy, Debug, PartialEq, Eq)]
|
||||
#[cfg_attr(feature = "serde", derive(serde::Deserialize, serde::Serialize))]
|
||||
pub enum HintingTarget {
|
||||
/// Strongest hinting, designed for aliased 1-bit (black & white) rendering.
|
||||
///
|
||||
/// Snaps stems hard to the pixel grid for maximum sharpness. egui always
|
||||
/// renders anti-aliased, so in practice this looks much like [`Self::Smooth`]
|
||||
/// here; it mostly exists for completeness. Maps to `skrifa`'s `Target::Mono`.
|
||||
Mono,
|
||||
|
||||
/// Hinting tuned for anti-aliased rendering. This is what you normally want,
|
||||
/// and what egui uses by default. Maps to `skrifa`'s `Target::Smooth`.
|
||||
Smooth(SmoothHinting),
|
||||
}
|
||||
|
||||
impl Default for HintingTarget {
|
||||
fn default() -> Self {
|
||||
Self::Smooth(SmoothHinting::default())
|
||||
}
|
||||
}
|
||||
|
||||
impl From<HintingTarget> for skrifa::outline::Target {
|
||||
fn from(hinting_target: HintingTarget) -> Self {
|
||||
use skrifa::outline::SmoothMode;
|
||||
match hinting_target {
|
||||
HintingTarget::Mono => Self::Mono,
|
||||
HintingTarget::Smooth(SmoothHinting {
|
||||
light,
|
||||
symmetric_rendering,
|
||||
preserve_linear_metrics,
|
||||
}) => Self::Smooth {
|
||||
mode: if light {
|
||||
SmoothMode::Light
|
||||
} else {
|
||||
SmoothMode::Normal
|
||||
},
|
||||
symmetric_rendering,
|
||||
preserve_linear_metrics,
|
||||
},
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/// Tuning for [`HintingTarget::Smooth`], mirroring `skrifa`'s `Target::Smooth`.
|
||||
#[derive(Clone, Copy, Debug, PartialEq, Eq)]
|
||||
#[cfg_attr(feature = "serde", derive(serde::Deserialize, serde::Serialize))]
|
||||
pub struct SmoothHinting {
|
||||
/// Hint only lightly: snap stems vertically but leave horizontal shapes
|
||||
/// alone (`FreeType`'s "light" mode). Preserves the font's proportions
|
||||
/// better, at the cost of a little horizontal sharpness.
|
||||
///
|
||||
/// `false` uses the "normal" mode, which also fits horizontally.
|
||||
/// Maps to `SmoothMode::Light` (`true`) vs `SmoothMode::Normal` (`false`).
|
||||
pub light: bool,
|
||||
|
||||
/// Render a glyph the same way regardless of its sub-pixel position.
|
||||
///
|
||||
/// `true` makes glyphs position-independent (better for caching and
|
||||
/// animation), but a font's instructions may then widen stems and look
|
||||
/// slightly blurrier under an analytic rasterizer like egui's.
|
||||
///
|
||||
/// **Only affects fonts hinted via the TrueType interpreter** (i.e. fonts
|
||||
/// that ship their own instructions). It has no effect on the auto-hinter.
|
||||
/// Mirrors `Target::Smooth { symmetric_rendering }`.
|
||||
pub symmetric_rendering: bool,
|
||||
|
||||
/// Keep advance widths independent of hinting (don't grid-fit horizontally
|
||||
/// in a way that changes spacing).
|
||||
///
|
||||
/// `true` keeps inter-glyph spacing identical to the unhinted font, so
|
||||
/// layout never depends on hinting — but it also prevents horizontal
|
||||
/// grid-fitting, leaving vertical stems softer on low-dpi screens.
|
||||
///
|
||||
/// `false` lets the (auto)hinter snap horizontally for crisper stems.
|
||||
/// egui positions glyphs from the shaper's advances, not the hinted
|
||||
/// outline, so this mainly affects sharpness here, not layout.
|
||||
/// Mirrors `Target::Smooth { preserve_linear_metrics }`.
|
||||
pub preserve_linear_metrics: bool,
|
||||
}
|
||||
|
||||
impl Default for SmoothHinting {
|
||||
fn default() -> Self {
|
||||
// Matches the behavior egui had before the hinting target was configurable.
|
||||
// Note this means horizontal grid-fitting is opt-in (see `preserve_linear_metrics`).
|
||||
Self {
|
||||
light: false,
|
||||
symmetric_rendering: true,
|
||||
preserve_linear_metrics: true,
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// ----------------------------------------------------------------------------
|
||||
|
||||
pub type Blob = Arc<dyn AsRef<[u8]> + Send + Sync>;
|
||||
|
||||
fn blob_from_font_data(data: &FontData) -> Blob {
|
||||
|
||||
@@ -10,7 +10,8 @@ mod text_layout_types;
|
||||
pub use {
|
||||
fonts::{
|
||||
FontData, FontDefinitions, FontFamily, FontId, FontInsert, FontPriority, FontTweak,
|
||||
FontVariationAxis, Fonts, FontsImpl, FontsView, InsertFontFamily,
|
||||
FontVariationAxis, Fonts, FontsImpl, FontsView, HintingTarget, InsertFontFamily,
|
||||
SmoothHinting,
|
||||
},
|
||||
index::{ByteIndex, ByteRange, ByteRangeExt, CharIndex, CharRange, CharRangeExt},
|
||||
text_layout::*,
|
||||
|
||||
Reference in New Issue
Block a user