Compare commits
66 Commits
a5bdf7b4f5
...
main-backu
| Author | SHA1 | Date | |
|---|---|---|---|
| cb43b4a7b5 | |||
| cbcb17a35c | |||
| 5922d5d274 | |||
| 96322a4776 | |||
| 1f94bbca15 | |||
| 5ef15f0b5c | |||
| 020bfca731 | |||
| ac864ba054 | |||
| e93ba5d29b | |||
| feb137471d | |||
| 0622d190d1 | |||
| 2867678223 | |||
| 096ac9f789 | |||
| 3006ccd5a0 | |||
| 8a8bcc304a | |||
| 54c6f205e0 | |||
| 48fddf7b15 | |||
| 2733c2e7f4 | |||
| 9502123b89 | |||
| ca2d724bd8 | |||
| 38229ac5e9 | |||
| a11c838d2a | |||
| f9fe914c32 | |||
| 21e09f7155 | |||
| 0b37f73634 | |||
| c764f892a1 | |||
| 78f367530a | |||
| b539329420 | |||
| 3e93e8ce3b | |||
| 2fab4bf70b | |||
| 1a6be67af1 | |||
| fea45fc4f8 | |||
| 761bd6be80 | |||
| 8e6bd12da5 | |||
| 548a2d6f53 | |||
| 01edb8d575 | |||
| c498b19afb | |||
| 74a8e7b393 | |||
| 9c4b6ec425 | |||
| dc3f0b53d7 | |||
| b215592292 | |||
| 9c7ecc97df | |||
| 0fd4fbe61f | |||
| 6e489ceac3 | |||
| 21d51732e5 | |||
| f1c94ed438 | |||
| 493c2a94f0 | |||
| 3a3a36e2ea | |||
| 535c82bd81 | |||
| 64aa08c699 | |||
| 6f3edc8977 | |||
| 9ac87b82e9 | |||
| 74e4799ea9 | |||
| 5247bd9816 | |||
| 50c06b3a8a | |||
| 5ab62f2b3b | |||
| 6120f04c95 | |||
| 179de67386 | |||
| 3da1b63a50 | |||
| 6b79e08684 | |||
| 7d5e77df76 | |||
| 23b47a7e85 | |||
| f4c75ea941 | |||
| 58522f2ae0 | |||
| 2a0aa7a6c8 | |||
| bcd86c9c68 |
31
.env.example
Normal file
@ -0,0 +1,31 @@
|
|||||||
|
# Copy this file to .env.local for local development
|
||||||
|
# Then run: npm run dev:local
|
||||||
|
|
||||||
|
# Public base URL for your local dev server
|
||||||
|
PUBLIC_BASE_URL=http://localhost:4321
|
||||||
|
|
||||||
|
# OAuth (Gitea) settings for local development
|
||||||
|
# Create an OAuth2 Application in your Gitea with Redirect URI:
|
||||||
|
# http://localhost:4321/api/auth/callback
|
||||||
|
# Then paste the resulting Client ID/Secret below
|
||||||
|
OAUTH_PROVIDER=gitea
|
||||||
|
OAUTH_CLIENT_ID=
|
||||||
|
OAUTH_CLIENT_SECRET=
|
||||||
|
OAUTH_AUTHORIZE_URL=https://git.bookageek.ch/login/oauth/authorize
|
||||||
|
OAUTH_TOKEN_URL=https://git.bookageek.ch/login/oauth/access_token
|
||||||
|
OAUTH_USERINFO_URL=https://git.bookageek.ch/api/v1/user
|
||||||
|
|
||||||
|
# Optional access control
|
||||||
|
# OAUTH_ALLOWED_USERS=user1,user2
|
||||||
|
# OAUTH_ALLOWED_ORG=your-org
|
||||||
|
|
||||||
|
# Gitea API for committing content changes (service account PAT)
|
||||||
|
GITEA_BASE=https://git.bookageek.ch
|
||||||
|
GITEA_OWNER=
|
||||||
|
GITEA_REPO=
|
||||||
|
GITEA_TOKEN=
|
||||||
|
GIT_BRANCH=main
|
||||||
|
|
||||||
|
# Session and CSRF secrets (use random long strings in .env.local)
|
||||||
|
SESSION_SECRET=
|
||||||
|
CSRF_SECRET=
|
||||||
@ -1,26 +1,16 @@
|
|||||||
pipeline:
|
steps:
|
||||||
build:
|
|
||||||
image: node:20-alpine
|
|
||||||
commands:
|
|
||||||
- npm ci
|
|
||||||
- npm run build
|
|
||||||
when:
|
|
||||||
branch: main
|
|
||||||
event: [push, pull_request]
|
|
||||||
deploy:
|
deploy:
|
||||||
depends_on: [build]
|
image: node:20
|
||||||
image: flyio/flyctl:latest
|
environment:
|
||||||
secrets: [fly_api_token]
|
FLY_API_TOKEN:
|
||||||
|
from_secret: FLY_API_TOKEN
|
||||||
commands:
|
commands:
|
||||||
- flyctl deploy --remote-only
|
- curl -L https://fly.io/install.sh | sh
|
||||||
when:
|
- export PATH="$HOME/.fly/bin:$PATH"
|
||||||
branch: main
|
- flyctl deploy --config fly.toml --app gallus-pub
|
||||||
event: push
|
|
||||||
|
|
||||||
branches:
|
when:
|
||||||
include: [main, dev]
|
branch:
|
||||||
|
- main
|
||||||
cache:
|
event:
|
||||||
mount:
|
- push
|
||||||
- node_modules
|
|
||||||
- .npm
|
|
||||||
|
|||||||
4
fly.toml
@ -1,5 +1,5 @@
|
|||||||
app = "gallus-pub"
|
app = "gallus-pub"
|
||||||
primary_region = "fra" # Frankfurt region, change if needed
|
primary_region = "fra"
|
||||||
kill_signal = "SIGINT"
|
kill_signal = "SIGINT"
|
||||||
kill_timeout = 5
|
kill_timeout = 5
|
||||||
|
|
||||||
@ -26,7 +26,7 @@ kill_timeout = 5
|
|||||||
[[http_service.checks]]
|
[[http_service.checks]]
|
||||||
interval = "30s"
|
interval = "30s"
|
||||||
timeout = "5s"
|
timeout = "5s"
|
||||||
grace_period = "10s"
|
grace_period = "30s"
|
||||||
method = "GET"
|
method = "GET"
|
||||||
path = "/"
|
path = "/"
|
||||||
protocol = "http"
|
protocol = "http"
|
||||||
|
|||||||
|
Before Width: | Height: | Size: 506 KiB After Width: | Height: | Size: 706 KiB |
BIN
public/images/Event1.png
Normal file
|
After Width: | Height: | Size: 47 KiB |
BIN
public/images/Event2.png
Normal file
|
After Width: | Height: | Size: 50 KiB |
BIN
public/images/Event3.png
Normal file
|
After Width: | Height: | Size: 55 KiB |
BIN
public/images/Event4.png
Normal file
|
After Width: | Height: | Size: 160 KiB |
BIN
public/images/Gallery1.png
Normal file
|
After Width: | Height: | Size: 800 KiB |
BIN
public/images/Gallery2.png
Normal file
|
After Width: | Height: | Size: 747 KiB |
BIN
public/images/Gallery3.png
Normal file
|
After Width: | Height: | Size: 398 KiB |
BIN
public/images/Gallery4.png
Normal file
|
After Width: | Height: | Size: 604 KiB |
BIN
public/images/Gallery5.png
Normal file
|
After Width: | Height: | Size: 580 KiB |
BIN
public/images/Gallery6.png
Normal file
|
After Width: | Height: | Size: 117 KiB |
BIN
public/images/Gallery7.png
Normal file
|
After Width: | Height: | Size: 128 KiB |
BIN
public/images/Gallery8.png
Normal file
|
After Width: | Height: | Size: 116 KiB |
BIN
public/images/Gallery9.png
Normal file
|
After Width: | Height: | Size: 567 KiB |
|
Before Width: | Height: | Size: 110 KiB After Width: | Height: | Size: 122 KiB |
BIN
public/images/MonthlyHit.png
Normal file
|
After Width: | Height: | Size: 250 KiB |
BIN
public/images/Whiskey1.png
Normal file
|
After Width: | Height: | Size: 184 KiB |
BIN
public/images/Whiskey2.png
Normal file
|
After Width: | Height: | Size: 179 KiB |
BIN
public/images/Whiskey3.png
Normal file
|
After Width: | Height: | Size: 196 KiB |
BIN
public/pdf/Getraenke_Gallus_2025.pdf
Normal file
@ -1,6 +1,6 @@
|
|||||||
---
|
---
|
||||||
import Layout from "../components/Layout.astro";
|
import Layout from "../components/Layout.astro";
|
||||||
import "../../styles/components/ContactForm.css";
|
import "../styles/components/ContactForm.css";
|
||||||
---
|
---
|
||||||
|
|
||||||
<Layout>
|
<Layout>
|
||||||
|
|||||||
@ -1,35 +1,44 @@
|
|||||||
---
|
---
|
||||||
import "../../styles/components/Drinks.css"
|
import "../styles/components/Drinks.css"
|
||||||
|
|
||||||
const { id } = Astro.props;
|
const { id } = Astro.props;
|
||||||
---
|
---
|
||||||
<section id={id} class="Drinks">
|
<section id={id} class="Drinks">
|
||||||
<h2 class="title">Drinks</h2>
|
<h2 class="title">Drinks</h2>
|
||||||
|
|
||||||
<a href="/pdf/Menu.pdf" class="card-link" target="_blank" rel="noopener noreferrer">Getränkekarte</a>
|
<p class="note">
|
||||||
|
Ob ein frisch gezapftes Pint, ein edler Tropfen Whiskey oder ein gemütliches Glas Wein – hier kannst du in entspannter
|
||||||
|
Atmosphäre das Leben genießen. Natürlich dürfen auch Cocktails nicht fehlen. Vieles kreieren wir auch selber - Sláinte!
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<a href="/pdf/Getraenke_Gallus_2025.pdf" class="card-link" target="_blank" rel="noopener noreferrer">Getränkekarte</a>
|
||||||
|
|
||||||
<h3 class="monats-hit">Monats Hit</h3>
|
<h3 class="monats-hit">Monats Hit</h3>
|
||||||
|
|
||||||
<div class="mate-vodka">
|
<div class="mate-vodka">
|
||||||
<div class="circle" title="Mate Vodka">
|
<div class="circle" title="Mate Vodka">
|
||||||
<span class="circle-label">Mate Vodka</span>
|
<img src="/images/MonthlyHit.png" alt="Monats Hit" class="circle-image" />
|
||||||
|
<span class="circle-label"></span>
|
||||||
</div>
|
</div>
|
||||||
<div>Mate Vodka</div>
|
<div>Mate Vodka</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="circle-row">
|
|
||||||
<div class="circle" title="Bier">
|
|
||||||
<span class="circle-label">Bier</span>
|
|
||||||
</div>
|
|
||||||
<div class="circle" title="Wein">
|
|
||||||
<span class="circle-label">Wein</span>
|
|
||||||
</div>
|
|
||||||
<div class="circle" title="Cocktails">
|
|
||||||
<span class="circle-label">Cocktails</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<p class="note">
|
<p class="note">
|
||||||
Wir bieten eine Auswahl an erlesenen Getränken für jeden Geschmack. Besuche uns und entdecke unsere saisonalen Spezialitäten und Klassiker.
|
Für Whisky-Liebhaber haben wir erlesene Sorten aus Schottland und Irland im Angebot.
|
||||||
</p>
|
</p>
|
||||||
|
|
||||||
|
<div class="circle-row">
|
||||||
|
<div class="circle whiskey-circle" title="Whiskey 1">
|
||||||
|
<img src="/images/Whiskey1.png" alt="Whiskey 1" class="circle-image" />
|
||||||
|
<span class="circle-label"></span>
|
||||||
|
</div>
|
||||||
|
<div class="circle whiskey-circle" title="Whiskey 2">
|
||||||
|
<img src="/images/Whiskey2.png" alt="Whiskey 2" class="circle-image" />
|
||||||
|
<span class="circle-label"></span>
|
||||||
|
</div>
|
||||||
|
<div class="circle whiskey-circle" title="Whiskey 3">
|
||||||
|
<img src="/images/Whiskey3.png" alt="Whiskey 3" class="circle-image" />
|
||||||
|
<span class="circle-label"></span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
</section>
|
</section>
|
||||||
|
|||||||
@ -13,7 +13,7 @@ const { events = [], id }: { events?: Event[]; id?: string } = Astro.props as {
|
|||||||
events?: Event[];
|
events?: Event[];
|
||||||
id?: string;
|
id?: string;
|
||||||
};
|
};
|
||||||
import "../../styles/components/EventsGrid.css";
|
import "../styles/components/EventsGrid.css";
|
||||||
---
|
---
|
||||||
|
|
||||||
<h2 class="section-title">Events</h2>
|
<h2 class="section-title">Events</h2>
|
||||||
|
|||||||
@ -1,28 +1,81 @@
|
|||||||
---
|
---
|
||||||
// src/components/Header.astro
|
// src/components/Header.astro
|
||||||
const { url } = Astro;
|
const { url } = Astro;
|
||||||
import "../../styles/components/Header.css";
|
import "../styles/components/Header.css";
|
||||||
---
|
---
|
||||||
|
|
||||||
<header class="header">
|
<header class="header">
|
||||||
|
<!-- Desktop Layout -->
|
||||||
|
<div class="desktop-layout">
|
||||||
<div class="logo-container">
|
<div class="logo-container">
|
||||||
<a href="/">
|
<a href="/">
|
||||||
<img src="/images/Logo.png" alt="Logo" class="logo" />
|
<img src="/images/Logo.png" alt="Logo" class="logo" />
|
||||||
</a>
|
</a>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<!-- Hauptnavigation: immer Home, About, Contact -->
|
|
||||||
<nav class="nav-main">
|
<nav class="nav-main">
|
||||||
<div>
|
<div class="desktop-menu">
|
||||||
<a href="/#hero">Home</a>
|
<a href="/#hero">Home</a>
|
||||||
<a href="/#events">Events</a>
|
<a href="/#events">Events</a>
|
||||||
<a href="/#gallery">Galerie</a>
|
<a href="/#gallery">Galerie</a>
|
||||||
<a href="/#drinks">Drinks</a>
|
<a href="/#drinks">Drinks</a>
|
||||||
<a href="/#openings">Openings</a>
|
<a href="/#footer">Contact</a>
|
||||||
<!--<a href="/#about">About</a>
|
<!--<a href="/#about">About</a>
|
||||||
<a href="/#contact">Contact</a>-->
|
<a href="/#contact">Contact</a>-->
|
||||||
</div>
|
</div>
|
||||||
</nav>
|
</nav>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Mobile Layout -->
|
||||||
|
<div class="mobile-layout">
|
||||||
|
<!-- Centered Logo -->
|
||||||
|
<div class="mobile-logo-container">
|
||||||
|
<a href="/">
|
||||||
|
<img src="/images/Logo.png" alt="Logo" class="logo" />
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Burger Menu Below Logo -->
|
||||||
|
<div class="burger-menu">
|
||||||
|
<div class="burger-icon">
|
||||||
|
<span></span>
|
||||||
|
<span></span>
|
||||||
|
<span></span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Mobile Navigation Menu (Dropdown) -->
|
||||||
|
<div class="mobile-menu">
|
||||||
|
<a href="/#hero">Home</a>
|
||||||
|
<a href="/#events">Events</a>
|
||||||
|
<a href="/#gallery">Galerie</a>
|
||||||
|
<a href="/#drinks">Drinks</a>
|
||||||
|
<a href="/#footer">Contact</a>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
</header>
|
</header>
|
||||||
|
|
||||||
<div class="header-spacer"></div>
|
<div class="header-spacer"></div>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
// Toggle mobile menu when burger icon is clicked
|
||||||
|
document.addEventListener('DOMContentLoaded', () => {
|
||||||
|
const burgerIcon = document.querySelector('.burger-icon');
|
||||||
|
const mobileMenu = document.querySelector('.mobile-menu');
|
||||||
|
const mobileMenuLinks = document.querySelectorAll('.mobile-menu a');
|
||||||
|
|
||||||
|
// Toggle menu when burger icon is clicked
|
||||||
|
burgerIcon.addEventListener('click', () => {
|
||||||
|
burgerIcon.classList.toggle('active');
|
||||||
|
mobileMenu.classList.toggle('active');
|
||||||
|
});
|
||||||
|
|
||||||
|
// Close menu when a navigation link is clicked
|
||||||
|
mobileMenuLinks.forEach(link => {
|
||||||
|
link.addEventListener('click', () => {
|
||||||
|
burgerIcon.classList.remove('active');
|
||||||
|
mobileMenu.classList.remove('active');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|||||||
@ -1,6 +1,6 @@
|
|||||||
---
|
---
|
||||||
// src/components/Hero.astro
|
// src/components/Hero.astro
|
||||||
import "../../styles/components/Hero.css"
|
import "../styles/components/Hero.css"
|
||||||
|
|
||||||
const { id } = Astro.props;
|
const { id } = Astro.props;
|
||||||
---
|
---
|
||||||
|
|||||||
@ -1,7 +1,6 @@
|
|||||||
---
|
---
|
||||||
// src/components/HoverCard.astro
|
import "../styles/components/HoverCard.css";
|
||||||
import "../../styles/components/HoverCard.css";
|
const {title, description, image = "", date} = Astro.props;
|
||||||
const { title, description, image = "", date } = Astro.props;
|
|
||||||
---
|
---
|
||||||
|
|
||||||
<article class="hover-card">
|
<article class="hover-card">
|
||||||
@ -10,6 +9,40 @@ const { title, description, image = "", date } = Astro.props;
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="hover-text">
|
<div class="hover-text">
|
||||||
|
<div>
|
||||||
<p set:html={description} />
|
<p set:html={description} />
|
||||||
</div>
|
</div>
|
||||||
|
</div>
|
||||||
</article>
|
</article>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
document.addEventListener('DOMContentLoaded', () => {
|
||||||
|
const hoverCards = document.querySelectorAll('.hover-card');
|
||||||
|
|
||||||
|
hoverCards.forEach(card => {
|
||||||
|
card.addEventListener('click', (e) => {
|
||||||
|
// Only toggle on mobile devices
|
||||||
|
if (window.innerWidth <= 768) {
|
||||||
|
e.preventDefault();
|
||||||
|
|
||||||
|
// Close all other active cards first
|
||||||
|
hoverCards.forEach(otherCard => {
|
||||||
|
if (otherCard !== card) {
|
||||||
|
otherCard.classList.remove('active');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// Toggle current card
|
||||||
|
card.classList.toggle('active');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// Close card when clicking outside (mobile only)
|
||||||
|
document.addEventListener('click', (e) => {
|
||||||
|
if (window.innerWidth <= 768 && !card.contains(e.target)) {
|
||||||
|
card.classList.remove('active');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
</script>
|
||||||
@ -1,6 +1,6 @@
|
|||||||
---
|
---
|
||||||
// src/components/ImageCarousel.astro
|
// src/components/ImageCarousel.astro
|
||||||
import "../../styles/components/ImageCarousel.css";
|
import "../styles/components/ImageCarousel.css";
|
||||||
|
|
||||||
interface Image {
|
interface Image {
|
||||||
src: string;
|
src: string;
|
||||||
|
|||||||
@ -2,7 +2,9 @@
|
|||||||
// src/components/Layout.astro
|
// src/components/Layout.astro
|
||||||
import Header from "./Header.astro";
|
import Header from "./Header.astro";
|
||||||
import Footer from "./Footer.astro";
|
import Footer from "./Footer.astro";
|
||||||
import "../../styles/components/Layout.css"
|
import "../styles/components/Layout.css"
|
||||||
|
import "../styles/variables.css"
|
||||||
|
import "../styles/index.css"
|
||||||
---
|
---
|
||||||
|
|
||||||
<!doctype html>
|
<!doctype html>
|
||||||
@ -14,8 +16,6 @@ import "../../styles/components/Layout.css"
|
|||||||
<meta charset="UTF-8" />
|
<meta charset="UTF-8" />
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
||||||
<title>Gallus Pub</title>
|
<title>Gallus Pub</title>
|
||||||
<link rel="stylesheet" href="/styles/variables.css" />
|
|
||||||
<link rel="stylesheet" href="/styles/index.css" />
|
|
||||||
</head>
|
</head>
|
||||||
|
|
||||||
<body>
|
<body>
|
||||||
|
|||||||
@ -1,6 +1,6 @@
|
|||||||
---
|
---
|
||||||
// src/components/Welcome.astro
|
// src/components/Welcome.astro
|
||||||
import "../../styles/components/Welcome.css"
|
import "../styles/components/Welcome.css"
|
||||||
|
|
||||||
const { id } = Astro.props;
|
const { id } = Astro.props;
|
||||||
---
|
---
|
||||||
@ -9,7 +9,8 @@ const { id } = Astro.props;
|
|||||||
|
|
||||||
<div class="welcome-text">
|
<div class="welcome-text">
|
||||||
|
|
||||||
<h2>Herzlich willkommen im Gallus Pub!</h2>
|
<h2>Herzlich willkommen im</h2>
|
||||||
|
<h2>Gallus Pub!</h2>
|
||||||
|
|
||||||
<p>
|
<p>
|
||||||
Wie die meisten bereits wissen, ist hier jeder willkommen - ob jung
|
Wie die meisten bereits wissen, ist hier jeder willkommen - ob jung
|
||||||
@ -51,7 +52,7 @@ const { id } = Astro.props;
|
|||||||
|
|
||||||
|
|
||||||
<div class="welcome-image">
|
<div class="welcome-image">
|
||||||
<img src="/images/Welcome.png" alt="Welcome backgrount image" />
|
<img src="/images/Welcome.png" alt="Welcome background image" />
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
</section>
|
</section>
|
||||||
|
|||||||
26
src/content/events.json
Normal file
@ -0,0 +1,26 @@
|
|||||||
|
[
|
||||||
|
{
|
||||||
|
"image": "/images/karaoke.jpg",
|
||||||
|
"title": "Karaoke",
|
||||||
|
"date": "Mittwoch - Samstag",
|
||||||
|
"description": "Bei uns gibt es Karaoke Mi-Sa!! <br>\nSeid ihr eine Gruppe und lieber unter euch? ..unseren 2.Stock kannst du auch mieten ;) <br>\nReserviere am besten gleich per Whatsapp <a href=\"tel:+41772322770\">077 232 27 70</a>"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"image": "/images/pub_quiz.jpg",
|
||||||
|
"title": "Pub Quiz",
|
||||||
|
"date": "Jeden Freitag",
|
||||||
|
"description": "Jeden Freitag findet unser <b>Pub Quiz</b> statt. Gespielt wird tischweise in 3-4 Runden. <br>\nJede Woche gibt es ein anderes Thema. Es geht um Ruhm und Ehre und zusätzlich werden die Sieger der Herzen durch das Publikum gekürt! <3 <br>\nAuch Einzelpersonen sind herzlich willkommen! <br>\n*zum mitmachen minimum 1 Getränk konsumieren oder 5CHF"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"image": "/images/crepes_sucette.jpg",
|
||||||
|
"title": "Crepes Sucette <br /> Live Music im Gallus Pub!",
|
||||||
|
"date": "Do, 04. September 2025",
|
||||||
|
"description": "<b>20:00 Uhr</b> <br>\n<a href=\"Metzgergasse 13, 9000 St. Gallen\">Metzgergasse 13, 9000 St. Gallen</a> <br>\nErlebt einen musikalischen Abend mit der Band <b>Crepes Sucette</b> <br>\nJetzt reservieren: <a href=\"tel:+41772322770\">077 232 27 70</a>"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"image": "/images/kevin_mcflannigan.jpeg",
|
||||||
|
"title": "Kevin McFlannigan <br> Live Music im Gallus Pub!",
|
||||||
|
"date": "Sa, 27. September 2025",
|
||||||
|
"description": "<b>ab 20:00 Uhr</b> <br>\nSinger & Songwriter Kevin McFlannigan <br>\nEintritt ist Frei / Hutkollekte <br>"
|
||||||
|
}
|
||||||
|
]
|
||||||
12
src/content/gallery.json
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
[
|
||||||
|
{ "src": "/images/Logo.png", "alt": "Erstes Bild" },
|
||||||
|
{ "src": "/images/Logo.png", "alt": "Zweites Bild" },
|
||||||
|
{ "src": "/images/Logo.png", "alt": "Drittes Bild" },
|
||||||
|
{ "src": "/images/Logo.png", "alt": "Viertes Bild" },
|
||||||
|
{ "src": "/images/Logo.png", "alt": "Fünftes Bild" },
|
||||||
|
{ "src": "/images/Logo.png", "alt": "Sechstes Bild" },
|
||||||
|
{ "src": "/images/Logo.png", "alt": "Siebtes Bild" },
|
||||||
|
{ "src": "/images/Logo.png", "alt": "Achtes Bild" },
|
||||||
|
{ "src": "/images/Logo.png", "alt": "Neuntes Bild" },
|
||||||
|
{ "src": "/images/Logo.png", "alt": "Zehntes Bild" }
|
||||||
|
]
|
||||||
94
src/pages/admin/index.astro
Normal file
@ -0,0 +1,94 @@
|
|||||||
|
---
|
||||||
|
import Layout from "../../components/Layout.astro";
|
||||||
|
import eventsData from "../../content/events.json";
|
||||||
|
import imagesData from "../../content/gallery.json";
|
||||||
|
import { getSessionFromRequest } from "../../utils/session";
|
||||||
|
|
||||||
|
const session = getSessionFromRequest(Astro.request);
|
||||||
|
if (!session?.user) {
|
||||||
|
// Not logged in: redirect to OAuth login
|
||||||
|
return Astro.redirect("/api/auth/login");
|
||||||
|
}
|
||||||
|
const csrf = session.csrf;
|
||||||
|
const events = eventsData;
|
||||||
|
const images = imagesData;
|
||||||
|
---
|
||||||
|
|
||||||
|
<Layout>
|
||||||
|
<section>
|
||||||
|
<h1>Admin</h1>
|
||||||
|
<p>Eingeloggt als {session.user.login}</p>
|
||||||
|
<form id="editor">
|
||||||
|
<h2>Events (JSON)</h2>
|
||||||
|
<textarea id="events" name="events" rows="16" style="width:100%;font-family:monospace;">{JSON.stringify(events, null, 2)}</textarea>
|
||||||
|
|
||||||
|
<h2>Galerie (JSON)</h2>
|
||||||
|
<textarea id="images" name="images" rows="10" style="width:100%;font-family:monospace;">{JSON.stringify(images, null, 2)}</textarea>
|
||||||
|
|
||||||
|
<h2>Bilder hochladen</h2>
|
||||||
|
<input type="file" id="fileInput" multiple accept="image/*" />
|
||||||
|
|
||||||
|
<div style="margin-top:1rem;display:flex;gap:.5rem;">
|
||||||
|
<button id="saveBtn" type="button">Speichern</button>
|
||||||
|
<button id="logoutBtn" type="button">Logout</button>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<meta name="csrf" content={csrf} />
|
||||||
|
<script type="module">
|
||||||
|
const csrf = document.querySelector('meta[name="csrf"]').content;
|
||||||
|
|
||||||
|
async function uploadFiles(files){
|
||||||
|
const uploads = [];
|
||||||
|
for (const file of files){
|
||||||
|
const arrayBuffer = await file.arrayBuffer();
|
||||||
|
const base64 = btoa(String.fromCharCode(...new Uint8Array(arrayBuffer)));
|
||||||
|
uploads.push({ path: `public/images/${file.name}`, content: base64 });
|
||||||
|
}
|
||||||
|
return uploads;
|
||||||
|
}
|
||||||
|
|
||||||
|
async function save(){
|
||||||
|
let events, images;
|
||||||
|
try{
|
||||||
|
events = JSON.parse(document.getElementById('events').value);
|
||||||
|
images = JSON.parse(document.getElementById('images').value);
|
||||||
|
}catch(e){
|
||||||
|
alert('JSON fehlerhaft: ' + e.message);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const files = [
|
||||||
|
{ path: 'src/content/events.json', content: JSON.stringify(events, null, 2) },
|
||||||
|
{ path: 'src/content/gallery.json', content: JSON.stringify(images, null, 2) },
|
||||||
|
];
|
||||||
|
|
||||||
|
const input = document.getElementById('fileInput');
|
||||||
|
if (input.files && input.files.length){
|
||||||
|
const imageFiles = await uploadFiles(input.files);
|
||||||
|
files.push(...imageFiles);
|
||||||
|
}
|
||||||
|
|
||||||
|
const res = await fetch('/api/save', {
|
||||||
|
method: 'POST',
|
||||||
|
headers: { 'Content-Type': 'application/json', 'X-CSRF': csrf },
|
||||||
|
body: JSON.stringify({ files, message: 'Admin-Inhalte aktualisiert' })
|
||||||
|
});
|
||||||
|
if (!res.ok){
|
||||||
|
const t = await res.text();
|
||||||
|
alert('Fehler beim Speichern: ' + t);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
alert('Gespeichert! Build wird gestartet.');
|
||||||
|
// optional: Seite neu laden
|
||||||
|
location.reload();
|
||||||
|
}
|
||||||
|
|
||||||
|
document.getElementById('saveBtn').addEventListener('click', save);
|
||||||
|
document.getElementById('logoutBtn').addEventListener('click', async () => {
|
||||||
|
await fetch('/api/auth/logout', { method: 'POST' });
|
||||||
|
location.href = '/';
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
</Layout>
|
||||||
116
src/pages/api/auth/callback.ts
Normal file
@ -0,0 +1,116 @@
|
|||||||
|
import type { APIRoute } from "astro";
|
||||||
|
import { createSessionCookie, sessionCookieHeader, randomToken } from "../../../utils/session";
|
||||||
|
|
||||||
|
export const GET: APIRoute = async ({ request }) => {
|
||||||
|
const url = new URL(request.url);
|
||||||
|
const code = url.searchParams.get("code");
|
||||||
|
const state = url.searchParams.get("state");
|
||||||
|
const cookie = request.headers.get("cookie") || "";
|
||||||
|
const stateCookie = cookie.match(/(?:^|; )oauth_state=([^;]+)/)?.[1];
|
||||||
|
|
||||||
|
if (!code || !state || !stateCookie || stateCookie !== state) {
|
||||||
|
return new Response("Invalid OAuth state", { status: 400 });
|
||||||
|
}
|
||||||
|
|
||||||
|
const clientId = process.env.OAUTH_CLIENT_ID;
|
||||||
|
const clientSecret = process.env.OAUTH_CLIENT_SECRET;
|
||||||
|
const tokenUrlRaw = process.env.OAUTH_TOKEN_URL;
|
||||||
|
const userinfoUrl = process.env.OAUTH_USERINFO_URL;
|
||||||
|
if (!clientId || !clientSecret || !tokenUrlRaw || !userinfoUrl) {
|
||||||
|
return new Response("OAuth not fully configured. Please set OAUTH_CLIENT_ID, OAUTH_CLIENT_SECRET, OAUTH_TOKEN_URL, OAUTH_USERINFO_URL.", { status: 500 });
|
||||||
|
}
|
||||||
|
|
||||||
|
// Compute redirect_uri consistent with login, robust against invalid PUBLIC_BASE_URL
|
||||||
|
let redirectUri: string;
|
||||||
|
try {
|
||||||
|
if (process.env.PUBLIC_BASE_URL) {
|
||||||
|
const base = new URL(process.env.PUBLIC_BASE_URL);
|
||||||
|
redirectUri = new URL("/api/auth/callback", base).toString();
|
||||||
|
} else {
|
||||||
|
const reqUrl = new URL(request.url);
|
||||||
|
redirectUri = new URL("/api/auth/callback", `${reqUrl.protocol}//${reqUrl.host}`).toString();
|
||||||
|
}
|
||||||
|
} catch {
|
||||||
|
const reqUrl = new URL(request.url);
|
||||||
|
redirectUri = new URL("/api/auth/callback", `${reqUrl.protocol}//${reqUrl.host}`).toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Validate token URL
|
||||||
|
let tokenUrl: URL;
|
||||||
|
try {
|
||||||
|
tokenUrl = new URL(tokenUrlRaw);
|
||||||
|
} catch {
|
||||||
|
return new Response("Invalid OAUTH_TOKEN_URL", { status: 500 });
|
||||||
|
}
|
||||||
|
|
||||||
|
// Exchange code for token
|
||||||
|
const params = new URLSearchParams();
|
||||||
|
params.set("client_id", clientId);
|
||||||
|
params.set("client_secret", clientSecret);
|
||||||
|
params.set("code", code);
|
||||||
|
params.set("grant_type", "authorization_code");
|
||||||
|
params.set("redirect_uri", redirectUri);
|
||||||
|
|
||||||
|
const tokenRes = await fetch(tokenUrl, {
|
||||||
|
method: "POST",
|
||||||
|
headers: { "Content-Type": "application/x-www-form-urlencoded" },
|
||||||
|
body: params.toString(),
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!tokenRes.ok) {
|
||||||
|
const t = await tokenRes.text();
|
||||||
|
return new Response(`OAuth token error: ${tokenRes.status} ${t}`, { status: 500 });
|
||||||
|
}
|
||||||
|
|
||||||
|
const tokenData = await tokenRes.json().catch(async () => {
|
||||||
|
// Some Gitea versions return application/x-www-form-urlencoded
|
||||||
|
const text = await tokenRes.text();
|
||||||
|
const usp = new URLSearchParams(text);
|
||||||
|
return Object.fromEntries(usp.entries());
|
||||||
|
});
|
||||||
|
const accessToken = tokenData.access_token || tokenData["access_token"];
|
||||||
|
if (!accessToken) {
|
||||||
|
return new Response("No access token", { status: 500 });
|
||||||
|
}
|
||||||
|
|
||||||
|
const userRes = await fetch(userinfoUrl, {
|
||||||
|
headers: { Authorization: `token ${accessToken}` },
|
||||||
|
});
|
||||||
|
if (!userRes.ok) {
|
||||||
|
const t = await userRes.text();
|
||||||
|
return new Response(`Userinfo error: ${userRes.status} ${t}`, { status: 500 });
|
||||||
|
}
|
||||||
|
const user = await userRes.json();
|
||||||
|
|
||||||
|
// Optional allowlist
|
||||||
|
const allowedUsers = (process.env.OAUTH_ALLOWED_USERS || "").split(/[\,\s]+/).filter(Boolean);
|
||||||
|
const allowedOrg = process.env.OAUTH_ALLOWED_ORG;
|
||||||
|
if (allowedUsers.length && !allowedUsers.includes(user.login)) {
|
||||||
|
return new Response("Forbidden", { status: 403 });
|
||||||
|
}
|
||||||
|
if (allowedOrg) {
|
||||||
|
// Best-effort org check
|
||||||
|
try {
|
||||||
|
const orgsRes = await fetch(process.env.GITEA_BASE + "/api/v1/user/orgs", {
|
||||||
|
headers: { Authorization: `token ${accessToken}` },
|
||||||
|
});
|
||||||
|
if (orgsRes.ok) {
|
||||||
|
const orgs = await orgsRes.json();
|
||||||
|
const inOrg = Array.isArray(orgs) && orgs.some((o: any) => o.username === allowedOrg || o.login === allowedOrg || o.name === allowedOrg);
|
||||||
|
if (!inOrg) return new Response("Forbidden (org)", { status: 403 });
|
||||||
|
}
|
||||||
|
} catch {}
|
||||||
|
}
|
||||||
|
|
||||||
|
const csrf = randomToken(16);
|
||||||
|
const sessionValue = createSessionCookie({
|
||||||
|
user: { id: user.id, login: user.login, name: user.full_name || user.username || user.login, email: user.email },
|
||||||
|
csrf,
|
||||||
|
});
|
||||||
|
|
||||||
|
const headers = new Headers();
|
||||||
|
headers.set("Set-Cookie", sessionCookieHeader(sessionValue));
|
||||||
|
headers.append("Set-Cookie", "oauth_state=; Path=/; HttpOnly; Secure; SameSite=Lax; Max-Age=0");
|
||||||
|
headers.set("Location", "/admin");
|
||||||
|
return new Response(null, { status: 302, headers });
|
||||||
|
};
|
||||||
53
src/pages/api/auth/login.ts
Normal file
@ -0,0 +1,53 @@
|
|||||||
|
import type { APIRoute } from "astro";
|
||||||
|
import { randomToken, setTempCookie } from "../../../utils/session";
|
||||||
|
|
||||||
|
export const GET: APIRoute = async ({ request }) => {
|
||||||
|
const clientId = process.env.OAUTH_CLIENT_ID;
|
||||||
|
const authorizeUrlRaw = process.env.OAUTH_AUTHORIZE_URL;
|
||||||
|
if (!clientId || !authorizeUrlRaw) {
|
||||||
|
return new Response(
|
||||||
|
"OAuth not configured. Please set OAUTH_CLIENT_ID and OAUTH_AUTHORIZE_URL (and related secrets) for local dev.",
|
||||||
|
{ status: 500 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Determine callback URL
|
||||||
|
let finalRedirect: string;
|
||||||
|
try {
|
||||||
|
if (process.env.PUBLIC_BASE_URL) {
|
||||||
|
// Ensure PUBLIC_BASE_URL is an absolute URL
|
||||||
|
const base = new URL(process.env.PUBLIC_BASE_URL);
|
||||||
|
finalRedirect = new URL("/api/auth/callback", base).toString();
|
||||||
|
} else {
|
||||||
|
// Fallback to current request origin
|
||||||
|
const reqUrl = new URL(request.url);
|
||||||
|
finalRedirect = new URL("/api/auth/callback", `${reqUrl.protocol}//${reqUrl.host}`).toString();
|
||||||
|
}
|
||||||
|
} catch {
|
||||||
|
// As a last resort, use request URL
|
||||||
|
const reqUrl = new URL(request.url);
|
||||||
|
finalRedirect = new URL("/api/auth/callback", `${reqUrl.protocol}//${reqUrl.host}`).toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Validate authorize URL
|
||||||
|
let authorizeUrl: URL;
|
||||||
|
try {
|
||||||
|
authorizeUrl = new URL(authorizeUrlRaw);
|
||||||
|
} catch {
|
||||||
|
return new Response("Invalid OAUTH_AUTHORIZE_URL", { status: 500 });
|
||||||
|
}
|
||||||
|
|
||||||
|
const state = randomToken(16);
|
||||||
|
authorizeUrl.searchParams.set("client_id", clientId);
|
||||||
|
authorizeUrl.searchParams.set("redirect_uri", finalRedirect);
|
||||||
|
authorizeUrl.searchParams.set("response_type", "code");
|
||||||
|
authorizeUrl.searchParams.set("state", state);
|
||||||
|
|
||||||
|
return new Response(null, {
|
||||||
|
status: 302,
|
||||||
|
headers: {
|
||||||
|
Location: authorizeUrl.toString(),
|
||||||
|
"Set-Cookie": setTempCookie("oauth_state", state),
|
||||||
|
},
|
||||||
|
});
|
||||||
|
};
|
||||||
11
src/pages/api/auth/logout.ts
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
import type { APIRoute } from "astro";
|
||||||
|
import { clearCookieHeader } from "../../../utils/session";
|
||||||
|
|
||||||
|
export const POST: APIRoute = async () => {
|
||||||
|
return new Response(null, {
|
||||||
|
status: 204,
|
||||||
|
headers: {
|
||||||
|
"Set-Cookie": clearCookieHeader(),
|
||||||
|
},
|
||||||
|
});
|
||||||
|
};
|
||||||
97
src/pages/api/save.ts
Normal file
@ -0,0 +1,97 @@
|
|||||||
|
import type { APIRoute } from "astro";
|
||||||
|
import { getSessionFromRequest } from "../../utils/session";
|
||||||
|
|
||||||
|
const GITEA_BASE = process.env.GITEA_BASE!;
|
||||||
|
const GITEA_OWNER = process.env.GITEA_OWNER!;
|
||||||
|
const GITEA_REPO = process.env.GITEA_REPO!;
|
||||||
|
const GITEA_TOKEN = process.env.GITEA_TOKEN!;
|
||||||
|
const DEFAULT_BRANCH = process.env.GIT_BRANCH || "main";
|
||||||
|
|
||||||
|
function isAllowedPath(path: string) {
|
||||||
|
if (path === "src/content/events.json") return true;
|
||||||
|
if (path === "src/content/gallery.json") return true;
|
||||||
|
if (path.startsWith("public/images/")) {
|
||||||
|
return /\.(png|jpg|jpeg|gif|webp|svg)$/i.test(path);
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
async function getShaIfExists(path: string): Promise<string | undefined> {
|
||||||
|
const url = `${GITEA_BASE}/api/v1/repos/${encodeURIComponent(GITEA_OWNER)}/${encodeURIComponent(GITEA_REPO)}/contents/${encodeURIComponent(path)}?ref=${encodeURIComponent(DEFAULT_BRANCH)}`;
|
||||||
|
const res = await fetch(url, {
|
||||||
|
headers: { Authorization: `token ${GITEA_TOKEN}` },
|
||||||
|
});
|
||||||
|
if (res.status === 404) return undefined;
|
||||||
|
if (!res.ok) throw new Error(`Gitea get sha error ${res.status}`);
|
||||||
|
const data = await res.json();
|
||||||
|
return data.sha;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const POST: APIRoute = async ({ request }) => {
|
||||||
|
const session = getSessionFromRequest(request);
|
||||||
|
if (!session?.user) return new Response(JSON.stringify({ error: "unauthorized" }), { status: 401 });
|
||||||
|
|
||||||
|
// CSRF header required
|
||||||
|
const csrfHeader = request.headers.get("x-csrf") || request.headers.get("X-CSRF");
|
||||||
|
if (!csrfHeader || csrfHeader !== session.csrf) {
|
||||||
|
return new Response(JSON.stringify({ error: "invalid csrf" }), { status: 403 });
|
||||||
|
}
|
||||||
|
|
||||||
|
let payload: any;
|
||||||
|
try {
|
||||||
|
payload = await request.json();
|
||||||
|
} catch {
|
||||||
|
return new Response(JSON.stringify({ error: "invalid json" }), { status: 400 });
|
||||||
|
}
|
||||||
|
if (!payload || !Array.isArray(payload.files)) {
|
||||||
|
return new Response(JSON.stringify({ error: "missing files" }), { status: 400 });
|
||||||
|
}
|
||||||
|
|
||||||
|
const results: any[] = [];
|
||||||
|
for (const file of payload.files) {
|
||||||
|
const path = String(file.path || "");
|
||||||
|
if (!isAllowedPath(path)) {
|
||||||
|
return new Response(JSON.stringify({ error: `path not allowed: ${path}` }), { status: 400 });
|
||||||
|
}
|
||||||
|
let contentBase64: string;
|
||||||
|
if (path.startsWith("public/images/")) {
|
||||||
|
// Expect already base64 string of binary
|
||||||
|
contentBase64 = String(file.content || "");
|
||||||
|
// Remove possible data URL prefix
|
||||||
|
const match = contentBase64.match(/^data:[^;]+;base64,(.*)$/);
|
||||||
|
if (match) contentBase64 = match[1];
|
||||||
|
} else {
|
||||||
|
// Text file
|
||||||
|
contentBase64 = Buffer.from(String(file.content ?? ""), "utf-8").toString("base64");
|
||||||
|
}
|
||||||
|
|
||||||
|
const sha = await getShaIfExists(path).catch(() => undefined);
|
||||||
|
const url = `${GITEA_BASE}/api/v1/repos/${encodeURIComponent(GITEA_OWNER)}/${encodeURIComponent(GITEA_REPO)}/contents/${encodeURIComponent(path)}`;
|
||||||
|
const body: any = {
|
||||||
|
content: contentBase64,
|
||||||
|
message: payload.message || `Update ${path}`,
|
||||||
|
branch: DEFAULT_BRANCH,
|
||||||
|
};
|
||||||
|
if (sha) body.sha = sha;
|
||||||
|
if (session.user) {
|
||||||
|
body.author = { name: session.user.name || session.user.login, email: session.user.email || `${session.user.login}@users.noreply` };
|
||||||
|
body.committer = { name: session.user.name || session.user.login, email: session.user.email || `${session.user.login}@users.noreply` };
|
||||||
|
}
|
||||||
|
|
||||||
|
const res = await fetch(url, {
|
||||||
|
method: "PUT",
|
||||||
|
headers: {
|
||||||
|
Authorization: `token ${GITEA_TOKEN}`,
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
},
|
||||||
|
body: JSON.stringify(body),
|
||||||
|
});
|
||||||
|
if (!res.ok) {
|
||||||
|
const t = await res.text();
|
||||||
|
return new Response(JSON.stringify({ error: `Gitea error for ${path}: ${res.status} ${t}` }), { status: 500 });
|
||||||
|
}
|
||||||
|
results.push(await res.json());
|
||||||
|
}
|
||||||
|
|
||||||
|
return new Response(JSON.stringify({ ok: true, results }), { status: 200, headers: { "Content-Type": "application/json" } });
|
||||||
|
};
|
||||||
BIN
src/public/Logo.png
Normal file
|
After Width: | Height: | Size: 110 KiB |
9
src/public/favicon.svg
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 128 128">
|
||||||
|
<path d="M50.4 78.5a75.1 75.1 0 0 0-28.5 6.9l24.2-65.7c.7-2 1.9-3.2 3.4-3.2h29c1.5 0 2.7 1.2 3.4 3.2l24.2 65.7s-11.6-7-28.5-7L67 45.5c-.4-1.7-1.6-2.8-2.9-2.8-1.3 0-2.5 1.1-2.9 2.7L50.4 78.5Zm-1.1 28.2Zm-4.2-20.2c-2 6.6-.6 15.8 4.2 20.2a17.5 17.5 0 0 1 .2-.7 5.5 5.5 0 0 1 5.7-4.5c2.8.1 4.3 1.5 4.7 4.7.2 1.1.2 2.3.2 3.5v.4c0 2.7.7 5.2 2.2 7.4a13 13 0 0 0 5.7 4.9v-.3l-.2-.3c-1.8-5.6-.5-9.5 4.4-12.8l1.5-1a73 73 0 0 0 3.2-2.2 16 16 0 0 0 6.8-11.4c.3-2 .1-4-.6-6l-.8.6-1.6 1a37 37 0 0 1-22.4 2.7c-5-.7-9.7-2-13.2-6.2Z" />
|
||||||
|
<style>
|
||||||
|
path { fill: #000; }
|
||||||
|
@media (prefers-color-scheme: dark) {
|
||||||
|
path { fill: #FFF; }
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 749 B |
BIN
src/public/images/Background.png
Normal file
|
After Width: | Height: | Size: 506 KiB |
BIN
src/public/images/Logo.png
Normal file
|
After Width: | Height: | Size: 110 KiB |
BIN
src/public/images/Welcome.png
Normal file
|
After Width: | Height: | Size: 57 KiB |
|
Before Width: | Height: | Size: 124 KiB After Width: | Height: | Size: 124 KiB |
BIN
src/public/images/karaoke.jpg
Normal file
|
After Width: | Height: | Size: 94 KiB |
|
Before Width: | Height: | Size: 384 KiB After Width: | Height: | Size: 384 KiB |
BIN
src/public/images/pub_quiz.jpg
Normal file
|
After Width: | Height: | Size: 49 KiB |
BIN
src/public/pdf/Menu.pdf
Normal file
@ -1,5 +1,3 @@
|
|||||||
/* styles/components/ContactForm.css */
|
|
||||||
|
|
||||||
.contact-container {
|
.contact-container {
|
||||||
max-width: 800px;
|
max-width: 800px;
|
||||||
margin: 2rem auto;
|
margin: 2rem auto;
|
||||||
190
src/styles/components/Header.css
Normal file
@ -0,0 +1,190 @@
|
|||||||
|
.header {
|
||||||
|
position: fixed;
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
right: 0;
|
||||||
|
background-color: #0e0c0c;
|
||||||
|
z-index: 1000;
|
||||||
|
box-shadow: 0 2px 5px rgba(0, 0, 0, 0.1);
|
||||||
|
backdrop-filter: blur(5px);
|
||||||
|
width: 100%;
|
||||||
|
border-bottom: 1px solid #444;
|
||||||
|
}
|
||||||
|
|
||||||
|
.header-spacer {
|
||||||
|
height: 70px;
|
||||||
|
/* Should match the header height */
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Desktop Layout */
|
||||||
|
.desktop-layout {
|
||||||
|
display: flex;
|
||||||
|
justify-content: space-between;
|
||||||
|
align-items: center;
|
||||||
|
padding: 1rem 2rem;
|
||||||
|
height: 70px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.logo-container {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.logo {
|
||||||
|
margin-left: 2em;
|
||||||
|
height: 4em;
|
||||||
|
width: auto;
|
||||||
|
transition: transform 0.3s ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
.logo:hover {
|
||||||
|
transform: scale(1.05);
|
||||||
|
}
|
||||||
|
|
||||||
|
.nav-main {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.nav-main a {
|
||||||
|
margin: 0 1rem;
|
||||||
|
color: white;
|
||||||
|
text-decoration: none;
|
||||||
|
font-size: 1rem;
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
|
||||||
|
.nav-main a:hover {
|
||||||
|
color: #ffa500;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Mobile Layout */
|
||||||
|
.mobile-layout {
|
||||||
|
display: none;
|
||||||
|
flex-direction: column;
|
||||||
|
align-items: center;
|
||||||
|
padding: 1rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.mobile-logo-container {
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
margin-bottom: 0.5rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.mobile-logo-container .logo {
|
||||||
|
margin: 0;
|
||||||
|
height: 3.5em;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Burger Menu Styles */
|
||||||
|
.burger-menu {
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
cursor: pointer;
|
||||||
|
margin: 0.5rem 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.burger-icon {
|
||||||
|
width: 30px;
|
||||||
|
height: 24px;
|
||||||
|
position: relative;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
justify-content: space-between;
|
||||||
|
}
|
||||||
|
|
||||||
|
.burger-icon span {
|
||||||
|
display: block;
|
||||||
|
height: 3px;
|
||||||
|
width: 100%;
|
||||||
|
background-color: white;
|
||||||
|
border-radius: 3px;
|
||||||
|
transition: all 0.3s ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
.burger-icon.active span:nth-child(1) {
|
||||||
|
transform: translateY(10.5px) rotate(45deg);
|
||||||
|
}
|
||||||
|
|
||||||
|
.burger-icon.active span:nth-child(2) {
|
||||||
|
opacity: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.burger-icon.active span:nth-child(3) {
|
||||||
|
transform: translateY(-10.5px) rotate(-45deg);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Mobile Menu Styles */
|
||||||
|
.mobile-menu {
|
||||||
|
display: none;
|
||||||
|
width: 100%;
|
||||||
|
background-color: #0e0c0c;
|
||||||
|
flex-direction: column;
|
||||||
|
align-items: center;
|
||||||
|
padding: 0;
|
||||||
|
box-shadow: 0 5px 10px rgba(0, 0, 0, 0.2);
|
||||||
|
max-height: 0;
|
||||||
|
overflow: hidden;
|
||||||
|
transition: max-height 0.3s ease;
|
||||||
|
z-index: 999;
|
||||||
|
}
|
||||||
|
|
||||||
|
.mobile-menu.active {
|
||||||
|
max-height: 300px; /* Adjust based on content */
|
||||||
|
padding: 0.5rem 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.mobile-menu a {
|
||||||
|
margin: 0.5rem 0;
|
||||||
|
color: white;
|
||||||
|
text-decoration: none;
|
||||||
|
font-size: 1.2rem;
|
||||||
|
text-align: center;
|
||||||
|
width: 100%;
|
||||||
|
padding: 0.5rem 0;
|
||||||
|
border-bottom: 1px solid #333;
|
||||||
|
}
|
||||||
|
|
||||||
|
.mobile-menu a:last-child {
|
||||||
|
border-bottom: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.mobile-menu a:hover {
|
||||||
|
color: #ffa500;
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (max-width: 768px) {
|
||||||
|
.header-spacer {
|
||||||
|
height: 120px; /* Adjusted for the taller mobile header */
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Hide desktop layout, show mobile layout */
|
||||||
|
.desktop-layout {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.mobile-layout {
|
||||||
|
display: flex;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Show mobile menu when active */
|
||||||
|
.mobile-menu.active {
|
||||||
|
display: flex;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (max-width: 480px) {
|
||||||
|
.header-spacer {
|
||||||
|
height: 110px; /* Slightly smaller for very small screens */
|
||||||
|
}
|
||||||
|
|
||||||
|
.mobile-logo-container .logo {
|
||||||
|
height: 3em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.burger-icon {
|
||||||
|
width: 25px;
|
||||||
|
height: 20px;
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -1,4 +1,3 @@
|
|||||||
/* === Hero === */
|
|
||||||
.hero-overlay {
|
.hero-overlay {
|
||||||
background-image: linear-gradient(to bottom, rgba(0, 0, 0, 0.3), rgba(0, 0, 0, 1)), url('/images/Background.png');
|
background-image: linear-gradient(to bottom, rgba(0, 0, 0, 0.3), rgba(0, 0, 0, 1)), url('/images/Background.png');
|
||||||
background-size: cover;
|
background-size: cover;
|
||||||
@ -12,7 +12,8 @@ html {
|
|||||||
|
|
||||||
body {
|
body {
|
||||||
font-family: var(--font-family-primary), serif;
|
font-family: var(--font-family-primary), serif;
|
||||||
background-color: var(--color-background);
|
background-color: var(--color-background, #000000);
|
||||||
|
background: #000000;
|
||||||
color: var(--color-text);
|
color: var(--color-text);
|
||||||
line-height: var(--line-height);
|
line-height: var(--line-height);
|
||||||
}
|
}
|
||||||
@ -1,6 +1,6 @@
|
|||||||
:root {
|
:root {
|
||||||
/* Colors */
|
/* Colors */
|
||||||
--color-background: #000;
|
--color-background: #000000 !important;
|
||||||
--color-text: #f5f5f5;
|
--color-text: #f5f5f5;
|
||||||
--color-accent-green: #213b28;
|
--color-accent-green: #213b28;
|
||||||
--color-accent-beige: #ceb39b;
|
--color-accent-beige: #ceb39b;
|
||||||
74
src/utils/session.ts
Normal file
@ -0,0 +1,74 @@
|
|||||||
|
import crypto from "node:crypto";
|
||||||
|
|
||||||
|
export type SessionData = {
|
||||||
|
user?: {
|
||||||
|
id: number;
|
||||||
|
login: string;
|
||||||
|
name?: string;
|
||||||
|
email?: string;
|
||||||
|
};
|
||||||
|
csrf?: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
const COOKIE_NAME = "gp_session";
|
||||||
|
|
||||||
|
function b64url(input: Buffer | string) {
|
||||||
|
return Buffer.from(input)
|
||||||
|
.toString("base64")
|
||||||
|
.replace(/=/g, "")
|
||||||
|
.replace(/\+/g, "-")
|
||||||
|
.replace(/\//g, "_");
|
||||||
|
}
|
||||||
|
|
||||||
|
function sign(payload: string, secret: string) {
|
||||||
|
return crypto.createHmac("sha256", secret).update(payload).digest("base64url");
|
||||||
|
}
|
||||||
|
|
||||||
|
export function createSessionCookie(data: SessionData, secret = process.env.SESSION_SECRET || "") {
|
||||||
|
const payload = b64url(JSON.stringify(data));
|
||||||
|
const sig = sign(payload, secret);
|
||||||
|
return `${payload}.${sig}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function parseSessionCookie(cookieValue: string | undefined, secret = process.env.SESSION_SECRET || ""): SessionData | undefined {
|
||||||
|
if (!cookieValue) return undefined;
|
||||||
|
const [payload, sig] = cookieValue.split(".");
|
||||||
|
if (!payload || !sig) return undefined;
|
||||||
|
const expected = sign(payload, secret);
|
||||||
|
if (!crypto.timingSafeEqual(Buffer.from(sig), Buffer.from(expected))) return undefined;
|
||||||
|
try {
|
||||||
|
const json = Buffer.from(payload.replace(/-/g, "+").replace(/_/g, "/"), "base64").toString("utf-8");
|
||||||
|
return JSON.parse(json);
|
||||||
|
} catch {
|
||||||
|
return undefined;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export function clearCookieHeader(name = COOKIE_NAME) {
|
||||||
|
return `${name}=; Path=/; HttpOnly; Secure; SameSite=Lax; Max-Age=0`;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function sessionCookieHeader(value: string, name = COOKIE_NAME) {
|
||||||
|
// 7 days
|
||||||
|
const maxAge = 60 * 60 * 24 * 7;
|
||||||
|
return `${name}=${value}; Path=/; HttpOnly; Secure; SameSite=Lax; Max-Age=${maxAge}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function getSessionFromRequest(req: Request, secret = process.env.SESSION_SECRET || ""): SessionData | undefined {
|
||||||
|
const cookie = req.headers.get("cookie") || "";
|
||||||
|
const match = cookie.match(/(?:^|; )gp_session=([^;]+)/);
|
||||||
|
if (!match) return undefined;
|
||||||
|
return parseSessionCookie(match[1], secret);
|
||||||
|
}
|
||||||
|
|
||||||
|
export function randomToken(bytes = 32) {
|
||||||
|
return crypto.randomBytes(bytes).toString("hex");
|
||||||
|
}
|
||||||
|
|
||||||
|
export const COOKIE_NAME_STATE = "oauth_state";
|
||||||
|
|
||||||
|
export function setTempCookie(name: string, value: string) {
|
||||||
|
// short lived: 10 minutes
|
||||||
|
const maxAge = 60 * 10;
|
||||||
|
return `${name}=${value}; Path=/; HttpOnly; Secure; SameSite=Lax; Max-Age=${maxAge}`;
|
||||||
|
}
|
||||||
@ -1,92 +0,0 @@
|
|||||||
.header {
|
|
||||||
position: fixed;
|
|
||||||
top: 0;
|
|
||||||
left: 0;
|
|
||||||
right: 0;
|
|
||||||
background-color: #0e0c0c;
|
|
||||||
z-index: 1000;
|
|
||||||
box-shadow: 0 2px 5px rgba(0, 0, 0, 0.1);
|
|
||||||
backdrop-filter: blur(5px);
|
|
||||||
display: flex;
|
|
||||||
justify-content: space-between;
|
|
||||||
align-items: center;
|
|
||||||
padding: 1rem 2rem;
|
|
||||||
height: 70px;
|
|
||||||
width: 100%;
|
|
||||||
border-bottom: 1px solid #444;
|
|
||||||
}
|
|
||||||
|
|
||||||
.header-spacer {
|
|
||||||
height: 70px;
|
|
||||||
/* Should match the header height */
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
.logo-container {
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
}
|
|
||||||
|
|
||||||
.logo {
|
|
||||||
margin-left: 2em;
|
|
||||||
height: 4em;
|
|
||||||
width: auto;
|
|
||||||
transition: transform 0.3s ease;
|
|
||||||
}
|
|
||||||
|
|
||||||
.logo:hover {
|
|
||||||
transform: scale(1.05);
|
|
||||||
}
|
|
||||||
|
|
||||||
.nav-main {
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
}
|
|
||||||
|
|
||||||
.nav-main a {
|
|
||||||
margin: 0 1rem;
|
|
||||||
color: white;
|
|
||||||
text-decoration: none;
|
|
||||||
font-size: 1rem;
|
|
||||||
cursor: pointer;
|
|
||||||
}
|
|
||||||
|
|
||||||
.nav-main a:hover {
|
|
||||||
color: #ffa500;
|
|
||||||
}
|
|
||||||
|
|
||||||
@media (max-width: 768px) {
|
|
||||||
.header {
|
|
||||||
height: 65px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.logo {
|
|
||||||
margin-left: 1em;
|
|
||||||
height: 3em;
|
|
||||||
}
|
|
||||||
|
|
||||||
.header-spacer {
|
|
||||||
height: 65px;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@media (max-width: 480px) {
|
|
||||||
.header {
|
|
||||||
padding: 0.5rem;
|
|
||||||
height: 60px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.logo {
|
|
||||||
margin-left: 0.5em;
|
|
||||||
height: 2.5em;
|
|
||||||
}
|
|
||||||
|
|
||||||
.nav-main a {
|
|
||||||
margin: 0 0.5rem;
|
|
||||||
font-size: 0.9rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
.header-spacer {
|
|
||||||
height: 60px;
|
|
||||||
}
|
|
||||||
}
|
|
||||||