2011-05-29 11:17:53 +08:00
|
|
|
//
|
2011-06-28 10:11:32 +08:00
|
|
|
// Blackfriday Markdown Processor
|
|
|
|
// Available at http://github.com/russross/blackfriday
|
|
|
|
//
|
|
|
|
// Copyright © 2011 Russ Ross <russ@russross.com>.
|
2011-06-29 01:30:10 +08:00
|
|
|
// Distributed under the Simplified BSD License.
|
2011-06-28 10:11:32 +08:00
|
|
|
// See README.md for details.
|
2011-05-29 11:17:53 +08:00
|
|
|
//
|
|
|
|
|
|
|
|
//
|
|
|
|
//
|
|
|
|
// HTML rendering backend
|
|
|
|
//
|
|
|
|
//
|
|
|
|
|
|
|
|
package blackfriday
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"fmt"
|
|
|
|
"strconv"
|
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
HTML_SKIP_HTML = 1 << iota
|
|
|
|
HTML_SKIP_STYLE
|
|
|
|
HTML_SKIP_IMAGES
|
|
|
|
HTML_SKIP_LINKS
|
|
|
|
HTML_SAFELINK
|
|
|
|
HTML_TOC
|
2011-06-30 00:36:56 +08:00
|
|
|
HTML_OMIT_CONTENTS
|
2011-06-30 00:08:56 +08:00
|
|
|
HTML_COMPLETE_PAGE
|
2011-05-29 11:17:53 +08:00
|
|
|
HTML_GITHUB_BLOCKCODE
|
|
|
|
HTML_USE_XHTML
|
|
|
|
HTML_USE_SMARTYPANTS
|
|
|
|
HTML_SMARTYPANTS_FRACTIONS
|
|
|
|
HTML_SMARTYPANTS_LATEX_DASHES
|
|
|
|
)
|
|
|
|
|
|
|
|
type htmlOptions struct {
|
2011-06-30 00:08:56 +08:00
|
|
|
flags int // HTML_* options
|
2011-06-26 23:51:36 +08:00
|
|
|
closeTag string // how to end singleton tags: either " />\n" or ">\n"
|
2011-06-30 00:08:56 +08:00
|
|
|
title string // document title
|
|
|
|
css string // optional css file url (used with HTML_COMPLETE_PAGE)
|
|
|
|
|
|
|
|
// table of contents data
|
2011-06-30 00:36:56 +08:00
|
|
|
tocMarker int
|
2011-06-30 00:08:56 +08:00
|
|
|
headerCount int
|
|
|
|
currentLevel int
|
|
|
|
toc *bytes.Buffer
|
|
|
|
|
2011-05-29 11:17:53 +08:00
|
|
|
smartypants *SmartypantsRenderer
|
|
|
|
}
|
|
|
|
|
2011-05-31 11:44:52 +08:00
|
|
|
var xhtmlClose = " />\n"
|
|
|
|
var htmlClose = ">\n"
|
2011-05-29 11:17:53 +08:00
|
|
|
|
2011-06-30 00:08:56 +08:00
|
|
|
func HtmlRenderer(flags int, title string, css string) *Renderer {
|
2011-05-29 11:17:53 +08:00
|
|
|
// configure the rendering engine
|
|
|
|
r := new(Renderer)
|
2011-06-29 09:46:35 +08:00
|
|
|
r.BlockCode = htmlBlockCode
|
2011-05-31 11:44:52 +08:00
|
|
|
r.BlockQuote = htmlBlockQuote
|
2011-06-29 09:46:35 +08:00
|
|
|
r.BlockHtml = htmlBlockHtml
|
2011-05-31 11:44:52 +08:00
|
|
|
r.Header = htmlHeader
|
|
|
|
r.HRule = htmlHRule
|
|
|
|
r.List = htmlList
|
|
|
|
r.ListItem = htmlListItem
|
|
|
|
r.Paragraph = htmlParagraph
|
|
|
|
r.Table = htmlTable
|
|
|
|
r.TableRow = htmlTableRow
|
|
|
|
r.TableCell = htmlTableCell
|
|
|
|
|
|
|
|
r.AutoLink = htmlAutoLink
|
|
|
|
r.CodeSpan = htmlCodeSpan
|
|
|
|
r.DoubleEmphasis = htmlDoubleEmphasis
|
|
|
|
r.Emphasis = htmlEmphasis
|
2011-06-29 09:46:35 +08:00
|
|
|
r.Image = htmlImage
|
2011-05-31 11:44:52 +08:00
|
|
|
r.LineBreak = htmlLineBreak
|
2011-06-29 09:46:35 +08:00
|
|
|
r.Link = htmlLink
|
2011-05-31 11:44:52 +08:00
|
|
|
r.RawHtmlTag = htmlRawTag
|
|
|
|
r.TripleEmphasis = htmlTripleEmphasis
|
|
|
|
r.StrikeThrough = htmlStrikeThrough
|
2011-06-30 00:08:56 +08:00
|
|
|
|
|
|
|
r.Entity = htmlEntity
|
2011-06-29 09:46:35 +08:00
|
|
|
r.NormalText = htmlNormalText
|
2011-05-29 11:17:53 +08:00
|
|
|
|
2011-06-30 00:08:56 +08:00
|
|
|
r.DocumentHeader = htmlDocumentHeader
|
|
|
|
r.DocumentFooter = htmlDocumentFooter
|
|
|
|
|
2011-05-31 11:44:52 +08:00
|
|
|
closeTag := htmlClose
|
2011-05-29 11:17:53 +08:00
|
|
|
if flags&HTML_USE_XHTML != 0 {
|
2011-05-31 11:44:52 +08:00
|
|
|
closeTag = xhtmlClose
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
|
|
|
|
2011-06-30 00:08:56 +08:00
|
|
|
r.Opaque = &htmlOptions{
|
|
|
|
flags: flags,
|
|
|
|
closeTag: closeTag,
|
|
|
|
title: title,
|
|
|
|
css: css,
|
2011-05-29 11:17:53 +08:00
|
|
|
|
2011-06-30 00:08:56 +08:00
|
|
|
headerCount: 0,
|
|
|
|
currentLevel: 0,
|
2011-06-30 00:36:56 +08:00
|
|
|
toc: new(bytes.Buffer),
|
2011-05-29 11:17:53 +08:00
|
|
|
|
2011-06-30 00:08:56 +08:00
|
|
|
smartypants: Smartypants(flags),
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
|
|
|
return r
|
|
|
|
}
|
|
|
|
|
2011-05-31 01:06:20 +08:00
|
|
|
func attrEscape(out *bytes.Buffer, src []byte) {
|
2011-06-25 09:11:06 +08:00
|
|
|
org := 0
|
|
|
|
for i, ch := range src {
|
2011-06-26 05:02:46 +08:00
|
|
|
// using if statements is a bit faster than a switch statement.
|
|
|
|
// as the compiler improves, this should be unnecessary
|
|
|
|
// this is only worthwhile because attrEscape is the single
|
|
|
|
// largest CPU user in normal use
|
|
|
|
if ch == '"' {
|
2011-06-25 09:11:06 +08:00
|
|
|
if i > org {
|
|
|
|
// copy all the normal characters since the last escape
|
|
|
|
out.Write(src[org:i])
|
|
|
|
}
|
|
|
|
org = i + 1
|
2011-06-26 05:02:46 +08:00
|
|
|
out.WriteString(""")
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if ch == '&' {
|
2011-06-25 09:11:06 +08:00
|
|
|
if i > org {
|
|
|
|
out.Write(src[org:i])
|
|
|
|
}
|
|
|
|
org = i + 1
|
2011-06-26 05:02:46 +08:00
|
|
|
out.WriteString("&")
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if ch == '<' {
|
2011-06-25 09:11:06 +08:00
|
|
|
if i > org {
|
|
|
|
out.Write(src[org:i])
|
|
|
|
}
|
|
|
|
org = i + 1
|
2011-06-26 05:02:46 +08:00
|
|
|
out.WriteString("<")
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if ch == '>' {
|
2011-06-25 09:11:06 +08:00
|
|
|
if i > org {
|
|
|
|
out.Write(src[org:i])
|
|
|
|
}
|
|
|
|
org = i + 1
|
2011-06-26 05:02:46 +08:00
|
|
|
out.WriteString(">")
|
|
|
|
continue
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
|
|
|
}
|
2011-06-25 09:11:06 +08:00
|
|
|
if org < len(src) {
|
|
|
|
out.Write(src[org:])
|
|
|
|
}
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
|
|
|
|
2011-06-25 22:20:08 +08:00
|
|
|
func htmlHeader(out *bytes.Buffer, text func() bool, level int, opaque interface{}) {
|
2011-05-29 11:17:53 +08:00
|
|
|
options := opaque.(*htmlOptions)
|
2011-06-25 22:20:08 +08:00
|
|
|
marker := out.Len()
|
2011-05-29 11:17:53 +08:00
|
|
|
|
2011-06-25 22:20:08 +08:00
|
|
|
if marker > 0 {
|
2011-05-31 01:06:20 +08:00
|
|
|
out.WriteByte('\n')
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
|
|
|
|
2011-05-30 07:00:31 +08:00
|
|
|
if options.flags&HTML_TOC != 0 {
|
2011-06-30 00:36:56 +08:00
|
|
|
// headerCount is incremented in htmlTocHeader
|
2011-06-30 00:08:56 +08:00
|
|
|
out.WriteString(fmt.Sprintf("<h%d id=\"toc_%d\">", level, options.headerCount))
|
2011-05-29 11:17:53 +08:00
|
|
|
} else {
|
2011-05-31 01:06:20 +08:00
|
|
|
out.WriteString(fmt.Sprintf("<h%d>", level))
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
|
|
|
|
2011-06-30 00:36:56 +08:00
|
|
|
tocMarker := out.Len()
|
2011-06-25 22:20:08 +08:00
|
|
|
if !text() {
|
|
|
|
out.Truncate(marker)
|
|
|
|
return
|
|
|
|
}
|
2011-06-30 00:08:56 +08:00
|
|
|
|
|
|
|
// are we building a table of contents?
|
|
|
|
if options.flags&HTML_TOC != 0 {
|
2011-06-30 00:36:56 +08:00
|
|
|
htmlTocHeader(out.Bytes()[tocMarker:], level, opaque)
|
2011-06-30 00:08:56 +08:00
|
|
|
}
|
|
|
|
|
2011-05-31 01:06:20 +08:00
|
|
|
out.WriteString(fmt.Sprintf("</h%d>\n", level))
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
|
|
|
|
2011-06-29 09:46:35 +08:00
|
|
|
func htmlBlockHtml(out *bytes.Buffer, text []byte, opaque interface{}) {
|
|
|
|
options := opaque.(*htmlOptions)
|
|
|
|
if options.flags&HTML_SKIP_HTML != 0 {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2011-05-29 11:17:53 +08:00
|
|
|
sz := len(text)
|
|
|
|
for sz > 0 && text[sz-1] == '\n' {
|
|
|
|
sz--
|
|
|
|
}
|
|
|
|
org := 0
|
|
|
|
for org < sz && text[org] == '\n' {
|
|
|
|
org++
|
|
|
|
}
|
|
|
|
if org >= sz {
|
|
|
|
return
|
|
|
|
}
|
2011-05-31 01:06:20 +08:00
|
|
|
if out.Len() > 0 {
|
|
|
|
out.WriteByte('\n')
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
2011-05-31 01:06:20 +08:00
|
|
|
out.Write(text[org:sz])
|
|
|
|
out.WriteByte('\n')
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
|
|
|
|
2011-05-31 11:44:52 +08:00
|
|
|
func htmlHRule(out *bytes.Buffer, opaque interface{}) {
|
2011-05-29 11:17:53 +08:00
|
|
|
options := opaque.(*htmlOptions)
|
|
|
|
|
2011-05-31 01:06:20 +08:00
|
|
|
if out.Len() > 0 {
|
|
|
|
out.WriteByte('\n')
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
2011-05-31 01:06:20 +08:00
|
|
|
out.WriteString("<hr")
|
2011-05-31 11:44:52 +08:00
|
|
|
out.WriteString(options.closeTag)
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
|
|
|
|
2011-05-31 11:44:52 +08:00
|
|
|
func htmlBlockCode(out *bytes.Buffer, text []byte, lang string, opaque interface{}) {
|
2011-06-29 09:46:35 +08:00
|
|
|
options := opaque.(*htmlOptions)
|
|
|
|
if options.flags&HTML_GITHUB_BLOCKCODE != 0 {
|
|
|
|
htmlBlockCodeGithub(out, text, lang, opaque)
|
|
|
|
} else {
|
|
|
|
htmlBlockCodeNormal(out, text, lang, opaque)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func htmlBlockCodeNormal(out *bytes.Buffer, text []byte, lang string, opaque interface{}) {
|
2011-05-31 01:06:20 +08:00
|
|
|
if out.Len() > 0 {
|
|
|
|
out.WriteByte('\n')
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if lang != "" {
|
2011-05-31 01:06:20 +08:00
|
|
|
out.WriteString("<pre><code class=\"")
|
2011-05-29 11:17:53 +08:00
|
|
|
|
|
|
|
for i, cls := 0, 0; i < len(lang); i, cls = i+1, cls+1 {
|
|
|
|
for i < len(lang) && isspace(lang[i]) {
|
|
|
|
i++
|
|
|
|
}
|
|
|
|
|
|
|
|
if i < len(lang) {
|
|
|
|
org := i
|
|
|
|
for i < len(lang) && !isspace(lang[i]) {
|
|
|
|
i++
|
|
|
|
}
|
|
|
|
|
|
|
|
if lang[org] == '.' {
|
|
|
|
org++
|
|
|
|
}
|
|
|
|
|
|
|
|
if cls > 0 {
|
2011-05-31 01:06:20 +08:00
|
|
|
out.WriteByte(' ')
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
2011-05-31 01:06:20 +08:00
|
|
|
attrEscape(out, []byte(lang[org:]))
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-05-31 01:06:20 +08:00
|
|
|
out.WriteString("\">")
|
2011-05-29 11:17:53 +08:00
|
|
|
} else {
|
2011-05-31 01:06:20 +08:00
|
|
|
out.WriteString("<pre><code>")
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if len(text) > 0 {
|
2011-05-31 01:06:20 +08:00
|
|
|
attrEscape(out, text)
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
|
|
|
|
2011-05-31 01:06:20 +08:00
|
|
|
out.WriteString("</code></pre>\n")
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* GitHub style code block:
|
|
|
|
*
|
|
|
|
* <pre lang="LANG"><code>
|
|
|
|
* ...
|
|
|
|
* </pre></code>
|
|
|
|
*
|
|
|
|
* Unlike other parsers, we store the language identifier in the <pre>,
|
|
|
|
* and don't let the user generate custom classes.
|
|
|
|
*
|
|
|
|
* The language identifier in the <pre> block gets postprocessed and all
|
|
|
|
* the code inside gets syntax highlighted with Pygments. This is much safer
|
|
|
|
* than letting the user specify a CSS class for highlighting.
|
|
|
|
*
|
|
|
|
* Note that we only generate HTML for the first specifier.
|
|
|
|
* E.g.
|
|
|
|
* ~~~~ {.python .numbered} => <pre lang="python"><code>
|
|
|
|
*/
|
2011-05-31 11:44:52 +08:00
|
|
|
func htmlBlockCodeGithub(out *bytes.Buffer, text []byte, lang string, opaque interface{}) {
|
2011-05-31 01:06:20 +08:00
|
|
|
if out.Len() > 0 {
|
|
|
|
out.WriteByte('\n')
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if len(lang) > 0 {
|
2011-05-31 01:06:20 +08:00
|
|
|
out.WriteString("<pre lang=\"")
|
2011-05-29 11:17:53 +08:00
|
|
|
|
|
|
|
i := 0
|
|
|
|
for i < len(lang) && !isspace(lang[i]) {
|
|
|
|
i++
|
|
|
|
}
|
|
|
|
|
|
|
|
if lang[0] == '.' {
|
2011-05-31 01:06:20 +08:00
|
|
|
attrEscape(out, []byte(lang[1:i]))
|
2011-05-29 11:17:53 +08:00
|
|
|
} else {
|
2011-05-31 01:06:20 +08:00
|
|
|
attrEscape(out, []byte(lang[:i]))
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
|
|
|
|
2011-05-31 01:06:20 +08:00
|
|
|
out.WriteString("\"><code>")
|
2011-05-29 11:17:53 +08:00
|
|
|
} else {
|
2011-05-31 01:06:20 +08:00
|
|
|
out.WriteString("<pre><code>")
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if len(text) > 0 {
|
2011-05-31 01:06:20 +08:00
|
|
|
attrEscape(out, text)
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
|
|
|
|
2011-05-31 01:06:20 +08:00
|
|
|
out.WriteString("</code></pre>\n")
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-05-31 11:44:52 +08:00
|
|
|
func htmlBlockQuote(out *bytes.Buffer, text []byte, opaque interface{}) {
|
2011-05-31 01:06:20 +08:00
|
|
|
out.WriteString("<blockquote>\n")
|
|
|
|
out.Write(text)
|
|
|
|
out.WriteString("</blockquote>")
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
|
|
|
|
2011-05-31 01:06:20 +08:00
|
|
|
func htmlTable(out *bytes.Buffer, header []byte, body []byte, columnData []int, opaque interface{}) {
|
|
|
|
if out.Len() > 0 {
|
|
|
|
out.WriteByte('\n')
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
2011-05-31 01:06:20 +08:00
|
|
|
out.WriteString("<table><thead>\n")
|
|
|
|
out.Write(header)
|
|
|
|
out.WriteString("\n</thead><tbody>\n")
|
|
|
|
out.Write(body)
|
|
|
|
out.WriteString("\n</tbody></table>")
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
|
|
|
|
2011-05-31 01:06:20 +08:00
|
|
|
func htmlTableRow(out *bytes.Buffer, text []byte, opaque interface{}) {
|
|
|
|
if out.Len() > 0 {
|
|
|
|
out.WriteByte('\n')
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
2011-05-31 01:06:20 +08:00
|
|
|
out.WriteString("<tr>\n")
|
|
|
|
out.Write(text)
|
|
|
|
out.WriteString("\n</tr>")
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
|
|
|
|
2011-05-31 01:06:20 +08:00
|
|
|
func htmlTableCell(out *bytes.Buffer, text []byte, align int, opaque interface{}) {
|
|
|
|
if out.Len() > 0 {
|
|
|
|
out.WriteByte('\n')
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
|
|
|
switch align {
|
|
|
|
case TABLE_ALIGNMENT_LEFT:
|
2011-05-31 01:06:20 +08:00
|
|
|
out.WriteString("<td align=\"left\">")
|
2011-05-29 11:17:53 +08:00
|
|
|
case TABLE_ALIGNMENT_RIGHT:
|
2011-05-31 01:06:20 +08:00
|
|
|
out.WriteString("<td align=\"right\">")
|
2011-05-29 11:17:53 +08:00
|
|
|
case TABLE_ALIGNMENT_CENTER:
|
2011-05-31 01:06:20 +08:00
|
|
|
out.WriteString("<td align=\"center\">")
|
2011-05-29 11:17:53 +08:00
|
|
|
default:
|
2011-05-31 01:06:20 +08:00
|
|
|
out.WriteString("<td>")
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
|
|
|
|
2011-05-31 01:06:20 +08:00
|
|
|
out.Write(text)
|
|
|
|
out.WriteString("</td>")
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
|
|
|
|
2011-06-26 05:02:46 +08:00
|
|
|
func htmlList(out *bytes.Buffer, text func() bool, flags int, opaque interface{}) {
|
|
|
|
marker := out.Len()
|
|
|
|
|
|
|
|
if marker > 0 {
|
2011-05-31 01:06:20 +08:00
|
|
|
out.WriteByte('\n')
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
|
|
|
if flags&LIST_TYPE_ORDERED != 0 {
|
2011-05-31 01:06:20 +08:00
|
|
|
out.WriteString("<ol>\n")
|
2011-05-29 11:17:53 +08:00
|
|
|
} else {
|
2011-05-31 01:06:20 +08:00
|
|
|
out.WriteString("<ul>\n")
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
2011-06-26 05:02:46 +08:00
|
|
|
if !text() {
|
|
|
|
out.Truncate(marker)
|
|
|
|
return
|
|
|
|
}
|
2011-05-29 11:17:53 +08:00
|
|
|
if flags&LIST_TYPE_ORDERED != 0 {
|
2011-05-31 01:06:20 +08:00
|
|
|
out.WriteString("</ol>\n")
|
2011-05-29 11:17:53 +08:00
|
|
|
} else {
|
2011-05-31 01:06:20 +08:00
|
|
|
out.WriteString("</ul>\n")
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-05-31 11:44:52 +08:00
|
|
|
func htmlListItem(out *bytes.Buffer, text []byte, flags int, opaque interface{}) {
|
2011-05-31 01:06:20 +08:00
|
|
|
out.WriteString("<li>")
|
2011-05-29 11:17:53 +08:00
|
|
|
size := len(text)
|
|
|
|
for size > 0 && text[size-1] == '\n' {
|
|
|
|
size--
|
|
|
|
}
|
2011-05-31 01:06:20 +08:00
|
|
|
out.Write(text[:size])
|
|
|
|
out.WriteString("</li>\n")
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
|
|
|
|
2011-06-27 07:21:11 +08:00
|
|
|
func htmlParagraph(out *bytes.Buffer, text func() bool, opaque interface{}) {
|
|
|
|
marker := out.Len()
|
|
|
|
if marker > 0 {
|
2011-05-31 01:06:20 +08:00
|
|
|
out.WriteByte('\n')
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
|
|
|
|
2011-05-31 01:06:20 +08:00
|
|
|
out.WriteString("<p>")
|
2011-06-27 07:21:11 +08:00
|
|
|
if !text() {
|
|
|
|
out.Truncate(marker)
|
|
|
|
return
|
|
|
|
}
|
2011-05-31 01:06:20 +08:00
|
|
|
out.WriteString("</p>\n")
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
|
|
|
|
2011-06-29 09:46:35 +08:00
|
|
|
func htmlAutoLink(out *bytes.Buffer, link []byte, kind int, opaque interface{}) bool {
|
2011-05-29 11:17:53 +08:00
|
|
|
options := opaque.(*htmlOptions)
|
|
|
|
|
|
|
|
if len(link) == 0 {
|
2011-06-29 09:46:35 +08:00
|
|
|
return false
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
2011-05-30 07:00:31 +08:00
|
|
|
if options.flags&HTML_SAFELINK != 0 && !isSafeLink(link) && kind != LINK_TYPE_EMAIL {
|
2011-06-29 09:46:35 +08:00
|
|
|
return false
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
|
|
|
|
2011-05-31 01:06:20 +08:00
|
|
|
out.WriteString("<a href=\"")
|
2011-05-29 11:17:53 +08:00
|
|
|
if kind == LINK_TYPE_EMAIL {
|
2011-05-31 01:06:20 +08:00
|
|
|
out.WriteString("mailto:")
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
2011-06-25 01:50:03 +08:00
|
|
|
attrEscape(out, link)
|
2011-05-31 01:06:20 +08:00
|
|
|
out.WriteString("\">")
|
2011-05-29 11:17:53 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Pretty print: if we get an email address as
|
|
|
|
* an actual URI, e.g. `mailto:foo@bar.com`, we don't
|
|
|
|
* want to print the `mailto:` prefix
|
|
|
|
*/
|
2011-06-01 01:49:49 +08:00
|
|
|
switch {
|
2011-05-31 05:36:31 +08:00
|
|
|
case bytes.HasPrefix(link, []byte("mailto://")):
|
|
|
|
attrEscape(out, link[9:])
|
2011-06-01 01:49:49 +08:00
|
|
|
case bytes.HasPrefix(link, []byte("mailto:")):
|
2011-05-31 01:06:20 +08:00
|
|
|
attrEscape(out, link[7:])
|
2011-05-31 05:36:31 +08:00
|
|
|
default:
|
2011-05-31 01:06:20 +08:00
|
|
|
attrEscape(out, link)
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
|
|
|
|
2011-05-31 01:06:20 +08:00
|
|
|
out.WriteString("</a>")
|
2011-05-29 11:17:53 +08:00
|
|
|
|
2011-06-29 09:46:35 +08:00
|
|
|
return true
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
|
|
|
|
2011-06-29 09:46:35 +08:00
|
|
|
func htmlCodeSpan(out *bytes.Buffer, text []byte, opaque interface{}) bool {
|
2011-05-31 01:06:20 +08:00
|
|
|
out.WriteString("<code>")
|
|
|
|
attrEscape(out, text)
|
|
|
|
out.WriteString("</code>")
|
2011-06-29 09:46:35 +08:00
|
|
|
return true
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
|
|
|
|
2011-06-29 09:46:35 +08:00
|
|
|
func htmlDoubleEmphasis(out *bytes.Buffer, text []byte, opaque interface{}) bool {
|
2011-05-29 11:17:53 +08:00
|
|
|
if len(text) == 0 {
|
2011-06-29 09:46:35 +08:00
|
|
|
return false
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
2011-05-31 01:06:20 +08:00
|
|
|
out.WriteString("<strong>")
|
|
|
|
out.Write(text)
|
|
|
|
out.WriteString("</strong>")
|
2011-06-29 09:46:35 +08:00
|
|
|
return true
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
|
|
|
|
2011-06-29 09:46:35 +08:00
|
|
|
func htmlEmphasis(out *bytes.Buffer, text []byte, opaque interface{}) bool {
|
2011-05-29 11:17:53 +08:00
|
|
|
if len(text) == 0 {
|
2011-06-29 09:46:35 +08:00
|
|
|
return false
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
2011-05-31 01:06:20 +08:00
|
|
|
out.WriteString("<em>")
|
|
|
|
out.Write(text)
|
|
|
|
out.WriteString("</em>")
|
2011-06-29 09:46:35 +08:00
|
|
|
return true
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
|
|
|
|
2011-06-29 09:46:35 +08:00
|
|
|
func htmlImage(out *bytes.Buffer, link []byte, title []byte, alt []byte, opaque interface{}) bool {
|
2011-05-29 11:17:53 +08:00
|
|
|
options := opaque.(*htmlOptions)
|
2011-06-29 09:46:35 +08:00
|
|
|
if options.flags&HTML_SKIP_IMAGES != 0 {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2011-05-29 11:17:53 +08:00
|
|
|
if len(link) == 0 {
|
2011-06-29 09:46:35 +08:00
|
|
|
return false
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
2011-05-31 01:06:20 +08:00
|
|
|
out.WriteString("<img src=\"")
|
|
|
|
attrEscape(out, link)
|
|
|
|
out.WriteString("\" alt=\"")
|
2011-05-29 11:17:53 +08:00
|
|
|
if len(alt) > 0 {
|
2011-05-31 01:06:20 +08:00
|
|
|
attrEscape(out, alt)
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
|
|
|
if len(title) > 0 {
|
2011-05-31 01:06:20 +08:00
|
|
|
out.WriteString("\" title=\"")
|
|
|
|
attrEscape(out, title)
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
|
|
|
|
2011-05-31 01:06:20 +08:00
|
|
|
out.WriteByte('"')
|
2011-05-31 11:44:52 +08:00
|
|
|
out.WriteString(options.closeTag)
|
2011-06-29 09:46:35 +08:00
|
|
|
return true
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
|
|
|
|
2011-06-29 09:46:35 +08:00
|
|
|
func htmlLineBreak(out *bytes.Buffer, opaque interface{}) bool {
|
2011-05-29 11:17:53 +08:00
|
|
|
options := opaque.(*htmlOptions)
|
2011-05-31 01:06:20 +08:00
|
|
|
out.WriteString("<br")
|
2011-05-31 11:44:52 +08:00
|
|
|
out.WriteString(options.closeTag)
|
2011-06-29 09:46:35 +08:00
|
|
|
return true
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
|
|
|
|
2011-06-29 09:46:35 +08:00
|
|
|
func htmlLink(out *bytes.Buffer, link []byte, title []byte, content []byte, opaque interface{}) bool {
|
2011-05-29 11:17:53 +08:00
|
|
|
options := opaque.(*htmlOptions)
|
2011-06-29 09:46:35 +08:00
|
|
|
if options.flags&HTML_SKIP_LINKS != 0 {
|
|
|
|
return false
|
|
|
|
}
|
2011-05-29 11:17:53 +08:00
|
|
|
|
2011-05-30 07:00:31 +08:00
|
|
|
if options.flags&HTML_SAFELINK != 0 && !isSafeLink(link) {
|
2011-06-29 09:46:35 +08:00
|
|
|
return false
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
|
|
|
|
2011-05-31 01:06:20 +08:00
|
|
|
out.WriteString("<a href=\"")
|
2011-06-25 01:50:03 +08:00
|
|
|
attrEscape(out, link)
|
2011-05-29 11:17:53 +08:00
|
|
|
if len(title) > 0 {
|
2011-05-31 01:06:20 +08:00
|
|
|
out.WriteString("\" title=\"")
|
|
|
|
attrEscape(out, title)
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
2011-05-31 01:06:20 +08:00
|
|
|
out.WriteString("\">")
|
2011-06-25 01:50:03 +08:00
|
|
|
out.Write(content)
|
2011-05-31 01:06:20 +08:00
|
|
|
out.WriteString("</a>")
|
2011-06-29 09:46:35 +08:00
|
|
|
return true
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
|
|
|
|
2011-06-29 09:46:35 +08:00
|
|
|
func htmlRawTag(out *bytes.Buffer, text []byte, opaque interface{}) bool {
|
2011-05-29 11:17:53 +08:00
|
|
|
options := opaque.(*htmlOptions)
|
2011-05-30 07:00:31 +08:00
|
|
|
if options.flags&HTML_SKIP_HTML != 0 {
|
2011-06-29 09:46:35 +08:00
|
|
|
return true
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
2011-05-30 07:00:31 +08:00
|
|
|
if options.flags&HTML_SKIP_STYLE != 0 && isHtmlTag(text, "style") {
|
2011-06-29 09:46:35 +08:00
|
|
|
return true
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
2011-05-30 07:00:31 +08:00
|
|
|
if options.flags&HTML_SKIP_LINKS != 0 && isHtmlTag(text, "a") {
|
2011-06-29 09:46:35 +08:00
|
|
|
return true
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
2011-05-30 07:00:31 +08:00
|
|
|
if options.flags&HTML_SKIP_IMAGES != 0 && isHtmlTag(text, "img") {
|
2011-06-29 09:46:35 +08:00
|
|
|
return true
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
2011-05-31 01:06:20 +08:00
|
|
|
out.Write(text)
|
2011-06-29 09:46:35 +08:00
|
|
|
return true
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
|
|
|
|
2011-06-29 09:46:35 +08:00
|
|
|
func htmlTripleEmphasis(out *bytes.Buffer, text []byte, opaque interface{}) bool {
|
2011-05-29 11:17:53 +08:00
|
|
|
if len(text) == 0 {
|
2011-06-29 09:46:35 +08:00
|
|
|
return false
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
2011-05-31 01:06:20 +08:00
|
|
|
out.WriteString("<strong><em>")
|
|
|
|
out.Write(text)
|
|
|
|
out.WriteString("</em></strong>")
|
2011-06-29 09:46:35 +08:00
|
|
|
return true
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
|
|
|
|
2011-06-29 09:46:35 +08:00
|
|
|
func htmlStrikeThrough(out *bytes.Buffer, text []byte, opaque interface{}) bool {
|
2011-05-29 11:17:53 +08:00
|
|
|
if len(text) == 0 {
|
2011-06-29 09:46:35 +08:00
|
|
|
return false
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
2011-05-31 01:06:20 +08:00
|
|
|
out.WriteString("<del>")
|
|
|
|
out.Write(text)
|
|
|
|
out.WriteString("</del>")
|
2011-06-29 09:46:35 +08:00
|
|
|
return true
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
|
|
|
|
2011-06-30 00:08:56 +08:00
|
|
|
func htmlEntity(out *bytes.Buffer, entity []byte, opaque interface{}) {
|
|
|
|
out.Write(entity)
|
|
|
|
}
|
|
|
|
|
2011-05-31 01:06:20 +08:00
|
|
|
func htmlNormalText(out *bytes.Buffer, text []byte, opaque interface{}) {
|
2011-06-29 09:46:35 +08:00
|
|
|
options := opaque.(*htmlOptions)
|
|
|
|
if options.flags&HTML_USE_SMARTYPANTS != 0 {
|
|
|
|
htmlSmartypants(out, text, opaque)
|
|
|
|
} else {
|
|
|
|
attrEscape(out, text)
|
|
|
|
}
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
|
|
|
|
2011-06-30 00:08:56 +08:00
|
|
|
func htmlDocumentHeader(out *bytes.Buffer, opaque interface{}) {
|
|
|
|
options := opaque.(*htmlOptions)
|
|
|
|
if options.flags&HTML_COMPLETE_PAGE == 0 {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
ending := ""
|
|
|
|
if options.flags&HTML_USE_XHTML != 0 {
|
|
|
|
out.WriteString("<!DOCTYPE html PUBLIC \"-//W3C//DTD XHTML 1.0 Transitional//EN\" ")
|
|
|
|
out.WriteString("\"http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd\">\n")
|
|
|
|
out.WriteString("<html xmlns=\"http://www.w3.org/1999/xhtml\">\n")
|
|
|
|
ending = " /"
|
|
|
|
} else {
|
|
|
|
out.WriteString("<!DOCTYPE html PUBLIC \"-//W3C//DTD HTML 4.01//EN\" ")
|
|
|
|
out.WriteString("\"http://www.w3.org/TR/html4/strict.dtd\">\n")
|
|
|
|
out.WriteString("<html>\n")
|
|
|
|
}
|
|
|
|
out.WriteString("<head>\n")
|
|
|
|
out.WriteString(" <title>")
|
|
|
|
htmlNormalText(out, []byte(options.title), opaque)
|
|
|
|
out.WriteString("</title>\n")
|
|
|
|
out.WriteString(" <meta name=\"GENERATOR\" content=\"Blackfriday Markdown Processor v")
|
|
|
|
out.WriteString(VERSION)
|
|
|
|
out.WriteString("\"")
|
|
|
|
out.WriteString(ending)
|
|
|
|
out.WriteString(">\n")
|
|
|
|
out.WriteString(" <meta http-equiv=\"Content-Type\" content=\"text/html; charset=utf-8\"")
|
|
|
|
out.WriteString(ending)
|
|
|
|
out.WriteString(">\n")
|
|
|
|
if options.css != "" {
|
|
|
|
out.WriteString(" <link rel=\"stylesheet\" type=\"text/css\" href=\"")
|
|
|
|
attrEscape(out, []byte(options.css))
|
|
|
|
out.WriteString("\"")
|
|
|
|
out.WriteString(ending)
|
|
|
|
out.WriteString(">\n")
|
|
|
|
}
|
|
|
|
out.WriteString("</head>\n")
|
|
|
|
out.WriteString("<body>\n")
|
2011-06-30 00:36:56 +08:00
|
|
|
|
|
|
|
options.tocMarker = out.Len()
|
2011-06-30 00:08:56 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
func htmlDocumentFooter(out *bytes.Buffer, opaque interface{}) {
|
|
|
|
options := opaque.(*htmlOptions)
|
2011-06-30 00:36:56 +08:00
|
|
|
|
|
|
|
// finalize and insert the table of contents
|
|
|
|
if options.flags&HTML_TOC != 0 {
|
|
|
|
htmlTocFinalize(opaque)
|
|
|
|
|
|
|
|
// now we have to insert the table of contents into the document
|
|
|
|
var temp bytes.Buffer
|
|
|
|
|
|
|
|
// start by making a copy of everything after the document header
|
|
|
|
temp.Write(out.Bytes()[options.tocMarker:])
|
|
|
|
|
|
|
|
// now clear the copied material from the main output buffer
|
|
|
|
out.Truncate(options.tocMarker)
|
|
|
|
|
|
|
|
// insert the table of contents
|
|
|
|
out.Write(options.toc.Bytes())
|
|
|
|
|
|
|
|
// write out everything that came after it
|
|
|
|
if options.flags&HTML_OMIT_CONTENTS == 0 {
|
|
|
|
out.Write(temp.Bytes())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if options.flags&HTML_COMPLETE_PAGE != 0 {
|
|
|
|
out.WriteString("\n</body>\n")
|
|
|
|
out.WriteString("</html>\n")
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
2011-06-30 00:08:56 +08:00
|
|
|
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
|
|
|
|
2011-06-30 00:36:56 +08:00
|
|
|
func htmlTocHeader(text []byte, level int, opaque interface{}) {
|
|
|
|
options := opaque.(*htmlOptions)
|
|
|
|
|
|
|
|
for level > options.currentLevel {
|
|
|
|
switch {
|
|
|
|
case bytes.HasSuffix(options.toc.Bytes(), []byte("</li>\n")):
|
|
|
|
size := options.toc.Len()
|
|
|
|
options.toc.Truncate(size - len("</li>\n"))
|
|
|
|
|
|
|
|
case options.currentLevel > 0:
|
|
|
|
options.toc.WriteString("<li>")
|
|
|
|
}
|
|
|
|
options.toc.WriteString("\n<ul>\n")
|
|
|
|
options.currentLevel++
|
|
|
|
}
|
|
|
|
|
|
|
|
for level < options.currentLevel {
|
|
|
|
options.toc.WriteString("</ul>")
|
|
|
|
if options.currentLevel > 1 {
|
|
|
|
options.toc.WriteString("</li>\n")
|
|
|
|
}
|
|
|
|
options.currentLevel--
|
|
|
|
}
|
|
|
|
|
|
|
|
options.toc.WriteString("<li><a href=\"#toc_")
|
|
|
|
options.toc.WriteString(strconv.Itoa(options.headerCount))
|
|
|
|
options.toc.WriteString("\">")
|
|
|
|
options.headerCount++
|
|
|
|
|
|
|
|
options.toc.Write(text)
|
|
|
|
|
|
|
|
options.toc.WriteString("</a></li>\n")
|
|
|
|
}
|
|
|
|
|
|
|
|
func htmlTocFinalize(opaque interface{}) {
|
2011-05-29 11:17:53 +08:00
|
|
|
options := opaque.(*htmlOptions)
|
2011-06-30 00:08:56 +08:00
|
|
|
for options.currentLevel > 1 {
|
2011-06-30 00:36:56 +08:00
|
|
|
options.toc.WriteString("</ul></li>\n")
|
2011-06-30 00:08:56 +08:00
|
|
|
options.currentLevel--
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
|
|
|
|
2011-06-30 00:08:56 +08:00
|
|
|
if options.currentLevel > 0 {
|
2011-06-30 00:36:56 +08:00
|
|
|
options.toc.WriteString("</ul>\n")
|
2011-05-29 11:17:53 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-05-30 07:00:31 +08:00
|
|
|
func isHtmlTag(tag []byte, tagname string) bool {
|
2011-05-29 11:17:53 +08:00
|
|
|
i := 0
|
|
|
|
if i < len(tag) && tag[0] != '<' {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
i++
|
|
|
|
for i < len(tag) && isspace(tag[i]) {
|
|
|
|
i++
|
|
|
|
}
|
|
|
|
|
|
|
|
if i < len(tag) && tag[i] == '/' {
|
|
|
|
i++
|
|
|
|
}
|
|
|
|
|
|
|
|
for i < len(tag) && isspace(tag[i]) {
|
|
|
|
i++
|
|
|
|
}
|
|
|
|
|
2011-06-29 06:02:12 +08:00
|
|
|
j := i
|
|
|
|
for ; i < len(tag); i, j = i+1, j+1 {
|
|
|
|
if j >= len(tagname) {
|
2011-05-29 11:17:53 +08:00
|
|
|
break
|
|
|
|
}
|
|
|
|
|
2011-06-29 06:02:12 +08:00
|
|
|
if tag[i] != tagname[j] {
|
2011-05-29 11:17:53 +08:00
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if i == len(tag) {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
return isspace(tag[i]) || tag[i] == '>'
|
|
|
|
}
|