mirror of
https://github.com/Theodor-Springmann-Stiftung/musenalm.git
synced 2025-10-29 17:25:32 +00:00
223 lines
5.0 KiB
Go
223 lines
5.0 KiB
Go
package pages
|
|
|
|
import (
|
|
"sort"
|
|
|
|
"github.com/Theodor-Springmann-Stiftung/musenalm/app"
|
|
"github.com/Theodor-Springmann-Stiftung/musenalm/dbmodels"
|
|
"github.com/Theodor-Springmann-Stiftung/musenalm/helpers/datatypes"
|
|
"github.com/Theodor-Springmann-Stiftung/musenalm/pagemodels"
|
|
"github.com/Theodor-Springmann-Stiftung/musenalm/templating"
|
|
"github.com/pocketbase/pocketbase/core"
|
|
"github.com/pocketbase/pocketbase/tools/router"
|
|
)
|
|
|
|
const (
|
|
URL_ALMANACH = "/almanach/{id}/"
|
|
TEMPLATE_ALMANACH = "/almanach/"
|
|
)
|
|
|
|
func init() {
|
|
rp := &AlmanachPage{
|
|
StaticPage: pagemodels.StaticPage{
|
|
Name: pagemodels.P_REIHEN_NAME,
|
|
URL: URL_ALMANACH,
|
|
Template: TEMPLATE_ALMANACH,
|
|
Layout: templating.DEFAULT_LAYOUT_NAME,
|
|
},
|
|
}
|
|
app.Register(rp)
|
|
}
|
|
|
|
type AlmanachPage struct {
|
|
pagemodels.StaticPage
|
|
}
|
|
|
|
func (p *AlmanachPage) Setup(router *router.Router[*core.RequestEvent], app core.App, engine *templating.Engine) error {
|
|
router.GET(p.URL, func(e *core.RequestEvent) error {
|
|
id := e.Request.PathValue("id")
|
|
data := make(map[string]interface{})
|
|
filters := NewBeitraegeFilterParameters(e)
|
|
result, err := NewAlmanachResult(app, id, filters)
|
|
if err != nil {
|
|
engine.Response404(e, err, nil)
|
|
}
|
|
data["result"] = result
|
|
data["filters"] = filters
|
|
|
|
abbrs, err := pagemodels.GetAbks(app)
|
|
if err == nil {
|
|
data["abbrs"] = abbrs
|
|
}
|
|
|
|
return engine.Response200(e, p.Template, data)
|
|
})
|
|
|
|
return nil
|
|
}
|
|
|
|
type AlmanachResult struct {
|
|
Entry *dbmodels.Entry
|
|
Places []*dbmodels.Place
|
|
Series []*dbmodels.Series
|
|
Contents []*dbmodels.Content
|
|
Agents map[string]*dbmodels.Agent // <- Key is agent id
|
|
EntriesSeries map[string]*dbmodels.REntriesSeries // <- Key is series id
|
|
EntriesAgents []*dbmodels.REntriesAgents
|
|
ContentsAgents map[string][]*dbmodels.RContentsAgents // <- Key is content id
|
|
|
|
Types []string
|
|
HasScans bool
|
|
}
|
|
|
|
func NewAlmanachResult(app core.App, id string, params BeitraegeFilterParameters) (*AlmanachResult, error) {
|
|
// INFO: what about sql.ErrNoRows?
|
|
// We don't get sql.ErrNoRows here, since dbx converts every empty slice or
|
|
// empty id to a WHERE 0=1 query, which will not error.
|
|
entry, err := dbmodels.Entries_MusenalmID(app, id)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
places, err := dbmodels.Places_IDs(app, datatypes.ToAny(entry.Places()))
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
srelations, err := dbmodels.REntriesSeries_Entry(app, entry.Id)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
sids := []any{}
|
|
srelationsMap := map[string]*dbmodels.REntriesSeries{}
|
|
for _, r := range srelations {
|
|
sids = append(sids, r.Series())
|
|
srelationsMap[r.Series()] = r
|
|
}
|
|
|
|
series, err := dbmodels.Series_IDs(app, sids)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
contents, err := dbmodels.Contents_Entry(app, entry.Id)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
types := Types_Contents(contents)
|
|
|
|
hs := HasScans(contents)
|
|
|
|
if params.OnlyScans {
|
|
cscans := []*dbmodels.Content{}
|
|
for _, c := range contents {
|
|
if len(c.Scans()) > 0 {
|
|
cscans = append(cscans, c)
|
|
}
|
|
}
|
|
contents = cscans
|
|
}
|
|
|
|
if params.Type != "" {
|
|
cfiltered := []*dbmodels.Content{}
|
|
outer:
|
|
for _, c := range contents {
|
|
for _, t := range c.MusenalmType() {
|
|
if t == params.Type {
|
|
cfiltered = append(cfiltered, c)
|
|
continue outer
|
|
}
|
|
}
|
|
}
|
|
contents = cfiltered
|
|
}
|
|
|
|
dbmodels.Sort_Contents_Numbering(contents)
|
|
|
|
contentsagents, err := dbmodels.RContentsAgents_Contents(app, dbmodels.Ids(contents))
|
|
caids := []any{}
|
|
caMap := map[string][]*dbmodels.RContentsAgents{}
|
|
for _, r := range contentsagents {
|
|
caids = append(caids, r.Agent())
|
|
caMap[r.Content()] = append(caMap[r.Content()], r)
|
|
}
|
|
|
|
entriesagents, err := dbmodels.REntriesAgents_Entry(app, entry.Id)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
for _, r := range entriesagents {
|
|
caids = append(caids, r.Agent())
|
|
}
|
|
|
|
agents, err := dbmodels.Agents_IDs(app, caids)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
agentsMap := map[string]*dbmodels.Agent{}
|
|
for _, a := range agents {
|
|
agentsMap[a.Id] = a
|
|
}
|
|
|
|
ret := &AlmanachResult{
|
|
Entry: entry,
|
|
Places: places,
|
|
Series: series,
|
|
Contents: contents,
|
|
Agents: agentsMap,
|
|
EntriesSeries: srelationsMap,
|
|
EntriesAgents: entriesagents,
|
|
ContentsAgents: caMap,
|
|
Types: types,
|
|
HasScans: hs,
|
|
}
|
|
|
|
ret.Collections()
|
|
return ret, nil
|
|
|
|
}
|
|
|
|
func (r *AlmanachResult) Collections() {
|
|
ids := []int{}
|
|
collections := []*dbmodels.Content{}
|
|
for _, s := range r.Contents {
|
|
ids = append(ids, s.MusenalmID())
|
|
for _, t := range s.MusenalmType() {
|
|
if t == "Sammlung" {
|
|
collections = append(collections, s)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
func Types_Contents(contents []*dbmodels.Content) []string {
|
|
types := map[string]bool{}
|
|
for _, c := range contents {
|
|
for _, t := range c.MusenalmType() {
|
|
types[t] = true
|
|
}
|
|
}
|
|
|
|
ret := make([]string, 0, len(types))
|
|
for t, _ := range types {
|
|
ret = append(ret, t)
|
|
}
|
|
|
|
sort.Strings(ret)
|
|
|
|
return ret
|
|
}
|
|
|
|
func HasScans(contents []*dbmodels.Content) bool {
|
|
for _, c := range contents {
|
|
if len(c.Scans()) > 0 {
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|