renderer.go 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277
  1. package markdown
  2. import (
  3. "strconv"
  4. "strings"
  5. "github.com/alecthomas/chroma/v2"
  6. "github.com/alecthomas/chroma/v2/lexers"
  7. "github.com/alecthomas/chroma/v2/styles"
  8. "github.com/ayn2op/discordo/internal/config"
  9. "github.com/ayn2op/discordo/internal/ui"
  10. "github.com/ayn2op/tview"
  11. "github.com/diamondburned/ningen/v3/discordmd"
  12. "github.com/gdamore/tcell/v3"
  13. "github.com/yuin/goldmark/ast"
  14. )
  15. type Renderer struct {
  16. cfg *config.Config
  17. listIx *int
  18. listNested int
  19. }
  20. const codeBlockIndent = " "
  21. func NewRenderer(cfg *config.Config) *Renderer {
  22. return &Renderer{cfg: cfg}
  23. }
  24. func (r *Renderer) RenderLines(source []byte, node ast.Node, base tcell.Style) []tview.Line {
  25. r.listIx = nil
  26. r.listNested = 0
  27. builder := tview.NewLineBuilder()
  28. styleStack := []tcell.Style{base}
  29. currentStyle := func() tcell.Style {
  30. return styleStack[len(styleStack)-1]
  31. }
  32. pushStyle := func(style tcell.Style) {
  33. styleStack = append(styleStack, style)
  34. }
  35. popStyle := func() {
  36. if len(styleStack) > 1 {
  37. styleStack = styleStack[:len(styleStack)-1]
  38. }
  39. }
  40. theme := r.cfg.Theme.MessagesList
  41. _ = ast.Walk(node, func(node ast.Node, entering bool) (ast.WalkStatus, error) {
  42. switch node := node.(type) {
  43. case *ast.Document:
  44. // noop
  45. case *ast.Heading:
  46. if entering {
  47. builder.Write(strings.Repeat("#", node.Level)+" ", currentStyle())
  48. } else {
  49. builder.NewLine()
  50. }
  51. case *ast.Text:
  52. if entering {
  53. builder.Write(string(node.Segment.Value(source)), currentStyle())
  54. switch {
  55. case node.HardLineBreak():
  56. builder.NewLine()
  57. builder.NewLine()
  58. case node.SoftLineBreak():
  59. builder.NewLine()
  60. }
  61. }
  62. case *ast.FencedCodeBlock:
  63. if entering {
  64. builder.NewLine()
  65. r.renderFencedCodeBlock(builder, source, node, currentStyle())
  66. }
  67. case *ast.AutoLink:
  68. if entering {
  69. style := ui.MergeStyle(currentStyle(), theme.URLStyle.Style)
  70. builder.Write(string(node.URL(source)), style)
  71. }
  72. case *ast.Link:
  73. if entering {
  74. pushStyle(ui.MergeStyle(currentStyle(), theme.URLStyle.Style))
  75. } else {
  76. popStyle()
  77. }
  78. case *ast.List:
  79. if node.IsOrdered() {
  80. start := node.Start
  81. r.listIx = &start
  82. } else {
  83. r.listIx = nil
  84. }
  85. if entering {
  86. builder.NewLine()
  87. r.listNested++
  88. } else {
  89. r.listNested--
  90. }
  91. case *ast.ListItem:
  92. if entering {
  93. builder.Write(strings.Repeat(" ", r.listNested-1), currentStyle())
  94. if r.listIx != nil {
  95. builder.Write(strconv.Itoa(*r.listIx)+". ", currentStyle())
  96. *r.listIx++
  97. } else {
  98. builder.Write("- ", currentStyle())
  99. }
  100. } else {
  101. builder.NewLine()
  102. }
  103. case *discordmd.Inline:
  104. if entering {
  105. pushStyle(applyInlineAttr(currentStyle(), node.Attr))
  106. } else {
  107. popStyle()
  108. }
  109. case *discordmd.Mention:
  110. if entering {
  111. style := ui.MergeStyle(currentStyle(), theme.MentionStyle.Style)
  112. style = style.Bold(true)
  113. builder.Write(mentionText(node), style)
  114. }
  115. case *discordmd.Emoji:
  116. if entering {
  117. style := ui.MergeStyle(currentStyle(), theme.EmojiStyle.Style)
  118. builder.Write(":"+node.Name+":", style)
  119. }
  120. }
  121. return ast.WalkContinue, nil
  122. })
  123. return builder.Finish()
  124. }
  125. func (r *Renderer) renderFencedCodeBlock(builder *tview.LineBuilder, source []byte, node *ast.FencedCodeBlock, base tcell.Style) {
  126. var code strings.Builder
  127. lines := node.Lines()
  128. for i := range lines.Len() {
  129. line := lines.At(i)
  130. code.Write(line.Value(source))
  131. }
  132. language := strings.TrimSpace(string(node.Language(source)))
  133. lexer := lexers.Get(language)
  134. declaredLanguageSupported := lexer != nil
  135. // Detect the language from its content.
  136. var analyzed bool
  137. if lexer == nil {
  138. lexer = lexers.Analyse(code.String())
  139. analyzed = lexer != nil
  140. }
  141. if lexer == nil {
  142. lexer = lexers.Fallback
  143. }
  144. // At this point, it should be noted that some lexers can be extremely chatty.
  145. // To mitigate this, use the coalescing lexer to coalesce runs of identical token types into a single token.
  146. lexer = chroma.Coalesce(lexer)
  147. // Show a fallback header when the language is omitted or unknown.
  148. headerStyle := base.Dim(true)
  149. if analyzed {
  150. builder.Write(codeBlockIndent+"code: analyzed", headerStyle)
  151. builder.NewLine()
  152. } else if language == "" {
  153. builder.Write(codeBlockIndent+"code", headerStyle)
  154. builder.NewLine()
  155. } else if !declaredLanguageSupported {
  156. builder.Write(codeBlockIndent+"code: "+language, headerStyle)
  157. builder.NewLine()
  158. }
  159. iterator, err := lexer.Tokenise(nil, code.String())
  160. if err != nil {
  161. for i := range lines.Len() {
  162. line := lines.At(i)
  163. builder.Write(codeBlockIndent+string(line.Value(source)), base)
  164. }
  165. return
  166. }
  167. theme := styles.Get(r.cfg.Markdown.Theme)
  168. if theme == nil {
  169. theme = styles.Fallback
  170. }
  171. builder.Write(codeBlockIndent, base)
  172. for token := iterator(); token != chroma.EOF; token = iterator() {
  173. style := applyChromaStyle(base, theme.Get(token.Type))
  174. // Chroma tokens may include embedded newlines, so split and re-emit with indentation on each visual line.
  175. parts := strings.Split(token.Value, "\n")
  176. for i, part := range parts {
  177. if i > 0 {
  178. builder.NewLine()
  179. builder.Write(codeBlockIndent, base)
  180. }
  181. if part != "" {
  182. builder.Write(part, style)
  183. }
  184. }
  185. }
  186. }
  187. func applyChromaStyle(base tcell.Style, entry chroma.StyleEntry) tcell.Style {
  188. style := base
  189. if entry.Colour.IsSet() {
  190. style = style.Foreground(tcell.NewRGBColor(
  191. int32(entry.Colour.Red()),
  192. int32(entry.Colour.Green()),
  193. int32(entry.Colour.Blue()),
  194. ))
  195. }
  196. // Intentionally do not apply token background colors so code blocks keep the user's terminal/chat background.
  197. // if entry.Background.IsSet() {
  198. // style = style.Background(tcell.NewRGBColor(
  199. // int32(entry.Background.Red()),
  200. // int32(entry.Background.Green()),
  201. // int32(entry.Background.Blue()),
  202. // ))
  203. // }
  204. switch entry.Bold {
  205. case chroma.Yes:
  206. style = style.Bold(true)
  207. case chroma.No:
  208. style = style.Bold(false)
  209. }
  210. switch entry.Italic {
  211. case chroma.Yes:
  212. style = style.Italic(true)
  213. case chroma.No:
  214. style = style.Italic(false)
  215. }
  216. switch entry.Underline {
  217. case chroma.Yes:
  218. style = style.Underline(true)
  219. case chroma.No:
  220. style = style.Underline(false)
  221. }
  222. return style
  223. }
  224. func mentionText(node *discordmd.Mention) string {
  225. switch {
  226. case node.Channel != nil:
  227. return "#" + node.Channel.Name
  228. case node.GuildUser != nil:
  229. name := node.GuildUser.DisplayOrUsername()
  230. if member := node.GuildUser.Member; member != nil && member.Nick != "" {
  231. name = member.Nick
  232. }
  233. return "@" + name
  234. case node.GuildRole != nil:
  235. return "@" + node.GuildRole.Name
  236. default:
  237. return ""
  238. }
  239. }
  240. func applyInlineAttr(style tcell.Style, attr discordmd.Attribute) tcell.Style {
  241. switch attr {
  242. case discordmd.AttrBold:
  243. return style.Bold(true)
  244. case discordmd.AttrItalics:
  245. return style.Italic(true)
  246. case discordmd.AttrUnderline:
  247. return style.Underline(true)
  248. case discordmd.AttrStrikethrough:
  249. return style.StrikeThrough(true)
  250. case discordmd.AttrMonospace:
  251. return style.Reverse(true)
  252. }
  253. return style
  254. }