- store.RecentDocuments(since, limit) returns dated item_links + parent item - web/dashboard.go handler aggregates VTODOs + Gitea issues + dated links across every linked item, fanout via 4-worker goroutine pool, 60s TTL cache keyed by encoded TreeFilter - Tasks card: bucketed Overdue/Today/Tomorrow/Week/NoDue, sort by bucket then due asc; ✓ button completes via existing PutTodo path + busts cache - Issues card: read-only, reuses GiteaDeps.Cache - Recent docs card: last-30d event_date links, canonical PER rendered - Filter chips on top reuse tree_filter URL params (tag/mgmt/has) - nav adds "dashboard" link; design.md §"Dashboard" documents the surface - 4 integration tests (empty render, dated-link surfacing, tag filter, cache hit)
123 lines
5.3 KiB
Cheetah
123 lines
5.3 KiB
Cheetah
{{define "dashboard-section"}}
|
|
<section id="dashboard-section" class="dashboard">
|
|
|
|
<section class="tagbar" id="dashboard-filterbar">
|
|
<form id="dashboard-filter" class="search"
|
|
hx-get="/dashboard"
|
|
hx-target="#dashboard-section"
|
|
hx-swap="outerHTML"
|
|
hx-trigger="change from:select"
|
|
hx-push-url="true">
|
|
<label>tag
|
|
<select name="tag" multiple size="3">
|
|
{{$sel := .Filter.Tags}}
|
|
{{range $.Filter.Tags}}<option value="{{.}}" selected>{{.}}</option>{{end}}
|
|
</select>
|
|
</label>
|
|
<label>mgmt
|
|
<select name="mgmt" multiple size="4">
|
|
{{$selM := .Filter.Management}}
|
|
<option value="mai" {{if contains $selM "mai"}}selected{{end}}>mai</option>
|
|
<option value="self" {{if contains $selM "self"}}selected{{end}}>self</option>
|
|
<option value="external" {{if contains $selM "external"}}selected{{end}}>external</option>
|
|
</select>
|
|
</label>
|
|
<label>has
|
|
<select name="has" multiple size="2">
|
|
{{$selH := .Filter.HasLinks}}
|
|
<option value="caldav-list" {{if contains $selH "caldav-list"}}selected{{end}}>caldav</option>
|
|
<option value="gitea-repo" {{if contains $selH "gitea-repo"}}selected{{end}}>gitea</option>
|
|
</select>
|
|
</label>
|
|
{{if .Filter.Active}}<a class="clear" href="/dashboard">clear filters</a>{{end}}
|
|
</form>
|
|
<p class="counts muted">
|
|
{{if .P.Cached}}<small title="Served from 60s in-memory cache">cached</small>
|
|
{{else}}<small>fresh — built {{.P.BuiltAt.Format "15:04:05"}}</small>{{end}}
|
|
</p>
|
|
</section>
|
|
|
|
<section class="dash-grid">
|
|
|
|
<article class="card card-tasks">
|
|
<header>
|
|
<h2>Open tasks <small class="muted">({{.P.TaskTotal}})</small></h2>
|
|
{{if or .P.TaskGroups.Overdue .P.TaskGroups.Today .P.TaskGroups.Tomorrow .P.TaskGroups.Week .P.TaskGroups.NoDue}}
|
|
<p class="task-groups muted">
|
|
{{if .P.TaskGroups.Overdue}}<span class="overdue">Overdue ({{.P.TaskGroups.Overdue}})</span>{{end}}
|
|
{{if .P.TaskGroups.Today}}<span>Today ({{.P.TaskGroups.Today}})</span>{{end}}
|
|
{{if .P.TaskGroups.Tomorrow}}<span>Tomorrow ({{.P.TaskGroups.Tomorrow}})</span>{{end}}
|
|
{{if .P.TaskGroups.Week}}<span>This week ({{.P.TaskGroups.Week}})</span>{{end}}
|
|
{{if .P.TaskGroups.NoDue}}<span>No due ({{.P.TaskGroups.NoDue}})</span>{{end}}
|
|
</p>
|
|
{{end}}
|
|
</header>
|
|
{{if .P.Tasks}}
|
|
<ul class="task-list">
|
|
{{range .P.Tasks}}
|
|
<li class="task-row bucket-{{.Bucket}}" id="task-{{.Todo.UID}}">
|
|
<form class="check"
|
|
hx-post="/dashboard/task/done"
|
|
hx-target="#dashboard-section"
|
|
hx-swap="outerHTML">
|
|
<input type="hidden" name="calendar_url" value="{{.CalendarURL}}">
|
|
<input type="hidden" name="uid" value="{{.Todo.UID}}">
|
|
<button type="submit" title="mark complete">✓</button>
|
|
</form>
|
|
<a class="proj" href="/i/{{.Item.PrimaryPath}}">{{.Item.PrimaryPath}}</a>
|
|
<span class="summary">{{.Todo.Summary}}</span>
|
|
{{if .DueRel}}<span class="due {{if eq .Bucket "overdue"}}bad{{end}}" title="{{if .Todo.Due}}{{.Todo.Due.Format "2006-01-02"}}{{end}}">{{.DueRel}}</span>{{end}}
|
|
</li>
|
|
{{end}}
|
|
</ul>
|
|
{{else}}
|
|
<p class="empty muted">Nothing open. Nice.</p>
|
|
{{end}}
|
|
</article>
|
|
|
|
<article class="card card-issues">
|
|
<header>
|
|
<h2>Open issues <small class="muted">({{.P.IssueTotal}})</small></h2>
|
|
</header>
|
|
{{if .P.Issues}}
|
|
<ul class="issue-list">
|
|
{{range .P.Issues}}
|
|
<li class="issue-row">
|
|
<a class="proj" href="/i/{{.Item.PrimaryPath}}">{{.Item.PrimaryPath}}</a>
|
|
<a class="iss" href="{{.Issue.HTMLURL}}" target="_blank" rel="noopener">#{{.Issue.Number}} {{.Issue.Title}}</a>
|
|
{{range .Issue.Labels}}<span class="label">{{.}}</span>{{end}}
|
|
{{if .Issue.Assignees}}<small class="muted">@ {{range $i, $a := .Issue.Assignees}}{{if $i}}, {{end}}{{$a}}{{end}}</small>{{end}}
|
|
<span class="upd muted" title="{{.Issue.UpdatedAt.Format "2006-01-02 15:04"}}">{{.UpdRel}}</span>
|
|
</li>
|
|
{{end}}
|
|
</ul>
|
|
{{else}}
|
|
<p class="empty muted">No open issues across linked repos.</p>
|
|
{{end}}
|
|
</article>
|
|
|
|
<article class="card card-docs">
|
|
<header>
|
|
<h2>Recent documents <small class="muted">({{.P.RecentDocsTotal}}, last 30d)</small></h2>
|
|
</header>
|
|
{{if .P.RecentDocs}}
|
|
<ul class="doc-list">
|
|
{{range .P.RecentDocs}}
|
|
<li class="doc-row">
|
|
<span class="per">{{.PER}}</span>
|
|
<span class="ref-type ref-type-{{.Link.RefType}}">{{.Link.RefType}}</span>
|
|
{{if .Link.Note}}<span class="note">{{deref .Link.Note}}</span>{{end}}
|
|
<a class="ref-id" href="{{.Link.RefID}}" target="_blank" rel="noopener">{{.Link.RefID}}</a>
|
|
<a class="proj muted" href="/i/{{.ItemPath}}">{{.ItemPath}}</a>
|
|
</li>
|
|
{{end}}
|
|
</ul>
|
|
{{else}}
|
|
<p class="empty muted">Nothing dated in the last 30 days.</p>
|
|
{{end}}
|
|
</article>
|
|
|
|
</section>
|
|
</section>
|
|
{{end}}
|