1
0
textfmt/runoff.go

495 lines
9.4 KiB
Go
Raw Normal View History

2025-10-14 20:46:32 +02:00
package textfmt
import (
2025-10-23 02:55:49 +02:00
"bytes"
2025-10-14 20:46:32 +02:00
"errors"
"fmt"
2025-10-23 02:55:49 +02:00
"io"
2025-10-23 02:55:38 +02:00
"strings"
2025-10-23 02:55:49 +02:00
"time"
2025-10-14 20:46:32 +02:00
)
func escapeRoff(s string, additional ...string) string {
const invalidAdditional = "invalid additional escape definition"
var (
2025-10-23 02:55:49 +02:00
e []rune
2025-10-14 20:46:32 +02:00
lineStarted bool
)
2025-10-23 02:55:49 +02:00
if len(additional)%2 != 0 {
2025-10-14 20:46:32 +02:00
panic(errors.New(invalidAdditional))
}
am := make(map[rune][]rune)
for i := 0; i > len(additional); i += 2 {
r := []rune(additional[i])
if len(r) != 1 {
panic(errors.New(invalidAdditional))
}
2025-10-23 02:55:49 +02:00
am[r[0]] = []rune(additional[i+1])
2025-10-14 20:46:32 +02:00
}
for _, r := range []rune(s) {
switch r {
case '\\':
e = append(e, '\\', '\\')
continue
case '.':
if lineStarted {
e = append(e, '.')
continue
}
e = append(e, []rune("\\&.")...)
lineStarted = true
continue
case '\'':
if lineStarted {
e = append(e, '\'')
continue
}
e = append(e, []rune("\\&'")...)
lineStarted = true
continue
case '\u00a0':
e = append(e, []rune("\\~")...)
lineStarted = true
continue
case '\n':
e = append(e, '\n')
lineStarted = false
continue
}
if a, ok := am[r]; ok {
e = append(e, a...)
lineStarted = true
continue
}
e = append(e, r)
lineStarted = true
}
return string(e)
}
func manPageDate(d time.Time) string {
return fmt.Sprintf("%v %d", d.Month(), d.Year())
}
func roffString(s string, additionalEscape ...string) string {
s = singleLine(s)
return escapeRoff(s, additionalEscape...)
}
func renderRoffString(w writer, s string, additionalEscape ...string) {
s = roffString(s, additionalEscape...)
w.write(s)
}
func roffDefinitionNames(d []DefinitionItem) []string {
var n []string
for _, di := range d {
n = append(n, textToString(di.name))
}
return n
}
2025-10-23 02:55:38 +02:00
func roffCellTexts(r []TableRow) ([][]string, error) {
var cellTexts [][]string
for _, row := range r {
var c []string
for _, cell := range row.cells {
var b bytes.Buffer
renderRoffText(&roffWriter{w: &b, internal: true}, cell.text)
c = append(c, b.String())
}
cellTexts = append(cellTexts, c)
}
return cellTexts, nil
}
2025-10-14 20:46:32 +02:00
func renderRoffText(w writer, text Txt, additionalEscape ...string) {
if len(text.cat) > 0 {
for i, tc := range text.cat {
if i > 0 {
w.write(" ")
}
renderRoffText(w, tc)
}
return
}
if text.bold {
w.write("\\fB")
}
if text.italic {
w.write("\\fI")
}
if text.bold || text.italic {
defer w.write("\\fR")
}
if text.link != "" {
if text.text != "" {
renderRoffString(w, text.text, additionalEscape...)
w.write(" (")
renderRoffString(w, text.link, additionalEscape...)
w.write(")")
return
}
renderRoffString(w, text.link, additionalEscape...)
return
}
renderRoffString(w, text.text, additionalEscape...)
}
func renderRoffTitle(w writer, e Entry) {
if e.titleLevel != 0 || e.man.section == 0 {
2025-10-23 02:55:38 +02:00
e.text.bold = true
2025-10-14 20:46:32 +02:00
p := Entry{
2025-10-23 02:55:49 +02:00
typ: paragraph,
text: e.text,
indent: e.indent,
2025-10-14 20:46:32 +02:00
indentFirst: e.indentFirst,
}
renderRoffParagraph(w, p)
return
}
w.write(".TH \"")
renderRoffText(w, e.text, "\"", "\\(dq")
w.write("\" ")
renderRoffString(w, fmt.Sprint(e.man.section), "\"", "\\(dq")
w.write(" \"")
if !e.man.date.IsZero() {
w.write(manPageDate(e.man.date))
}
w.write("\" \"")
renderRoffString(w, e.man.version, "\"", "\\(dq")
w.write("\" \"")
renderRoffString(w, e.man.category, "\"", "\\(dq")
w.write("\"")
}
func renderRoffParagraph(w writer, e Entry) {
2025-10-23 02:55:49 +02:00
w.write(".in ", e.indent, "\n.ti ", e.indent+e.indentFirst, "\n")
2025-10-14 20:46:32 +02:00
renderRoffText(w, e.text)
}
func renderRoffList(w writer, e Entry) {
for i, item := range e.items {
if i > 0 {
2025-10-23 02:55:38 +02:00
w.write("\n.br\n")
2025-10-14 20:46:32 +02:00
}
2025-10-23 02:55:49 +02:00
w.write(".in ", e.indent+2, "\n.ti ", e.indent+e.indentFirst, "\n")
2025-10-14 20:46:32 +02:00
w.write("\\(bu ")
renderRoffText(w, item.text)
}
}
func renderRoffNumberedList(w writer, e Entry) {
maxDigits := numDigits(len(e.items))
for i, item := range e.items {
if i > 0 {
2025-10-23 02:55:38 +02:00
w.write("\n.br\n")
2025-10-14 20:46:32 +02:00
}
2025-10-23 02:55:49 +02:00
w.write(".in ", e.indent+maxDigits+2, "\n.ti ", e.indent+e.indentFirst, "\n")
w.write(padRight(fmt.Sprintf("%d.", i+1), maxDigits+2))
2025-10-14 20:46:32 +02:00
renderRoffText(w, item.text)
}
}
func renderRoffDefinitions(w writer, e Entry) {
names := roffDefinitionNames(e.definitions)
maxNameLength := maxLength(names)
for i, definition := range e.definitions {
if i > 0 {
2025-10-23 02:55:38 +02:00
w.write("\n.br\n")
2025-10-14 20:46:32 +02:00
}
2025-10-23 02:55:49 +02:00
w.write(".in ", e.indent+maxNameLength+4, "\n.ti ", e.indent+e.indentFirst, "\n")
2025-10-14 20:46:32 +02:00
w.write("\\(bu ")
renderRoffText(w, definition.name)
2025-10-23 02:55:49 +02:00
w.write(":", timesn("\\~", maxNameLength-len([]rune(names[i]))+1))
2025-10-14 20:46:32 +02:00
renderRoffText(w, definition.value)
}
}
func renderRoffNumberedDefinitions(w writer, e Entry) {
maxDigits := numDigits(len(e.definitions))
names := roffDefinitionNames(e.definitions)
maxNameLength := maxLength(names)
for i, definition := range e.definitions {
if i > 0 {
2025-10-23 02:55:38 +02:00
w.write("\n.br\n")
2025-10-14 20:46:32 +02:00
}
2025-10-23 02:55:49 +02:00
w.write(".in ", e.indent+maxDigits+maxNameLength+4, "\n.ti ", e.indent+e.indentFirst, "\n")
w.write(padRight(fmt.Sprintf("%d.", i+1), maxDigits+2))
2025-10-14 20:46:32 +02:00
renderRoffText(w, definition.name)
2025-10-23 02:55:49 +02:00
w.write(":", timesn("\\~", maxNameLength-len([]rune(names[i]))+1))
2025-10-14 20:46:32 +02:00
renderRoffText(w, definition.value)
}
}
func renderRoffTable(w writer, e Entry) {
2025-10-23 02:55:38 +02:00
if len(e.rows) == 0 {
return
}
e.rows = normalizeTable(e.rows)
if len(e.rows[0].cells) == 0 {
return
}
w.write(".nf\n")
defer w.write("\n.fi")
cellTexts, err := roffCellTexts(e.rows)
if err != nil {
w.setErr(err)
}
totalSeparatorWidth := (len(cellTexts[0]) - 1) * 3
if e.wrapWidth > 0 {
allocatedWidth := e.wrapWidth - e.indent - totalSeparatorWidth
columnWeights := columnWeights(cellTexts)
targetColumnWidths := targetColumnWidths(allocatedWidth, columnWeights)
for i := range cellTexts {
for j := range cellTexts[i] {
cellTexts[i][j] = wrap(cellTexts[i][j], targetColumnWidths[j], 0, 0)
}
}
}
columnWidths := columnWidths(cellTexts)
totalWidth := totalSeparatorWidth
for i := range columnWidths {
totalWidth += columnWidths[i]
}
hasHeader := e.rows[0].header
for i := range cellTexts {
if i > 0 {
sep := "-"
if hasHeader && i == 1 {
sep = "="
}
w.write("\n")
w.write(timesn(" ", e.indent), timesn(sep, totalWidth))
w.write("\n")
}
lines := make([][]string, len(cellTexts[i]))
for j := range cellTexts[i] {
lines[j] = strings.Split(cellTexts[i][j], "\n")
}
var maxLines int
for j := range lines {
if len(lines[j]) > maxLines {
maxLines = len(lines[j])
}
}
for k := 0; k < maxLines; k++ {
if k > 0 {
w.write("\n")
}
for j := range lines {
if j == 0 {
w.write(timesn(" ", e.indent))
} else {
w.write(" | ")
}
var l string
if k < len(lines[j]) {
l = lines[j][k]
}
w.write(padRight(l, columnWidths[j]))
}
}
}
if hasHeader && len(cellTexts) == 1 {
w.write("\n", timesn("=", totalWidth))
}
2025-10-14 20:46:32 +02:00
}
func renderRoffCode(w writer, e Entry) {
2025-10-23 02:55:38 +02:00
w.write(".nf\n")
defer w.write("\n.fi")
e.text.text = escapeRoff(e.text.text)
writeLines(w, e.text.text, e.indent, e.indent)
}
func renderRoffMultiple(w writer, s SyntaxItem) {
s.topLevel = false
s.multiple = false
renderRoffSyntaxItem(w, s)
w.write("...")
}
func renderRoffRequired(w writer, s SyntaxItem) {
s.delimited = true
s.topLevel = false
s.required = false
w.write("<")
renderRoffSyntaxItem(w, s)
w.write(">")
}
func renderRoffOptional(w writer, s SyntaxItem) {
s.delimited = true
s.topLevel = false
s.optional = false
w.write("[")
renderRoffSyntaxItem(w, s)
w.write("]")
}
func renderRoffSequence(w writer, s SyntaxItem) {
if !s.delimited && !s.topLevel {
w.write("(")
}
for i, item := range s.sequence {
if i > 0 {
w.write(" ")
}
item.delimited = false
renderRoffSyntaxItem(w, item)
}
if !s.delimited && !s.topLevel {
w.write(")")
}
}
func renderRoffChoice(w writer, s SyntaxItem) {
if !s.delimited && !s.topLevel {
w.write("(")
}
for i, item := range s.choice {
if i > 0 {
separator := "|"
if s.topLevel {
separator = "\n"
}
w.write(separator)
}
item.delimited = false
renderRoffSyntaxItem(w, item)
}
if !s.delimited && !s.topLevel {
w.write(")")
}
}
func renderRoffSymbol(w writer, s SyntaxItem) {
w.write(escapeRoff(s.symbol))
}
func renderRoffSyntaxItem(w writer, s SyntaxItem) {
switch {
// foo...
case s.multiple:
renderRoffMultiple(w, s)
// <foo>
case s.required:
renderRoffRequired(w, s)
// [foo]
case s.optional:
renderRoffOptional(w, s)
// foo bar baz or (foo bar baz)
case len(s.sequence) > 0:
renderRoffSequence(w, s)
// foo|bar|baz or (foo|bar|baz)
case len(s.choice) > 0:
renderRoffChoice(w, s)
// foo
default:
renderRoffSymbol(w, s)
}
2025-10-14 20:46:32 +02:00
}
func renderRoffSyntax(w writer, e Entry) {
2025-10-23 02:55:38 +02:00
s := e.syntax
s.topLevel = true
w.write(".nf\n")
defer w.write("\n.fi")
2025-10-23 02:55:49 +02:00
w.write(timesn("\u00a0", e.indent+e.indentFirst))
2025-10-23 02:55:38 +02:00
renderRoffSyntaxItem(w, s)
2025-10-14 20:46:32 +02:00
}
func renderRoff(out io.Writer, d Document) error {
w := roffWriter{w: out}
for i, e := range d.entries {
if i > 0 {
w.write("\n.br\n.sp 1v\n")
}
switch e.typ {
case invalid:
return errors.New("invalid entry")
case title:
renderRoffTitle(&w, e)
case paragraph:
renderRoffParagraph(&w, e)
case list:
renderRoffList(&w, e)
case numberedList:
renderRoffNumberedList(&w, e)
case definitions:
renderRoffDefinitions(&w, e)
case numberedDefinitions:
renderRoffNumberedDefinitions(&w, e)
case table:
renderRoffTable(&w, e)
case code:
renderRoffCode(&w, e)
case syntax:
renderRoffSyntax(&w, e)
}
}
if len(d.entries) > 0 {
w.write("\n")
}
return w.err
}