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:
parent
37141d5b5a
commit
02da1dfe9d
@ -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
33
html.go
@ -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 {
|
||||||
|
@ -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 “some” quoted “text”.</p>\n"}
|
"<p>two pair of “some” quoted “text”.</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 «some» quoted «text».</p>\n"}
|
"<p>two pair of «some» quoted «text».</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 — 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{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 – 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 | 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) {
|
||||||
|
11
markdown.go
11
markdown.go
@ -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
|
||||||
|
@ -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 {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user