mirror of
https://github.com/JanDeDobbeleer/oh-my-posh.git
synced 2025-01-11 11:17:27 -08:00
feat: improved theme png rendering
* feat: improved theme png rendering Overall this improves glyph rendering in the theme PNGs for docs. Changes: - Replaces VictorMono with latest Hack nerd font - When calculating overall width, count glyphs as double wide - When rendering a glyph, allow ~1.7x space (this seemed right testing many) I hope this is a net improvement that helps theme renders be a bit more accurate and shine! * move to 2x for overall width expansion 1.7 introduces some odd offset problems with some backgrounds - 2x is stable. * Based on data from: https://github.com/ryanoasis/nerd-fonts/wiki/Glyph-Sets-and-Code-Points ...checking those points for nerd font glyphs in play I'm unfamiliar with Go here and maybe there's a better way to implement, but the results are much improved. * tweak ranges and bg position for font diff * exclude pixelated (e0c4-e0c7) from doube-width * simplify cyclomatic complexity Go checks were unhappy - this isn't a hot path so breaking it out into a format that matches the docs, for easier maintenance. * I have no idea what I'm doing ...but I hope this works - bit simpler layout to maintain too. * fix: add comments Good PR feedback - adding some more commentary for anyone coming across these bits later * typo
This commit is contained in:
parent
4eb3b439ba
commit
e27d34a1cb
BIN
src/font/Hack-Nerd-Bold.ttf
Normal file
BIN
src/font/Hack-Nerd-Bold.ttf
Normal file
Binary file not shown.
BIN
src/font/Hack-Nerd-Italic.ttf
Normal file
BIN
src/font/Hack-Nerd-Italic.ttf
Normal file
Binary file not shown.
BIN
src/font/Hack-Nerd-Regular.ttf
Normal file
BIN
src/font/Hack-Nerd-Regular.ttf
Normal file
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
84
src/image.go
84
src/image.go
|
@ -67,14 +67,14 @@ const (
|
||||||
link = "link"
|
link = "link"
|
||||||
)
|
)
|
||||||
|
|
||||||
//go:embed font/VictorMono-Bold.ttf
|
//go:embed font/Hack-Nerd-Bold.ttf
|
||||||
var victorMonoBold []byte
|
var hackBold []byte
|
||||||
|
|
||||||
//go:embed font/VictorMono-Regular.ttf
|
//go:embed font/Hack-Nerd-Regular.ttf
|
||||||
var victorMonoRegular []byte
|
var hackRegular []byte
|
||||||
|
|
||||||
//go:embed font/VictorMono-Italic.ttf
|
//go:embed font/Hack-Nerd-Italic.ttf
|
||||||
var victorMonoItalic []byte
|
var hackItalic []byte
|
||||||
|
|
||||||
type RGB struct {
|
type RGB struct {
|
||||||
r int
|
r int
|
||||||
|
@ -135,9 +135,9 @@ func (ir *ImageRenderer) init() {
|
||||||
|
|
||||||
ir.cleanContent()
|
ir.cleanContent()
|
||||||
|
|
||||||
fontRegular, _ := truetype.Parse(victorMonoRegular)
|
fontRegular, _ := truetype.Parse(hackRegular)
|
||||||
fontBold, _ := truetype.Parse(victorMonoBold)
|
fontBold, _ := truetype.Parse(hackBold)
|
||||||
fontItalic, _ := truetype.Parse(victorMonoItalic)
|
fontItalic, _ := truetype.Parse(hackItalic)
|
||||||
fontFaceOptions := &truetype.Options{Size: f * 12, DPI: 144}
|
fontFaceOptions := &truetype.Options{Size: f * 12, DPI: 144}
|
||||||
|
|
||||||
ir.defaultForegroundColor = &RGB{255, 255, 255}
|
ir.defaultForegroundColor = &RGB{255, 255, 255}
|
||||||
|
@ -188,10 +188,65 @@ func (ir *ImageRenderer) fontHeight() float64 {
|
||||||
return float64(ir.regular.Metrics().Height >> 6)
|
return float64(ir.regular.Metrics().Height >> 6)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type RuneRange struct {
|
||||||
|
Start rune
|
||||||
|
End rune
|
||||||
|
}
|
||||||
|
|
||||||
|
// If we're a Nerd Font code point, treat as double width
|
||||||
|
var doubleWidthRunes = []RuneRange{
|
||||||
|
// Seti-UI + Custom range
|
||||||
|
{Start: '\ue5fa', End: '\ue62b'},
|
||||||
|
// Devicons
|
||||||
|
{Start: '\ue700', End: '\ue7c5'},
|
||||||
|
// Font Awesome
|
||||||
|
{Start: '\uf000', End: '\uf2e0'},
|
||||||
|
// Font Awesome Extension
|
||||||
|
{Start: '\ue200', End: '\ue2a9'},
|
||||||
|
// Material Design Icons
|
||||||
|
{Start: '\uf500', End: '\ufd46'},
|
||||||
|
// Weather
|
||||||
|
{Start: '\ue300', End: '\ue3eb'},
|
||||||
|
// Octicons
|
||||||
|
{Start: '\uf400', End: '\uf4a8'},
|
||||||
|
{Start: '\u2665', End: '\u2665'},
|
||||||
|
{Start: '\u26A1', End: '\u26A1'},
|
||||||
|
{Start: '\uf27c', End: '\uf27c'},
|
||||||
|
// Powerline Extra Symbols (intentionally excluding single width bubbles (e0b4-e0b7) and pixelated (e0c4-e0c7))
|
||||||
|
{Start: '\ue0a3', End: '\ue0a3'},
|
||||||
|
{Start: '\ue0b8', End: '\ue0c3'},
|
||||||
|
{Start: '\ue0c8', End: '\ue0c8'},
|
||||||
|
{Start: '\ue0ca', End: '\ue0ca'},
|
||||||
|
{Start: '\ue0cc', End: '\ue0d2'},
|
||||||
|
{Start: '\ue0d4', End: '\ue0d4'},
|
||||||
|
// IEC Power Symbols
|
||||||
|
{Start: '\u23fb', End: '\u23fe'},
|
||||||
|
{Start: '\u2b58', End: '\u2b58'},
|
||||||
|
// Font Logos
|
||||||
|
{Start: '\uf300', End: '\uf313'},
|
||||||
|
// Pomicons
|
||||||
|
{Start: '\ue000', End: '\ue00d'},
|
||||||
|
}
|
||||||
|
|
||||||
|
// This is getting how many additional characters of width to allocate when drawing
|
||||||
|
// e.g. for characters that are 2 or more wide. A standard character will return 0
|
||||||
|
// Nerd Font glyphs will return 1, since most are double width
|
||||||
|
func (ir *ImageRenderer) runeAdditionalWidth(r rune) int {
|
||||||
|
for _, runeRange := range doubleWidthRunes {
|
||||||
|
if runeRange.Start <= r && r <= runeRange.End {
|
||||||
|
return 1
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
func (ir *ImageRenderer) calculateWidth() int {
|
func (ir *ImageRenderer) calculateWidth() int {
|
||||||
longest := 0
|
longest := 0
|
||||||
for _, line := range strings.Split(ir.ansiString, "\n") {
|
for _, line := range strings.Split(ir.ansiString, "\n") {
|
||||||
length := ir.ansi.lenWithoutANSI(line)
|
length := ir.ansi.lenWithoutANSI(line)
|
||||||
|
for _, char := range line {
|
||||||
|
length += ir.runeAdditionalWidth(char)
|
||||||
|
}
|
||||||
if length > longest {
|
if length > longest {
|
||||||
longest = length
|
longest = length
|
||||||
}
|
}
|
||||||
|
@ -320,9 +375,18 @@ func (ir *ImageRenderer) SavePNG(path string) error {
|
||||||
}
|
}
|
||||||
|
|
||||||
w, h := dc.MeasureString(str)
|
w, h := dc.MeasureString(str)
|
||||||
|
// The gg library unfortunately returns a single character width for *all* glyphs in a font.
|
||||||
|
// So if we know the glyph to occupy n additional characters in width, allocate that area
|
||||||
|
// e.g. this will double the space for Nerd Fonts, but some could even be 3 or 4 wide
|
||||||
|
// If there's 0 additional characters of width (the common case), this won't add anything
|
||||||
|
w += (w * float64(ir.runeAdditionalWidth(runes[0])))
|
||||||
|
|
||||||
if ir.backgroundColor != nil {
|
if ir.backgroundColor != nil {
|
||||||
dc.SetRGB255(ir.backgroundColor.r, ir.backgroundColor.g, ir.backgroundColor.b)
|
dc.SetRGB255(ir.backgroundColor.r, ir.backgroundColor.g, ir.backgroundColor.b)
|
||||||
dc.DrawRectangle(x, y-h, w, h+12)
|
// The background for a character needs love to align to the font we're using
|
||||||
|
// Not all fonts are rendered the same height or starting position,
|
||||||
|
// so we're shifting the background rectangles vertically to correct
|
||||||
|
dc.DrawRectangle(x, y-h+3, w, h+9)
|
||||||
dc.Fill()
|
dc.Fill()
|
||||||
}
|
}
|
||||||
if ir.foregroundColor != nil {
|
if ir.foregroundColor != nil {
|
||||||
|
|
Loading…
Reference in a new issue