1
0
mirror of https://github.com/osmarks/mycorrhiza.git synced 2025-01-07 10:20:26 +00:00

Add clickable diffs to recent changes view (#215)

* Add clickable diffs to recent changes view

* Reuse entries

* Insert non-breaking space

* Show hash only once, remove parenthesis
This commit is contained in:
Danila Gorelko 2024-03-10 22:39:56 +03:00 committed by GitHub
parent 210615efa2
commit afe2f0c9e2
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
3 changed files with 237 additions and 116 deletions

View File

@ -24,7 +24,9 @@
<time class="recent-changes__entry__time"> <time class="recent-changes__entry__time">
{{ $time.Format "15:04 UTC" }} {{ $time.Format "15:04 UTC" }}
</time> </time>
<span class="recent-changes__entry__message">{{$entry.Hash}}</span> <span class="recent-changes__entry__message">
{{$entry.HyphaeDiffsHTML}}
</span>
{{ if $entry.Username | ne "anon" }} {{ if $entry.Username | ne "anon" }}
<span class="recent-changes__entry__author"> <span class="recent-changes__entry__author">
&mdash; <a href="/hypha/{{$userHypha}}/{{$entry.Username}}" rel="author">{{$entry.Username}}</a> &mdash; <a href="/hypha/{{$userHypha}}/{{$entry.Username}}" rel="author">{{$entry.Username}}</a>

View File

@ -13,6 +13,28 @@ HyphaeLinksHTML returns a comma-separated list of hyphae that were affected by t
{% endstripspace %} {% endstripspace %}
{% endfunc %} {% endfunc %}
HyphaeDiffsHTML returns a comma-separated list of diffs links of current revision for every affected file as HTML string.
{% func (rev Revision) HyphaeDiffsHTML() %}
{% code entries := rev.hyphaeAffected() %}
{% stripspace %}
{% if len(entries) == 1 %}
<a href="/primitive-diff/{%s rev.Hash %}/{%s entries[0] %}">{%s rev.Hash %}</a>
{% else %}
{% for i, hyphaName := range entries %}
{% if i > 0 %}
<span aria-hidden="true">, </span>
{% endif %}
<a href="/primitive-diff/{%s rev.Hash %}/{%s hyphaName %}">
{% if i == 0 %}
{%s rev.Hash %}&nbsp;
{% endif %}
{%s hyphaName %}</a>
{% endfor %}
{% endif %}
{% endstripspace %}
{% endfunc %}
descriptionForFeed generates a good enough HTML contents for a web feed. descriptionForFeed generates a good enough HTML contents for a web feed.
{% func (rev *Revision) descriptionForFeed() %} {% func (rev *Revision) descriptionForFeed() %}
<p><b>{%s rev.Message %}</b> (by {%s rev.Username %} at {%s rev.TimeString() %})</p> <p><b>{%s rev.Message %}</b> (by {%s rev.Username %} at {%s rev.TimeString() %})</p>

View File

@ -82,245 +82,342 @@ func (rev Revision) HyphaeLinksHTML() string {
//line history/view.qtpl:14 //line history/view.qtpl:14
} }
// descriptionForFeed generates a good enough HTML contents for a web feed. // HyphaeDiffsHTML returns a comma-separated list of diffs links of current revision for every affected file as HTML string.
//line history/view.qtpl:17 //line history/view.qtpl:18
func (rev *Revision) streamdescriptionForFeed(qw422016 *qt422016.Writer) { func (rev Revision) StreamHyphaeDiffsHTML(qw422016 *qt422016.Writer) {
//line history/view.qtpl:17 //line history/view.qtpl:18
qw422016.N().S(` qw422016.N().S(`
<p><b>`) `)
//line history/view.qtpl:18
qw422016.E().S(rev.Message)
//line history/view.qtpl:18
qw422016.N().S(`</b> (by `)
//line history/view.qtpl:18
qw422016.E().S(rev.Username)
//line history/view.qtpl:18
qw422016.N().S(` at `)
//line history/view.qtpl:18
qw422016.E().S(rev.TimeString())
//line history/view.qtpl:18
qw422016.N().S(`)</p>
<p>Hyphae affected: `)
//line history/view.qtpl:19 //line history/view.qtpl:19
rev.StreamHyphaeLinksHTML(qw422016) entries := rev.hyphaeAffected()
//line history/view.qtpl:19 //line history/view.qtpl:19
qw422016.N().S(`</p> qw422016.N().S(`
<pre><code>`)
//line history/view.qtpl:20
qw422016.E().S(rev.textDiff())
//line history/view.qtpl:20
qw422016.N().S(`</code></pre>
`) `)
//line history/view.qtpl:21 //line history/view.qtpl:21
if len(entries) == 1 {
//line history/view.qtpl:21
qw422016.N().S(`<a href="/primitive-diff/`)
//line history/view.qtpl:22
qw422016.E().S(rev.Hash)
//line history/view.qtpl:22
qw422016.N().S(`/`)
//line history/view.qtpl:22
qw422016.E().S(entries[0])
//line history/view.qtpl:22
qw422016.N().S(`">`)
//line history/view.qtpl:22
qw422016.E().S(rev.Hash)
//line history/view.qtpl:22
qw422016.N().S(`</a>`)
//line history/view.qtpl:23
} else {
//line history/view.qtpl:24
for i, hyphaName := range entries {
//line history/view.qtpl:25
if i > 0 {
//line history/view.qtpl:25
qw422016.N().S(`<span aria-hidden="true">, </span>`)
//line history/view.qtpl:27
}
//line history/view.qtpl:27
qw422016.N().S(`<a href="/primitive-diff/`)
//line history/view.qtpl:28
qw422016.E().S(rev.Hash)
//line history/view.qtpl:28
qw422016.N().S(`/`)
//line history/view.qtpl:28
qw422016.E().S(hyphaName)
//line history/view.qtpl:28
qw422016.N().S(`">`)
//line history/view.qtpl:29
if i == 0 {
//line history/view.qtpl:30
qw422016.E().S(rev.Hash)
//line history/view.qtpl:30
qw422016.N().S(`&nbsp;`)
//line history/view.qtpl:31
}
//line history/view.qtpl:32
qw422016.E().S(hyphaName)
//line history/view.qtpl:32
qw422016.N().S(`</a>`)
//line history/view.qtpl:33
}
//line history/view.qtpl:34
}
//line history/view.qtpl:35
qw422016.N().S(`
`)
//line history/view.qtpl:36
} }
//line history/view.qtpl:21 //line history/view.qtpl:36
func (rev *Revision) writedescriptionForFeed(qq422016 qtio422016.Writer) { func (rev Revision) WriteHyphaeDiffsHTML(qq422016 qtio422016.Writer) {
//line history/view.qtpl:21 //line history/view.qtpl:36
qw422016 := qt422016.AcquireWriter(qq422016) qw422016 := qt422016.AcquireWriter(qq422016)
//line history/view.qtpl:21 //line history/view.qtpl:36
rev.streamdescriptionForFeed(qw422016) rev.StreamHyphaeDiffsHTML(qw422016)
//line history/view.qtpl:21 //line history/view.qtpl:36
qt422016.ReleaseWriter(qw422016) qt422016.ReleaseWriter(qw422016)
//line history/view.qtpl:21 //line history/view.qtpl:36
} }
//line history/view.qtpl:21 //line history/view.qtpl:36
func (rev *Revision) descriptionForFeed() string { func (rev Revision) HyphaeDiffsHTML() string {
//line history/view.qtpl:21 //line history/view.qtpl:36
qb422016 := qt422016.AcquireByteBuffer() qb422016 := qt422016.AcquireByteBuffer()
//line history/view.qtpl:21 //line history/view.qtpl:36
rev.writedescriptionForFeed(qb422016) rev.WriteHyphaeDiffsHTML(qb422016)
//line history/view.qtpl:21 //line history/view.qtpl:36
qs422016 := string(qb422016.B) qs422016 := string(qb422016.B)
//line history/view.qtpl:21 //line history/view.qtpl:36
qt422016.ReleaseByteBuffer(qb422016) qt422016.ReleaseByteBuffer(qb422016)
//line history/view.qtpl:21 //line history/view.qtpl:36
return qs422016 return qs422016
//line history/view.qtpl:21 //line history/view.qtpl:36
}
// descriptionForFeed generates a good enough HTML contents for a web feed.
//line history/view.qtpl:39
func (rev *Revision) streamdescriptionForFeed(qw422016 *qt422016.Writer) {
//line history/view.qtpl:39
qw422016.N().S(`
<p><b>`)
//line history/view.qtpl:40
qw422016.E().S(rev.Message)
//line history/view.qtpl:40
qw422016.N().S(`</b> (by `)
//line history/view.qtpl:40
qw422016.E().S(rev.Username)
//line history/view.qtpl:40
qw422016.N().S(` at `)
//line history/view.qtpl:40
qw422016.E().S(rev.TimeString())
//line history/view.qtpl:40
qw422016.N().S(`)</p>
<p>Hyphae affected: `)
//line history/view.qtpl:41
rev.StreamHyphaeLinksHTML(qw422016)
//line history/view.qtpl:41
qw422016.N().S(`</p>
<pre><code>`)
//line history/view.qtpl:42
qw422016.E().S(rev.textDiff())
//line history/view.qtpl:42
qw422016.N().S(`</code></pre>
`)
//line history/view.qtpl:43
}
//line history/view.qtpl:43
func (rev *Revision) writedescriptionForFeed(qq422016 qtio422016.Writer) {
//line history/view.qtpl:43
qw422016 := qt422016.AcquireWriter(qq422016)
//line history/view.qtpl:43
rev.streamdescriptionForFeed(qw422016)
//line history/view.qtpl:43
qt422016.ReleaseWriter(qw422016)
//line history/view.qtpl:43
}
//line history/view.qtpl:43
func (rev *Revision) descriptionForFeed() string {
//line history/view.qtpl:43
qb422016 := qt422016.AcquireByteBuffer()
//line history/view.qtpl:43
rev.writedescriptionForFeed(qb422016)
//line history/view.qtpl:43
qs422016 := string(qb422016.B)
//line history/view.qtpl:43
qt422016.ReleaseByteBuffer(qb422016)
//line history/view.qtpl:43
return qs422016
//line history/view.qtpl:43
} }
// WithRevisions returns an html representation of `revs` that is meant to be inserted in a history page. // WithRevisions returns an html representation of `revs` that is meant to be inserted in a history page.
//line history/view.qtpl:24 //line history/view.qtpl:46
func StreamWithRevisions(qw422016 *qt422016.Writer, hyphaName string, revs []Revision) { func StreamWithRevisions(qw422016 *qt422016.Writer, hyphaName string, revs []Revision) {
//line history/view.qtpl:24 //line history/view.qtpl:46
qw422016.N().S(` qw422016.N().S(`
`) `)
//line history/view.qtpl:25 //line history/view.qtpl:47
for _, grp := range groupRevisionsByMonth(revs) { for _, grp := range groupRevisionsByMonth(revs) {
//line history/view.qtpl:25 //line history/view.qtpl:47
qw422016.N().S(` qw422016.N().S(`
`) `)
//line history/view.qtpl:27 //line history/view.qtpl:49
currentYear := grp[0].Time.Year() currentYear := grp[0].Time.Year()
currentMonth := grp[0].Time.Month() currentMonth := grp[0].Time.Month()
sectionId := fmt.Sprintf("%04d-%02d", currentYear, currentMonth) sectionId := fmt.Sprintf("%04d-%02d", currentYear, currentMonth)
//line history/view.qtpl:30 //line history/view.qtpl:52
qw422016.N().S(` qw422016.N().S(`
<section class="history__month"> <section class="history__month">
<a href="#`) <a href="#`)
//line history/view.qtpl:32 //line history/view.qtpl:54
qw422016.E().S(sectionId) qw422016.E().S(sectionId)
//line history/view.qtpl:32 //line history/view.qtpl:54
qw422016.N().S(`" class="history__month-anchor"> qw422016.N().S(`" class="history__month-anchor">
<h2 id="`) <h2 id="`)
//line history/view.qtpl:33 //line history/view.qtpl:55
qw422016.E().S(sectionId) qw422016.E().S(sectionId)
//line history/view.qtpl:33 //line history/view.qtpl:55
qw422016.N().S(`" class="history__month-title">`) qw422016.N().S(`" class="history__month-title">`)
//line history/view.qtpl:33 //line history/view.qtpl:55
qw422016.N().D(currentYear) qw422016.N().D(currentYear)
//line history/view.qtpl:33 //line history/view.qtpl:55
qw422016.N().S(` `) qw422016.N().S(` `)
//line history/view.qtpl:33 //line history/view.qtpl:55
qw422016.E().S(currentMonth.String()) qw422016.E().S(currentMonth.String())
//line history/view.qtpl:33 //line history/view.qtpl:55
qw422016.N().S(`</h2> qw422016.N().S(`</h2>
</a> </a>
<ul class="history__entries"> <ul class="history__entries">
`) `)
//line history/view.qtpl:36 //line history/view.qtpl:58
for _, rev := range grp { for _, rev := range grp {
//line history/view.qtpl:36 //line history/view.qtpl:58
qw422016.N().S(` qw422016.N().S(`
`) `)
//line history/view.qtpl:37 //line history/view.qtpl:59
rev.streamasHistoryEntry(qw422016, hyphaName) rev.streamasHistoryEntry(qw422016, hyphaName)
//line history/view.qtpl:37 //line history/view.qtpl:59
qw422016.N().S(` qw422016.N().S(`
`) `)
//line history/view.qtpl:38 //line history/view.qtpl:60
} }
//line history/view.qtpl:38 //line history/view.qtpl:60
qw422016.N().S(` qw422016.N().S(`
</ul> </ul>
</section> </section>
`) `)
//line history/view.qtpl:41 //line history/view.qtpl:63
} }
//line history/view.qtpl:41 //line history/view.qtpl:63
qw422016.N().S(` qw422016.N().S(`
`) `)
//line history/view.qtpl:42 //line history/view.qtpl:64
} }
//line history/view.qtpl:42 //line history/view.qtpl:64
func WriteWithRevisions(qq422016 qtio422016.Writer, hyphaName string, revs []Revision) { func WriteWithRevisions(qq422016 qtio422016.Writer, hyphaName string, revs []Revision) {
//line history/view.qtpl:42 //line history/view.qtpl:64
qw422016 := qt422016.AcquireWriter(qq422016) qw422016 := qt422016.AcquireWriter(qq422016)
//line history/view.qtpl:42 //line history/view.qtpl:64
StreamWithRevisions(qw422016, hyphaName, revs) StreamWithRevisions(qw422016, hyphaName, revs)
//line history/view.qtpl:42 //line history/view.qtpl:64
qt422016.ReleaseWriter(qw422016) qt422016.ReleaseWriter(qw422016)
//line history/view.qtpl:42 //line history/view.qtpl:64
} }
//line history/view.qtpl:42 //line history/view.qtpl:64
func WithRevisions(hyphaName string, revs []Revision) string { func WithRevisions(hyphaName string, revs []Revision) string {
//line history/view.qtpl:42 //line history/view.qtpl:64
qb422016 := qt422016.AcquireByteBuffer() qb422016 := qt422016.AcquireByteBuffer()
//line history/view.qtpl:42 //line history/view.qtpl:64
WriteWithRevisions(qb422016, hyphaName, revs) WriteWithRevisions(qb422016, hyphaName, revs)
//line history/view.qtpl:42 //line history/view.qtpl:64
qs422016 := string(qb422016.B) qs422016 := string(qb422016.B)
//line history/view.qtpl:42 //line history/view.qtpl:64
qt422016.ReleaseByteBuffer(qb422016) qt422016.ReleaseByteBuffer(qb422016)
//line history/view.qtpl:42 //line history/view.qtpl:64
return qs422016 return qs422016
//line history/view.qtpl:42 //line history/view.qtpl:64
} }
//line history/view.qtpl:44 //line history/view.qtpl:66
func (rev *Revision) streamasHistoryEntry(qw422016 *qt422016.Writer, hyphaName string) { func (rev *Revision) streamasHistoryEntry(qw422016 *qt422016.Writer, hyphaName string) {
//line history/view.qtpl:44 //line history/view.qtpl:66
qw422016.N().S(` qw422016.N().S(`
<li class="history__entry"> <li class="history__entry">
<a class="history-entry" href="/rev/`) <a class="history-entry" href="/rev/`)
//line history/view.qtpl:46 //line history/view.qtpl:68
qw422016.E().S(rev.Hash) qw422016.E().S(rev.Hash)
//line history/view.qtpl:46 //line history/view.qtpl:68
qw422016.N().S(`/`) qw422016.N().S(`/`)
//line history/view.qtpl:46 //line history/view.qtpl:68
qw422016.E().S(hyphaName) qw422016.E().S(hyphaName)
//line history/view.qtpl:46 //line history/view.qtpl:68
qw422016.N().S(`"> qw422016.N().S(`">
<time class="history-entry__time">`) <time class="history-entry__time">`)
//line history/view.qtpl:47 //line history/view.qtpl:69
qw422016.E().S(rev.timeToDisplay()) qw422016.E().S(rev.timeToDisplay())
//line history/view.qtpl:47 //line history/view.qtpl:69
qw422016.N().S(`</time> qw422016.N().S(`</time>
</a> </a>
<span class="history-entry__hash"><a href="/primitive-diff/`) <span class="history-entry__hash"><a href="/primitive-diff/`)
//line history/view.qtpl:49 //line history/view.qtpl:71
qw422016.E().S(rev.Hash) qw422016.E().S(rev.Hash)
//line history/view.qtpl:49 //line history/view.qtpl:71
qw422016.N().S(`/`) qw422016.N().S(`/`)
//line history/view.qtpl:49 //line history/view.qtpl:71
qw422016.E().S(hyphaName) qw422016.E().S(hyphaName)
//line history/view.qtpl:49 //line history/view.qtpl:71
qw422016.N().S(`">`) qw422016.N().S(`">`)
//line history/view.qtpl:49 //line history/view.qtpl:71
qw422016.E().S(rev.Hash) qw422016.E().S(rev.Hash)
//line history/view.qtpl:49 //line history/view.qtpl:71
qw422016.N().S(`</a></span> qw422016.N().S(`</a></span>
<span class="history-entry__msg">`) <span class="history-entry__msg">`)
//line history/view.qtpl:50 //line history/view.qtpl:72
qw422016.E().S(rev.Message) qw422016.E().S(rev.Message)
//line history/view.qtpl:50 //line history/view.qtpl:72
qw422016.N().S(`</span> qw422016.N().S(`</span>
`) `)
//line history/view.qtpl:51 //line history/view.qtpl:73
if rev.Username != "anon" { if rev.Username != "anon" {
//line history/view.qtpl:51 //line history/view.qtpl:73
qw422016.N().S(` qw422016.N().S(`
<span class="history-entry__author">by <a href="/hypha/`) <span class="history-entry__author">by <a href="/hypha/`)
//line history/view.qtpl:52 //line history/view.qtpl:74
qw422016.E().S(cfg.UserHypha) qw422016.E().S(cfg.UserHypha)
//line history/view.qtpl:52 //line history/view.qtpl:74
qw422016.N().S(`/`) qw422016.N().S(`/`)
//line history/view.qtpl:52 //line history/view.qtpl:74
qw422016.E().S(rev.Username) qw422016.E().S(rev.Username)
//line history/view.qtpl:52 //line history/view.qtpl:74
qw422016.N().S(`" rel="author">`) qw422016.N().S(`" rel="author">`)
//line history/view.qtpl:52 //line history/view.qtpl:74
qw422016.E().S(rev.Username) qw422016.E().S(rev.Username)
//line history/view.qtpl:52 //line history/view.qtpl:74
qw422016.N().S(`</a></span> qw422016.N().S(`</a></span>
`) `)
//line history/view.qtpl:53 //line history/view.qtpl:75
} }
//line history/view.qtpl:53 //line history/view.qtpl:75
qw422016.N().S(` qw422016.N().S(`
</li> </li>
`) `)
//line history/view.qtpl:55 //line history/view.qtpl:77
} }
//line history/view.qtpl:55 //line history/view.qtpl:77
func (rev *Revision) writeasHistoryEntry(qq422016 qtio422016.Writer, hyphaName string) { func (rev *Revision) writeasHistoryEntry(qq422016 qtio422016.Writer, hyphaName string) {
//line history/view.qtpl:55 //line history/view.qtpl:77
qw422016 := qt422016.AcquireWriter(qq422016) qw422016 := qt422016.AcquireWriter(qq422016)
//line history/view.qtpl:55 //line history/view.qtpl:77
rev.streamasHistoryEntry(qw422016, hyphaName) rev.streamasHistoryEntry(qw422016, hyphaName)
//line history/view.qtpl:55 //line history/view.qtpl:77
qt422016.ReleaseWriter(qw422016) qt422016.ReleaseWriter(qw422016)
//line history/view.qtpl:55 //line history/view.qtpl:77
} }
//line history/view.qtpl:55 //line history/view.qtpl:77
func (rev *Revision) asHistoryEntry(hyphaName string) string { func (rev *Revision) asHistoryEntry(hyphaName string) string {
//line history/view.qtpl:55 //line history/view.qtpl:77
qb422016 := qt422016.AcquireByteBuffer() qb422016 := qt422016.AcquireByteBuffer()
//line history/view.qtpl:55 //line history/view.qtpl:77
rev.writeasHistoryEntry(qb422016, hyphaName) rev.writeasHistoryEntry(qb422016, hyphaName)
//line history/view.qtpl:55 //line history/view.qtpl:77
qs422016 := string(qb422016.B) qs422016 := string(qb422016.B)
//line history/view.qtpl:55 //line history/view.qtpl:77
qt422016.ReleaseByteBuffer(qb422016) qt422016.ReleaseByteBuffer(qb422016)
//line history/view.qtpl:55 //line history/view.qtpl:77
return qs422016 return qs422016
//line history/view.qtpl:55 //line history/view.qtpl:77
} }