24 Commits
1.0.1 ... 1.0.7

Author SHA1 Message Date
1652e54ac6 chore: release 1.0.7
Some checks failed
Build release images / build-container (push) Failing after 11s
2025-04-04 22:28:35 +02:00
52528d31a4 chore: release 1.0.6 2025-04-04 22:27:47 +02:00
e6ff8ef80b ci: move to gitea actions 2025-04-04 22:27:16 +02:00
aa707ff0cc chore: update readme 2025-04-04 22:27:05 +02:00
dc0488b1b7 chore(deps). bump 2025-04-04 22:26:59 +02:00
4920da1df1 ci: change release message 2025-04-04 22:26:45 +02:00
f9a84f798b fix: i18n 2025-04-04 22:18:59 +02:00
c6f7210196 feat: show org names for teams_distance slide 2025-04-04 22:16:59 +02:00
c286969a9d chore(deps): pnpm@10 2025-04-04 22:15:35 +02:00
14ae9e49fb fix(login): bg image 2025-04-04 22:15:08 +02:00
4874b22796 🚀Bumped version to 1.0.5 2024-12-18 18:20:27 +01:00
72d34cbfd7 chore: remove husky 2024-12-18 18:20:08 +01:00
b10e964ad9 feat(footer): cleaned up text 2024-12-18 18:19:03 +01:00
919b2956ab feat: improved background img import 2024-12-18 18:18:46 +01:00
c4ad18cb4f chore(deps): pnpm@9 + node@23 2024-12-18 18:18:27 +01:00
ab4f82ccf1 chore(deps): bump some 2024-12-18 18:18:14 +01:00
1235776a62 feat(ci)!: Switched to woodpecker
All checks were successful
ci/woodpecker/push/build Pipeline was successful
2023-11-06 20:04:30 +01:00
4a1e26663e 🚀Bumped version to 1.0.4
All checks were successful
continuous-integration/drone/push Build is passing
2023-04-15 23:27:06 +02:00
2bdbd00189 fix: formatting of total km
All checks were successful
continuous-integration/drone/push Build is passing
2023-04-15 23:26:48 +02:00
85e7b7c231 🚀Bumped version to 1.0.3
All checks were successful
continuous-integration/drone/push Build is passing
2023-04-15 23:01:24 +02:00
d5f685a826 cleanup loading 2023-04-15 23:01:07 +02:00
88449174a1 drop laptime runner stats, fix runner distance page 2023-04-15 23:00:44 +02:00
d30be90102 🚀Bumped version to 1.0.2
All checks were successful
continuous-integration/drone/push Build is passing
2023-04-15 22:47:58 +02:00
7e3570e923 updated release config
All checks were successful
continuous-integration/drone/push Build is passing
2023-04-15 22:47:42 +02:00
14 changed files with 2742 additions and 2447 deletions

View File

@@ -1,101 +0,0 @@
---
kind: secret
name: docker_username
get:
path: odit-registry-builder
name: username
---
kind: secret
name: docker_password
get:
path: odit-registry-builder
name: password
---
kind: secret
name: npm_url
get:
path: odit-npm-cache
name: url
---
kind: pipeline
type: kubernetes
name: build:dev
steps:
- name: build dev
image: plugins/docker
depends_on: [clone]
settings:
username:
from_secret: docker_username
password:
from_secret: docker_password
build_args:
- NPM_REGISTRY_URL:
from_secret: npm_url
repo: registry.odit.services/lfk/beamershow
tags:
- dev
registry: registry.odit.services
mtu: 1000
trigger:
branch:
- dev
event:
- push
---
kind: pipeline
type: kubernetes
name: build:latest
steps:
- name: build latest
image: plugins/docker
depends_on: [clone]
settings:
username:
from_secret: docker_username
password:
from_secret: docker_password
build_args:
- NPM_REGISTRY_URL:
from_secret: npm_url
repo: registry.odit.services/lfk/beamershow
tags:
- latest
registry: registry.odit.services
mtu: 1000
trigger:
branch:
- main
event:
- push
---
kind: pipeline
type: kubernetes
name: build:tags
steps:
- name: build $DRONE_TAG
image: plugins/docker
depends_on: [clone]
settings:
username:
from_secret: docker_username
password:
from_secret: docker_password
build_args:
- NPM_REGISTRY_URL:
from_secret: npm_url
repo: registry.odit.services/lfk/beamershow
tags:
- '${DRONE_TAG}'
registry: registry.odit.services
mtu: 1000
trigger:
event:
- tag

View File

