added URL scheme/deep linking

This commit is contained in:
cpu
2025-03-24 00:43:55 +01:00
parent 8a6947f4ea
commit 2838df5e05
8 changed files with 1142 additions and 31 deletions

155
sw.js
View File

@@ -1,52 +1,153 @@
// Updated service worker code - sw.js
const CACHE_NAME = 'timer-cache-v1';
const urlsToCache = [
// Service Worker version
const CACHE_VERSION = 'v1.0.0';
const CACHE_NAME = `game-timer-${CACHE_VERSION}`;
// Files to cache
const CACHE_FILES = [
'/',
'/index.html',
'/styles.css',
'/apps.js',
'/app.js',
'/audio.js',
'/styles.css',
'/manifest.json',
'/icons/android-chrome-192x192.png',
'/icons/android-chrome-512x512.png',
'/icons/apple-touch-icon.png',
'/icons/favicon-32x32.png',
'/icons/favicon-16x16.png',
'/favicon.ico',
'/manifest.json',
'/site.webmanifest'
'/icons/favicon-16x16.png'
];
// Install event - Cache files
self.addEventListener('install', event => {
console.log('[ServiceWorker] Install');
event.waitUntil(
caches.open(CACHE_NAME)
.then(cache => {
console.log('Opened cache');
// Use individual cache.add calls in a Promise.all to handle failures better
return Promise.all(
urlsToCache.map(url => {
return cache.add(url).catch(err => {
console.log('Failed to cache:', url, err);
// Continue despite individual failures
return Promise.resolve();
});
})
);
console.log('[ServiceWorker] Caching app shell');
return cache.addAll(CACHE_FILES);
})
.then(() => {
console.log('[ServiceWorker] Skip waiting on install');
return self.skipWaiting();
})
);
});
// Activate event - Clean old caches
self.addEventListener('activate', event => {
console.log('[ServiceWorker] Activate');
event.waitUntil(
caches.keys().then(keyList => {
return Promise.all(keyList.map(key => {
if (key !== CACHE_NAME) {
console.log('[ServiceWorker] Removing old cache', key);
return caches.delete(key);
}
}));
})
.then(() => {
console.log('[ServiceWorker] Claiming clients');
return self.clients.claim();
})
);
});
// Fetch event - Serve from cache, fallback to network
self.addEventListener('fetch', event => {
console.log('[ServiceWorker] Fetch', event.request.url);
// For navigation requests that include our deep link parameters,
// skip the cache and go straight to network
if (event.request.mode === 'navigate') {
const url = new URL(event.request.url);
// Check if request has action parameter or hash
if (url.searchParams.has('action') || url.hash.includes('action=')) {
console.log('[ServiceWorker] Processing deep link navigation');
return;
}
}
event.respondWith(
caches.match(event.request)
.then(response => {
// Cache hit - return response
if (response) {
return response;
}
return fetch(event.request);
return response || fetch(event.request)
.then(res => {
// Check if we should cache this response
if (shouldCacheResponse(event.request, res)) {
return caches.open(CACHE_NAME)
.then(cache => {
console.log('[ServiceWorker] Caching new resource:', event.request.url);
cache.put(event.request, res.clone());
return res;
});
}
return res;
});
})
.catch(err => {
console.log('Fetch handler failed:', err);
.catch(error => {
console.log('[ServiceWorker] Fetch failed; returning offline page', error);
// You could return a custom offline page here
})
);
});
// Helper function to determine if a response should be cached
function shouldCacheResponse(request, response) {
// Only cache GET requests
if (request.method !== 'GET') return false;
// Don't cache errors
if (!response || response.status !== 200) return false;
// Check if URL should be cached
const url = new URL(request.url);
// Don't cache query parameters (except common ones for content)
if (url.search && !url.search.match(/\?(v|version|cache)=/)) return false;
return true;
}
// Handle deep links from other apps (including Flic)
self.addEventListener('message', event => {
if (event.data && event.data.type === 'PROCESS_ACTION') {
const action = event.data.action;
console.log('[ServiceWorker] Received action message:', action);
// Broadcast the action to all clients
self.clients.matchAll().then(clients => {
clients.forEach(client => {
client.postMessage({
type: 'ACTION',
action: action
});
});
});
}
});
// This helps with navigation after app is installed
self.addEventListener('notificationclick', event => {
event.notification.close();
// This looks to see if the current is already open and focuses if it is
event.waitUntil(
self.clients.matchAll({
type: 'window'
})
.then(clientList => {
// Check if there is already a window/tab open with the target URL
for (const client of clientList) {
// If so, just focus it
if (client.url.startsWith(self.location.origin) && 'focus' in client) {
return client.focus();
}
}
// If not, open a new window/tab
if (self.clients.openWindow) {
return self.clients.openWindow('/');
}
})
);
});