GoBlog/markdown.go

118 lines
2.8 KiB
Go
Raw Normal View History

2020-07-28 19:38:12 +00:00
package main
import (
"bytes"
2020-09-18 11:11:25 +00:00
kemoji "github.com/kyokomi/emoji"
2020-07-28 19:38:12 +00:00
"github.com/yuin/goldmark"
"github.com/yuin/goldmark-emoji"
"github.com/yuin/goldmark-emoji/definition"
2020-09-21 14:53:20 +00:00
"github.com/yuin/goldmark/ast"
2020-07-28 19:38:12 +00:00
"github.com/yuin/goldmark/extension"
"github.com/yuin/goldmark/parser"
2020-09-21 14:53:20 +00:00
"github.com/yuin/goldmark/renderer"
2020-07-28 19:38:12 +00:00
"github.com/yuin/goldmark/renderer/html"
2020-09-21 14:53:20 +00:00
"github.com/yuin/goldmark/util"
2020-09-18 11:11:25 +00:00
"strings"
"sync"
2020-07-28 19:38:12 +00:00
)
2020-09-18 11:11:25 +00:00
var emojilib definition.Emojis
var emojiOnce sync.Once
2020-07-28 19:38:12 +00:00
var markdown goldmark.Markdown
2020-08-24 19:09:30 +00:00
func initMarkdown() {
2020-07-28 19:38:12 +00:00
markdown = goldmark.New(
goldmark.WithRendererOptions(
html.WithUnsafe(),
),
goldmark.WithParserOptions(
parser.WithAutoHeadingID(),
),
goldmark.WithExtensions(
extension.GFM,
extension.Footnote,
extension.Typographer,
// Emojis
emoji.New(
2020-09-21 14:53:20 +00:00
emoji.WithEmojis(emojiGoLib()),
),
2020-09-21 14:53:20 +00:00
// Links
newLinkExtension(),
2020-07-28 19:38:12 +00:00
),
)
}
2020-07-29 14:41:36 +00:00
func renderMarkdown(source string) (content []byte, err error) {
2020-07-28 19:38:12 +00:00
var buffer bytes.Buffer
err = markdown.Convert([]byte(source), &buffer)
2020-08-24 19:09:30 +00:00
content = buffer.Bytes()
2020-07-28 19:38:12 +00:00
return
}
2020-09-18 11:11:25 +00:00
2020-09-21 14:53:20 +00:00
// Extensions etc...
// All emojis from emoji lib
func emojiGoLib() definition.Emojis {
2020-09-18 11:11:25 +00:00
emojiOnce.Do(func() {
var emojis []definition.Emoji
for shotcode, e := range kemoji.CodeMap() {
emojis = append(emojis, definition.NewEmoji(e, []rune(e), strings.ReplaceAll(shotcode, ":", "")))
}
emojilib = definition.NewEmojis(emojis...)
})
return emojilib
}
2020-09-21 14:53:20 +00:00
// Links
type linkExtension struct{}
func newLinkExtension() goldmark.Extender {
return &linkExtension{}
}
func (l *linkExtension) Extend(m goldmark.Markdown) {
m.Renderer().AddOptions(renderer.WithNodeRenderers(
util.Prioritized(newLinkRenderer(), 500),
))
}
type linkRenderer struct{}
func (l *linkRenderer) RegisterFuncs(r renderer.NodeRendererFuncRegisterer) {
r.Register(ast.KindLink, l.renderLink)
}
func (l *linkRenderer) renderLink(w util.BufWriter, _ []byte, node ast.Node, entering bool) (ast.WalkStatus, error) {
n := node.(*ast.Link)
if entering {
// Make URL absolute if it's relative
newDestination := string(util.URLEscape(n.Destination, true))
if strings.HasPrefix(newDestination, "/") {
newDestination = appConfig.Server.PublicAddress + newDestination
}
// Write URL
_, _ = w.WriteString("<a href=\"")
_, _ = w.Write(util.EscapeHTML([]byte(newDestination)))
_ = w.WriteByte('"')
// Open external links (links that start with "http") in new tab
if strings.HasPrefix(string(n.Destination), "http") {
_, _ = w.WriteString(` target="_blank" rel="noopener"`)
}
// Title
if n.Title != nil {
_, _ = w.WriteString(` title="`)
_, _ = w.Write(n.Title)
_ = w.WriteByte('"')
}
_ = w.WriteByte('>')
} else {
_, _ = w.WriteString("</a>")
}
return ast.WalkContinue, nil
}
func newLinkRenderer() renderer.NodeRenderer {
return &linkRenderer{}
}