renderer.go 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165
  1. package markdown
  2. import (
  3. "fmt"
  4. "io"
  5. "github.com/diamondburned/ningen/v3/discordmd"
  6. "github.com/yuin/goldmark/ast"
  7. gmr "github.com/yuin/goldmark/renderer"
  8. )
  9. var DefaultRenderer = newRenderer()
  10. type renderer struct {
  11. config *gmr.Config
  12. }
  13. func newRenderer() *renderer {
  14. config := gmr.NewConfig()
  15. return &renderer{config}
  16. }
  17. // AddOptions implements renderer.Renderer.
  18. func (r *renderer) AddOptions(opts ...gmr.Option) {
  19. for _, opt := range opts {
  20. opt.SetConfig(r.config)
  21. }
  22. }
  23. func (r *renderer) Render(w io.Writer, source []byte, n ast.Node) error {
  24. return ast.Walk(n, func(n ast.Node, entering bool) (ast.WalkStatus, error) {
  25. switch n := n.(type) {
  26. case *ast.Document:
  27. // noop
  28. case *ast.Heading:
  29. r.renderHeading(w)
  30. case *ast.Text:
  31. r.renderText(w, n, entering, source)
  32. case *ast.FencedCodeBlock:
  33. r.renderFencedCodeBlock(w, n, entering, source)
  34. case *ast.AutoLink:
  35. r.renderAutoLink(w, n, entering, source)
  36. case *ast.Link:
  37. r.renderLink(w, n, entering)
  38. case *discordmd.Inline:
  39. r.renderInline(w, n, entering)
  40. case *discordmd.Mention:
  41. r.renderMention(w, n, entering)
  42. case *discordmd.Emoji:
  43. r.renderEmoji(w, n, entering)
  44. }
  45. return ast.WalkContinue, nil
  46. })
  47. }
  48. func (r *renderer) renderHeading(w io.Writer) {
  49. io.WriteString(w, "\n")
  50. }
  51. func (r *renderer) renderFencedCodeBlock(w io.Writer, n *ast.FencedCodeBlock, entering bool, source []byte) {
  52. io.WriteString(w, "\n")
  53. if entering {
  54. // body
  55. for i := range n.Lines().Len() {
  56. line := n.Lines().At(i)
  57. io.WriteString(w, "| ")
  58. w.Write(line.Value(source))
  59. }
  60. }
  61. }
  62. func (r *renderer) renderAutoLink(w io.Writer, n *ast.AutoLink, entering bool, source []byte) {
  63. if entering {
  64. linkColor := r.config.Options["linkColor"].(string)
  65. io.WriteString(w, "["+linkColor+"]")
  66. w.Write(n.URL(source))
  67. } else {
  68. io.WriteString(w, "[-::]")
  69. }
  70. }
  71. func (r *renderer) renderLink(w io.Writer, n *ast.Link, entering bool) {
  72. if entering {
  73. linkColor := r.config.Options["linkColor"].(string)
  74. io.WriteString(w, fmt.Sprintf("[%s:::%s]", linkColor, n.Destination))
  75. } else {
  76. io.WriteString(w, "[-:::-]")
  77. }
  78. }
  79. func (r *renderer) renderText(w io.Writer, n *ast.Text, entering bool, source []byte) {
  80. if entering {
  81. w.Write(n.Segment.Value(source))
  82. switch {
  83. case n.HardLineBreak():
  84. io.WriteString(w, "\n\n")
  85. case n.SoftLineBreak():
  86. io.WriteString(w, "\n")
  87. }
  88. }
  89. }
  90. func (r *renderer) renderInline(w io.Writer, n *discordmd.Inline, entering bool) {
  91. if entering {
  92. switch n.Attr {
  93. case discordmd.AttrBold:
  94. io.WriteString(w, "[::b]")
  95. case discordmd.AttrItalics:
  96. io.WriteString(w, "[::i]")
  97. case discordmd.AttrUnderline:
  98. io.WriteString(w, "[::u]")
  99. case discordmd.AttrStrikethrough:
  100. io.WriteString(w, "[::s]")
  101. case discordmd.AttrMonospace:
  102. io.WriteString(w, "[::r]")
  103. }
  104. } else {
  105. switch n.Attr {
  106. case discordmd.AttrBold:
  107. io.WriteString(w, "[::B]")
  108. case discordmd.AttrItalics:
  109. io.WriteString(w, "[::I]")
  110. case discordmd.AttrUnderline:
  111. io.WriteString(w, "[::U]")
  112. case discordmd.AttrStrikethrough:
  113. io.WriteString(w, "[::S]")
  114. case discordmd.AttrMonospace:
  115. io.WriteString(w, "[::R]")
  116. }
  117. }
  118. }
  119. func (r *renderer) renderMention(w io.Writer, n *discordmd.Mention, entering bool) {
  120. if entering {
  121. io.WriteString(w, "[::b]")
  122. switch {
  123. case n.Channel != nil:
  124. io.WriteString(w, "#"+n.Channel.Name)
  125. case n.GuildUser != nil:
  126. username := n.GuildUser.DisplayOrUsername()
  127. if r.config.Options["showNicknames"].(bool) && n.GuildUser.Member != nil && n.GuildUser.Member.Nick != "" {
  128. username = n.GuildUser.Member.Nick
  129. }
  130. io.WriteString(w, "@"+username)
  131. case n.GuildRole != nil:
  132. io.WriteString(w, "@"+n.GuildRole.Name)
  133. }
  134. } else {
  135. io.WriteString(w, "[::B]")
  136. }
  137. }
  138. func (r *renderer) renderEmoji(w io.Writer, n *discordmd.Emoji, entering bool) {
  139. if entering {
  140. emojiColor := r.config.Options["emojiColor"].(string)
  141. io.WriteString(w, "["+emojiColor+"]")
  142. io.WriteString(w, ":"+n.Name+":")
  143. } else {
  144. io.WriteString(w, "[-]")
  145. }
  146. }