vendor/github.com/cpuguy83/go-md2man/md2man/roff.go
changeset 251 1c52a0eeb952
equal deleted inserted replaced
250:c040f992052f 251:1c52a0eeb952
       
     1 package md2man
       
     2 
       
     3 import (
       
     4 	"bytes"
       
     5 	"fmt"
       
     6 	"html"
       
     7 	"strings"
       
     8 
       
     9 	"github.com/russross/blackfriday"
       
    10 )
       
    11 
       
    12 type roffRenderer struct {
       
    13 	ListCounters []int
       
    14 }
       
    15 
       
    16 // RoffRenderer creates a new blackfriday Renderer for generating roff documents
       
    17 // from markdown
       
    18 func RoffRenderer(flags int) blackfriday.Renderer {
       
    19 	return &roffRenderer{}
       
    20 }
       
    21 
       
    22 func (r *roffRenderer) GetFlags() int {
       
    23 	return 0
       
    24 }
       
    25 
       
    26 func (r *roffRenderer) TitleBlock(out *bytes.Buffer, text []byte) {
       
    27 	out.WriteString(".TH ")
       
    28 
       
    29 	splitText := bytes.Split(text, []byte("\n"))
       
    30 	for i, line := range splitText {
       
    31 		line = bytes.TrimPrefix(line, []byte("% "))
       
    32 		if i == 0 {
       
    33 			line = bytes.Replace(line, []byte("("), []byte("\" \""), 1)
       
    34 			line = bytes.Replace(line, []byte(")"), []byte("\" \""), 1)
       
    35 		}
       
    36 		line = append([]byte("\""), line...)
       
    37 		line = append(line, []byte("\" ")...)
       
    38 		out.Write(line)
       
    39 	}
       
    40 	out.WriteString("\n")
       
    41 
       
    42 	// disable hyphenation
       
    43 	out.WriteString(".nh\n")
       
    44 	// disable justification (adjust text to left margin only)
       
    45 	out.WriteString(".ad l\n")
       
    46 }
       
    47 
       
    48 func (r *roffRenderer) BlockCode(out *bytes.Buffer, text []byte, lang string) {
       
    49 	out.WriteString("\n.PP\n.RS\n\n.nf\n")
       
    50 	escapeSpecialChars(out, text)
       
    51 	out.WriteString("\n.fi\n.RE\n")
       
    52 }
       
    53 
       
    54 func (r *roffRenderer) BlockQuote(out *bytes.Buffer, text []byte) {
       
    55 	out.WriteString("\n.PP\n.RS\n")
       
    56 	out.Write(text)
       
    57 	out.WriteString("\n.RE\n")
       
    58 }
       
    59 
       
    60 func (r *roffRenderer) BlockHtml(out *bytes.Buffer, text []byte) { // nolint: golint
       
    61 	out.Write(text)
       
    62 }
       
    63 
       
    64 func (r *roffRenderer) Header(out *bytes.Buffer, text func() bool, level int, id string) {
       
    65 	marker := out.Len()
       
    66 
       
    67 	switch {
       
    68 	case marker == 0:
       
    69 		// This is the doc header
       
    70 		out.WriteString(".TH ")
       
    71 	case level == 1:
       
    72 		out.WriteString("\n\n.SH ")
       
    73 	case level == 2:
       
    74 		out.WriteString("\n.SH ")
       
    75 	default:
       
    76 		out.WriteString("\n.SS ")
       
    77 	}
       
    78 
       
    79 	if !text() {
       
    80 		out.Truncate(marker)
       
    81 		return
       
    82 	}
       
    83 }
       
    84 
       
    85 func (r *roffRenderer) HRule(out *bytes.Buffer) {
       
    86 	out.WriteString("\n.ti 0\n\\l'\\n(.lu'\n")
       
    87 }
       
    88 
       
    89 func (r *roffRenderer) List(out *bytes.Buffer, text func() bool, flags int) {
       
    90 	marker := out.Len()
       
    91 	r.ListCounters = append(r.ListCounters, 1)
       
    92 	out.WriteString("\n.RS\n")
       
    93 	if !text() {
       
    94 		out.Truncate(marker)
       
    95 		return
       
    96 	}
       
    97 	r.ListCounters = r.ListCounters[:len(r.ListCounters)-1]
       
    98 	out.WriteString("\n.RE\n")
       
    99 }
       
   100 
       
   101 func (r *roffRenderer) ListItem(out *bytes.Buffer, text []byte, flags int) {
       
   102 	if flags&blackfriday.LIST_TYPE_ORDERED != 0 {
       
   103 		out.WriteString(fmt.Sprintf(".IP \"%3d.\" 5\n", r.ListCounters[len(r.ListCounters)-1]))
       
   104 		r.ListCounters[len(r.ListCounters)-1]++
       
   105 	} else {
       
   106 		out.WriteString(".IP \\(bu 2\n")
       
   107 	}
       
   108 	out.Write(text)
       
   109 	out.WriteString("\n")
       
   110 }
       
   111 
       
   112 func (r *roffRenderer) Paragraph(out *bytes.Buffer, text func() bool) {
       
   113 	marker := out.Len()
       
   114 	out.WriteString("\n.PP\n")
       
   115 	if !text() {
       
   116 		out.Truncate(marker)
       
   117 		return
       
   118 	}
       
   119 	if marker != 0 {
       
   120 		out.WriteString("\n")
       
   121 	}
       
   122 }
       
   123 
       
   124 func (r *roffRenderer) Table(out *bytes.Buffer, header []byte, body []byte, columnData []int) {
       
   125 	out.WriteString("\n.TS\nallbox;\n")
       
   126 
       
   127 	maxDelims := 0
       
   128 	lines := strings.Split(strings.TrimRight(string(header), "\n")+"\n"+strings.TrimRight(string(body), "\n"), "\n")
       
   129 	for _, w := range lines {
       
   130 		curDelims := strings.Count(w, "\t")
       
   131 		if curDelims > maxDelims {
       
   132 			maxDelims = curDelims
       
   133 		}
       
   134 	}
       
   135 	out.Write([]byte(strings.Repeat("l ", maxDelims+1) + "\n"))
       
   136 	out.Write([]byte(strings.Repeat("l ", maxDelims+1) + ".\n"))
       
   137 	out.Write(header)
       
   138 	if len(header) > 0 {
       
   139 		out.Write([]byte("\n"))
       
   140 	}
       
   141 
       
   142 	out.Write(body)
       
   143 	out.WriteString("\n.TE\n")
       
   144 }
       
   145 
       
   146 func (r *roffRenderer) TableRow(out *bytes.Buffer, text []byte) {
       
   147 	if out.Len() > 0 {
       
   148 		out.WriteString("\n")
       
   149 	}
       
   150 	out.Write(text)
       
   151 }
       
   152 
       
   153 func (r *roffRenderer) TableHeaderCell(out *bytes.Buffer, text []byte, align int) {
       
   154 	if out.Len() > 0 {
       
   155 		out.WriteString("\t")
       
   156 	}
       
   157 	if len(text) == 0 {
       
   158 		text = []byte{' '}
       
   159 	}
       
   160 	out.Write([]byte("\\fB\\fC" + string(text) + "\\fR"))
       
   161 }
       
   162 
       
   163 func (r *roffRenderer) TableCell(out *bytes.Buffer, text []byte, align int) {
       
   164 	if out.Len() > 0 {
       
   165 		out.WriteString("\t")
       
   166 	}
       
   167 	if len(text) > 30 {
       
   168 		text = append([]byte("T{\n"), text...)
       
   169 		text = append(text, []byte("\nT}")...)
       
   170 	}
       
   171 	if len(text) == 0 {
       
   172 		text = []byte{' '}
       
   173 	}
       
   174 	out.Write(text)
       
   175 }
       
   176 
       
   177 func (r *roffRenderer) Footnotes(out *bytes.Buffer, text func() bool) {
       
   178 
       
   179 }
       
   180 
       
   181 func (r *roffRenderer) FootnoteItem(out *bytes.Buffer, name, text []byte, flags int) {
       
   182 
       
   183 }
       
   184 
       
   185 func (r *roffRenderer) AutoLink(out *bytes.Buffer, link []byte, kind int) {
       
   186 	out.WriteString("\n\\[la]")
       
   187 	out.Write(link)
       
   188 	out.WriteString("\\[ra]")
       
   189 }
       
   190 
       
   191 func (r *roffRenderer) CodeSpan(out *bytes.Buffer, text []byte) {
       
   192 	out.WriteString("\\fB\\fC")
       
   193 	escapeSpecialChars(out, text)
       
   194 	out.WriteString("\\fR")
       
   195 }
       
   196 
       
   197 func (r *roffRenderer) DoubleEmphasis(out *bytes.Buffer, text []byte) {
       
   198 	out.WriteString("\\fB")
       
   199 	out.Write(text)
       
   200 	out.WriteString("\\fP")
       
   201 }
       
   202 
       
   203 func (r *roffRenderer) Emphasis(out *bytes.Buffer, text []byte) {
       
   204 	out.WriteString("\\fI")
       
   205 	out.Write(text)
       
   206 	out.WriteString("\\fP")
       
   207 }
       
   208 
       
   209 func (r *roffRenderer) Image(out *bytes.Buffer, link []byte, title []byte, alt []byte) {
       
   210 }
       
   211 
       
   212 func (r *roffRenderer) LineBreak(out *bytes.Buffer) {
       
   213 	out.WriteString("\n.br\n")
       
   214 }
       
   215 
       
   216 func (r *roffRenderer) Link(out *bytes.Buffer, link []byte, title []byte, content []byte) {
       
   217 	out.Write(content)
       
   218 	r.AutoLink(out, link, 0)
       
   219 }
       
   220 
       
   221 func (r *roffRenderer) RawHtmlTag(out *bytes.Buffer, tag []byte) { // nolint: golint
       
   222 	out.Write(tag)
       
   223 }
       
   224 
       
   225 func (r *roffRenderer) TripleEmphasis(out *bytes.Buffer, text []byte) {
       
   226 	out.WriteString("\\s+2")
       
   227 	out.Write(text)
       
   228 	out.WriteString("\\s-2")
       
   229 }
       
   230 
       
   231 func (r *roffRenderer) StrikeThrough(out *bytes.Buffer, text []byte) {
       
   232 }
       
   233 
       
   234 func (r *roffRenderer) FootnoteRef(out *bytes.Buffer, ref []byte, id int) {
       
   235 
       
   236 }
       
   237 
       
   238 func (r *roffRenderer) Entity(out *bytes.Buffer, entity []byte) {
       
   239 	out.WriteString(html.UnescapeString(string(entity)))
       
   240 }
       
   241 
       
   242 func (r *roffRenderer) NormalText(out *bytes.Buffer, text []byte) {
       
   243 	escapeSpecialChars(out, text)
       
   244 }
       
   245 
       
   246 func (r *roffRenderer) DocumentHeader(out *bytes.Buffer) {
       
   247 }
       
   248 
       
   249 func (r *roffRenderer) DocumentFooter(out *bytes.Buffer) {
       
   250 }
       
   251 
       
   252 func needsBackslash(c byte) bool {
       
   253 	for _, r := range []byte("-_&\\~") {
       
   254 		if c == r {
       
   255 			return true
       
   256 		}
       
   257 	}
       
   258 	return false
       
   259 }
       
   260 
       
   261 func escapeSpecialChars(out *bytes.Buffer, text []byte) {
       
   262 	for i := 0; i < len(text); i++ {
       
   263 		// escape initial apostrophe or period
       
   264 		if len(text) >= 1 && (text[0] == '\'' || text[0] == '.') {
       
   265 			out.WriteString("\\&")
       
   266 		}
       
   267 
       
   268 		// directly copy normal characters
       
   269 		org := i
       
   270 
       
   271 		for i < len(text) && !needsBackslash(text[i]) {
       
   272 			i++
       
   273 		}
       
   274 		if i > org {
       
   275 			out.Write(text[org:i])
       
   276 		}
       
   277 
       
   278 		// escape a character
       
   279 		if i >= len(text) {
       
   280 			break
       
   281 		}
       
   282 		out.WriteByte('\\')
       
   283 		out.WriteByte(text[i])
       
   284 	}
       
   285 }