Add a year navigator to the seasons’ calendar

This commit is contained in:
jordi fita mas 2023-09-29 18:20:16 +02:00
parent 47ec317010
commit 6939670dfc
4 changed files with 133 additions and 70 deletions

View File

@ -9,6 +9,9 @@ import (
"context" "context"
"fmt" "fmt"
"net/http" "net/http"
"net/url"
"strconv"
"strings"
"time" "time"
"github.com/jackc/pgtype" "github.com/jackc/pgtype"
@ -56,10 +59,12 @@ func (h *AdminHandler) Handler(user *auth.User, company *auth.Company, conn *dat
} }
case "range": case "range":
switch r.Method { switch r.Method {
case http.MethodGet:
serveSeasonCalendar(w, r, user, company, conn)
case http.MethodPut: case http.MethodPut:
updateSeasonCalendar(w, r, user, company, conn) updateSeasonCalendar(w, r, user, company, conn)
default: default:
httplib.MethodNotAllowed(w, r, http.MethodGet) httplib.MethodNotAllowed(w, r, http.MethodGet, http.MethodPut)
} }
default: default:
if !uuid.Valid(head) { if !uuid.Valid(head) {
@ -91,7 +96,7 @@ func serveSeasonIndex(w http.ResponseWriter, r *http.Request, user *auth.User, c
if err != nil { if err != nil {
panic(err) panic(err)
} }
calendar, err := collectSeasonCalendar(r.Context(), company, conn, 2023) calendar, err := collectSeasonCalendar(r.Context(), company, conn, getCalendarYear(r.URL.Query()))
if err != nil { if err != nil {
panic(err) panic(err)
} }
@ -103,6 +108,16 @@ func serveSeasonIndex(w http.ResponseWriter, r *http.Request, user *auth.User, c
page.MustRender(w, r, user, company) page.MustRender(w, r, user, company)
} }
func getCalendarYear(query url.Values) int {
yearStr := strings.TrimSpace(query.Get("year"))
if yearStr != "" {
if year, err := strconv.Atoi(yearStr); err == nil {
return year
}
}
return time.Now().Year()
}
func collectSeasonEntries(ctx context.Context, company *auth.Company, conn *database.Conn) ([]*seasonEntry, error) { func collectSeasonEntries(ctx context.Context, company *auth.Company, conn *database.Conn) ([]*seasonEntry, error) {
rows, err := conn.Query(ctx, ` rows, err := conn.Query(ctx, `
select slug select slug
@ -167,7 +182,9 @@ func collectSeasonCalendar(ctx context.Context, company *auth.Company, conn *dat
var month *seasonMonth var month *seasonMonth
var week seasonWeek var week seasonWeek
calendar := &seasonCalendar{} calendar := &seasonCalendar{
Year: year,
}
weekday := int(time.Monday) weekday := int(time.Monday)
for rows.Next() { for rows.Next() {
day := &seasonDay{} day := &seasonDay{}
@ -213,6 +230,7 @@ func collectSeasonCalendar(ctx context.Context, company *auth.Company, conn *dat
} }
type seasonCalendar struct { type seasonCalendar struct {
Year int
Months []*seasonMonth Months []*seasonMonth
Form *calendarForm Form *calendarForm
} }
@ -327,6 +345,11 @@ func (f *seasonForm) MustRender(w http.ResponseWriter, r *http.Request, user *au
template.MustRenderAdmin(w, r, user, company, "season/form.gohtml", f) template.MustRenderAdmin(w, r, user, company, "season/form.gohtml", f)
} }
func serveSeasonCalendar(w http.ResponseWriter, r *http.Request, user *auth.User, company *auth.Company, conn *database.Conn) {
f := newCalendarForm(r.Context(), company, conn)
f.MustRender(w, r, user, company, conn, getCalendarYear(r.URL.Query()))
}
func updateSeasonCalendar(w http.ResponseWriter, r *http.Request, user *auth.User, company *auth.Company, conn *database.Conn) { func updateSeasonCalendar(w http.ResponseWriter, r *http.Request, user *auth.User, company *auth.Company, conn *database.Conn) {
f := newCalendarForm(r.Context(), company, conn) f := newCalendarForm(r.Context(), company, conn)
if ok, err := form.Handle(f, w, r, user); err != nil { if ok, err := form.Handle(f, w, r, user); err != nil {
@ -346,7 +369,7 @@ func updateSeasonCalendar(w http.ResponseWriter, r *http.Request, user *auth.Use
f.StartDate.Val = "" f.StartDate.Val = ""
f.EndDate.Val = "" f.EndDate.Val = ""
} }
f.MustRender(w, r, user, company, conn) f.MustRender(w, r, user, company, conn, getCalendarYear(r.Form))
} }
type calendarForm struct { type calendarForm struct {
@ -427,8 +450,8 @@ func (f *calendarForm) Valid(l *locale.Locale) bool {
return v.AllOK return v.AllOK
} }
func (f *calendarForm) MustRender(w http.ResponseWriter, r *http.Request, user *auth.User, company *auth.Company, conn *database.Conn) { func (f *calendarForm) MustRender(w http.ResponseWriter, r *http.Request, user *auth.User, company *auth.Company, conn *database.Conn, year int) {
calendar, err := collectSeasonCalendar(r.Context(), company, conn, 2023) calendar, err := collectSeasonCalendar(r.Context(), company, conn, year)
if err != nil { if err != nil {
panic(err) panic(err)
} }

View File

@ -78,6 +78,12 @@ func mustRenderLayout(w io.Writer, user *auth.User, company *auth.Company, templ
"queryEscape": func(s string) string { "queryEscape": func(s string) string {
return url.QueryEscape(s) return url.QueryEscape(s)
}, },
"inc": func(i int) int {
return i + 1
},
"dec": func(i int) int {
return i - 1
},
}) })
templates = append(templates, "form.gohtml") templates = append(templates, "form.gohtml")
files := make([]string, len(templates)) files := make([]string, len(templates))

View File

@ -228,14 +228,14 @@ body > a[href="#content"]:focus {
} }
/* header */ /* header */
header { body > header {
display: flex; display: flex;
justify-content: space-between; justify-content: space-between;
align-items: center; align-items: center;
background-color: var(--camper--header--background-color); background-color: var(--camper--header--background-color);
} }
header, body > nav a { body > header, body > nav a {
padding: 0 3rem; padding: 0 3rem;
} }
@ -497,7 +497,38 @@ textarea {
background-color: #ffeeaa; background-color: #ffeeaa;
} }
.season-calendar { /* calendar */
.season-calendar button {
display: flex;
gap: 1em;
border: none;
cursor: pointer;
}
.season-calendar form button:first-child, .season-calendar > header button {
min-width: 0;
}
.season-calendar > header {
display: flex;
gap: 2rem;
justify-content: center;
align-items: center;
}
.season-calendar > header button:first-of-type {
order: -1;
}
.season-calendar > header button:first-of-type::before {
content: "←";
}
.season-calendar > header button:last-of-type::before {
content: "→";
}
.season-calendar > div {
display: grid; display: grid;
grid-template-columns: repeat(3, auto); grid-template-columns: repeat(3, auto);
grid-auto-rows: 1fr; grid-auto-rows: 1fr;
@ -507,7 +538,7 @@ textarea {
} }
@media (max-width: 48rem) { @media (max-width: 48rem) {
.season-calendar { .season-calendar > div {
display: flex; display: flex;
flex-direction: column; flex-direction: column;
} }
@ -553,15 +584,7 @@ textarea {
height: .8rem; height: .8rem;
} }
.season-calendar form button {
display: flex;
gap: 1em;
border: none;
cursor: pointer;
}
.season-calendar form button:first-child { .season-calendar form button:first-child {
min-width: 0;
position: absolute; position: absolute;
top: 0; top: 0;
right: 0; right: 0;

View File

@ -1,5 +1,13 @@
<div class="season-calendar" data-hx-target="this" data-hx-swap="outerHTML"> <article class="season-calendar" data-hx-target="this" data-hx-swap="outerHTML">
{{- /*gotype: dev.tandem.ws/tandem/camper/pkg/season.seasonCalendar*/ -}} {{- /*gotype: dev.tandem.ws/tandem/camper/pkg/season.seasonCalendar*/ -}}
<header>
<h3>{{ .Year }}</h3>
<button type="button"
data-hx-get="/admin/seasons/range?year={{ dec .Year }}"><span class="sr-only">{{ pgettext "Prev" "navigation" }}</span></button>
<button type="button"
data-hx-get="/admin/seasons/range?year={{ inc .Year }}"><span class="sr-only">{{ pgettext "Next" "navigation" }}</span></button>
</header>
<div>
{{ range .Months -}} {{ range .Months -}}
<table class="month"> <table class="month">
<caption>{{ pgettext .Name "month" }}</caption> <caption>{{ pgettext .Name "month" }}</caption>
@ -34,14 +42,16 @@
<dialog> <dialog>
<form data-hx-put="/admin/seasons/range"> <form data-hx-put="/admin/seasons/range">
{{ CSRFInput }} {{ CSRFInput }}
<input type="hidden" name="year" value="{{ $.Year }}">
{{ with .StartDate }}<input type="hidden" name="{{ .Name }}" value="{{ .Val }}">{{ end }} {{ with .StartDate }}<input type="hidden" name="{{ .Name }}" value="{{ .Val }}">{{ end }}
{{ with .EndDate }}<input type="hidden" name="{{ .Name }}" value="{{ .Val }}">{{ end }} {{ with .EndDate }}<input type="hidden" name="{{ .Name }}" value="{{ .Val }}">{{ end }}
<footer> <footer>
<button type="submit" name="season_id" value=""><span class="sr-only">{{( pgettext "Cancel" "action" )}}</span></button> <button type="submit"><span class="sr-only">{{( pgettext "Cancel" "action" )}}</span></button>
{{ range .Seasons -}} {{ range .Seasons -}}
<button type="submit" name="season_id" value="{{ .Slug }}"> <button type="submit" name="season_id" value="{{ .Slug }}">
<svg width="20px" height="20px"> <svg width="20px" height="20px">
<circle cx="50%" cy="50%" r="49%" fill="{{ .Color }}" stroke="#000" stroke-width=".5"/> <circle cx="50%" cy="50%" r="49%" fill="{{ .Color }}" stroke="#000"
stroke-width=".5"/>
</svg> </svg>
{{ .Name }} {{ .Name }}
</button> </button>
@ -50,9 +60,10 @@
</form> </form>
</dialog> </dialog>
{{ end }} {{ end }}
</div>
<script type="module"> <script type="module">
import {setupCalendar} from "/static/camper.js"; import {setupCalendar} from "/static/camper.js";
setupCalendar(document.querySelector('.season-calendar')) setupCalendar(document.querySelector('.season-calendar'))
</script> </script>
</div> </article>