1
0
mirror of https://github.com/danog/blackfriday.git synced 2025-01-22 21:31:20 +01:00

Factor Smartypants to HTML

Smartypants is HTML-specific.
There is no need to run Smartypants from `Render()`.
This simplifies extensions built upon the HTML renderer.
This commit is contained in:
Pierre Neidhardt 2016-08-10 09:56:37 +05:30
parent 37141d5b5a
commit 02da1dfe9d
5 changed files with 61 additions and 93 deletions

View File

@ -106,7 +106,7 @@ func doLinkTestsInline(t *testing.T, tests []string) {
HTMLRendererParameters: params, HTMLRendererParameters: params,
}) })
doTestsInlineParam(t, transformTests, TestParams{ doTestsInlineParam(t, transformTests, TestParams{
HTMLFlags: CommonHTMLFlags, HTMLFlags: UseXHTML,
HTMLRendererParameters: params, HTMLRendererParameters: params,
}) })
} }

33
html.go
View File

@ -41,6 +41,11 @@ const (
CompletePage // Generate a complete HTML page CompletePage // Generate a complete HTML page
UseXHTML // Generate XHTML output instead of HTML UseXHTML // Generate XHTML output instead of HTML
FootnoteReturnLinks // Generate a link at the end of a footnote to return to the source FootnoteReturnLinks // Generate a link at the end of a footnote to return to the source
Smartypants // Enable smart punctuation substitutions
SmartypantsFractions // Enable smart fractions (with Smartypants)
SmartypantsDashes // Enable smart dashes (with Smartypants)
SmartypantsLatexDashes // Enable LaTeX-style dashes (with Smartypants)
SmartypantsAngledQuotes // Enable angled double quotes (with Smartypants) for double quotes rendering
TagName = "[A-Za-z][A-Za-z0-9-]*" TagName = "[A-Za-z][A-Za-z0-9-]*"
AttributeName = "[a-zA-Z_:][a-zA-Z0-9:._-]*" AttributeName = "[a-zA-Z_:][a-zA-Z0-9:._-]*"
@ -102,6 +107,8 @@ type HTMLRenderer struct {
lastOutputLen int lastOutputLen int
disableTags int disableTags int
sr *SPRenderer
} }
const ( const (
@ -127,6 +134,8 @@ func NewHTMLRenderer(params HTMLRendererParameters) *HTMLRenderer {
closeTag: closeTag, closeTag: closeTag,
headerIDs: make(map[string]int), headerIDs: make(map[string]int),
sr: NewSmartypantsRenderer(params.Flags),
} }
} }
@ -402,6 +411,10 @@ func (r *HTMLRenderer) RenderNode(w io.Writer, node *Node, entering bool) WalkSt
attrs := []string{} attrs := []string{}
switch node.Type { switch node.Type {
case Text: case Text:
node.Literal = esc(node.Literal)
if r.Flags&Smartypants != 0 {
node.Literal = r.sr.Process(node.Literal)
}
r.out(w, node.Literal) r.out(w, node.Literal)
break break
case Softbreak: case Softbreak:
@ -705,7 +718,7 @@ func (r *HTMLRenderer) RenderNode(w io.Writer, node *Node, entering bool) WalkSt
return GoToNext return GoToNext
} }
func (r *HTMLRenderer) writeDocumentHeader(w *bytes.Buffer, sr *SPRenderer) { func (r *HTMLRenderer) writeDocumentHeader(w *bytes.Buffer) {
if r.Flags&CompletePage == 0 { if r.Flags&CompletePage == 0 {
return return
} }
@ -721,8 +734,8 @@ func (r *HTMLRenderer) writeDocumentHeader(w *bytes.Buffer, sr *SPRenderer) {
} }
w.WriteString("<head>\n") w.WriteString("<head>\n")
w.WriteString(" <title>") w.WriteString(" <title>")
if r.Extensions&Smartypants != 0 { if r.Flags&Smartypants != 0 {
w.Write(sr.Process([]byte(r.Title))) w.Write(r.sr.Process([]byte(r.Title)))
} else { } else {
w.Write(esc([]byte(r.Title))) w.Write(esc([]byte(r.Title)))
} }
@ -817,20 +830,8 @@ func (r *HTMLRenderer) writeDocumentFooter(w *bytes.Buffer) {
func (r *HTMLRenderer) Render(ast *Node) []byte { func (r *HTMLRenderer) Render(ast *Node) []byte {
//println("render_Blackfriday") //println("render_Blackfriday")
//dump(ast) //dump(ast)
// Run Smartypants if it's enabled or simply escape text if not
sr := NewSmartypantsRenderer(r.Extensions)
ast.Walk(func(node *Node, entering bool) WalkStatus {
if node.Type == Text {
if r.Extensions&Smartypants != 0 {
node.Literal = sr.Process(node.Literal)
} else {
node.Literal = esc(node.Literal)
}
}
return GoToNext
})
var buff bytes.Buffer var buff bytes.Buffer
r.writeDocumentHeader(&buff, sr) r.writeDocumentHeader(&buff)
if r.Extensions&TOC != 0 || r.Extensions&OmitContents != 0 { if r.Extensions&TOC != 0 || r.Extensions&OmitContents != 0 {
r.writeTOC(&buff, ast) r.writeTOC(&buff, ast)
if r.Extensions&OmitContents != 0 { if r.Extensions&OmitContents != 0 {

View File

@ -1012,11 +1012,7 @@ func TestInlineComments(t *testing.T) {
"blahblah\n<!--- foo -->\nrhubarb\n", "blahblah\n<!--- foo -->\nrhubarb\n",
"<p>blahblah\n<!--- foo -->\nrhubarb</p>\n", "<p>blahblah\n<!--- foo -->\nrhubarb</p>\n",
} }
doTestsInlineParam(t, tests, TestParams{ doTestsInlineParam(t, tests, TestParams{HTMLFlags: Smartypants | SmartypantsDashes})
Options: Options{
Extensions: Smartypants | SmartypantsDashes,
},
})
} }
func TestSmartDoubleQuotes(t *testing.T) { func TestSmartDoubleQuotes(t *testing.T) {
@ -1028,11 +1024,7 @@ func TestSmartDoubleQuotes(t *testing.T) {
"two pair of \"some\" quoted \"text\".\n", "two pair of \"some\" quoted \"text\".\n",
"<p>two pair of &ldquo;some&rdquo; quoted &ldquo;text&rdquo;.</p>\n"} "<p>two pair of &ldquo;some&rdquo; quoted &ldquo;text&rdquo;.</p>\n"}
doTestsInlineParam(t, tests, TestParams{ doTestsInlineParam(t, tests, TestParams{HTMLFlags: Smartypants})
Options: Options{
Extensions: Smartypants,
},
})
} }
func TestSmartAngledDoubleQuotes(t *testing.T) { func TestSmartAngledDoubleQuotes(t *testing.T) {
@ -1044,11 +1036,7 @@ func TestSmartAngledDoubleQuotes(t *testing.T) {
"two pair of \"some\" quoted \"text\".\n", "two pair of \"some\" quoted \"text\".\n",
"<p>two pair of &laquo;some&raquo; quoted &laquo;text&raquo;.</p>\n"} "<p>two pair of &laquo;some&raquo; quoted &laquo;text&raquo;.</p>\n"}
doTestsInlineParam(t, tests, TestParams{ doTestsInlineParam(t, tests, TestParams{HTMLFlags: Smartypants | SmartypantsAngledQuotes})
Options: Options{
Extensions: Smartypants | SmartypantsAngledQuotes,
},
})
} }
func TestSmartFractions(t *testing.T) { func TestSmartFractions(t *testing.T) {
@ -1058,11 +1046,7 @@ func TestSmartFractions(t *testing.T) {
"1/2/2015, 1/4/2015, 3/4/2015; 2015/1/2, 2015/1/4, 2015/3/4.\n", "1/2/2015, 1/4/2015, 3/4/2015; 2015/1/2, 2015/1/4, 2015/3/4.\n",
"<p>1/2/2015, 1/4/2015, 3/4/2015; 2015/1/2, 2015/1/4, 2015/3/4.</p>\n"} "<p>1/2/2015, 1/4/2015, 3/4/2015; 2015/1/2, 2015/1/4, 2015/3/4.</p>\n"}
doTestsInlineParam(t, tests, TestParams{ doTestsInlineParam(t, tests, TestParams{HTMLFlags: Smartypants})
Options: Options{
Extensions: Smartypants,
},
})
tests = []string{ tests = []string{
"1/2, 2/3, 81/100 and 1000000/1048576.\n", "1/2, 2/3, 81/100 and 1000000/1048576.\n",
@ -1070,11 +1054,7 @@ func TestSmartFractions(t *testing.T) {
"1/2/2015, 1/4/2015, 3/4/2015; 2015/1/2, 2015/1/4, 2015/3/4.\n", "1/2/2015, 1/4/2015, 3/4/2015; 2015/1/2, 2015/1/4, 2015/3/4.\n",
"<p>1/2/2015, 1/4/2015, 3/4/2015; 2015/1/2, 2015/1/4, 2015/3/4.</p>\n"} "<p>1/2/2015, 1/4/2015, 3/4/2015; 2015/1/2, 2015/1/4, 2015/3/4.</p>\n"}
doTestsInlineParam(t, tests, TestParams{ doTestsInlineParam(t, tests, TestParams{HTMLFlags: Smartypants | SmartypantsFractions})
Options: Options{
Extensions: Smartypants | SmartypantsFractions,
},
})
} }
func TestDisableSmartDashes(t *testing.T) { func TestDisableSmartDashes(t *testing.T) {
@ -1093,7 +1073,7 @@ func TestDisableSmartDashes(t *testing.T) {
"<p>foo &mdash; bar</p>\n", "<p>foo &mdash; bar</p>\n",
"foo --- bar\n", "foo --- bar\n",
"<p>foo &mdash;&ndash; bar</p>\n", "<p>foo &mdash;&ndash; bar</p>\n",
}, TestParams{Options: Options{Extensions: Smartypants | SmartypantsDashes}}) }, TestParams{HTMLFlags: Smartypants | SmartypantsDashes})
doTestsInlineParam(t, []string{ doTestsInlineParam(t, []string{
"foo - bar\n", "foo - bar\n",
"<p>foo - bar</p>\n", "<p>foo - bar</p>\n",
@ -1101,11 +1081,7 @@ func TestDisableSmartDashes(t *testing.T) {
"<p>foo &ndash; bar</p>\n", "<p>foo &ndash; bar</p>\n",
"foo --- bar\n", "foo --- bar\n",
"<p>foo &mdash; bar</p>\n", "<p>foo &mdash; bar</p>\n",
}, TestParams{ }, TestParams{HTMLFlags: Smartypants | SmartypantsLatexDashes | SmartypantsDashes})
Options: Options{
Extensions: Smartypants | SmartypantsLatexDashes | SmartypantsDashes,
},
})
doTestsInlineParam(t, []string{ doTestsInlineParam(t, []string{
"foo - bar\n", "foo - bar\n",
"<p>foo - bar</p>\n", "<p>foo - bar</p>\n",
@ -1113,9 +1089,7 @@ func TestDisableSmartDashes(t *testing.T) {
"<p>foo -- bar</p>\n", "<p>foo -- bar</p>\n",
"foo --- bar\n", "foo --- bar\n",
"<p>foo --- bar</p>\n", "<p>foo --- bar</p>\n",
}, TestParams{ }, TestParams{HTMLFlags: Smartypants | SmartypantsLatexDashes})
Options: Options{Extensions: Smartypants | SmartypantsLatexDashes},
})
} }
func TestSkipLinks(t *testing.T) { func TestSkipLinks(t *testing.T) {

View File

@ -53,20 +53,15 @@ const (
AutoHeaderIDs // Create the header ID from the text AutoHeaderIDs // Create the header ID from the text
BackslashLineBreak // Translate trailing backslashes into line breaks BackslashLineBreak // Translate trailing backslashes into line breaks
DefinitionLists // Render definition lists DefinitionLists // Render definition lists
Smartypants // Enable smart punctuation substitutions
SmartypantsFractions // Enable smart fractions (with Smartypants)
SmartypantsDashes // Enable smart dashes (with Smartypants)
SmartypantsLatexDashes // Enable LaTeX-style dashes (with Smartypants)
SmartypantsAngledQuotes // Enable angled double quotes (with Smartypants) for double quotes rendering
TOC // Generate a table of contents TOC // Generate a table of contents
OmitContents // Skip the main contents (for a standalone table of contents) OmitContents // Skip the main contents (for a standalone table of contents)
CommonHTMLFlags HTMLFlags = UseXHTML CommonHTMLFlags HTMLFlags = UseXHTML | Smartypants |
SmartypantsFractions | SmartypantsDashes | SmartypantsLatexDashes
CommonExtensions Extensions = NoIntraEmphasis | Tables | FencedCode | CommonExtensions Extensions = NoIntraEmphasis | Tables | FencedCode |
Autolink | Strikethrough | SpaceHeaders | HeaderIDs | Autolink | Strikethrough | SpaceHeaders | HeaderIDs |
BackslashLineBreak | DefinitionLists | Smartypants | BackslashLineBreak | DefinitionLists
SmartypantsFractions | SmartypantsDashes | SmartypantsLatexDashes
) )
// DefaultOptions is a convenience variable with all the options that are // DefaultOptions is a convenience variable with all the options that are

View File

@ -368,7 +368,7 @@ func (r *SPRenderer) smartLeftAngle(out *bytes.Buffer, previousChar byte, text [
type smartCallback func(out *bytes.Buffer, previousChar byte, text []byte) int type smartCallback func(out *bytes.Buffer, previousChar byte, text []byte) int
// NewSmartypantsRenderer constructs a Smartypants renderer object. // NewSmartypantsRenderer constructs a Smartypants renderer object.
func NewSmartypantsRenderer(flags Extensions) *SPRenderer { func NewSmartypantsRenderer(flags HTMLFlags) *SPRenderer {
var r SPRenderer var r SPRenderer
if flags&SmartypantsAngledQuotes == 0 { if flags&SmartypantsAngledQuotes == 0 {
r.callbacks['"'] = r.smartDoubleQuote r.callbacks['"'] = r.smartDoubleQuote
@ -403,8 +403,6 @@ func NewSmartypantsRenderer(flags Extensions) *SPRenderer {
// Process is the entry point of the Smartypants renderer. // Process is the entry point of the Smartypants renderer.
func (r *SPRenderer) Process(text []byte) []byte { func (r *SPRenderer) Process(text []byte) []byte {
var buff bytes.Buffer var buff bytes.Buffer
// first do normal entity escaping
text = esc(text)
mark := 0 mark := 0
for i := 0; i < len(text); i++ { for i := 0; i < len(text); i++ {
if action := r.callbacks[text[i]]; action != nil { if action := r.callbacks[text[i]]; action != nil {