This commit is contained in:
Laucha1312
2026-06-04 15:19:42 -03:00
parent 90c5f85512
commit fdd0fef3f0
37 changed files with 4912 additions and 0 deletions
+134
View File
@@ -0,0 +1,134 @@
const CACHE_NAME = 'onapb-v4';
// Recursos estáticos que se cachean en la instalación
const STATIC_ASSETS = [
'/',
'/eventos',
'/torneos',
'/offline',
'/manifest.json',
];
// ── Install: pre-cachear recursos estáticos ──
self.addEventListener('install', (event) => {
self.skipWaiting();
event.waitUntil(
caches.open(CACHE_NAME).then((cache) => {
return cache.addAll(STATIC_ASSETS).catch(() => {
// Si algún recurso falla, no bloquear la instalación
console.warn('[SW] Algún recurso estático no pudo cachearse.');
});
})
);
});
// ── Activate: limpiar caches viejas ──
self.addEventListener('activate', (event) => {
event.waitUntil(
caches.keys().then((keys) =>
Promise.all(
keys
.filter((key) => key !== CACHE_NAME)
.map((key) => caches.delete(key))
)
).then(() => self.clients.claim())
);
});
// ── Fetch: Network-first para páginas, Cache-first para assets ──
self.addEventListener('fetch', (event) => {
const { request } = event;
const url = new URL(request.url);
// Solo manejar peticiones del mismo origen
if (url.origin !== location.origin) return;
// Ignorar peticiones POST, PUT, DELETE, admin, notificaciones en tiempo real
if (request.method !== 'GET') return;
if (url.pathname.startsWith('/admin')) return;
if (url.pathname.startsWith('/notificaciones/count')) return;
// Assets estáticos (JS, CSS, imágenes, fuentes): Cache-first
if (
url.pathname.match(/\.(js|css|png|jpg|jpeg|gif|svg|webp|woff2?|ico)$/)
) {
event.respondWith(
caches.match(request).then((cached) => {
if (cached) return cached;
return fetch(request).then((response) => {
if (response && response.status === 200) {
const clone = response.clone();
caches.open(CACHE_NAME).then((cache) => cache.put(request, clone));
}
return response;
});
})
);
return;
}
// Páginas HTML: Network-first, fallback a cache, luego /offline
event.respondWith(
fetch(request)
.then((response) => {
if (response && response.status === 200) {
const clone = response.clone();
caches.open(CACHE_NAME).then((cache) => cache.put(request, clone));
}
return response;
})
.catch(() =>
caches.match(request).then(
(cached) => cached || caches.match('/offline')
)
)
);
});
// ── Web Push Notifications ──
self.addEventListener('push', (event) => {
let data = { title: 'OnAPB', body: 'Tenés una nueva notificación' };
if (event.data) {
try {
data = event.data.json();
} catch (e) {
data.body = event.data.text();
}
}
const options = {
body: data.body,
icon: '/icons/icon-192.png?v=4',
badge: '/icons/icon-72.png?v=4',
data: {
url: data.url || '/'
},
vibrate: [100, 50, 100]
};
event.waitUntil(
self.registration.showNotification(data.title, options)
);
});
self.addEventListener('notificationclick', (event) => {
event.notification.close();
const targetUrl = event.notification.data.url || '/';
event.waitUntil(
clients.matchAll({ type: 'window', includeUncontrolled: true }).then((clientList) => {
// Si ya hay una ventana abierta con la misma URL, enfocarla
for (const client of clientList) {
if (client.url === targetUrl && 'focus' in client) {
return client.focus();
}
}
// Si no, abrir una nueva
if (clients.openWindow) {
return clients.openWindow(targetUrl);
}
})
);
});