summaryrefslogtreecommitdiffstats
path: root/utils/markdown/fenced_code.go
blob: 8b2ebd4fe010327612879c362925d9371e3ab8f9 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
// Copyright (c) 2017-present Mattermost, Inc. All Rights Reserved.
// See License.txt for license information.

package markdown

import (
	"strings"
)

type FencedCodeLine struct {
	Indentation int
	Range       Range
}

type FencedCode struct {
	blockBase
	markdown           string
	didSeeClosingFence bool

	Indentation  int
	OpeningFence Range
	RawInfo      Range
	RawCode      []FencedCodeLine
}

func (b *FencedCode) Code() (result string) {
	for _, code := range b.RawCode {
		result += strings.Repeat(" ", code.Indentation) + b.markdown[code.Range.Position:code.Range.End]
	}
	return
}

func (b *FencedCode) Info() string {
	return Unescape(b.markdown[b.RawInfo.Position:b.RawInfo.End])
}

func (b *FencedCode) Continuation(indentation int, r Range) *continuation {
	if b.didSeeClosingFence {
		return nil
	}
	return &continuation{
		Indentation: indentation,
		Remaining:   r,
	}
}

func (b *FencedCode) AddLine(indentation int, r Range) bool {
	s := b.markdown[r.Position:r.End]
	if indentation <= 3 && strings.HasPrefix(s, b.markdown[b.OpeningFence.Position:b.OpeningFence.End]) {
		suffix := strings.TrimSpace(s[b.OpeningFence.End-b.OpeningFence.Position:])
		isClosingFence := true
		for _, c := range suffix {
			if c != rune(s[0]) {
				isClosingFence = false
				break
			}
		}
		if isClosingFence {
			b.didSeeClosingFence = true
			return true
		}
	}

	if indentation >= b.Indentation {
		indentation -= b.Indentation
	} else {
		indentation = 0
	}

	b.RawCode = append(b.RawCode, FencedCodeLine{
		Indentation: indentation,
		Range:       r,
	})
	return true
}

func (b *FencedCode) AllowsBlockStarts() bool {
	return false
}

func fencedCodeStart(markdown string, indentation int, r Range, matchedBlocks, unmatchedBlocks []Block) []Block {
	s := markdown[r.Position:r.End]

	if !strings.HasPrefix(s, "```") && !strings.HasPrefix(s, "~~~") {
		return nil
	}

	fenceCharacter := rune(s[0])
	fenceLength := 3
	for _, c := range s[3:] {
		if c == fenceCharacter {
			fenceLength++
		} else {
			break
		}
	}

	for i := r.Position + fenceLength; i < r.End; i++ {
		if markdown[i] == '`' {
			return nil
		}
	}

	return []Block{
		&FencedCode{
			markdown:     markdown,
			Indentation:  indentation,
			RawInfo:      trimRightSpace(markdown, Range{r.Position + fenceLength, r.End}),
			OpeningFence: Range{r.Position, r.Position + fenceLength},
		},
	}
}