2021-04-04 11:28:41 -07:00
|
|
|
// Copyright © 2020 The Homeport Team
|
|
|
|
//
|
|
|
|
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
// of this software and associated documentation files (the "Software"), to deal
|
|
|
|
// in the Software without restriction, including without limitation the rights
|
|
|
|
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
// copies of the Software, and to permit persons to whom the Software is
|
|
|
|
// furnished to do so, subject to the following conditions:
|
|
|
|
//
|
|
|
|
// The above copyright notice and this permission notice shall be included in
|
|
|
|
// all copies or substantial portions of the Software.
|
|
|
|
//
|
|
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
|
|
// THE SOFTWARE.
|
|
|
|
|
|
|
|
// https://github.com/homeport/termshot
|
|
|
|
|
2022-01-26 23:38:46 -08:00
|
|
|
package engine
|
2021-04-04 11:28:41 -07:00
|
|
|
|
|
|
|
import (
|
|
|
|
_ "embed"
|
|
|
|
"fmt"
|
|
|
|
"math"
|
2022-01-26 04:09:21 -08:00
|
|
|
"oh-my-posh/color"
|
2022-01-26 01:23:18 -08:00
|
|
|
"oh-my-posh/regex"
|
2021-04-04 11:28:41 -07:00
|
|
|
"strconv"
|
|
|
|
"strings"
|
2022-02-04 10:34:44 -08:00
|
|
|
"unicode/utf8"
|
2021-04-04 11:28:41 -07:00
|
|
|
|
|
|
|
"github.com/esimov/stackblur-go"
|
|
|
|
"github.com/fogleman/gg"
|
|
|
|
"github.com/golang/freetype/truetype"
|
|
|
|
"golang.org/x/image/font"
|
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
red = "#ED655A"
|
|
|
|
yellow = "#E1C04C"
|
|
|
|
green = "#71BD47"
|
|
|
|
|
|
|
|
// known ansi sequences
|
2021-05-21 11:01:08 -07:00
|
|
|
|
|
|
|
fg = "FG"
|
|
|
|
bg = "BG"
|
2022-05-09 18:47:27 -07:00
|
|
|
bc = "BC" // for base 16 colors
|
2021-05-21 11:01:08 -07:00
|
|
|
str = "STR"
|
|
|
|
url = "URL"
|
2021-04-04 11:28:41 -07:00
|
|
|
invertedColor = "inverted"
|
|
|
|
invertedColorSingle = "invertedsingle"
|
|
|
|
fullColor = "full"
|
|
|
|
foreground = "foreground"
|
2022-05-09 18:47:27 -07:00
|
|
|
background = "background"
|
2021-04-04 11:28:41 -07:00
|
|
|
reset = "reset"
|
|
|
|
bold = "bold"
|
|
|
|
boldReset = "boldr"
|
|
|
|
italic = "italic"
|
|
|
|
italicReset = "italicr"
|
|
|
|
underline = "underline"
|
|
|
|
underlineReset = "underliner"
|
2022-05-22 21:33:32 -07:00
|
|
|
overline = "overline"
|
|
|
|
overlineReset = "overliner"
|
2021-04-04 11:28:41 -07:00
|
|
|
strikethrough = "strikethrough"
|
|
|
|
strikethroughReset = "strikethroughr"
|
|
|
|
color16 = "color16"
|
|
|
|
left = "left"
|
|
|
|
osc99 = "osc99"
|
2022-07-13 04:53:55 -07:00
|
|
|
osc7 = "osc7"
|
2022-11-24 14:45:44 -08:00
|
|
|
osc51 = "osc51"
|
2021-04-04 11:28:41 -07:00
|
|
|
lineChange = "linechange"
|
2022-01-26 22:44:35 -08:00
|
|
|
consoleTitle = "title"
|
2021-04-04 11:28:41 -07:00
|
|
|
link = "link"
|
|
|
|
)
|
|
|
|
|
2021-09-18 06:37:15 -07:00
|
|
|
//go:embed font/Hack-Nerd-Bold.ttf
|
|
|
|
var hackBold []byte
|
2021-04-04 11:28:41 -07:00
|
|
|
|
2021-09-18 06:37:15 -07:00
|
|
|
//go:embed font/Hack-Nerd-Regular.ttf
|
|
|
|
var hackRegular []byte
|
2021-04-04 11:28:41 -07:00
|
|
|
|
2021-09-18 06:37:15 -07:00
|
|
|
//go:embed font/Hack-Nerd-Italic.ttf
|
|
|
|
var hackItalic []byte
|
2021-04-04 11:28:41 -07:00
|
|
|
|
|
|
|
type RGB struct {
|
|
|
|
r int
|
|
|
|
g int
|
|
|
|
b int
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewRGBColor(ansiColor string) *RGB {
|
|
|
|
colors := strings.Split(ansiColor, ";")
|
|
|
|
b, _ := strconv.Atoi(colors[2])
|
2021-12-18 10:30:31 -08:00
|
|
|
g, _ := strconv.Atoi(colors[1])
|
|
|
|
r, _ := strconv.Atoi(colors[0])
|
2021-04-04 11:28:41 -07:00
|
|
|
return &RGB{
|
|
|
|
r: r,
|
|
|
|
g: g,
|
|
|
|
b: b,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
type ImageRenderer struct {
|
2022-01-26 23:38:46 -08:00
|
|
|
AnsiString string
|
|
|
|
Author string
|
|
|
|
CursorPadding int
|
|
|
|
RPromptOffset int
|
|
|
|
BgColor string
|
|
|
|
Ansi *color.Ansi
|
|
|
|
|
2022-05-05 09:06:31 -07:00
|
|
|
Path string
|
2021-04-04 11:28:41 -07:00
|
|
|
|
|
|
|
factor float64
|
|
|
|
|
|
|
|
columns int
|
|
|
|
rows int
|
|
|
|
|
|
|
|
defaultForegroundColor *RGB
|
|
|
|
defaultBackgroundColor *RGB
|
|
|
|
|
|
|
|
shadowBaseColor string
|
|
|
|
shadowRadius uint8
|
|
|
|
shadowOffsetX float64
|
|
|
|
shadowOffsetY float64
|
|
|
|
|
|
|
|
padding float64
|
|
|
|
margin float64
|
|
|
|
|
|
|
|
regular font.Face
|
|
|
|
bold font.Face
|
|
|
|
italic font.Face
|
|
|
|
lineSpacing float64
|
|
|
|
|
|
|
|
// canvas switches
|
|
|
|
style string
|
|
|
|
backgroundColor *RGB
|
|
|
|
foregroundColor *RGB
|
|
|
|
ansiSequenceRegexMap map[string]string
|
|
|
|
}
|
|
|
|
|
2022-01-26 23:38:46 -08:00
|
|
|
func (ir *ImageRenderer) Init(config string) {
|
2022-05-05 09:06:31 -07:00
|
|
|
if ir.Path == "" {
|
|
|
|
match := regex.FindNamedRegexMatch(`.*(\/|\\)(?P<STR>.+)\.(json|yaml|yml|toml)`, config)
|
|
|
|
ir.Path = fmt.Sprintf("%s.png", strings.TrimSuffix(match[str], ".omp"))
|
|
|
|
}
|
2022-01-26 23:38:46 -08:00
|
|
|
|
2021-04-04 11:28:41 -07:00
|
|
|
f := 2.0
|
|
|
|
|
|
|
|
ir.cleanContent()
|
|
|
|
|
2021-09-18 06:37:15 -07:00
|
|
|
fontRegular, _ := truetype.Parse(hackRegular)
|
|
|
|
fontBold, _ := truetype.Parse(hackBold)
|
|
|
|
fontItalic, _ := truetype.Parse(hackItalic)
|
2021-04-04 11:28:41 -07:00
|
|
|
fontFaceOptions := &truetype.Options{Size: f * 12, DPI: 144}
|
|
|
|
|
|
|
|
ir.defaultForegroundColor = &RGB{255, 255, 255}
|
|
|
|
ir.defaultBackgroundColor = &RGB{21, 21, 21}
|
|
|
|
|
|
|
|
ir.factor = f
|
|
|
|
|
|
|
|
ir.columns = 80
|
|
|
|
ir.rows = 25
|
|
|
|
|
|
|
|
ir.margin = f * 48
|
|
|
|
ir.padding = f * 24
|
|
|
|
|
|
|
|
ir.shadowBaseColor = "#10101066"
|
|
|
|
ir.shadowRadius = uint8(math.Min(f*16, 255))
|
|
|
|
ir.shadowOffsetX = f * 16
|
|
|
|
ir.shadowOffsetY = f * 16
|
|
|
|
|
|
|
|
ir.regular = truetype.NewFace(fontRegular, fontFaceOptions)
|
|
|
|
ir.bold = truetype.NewFace(fontBold, fontFaceOptions)
|
|
|
|
ir.italic = truetype.NewFace(fontItalic, fontFaceOptions)
|
|
|
|
ir.lineSpacing = 1.2
|
|
|
|
|
|
|
|
ir.ansiSequenceRegexMap = map[string]string{
|
|
|
|
invertedColor: `^(?P<STR>(\x1b\[38;2;(?P<BG>(\d+;?){3});49m){1}(\x1b\[7m))`,
|
|
|
|
invertedColorSingle: `^(?P<STR>\x1b\[(?P<BG>\d{2,3});49m\x1b\[7m)`,
|
|
|
|
fullColor: `^(?P<STR>(\x1b\[48;2;(?P<BG>(\d+;?){3})m)(\x1b\[38;2;(?P<FG>(\d+;?){3})m))`,
|
|
|
|
foreground: `^(?P<STR>(\x1b\[38;2;(?P<FG>(\d+;?){3})m))`,
|
2022-05-09 18:47:27 -07:00
|
|
|
background: `^(?P<STR>(\x1b\[48;2;(?P<BG>(\d+;?){3})m))`,
|
2021-04-04 11:28:41 -07:00
|
|
|
reset: `^(?P<STR>\x1b\[0m)`,
|
|
|
|
bold: `^(?P<STR>\x1b\[1m)`,
|
|
|
|
boldReset: `^(?P<STR>\x1b\[22m)`,
|
|
|
|
italic: `^(?P<STR>\x1b\[3m)`,
|
|
|
|
italicReset: `^(?P<STR>\x1b\[23m)`,
|
|
|
|
underline: `^(?P<STR>\x1b\[4m)`,
|
|
|
|
underlineReset: `^(?P<STR>\x1b\[24m)`,
|
2022-05-22 21:33:32 -07:00
|
|
|
overline: `^(?P<STR>\x1b\[53m)`,
|
|
|
|
overlineReset: `^(?P<STR>\x1b\[55m)`,
|
2021-04-04 11:28:41 -07:00
|
|
|
strikethrough: `^(?P<STR>\x1b\[9m)`,
|
|
|
|
strikethroughReset: `^(?P<STR>\x1b\[29m)`,
|
2022-05-22 21:33:32 -07:00
|
|
|
color16: `^(?P<STR>\x1b\[(?P<BC>[349][0-7]|10[0-7]|39)m)`,
|
2021-04-04 11:28:41 -07:00
|
|
|
left: `^(?P<STR>\x1b\[(\d{1,3})D)`,
|
|
|
|
osc99: `^(?P<STR>\x1b\]9;9;(.+)\x1b\\)`,
|
2022-07-13 04:53:55 -07:00
|
|
|
osc7: `^(?P<STR>\x1b\]7;(.+)\x1b\\)`,
|
2022-11-24 14:45:44 -08:00
|
|
|
osc51: `^(?P<STR>\x1b\]51;A(.+)\x1b\\)`,
|
2021-04-04 11:28:41 -07:00
|
|
|
lineChange: `^(?P<STR>\x1b\[(\d)[FB])`,
|
2022-01-26 22:44:35 -08:00
|
|
|
consoleTitle: `^(?P<STR>\x1b\]0;(.+)\007)`,
|
2022-02-05 11:54:05 -08:00
|
|
|
link: fmt.Sprintf(`^%s`, regex.LINK),
|
2021-04-04 11:28:41 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ir *ImageRenderer) fontHeight() float64 {
|
|
|
|
return float64(ir.regular.Metrics().Height >> 6)
|
|
|
|
}
|
|
|
|
|
2021-09-18 06:37:15 -07:00
|
|
|
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
|
2021-09-21 11:11:38 -07:00
|
|
|
{Start: '\uf300', End: '\uf31c'},
|
2021-09-18 06:37:15 -07:00
|
|
|
// 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
|
|
|
|
}
|
|
|
|
|
2022-02-03 08:36:37 -08:00
|
|
|
func (ir *ImageRenderer) lenWithoutANSI(text string) int {
|
|
|
|
if len(text) == 0 {
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
// replace hyperlinks(file/http/https)
|
|
|
|
regexStr := ir.ansiSequenceRegexMap[link]
|
|
|
|
matches := regex.FindAllNamedRegexMatch(regexStr, text)
|
|
|
|
for _, match := range matches {
|
|
|
|
text = strings.ReplaceAll(text, match[str], match[url])
|
|
|
|
}
|
|
|
|
// replace console title
|
|
|
|
regexStr = ir.ansiSequenceRegexMap[consoleTitle]
|
|
|
|
matches = regex.FindAllNamedRegexMatch(regexStr, text)
|
|
|
|
for _, match := range matches {
|
|
|
|
text = strings.ReplaceAll(text, match[str], "")
|
|
|
|
}
|
2022-02-11 11:50:02 -08:00
|
|
|
stripped := regex.ReplaceAllString(color.AnsiRegex, text, "")
|
2022-02-04 10:34:44 -08:00
|
|
|
length := utf8.RuneCountInString(stripped)
|
|
|
|
for _, rune := range stripped {
|
2022-02-03 08:36:37 -08:00
|
|
|
length += ir.runeAdditionalWidth(rune)
|
|
|
|
}
|
|
|
|
return length
|
|
|
|
}
|
|
|
|
|
2021-04-04 11:28:41 -07:00
|
|
|
func (ir *ImageRenderer) calculateWidth() int {
|
|
|
|
longest := 0
|
2022-01-26 23:38:46 -08:00
|
|
|
for _, line := range strings.Split(ir.AnsiString, "\n") {
|
2022-02-03 08:36:37 -08:00
|
|
|
length := ir.lenWithoutANSI(line)
|
2021-04-04 11:28:41 -07:00
|
|
|
if length > longest {
|
|
|
|
longest = length
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return longest
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ir *ImageRenderer) cleanContent() {
|
|
|
|
rPromptAnsi := "\x1b7\x1b[1000C"
|
2022-01-26 23:38:46 -08:00
|
|
|
hasRPrompt := strings.Contains(ir.AnsiString, rPromptAnsi)
|
2021-04-04 11:28:41 -07:00
|
|
|
// clean abundance of empty lines
|
2022-01-26 23:38:46 -08:00
|
|
|
ir.AnsiString = strings.Trim(ir.AnsiString, "\n")
|
|
|
|
ir.AnsiString = "\n" + ir.AnsiString
|
2021-04-04 11:28:41 -07:00
|
|
|
// clean string before render
|
2022-01-26 23:38:46 -08:00
|
|
|
ir.AnsiString = strings.ReplaceAll(ir.AnsiString, "\x1b[m", "\x1b[0m")
|
|
|
|
ir.AnsiString = strings.ReplaceAll(ir.AnsiString, "\x1b[K", "")
|
|
|
|
ir.AnsiString = strings.ReplaceAll(ir.AnsiString, "\x1b[1F", "")
|
|
|
|
ir.AnsiString = strings.ReplaceAll(ir.AnsiString, "\x1b8", "")
|
|
|
|
ir.AnsiString = strings.ReplaceAll(ir.AnsiString, "\u2800", " ")
|
2021-04-04 11:28:41 -07:00
|
|
|
// replace rprompt with adding and mark right aligned blocks with a pointer
|
2022-01-26 23:38:46 -08:00
|
|
|
ir.AnsiString = strings.ReplaceAll(ir.AnsiString, rPromptAnsi, fmt.Sprintf("_%s", strings.Repeat(" ", ir.CursorPadding)))
|
|
|
|
ir.AnsiString = strings.ReplaceAll(ir.AnsiString, "\x1b[1000C", strings.Repeat(" ", ir.RPromptOffset))
|
2021-04-04 11:28:41 -07:00
|
|
|
if !hasRPrompt {
|
2022-01-26 23:38:46 -08:00
|
|
|
ir.AnsiString += fmt.Sprintf("_%s", strings.Repeat(" ", ir.CursorPadding))
|
2021-04-04 11:28:41 -07:00
|
|
|
}
|
|
|
|
// add watermarks
|
2022-02-25 08:12:02 -08:00
|
|
|
ir.AnsiString += "\n\n\x1b[1mohmyposh.dev\x1b[22m"
|
2022-01-26 23:38:46 -08:00
|
|
|
if len(ir.Author) > 0 {
|
|
|
|
createdBy := fmt.Sprintf(" by \x1b[1m%s\x1b[22m", ir.Author)
|
|
|
|
ir.AnsiString += createdBy
|
2021-04-04 11:28:41 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ir *ImageRenderer) measureContent() (width, height float64) {
|
|
|
|
// get the longest line
|
|
|
|
linewidth := ir.calculateWidth()
|
|
|
|
// width, taken from the longest line
|
|
|
|
tmpDrawer := &font.Drawer{Face: ir.regular}
|
|
|
|
advance := tmpDrawer.MeasureString(strings.Repeat(" ", linewidth))
|
|
|
|
width = float64(advance >> 6)
|
|
|
|
// height, lines times font height and line spacing
|
2022-01-26 23:38:46 -08:00
|
|
|
height = float64(len(strings.Split(ir.AnsiString, "\n"))) * ir.fontHeight() * ir.lineSpacing
|
2021-04-04 11:28:41 -07:00
|
|
|
return width, height
|
|
|
|
}
|
|
|
|
|
2022-01-26 23:38:46 -08:00
|
|
|
func (ir *ImageRenderer) SavePNG() error {
|
2021-04-04 11:28:41 -07:00
|
|
|
var f = func(value float64) float64 { return ir.factor * value }
|
|
|
|
|
|
|
|
var (
|
|
|
|
corner = f(6)
|
|
|
|
radius = f(9)
|
|
|
|
distance = f(25)
|
|
|
|
)
|
|
|
|
|
|
|
|
contentWidth, contentHeight := ir.measureContent()
|
|
|
|
|
|
|
|
// Make sure the output window is big enough in case no content or very few
|
|
|
|
// content will be rendered
|
|
|
|
contentWidth = math.Max(contentWidth, 3*distance+3*radius)
|
|
|
|
|
|
|
|
marginX, marginY := ir.margin, ir.margin
|
|
|
|
paddingX, paddingY := ir.padding, ir.padding
|
|
|
|
|
|
|
|
xOffset := marginX
|
|
|
|
yOffset := marginY
|
|
|
|
titleOffset := f(40)
|
|
|
|
|
|
|
|
width := contentWidth + 2*marginX + 2*paddingX
|
|
|
|
height := contentHeight + 2*marginY + 2*paddingY + titleOffset
|
|
|
|
|
|
|
|
dc := gg.NewContext(int(width), int(height))
|
|
|
|
|
|
|
|
xOffset -= ir.shadowOffsetX / 2
|
|
|
|
yOffset -= ir.shadowOffsetY / 2
|
|
|
|
|
|
|
|
bc := gg.NewContext(int(width), int(height))
|
|
|
|
bc.DrawRoundedRectangle(xOffset+ir.shadowOffsetX, yOffset+ir.shadowOffsetY, width-2*marginX, height-2*marginY, corner)
|
|
|
|
bc.SetHexColor(ir.shadowBaseColor)
|
|
|
|
bc.Fill()
|
|
|
|
|
2021-11-26 05:11:31 -08:00
|
|
|
// var done = make(chan struct{}, ir.shadowRadius)
|
2022-03-29 10:59:33 -07:00
|
|
|
shadow, err := stackblur.Process(
|
2021-04-04 11:28:41 -07:00
|
|
|
bc.Image(),
|
|
|
|
uint32(ir.shadowRadius),
|
|
|
|
)
|
|
|
|
|
2021-11-26 05:11:31 -08:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// <-done
|
2021-04-04 11:28:41 -07:00
|
|
|
dc.DrawImage(shadow, 0, 0)
|
|
|
|
|
|
|
|
// Draw rounded rectangle with outline and three button to produce the
|
|
|
|
// impression of a window with controls and a content area
|
|
|
|
dc.DrawRoundedRectangle(xOffset, yOffset, width-2*marginX, height-2*marginY, corner)
|
2022-01-26 23:38:46 -08:00
|
|
|
dc.SetHexColor(ir.BgColor)
|
2021-04-04 11:28:41 -07:00
|
|
|
dc.Fill()
|
|
|
|
|
|
|
|
dc.DrawRoundedRectangle(xOffset, yOffset, width-2*marginX, height-2*marginY, corner)
|
|
|
|
dc.SetHexColor("#404040")
|
|
|
|
dc.SetLineWidth(f(1))
|
|
|
|
dc.Stroke()
|
|
|
|
|
|
|
|
for i, color := range []string{red, yellow, green} {
|
|
|
|
dc.DrawCircle(xOffset+paddingX+float64(i)*distance+f(4), yOffset+paddingY+f(4), radius)
|
|
|
|
dc.SetHexColor(color)
|
|
|
|
dc.Fill()
|
|
|
|
}
|
|
|
|
|
|
|
|
// Apply the actual text into the prepared content area of the window
|
|
|
|
var x, y float64 = xOffset + paddingX, yOffset + paddingY + titleOffset + ir.fontHeight()
|
|
|
|
|
2022-01-26 23:38:46 -08:00
|
|
|
for len(ir.AnsiString) != 0 {
|
2021-04-04 11:28:41 -07:00
|
|
|
if !ir.shouldPrint() {
|
|
|
|
continue
|
|
|
|
}
|
2022-01-26 23:38:46 -08:00
|
|
|
runes := []rune(ir.AnsiString)
|
2021-12-18 10:30:31 -08:00
|
|
|
if len(runes) == 0 {
|
|
|
|
continue
|
|
|
|
}
|
2021-04-04 11:28:41 -07:00
|
|
|
str := string(runes[0:1])
|
2022-01-26 23:38:46 -08:00
|
|
|
ir.AnsiString = string(runes[1:])
|
2021-04-04 11:28:41 -07:00
|
|
|
switch ir.style {
|
|
|
|
case bold:
|
|
|
|
dc.SetFontFace(ir.bold)
|
|
|
|
case italic:
|
|
|
|
dc.SetFontFace(ir.italic)
|
|
|
|
default:
|
|
|
|
dc.SetFontFace(ir.regular)
|
|
|
|
}
|
|
|
|
|
|
|
|
w, h := dc.MeasureString(str)
|
2021-09-18 06:37:15 -07:00
|
|
|
// 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])))
|
|
|
|
|
2021-04-04 11:28:41 -07:00
|
|
|
if ir.backgroundColor != nil {
|
|
|
|
dc.SetRGB255(ir.backgroundColor.r, ir.backgroundColor.g, ir.backgroundColor.b)
|
2021-09-18 06:37:15 -07:00
|
|
|
// 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)
|
2021-04-04 11:28:41 -07:00
|
|
|
dc.Fill()
|
|
|
|
}
|
|
|
|
if ir.foregroundColor != nil {
|
|
|
|
dc.SetRGB255(ir.foregroundColor.r, ir.foregroundColor.g, ir.foregroundColor.b)
|
|
|
|
} else {
|
|
|
|
dc.SetRGB255(ir.defaultForegroundColor.r, ir.defaultForegroundColor.g, ir.defaultForegroundColor.b)
|
|
|
|
}
|
|
|
|
|
|
|
|
if str == "\n" {
|
|
|
|
x = xOffset + paddingX
|
|
|
|
y += h * ir.lineSpacing
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
dc.DrawString(str, x, y)
|
|
|
|
|
|
|
|
if ir.style == underline {
|
|
|
|
dc.DrawLine(x, y+f(4), x+w, y+f(4))
|
|
|
|
dc.SetLineWidth(f(1))
|
|
|
|
dc.Stroke()
|
|
|
|
}
|
|
|
|
|
2022-05-22 21:33:32 -07:00
|
|
|
if ir.style == overline {
|
|
|
|
dc.DrawLine(x, y-f(22), x+w, y-f(22))
|
|
|
|
dc.SetLineWidth(f(1))
|
|
|
|
dc.Stroke()
|
|
|
|
}
|
|
|
|
|
2021-04-04 11:28:41 -07:00
|
|
|
x += w
|
|
|
|
}
|
|
|
|
|
2022-05-05 09:06:31 -07:00
|
|
|
return dc.SavePNG(ir.Path)
|
2021-04-04 11:28:41 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
func (ir *ImageRenderer) shouldPrint() bool {
|
2022-01-26 01:23:18 -08:00
|
|
|
for sequence, re := range ir.ansiSequenceRegexMap {
|
2022-01-26 23:38:46 -08:00
|
|
|
match := regex.FindNamedRegexMatch(re, ir.AnsiString)
|
2021-04-04 11:28:41 -07:00
|
|
|
if len(match) == 0 {
|
|
|
|
continue
|
|
|
|
}
|
2022-01-26 23:38:46 -08:00
|
|
|
ir.AnsiString = strings.TrimPrefix(ir.AnsiString, match[str])
|
2021-04-04 11:28:41 -07:00
|
|
|
switch sequence {
|
|
|
|
case invertedColor:
|
|
|
|
ir.foregroundColor = ir.defaultBackgroundColor
|
2021-05-21 11:01:08 -07:00
|
|
|
ir.backgroundColor = NewRGBColor(match[bg])
|
2021-04-04 11:28:41 -07:00
|
|
|
return false
|
|
|
|
case invertedColorSingle:
|
|
|
|
ir.foregroundColor = ir.defaultBackgroundColor
|
2022-01-26 04:09:21 -08:00
|
|
|
bgColor, _ := strconv.Atoi(match[bg])
|
|
|
|
bgColor += 10
|
|
|
|
ir.setBase16Color(fmt.Sprint(bgColor))
|
2021-04-04 11:28:41 -07:00
|
|
|
return false
|
|
|
|
case fullColor:
|
2021-05-21 11:01:08 -07:00
|
|
|
ir.foregroundColor = NewRGBColor(match[fg])
|
|
|
|
ir.backgroundColor = NewRGBColor(match[bg])
|
2021-04-04 11:28:41 -07:00
|
|
|
return false
|
|
|
|
case foreground:
|
2021-05-21 11:01:08 -07:00
|
|
|
ir.foregroundColor = NewRGBColor(match[fg])
|
2021-04-04 11:28:41 -07:00
|
|
|
return false
|
2022-05-09 18:47:27 -07:00
|
|
|
case background:
|
|
|
|
ir.backgroundColor = NewRGBColor(match[bg])
|
|
|
|
return false
|
2021-04-04 11:28:41 -07:00
|
|
|
case reset:
|
|
|
|
ir.foregroundColor = ir.defaultForegroundColor
|
|
|
|
ir.backgroundColor = nil
|
|
|
|
return false
|
2022-05-22 21:33:32 -07:00
|
|
|
case bold, italic, underline, overline:
|
2021-04-04 11:28:41 -07:00
|
|
|
ir.style = sequence
|
|
|
|
return false
|
2022-05-22 21:33:32 -07:00
|
|
|
case boldReset, italicReset, underlineReset, overlineReset:
|
2021-04-04 11:28:41 -07:00
|
|
|
ir.style = ""
|
|
|
|
return false
|
2022-11-24 14:45:44 -08:00
|
|
|
case strikethrough, strikethroughReset, left, osc99, osc7, osc51, lineChange, consoleTitle:
|
2021-04-04 11:28:41 -07:00
|
|
|
return false
|
|
|
|
case color16:
|
2022-05-09 18:47:27 -07:00
|
|
|
ir.setBase16Color(match[bc])
|
2021-04-04 11:28:41 -07:00
|
|
|
return false
|
|
|
|
case link:
|
2022-01-26 23:38:46 -08:00
|
|
|
ir.AnsiString = match[url] + ir.AnsiString
|
2021-04-04 11:28:41 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ir *ImageRenderer) setBase16Color(colorStr string) {
|
2022-01-26 04:09:21 -08:00
|
|
|
tempColor := ir.defaultForegroundColor
|
2021-04-04 11:28:41 -07:00
|
|
|
colorInt, err := strconv.Atoi(colorStr)
|
|
|
|
if err != nil {
|
2022-01-26 04:09:21 -08:00
|
|
|
ir.foregroundColor = tempColor
|
2021-04-04 11:28:41 -07:00
|
|
|
}
|
|
|
|
switch colorInt {
|
|
|
|
case 30, 40: // Black
|
2022-01-26 04:09:21 -08:00
|
|
|
tempColor = &RGB{1, 1, 1}
|
2021-04-04 11:28:41 -07:00
|
|
|
case 31, 41: // Red
|
2022-01-26 04:09:21 -08:00
|
|
|
tempColor = &RGB{222, 56, 43}
|
2021-04-04 11:28:41 -07:00
|
|
|
case 32, 42: // Green
|
2022-01-26 04:09:21 -08:00
|
|
|
tempColor = &RGB{57, 181, 74}
|
2021-04-04 11:28:41 -07:00
|
|
|
case 33, 43: // Yellow
|
2022-01-26 04:09:21 -08:00
|
|
|
tempColor = &RGB{255, 199, 6}
|
2021-04-04 11:28:41 -07:00
|
|
|
case 34, 44: // Blue
|
2022-01-26 04:09:21 -08:00
|
|
|
tempColor = &RGB{0, 111, 184}
|
2021-04-04 11:28:41 -07:00
|
|
|
case 35, 45: // Magenta
|
2022-01-26 04:09:21 -08:00
|
|
|
tempColor = &RGB{118, 38, 113}
|
2021-04-04 11:28:41 -07:00
|
|
|
case 36, 46: // Cyan
|
2022-01-26 04:09:21 -08:00
|
|
|
tempColor = &RGB{44, 181, 233}
|
2021-04-04 11:28:41 -07:00
|
|
|
case 37, 47: // White
|
2022-01-26 04:09:21 -08:00
|
|
|
tempColor = &RGB{204, 204, 204}
|
2021-04-04 11:28:41 -07:00
|
|
|
case 90, 100: // Bright Black (Gray)
|
2022-01-26 04:09:21 -08:00
|
|
|
tempColor = &RGB{128, 128, 128}
|
2021-04-04 11:28:41 -07:00
|
|
|
case 91, 101: // Bright Red
|
2022-01-26 04:09:21 -08:00
|
|
|
tempColor = &RGB{255, 0, 0}
|
2021-04-04 11:28:41 -07:00
|
|
|
case 92, 102: // Bright Green
|
2022-01-26 04:09:21 -08:00
|
|
|
tempColor = &RGB{0, 255, 0}
|
2021-04-04 11:28:41 -07:00
|
|
|
case 93, 103: // Bright Yellow
|
2022-01-26 04:09:21 -08:00
|
|
|
tempColor = &RGB{255, 255, 0}
|
2021-04-04 11:28:41 -07:00
|
|
|
case 94, 104: // Bright Blue
|
2022-01-26 04:09:21 -08:00
|
|
|
tempColor = &RGB{0, 0, 255}
|
2021-04-04 11:28:41 -07:00
|
|
|
case 95, 105: // Bright Magenta
|
2022-01-26 04:09:21 -08:00
|
|
|
tempColor = &RGB{255, 0, 255}
|
2021-04-04 11:28:41 -07:00
|
|
|
case 96, 106: // Bright Cyan
|
2022-01-26 04:09:21 -08:00
|
|
|
tempColor = &RGB{101, 194, 205}
|
2021-04-04 11:28:41 -07:00
|
|
|
case 97, 107: // Bright White
|
2022-01-26 04:09:21 -08:00
|
|
|
tempColor = &RGB{255, 255, 255}
|
2021-04-04 11:28:41 -07:00
|
|
|
}
|
|
|
|
if colorInt < 40 || (colorInt >= 90 && colorInt < 100) {
|
2022-01-26 04:09:21 -08:00
|
|
|
ir.foregroundColor = tempColor
|
2021-04-04 11:28:41 -07:00
|
|
|
return
|
|
|
|
}
|
2022-01-26 04:09:21 -08:00
|
|
|
ir.backgroundColor = tempColor
|
2021-04-04 11:28:41 -07:00
|
|
|
}
|