Skip to content

Commit 8a1613a

Browse files
committed
Fonts: OversampleH/OversampleV value defaults to 0 for automatic selection.
1 parent 4211fdc commit 8a1613a

File tree

7 files changed

+38
-21
lines changed

7 files changed

+38
-21
lines changed

docs/CHANGELOG.txt

+7-3
Original file line numberDiff line numberDiff line change
@@ -76,11 +76,15 @@ Other changes:
7676
styles as the current look is not right (but ImGuiCol_TabSelectedOverline stays the same).
7777
- Debug Tools: Tweaked font preview.
7878
- ImDrawList: texture baked storage for thick line reduced from ~64x64 to ~32x32. (#3245)
79-
- ImFontAtlas: made calling ClearFonts() call ClearInputData(), as calling
80-
one without the other is never correct. (#8174, #6556, #6336, #4723)
81-
- ImFont: IndexLookup[] table hold 16-bit values even in ImWchar32 mode,
79+
- Fonts: IndexLookup[] table hold 16-bit values even in ImWchar32 mode,
8280
as it accounts for number of glyphs in same font. This is favorable to
8381
CalcTextSize() calls touching less memory.
82+
- Fonts: OversampleH/OversampleV defaults to 0 for automatic selection.
83+
- OversampleH == 0 --> use 1 or 2 depending on font size and use of PixelSnapH.
84+
- OversampleV == 0 --> always use 1.
85+
This also
86+
- ImFontAtlas: made calling ClearFonts() call ClearInputData(), as calling
87+
one without the other is never correct. (#8174, #6556, #6336, #4723)
8488
- Examples: DirectX12: Reduced number of frame in flight from 3 to 2 in
8589
provided example, to reduce latency.
8690
- Examples: Vulkan: better handle VK_SUBOPTIMAL_KHR being returned by

docs/FONTS.md

-2
Original file line numberDiff line numberDiff line change
@@ -110,8 +110,6 @@ ImGui::PopFont();
110110
**For advanced options create a ImFontConfig structure and pass it to the AddFont() function (it will be copied internally):**
111111
```cpp
112112
ImFontConfig config;
113-
config.OversampleH = 2;
114-
config.OversampleV = 1;
115113
config.GlyphExtraSpacing.x = 1.0f;
116114
ImFont* font = io.Fonts->AddFontFromFileTTF("font.ttf", size_pixels, &config);
117115
```

imgui.cpp

+7-3
Original file line numberDiff line numberDiff line change
@@ -16184,9 +16184,13 @@ void ImGui::DebugNodeFont(ImFont* font)
1618416184
Text("Texture Area: about %d px ~%dx%d px", font->MetricsTotalSurface, surface_sqrt, surface_sqrt);
1618516185
for (int config_i = 0; config_i < font->ConfigDataCount; config_i++)
1618616186
if (font->ConfigData)
16187-
if (const ImFontConfig* cfg = &font->ConfigData[config_i])
16188-
BulletText("Input %d: \'%s\', Oversample: (%d,%d), PixelSnapH: %d, Offset: (%.1f,%.1f)",
16189-
config_i, cfg->Name, cfg->OversampleH, cfg->OversampleV, cfg->PixelSnapH, cfg->GlyphOffset.x, cfg->GlyphOffset.y);
16187+
{
16188+
const ImFontConfig* cfg = &font->ConfigData[config_i];
16189+
int oversample_h, oversample_v;
16190+
ImFontAtlasBuildGetOversampleFactors(cfg, &oversample_h, &oversample_v);
16191+
BulletText("Input %d: \'%s\', Oversample: (%d=>%d,%d=>%d), PixelSnapH: %d, Offset: (%.1f,%.1f)",
16192+
config_i, cfg->Name, cfg->OversampleH, oversample_h, cfg->OversampleV, oversample_v, cfg->PixelSnapH, cfg->GlyphOffset.x, cfg->GlyphOffset.y);
16193+
}
1619016194

1619116195
// Display all glyphs of the fonts in separate pages of 256 characters
1619216196
{

imgui.h

+2-2
Original file line numberDiff line numberDiff line change
@@ -3254,8 +3254,8 @@ struct ImFontConfig
32543254
bool MergeMode; // false // Merge into previous ImFont, so you can combine multiple inputs font into one ImFont (e.g. ASCII font + icons + Japanese glyphs). You may want to use GlyphOffset.y when merge font of different heights.
32553255
bool PixelSnapH; // false // Align every glyph AdvanceX to pixel boundaries. Useful e.g. if you are merging a non-pixel aligned font with the default font. If enabled, you can set OversampleH/V to 1.
32563256
int FontNo; // 0 // Index of font within TTF/OTF file
3257-
int OversampleH; // 2 // Rasterize at higher quality for sub-pixel positioning. Note the difference between 2 and 3 is minimal. You can reduce this to 1 for large glyphs save memory. Read https://github.com/nothings/stb/blob/master/tests/oversample/README.md for details.
3258-
int OversampleV; // 1 // Rasterize at higher quality for sub-pixel positioning. This is not really useful as we don't use sub-pixel positions on the Y axis.
3257+
int OversampleH; // 0 (2) // Rasterize at higher quality for sub-pixel positioning. 0 == auto == 1 or 2 depending on size. Note the difference between 2 and 3 is minimal. You can reduce this to 1 for large glyphs save memory. Read https://github.com/nothings/stb/blob/master/tests/oversample/README.md for details.
3258+
int OversampleV; // 0 (1) // Rasterize at higher quality for sub-pixel positioning. 0 == auto == 1. This is not really useful as we don't use sub-pixel positions on the Y axis.
32593259
float SizePixels; // // Size in pixels for rasterizer (more or less maps to the resulting font height).
32603260
ImVec2 GlyphExtraSpacing; // 0, 0 // Extra spacing (in pixels) between glyphs when rendered: essentially add to glyph->AdvanceX. Only X axis is supported for now.
32613261
ImVec2 GlyphOffset; // 0, 0 // Offset all glyphs from this font input.

imgui_draw.cpp

+20-10
Original file line numberDiff line numberDiff line change
@@ -2374,8 +2374,8 @@ ImFontConfig::ImFontConfig()
23742374
{
23752375
memset(this, 0, sizeof(*this));
23762376
FontDataOwnedByAtlas = true;
2377-
OversampleH = 2;
2378-
OversampleV = 1;
2377+
OversampleH = 0; // Auto == 1 or 2 depending on size
2378+
OversampleV = 0; // Auto == 1
23792379
GlyphMaxAdvanceX = FLT_MAX;
23802380
RasterizerMultiply = 1.0f;
23812381
RasterizerDensity = 1.0f;
@@ -2573,8 +2573,7 @@ ImFont* ImFontAtlas::AddFont(const ImFontConfig* font_cfg)
25732573
IM_ASSERT(!Locked && "Cannot modify a locked ImFontAtlas between NewFrame() and EndFrame/Render()!");
25742574
IM_ASSERT(font_cfg->FontData != NULL && font_cfg->FontDataSize > 0);
25752575
IM_ASSERT(font_cfg->SizePixels > 0.0f && "Is ImFontConfig struct correctly initialized?");
2576-
IM_ASSERT(font_cfg->OversampleH > 0 && font_cfg->OversampleV > 0 && "Is ImFontConfig struct correctly initialized?");
2577-
IM_ASSERT(font_cfg->RasterizerDensity > 0.0f);
2576+
IM_ASSERT(font_cfg->RasterizerDensity > 0.0f && "Is ImFontConfig struct correctly initialized?");
25782577

25792578
// Create new font
25802579
if (!font_cfg->MergeMode)
@@ -2858,6 +2857,13 @@ static void UnpackBitVectorToFlatIndexList(const ImBitVector* in, ImVector<int>*
28582857
out->push_back((int)(((it - it_begin) << 5) + bit_n));
28592858
}
28602859

2860+
void ImFontAtlasBuildGetOversampleFactors(const ImFontConfig* cfg, int* out_oversample_h, int* out_oversample_v)
2861+
{
2862+
// Automatically disable horizontal oversampling over size 32
2863+
*out_oversample_h = (cfg->OversampleH != 0) ? cfg->OversampleH : (cfg->SizePixels * cfg->RasterizerDensity > 32.0f || cfg->PixelSnapH) ? 1 : 2;
2864+
*out_oversample_v = (cfg->OversampleV != 0) ? cfg->OversampleV : 1;
2865+
}
2866+
28612867
static bool ImFontAtlasBuildWithStbTruetype(ImFontAtlas* atlas)
28622868
{
28632869
IM_ASSERT(atlas->ConfigData.Size > 0);
@@ -2985,15 +2991,19 @@ static bool ImFontAtlasBuildWithStbTruetype(ImFontAtlas* atlas)
29852991
buf_rects_out_n += src_tmp.GlyphsCount;
29862992
buf_packedchars_out_n += src_tmp.GlyphsCount;
29872993

2988-
// Convert our ranges in the format stb_truetype wants
2994+
// Automatic selection of oversampling parameters
29892995
ImFontConfig& cfg = atlas->ConfigData[src_i];
2996+
int oversample_h, oversample_v;
2997+
ImFontAtlasBuildGetOversampleFactors(&cfg, &oversample_h, &oversample_v);
2998+
2999+
// Convert our ranges in the format stb_truetype wants
29903000
src_tmp.PackRange.font_size = cfg.SizePixels * cfg.RasterizerDensity;
29913001
src_tmp.PackRange.first_unicode_codepoint_in_range = 0;
29923002
src_tmp.PackRange.array_of_unicode_codepoints = src_tmp.GlyphsList.Data;
29933003
src_tmp.PackRange.num_chars = src_tmp.GlyphsList.Size;
29943004
src_tmp.PackRange.chardata_for_range = src_tmp.PackedChars;
2995-
src_tmp.PackRange.h_oversample = (unsigned char)cfg.OversampleH;
2996-
src_tmp.PackRange.v_oversample = (unsigned char)cfg.OversampleV;
3005+
src_tmp.PackRange.h_oversample = (unsigned char)oversample_h;
3006+
src_tmp.PackRange.v_oversample = (unsigned char)oversample_v;
29973007

29983008
// Gather the sizes of all rectangles we will need to pack (this loop is based on stbtt_PackFontRangesGatherRects)
29993009
const float scale = (cfg.SizePixels > 0.0f) ? stbtt_ScaleForPixelHeight(&src_tmp.FontInfo, cfg.SizePixels * cfg.RasterizerDensity) : stbtt_ScaleForMappingEmToPixels(&src_tmp.FontInfo, -cfg.SizePixels * cfg.RasterizerDensity);
@@ -3002,9 +3012,9 @@ static bool ImFontAtlasBuildWithStbTruetype(ImFontAtlas* atlas)
30023012
int x0, y0, x1, y1;
30033013
const int glyph_index_in_font = stbtt_FindGlyphIndex(&src_tmp.FontInfo, src_tmp.GlyphsList[glyph_i]);
30043014
IM_ASSERT(glyph_index_in_font != 0);
3005-
stbtt_GetGlyphBitmapBoxSubpixel(&src_tmp.FontInfo, glyph_index_in_font, scale * cfg.OversampleH, scale * cfg.OversampleV, 0, 0, &x0, &y0, &x1, &y1);
3006-
src_tmp.Rects[glyph_i].w = (stbrp_coord)(x1 - x0 + pack_padding + cfg.OversampleH - 1);
3007-
src_tmp.Rects[glyph_i].h = (stbrp_coord)(y1 - y0 + pack_padding + cfg.OversampleV - 1);
3015+
stbtt_GetGlyphBitmapBoxSubpixel(&src_tmp.FontInfo, glyph_index_in_font, scale * oversample_h, scale * oversample_v, 0, 0, &x0, &y0, &x1, &y1);
3016+
src_tmp.Rects[glyph_i].w = (stbrp_coord)(x1 - x0 + pack_padding + oversample_h - 1);
3017+
src_tmp.Rects[glyph_i].h = (stbrp_coord)(y1 - y0 + pack_padding + oversample_v - 1);
30083018
total_surface += src_tmp.Rects[glyph_i].w * src_tmp.Rects[glyph_i].h;
30093019
}
30103020
}

imgui_internal.h

+1
Original file line numberDiff line numberDiff line change
@@ -3571,6 +3571,7 @@ IMGUI_API void ImFontAtlasBuildRender8bppRectFromString(ImFontAtlas* atlas,
35713571
IMGUI_API void ImFontAtlasBuildRender32bppRectFromString(ImFontAtlas* atlas, int x, int y, int w, int h, const char* in_str, char in_marker_char, unsigned int in_marker_pixel_value);
35723572
IMGUI_API void ImFontAtlasBuildMultiplyCalcLookupTable(unsigned char out_table[256], float in_multiply_factor);
35733573
IMGUI_API void ImFontAtlasBuildMultiplyRectAlpha8(const unsigned char table[256], unsigned char* pixels, int x, int y, int w, int h, int stride);
3574+
IMGUI_API void ImFontAtlasBuildGetOversampleFactors(const ImFontConfig* cfg, int* out_oversample_h, int* out_oversample_v);
35743575

35753576
//-----------------------------------------------------------------------------
35763577
// [SECTION] Test Engine specific hooks (imgui_test_engine)

misc/freetype/imgui_freetype.cpp

+1-1
Original file line numberDiff line numberDiff line change
@@ -33,7 +33,7 @@
3333
// - For correct results you need to be using sRGB and convert to linear space in the pixel shader output.
3434
// - The default dear imgui styles will be impacted by this change (alpha values will need tweaking).
3535

36-
// FIXME: cfg.OversampleH, OversampleV are not supported (but perhaps not so necessary with this rasterizer).
36+
// FIXME: cfg.OversampleH, OversampleV are not supported, but generally not necessary with this rasterizer because Hinting makes everything look better.
3737

3838
#include "imgui.h"
3939
#ifndef IMGUI_DISABLE

0 commit comments

Comments
 (0)