-
Notifications
You must be signed in to change notification settings - Fork 49
/
Copy pathline.go
117 lines (97 loc) · 2.54 KB
/
line.go
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
113
114
115
116
117
package ace
import (
"fmt"
"strings"
)
const unicodeSpace = 32
const indentTop = 0
// line represents a line of codes.
type line struct {
no int
str string
indent int
tokens []string
opts *Options
file *File
}
// isEmpty returns true if the line is empty.
func (l *line) isEmpty() bool {
return strings.TrimSpace(l.str) == ""
}
// isTopIndent returns true if the line's indent is the top level.
func (l *line) isTopIndent() bool {
return l.indent == indentTop
}
// isHelperMethod returns true if the line is a helper method.
func (l *line) isHelperMethod() bool {
return len(l.tokens) > 1 && l.tokens[0] == equal
}
// isHelperMethodOf returns true if the line is a specified helper method.
func (l *line) isHelperMethodOf(name string) bool {
return l.isHelperMethod() && l.tokens[1] == name
}
// isPlainText returns true if the line is a plain text.
func (l *line) isPlainText() bool {
return len(l.tokens) > 0 && (l.tokens[0] == pipe || l.tokens[0] == doublePipe)
}
// isComment returns true if the line is a comment.
func (l *line) isComment() bool {
return len(l.tokens) > 0 && l.tokens[0] == slash
}
// isHTMLComment returns true if the line is an HTML comment.
func (l *line) isHTMLComment() bool {
return len(l.tokens) > 0 && l.tokens[0] == slash+slash
}
// isAction returns true if the line is an action.
func (l *line) isAction() bool {
str := strings.TrimSpace(l.str)
return strings.HasPrefix(str, l.opts.DelimLeft) && strings.HasSuffix(str, l.opts.DelimRight)
}
// fileName returns the file name.
func (l *line) fileName() string {
return l.file.path + dot + l.opts.Extension
}
// childOf returns true if the line is a child of the element.
func (l *line) childOf(parent element) (bool, error) {
var ok bool
var err error
switch {
case l.isEmpty():
ok = true
case parent.ContainPlainText():
switch {
case parent.Base().ln.indent < l.indent:
ok = true
}
default:
switch {
case l.indent == parent.Base().ln.indent+1:
ok = true
case l.indent > parent.Base().ln.indent+1:
err = fmt.Errorf("the indent is invalid [file: %s][line: %d]", l.fileName(), l.no)
}
}
return ok, err
}
// newLine creates and returns a line.
func newLine(no int, str string, opts *Options, f *File) *line {
return &line{
no: no,
str: str,
indent: indent(str),
tokens: strings.Split(strings.TrimLeft(str, space), space),
opts: opts,
file: f,
}
}
// indent returns the line's indent.
func indent(str string) int {
var i int
for _, b := range str {
if b != unicodeSpace {
break
}
i++
}
return i / 2
}