Wire up a web manifest, service worker and icon routes so the player can be installed on mobile. The static manifest stays generic; the page regenerates it at runtime from STATION_NAME so an instance keeps its name. The service worker only caches the app shell, never the live stream or the playback APIs. Icons (192/512, maskable and apple-touch) are rasterized from the favicon. Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
61 lines
2.1 KiB
JavaScript
61 lines
2.1 KiB
JavaScript
// Service worker minimal : son seul rôle indispensable est de rendre la webapp
|
|
// « installable » (critère PWA d'Android/Chrome) en présentant un gestionnaire
|
|
// de requêtes. Accessoirement, il met en cache la coquille de l'application
|
|
// (page, icônes, manifest) pour que la fenêtre autonome s'ouvre instantanément,
|
|
// même hors ligne.
|
|
//
|
|
// Rien de ce qui est « vivant » ne passe par le cache : le flux audio
|
|
// (/radio.mp3) et toutes les API de lecture (/nowplaying, /history, /queue,
|
|
// /skip…) vont toujours directement au réseau. On se contente donc de gérer les
|
|
// quelques ressources statiques de la coquille.
|
|
|
|
const CACHE = "nemufm-shell-v1";
|
|
const SHELL = [
|
|
"/",
|
|
"/manifest.webmanifest",
|
|
"/favicon.svg",
|
|
"/icon-192.png",
|
|
"/icon-512.png",
|
|
"/icon-maskable-512.png",
|
|
"/apple-touch-icon.png",
|
|
];
|
|
|
|
self.addEventListener("install", (e) => {
|
|
e.waitUntil(caches.open(CACHE).then((c) => c.addAll(SHELL)));
|
|
self.skipWaiting();
|
|
});
|
|
|
|
self.addEventListener("activate", (e) => {
|
|
// Purge des anciennes versions de coquille.
|
|
e.waitUntil(
|
|
caches.keys().then((keys) =>
|
|
Promise.all(keys.filter((k) => k !== CACHE).map((k) => caches.delete(k)))
|
|
)
|
|
);
|
|
self.clients.claim();
|
|
});
|
|
|
|
self.addEventListener("fetch", (e) => {
|
|
const req = e.request;
|
|
if (req.method !== "GET") return;
|
|
const url = new URL(req.url);
|
|
|
|
// Hors périmètre (CDN d'arrière-plan…) ou ressources vivantes : on laisse le
|
|
// navigateur faire, sans interception ni cache.
|
|
if (url.origin !== location.origin) return;
|
|
const shellPaths = new Set(SHELL);
|
|
const isShell = url.pathname === "/" ? true : shellPaths.has(url.pathname);
|
|
if (!isShell) return; // flux et API : réseau direct, pas de cache.
|
|
|
|
// Coquille : réseau d'abord (pour récupérer une version fraîche), repli sur le
|
|
// cache si le réseau manque, avec mise à jour du cache au passage.
|
|
e.respondWith(
|
|
fetch(req)
|
|
.then((resp) => {
|
|
const copy = resp.clone();
|
|
caches.open(CACHE).then((c) => c.put(req, copy)).catch(() => {});
|
|
return resp;
|
|
})
|
|
.catch(() => caches.match(req).then((r) => r || caches.match("/")))
|
|
);
|
|
});
|