rebuild websites with svelte
still early WIP Signed-off-by: Peter Siegmund <developer@mars3142.org>
This commit is contained in:
12
firmware/website/.run/dev.run.xml
Normal file
12
firmware/website/.run/dev.run.xml
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
<component name="ProjectRunConfigurationManager">
|
||||||
|
<configuration default="false" name="dev" type="js.build_tools.npm" nameIsGenerated="true">
|
||||||
|
<package-json value="$PROJECT_DIR$/package.json" />
|
||||||
|
<command value="run" />
|
||||||
|
<scripts>
|
||||||
|
<script value="dev" />
|
||||||
|
</scripts>
|
||||||
|
<node-interpreter value="project" />
|
||||||
|
<envs />
|
||||||
|
<method v="2" />
|
||||||
|
</configuration>
|
||||||
|
</component>
|
||||||
@@ -8,7 +8,7 @@
|
|||||||
<meta name="apple-mobile-web-app-capable" content="yes">
|
<meta name="apple-mobile-web-app-capable" content="yes">
|
||||||
<meta name="apple-mobile-web-app-status-bar-style" content="black-translucent">
|
<meta name="apple-mobile-web-app-status-bar-style" content="black-translucent">
|
||||||
<link rel="icon" type="image/svg+xml" href="favicon.svg">
|
<link rel="icon" type="image/svg+xml" href="favicon.svg">
|
||||||
<title>🚂 System Control</title>
|
<title>System Control</title>
|
||||||
</head>
|
</head>
|
||||||
|
|
||||||
<body>
|
<body>
|
||||||
|
|||||||
1560
firmware/website/package-lock.json
generated
Normal file
1560
firmware/website/package-lock.json
generated
Normal file
File diff suppressed because it is too large
Load Diff
20
firmware/website/package.json
Normal file
20
firmware/website/package.json
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
{
|
||||||
|
"name": "learn-svelte",
|
||||||
|
"type": "module",
|
||||||
|
"scripts": {
|
||||||
|
"dev": "vite",
|
||||||
|
"build": "vite build",
|
||||||
|
"preview": "vite preview"
|
||||||
|
},
|
||||||
|
"devDependencies": {
|
||||||
|
"@sveltejs/vite-plugin-svelte": "^6.1.1",
|
||||||
|
"svelte": "^5.38.1",
|
||||||
|
"vite": "^7.1.2",
|
||||||
|
"vite-plugin-compression": "^0.5.1"
|
||||||
|
},
|
||||||
|
"dependencies": {
|
||||||
|
"@fontsource/atkinson-hyperlegible": "^5.2.6",
|
||||||
|
"@picocss/pico": "^2.1.1",
|
||||||
|
"gsap": "^3.13.0"
|
||||||
|
}
|
||||||
|
}
|
||||||
27
firmware/website/src/App.svelte
Normal file
27
firmware/website/src/App.svelte
Normal file
@@ -0,0 +1,27 @@
|
|||||||
|
<script lang="ts">
|
||||||
|
import Header from "./compoents/Header.svelte";
|
||||||
|
import Index from "./Index.svelte";
|
||||||
|
import Captive from "./Captive.svelte";
|
||||||
|
import { onMount } from "svelte";
|
||||||
|
import { writable } from "svelte/store";
|
||||||
|
|
||||||
|
const isCaptive = writable(false);
|
||||||
|
|
||||||
|
function checkHash() {
|
||||||
|
isCaptive.set(window.location.hash === "#/captive");
|
||||||
|
}
|
||||||
|
|
||||||
|
onMount(() => {
|
||||||
|
checkHash();
|
||||||
|
window.addEventListener("hashchange", checkHash);
|
||||||
|
return () => window.removeEventListener("hashchange", checkHash);
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<Header />
|
||||||
|
|
||||||
|
{#if $isCaptive}
|
||||||
|
<Captive />
|
||||||
|
{:else}
|
||||||
|
<Index />
|
||||||
|
{/if}
|
||||||
5
firmware/website/src/Captive.svelte
Normal file
5
firmware/website/src/Captive.svelte
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
<script lang="ts">
|
||||||
|
import { t } from "./i18n/store";
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<h1>{$t("welcome")} - Captive Portal</h1>
|
||||||
5
firmware/website/src/Index.svelte
Normal file
5
firmware/website/src/Index.svelte
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
<script lang="ts">
|
||||||
|
import { t } from "./i18n/store";
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<h1>{$t("welcome")}</h1>
|
||||||
70
firmware/website/src/app.css
Normal file
70
firmware/website/src/app.css
Normal file
@@ -0,0 +1,70 @@
|
|||||||
|
:root {
|
||||||
|
--bg-color: #1a1a2e;
|
||||||
|
--card-bg: #16213e;
|
||||||
|
--accent: #0f3460;
|
||||||
|
--text: #eaeaea;
|
||||||
|
--text-muted: #a0a0a0;
|
||||||
|
--success: #00d26a;
|
||||||
|
--error: #ff6b6b;
|
||||||
|
--border: #2a2a4a;
|
||||||
|
--input-bg: #1a1a2e;
|
||||||
|
--shadow: rgba(0, 0, 0, 0.3);
|
||||||
|
--primary: #c41e3a;
|
||||||
|
}
|
||||||
|
|
||||||
|
[data-theme="light"] {
|
||||||
|
--bg-color: #faf8f5;
|
||||||
|
--card-bg: #ffffff;
|
||||||
|
--accent: #fef2f2;
|
||||||
|
--text: #1a1a2e;
|
||||||
|
--text-muted: #6b7280;
|
||||||
|
--success: #c41e3a;
|
||||||
|
--error: #dc2626;
|
||||||
|
--border: #e5d9d0;
|
||||||
|
--input-bg: #ffffff;
|
||||||
|
--shadow: rgba(196, 30, 58, 0.1);
|
||||||
|
--primary: #c41e3a;
|
||||||
|
}
|
||||||
|
|
||||||
|
[data-theme="light"] .header h1 {
|
||||||
|
color: var(--primary);
|
||||||
|
}
|
||||||
|
|
||||||
|
* {
|
||||||
|
margin: 0;
|
||||||
|
padding: 0;
|
||||||
|
box-sizing: border-box;
|
||||||
|
}
|
||||||
|
|
||||||
|
body {
|
||||||
|
font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif;
|
||||||
|
background: var(--bg-color);
|
||||||
|
color: var(--text);
|
||||||
|
min-height: 100vh;
|
||||||
|
transition: background 0.3s, color 0.3s;
|
||||||
|
padding: 12px;
|
||||||
|
}
|
||||||
|
|
||||||
|
h1 {
|
||||||
|
font-size: 1.5rem;
|
||||||
|
color: var(--text);
|
||||||
|
}
|
||||||
|
|
||||||
|
#app {
|
||||||
|
max-width: 900px;
|
||||||
|
margin: 0 auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (max-width: 600px) {
|
||||||
|
body {
|
||||||
|
padding: 6px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@supports (padding: max(0px)) {
|
||||||
|
body {
|
||||||
|
padding-left: max(12px, env(safe-area-inset-left));
|
||||||
|
padding-right: max(12px, env(safe-area-inset-right));
|
||||||
|
padding-bottom: max(12px, env(safe-area-inset-bottom));
|
||||||
|
}
|
||||||
|
}
|
||||||
133
firmware/website/src/compoents/Header.svelte
Normal file
133
firmware/website/src/compoents/Header.svelte
Normal file
@@ -0,0 +1,133 @@
|
|||||||
|
<script lang="ts">
|
||||||
|
import Toggle from "./Toggle.svelte";
|
||||||
|
import {toggleTheme} from "../theme";
|
||||||
|
import {onMount} from "svelte";
|
||||||
|
import {writable} from "svelte/store";
|
||||||
|
import {lang, t} from "../i18n/store";
|
||||||
|
|
||||||
|
const theme = writable<"dark" | "light">("dark");
|
||||||
|
|
||||||
|
function applyInitialTheme() {
|
||||||
|
const userTheme = localStorage.getItem("theme");
|
||||||
|
if (userTheme) {
|
||||||
|
document.documentElement.setAttribute("data-theme", userTheme);
|
||||||
|
} else if (window.matchMedia("(prefers-color-scheme: light)").matches) {
|
||||||
|
document.documentElement.setAttribute("data-theme", "light");
|
||||||
|
} else {
|
||||||
|
document.documentElement.setAttribute("data-theme", "dark");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function updateThemeFromDom() {
|
||||||
|
const t = document.documentElement.getAttribute("data-theme");
|
||||||
|
theme.set(t === "light" ? "light" : "dark");
|
||||||
|
}
|
||||||
|
|
||||||
|
function handleThemeToggle() {
|
||||||
|
toggleTheme();
|
||||||
|
updateThemeFromDom();
|
||||||
|
}
|
||||||
|
|
||||||
|
let themeIcon = $state("🌙");
|
||||||
|
let themeLabel = $state("Dark");
|
||||||
|
let currentLangCode = $state($lang);
|
||||||
|
let currentLang = $state("Deutsch");
|
||||||
|
let currentFlag = $state("🇩🇪");
|
||||||
|
|
||||||
|
$effect(() => {
|
||||||
|
theme.subscribe(($theme) => {
|
||||||
|
themeIcon = $theme === "light" ? "☀️" : "🌙";
|
||||||
|
themeLabel = $theme === "light" ? "Light" : "Dark";
|
||||||
|
});
|
||||||
|
lang.subscribe(($lang) => {
|
||||||
|
currentLangCode = $lang;
|
||||||
|
currentLang = $lang === "de" ? "Deutsch" : "English";
|
||||||
|
currentFlag = $lang === "de" ? "🇩🇪" : "🇬🇧";
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
function handleLangChange(newLang: "de" | "en") {
|
||||||
|
lang.set(newLang);
|
||||||
|
|
||||||
|
localStorage.setItem("lang", newLang);
|
||||||
|
}
|
||||||
|
|
||||||
|
onMount(() => {
|
||||||
|
applyInitialTheme();
|
||||||
|
updateThemeFromDom();
|
||||||
|
window.addEventListener("storage", updateThemeFromDom);
|
||||||
|
|
||||||
|
// Listener für OS-Theme-Änderung
|
||||||
|
const mql = window.matchMedia("(prefers-color-scheme: light)");
|
||||||
|
const osThemeListener = () => {
|
||||||
|
// Nur reagieren, wenn kein User-Theme gesetzt ist
|
||||||
|
if (!localStorage.getItem("theme")) {
|
||||||
|
applyInitialTheme();
|
||||||
|
updateThemeFromDom();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
mql.addEventListener("change", osThemeListener);
|
||||||
|
|
||||||
|
return () => {
|
||||||
|
window.removeEventListener("storage", updateThemeFromDom);
|
||||||
|
mql.removeEventListener("change", osThemeListener);
|
||||||
|
};
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<div class="header">
|
||||||
|
<div class="header-controls">
|
||||||
|
<Toggle
|
||||||
|
label={currentLang}
|
||||||
|
icon={currentFlag}
|
||||||
|
ariaLabel="Sprache wechseln"
|
||||||
|
onClick={() => {
|
||||||
|
const newLang = currentLangCode === "de" ? "en" : "de";
|
||||||
|
handleLangChange(newLang);
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
|
||||||
|
<Toggle
|
||||||
|
label={themeLabel}
|
||||||
|
icon={themeIcon}
|
||||||
|
ariaLabel="Theme wechseln"
|
||||||
|
onClick={handleThemeToggle}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<h1>🚂 System Control</h1>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
.header {
|
||||||
|
display: flex;
|
||||||
|
justify-content: space-between;
|
||||||
|
align-items: center;
|
||||||
|
margin-bottom: 20px;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
gap: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.header-controls {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (max-width: 600px) {
|
||||||
|
.header {
|
||||||
|
flex-direction: column;
|
||||||
|
align-items: stretch;
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.header h1 {
|
||||||
|
order: 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (max-width: 380px) {
|
||||||
|
.header h1 {
|
||||||
|
font-size: 1.2rem;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
||||||
37
firmware/website/src/compoents/Toggle.svelte
Normal file
37
firmware/website/src/compoents/Toggle.svelte
Normal file
@@ -0,0 +1,37 @@
|
|||||||
|
<script lang="ts">
|
||||||
|
const { label, icon, ariaLabel, onClick } = $props<{
|
||||||
|
label: string;
|
||||||
|
icon: string;
|
||||||
|
ariaLabel: string;
|
||||||
|
onClick?: () => void;
|
||||||
|
}>();
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<button class="toggle" aria-label={ariaLabel} onclick={onClick}>
|
||||||
|
<span class="icon" id="icon">{icon}</span>
|
||||||
|
<span class="label" id="label">{label}</span>
|
||||||
|
</button>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
.toggle {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 6px;
|
||||||
|
background: var(--card-bg);
|
||||||
|
padding: 8px 12px;
|
||||||
|
border-radius: 20px;
|
||||||
|
border: 1px solid var(--border);
|
||||||
|
cursor: pointer;
|
||||||
|
transition: all 0.2s;
|
||||||
|
font-size: 0.85rem;
|
||||||
|
color: var(--text-muted);
|
||||||
|
}
|
||||||
|
|
||||||
|
.toggle:hover {
|
||||||
|
border-color: var(--success);
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon {
|
||||||
|
font-size: 1.2rem;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
6
firmware/website/src/i18n/de.json
Normal file
6
firmware/website/src/i18n/de.json
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
{
|
||||||
|
"hello": "Hallo Welt",
|
||||||
|
"welcome": "Willkommen",
|
||||||
|
"language": "Sprache",
|
||||||
|
"save": "Speichern"
|
||||||
|
}
|
||||||
6
firmware/website/src/i18n/en.json
Normal file
6
firmware/website/src/i18n/en.json
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
{
|
||||||
|
"hello": "Hello World",
|
||||||
|
"welcome": "Welcome",
|
||||||
|
"language": "Language",
|
||||||
|
"save": "Save"
|
||||||
|
}
|
||||||
12
firmware/website/src/i18n/index.ts
Normal file
12
firmware/website/src/i18n/index.ts
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
import de from './de.json';
|
||||||
|
import en from './en.json';
|
||||||
|
|
||||||
|
export const translations = { de, en };
|
||||||
|
|
||||||
|
export type Lang = keyof typeof translations;
|
||||||
|
|
||||||
|
export function getInitialLang(): Lang {
|
||||||
|
const navLang = navigator.language.slice(0, 2);
|
||||||
|
if (navLang in translations) return navLang as Lang;
|
||||||
|
return 'en';
|
||||||
|
}
|
||||||
16
firmware/website/src/i18n/store.ts
Normal file
16
firmware/website/src/i18n/store.ts
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
import { writable, derived } from 'svelte/store';
|
||||||
|
import { translations, getInitialLang, type Lang } from './index';
|
||||||
|
|
||||||
|
function getLang(): Lang {
|
||||||
|
const stored = localStorage.getItem('lang');
|
||||||
|
if (stored && stored in translations) return stored as Lang;
|
||||||
|
return getInitialLang();
|
||||||
|
}
|
||||||
|
|
||||||
|
export const lang = writable<Lang>(getLang());
|
||||||
|
|
||||||
|
export const t = derived(lang, $lang => {
|
||||||
|
return (key: string) => {
|
||||||
|
return translations[$lang][key] || key;
|
||||||
|
};
|
||||||
|
});
|
||||||
9
firmware/website/src/main.js
Normal file
9
firmware/website/src/main.js
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
import { mount } from 'svelte'
|
||||||
|
import App from './App.svelte'
|
||||||
|
import './app.css'
|
||||||
|
|
||||||
|
const app = mount(App, {
|
||||||
|
target: document.getElementById('app'),
|
||||||
|
})
|
||||||
|
|
||||||
|
export default app
|
||||||
18
firmware/website/src/theme.ts
Normal file
18
firmware/website/src/theme.ts
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
// src/theme.ts
|
||||||
|
export function setTheme(theme: 'light' | 'dark') {
|
||||||
|
document.documentElement.setAttribute('data-theme', theme);
|
||||||
|
localStorage.setItem('theme', theme);
|
||||||
|
|
||||||
|
const icon = document.getElementById('theme-icon');
|
||||||
|
const label = document.getElementById('theme-label');
|
||||||
|
const metaTheme = document.querySelector('meta[name="theme-color"]') as HTMLMetaElement | null;
|
||||||
|
|
||||||
|
if (icon) icon.textContent = theme === 'light' ? '☀️' : '🌙';
|
||||||
|
if (label) label.textContent = theme === 'light' ? 'Light' : 'Dark';
|
||||||
|
if (metaTheme) metaTheme.content = theme === 'light' ? '#f0f2f5' : '#1a1a2e';
|
||||||
|
}
|
||||||
|
|
||||||
|
export function toggleTheme() {
|
||||||
|
const current = document.documentElement.getAttribute('data-theme') as 'light' | 'dark' | null || 'dark';
|
||||||
|
setTheme(current === 'dark' ? 'light' : 'dark');
|
||||||
|
}
|
||||||
2
firmware/website/src/vite-env.d.ts
vendored
Normal file
2
firmware/website/src/vite-env.d.ts
vendored
Normal file
@@ -0,0 +1,2 @@
|
|||||||
|
/// <reference types="svelte" />
|
||||||
|
/// <reference types="vite/client" />
|
||||||
17
firmware/website/svelte.config.js
Normal file
17
firmware/website/svelte.config.js
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
import { vitePreprocess } from '@sveltejs/vite-plugin-svelte'
|
||||||
|
|
||||||
|
/** @type {import("@sveltejs/vite-plugin-svelte").SvelteConfig} */
|
||||||
|
export default {
|
||||||
|
// Consult https://svelte.dev/docs#compile-time-svelte-preprocess
|
||||||
|
// for more information about preprocessors
|
||||||
|
preprocess: vitePreprocess(),
|
||||||
|
compilerOptions: {
|
||||||
|
runes: true,
|
||||||
|
},
|
||||||
|
vitePlugin: {
|
||||||
|
inspector: {
|
||||||
|
showToggleButton: 'always',
|
||||||
|
toggleButtonPos: 'bottom-right'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user