Initial commit
This commit is contained in:
165
templates/dashboard.templ
Normal file
165
templates/dashboard.templ
Normal file
@@ -0,0 +1,165 @@
|
||||
package templates
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"time"
|
||||
|
||||
"veola/internal/db"
|
||||
"veola/internal/models"
|
||||
)
|
||||
|
||||
type DashboardData struct {
|
||||
Page
|
||||
Stats *db.DashboardStats
|
||||
RecentResults []ResultRow
|
||||
RecentAlerts []AlertRow
|
||||
}
|
||||
|
||||
type ResultRow struct {
|
||||
ItemID int64
|
||||
ItemName string
|
||||
Title string
|
||||
Price *float64
|
||||
Currency string
|
||||
Source string
|
||||
URL string
|
||||
FoundAt time.Time
|
||||
Alerted bool
|
||||
}
|
||||
|
||||
type AlertRow struct {
|
||||
ItemName string
|
||||
Price *float64
|
||||
Currency string
|
||||
FoundAt time.Time
|
||||
}
|
||||
|
||||
templ dashboardBody(d DashboardData) {
|
||||
<div hx-get="/dashboard/refresh" hx-trigger="every 60s" hx-swap="outerHTML">
|
||||
<h1 class="text-3xl font-semibold mb-6">Dashboard</h1>
|
||||
<div class="grid grid-cols-2 md:grid-cols-4 gap-4 mb-6">
|
||||
@statCard("Total Items", fmt.Sprintf("%d", d.Stats.TotalItems), "")
|
||||
@statCard("Active", fmt.Sprintf("%d", d.Stats.ActiveItems), "")
|
||||
@statCard("Results Today", fmt.Sprintf("%d", d.Stats.ResultsToday), "")
|
||||
@statCard("Alerts Today", fmt.Sprintf("%d", d.Stats.AlertsToday), "")
|
||||
</div>
|
||||
<div class="grid md:grid-cols-2 gap-4 mb-6">
|
||||
<div class="v-card p-5">
|
||||
<div class="v-muted text-sm uppercase tracking-wide">Potential Spend</div>
|
||||
<div class="font-mono text-4xl mt-2">{ fmt.Sprintf("$%.2f", d.Stats.PotentialSpend) }</div>
|
||||
<div class="v-muted text-sm mt-1">across { fmt.Sprintf("%d", d.Stats.PricedItemCount) } items</div>
|
||||
if d.Stats.UnpricedCount > 0 {
|
||||
<div class="v-muted text-xs mt-1">{ fmt.Sprintf("%d items not yet priced.", d.Stats.UnpricedCount) }</div>
|
||||
}
|
||||
</div>
|
||||
<div class="v-card p-5">
|
||||
<div class="v-muted text-sm uppercase tracking-wide">Money Saved</div>
|
||||
<div class="font-mono text-4xl mt-2 v-price-deal">{ fmt.Sprintf("$%.2f", d.Stats.MoneySaved) }</div>
|
||||
<div class="v-muted text-sm mt-1">across { fmt.Sprintf("%d", d.Stats.SavedItemCount) } items</div>
|
||||
</div>
|
||||
</div>
|
||||
<div class="grid md:grid-cols-2 gap-6">
|
||||
<div class="v-card p-5">
|
||||
<h2 class="font-semibold mb-3">Recent Results</h2>
|
||||
if len(d.RecentResults) == 0 {
|
||||
<div class="v-muted text-sm">No results yet.</div>
|
||||
} else {
|
||||
<table class="v-table">
|
||||
<thead>
|
||||
<tr><th>Item</th><th>Price</th><th>Source</th><th>Found</th></tr>
|
||||
</thead>
|
||||
<tbody>
|
||||
for _, r := range d.RecentResults {
|
||||
<tr>
|
||||
<td><a href={ templ.SafeURL(fmt.Sprintf("/items/%d/results", r.ItemID)) }>{ r.ItemName }</a></td>
|
||||
<td class="font-mono">{ fmtPrice(r.Price, r.Currency) }</td>
|
||||
<td>{ r.Source }</td>
|
||||
<td class="v-muted text-sm">{ humanTime(r.FoundAt) }</td>
|
||||
</tr>
|
||||
}
|
||||
</tbody>
|
||||
</table>
|
||||
}
|
||||
</div>
|
||||
<div class="v-card p-5">
|
||||
<h2 class="font-semibold mb-3">Recent Alerts</h2>
|
||||
if len(d.RecentAlerts) == 0 {
|
||||
<div class="v-muted text-sm">No alerts sent yet.</div>
|
||||
} else {
|
||||
<ul class="space-y-2">
|
||||
for _, a := range d.RecentAlerts {
|
||||
<li class="flex justify-between items-center border-b border-white/10 pb-2">
|
||||
<span>{ a.ItemName }</span>
|
||||
<span class="font-mono v-price-target">{ fmtPrice(a.Price, a.Currency) }</span>
|
||||
</li>
|
||||
}
|
||||
</ul>
|
||||
}
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
}
|
||||
|
||||
templ statCard(label, value, sub string) {
|
||||
<div class="v-card p-4">
|
||||
<div class="v-muted text-xs uppercase tracking-wide">{ label }</div>
|
||||
<div class="font-mono text-3xl mt-1">{ value }</div>
|
||||
if sub != "" {
|
||||
<div class="v-muted text-xs mt-1">{ sub }</div>
|
||||
}
|
||||
</div>
|
||||
}
|
||||
|
||||
templ Dashboard(d DashboardData) {
|
||||
@Layout(d.Page, dashboardBody(d))
|
||||
}
|
||||
|
||||
// Helpers used by multiple templates.
|
||||
|
||||
func fmtPrice(p *float64, currency string) string {
|
||||
if p == nil {
|
||||
return "—"
|
||||
}
|
||||
sym := currencySymbol(currency)
|
||||
return fmt.Sprintf("%s%.2f", sym, *p)
|
||||
}
|
||||
|
||||
func currencySymbol(c string) string {
|
||||
switch c {
|
||||
case "USD", "":
|
||||
return "$"
|
||||
case "GBP":
|
||||
return "£"
|
||||
case "EUR":
|
||||
return "€"
|
||||
case "JPY":
|
||||
return "¥"
|
||||
default:
|
||||
return c + " "
|
||||
}
|
||||
}
|
||||
|
||||
func humanTime(t time.Time) string {
|
||||
if t.IsZero() {
|
||||
return "—"
|
||||
}
|
||||
d := time.Since(t)
|
||||
switch {
|
||||
case d < time.Minute:
|
||||
return "just now"
|
||||
case d < time.Hour:
|
||||
m := int(d.Minutes())
|
||||
return fmt.Sprintf("%d minutes ago", m)
|
||||
case d < 24*time.Hour:
|
||||
h := int(d.Hours())
|
||||
return fmt.Sprintf("%d hours ago", h)
|
||||
case d < 30*24*time.Hour:
|
||||
d2 := int(d.Hours() / 24)
|
||||
return fmt.Sprintf("%d days ago", d2)
|
||||
default:
|
||||
return t.Format("2006-01-02")
|
||||
}
|
||||
}
|
||||
|
||||
// Used by item rendering.
|
||||
var _ = models.Item{}
|
||||
Reference in New Issue
Block a user