All checks were successful
Deploy baustelle-pwa / deploy (push) Successful in 1s
5.4 Mess-Werkzeug mit Skala-Kalibrierung: - Zwei neue Sketch-Tools: 📏⚙ Kalibrieren + 📏 Messen - Kalibrieren: 2 Punkte ziehen → reale Länge eingeben (cm/m/mm) - Messen: 2 Punkte ziehen → Live-Distanz-Label mit berechnetem Wert - pxPerUnit + unit werden im State gehalten, Linie im Canvas gezeichnet 5.9 Materialliste pro Auftrag: - 📦 Materialliste-Button im Auftrags-Detail - Modal: Bezeichnung + Menge + Einheit (Stk/m/m²/kg/l/Set/Pa/h) + Notiz - Live-Liste mit Löschen pro Eintrag 5.8 Tages-Baustellen-Map: - Neuer Bottom-Nav-Tab ☀️ Heute (ganz links) - Filtert Aufträge nach heutigem Datum - Route-Button öffnet Google Maps mit allen heutigen Adressen als Waypoints - Darunter noch die offenen Aufträge als Backup 4.d Benachrichtigungen via ntfy (statt VAPID): - Settings → Benachrichtigungen: ntfy-Server + Topic - EventSource auf /topic/sse → native Browser-Notifications - Test-Nachricht-Button sendet POST an das Topic - Auto-Reconnect bei Fehlern nach 10s - Auto-Start beim Boot wenn Topic gesetzt und Permission granted Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com> [deploy]
130 lines
5.3 KiB
PHP
130 lines
5.3 KiB
PHP
<?php
|
|
/* Baustelle PWA — Einstiegsseite.
|
|
* Cache-Versionierung via filemtime() nach Pattern claude-db #201.
|
|
* Bei jedem Deploy ändern sich die mtimes → neue URLs → Browser holt frische Files.
|
|
*/
|
|
$cssVersion = @filemtime(__DIR__.'/app.css') ?: time();
|
|
$jsVersion = max(
|
|
@filemtime(__DIR__.'/app.js') ?: time(),
|
|
@filemtime(__DIR__.'/lib/idb.js') ?: time(),
|
|
@filemtime(__DIR__.'/lib/api.js') ?: time(),
|
|
@filemtime(__DIR__.'/lib/offline.js')?: time(),
|
|
@filemtime(__DIR__.'/lib/router.js') ?: time()
|
|
);
|
|
$manifestVersion = @filemtime(__DIR__.'/manifest.webmanifest') ?: time();
|
|
$swVersion = max($cssVersion, $jsVersion, $manifestVersion);
|
|
|
|
// Einstiegsseite NIE cachen
|
|
header('Cache-Control: no-cache, no-store, must-revalidate');
|
|
header('Pragma: no-cache');
|
|
header('Expires: 0');
|
|
?>
|
|
<!doctype html>
|
|
<html lang="de">
|
|
<head>
|
|
<meta charset="utf-8">
|
|
<meta name="viewport" content="width=device-width,initial-scale=1,maximum-scale=1,user-scalable=no">
|
|
<meta name="theme-color" content="#1a1a1f">
|
|
<meta name="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">
|
|
<title>Baustelle</title>
|
|
<link rel="manifest" href="manifest.webmanifest?v=<?php echo $manifestVersion; ?>">
|
|
<link rel="icon" type="image/svg+xml" href="icons/icon.svg">
|
|
<link rel="icon" type="image/png" sizes="192x192" href="icons/icon-192.png">
|
|
<link rel="icon" type="image/png" sizes="512x512" href="icons/icon-512.png">
|
|
<link rel="apple-touch-icon" href="icons/icon-192.png">
|
|
<link rel="stylesheet" href="app.css?v=<?php echo $cssVersion; ?>">
|
|
</head>
|
|
<body>
|
|
|
|
<div id="app">
|
|
<header id="topbar">
|
|
<button id="back-btn" class="icon-btn" style="display:none">←</button>
|
|
<h1 id="page-title">Baustelle</h1>
|
|
<button id="install-btn" class="icon-btn" title="Als App installieren" style="display:none">📲</button>
|
|
<button id="help-btn" class="icon-btn" title="Hilfe">❓</button>
|
|
<span id="status-badge">🟢</span>
|
|
</header>
|
|
|
|
<main id="main"></main>
|
|
|
|
<nav id="bottom-nav" style="display:none">
|
|
<button data-route="today">☀️ Heute</button>
|
|
<button data-route="orders" class="active">📋 Aufträge</button>
|
|
<button data-route="customers">👥 Kunden</button>
|
|
<button data-route="reports">📑 Berichte</button>
|
|
<button data-route="settings">⚙️</button>
|
|
</nav>
|
|
</div>
|
|
|
|
<div id="toast-container"></div>
|
|
|
|
<script src="lib/idb.js?v=<?php echo $jsVersion; ?>"></script>
|
|
<script src="lib/api.js?v=<?php echo $jsVersion; ?>"></script>
|
|
<script src="lib/offline.js?v=<?php echo $jsVersion; ?>"></script>
|
|
<script src="lib/router.js?v=<?php echo $jsVersion; ?>"></script>
|
|
<script src="app.js?v=<?php echo $jsVersion; ?>"></script>
|
|
<script>
|
|
/* PWA Install Prompt abfangen */
|
|
let deferredInstallPrompt = null;
|
|
window.addEventListener('beforeinstallprompt', function (e) {
|
|
e.preventDefault();
|
|
deferredInstallPrompt = e;
|
|
var btn = document.getElementById('install-btn');
|
|
if (btn) btn.style.display = '';
|
|
console.log('PWA: Install-Prompt ist verfügbar');
|
|
});
|
|
document.addEventListener('DOMContentLoaded', function () {
|
|
var btn = document.getElementById('install-btn');
|
|
if (btn) btn.addEventListener('click', async function () {
|
|
if (!deferredInstallPrompt) {
|
|
alert('Installation nicht verfügbar. Falls du die App bereits installiert hast, öffne sie über das Home-Screen-Icon.\n\nAuf iPhone: Safari → Teilen → "Zum Home-Bildschirm".');
|
|
return;
|
|
}
|
|
deferredInstallPrompt.prompt();
|
|
var choice = await deferredInstallPrompt.userChoice;
|
|
if (choice.outcome === 'accepted') {
|
|
btn.style.display = 'none';
|
|
}
|
|
deferredInstallPrompt = null;
|
|
});
|
|
});
|
|
window.addEventListener('appinstalled', function () {
|
|
var btn = document.getElementById('install-btn');
|
|
if (btn) btn.style.display = 'none';
|
|
console.log('PWA: Installation abgeschlossen');
|
|
});
|
|
|
|
if ('serviceWorker' in navigator) {
|
|
window.addEventListener('load', function () {
|
|
navigator.serviceWorker.register('sw.js?v=<?php echo $swVersion; ?>').then(function (reg) {
|
|
// Periodisch updates prüfen
|
|
setInterval(function () { reg.update(); }, 60000);
|
|
// Bei Tab-Wechsel zurück: sofort prüfen
|
|
document.addEventListener('visibilitychange', function () {
|
|
if (document.visibilityState === 'visible') reg.update();
|
|
});
|
|
// Neuer SW installed → sofort aktivieren
|
|
reg.addEventListener('updatefound', function () {
|
|
var nw = reg.installing;
|
|
if (!nw) return;
|
|
nw.addEventListener('statechange', function () {
|
|
if (nw.state === 'installed' && navigator.serviceWorker.controller) {
|
|
nw.postMessage({ type: 'SKIP_WAITING' });
|
|
}
|
|
});
|
|
});
|
|
}).catch(function (e) { console.warn('SW reg failed', e); });
|
|
// Controller-Change → einmal neu laden
|
|
var reloaded = false;
|
|
navigator.serviceWorker.addEventListener('controllerchange', function () {
|
|
if (reloaded) return;
|
|
reloaded = true;
|
|
window.location.reload();
|
|
});
|
|
});
|
|
}
|
|
</script>
|
|
</body>
|
|
</html>
|