code_span.go 2.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384
  1. package parser
  2. import (
  3. "github.com/yuin/goldmark/ast"
  4. "github.com/yuin/goldmark/text"
  5. )
  6. type codeSpanParser struct {
  7. }
  8. var defaultCodeSpanParser = &codeSpanParser{}
  9. // NewCodeSpanParser return a new InlineParser that parses inline codes
  10. // surrounded by '`' .
  11. func NewCodeSpanParser() InlineParser {
  12. return defaultCodeSpanParser
  13. }
  14. func (s *codeSpanParser) Trigger() []byte {
  15. return []byte{'`'}
  16. }
  17. func (s *codeSpanParser) Parse(parent ast.Node, block text.Reader, pc Context) ast.Node {
  18. line, startSegment := block.PeekLine()
  19. opener := 0
  20. for ; opener < len(line) && line[opener] == '`'; opener++ {
  21. }
  22. block.Advance(opener)
  23. l, pos := block.Position()
  24. node := ast.NewCodeSpan()
  25. for {
  26. line, segment := block.PeekLine()
  27. if line == nil {
  28. block.SetPosition(l, pos)
  29. return ast.NewTextSegment(startSegment.WithStop(startSegment.Start + opener))
  30. }
  31. for i := 0; i < len(line); i++ {
  32. c := line[i]
  33. if c == '`' {
  34. oldi := i
  35. for ; i < len(line) && line[i] == '`'; i++ {
  36. }
  37. closure := i - oldi
  38. if closure == opener && (i >= len(line) || line[i] != '`') {
  39. segment = segment.WithStop(segment.Start + i - closure)
  40. if !segment.IsEmpty() {
  41. node.AppendChild(node, ast.NewRawTextSegment(segment))
  42. }
  43. block.Advance(i)
  44. goto end
  45. }
  46. }
  47. }
  48. node.AppendChild(node, ast.NewRawTextSegment(segment))
  49. block.AdvanceLine()
  50. }
  51. end:
  52. if !node.IsBlank(block.Source()) {
  53. // trim first halfspace and last halfspace
  54. segment := node.FirstChild().(*ast.Text).Segment
  55. shouldTrimmed := true
  56. if !(!segment.IsEmpty() && isSpaceOrNewline(block.Source()[segment.Start])) {
  57. shouldTrimmed = false
  58. }
  59. segment = node.LastChild().(*ast.Text).Segment
  60. if !(!segment.IsEmpty() && isSpaceOrNewline(block.Source()[segment.Stop-1])) {
  61. shouldTrimmed = false
  62. }
  63. if shouldTrimmed {
  64. t := node.FirstChild().(*ast.Text)
  65. segment := t.Segment
  66. t.Segment = segment.WithStart(segment.Start + 1)
  67. t = node.LastChild().(*ast.Text)
  68. segment = node.LastChild().(*ast.Text).Segment
  69. t.Segment = segment.WithStop(segment.Stop - 1)
  70. }
  71. }
  72. return node
  73. }
  74. func isSpaceOrNewline(c byte) bool {
  75. return c == ' ' || c == '\n'
  76. }