@@ -0,0 +1,33 @@
name: Build release images
on:
push:
tags:
- "*.*.*"
jobs:
build-container:
runs-on: ubuntu-latest
steps:
- name: Checkout
uses: actions/checkout@v4
- name: Set up Node.js
uses: actions/setup-node@v4
with:
node-version: 19
- run: npm i -g pnpm@10.7 && pnpm i
- run: pnpm licenses:export
- name: Login to registry
uses: docker/login-action@v3
with:
registry: registry.odit.services
username: ${{ vars.REGISTRY_USERNAME }}
password: ${{ secrets.REGISTRY_PASSWORD }}
- name: Set up Docker Buildx
uses: docker/setup-buildx-action@v3
- name: Build and push
uses: docker/build-push-action@v6
with:
push: true
tags: |
${{ vars.REGISTRY }}/lfk/beamershow:${{ github.ref_name }}
platforms: linux/amd64,linux/arm64

1
.husky/.gitignore vendored
View File

@@ -1 +0,0 @@
_

View File

@@ -1,5 +0,0 @@
#!/bin/sh
. "$(dirname "$0")/_/husky.sh"
yarn format
yarn license:export

View File

@@ -2,8 +2,61 @@
All notable changes to this project will be documented in this file. Dates are displayed in UTC. All notable changes to this project will be documented in this file. Dates are displayed in UTC.
#### [1.0.7](https://git.odit.services/lfk/beamershow/compare/1.0.6...1.0.7)
#### [1.0.6](https://git.odit.services/lfk/beamershow/compare/1.0.5...1.0.6)
> 4 April 2025
- feat: show org names for teams_distance slide [`c6f7210`](https://git.odit.services/lfk/beamershow/commit/c6f721019691cdc9f5492f920dbdd99e6b25ca9f)
- chore(deps). bump [`dc0488b`](https://git.odit.services/lfk/beamershow/commit/dc0488b1b7632326f4dd3bf5fff0ac8fa91fb128)
- ci: move to gitea actions [`e6ff8ef`](https://git.odit.services/lfk/beamershow/commit/e6ff8ef80b667692574319f20bebe6a4e872ca6e)
- chore: release 1.0.6 [`52528d3`](https://git.odit.services/lfk/beamershow/commit/52528d31a46f99bd44c9b7f18833159f9cd4db56)
- chore: update readme [`aa707ff`](https://git.odit.services/lfk/beamershow/commit/aa707ff0ccaa2cb4a69d3c7b396591cd40827c6d)
- chore(deps): pnpm@10 [`c286969`](https://git.odit.services/lfk/beamershow/commit/c286969a9de52519005013a6798f711cef26b465)
- ci: change release message [`4920da1`](https://git.odit.services/lfk/beamershow/commit/4920da1df1a51a071eeba44d8d2c1cb6c5f8d09c)
- fix: i18n [`f9a84f7`](https://git.odit.services/lfk/beamershow/commit/f9a84f798b1052530688aaabf654b08c2d466da5)
- fix(login): bg image [`14ae9e4`](https://git.odit.services/lfk/beamershow/commit/14ae9e49fbdc10a1c5681212713b166e333e4b4d)
#### [1.0.5](https://git.odit.services/lfk/beamershow/compare/1.0.4...1.0.5)
> 18 December 2024
- chore(deps): bump some [`ab4f82c`](https://git.odit.services/lfk/beamershow/commit/ab4f82ccf14a1569921845910e3b2a740f74aeea)
- feat(ci)!: Switched to woodpecker [`1235776`](https://git.odit.services/lfk/beamershow/commit/1235776a6255a3925f98ac6cacd3167f6e86e363)
- chore: remove husky [`72d34cb`](https://git.odit.services/lfk/beamershow/commit/72d34cbfd7f7e47f0239416469533aea772daa79)
- 🚀Bumped version to 1.0.5 [`4874b22`](https://git.odit.services/lfk/beamershow/commit/4874b227960c30b65d7bc822f987c9b957144953)
- chore(deps): pnpm@9 + node@23 [`c4ad18c`](https://git.odit.services/lfk/beamershow/commit/c4ad18cb4ff4dd13e50f4674d7b6016ed788ff63)
- feat(footer): cleaned up text [`b10e964`](https://git.odit.services/lfk/beamershow/commit/b10e964ad900fcfd2503884c248ea0f6d11a2fb9)
- feat: improved background img import [`919b295`](https://git.odit.services/lfk/beamershow/commit/919b2956ab5253a3697c14ac284f6da700afda91)
#### [1.0.4](https://git.odit.services/lfk/beamershow/compare/1.0.3...1.0.4)
> 15 April 2023
- 🚀Bumped version to 1.0.4 [`4a1e266`](https://git.odit.services/lfk/beamershow/commit/4a1e26663e29339d6373fa8c340ba2117d3ecf28)
- fix: formatting of total km [`2bdbd00`](https://git.odit.services/lfk/beamershow/commit/2bdbd001898b9dd5aa541f3cce90fc108e7e458c)
#### [1.0.3](https://git.odit.services/lfk/beamershow/compare/1.0.2...1.0.3)
> 15 April 2023
- 🚀Bumped version to 1.0.3 [`85e7b7c`](https://git.odit.services/lfk/beamershow/commit/85e7b7c231b92233b58373cd4ad56f514d6d789c)
- cleanup loading [`d5f685a`](https://git.odit.services/lfk/beamershow/commit/d5f685a8269737d148a06e0ec784c0074eac7d3f)
- drop laptime runner stats, fix runner distance page [`8844917`](https://git.odit.services/lfk/beamershow/commit/88449174a148ab7498075a15149388213a532c02)
#### [1.0.2](https://git.odit.services/lfk/beamershow/compare/1.0.1...1.0.2)
> 15 April 2023
- 🚀Bumped version to 1.0.2 [`d30be90`](https://git.odit.services/lfk/beamershow/commit/d30be90102b3065b16836adc44a686c677e3053d)
- updated release config [`7e3570e`](https://git.odit.services/lfk/beamershow/commit/7e3570e9231929f366047ff149364ec06344d2ee)
#### [1.0.1](https://git.odit.services/lfk/beamershow/compare/1.0.0...1.0.1) #### [1.0.1](https://git.odit.services/lfk/beamershow/compare/1.0.0...1.0.1)
> 15 April 2023
- 🚀Bumped version to 1.0.1 [`3ac0a3c`](https://git.odit.services/lfk/beamershow/commit/3ac0a3c142427adb3b3d461cd39a458685a85335)
- fix bg image? [`dc588e8`](https://git.odit.services/lfk/beamershow/commit/dc588e83c03d7e381f67c52b5fd5430fd0462742) - fix bg image? [`dc588e8`](https://git.odit.services/lfk/beamershow/commit/dc588e83c03d7e381f67c52b5fd5430fd0462742)
### [1.0.0](https://git.odit.services/lfk/beamershow/compare/0.1.5...1.0.0) ### [1.0.0](https://git.odit.services/lfk/beamershow/compare/0.1.5...1.0.0)

View File

@@ -1,9 +1,9 @@
FROM registry.odit.services/hub/library/node:19.8.1-alpine3.16 as build FROM registry.odit.services/hub/library/node:23.11.0-alpine3.21 as build
ARG NPM_REGISTRY_URL=https://registry.npmjs.org ARG NPM_REGISTRY_URL=https://registry.npmjs.org
WORKDIR /app WORKDIR /app
COPY . ./ COPY . ./
RUN npm config set registry $NPM_REGISTRY_URL && npm i -g pnpm@8 && pnpm i RUN npm config set registry $NPM_REGISTRY_URL && npm i -g pnpm@10.7 && pnpm i
RUN pnpm build RUN pnpm build
# final image # final image

View File

@@ -7,17 +7,17 @@ This is an API client for [https://git.odit.services/lfk/backend](@lfk/backend)
## Dev🛠 ## Dev🛠
### 🚀 Getting Started ### 🚀 Getting Started
``` ```
yarn pnpm i
``` ```
### Development ### Development
``` ```
yarn dev pnpm dev
/ /
yarn dev --open pnpm dev --open
``` ```
### Build ### Build
``` ```
yarn build pnpm build
``` ```
## Use (quickstart) 🔥 ## Use (quickstart) 🔥

View File

@@ -1,41 +1,39 @@
{ {
"name": "@lfk/beamershow", "name": "@lfk/beamershow",
"version": "1.0.1", "version": "1.0.7",
"scripts": { "scripts": {
"dev": "vite", "dev": "vite",
"build": "vite build", "build": "vite build",
"format": "prettier --write --plugin-search-dir=. ./**/*.html ./**/*.svelte", "format": "prettier --write --plugin-search-dir=. ./**/*.html ./**/*.svelte",
"prepare": "husky install",
"license:export": "license-exporter --markdown && git stage licenses.md", "license:export": "license-exporter --markdown && git stage licenses.md",
"release": "release-it --only-version" "release": "release-it --only-version"
}, },
"devDependencies": { "devDependencies": {
"@odit/license-exporter": "0.0.11", "@odit/license-exporter": "0.2.0",
"@philippdormann/release-it": "1.0.0",
"@svitejs/vite-plugin-svelte": "0.11.1", "@svitejs/vite-plugin-svelte": "0.11.1",
"@tsconfig/svelte": "1.0.10", "@tsconfig/svelte": "1.0.10",
"@types/html-minifier": "4.0.0", "@types/html-minifier": "4.0.5",
"axios": "0.21.1", "axios": "0.21.1",
"html-minifier": "4.0.0", "html-minifier": "4.0.0",
"husky": "5.1.3", "prettier": "3.5.3",
"prettier": "2.2.1", "prettier-plugin-svelte": "3.3.3",
"prettier-plugin-svelte": "2.2.0",
"release-it": "14.5.0",
"svelte": "3.36.0", "svelte": "3.36.0",
"svelte-i18n": "3.3.9", "svelte-i18n": "3.3.9",
"svelte-preprocess": "4.7.0", "svelte-preprocess": "4.7.0",
"validator": "13.5.2", "validator": "13.15.0",
"vite": "2.1.4", "vite": "2.1.4",
"vite-plugin-windicss": "1.8.10" "vite-plugin-windicss": "1.9.4"
}, },
"release-it": { "release-it": {
"git": { "git": {
"commit": true, "commit": true,
"requireCleanWorkingDir": false, "requireCleanWorkingDir": false,
"commitMessage": "🚀Bumped version to ${version}", "commitMessage": "chore: release ${version}",
"requireBranch": "main", "requireBranch": "main",
"push": false, "push": true,
"tag": true, "tag": true,
"tagName": null, "tagName": "${version}",
"tagAnnotation": "${version}" "tagAnnotation": "${version}"
}, },
"npm": { "npm": {
@@ -46,6 +44,6 @@
} }
}, },
"dependencies": { "dependencies": {
"@fontsource/athiti": "^4.5.10" "@fontsource/athiti": "5.2.5"
} }
} }

4127
pnpm-lock.yaml generated

File diff suppressed because it is too large Load Diff

3
pnpm-workspace.yaml Normal file
View File

@@ -0,0 +1,3 @@
onlyBuiltDependencies:
- esbuild
- svelte-preprocess

View File

@@ -1,417 +1,420 @@
<script> <script>
import axios from "axios"; import axios from "axios";
import bg from "../public/beamershow_background.png?import"; import bg from "/beamershow_background.png?inline";
import { _ } from "svelte-i18n"; import { _ } from "svelte-i18n";
import { slide } from "svelte/transition"; import { slide } from "svelte/transition";
import { apikey, api_endpoint, laptime_track } from "./store.js"; import { apikey, api_endpoint, laptime_track } from "./store.js";
function init(el) { $: pages = [
el.focus(); "general",
} "runners_distance",
$: pages = [ // "runners_laptime",
"general", "orgs_distance",
"runners_distance", // "teams_distance",
"runners_laptime", ];
"orgs_distance", $: current_page = "general";
// "teams_distance", $: general = {};
]; $: runners = [];
$: current_page = "general"; $: runners_by_laptime = [];
$: general = {}; $: orgs = [];
$: runners = []; $: teams = [];
$: runners_by_laptime = [];
$: orgs = [];
$: teams = [];
let time = new Date(); let time = new Date();
$: hours = (time.getHours() + "").padStart(2, "0"); $: hours = (time.getHours() + "").padStart(2, "0");
$: minutes = (time.getMinutes() + "").padStart(2, "0"); $: minutes = (time.getMinutes() + "").padStart(2, "0");
$: seconds = (time.getSeconds() + "").padStart(2, "0"); $: seconds = (time.getSeconds() + "").padStart(2, "0");
function format_laptime(laptime) { function format_laptime(laptime) {
if (laptime < 60) { if (laptime < 60) {
return `${laptime}s`; return `${laptime}s`;
} }
if (laptime < 3600) { if (laptime < 3600) {
return `${Math.floor(laptime / 60)}min ${ return `${Math.floor(laptime / 60)}min ${
laptime - Math.floor(laptime / 60) * 60 laptime - Math.floor(laptime / 60) * 60
}s`; }s`;
} }
return `${Math.floor(laptime / 3600)}h ${ return `${Math.floor(laptime / 3600)}h ${
laptime - Math.floor(laptime / 3600) * 3600 laptime - Math.floor(laptime / 3600) * 3600
}min ${ }min ${
laptime - laptime -
Math.floor(laptime / 3600) * 3600 - Math.floor(laptime / 3600) * 3600 -
Math.floor(laptime / 60) * 60 Math.floor(laptime / 60) * 60
}`; }`;
} }
function stats_general() { function stats_general() {
axios axios
.request({ .request({
method: "GET", method: "GET",
url: $api_endpoint + "api/stats/", url: $api_endpoint + "api/stats/",
headers: { Authorization: "Bearer " + $apikey }, headers: { Authorization: "Bearer " + $apikey },
}) })
.then(function ({ data }) { .then(function ({ data }) {
general = data; general = data;
}) })
.catch(function (e) { .catch(function (e) {
console.log(e); console.log(e);
}); });
} }
function stats_runners() { function stats_runners() {
axios axios
.request({ .request({
method: "GET", method: "GET",
url: $api_endpoint + "api/stats/runners/distance", url: $api_endpoint + "api/stats/runners/distance",
headers: { Authorization: "Bearer " + $apikey }, headers: { Authorization: "Bearer " + $apikey },
}) })
.then(function ({ data }) { .then(function ({ data }) {
runners = data; runners = data;
}) })
.catch(function (e) { .catch(function (e) {
console.log(e); console.log(e);
}); });
} }
function stats_runners_by_laptime() { function stats_runners_by_laptime() {
axios axios
.request({ .request({
method: "GET", method: "GET",
url: url:
$api_endpoint + "api/stats/runners/laptime?track=" + $laptime_track, $api_endpoint + "api/stats/runners/laptime?track=" + $laptime_track,
headers: { Authorization: "Bearer " + $apikey }, headers: { Authorization: "Bearer " + $apikey },
}) })
.then(function ({ data }) { .then(function ({ data }) {
runners_by_laptime = data; runners_by_laptime = data;
}) })
.catch(function (e) { .catch(function (e) {
console.log(e); console.log(e);
}); });
} }
function stats_orgs() { function stats_orgs() {
axios axios
.request({ .request({
method: "GET", method: "GET",
url: $api_endpoint + "api/stats/organizations/distance", url: $api_endpoint + "api/stats/organizations/distance",
headers: { Authorization: "Bearer " + $apikey }, headers: { Authorization: "Bearer " + $apikey },
}) })
.then(function ({ data }) { .then(function ({ data }) {
orgs = data; orgs = data;
}) })
.catch(function (e) { .catch(function (e) {
console.log(e); console.log(e);
}); });
} }
function stats_teams() { function stats_teams() {
axios axios
.request({ .request({
method: "GET", method: "GET",
url: $api_endpoint + "api/stats/teams/distance", url: $api_endpoint + "api/stats/teams/distance",
headers: { Authorization: "Bearer " + $apikey }, headers: { Authorization: "Bearer " + $apikey },
}) })
.then(function ({ data }) { .then(function ({ data }) {
teams = data; teams = data;
}) })
.catch(function (e) { .catch(function (e) {
console.log(e); console.log(e);
}); });
} }
Array.prototype.cycle = function (str) { Array.prototype.cycle = function (str) {
const i = this.indexOf(str); const i = this.indexOf(str);
if (i === -1) return undefined; if (i === -1) return undefined;
return this[(i + 1) % this.length]; return this[(i + 1) % this.length];
}; };
function fetch_all() { function fetch_all() {
stats_general(); stats_general();
stats_runners(); stats_runners();
stats_runners_by_laptime(); // stats_runners_by_laptime();
stats_orgs(); stats_orgs();
stats_teams(); stats_teams();
} }
fetch_all(); fetch_all();
setInterval(() => { setInterval(() => {
time = new Date(); time = new Date();
}, 1000); }, 1000);
setInterval(() => { setInterval(() => {
fetch_all(); fetch_all();
}, 90000); }, 50000);
setInterval(() => { setInterval(() => {
current_page = pages.cycle(current_page); current_page = pages.cycle(current_page);
}, 20000); }, 20000);
</script> </script>
<div <div
class="min-h-screen flex items-center justify-center bg-gray-100" class="min-h-screen flex items-center justify-center bg-gray-100"
style="background-image: url({bg});background-position: center;background-size: contain;background-repeat:no-repeat;" style="background-image: url({bg});background-position: center;background-size: contain;background-repeat:no-repeat;"
> >
<div class="w-full"> <div class="w-full">
<div class="w-3/4 xl:w-1/2 mx-auto"> <div class="w-3/4 xl:w-1/2 mx-auto">
<!-- --> <!-- -->
{#if current_page === "general"} {#if current_page === "general"}
<div transition:slide|local> <div transition:slide|local>
<h1 <h1
class="mr-6 text-5xl xl:text-7xl font-bold text-center text-gray-900" class="mr-6 text-5xl xl:text-7xl font-bold text-center text-gray-900"
> >
Statistiken {$_("statistiken")}
</h1> </h1>
<!-- --> <!-- -->
<div class="flex flex-wrap -mx-1 overflow-hidden mt-5"> <div class="flex flex-wrap -mx-1 overflow-hidden mt-5">
<div class="my-1 px-1 w-full overflow-hidden sm:w-1/2 md:w-1/3"> <div class="my-1 px-1 w-full overflow-hidden sm:w-1/2 md:w-1/3">
<h1 class="text-5xl font-bold text-center text-gray-900"> <h1 class="text-5xl font-bold text-center text-gray-900">
{general.total_runners || "0"} {general.total_runners || "0"}
</h1> </h1>
<h1 class="text-2xl font-semibold text-center text-gray-900"> <h1 class="text-2xl font-semibold text-center text-gray-900">
{$_("laeufer")} {$_("laeufer")}
</h1> </h1>
</div> </div>
<div class="my-1 px-1 w-full overflow-hidden sm:w-1/2 md:w-1/3"> <div class="my-1 px-1 w-full overflow-hidden sm:w-1/2 md:w-1/3">
<h1 class="text-5xl font-bold text-center text-gray-900"> <h1 class="text-5xl font-bold text-center text-gray-900">
{general.total_distance || "0"} {general.total_distance / 1000 || 0}
</h1> </h1>
<h1 class="text-2xl font-semibold text-center text-gray-900"> <h1 class="text-2xl font-semibold text-center text-gray-900">
{$_("kilometer-gesamt")} {$_("kilometer-gesamt")}
</h1> </h1>
</div> </div>
<div class="my-1 px-1 w-full overflow-hidden sm:w-1/2 md:w-1/3"> <div class="my-1 px-1 w-full overflow-hidden sm:w-1/2 md:w-1/3">
<h1 class="text-5xl font-bold text-center text-gray-900"> <h1 class="text-5xl font-bold text-center text-gray-900">
{parseFloat( {parseFloat(
((general.total_donation || 0) / 100).toFixed(2) ((general.total_donation || 0) / 100).toFixed(2)
).toLocaleString(undefined, { ).toLocaleString(undefined, {
minimumFractionDigits: 2, minimumFractionDigits: 2,
maximumFractionDigits: 2, maximumFractionDigits: 2,
}) || "0"} }) || "0"}
</h1> </h1>
<h1 class="text-2xl font-semibold text-center text-gray-900"> <h1 class="text-2xl font-semibold text-center text-gray-900">
{$_("spendensumme")} {$_("spendensumme")}
</h1> </h1>
</div> </div>
</div> </div>
</div> </div>
{:else if current_page === "runners_distance"} {:else if current_page === "runners_distance"}
<div transition:slide|local> <div transition:slide|local>
<h1 <h1
class="mr-6 text-5xl xl:text-7xl font-bold text-center text-gray-900 mb-3" class="mr-6 text-5xl xl:text-7xl font-bold text-center text-gray-900 mb-3"
> >
{$_("top-laeufer")} ({$_("distanz")}) {$_("top-laeufer")} ({$_("distanz")})
</h1> </h1>
{#if runners_by_laptime.length === 0} {#if runners.length === 0}
<p class="w-full text-center text-3xl font-semibold"> <p class="w-full text-center text-3xl font-semibold">
Noch keine Daten... Noch keine Daten...
</p> </p>
{:else} {:else}
<table <table
class="table font-semibold p-4 bg-white shadow rounded-lg w-full" class="table font-semibold p-4 bg-white shadow rounded-lg w-full"
> >
<thead> <thead>
<tr> <tr>
<th <th
class="border p-4 dark:border-dark-5 whitespace-nowrap font-normal text-gray-900" class="border p-4 dark:border-dark-5 whitespace-nowrap font-normal text-gray-900"
> >
{$_("platz")} {$_("platz")}
</th> </th>
<th <th
class="border p-4 dark:border-dark-5 whitespace-nowrap font-normal text-gray-900" class="border p-4 dark:border-dark-5 whitespace-nowrap font-normal text-gray-900"
> >
{$_("laeufer")} {$_("laeufer")}
</th> </th>
<th <th
class="border p-4 dark:border-dark-5 whitespace-nowrap font-normal text-gray-900" class="border p-4 dark:border-dark-5 whitespace-nowrap font-normal text-gray-900"
> >
{$_("organisation")} {$_("organisation")}
</th> </th>
<th <th
class="border p-4 dark:border-dark-5 whitespace-nowrap font-normal text-gray-900" class="border p-4 dark:border-dark-5 whitespace-nowrap font-normal text-gray-900"
> >
{$_("kilometer")} {$_("kilometer")}
</th> </th>
</tr> </tr>
</thead> </thead>
<tbody> <tbody>
{#each runners as r, i} {#each runners as r, i}
<tr class="text-gray-700"> <tr class="text-gray-700">
<td class="border p-4 dark:border-dark-5"> <td class="border p-4 dark:border-dark-5">
{i + 1} {i + 1}
</td> </td>
<td class="border p-4 dark:border-dark-5"> <td class="border p-4 dark:border-dark-5">
{r.firstname} {r.firstname}
{r.lastname} {r.lastname}
</td> </td>
<td class="border p-4 dark:border-dark-5"> <td class="border p-4 dark:border-dark-5">
{r.group.name} {r.group.name}
</td> </td>
<td class="border p-4 dark:border-dark-5"> <td class="border p-4 dark:border-dark-5">
{r.distance / 1000} km {r.distance / 1000} km
</td> </td>
</tr> </tr>
{/each} {/each}
</tbody> </tbody>
</table> </table>
{/if} {/if}
</div> </div>
{:else if current_page === "runners_laptime"} {:else if current_page === "runners_laptime"}
<div transition:slide|local> <div transition:slide|local>
<h1 <h1
class="mr-6 text-5xl xl:text-7xl font-bold text-center text-gray-900 mb-3" class="mr-6 text-5xl xl:text-7xl font-bold text-center text-gray-900 mb-3"
> >
{$_("top-laeufer")} ({$_("rundenzeit")}) {$_("top-laeufer")} ({$_("rundenzeit")})
</h1> </h1>
{#if runners_by_laptime.length === 0} {#if runners_by_laptime.length === 0}
<p class="w-full text-center text-3xl font-semibold"> <p class="w-full text-center text-3xl font-semibold">
Noch keine Daten... Noch keine Daten...
</p> </p>
{:else} {:else}
<table <table
class="table font-semibold p-4 bg-white shadow rounded-lg w-full" class="table font-semibold p-4 bg-white shadow rounded-lg w-full"
> >
<thead> <thead>
<tr> <tr>
<th <th
class="border p-4 dark:border-dark-5 whitespace-nowrap font-normal text-gray-900" class="border p-4 dark:border-dark-5 whitespace-nowrap font-normal text-gray-900"
> >
{$_("platz")} {$_("platz")}
</th> </th>
<th <th
class="border p-4 dark:border-dark-5 whitespace-nowrap font-normal text-gray-900" class="border p-4 dark:border-dark-5 whitespace-nowrap font-normal text-gray-900"
> >
{$_("laeufer")} {$_("laeufer")}
</th> </th>
<th <th
class="border p-4 dark:border-dark-5 whitespace-nowrap font-normal text-gray-900" class="border p-4 dark:border-dark-5 whitespace-nowrap font-normal text-gray-900"
> >
{$_("organisation")} {$_("organisation")}
</th> </th>
<th <th
class="border p-4 dark:border-dark-5 whitespace-nowrap font-normal text-gray-900" class="border p-4 dark:border-dark-5 whitespace-nowrap font-normal text-gray-900"
> >
{$_("schnellste-rundenzeit")} {$_("schnellste-rundenzeit")}
</th> </th>
</tr> </tr>
</thead> </thead>
<tbody> <tbody>
{#each runners_by_laptime as r, i} {#each runners_by_laptime as r, i}
<tr class="text-gray-700"> <tr class="text-gray-700">
<td class="border p-4 dark:border-dark-5"> <td class="border p-4 dark:border-dark-5">
{i + 1} {i + 1}
</td> </td>
<td class="border p-4 dark:border-dark-5"> <td class="border p-4 dark:border-dark-5">
{r.firstname} {r.firstname}
{r.lastname} {r.lastname}
</td> </td>
<td class="border p-4 dark:border-dark-5"> <td class="border p-4 dark:border-dark-5">
{r.group.name} {r.group.name}
</td> </td>
<td class="border p-4 dark:border-dark-5"> <td class="border p-4 dark:border-dark-5">
{format_laptime(r.minLaptime)} {format_laptime(r.minLaptime)}
</td> </td>
</tr> </tr>
{/each} {/each}
</tbody> </tbody>
</table> </table>
{/if} {/if}
</div> </div>
{:else if current_page === "orgs_distance"} {:else if current_page === "orgs_distance"}
<div transition:slide|local> <div transition:slide|local>
<h1 <h1
class="mr-6 text-5xl xl:text-7xl font-bold text-center text-gray-900 mb-3" class="mr-6 text-5xl xl:text-7xl font-bold text-center text-gray-900 mb-3"
> >
{$_("top-organisationen")} {$_("top-organisationen")}
</h1> </h1>
<table {#if orgs.length === 0}
class="table font-semibold p-4 bg-white shadow rounded-lg w-full" <p class="w-full text-center text-3xl font-semibold">
> Noch keine Daten...
<thead> </p>
<tr> {:else}
<th <table
class="border p-4 dark:border-dark-5 whitespace-nowrap font-normal text-gray-900" class="table font-semibold p-4 bg-white shadow rounded-lg w-full"
> >
{$_("platz")} <thead>
</th> <tr>
<th <th
class="border p-4 dark:border-dark-5 whitespace-nowrap font-normal text-gray-900" class="border p-4 dark:border-dark-5 whitespace-nowrap font-normal text-gray-900"
> >
{$_("organsiation")} {$_("platz")}
</th> </th>
<th <th
class="border p-4 dark:border-dark-5 whitespace-nowrap font-normal text-gray-900" class="border p-4 dark:border-dark-5 whitespace-nowrap font-normal text-gray-900"
> >
{$_("kilometer")} {$_("organsiation")}
</th> </th>
</tr> <th
</thead> class="border p-4 dark:border-dark-5 whitespace-nowrap font-normal text-gray-900"
<tbody> >
{#each orgs as o, i} {$_("kilometer")}
<tr class="text-gray-700"> </th>
<td class="border p-4 dark:border-dark-5"> </tr>
{i + 1} </thead>
</td> <tbody>
<td class="border p-4 dark:border-dark-5"> {#each orgs as o, i}
{o.name} <tr class="text-gray-700">
</td> <td class="border p-4 dark:border-dark-5">
<td class="border p-4 dark:border-dark-5"> {i + 1}
{o.distance / 1000} km </td>
</td> <td class="border p-4 dark:border-dark-5">
</tr> {o.name}
{/each} </td>
</tbody> <td class="border p-4 dark:border-dark-5">
</table> {o.distance / 1000} km
</div> </td>
{:else if current_page === "teams_distance"} </tr>
<div transition:slide|local> {/each}
<h1 </tbody>
class="mr-6 text-5xl xl:text-7xl font-bold text-center text-gray-900 mb-3" </table>
> {/if}
{$_("top-teams")} </div>
</h1> {:else if current_page === "teams_distance"}
<table <div transition:slide|local>
class="table font-semibold p-4 bg-white shadow rounded-lg w-full" <h1
> class="mr-6 text-5xl xl:text-7xl font-bold text-center text-gray-900 mb-3"
<thead> >
<tr> {$_("top-teams")}
<th </h1>
class="border p-4 dark:border-dark-5 whitespace-nowrap font-normal text-gray-900" <table
> class="table font-semibold p-4 bg-white shadow rounded-lg w-full"
{$_("platz")} >
</th> <thead>
<th <tr>
class="border p-4 dark:border-dark-5 whitespace-nowrap font-normal text-gray-900" <th
> class="border p-4 dark:border-dark-5 whitespace-nowrap font-normal text-gray-900"
{$_("team")} >
</th> {$_("platz")}
<th </th>
class="border p-4 dark:border-dark-5 whitespace-nowrap font-normal text-gray-900" <th
> class="border p-4 dark:border-dark-5 whitespace-nowrap font-normal text-gray-900"
{$_("kilometer")} >
</th> {$_("team")}
</tr> </th>
</thead> <th
<tbody> class="border p-4 dark:border-dark-5 whitespace-nowrap font-normal text-gray-900"
{#each teams as t, i} >
<tr class="text-gray-700"> {$_("kilometer")}
<td class="border p-4 dark:border-dark-5"> </th>
{i + 1} </tr>
</td> </thead>
<td class="border p-4 dark:border-dark-5"> <tbody>
{t.name} {#each teams as t, i}
</td> <tr class="text-gray-700">
<td class="border p-4 dark:border-dark-5"> <td class="border p-4 dark:border-dark-5">
{t.distance / 1000} km {i + 1}
</td> </td>
</tr> <td class="border p-4 dark:border-dark-5">
{/each} {t.parent.name}<br />
</tbody> {t.name}
</table> </td>
</div> <td class="border p-4 dark:border-dark-5">
{:else} {t.distance / 1000} km
<!-- content here --> </td>
{/if} </tr>
</div> {/each}
</div> </tbody>
<h1 </table>
class="text-6xl font-semibold text-right text-gray-900 font-mono top-2 w-full fixed pr-4 xl:top-6 xl:pr-8" </div>
> {:else}
{hours}:{minutes}:{seconds} <!-- content here -->
</h1> {/if}
<h1 </div>
class="text-xl xl:text-3xl font-medium text-center text-gray-900 font-mono bottom-2 xl:bottom-4 w-full fixed" </div>
> <h1
<span class="text-black font-extrabold">LfK!2023 </span>powered by class="text-6xl font-semibold text-right text-gray-900 font-mono top-2 w-full fixed pr-4 xl:top-6 xl:pr-8"
ODIT.Services >
</h1> {hours}:{minutes}:{seconds}
</h1>
<h1
class="text-xl xl:text-3xl font-medium text-center text-gray-900 font-mono bottom-2 xl:bottom-4 w-full fixed"
>
<span class="text-black font-extrabold">Lauf für Kaya!</span> - powered by ODIT.Services
</h1>
</div> </div>

View File

@@ -1,4 +1,5 @@
<script> <script>
import bg from "/beamershow_background.png?inline";
import { apikey, lang, api_endpoint, laptime_track, clear } from "./store.js"; import { apikey, lang, api_endpoint, laptime_track, clear } from "./store.js";
import isURL from "validator/lib/isURL"; import isURL from "validator/lib/isURL";
import isUUID from "validator/lib/isUUID"; import isUUID from "validator/lib/isUUID";
@@ -253,7 +254,7 @@
<img <img
alt="" alt=""
class="object-cover w-full h-screen hidden md:block" class="object-cover w-full h-screen hidden md:block"
src="https://source.unsplash.com/IXUM4cJynP0" src={bg}
/> />
</div> </div>
</div> </div>

View File

@@ -30,6 +30,7 @@
"spendensumme": "Spendensumme", "spendensumme": "Spendensumme",
"station_description": "Beschreibung der Scanstation", "station_description": "Beschreibung der Scanstation",
"station_id": "Scanstations-ID", "station_id": "Scanstations-ID",
"statistiken": "Statistiken",
"team": "Team", "team": "Team",
"the_provided_scan_station_is_disabled": "Die angegebene Scanstation ist deaktiviert.", "the_provided_scan_station_is_disabled": "Die angegebene Scanstation ist deaktiviert.",
"the_provided_scan_station_token_is_invalid": "Der angegebene Scanstation-Token ist ungültig.", "the_provided_scan_station_token_is_invalid": "Der angegebene Scanstation-Token ist ungültig.",

View File

@@ -30,6 +30,7 @@
"spendensumme": "Donations", "spendensumme": "Donations",
"station_description": "Station Description", "station_description": "Station Description",
"station_id": "Scanstation ID", "station_id": "Scanstation ID",
"statistiken": "Statistics",
"team": "Team", "team": "Team",
"the_provided_scan_station_is_disabled": "The provided scan station is disabled.", "the_provided_scan_station_is_disabled": "The provided scan station is disabled.",
"the_provided_scan_station_token_is_invalid": "The provided scan station token is invalid.", "the_provided_scan_station_token_is_invalid": "The provided scan station token is invalid.",