This project demonstrates how to blend text in a HLSL shader and have it look like native DirectWrite.
It implements both, ClearType (D2D1_TEXT_ANTIALIAS_MODE_CLEARTYPE
) and grayscale (D2D1_TEXT_ANTIALIAS_MODE_GRAYSCALE
) anti-aliasing.
This project is an extract of Windows Terminal. See LICENSE.
- Emojis
Emojis are luckily very simple to draw and I'll add support later:- Draw them with grayscale antialiasing (and not ClearType).
-
Don't use linear gamma space (meaning:
DWrite_GetRenderParams
). Instead callSetTextRenderingParams(nullptr)
before using the rendering target, to reset it to the regular, gamma corrected parameters. - Do a simple premultiplied alpha blend with the Emoji's RGBA values on your background color in your shader.
- This demo doesn't actually use a glyph atlas and only shows the "blending" part of the algorithm.
-
dwrite.hlsl contains all of the shader functions relevant for grayscale-antialiased alpha blending.
DWrite_GetGrayScaleCorrectedAlpha
is the entrypoint function that you need to call in your shader. -
dwrite.cpp contains support functions which are required to fill out the parameters for
DWrite_GetGrayScaleCorrectedAlpha
. - Draw your glyphs with DirectWrite into your texture atlas however you like them. These are the approaches I'm aware about:
-
ID2D1RenderTarget::DrawTextLayout
Direct2D provides various different kinds of render targets. It handles font fallback, provides you with metrics, etc. and is simple to use. However it's not particularly configurable, not the most performant solution and uses extra GPU/CPU memory for Direct2D's internal glyph atlas. This demo application uses this approach. -
Implement your own custom
IDWriteTextRenderer
Personally I'm not sure about the value of this approach. I feel like it's the worst of both worlds, as it still ties you into the Direct2D ecosystem and allows you to only marginally improve performance, while simultaneously being very difficult to implement. I'd not suggest anyone to use this. -
ID2D1RenderTarget::DrawGlyphRun
This is Direct2D's low level API, which ties better into DirectWrite. This API is preferable if you're implementing a glyph atlas, because it allows you to cache individual glyphs that you got fromIDWriteTextAnalyzer
. Here's what you need to call when using DirectWrite:IDWriteFontFallback::MapCharacters
IDWriteTextAnalyzer::AnalyzeBidi
IDWriteTextAnalyzer::AnalyzeScript
-
IDWriteTextAnalyzer1::GetTextComplexity
(optionally; allows you to skipGetGlyphs
) IDWriteTextAnalyzer::GetGlyphs
IDWriteTextAnalyzer::GetGlyphPlacements
You can call
ID2D1DeviceContext::GetGlyphRunWorldBounds
before callingDrawGlyphRun
to get pixel-precise boundaries for the given glyph. Unfortunately,GetGlyphRunWorldBounds
only works reliably for regular OpenType glyphs and not for SVG layers in COLRv0 emojis or for COLRv1 in general. -
IDWriteGlyphRunAnalysis::CreateAlphaTexture
This is the lowest level approach that is technically the best. It's what every serious DirectWrite application uses, including libraries like skia. It straight up yields rasterized glyphs and allows you to do your own anti-aliasing. Don't be fooled byDWRITE_TEXTURE_ALIASED_1x1
: It yields grayscale textures (allegedly). It effectively replaces theDrawGlyphRun
call in the previous point, but if you just replace it 1:1 you might notice a reduction in performance. This is because Direct2D internally uses a pool of upload heaps to efficiently send batches of glyphs up to the GPU memory. Preferably, you'd do something similar.
-