155 lines
3.8 KiB
Go
155 lines
3.8 KiB
Go
/*
|
|
* SPDX-FileCopyrightText: 2023 jordi fita mas <jfita@peritasoft.com>
|
|
* SPDX-License-Identifier: AGPL-3.0-only
|
|
*/
|
|
|
|
package template
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
gotemplate "html/template"
|
|
"net/http"
|
|
"sort"
|
|
|
|
"dev.tandem.ws/tandem/camper/pkg/auth"
|
|
"dev.tandem.ws/tandem/camper/pkg/database"
|
|
httplib "dev.tandem.ws/tandem/camper/pkg/http"
|
|
"dev.tandem.ws/tandem/camper/pkg/locale"
|
|
)
|
|
|
|
type PublicPage struct {
|
|
LocalizedAlternates []*LocalizedAlternate
|
|
Menu *siteMenu
|
|
CompanyAddress *address
|
|
OpeningDates gotemplate.HTML
|
|
}
|
|
|
|
func NewPublicPage() *PublicPage {
|
|
return &PublicPage{
|
|
CompanyAddress: &address{},
|
|
}
|
|
}
|
|
|
|
func (p *PublicPage) Setup(r *http.Request, user *auth.User, company *auth.Company, conn *database.Conn) {
|
|
schema := httplib.Protocol(r)
|
|
authority := httplib.Host(r)
|
|
_, path := httplib.ShiftPath(r.RequestURI)
|
|
for _, l := range company.Locales {
|
|
p.LocalizedAlternates = append(p.LocalizedAlternates, &LocalizedAlternate{
|
|
Lang: l.Language.String(),
|
|
Endonym: l.Endonym,
|
|
HRef: fmt.Sprintf("%s://%s/%s%s", schema, authority, l.Language, path),
|
|
})
|
|
}
|
|
sort.Slice(p.LocalizedAlternates, func(i, j int) bool { return p.LocalizedAlternates[i].Lang < p.LocalizedAlternates[j].Lang })
|
|
|
|
p.Menu = &siteMenu{
|
|
CampsiteTypes: mustCollectMenuItems(r.Context(), conn, user.Locale, `
|
|
select coalesce(i18n.name, campsite_type.name) as l10n_name
|
|
, '/campsites/types/' || slug
|
|
from campsite_type
|
|
left join campsite_type_i18n as i18n on campsite_type.campsite_type_id = i18n.campsite_type_id and i18n.lang_tag = $1
|
|
where company_id = $2
|
|
and active
|
|
order by position, l10n_name
|
|
`, user.Locale.Language, company.ID),
|
|
}
|
|
|
|
if err := conn.QueryRow(r.Context(), `
|
|
select coalesce(i18n.opening_dates, location.opening_dates)
|
|
from location
|
|
left join location_i18n as i18n on location.company_id = i18n.company_id and i18n.lang_tag = $1
|
|
where location.company_id = $2
|
|
`, user.Locale.Language, company.ID).Scan(&p.OpeningDates); err != nil {
|
|
if !database.ErrorIsNotFound(err) {
|
|
panic(err)
|
|
}
|
|
}
|
|
|
|
if err := p.CompanyAddress.FillFromDatabase(r.Context(), conn, user, company); err != nil {
|
|
panic(err)
|
|
}
|
|
}
|
|
|
|
type LocalizedAlternate struct {
|
|
Lang string
|
|
HRef string
|
|
Endonym string
|
|
}
|
|
|
|
type siteMenu struct {
|
|
CampsiteTypes []*menuItem
|
|
}
|
|
|
|
type menuItem struct {
|
|
Label string
|
|
HRef string
|
|
}
|
|
|
|
func mustCollectMenuItems(ctx context.Context, conn *database.Conn, loc *locale.Locale, sql string, args ...interface{}) []*menuItem {
|
|
rows, err := conn.Query(ctx, sql, args...)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
defer rows.Close()
|
|
|
|
localePath := "/" + loc.Language.String()
|
|
var items []*menuItem
|
|
for rows.Next() {
|
|
item := &menuItem{}
|
|
err = rows.Scan(&item.Label, &item.HRef)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
item.HRef = localePath + item.HRef
|
|
items = append(items, item)
|
|
}
|
|
if rows.Err() != nil {
|
|
panic(rows.Err())
|
|
}
|
|
|
|
return items
|
|
}
|
|
|
|
type address struct {
|
|
TradeName string
|
|
Address string
|
|
PostalCode string
|
|
Province string
|
|
City string
|
|
Country string
|
|
Phone string
|
|
Email string
|
|
RTCNumber string
|
|
}
|
|
|
|
func (addr *address) FillFromDatabase(ctx context.Context, conn *database.Conn, user *auth.User, company *auth.Company) error {
|
|
row := conn.QueryRow(ctx, `
|
|
select trade_name
|
|
, address
|
|
, postal_code
|
|
, province
|
|
, city
|
|
, coalesce(country_i18n.name, country.name) as country_name
|
|
, phone::text
|
|
, email::text
|
|
, rtc_number
|
|
from company
|
|
join country using (country_code)
|
|
left join country_i18n on country.country_code = country_i18n.country_code and country_i18n.lang_tag = $2
|
|
where company_id = $1
|
|
`, company.ID, user.Locale.Language)
|
|
return row.Scan(
|
|
&addr.TradeName,
|
|
&addr.Address,
|
|
&addr.PostalCode,
|
|
&addr.Province,
|
|
&addr.City,
|
|
&addr.Country,
|
|
&addr.Phone,
|
|
&addr.Email,
|
|
&addr.RTCNumber,
|
|
)
|
|
}
|