diff options
author | zeripath <art27@cantab.net> | 2021-08-14 00:16:56 +0100 |
---|---|---|
committer | GitHub <noreply@github.com> | 2021-08-14 01:16:56 +0200 |
commit | 23a87a003eb2b592ad5295f73d8d52d3f24069c6 (patch) | |
tree | 761cce20349f43b20702ee67d18629835db65a57 | |
parent | a4962a944002c3d3949520949f964aae7ff478ed (diff) | |
download | gitea-23a87a003eb2b592ad5295f73d8d52d3f24069c6.tar.gz gitea-23a87a003eb2b592ad5295f73d8d52d3f24069c6.zip |
Ensure empty lines are copiable and final new line too (#16678)
* Ensure empty lines are copiable and final new line too
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
* Add test and ensure spans closed
Signed-off-by: Andrew Thornton <art27@cantab.net>
-rw-r--r-- | modules/highlight/highlight.go | 13 | ||||
-rw-r--r-- | modules/highlight/highlight_test.go | 103 |
2 files changed, 116 insertions, 0 deletions
diff --git a/modules/highlight/highlight.go b/modules/highlight/highlight.go index 568035fbb7..ed0548578a 100644 --- a/modules/highlight/highlight.go +++ b/modules/highlight/highlight.go @@ -166,6 +166,11 @@ func File(numLines int, fileName string, code []byte) map[int]string { } htmlw.Flush() + finalNewLine := false + if len(code) > 0 { + finalNewLine = code[len(code)-1] == '\n' + } + m := make(map[int]string, numLines) for k, v := range strings.SplitN(htmlbuf.String(), "\n", numLines) { line := k + 1 @@ -173,9 +178,17 @@ func File(numLines int, fileName string, code []byte) map[int]string { //need to keep lines that are only \n so copy/paste works properly in browser if content == "" { content = "\n" + } else if content == `</span><span class="w">` { + content += "\n</span>" } + content = strings.TrimSuffix(content, `<span class="w">`) + content = strings.TrimPrefix(content, `</span>`) m[line] = content } + if finalNewLine { + m[numLines+1] = "<span class=\"w\">\n</span>" + } + return m } diff --git a/modules/highlight/highlight_test.go b/modules/highlight/highlight_test.go new file mode 100644 index 0000000000..7c5afaa52c --- /dev/null +++ b/modules/highlight/highlight_test.go @@ -0,0 +1,103 @@ +// Copyright 2021 The Gitea Authors. All rights reserved. +// Use of this source code is governed by a MIT-style +// license that can be found in the LICENSE file. + +package highlight + +import ( + "reflect" + "testing" + + "code.gitea.io/gitea/modules/setting" + "gopkg.in/ini.v1" +) + +func TestFile(t *testing.T) { + setting.Cfg = ini.Empty() + tests := []struct { + name string + numLines int + fileName string + code string + want map[int]string + }{ + { + name: ".drone.yml", + numLines: 12, + fileName: ".drone.yml", + code: `kind: pipeline +name: default + +steps: +- name: test + image: golang:1.13 + environment: + GOPROXY: https://goproxy.cn + commands: + - go get -u + - go build -v + - go test -v -race -coverprofile=coverage.txt -covermode=atomic +`, + want: map[int]string{ + 1: `<span class="nt">kind</span><span class="p">:</span><span class="w"> </span><span class="l">pipeline</span>`, + 2: `<span class="w"></span><span class="nt">name</span><span class="p">:</span><span class="w"> </span><span class="l">default</span>`, + 3: `<span class="w"> +</span>`, + 4: `<span class="w"></span><span class="nt">steps</span><span class="p">:</span>`, + 5: `<span class="w"></span>- <span class="nt">name</span><span class="p">:</span><span class="w"> </span><span class="l">test</span>`, + 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>`, + 7: `<span class="w"> </span><span class="nt">environment</span><span class="p">:</span>`, + 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>`, + 9: `<span class="w"> </span><span class="nt">commands</span><span class="p">:</span>`, + 10: `<span class="w"></span><span class="w"> </span>- <span class="l">go get -u</span>`, + 11: `<span class="w"> </span>- <span class="l">go build -v</span>`, + 12: `<span class="w"> </span>- <span class="l">go test -v -race -coverprofile=coverage.txt -covermode=atomic</span><span class="w"> +</span>`, + 13: `<span class="w"> +</span>`, + }, + }, + { + name: ".drone.yml - trailing space", + numLines: 13, + fileName: ".drone.yml", + code: `kind: pipeline +name: default ` + ` + +steps: +- name: test + image: golang:1.13 + environment: + GOPROXY: https://goproxy.cn + commands: + - go get -u + - go build -v + - go test -v -race -coverprofile=coverage.txt -covermode=atomic + `, + want: map[int]string{ + 1: `<span class="nt">kind</span><span class="p">:</span><span class="w"> </span><span class="l">pipeline</span>`, + 2: `<span class="w"></span><span class="nt">name</span><span class="p">:</span><span class="w"> </span><span class="l">default </span>`, + 3: `<span class="w"> +</span>`, + 4: `<span class="w"></span><span class="nt">steps</span><span class="p">:</span>`, + 5: `<span class="w"></span>- <span class="nt">name</span><span class="p">:</span><span class="w"> </span><span class="l">test</span>`, + 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>`, + 7: `<span class="w"> </span><span class="nt">environment</span><span class="p">:</span>`, + 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>`, + 9: `<span class="w"> </span><span class="nt">commands</span><span class="p">:</span>`, + 10: `<span class="w"></span><span class="w"> </span>- <span class="l">go get -u</span>`, + 11: `<span class="w"> </span>- <span class="l">go build -v</span>`, + 12: `<span class="w"> </span>- <span class="l">go test -v -race -coverprofile=coverage.txt -covermode=atomic</span>`, + 13: `<span class="w"> </span>`, + }, + }, + } + + for _, tt := range tests { + t.Run(tt.name, func(t *testing.T) { + if got := File(tt.numLines, tt.fileName, []byte(tt.code)); !reflect.DeepEqual(got, tt.want) { + t.Errorf("File() = %v, want %v", got, tt.want) + } + }) + } +} |