Skip to content

Commit d89029e

Browse files
authored
Ensure empty lines are copiable and final new line too (#16678) (#16692)
Backport #16678 When files are highlighted the newline character needs to be added in a whitespace compliant mode. Also ensure the final empty newline is rendered. Fix #16434 Signed-off-by: Andrew Thornton <[email protected]>
1 parent 62315ea commit d89029e

File tree

2 files changed

+116
-0
lines changed

2 files changed

+116
-0
lines changed

modules/highlight/highlight.go

Lines changed: 13 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -166,16 +166,29 @@ func File(numLines int, fileName string, code []byte) map[int]string {
166166
}
167167

168168
htmlw.Flush()
169+
finalNewLine := false
170+
if len(code) > 0 {
171+
finalNewLine = code[len(code)-1] == '\n'
172+
}
173+
169174
m := make(map[int]string, numLines)
170175
for k, v := range strings.SplitN(htmlbuf.String(), "\n", numLines) {
171176
line := k + 1
172177
content := string(v)
173178
//need to keep lines that are only \n so copy/paste works properly in browser
174179
if content == "" {
175180
content = "\n"
181+
} else if content == `</span><span class="w">` {
182+
content += "\n</span>"
176183
}
184+
content = strings.TrimSuffix(content, `<span class="w">`)
185+
content = strings.TrimPrefix(content, `</span>`)
177186
m[line] = content
178187
}
188+
if finalNewLine {
189+
m[numLines+1] = "<span class=\"w\">\n</span>"
190+
}
191+
179192
return m
180193
}
181194

modules/highlight/highlight_test.go

Lines changed: 103 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,103 @@
1+
// Copyright 2021 The Gitea Authors. All rights reserved.
2+
// Use of this source code is governed by a MIT-style
3+
// license that can be found in the LICENSE file.
4+
5+
package highlight
6+
7+
import (
8+
"reflect"
9+
"testing"
10+
11+
"code.gitea.io/gitea/modules/setting"
12+
"gopkg.in/ini.v1"
13+
)
14+
15+
func TestFile(t *testing.T) {
16+
setting.Cfg = ini.Empty()
17+
tests := []struct {
18+
name string
19+
numLines int
20+
fileName string
21+
code string
22+
want map[int]string
23+
}{
24+
{
25+
name: ".drone.yml",
26+
numLines: 12,
27+
fileName: ".drone.yml",
28+
code: `kind: pipeline
29+
name: default
30+
31+
steps:
32+
- name: test
33+
image: golang:1.13
34+
environment:
35+
GOPROXY: https://goproxy.cn
36+
commands:
37+
- go get -u
38+
- go build -v
39+
- go test -v -race -coverprofile=coverage.txt -covermode=atomic
40+
`,
41+
want: map[int]string{
42+
1: `<span class="nt">kind</span><span class="p">:</span><span class="w"> </span><span class="l">pipeline</span>`,
43+
2: `<span class="w"></span><span class="nt">name</span><span class="p">:</span><span class="w"> </span><span class="l">default</span>`,
44+
3: `<span class="w">
45+
</span>`,
46+
4: `<span class="w"></span><span class="nt">steps</span><span class="p">:</span>`,
47+
5: `<span class="w"></span>- <span class="nt">name</span><span class="p">:</span><span class="w"> </span><span class="l">test</span>`,
48+
6: `<span class="w"> </span><span class="nt">image</span><span class="p">:</span><span class="w"> </span><span class="l">golang:1.13</span>`,
49+
7: `<span class="w"> </span><span class="nt">environment</span><span class="p">:</span>`,
50+
8: `<span class="w"></span><span class="w"> </span><span class="nt">GOPROXY</span><span class="p">:</span><span class="w"> </span><span class="l">https://goproxy.cn</span>`,
51+
9: `<span class="w"> </span><span class="nt">commands</span><span class="p">:</span>`,
52+
10: `<span class="w"></span><span class="w"> </span>- <span class="l">go get -u</span>`,
53+
11: `<span class="w"> </span>- <span class="l">go build -v</span>`,
54+
12: `<span class="w"> </span>- <span class="l">go test -v -race -coverprofile=coverage.txt -covermode=atomic</span><span class="w">
55+
</span>`,
56+
13: `<span class="w">
57+
</span>`,
58+
},
59+
},
60+
{
61+
name: ".drone.yml - trailing space",
62+
numLines: 13,
63+
fileName: ".drone.yml",
64+
code: `kind: pipeline
65+
name: default ` + `
66+
67+
steps:
68+
- name: test
69+
image: golang:1.13
70+
environment:
71+
GOPROXY: https://goproxy.cn
72+
commands:
73+
- go get -u
74+
- go build -v
75+
- go test -v -race -coverprofile=coverage.txt -covermode=atomic
76+
`,
77+
want: map[int]string{
78+
1: `<span class="nt">kind</span><span class="p">:</span><span class="w"> </span><span class="l">pipeline</span>`,
79+
2: `<span class="w"></span><span class="nt">name</span><span class="p">:</span><span class="w"> </span><span class="l">default </span>`,
80+
3: `<span class="w">
81+
</span>`,
82+
4: `<span class="w"></span><span class="nt">steps</span><span class="p">:</span>`,
83+
5: `<span class="w"></span>- <span class="nt">name</span><span class="p">:</span><span class="w"> </span><span class="l">test</span>`,
84+
6: `<span class="w"> </span><span class="nt">image</span><span class="p">:</span><span class="w"> </span><span class="l">golang:1.13</span>`,
85+
7: `<span class="w"> </span><span class="nt">environment</span><span class="p">:</span>`,
86+
8: `<span class="w"></span><span class="w"> </span><span class="nt">GOPROXY</span><span class="p">:</span><span class="w"> </span><span class="l">https://goproxy.cn</span>`,
87+
9: `<span class="w"> </span><span class="nt">commands</span><span class="p">:</span>`,
88+
10: `<span class="w"></span><span class="w"> </span>- <span class="l">go get -u</span>`,
89+
11: `<span class="w"> </span>- <span class="l">go build -v</span>`,
90+
12: `<span class="w"> </span>- <span class="l">go test -v -race -coverprofile=coverage.txt -covermode=atomic</span>`,
91+
13: `<span class="w"> </span>`,
92+
},
93+
},
94+
}
95+
96+
for _, tt := range tests {
97+
t.Run(tt.name, func(t *testing.T) {
98+
if got := File(tt.numLines, tt.fileName, []byte(tt.code)); !reflect.DeepEqual(got, tt.want) {
99+
t.Errorf("File() = %v, want %v", got, tt.want)
100+
}
101+
})
102+
}
103+
}

0 commit comments

Comments
 (0)