| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127 |
- // Service Worker for PilotDeck PWA
- // Cache only manifest (needed for PWA install). HTML and JS are never pre-cached
- // so a rebuild + refresh always picks up the latest assets.
- // Bump this token whenever a cached asset's contents change (icons, manifest).
- // The activate handler below purges every cache whose name doesn't match,
- // so existing PWAs pick up the new visuals on the next page load.
- const CACHE_NAME = 'politdeck-v1';
- const urlsToCache = [
- '/manifest.json'
- ];
- // Install event
- self.addEventListener('install', event => {
- event.waitUntil(
- caches.open(CACHE_NAME)
- .then(cache => cache.addAll(urlsToCache))
- );
- self.skipWaiting();
- });
- // Fetch event — network-first for everything except hashed assets
- self.addEventListener('fetch', event => {
- const url = event.request.url;
- // Never intercept API requests or WebSocket upgrades
- if (url.includes('/api/') || url.includes('/ws')) {
- return;
- }
- // Navigation requests (HTML) — always go to network, no caching
- if (event.request.mode === 'navigate') {
- event.respondWith(
- fetch(event.request).catch(() => caches.match('/manifest.json').then(() =>
- new Response('<h1>Offline</h1><p>Please check your connection.</p>', {
- headers: { 'Content-Type': 'text/html' }
- })
- ))
- );
- return;
- }
- // Hashed assets (JS/CSS in /assets/) — cache-first since filenames change per build
- if (url.includes('/assets/')) {
- event.respondWith(
- caches.match(event.request).then(cached => {
- if (cached) return cached;
- return fetch(event.request).then(response => {
- const clone = response.clone();
- caches.open(CACHE_NAME).then(cache => cache.put(event.request, clone));
- return response;
- });
- })
- );
- return;
- }
- // Everything else — network-first
- event.respondWith(
- fetch(event.request).catch(() => caches.match(event.request))
- );
- });
- // Activate event — purge old caches
- self.addEventListener('activate', event => {
- event.waitUntil(
- caches.keys().then(cacheNames =>
- Promise.all(
- cacheNames
- .filter(name => name !== CACHE_NAME)
- .map(name => caches.delete(name))
- )
- )
- );
- self.clients.claim();
- });
- // Push notification event
- self.addEventListener('push', event => {
- if (!event.data) return;
- let payload;
- try {
- payload = event.data.json();
- } catch {
- payload = { title: 'PilotDeck', body: event.data.text() };
- }
- const options = {
- body: payload.body || '',
- icon: '/logo-256.png',
- badge: '/logo-128.png',
- data: payload.data || {},
- tag: payload.data?.tag || `${payload.data?.sessionId || 'global'}:${payload.data?.code || 'default'}`,
- renotify: true
- };
- event.waitUntil(
- self.registration.showNotification(payload.title || 'PilotDeck', options)
- );
- });
- // Notification click event
- self.addEventListener('notificationclick', event => {
- event.notification.close();
- const sessionId = event.notification.data?.sessionId;
- const provider = event.notification.data?.provider || null;
- const urlPath = sessionId ? `/session/${sessionId}` : '/';
- event.waitUntil(
- self.clients.matchAll({ type: 'window', includeUncontrolled: true }).then(async clientList => {
- for (const client of clientList) {
- if (client.url.includes(self.location.origin)) {
- await client.focus();
- client.postMessage({
- type: 'notification:navigate',
- sessionId: sessionId || null,
- provider,
- urlPath
- });
- return;
- }
- }
- return self.clients.openWindow(urlPath);
- })
- );
- });
|