mirror of
https://git.sr.ht/~adnano/kiln
synced 2025-01-07 17:33:55 +00:00
html: Make textToHTML accept an io.Reader
Use gemini.ParseLines to convert the Gemini text to HTML in one pass.
This commit is contained in:
parent
9154319862
commit
1ff05f49f5
|
@ -4,8 +4,6 @@ import (
|
||||||
"bytes"
|
"bytes"
|
||||||
pathpkg "path"
|
pathpkg "path"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
"git.sr.ht/~adnano/go-gemini"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
// Format represents an output format.
|
// Format represents an output format.
|
||||||
|
@ -31,8 +29,7 @@ func FormatHTML(p *Page, cfg *Config) (path string, content []byte) {
|
||||||
path = pathpkg.Join(p.Path, "index.html")
|
path = pathpkg.Join(p.Path, "index.html")
|
||||||
|
|
||||||
r := strings.NewReader(p.Content)
|
r := strings.NewReader(p.Content)
|
||||||
text := gemini.ParseText(r)
|
content = textToHTML(r)
|
||||||
content = textToHTML(text)
|
|
||||||
|
|
||||||
// html template context
|
// html template context
|
||||||
type htmlCtx struct {
|
type htmlCtx struct {
|
||||||
|
|
90
html.go
90
html.go
|
@ -4,73 +4,79 @@ import (
|
||||||
"bytes"
|
"bytes"
|
||||||
"fmt"
|
"fmt"
|
||||||
"html"
|
"html"
|
||||||
|
"io"
|
||||||
|
|
||||||
"git.sr.ht/~adnano/go-gemini"
|
"git.sr.ht/~adnano/go-gemini"
|
||||||
)
|
)
|
||||||
|
|
||||||
// textToHTML returns the Gemini text response as HTML.
|
// textToHTML returns the Gemini text response as HTML.
|
||||||
func textToHTML(text gemini.Text) []byte {
|
func textToHTML(r io.Reader) []byte {
|
||||||
var b bytes.Buffer
|
buf := &bytes.Buffer{}
|
||||||
var pre bool
|
hw := &HTMLWriter{
|
||||||
var list bool
|
out: buf,
|
||||||
for _, l := range text {
|
|
||||||
if _, ok := l.(gemini.LineListItem); ok {
|
|
||||||
if !list {
|
|
||||||
list = true
|
|
||||||
b.WriteString("<ul>\n")
|
|
||||||
}
|
}
|
||||||
} else if list {
|
defer hw.Finish()
|
||||||
list = false
|
gemini.ParseLines(r, hw.Handle)
|
||||||
b.WriteString("</ul>\n")
|
return buf.Bytes()
|
||||||
}
|
}
|
||||||
switch l.(type) {
|
|
||||||
|
type HTMLWriter struct {
|
||||||
|
out io.Writer
|
||||||
|
pre bool
|
||||||
|
list bool
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *HTMLWriter) Handle(line gemini.Line) {
|
||||||
|
if _, ok := line.(gemini.LineListItem); ok {
|
||||||
|
if !h.list {
|
||||||
|
h.list = true
|
||||||
|
fmt.Fprint(h.out, "<ul>\n")
|
||||||
|
}
|
||||||
|
} else if h.list {
|
||||||
|
h.list = false
|
||||||
|
fmt.Fprint(h.out, "</ul>\n")
|
||||||
|
}
|
||||||
|
switch line := line.(type) {
|
||||||
case gemini.LineLink:
|
case gemini.LineLink:
|
||||||
link := l.(gemini.LineLink)
|
url := html.EscapeString(line.URL)
|
||||||
url := html.EscapeString(link.URL)
|
name := html.EscapeString(line.Name)
|
||||||
name := html.EscapeString(link.Name)
|
|
||||||
if name == "" {
|
if name == "" {
|
||||||
name = url
|
name = url
|
||||||
}
|
}
|
||||||
fmt.Fprintf(&b, "<p><a href='%s'>%s</a></p>\n", url, name)
|
fmt.Fprintf(h.out, "<p><a href='%s'>%s</a></p>\n", url, name)
|
||||||
case gemini.LinePreformattingToggle:
|
case gemini.LinePreformattingToggle:
|
||||||
pre = !pre
|
h.pre = !h.pre
|
||||||
if pre {
|
if h.pre {
|
||||||
b.WriteString("<pre>\n")
|
fmt.Fprint(h.out, "<pre>\n")
|
||||||
} else {
|
} else {
|
||||||
b.WriteString("</pre>\n")
|
fmt.Fprint(h.out, "</pre>\n")
|
||||||
}
|
}
|
||||||
case gemini.LinePreformattedText:
|
case gemini.LinePreformattedText:
|
||||||
text := string(l.(gemini.LinePreformattedText))
|
fmt.Fprintf(h.out, "%s\n", html.EscapeString(string(line)))
|
||||||
fmt.Fprintf(&b, "%s\n", html.EscapeString(text))
|
|
||||||
case gemini.LineHeading1:
|
case gemini.LineHeading1:
|
||||||
text := string(l.(gemini.LineHeading1))
|
fmt.Fprintf(h.out, "<h1>%s</h1>\n", html.EscapeString(string(line)))
|
||||||
fmt.Fprintf(&b, "<h1>%s</h1>\n", html.EscapeString(text))
|
|
||||||
case gemini.LineHeading2:
|
case gemini.LineHeading2:
|
||||||
text := string(l.(gemini.LineHeading2))
|
fmt.Fprintf(h.out, "<h2>%s</h2>\n", html.EscapeString(string(line)))
|
||||||
fmt.Fprintf(&b, "<h2>%s</h2>\n", html.EscapeString(text))
|
|
||||||
case gemini.LineHeading3:
|
case gemini.LineHeading3:
|
||||||
text := string(l.(gemini.LineHeading3))
|
fmt.Fprintf(h.out, "<h3>%s</h3>\n", html.EscapeString(string(line)))
|
||||||
fmt.Fprintf(&b, "<h3>%s</h3>\n", html.EscapeString(text))
|
|
||||||
case gemini.LineListItem:
|
case gemini.LineListItem:
|
||||||
text := string(l.(gemini.LineListItem))
|
fmt.Fprintf(h.out, "<li>%s</li>\n", html.EscapeString(string(line)))
|
||||||
fmt.Fprintf(&b, "<li>%s</li>\n", html.EscapeString(text))
|
|
||||||
case gemini.LineQuote:
|
case gemini.LineQuote:
|
||||||
text := string(l.(gemini.LineQuote))
|
fmt.Fprintf(h.out, "<blockquote>%s</blockquote>\n", html.EscapeString(string(line)))
|
||||||
fmt.Fprintf(&b, "<blockquote>%s</blockquote>\n", html.EscapeString(text))
|
|
||||||
case gemini.LineText:
|
case gemini.LineText:
|
||||||
text := string(l.(gemini.LineText))
|
if line == "" {
|
||||||
if text == "" {
|
fmt.Fprint(h.out, "<br>\n")
|
||||||
b.WriteString("<br>\n")
|
|
||||||
} else {
|
} else {
|
||||||
fmt.Fprintf(&b, "<p>%s</p>\n", html.EscapeString(text))
|
fmt.Fprintf(h.out, "<p>%s</p>\n", html.EscapeString(string(line)))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if pre {
|
|
||||||
b.WriteString("</pre>\n")
|
func (h *HTMLWriter) Finish() {
|
||||||
|
if h.pre {
|
||||||
|
fmt.Fprint(h.out, "</pre>\n")
|
||||||
}
|
}
|
||||||
if list {
|
if h.list {
|
||||||
b.WriteString("</ul>\n")
|
fmt.Fprint(h.out, "</ul>\n")
|
||||||
}
|
}
|
||||||
return b.Bytes()
|
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue