501 lines
14 KiB
Go
501 lines
14 KiB
Go
|
package html
|
||
|
|
||
|
import (
|
||
|
"fmt"
|
||
|
"html"
|
||
|
"io"
|
||
|
"sort"
|
||
|
"strings"
|
||
|
|
||
|
"github.com/alecthomas/chroma"
|
||
|
)
|
||
|
|
||
|
// Option sets an option of the HTML formatter.
|
||
|
type Option func(f *Formatter)
|
||
|
|
||
|
// Standalone configures the HTML formatter for generating a standalone HTML document.
|
||
|
func Standalone(b bool) Option { return func(f *Formatter) { f.standalone = b } }
|
||
|
|
||
|
// ClassPrefix sets the CSS class prefix.
|
||
|
func ClassPrefix(prefix string) Option { return func(f *Formatter) { f.prefix = prefix } }
|
||
|
|
||
|
// WithClasses emits HTML using CSS classes, rather than inline styles.
|
||
|
func WithClasses(b bool) Option { return func(f *Formatter) { f.Classes = b } }
|
||
|
|
||
|
// WithAllClasses disables an optimisation that omits redundant CSS classes.
|
||
|
func WithAllClasses(b bool) Option { return func(f *Formatter) { f.allClasses = b } }
|
||
|
|
||
|
// TabWidth sets the number of characters for a tab. Defaults to 8.
|
||
|
func TabWidth(width int) Option { return func(f *Formatter) { f.tabWidth = width } }
|
||
|
|
||
|
// PreventSurroundingPre prevents the surrounding pre tags around the generated code.
|
||
|
func PreventSurroundingPre(b bool) Option {
|
||
|
return func(f *Formatter) {
|
||
|
if b {
|
||
|
f.preWrapper = nopPreWrapper
|
||
|
} else {
|
||
|
f.preWrapper = defaultPreWrapper
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// WithPreWrapper allows control of the surrounding pre tags.
|
||
|
func WithPreWrapper(wrapper PreWrapper) Option {
|
||
|
return func(f *Formatter) {
|
||
|
f.preWrapper = wrapper
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// WrapLongLines wraps long lines.
|
||
|
func WrapLongLines(b bool) Option {
|
||
|
return func(f *Formatter) {
|
||
|
f.wrapLongLines = b
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// WithLineNumbers formats output with line numbers.
|
||
|
func WithLineNumbers(b bool) Option {
|
||
|
return func(f *Formatter) {
|
||
|
f.lineNumbers = b
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// LineNumbersInTable will, when combined with WithLineNumbers, separate the line numbers
|
||
|
// and code in table td's, which make them copy-and-paste friendly.
|
||
|
func LineNumbersInTable(b bool) Option {
|
||
|
return func(f *Formatter) {
|
||
|
f.lineNumbersInTable = b
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// LinkableLineNumbers decorates the line numbers HTML elements with an "id"
|
||
|
// attribute so they can be linked.
|
||
|
func LinkableLineNumbers(b bool, prefix string) Option {
|
||
|
return func(f *Formatter) {
|
||
|
f.linkableLineNumbers = b
|
||
|
f.lineNumbersIDPrefix = prefix
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// HighlightLines higlights the given line ranges with the Highlight style.
|
||
|
//
|
||
|
// A range is the beginning and ending of a range as 1-based line numbers, inclusive.
|
||
|
func HighlightLines(ranges [][2]int) Option {
|
||
|
return func(f *Formatter) {
|
||
|
f.highlightRanges = ranges
|
||
|
sort.Sort(f.highlightRanges)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// BaseLineNumber sets the initial number to start line numbering at. Defaults to 1.
|
||
|
func BaseLineNumber(n int) Option {
|
||
|
return func(f *Formatter) {
|
||
|
f.baseLineNumber = n
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// New HTML formatter.
|
||
|
func New(options ...Option) *Formatter {
|
||
|
f := &Formatter{
|
||
|
baseLineNumber: 1,
|
||
|
preWrapper: defaultPreWrapper,
|
||
|
}
|
||
|
for _, option := range options {
|
||
|
option(f)
|
||
|
}
|
||
|
return f
|
||
|
}
|
||
|
|
||
|
// PreWrapper defines the operations supported in WithPreWrapper.
|
||
|
type PreWrapper interface {
|
||
|
// Start is called to write a start <pre> element.
|
||
|
// The code flag tells whether this block surrounds
|
||
|
// highlighted code. This will be false when surrounding
|
||
|
// line numbers.
|
||
|
Start(code bool, styleAttr string) string
|
||
|
|
||
|
// End is called to write the end </pre> element.
|
||
|
End(code bool) string
|
||
|
}
|
||
|
|
||
|
type preWrapper struct {
|
||
|
start func(code bool, styleAttr string) string
|
||
|
end func(code bool) string
|
||
|
}
|
||
|
|
||
|
func (p preWrapper) Start(code bool, styleAttr string) string {
|
||
|
return p.start(code, styleAttr)
|
||
|
}
|
||
|
|
||
|
func (p preWrapper) End(code bool) string {
|
||
|
return p.end(code)
|
||
|
}
|
||
|
|
||
|
var (
|
||
|
nopPreWrapper = preWrapper{
|
||
|
start: func(code bool, styleAttr string) string { return "" },
|
||
|
end: func(code bool) string { return "" },
|
||
|
}
|
||
|
defaultPreWrapper = preWrapper{
|
||
|
start: func(code bool, styleAttr string) string {
|
||
|
if code {
|
||
|
return fmt.Sprintf(`<pre tabindex="0"%s><code>`, styleAttr)
|
||
|
}
|
||
|
|
||
|
return fmt.Sprintf(`<pre tabindex="0"%s>`, styleAttr)
|
||
|
},
|
||
|
end: func(code bool) string {
|
||
|
if code {
|
||
|
return `</code></pre>`
|
||
|
}
|
||
|
|
||
|
return `</pre>`
|
||
|
},
|
||
|
}
|
||
|
)
|
||
|
|
||
|
// Formatter that generates HTML.
|
||
|
type Formatter struct {
|
||
|
standalone bool
|
||
|
prefix string
|
||
|
Classes bool // Exported field to detect when classes are being used
|
||
|
allClasses bool
|
||
|
preWrapper PreWrapper
|
||
|
tabWidth int
|
||
|
wrapLongLines bool
|
||
|
lineNumbers bool
|
||
|
lineNumbersInTable bool
|
||
|
linkableLineNumbers bool
|
||
|
lineNumbersIDPrefix string
|
||
|
highlightRanges highlightRanges
|
||
|
baseLineNumber int
|
||
|
}
|
||
|
|
||
|
type highlightRanges [][2]int
|
||
|
|
||
|
func (h highlightRanges) Len() int { return len(h) }
|
||
|
func (h highlightRanges) Swap(i, j int) { h[i], h[j] = h[j], h[i] }
|
||
|
func (h highlightRanges) Less(i, j int) bool { return h[i][0] < h[j][0] }
|
||
|
|
||
|
func (f *Formatter) Format(w io.Writer, style *chroma.Style, iterator chroma.Iterator) (err error) {
|
||
|
return f.writeHTML(w, style, iterator.Tokens())
|
||
|
}
|
||
|
|
||
|
// We deliberately don't use html/template here because it is two orders of magnitude slower (benchmarked).
|
||
|
//
|
||
|
// OTOH we need to be super careful about correct escaping...
|
||
|
func (f *Formatter) writeHTML(w io.Writer, style *chroma.Style, tokens []chroma.Token) (err error) { // nolint: gocyclo
|
||
|
css := f.styleToCSS(style)
|
||
|
if !f.Classes {
|
||
|
for t, style := range css {
|
||
|
css[t] = compressStyle(style)
|
||
|
}
|
||
|
}
|
||
|
if f.standalone {
|
||
|
fmt.Fprint(w, "<html>\n")
|
||
|
if f.Classes {
|
||
|
fmt.Fprint(w, "<style type=\"text/css\">\n")
|
||
|
err = f.WriteCSS(w, style)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
fmt.Fprintf(w, "body { %s; }\n", css[chroma.Background])
|
||
|
fmt.Fprint(w, "</style>")
|
||
|
}
|
||
|
fmt.Fprintf(w, "<body%s>\n", f.styleAttr(css, chroma.Background))
|
||
|
}
|
||
|
|
||
|
wrapInTable := f.lineNumbers && f.lineNumbersInTable
|
||
|
|
||
|
lines := chroma.SplitTokensIntoLines(tokens)
|
||
|
lineDigits := len(fmt.Sprintf("%d", f.baseLineNumber+len(lines)-1))
|
||
|
highlightIndex := 0
|
||
|
|
||
|
if wrapInTable {
|
||
|
// List line numbers in its own <td>
|
||
|
fmt.Fprintf(w, "<div%s>\n", f.styleAttr(css, chroma.PreWrapper))
|
||
|
fmt.Fprintf(w, "<table%s><tr>", f.styleAttr(css, chroma.LineTable))
|
||
|
fmt.Fprintf(w, "<td%s>\n", f.styleAttr(css, chroma.LineTableTD))
|
||
|
fmt.Fprintf(w, f.preWrapper.Start(false, f.styleAttr(css, chroma.PreWrapper)))
|
||
|
for index := range lines {
|
||
|
line := f.baseLineNumber + index
|
||
|
highlight, next := f.shouldHighlight(highlightIndex, line)
|
||
|
if next {
|
||
|
highlightIndex++
|
||
|
}
|
||
|
if highlight {
|
||
|
fmt.Fprintf(w, "<span%s>", f.styleAttr(css, chroma.LineHighlight))
|
||
|
}
|
||
|
|
||
|
fmt.Fprintf(w, "<span%s%s>%s\n</span>", f.styleAttr(css, chroma.LineNumbersTable), f.lineIDAttribute(line), f.lineTitleWithLinkIfNeeded(lineDigits, line))
|
||
|
|
||
|
if highlight {
|
||
|
fmt.Fprintf(w, "</span>")
|
||
|
}
|
||
|
}
|
||
|
fmt.Fprint(w, f.preWrapper.End(false))
|
||
|
fmt.Fprint(w, "</td>\n")
|
||
|
fmt.Fprintf(w, "<td%s>\n", f.styleAttr(css, chroma.LineTableTD, "width:100%"))
|
||
|
}
|
||
|
|
||
|
fmt.Fprintf(w, f.preWrapper.Start(true, f.styleAttr(css, chroma.PreWrapper)))
|
||
|
|
||
|
highlightIndex = 0
|
||
|
for index, tokens := range lines {
|
||
|
// 1-based line number.
|
||
|
line := f.baseLineNumber + index
|
||
|
highlight, next := f.shouldHighlight(highlightIndex, line)
|
||
|
if next {
|
||
|
highlightIndex++
|
||
|
}
|
||
|
|
||
|
// Start of Line
|
||
|
fmt.Fprint(w, `<span`)
|
||
|
if highlight {
|
||
|
// Line + LineHighlight
|
||
|
if f.Classes {
|
||
|
fmt.Fprintf(w, ` class="%s %s"`, f.class(chroma.Line), f.class(chroma.LineHighlight))
|
||
|
} else {
|
||
|
fmt.Fprintf(w, ` style="%s %s"`, css[chroma.Line], css[chroma.LineHighlight])
|
||
|
}
|
||
|
fmt.Fprint(w, `>`)
|
||
|
} else {
|
||
|
fmt.Fprintf(w, "%s>", f.styleAttr(css, chroma.Line))
|
||
|
}
|
||
|
|
||
|
// Line number
|
||
|
if f.lineNumbers && !wrapInTable {
|
||
|
fmt.Fprintf(w, "<span%s%s>%s</span>", f.styleAttr(css, chroma.LineNumbers), f.lineIDAttribute(line), f.lineTitleWithLinkIfNeeded(lineDigits, line))
|
||
|
}
|
||
|
|
||
|
fmt.Fprintf(w, `<span%s>`, f.styleAttr(css, chroma.CodeLine))
|
||
|
|
||
|
for _, token := range tokens {
|
||
|
html := html.EscapeString(token.String())
|
||
|
attr := f.styleAttr(css, token.Type)
|
||
|
if attr != "" {
|
||
|
html = fmt.Sprintf("<span%s>%s</span>", attr, html)
|
||
|
}
|
||
|
fmt.Fprint(w, html)
|
||
|
}
|
||
|
|
||
|
fmt.Fprint(w, `</span>`) // End of CodeLine
|
||
|
|
||
|
fmt.Fprint(w, `</span>`) // End of Line
|
||
|
}
|
||
|
|
||
|
fmt.Fprintf(w, f.preWrapper.End(true))
|
||
|
|
||
|
if wrapInTable {
|
||
|
fmt.Fprint(w, "</td></tr></table>\n")
|
||
|
fmt.Fprint(w, "</div>\n")
|
||
|
}
|
||
|
|
||
|
if f.standalone {
|
||
|
fmt.Fprint(w, "\n</body>\n")
|
||
|
fmt.Fprint(w, "</html>\n")
|
||
|
}
|
||
|
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (f *Formatter) lineIDAttribute(line int) string {
|
||
|
if !f.linkableLineNumbers {
|
||
|
return ""
|
||
|
}
|
||
|
return fmt.Sprintf(" id=\"%s\"", f.lineID(line))
|
||
|
}
|
||
|
|
||
|
func (f *Formatter) lineTitleWithLinkIfNeeded(lineDigits, line int) string {
|
||
|
title := fmt.Sprintf("%*d", lineDigits, line)
|
||
|
if !f.linkableLineNumbers {
|
||
|
return title
|
||
|
}
|
||
|
return fmt.Sprintf("<a style=\"outline: none; text-decoration:none; color:inherit\" href=\"#%s\">%s</a>", f.lineID(line), title)
|
||
|
}
|
||
|
|
||
|
func (f *Formatter) lineID(line int) string {
|
||
|
return fmt.Sprintf("%s%d", f.lineNumbersIDPrefix, line)
|
||
|
}
|
||
|
|
||
|
func (f *Formatter) shouldHighlight(highlightIndex, line int) (bool, bool) {
|
||
|
next := false
|
||
|
for highlightIndex < len(f.highlightRanges) && line > f.highlightRanges[highlightIndex][1] {
|
||
|
highlightIndex++
|
||
|
next = true
|
||
|
}
|
||
|
if highlightIndex < len(f.highlightRanges) {
|
||
|
hrange := f.highlightRanges[highlightIndex]
|
||
|
if line >= hrange[0] && line <= hrange[1] {
|
||
|
return true, next
|
||
|
}
|
||
|
}
|
||
|
return false, next
|
||
|
}
|
||
|
|
||
|
func (f *Formatter) class(t chroma.TokenType) string {
|
||
|
for t != 0 {
|
||
|
if cls, ok := chroma.StandardTypes[t]; ok {
|
||
|
if cls != "" {
|
||
|
return f.prefix + cls
|
||
|
}
|
||
|
return ""
|
||
|
}
|
||
|
t = t.Parent()
|
||
|
}
|
||
|
if cls := chroma.StandardTypes[t]; cls != "" {
|
||
|
return f.prefix + cls
|
||
|
}
|
||
|
return ""
|
||
|
}
|
||
|
|
||
|
func (f *Formatter) styleAttr(styles map[chroma.TokenType]string, tt chroma.TokenType, extraCSS ...string) string {
|
||
|
if f.Classes {
|
||
|
cls := f.class(tt)
|
||
|
if cls == "" {
|
||
|
return ""
|
||
|
}
|
||
|
return fmt.Sprintf(` class="%s"`, cls)
|
||
|
}
|
||
|
if _, ok := styles[tt]; !ok {
|
||
|
tt = tt.SubCategory()
|
||
|
if _, ok := styles[tt]; !ok {
|
||
|
tt = tt.Category()
|
||
|
if _, ok := styles[tt]; !ok {
|
||
|
return ""
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
css := []string{styles[tt]}
|
||
|
css = append(css, extraCSS...)
|
||
|
return fmt.Sprintf(` style="%s"`, strings.Join(css, ";"))
|
||
|
}
|
||
|
|
||
|
func (f *Formatter) tabWidthStyle() string {
|
||
|
if f.tabWidth != 0 && f.tabWidth != 8 {
|
||
|
return fmt.Sprintf("; -moz-tab-size: %[1]d; -o-tab-size: %[1]d; tab-size: %[1]d", f.tabWidth)
|
||
|
}
|
||
|
return ""
|
||
|
}
|
||
|
|
||
|
// WriteCSS writes CSS style definitions (without any surrounding HTML).
|
||
|
func (f *Formatter) WriteCSS(w io.Writer, style *chroma.Style) error {
|
||
|
css := f.styleToCSS(style)
|
||
|
// Special-case background as it is mapped to the outer ".chroma" class.
|
||
|
if _, err := fmt.Fprintf(w, "/* %s */ .%sbg { %s }\n", chroma.Background, f.prefix, css[chroma.Background]); err != nil {
|
||
|
return err
|
||
|
}
|
||
|
// Special-case PreWrapper as it is the ".chroma" class.
|
||
|
if _, err := fmt.Fprintf(w, "/* %s */ .%schroma { %s }\n", chroma.PreWrapper, f.prefix, css[chroma.PreWrapper]); err != nil {
|
||
|
return err
|
||
|
}
|
||
|
// Special-case code column of table to expand width.
|
||
|
if f.lineNumbers && f.lineNumbersInTable {
|
||
|
if _, err := fmt.Fprintf(w, "/* %s */ .%schroma .%s:last-child { width: 100%%; }",
|
||
|
chroma.LineTableTD, f.prefix, f.class(chroma.LineTableTD)); err != nil {
|
||
|
return err
|
||
|
}
|
||
|
}
|
||
|
// Special-case line number highlighting when targeted.
|
||
|
if f.lineNumbers || f.lineNumbersInTable {
|
||
|
targetedLineCSS := StyleEntryToCSS(style.Get(chroma.LineHighlight))
|
||
|
for _, tt := range []chroma.TokenType{chroma.LineNumbers, chroma.LineNumbersTable} {
|
||
|
fmt.Fprintf(w, "/* %s targeted by URL anchor */ .%schroma .%s:target { %s }\n", tt, f.prefix, f.class(tt), targetedLineCSS)
|
||
|
}
|
||
|
}
|
||
|
tts := []int{}
|
||
|
for tt := range css {
|
||
|
tts = append(tts, int(tt))
|
||
|
}
|
||
|
sort.Ints(tts)
|
||
|
for _, ti := range tts {
|
||
|
tt := chroma.TokenType(ti)
|
||
|
switch tt {
|
||
|
case chroma.Background, chroma.PreWrapper:
|
||
|
continue
|
||
|
}
|
||
|
class := f.class(tt)
|
||
|
if class == "" {
|
||
|
continue
|
||
|
}
|
||
|
styles := css[tt]
|
||
|
if _, err := fmt.Fprintf(w, "/* %s */ .%schroma .%s { %s }\n", tt, f.prefix, class, styles); err != nil {
|
||
|
return err
|
||
|
}
|
||
|
}
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (f *Formatter) styleToCSS(style *chroma.Style) map[chroma.TokenType]string {
|
||
|
classes := map[chroma.TokenType]string{}
|
||
|
bg := style.Get(chroma.Background)
|
||
|
// Convert the style.
|
||
|
for t := range chroma.StandardTypes {
|
||
|
entry := style.Get(t)
|
||
|
if t != chroma.Background {
|
||
|
entry = entry.Sub(bg)
|
||
|
}
|
||
|
if !f.allClasses && entry.IsZero() {
|
||
|
continue
|
||
|
}
|
||
|
classes[t] = StyleEntryToCSS(entry)
|
||
|
}
|
||
|
classes[chroma.Background] += f.tabWidthStyle()
|
||
|
classes[chroma.PreWrapper] += classes[chroma.Background] + `;`
|
||
|
// Make PreWrapper a grid to show highlight style with full width.
|
||
|
if len(f.highlightRanges) > 0 {
|
||
|
classes[chroma.PreWrapper] += `display: grid;`
|
||
|
}
|
||
|
// Make PreWrapper wrap long lines.
|
||
|
if f.wrapLongLines {
|
||
|
classes[chroma.PreWrapper] += `white-space: pre-wrap; word-break: break-word;`
|
||
|
}
|
||
|
lineNumbersStyle := `white-space: pre; user-select: none; margin-right: 0.4em; padding: 0 0.4em 0 0.4em;`
|
||
|
// All rules begin with default rules followed by user provided rules
|
||
|
classes[chroma.Line] = `display: flex;` + classes[chroma.Line]
|
||
|
classes[chroma.LineNumbers] = lineNumbersStyle + classes[chroma.LineNumbers]
|
||
|
classes[chroma.LineNumbersTable] = lineNumbersStyle + classes[chroma.LineNumbersTable]
|
||
|
classes[chroma.LineTable] = "border-spacing: 0; padding: 0; margin: 0; border: 0;" + classes[chroma.LineTable]
|
||
|
classes[chroma.LineTableTD] = "vertical-align: top; padding: 0; margin: 0; border: 0;" + classes[chroma.LineTableTD]
|
||
|
return classes
|
||
|
}
|
||
|
|
||
|
// StyleEntryToCSS converts a chroma.StyleEntry to CSS attributes.
|
||
|
func StyleEntryToCSS(e chroma.StyleEntry) string {
|
||
|
styles := []string{}
|
||
|
if e.Colour.IsSet() {
|
||
|
styles = append(styles, "color: "+e.Colour.String())
|
||
|
}
|
||
|
if e.Background.IsSet() {
|
||
|
styles = append(styles, "background-color: "+e.Background.String())
|
||
|
}
|
||
|
if e.Bold == chroma.Yes {
|
||
|
styles = append(styles, "font-weight: bold")
|
||
|
}
|
||
|
if e.Italic == chroma.Yes {
|
||
|
styles = append(styles, "font-style: italic")
|
||
|
}
|
||
|
if e.Underline == chroma.Yes {
|
||
|
styles = append(styles, "text-decoration: underline")
|
||
|
}
|
||
|
return strings.Join(styles, "; ")
|
||
|
}
|
||
|
|
||
|
// Compress CSS attributes - remove spaces, transform 6-digit colours to 3.
|
||
|
func compressStyle(s string) string {
|
||
|
parts := strings.Split(s, ";")
|
||
|
out := []string{}
|
||
|
for _, p := range parts {
|
||
|
p = strings.Join(strings.Fields(p), " ")
|
||
|
p = strings.Replace(p, ": ", ":", 1)
|
||
|
if strings.Contains(p, "#") {
|
||
|
c := p[len(p)-6:]
|
||
|
if c[0] == c[1] && c[2] == c[3] && c[4] == c[5] {
|
||
|
p = p[:len(p)-6] + c[0:1] + c[2:3] + c[4:5]
|
||
|
}
|
||
|
}
|
||
|
out = append(out, p)
|
||
|
}
|
||
|
return strings.Join(out, ";")
|
||
|
}
|