package d2asset import ( "image/color" "strings" "github.com/OpenDiablo2/OpenDiablo2/d2common/d2interface" "github.com/OpenDiablo2/OpenDiablo2/d2common/d2math" ) var _ d2interface.Font = &Font{} // Static check to confirm struct conforms to interface type fontGlyph struct { frame int width int height int } // Font represents a displayable font type Font struct { sheet d2interface.Animation glyphs map[rune]fontGlyph color color.Color } // SetColor sets the fonts color func (f *Font) SetColor(c color.Color) { f.color = c } // GetTextMetrics returns the dimensions of the Font element in pixels func (f *Font) GetTextMetrics(text string) (width, height int) { var ( lineWidth int lineHeight int totalWidth int totalHeight int ) for _, c := range text { if c == '\n' { totalWidth = d2math.MaxInt(totalWidth, lineWidth) totalHeight += lineHeight lineWidth = 0 lineHeight = 0 } else if glyph, ok := f.glyphs[c]; ok { lineWidth += glyph.width lineHeight = d2math.MaxInt(lineHeight, glyph.height) } } totalWidth = d2math.MaxInt(totalWidth, lineWidth) totalHeight += lineHeight return totalWidth, totalHeight } // RenderText prints a text using its configured style on a Surface (multi-lines are left-aligned, use label otherwise) func (f *Font) RenderText(text string, target d2interface.Surface) error { f.sheet.SetColorMod(f.color) lines := strings.Split(text, "\n") for _, line := range lines { var ( lineHeight int lineLength int ) for _, c := range line { glyph, ok := f.glyphs[c] if !ok { continue } if err := f.sheet.SetCurrentFrame(glyph.frame); err != nil { return err } if err := f.sheet.Render(target); err != nil { return err } lineHeight = d2math.MaxInt(lineHeight, glyph.height) lineLength++ target.PushTranslation(glyph.width, 0) } target.PopN(lineLength) target.PushTranslation(0, lineHeight) } target.PopN(len(lines)) return nil }