diff --git a/backend/src/services/file.js b/backend/src/services/file.js index 1576e00..3ba884b 100644 --- a/backend/src/services/file.js +++ b/backend/src/services/file.js @@ -335,6 +335,38 @@ const uploadFile = async (folder, req, res) => { } }; +/** + * Parse Range header value + * @param {string} rangeHeader - Range header value (e.g., "bytes=0-1000") + * @param {number} totalSize - Total file size + * @returns {{start: number, end: number} | null} + */ +const parseRangeHeader = (rangeHeader, totalSize) => { + if (!rangeHeader || !rangeHeader.startsWith('bytes=')) return null; + + const range = rangeHeader.slice(6); // Remove "bytes=" + const parts = range.split('-'); + + let start = parseInt(parts[0], 10); + let end = parts[1] ? parseInt(parts[1], 10) : totalSize - 1; + + // Handle suffix ranges (e.g., bytes=-500 means last 500 bytes) + if (isNaN(start)) { + start = totalSize - end; + end = totalSize - 1; + } + + // Validate range + if (isNaN(start) || isNaN(end) || start > end || start >= totalSize) { + return null; + } + + // Cap end to file size + end = Math.min(end, totalSize - 1); + + return { start, end }; +}; + const downloadFile = async (req, res) => { const provider = getFileStorageProvider(); const privateUrl = req.query.privateUrl; @@ -359,6 +391,7 @@ const downloadFile = async (req, res) => { } res.setHeader('Cross-Origin-Resource-Policy', 'cross-origin'); + res.setHeader('Accept-Ranges', 'bytes'); // Create AbortController for request cancellation const abortController = new AbortController(); @@ -393,14 +426,6 @@ const downloadFile = async (req, res) => { return res.status(304).end(); } - // Set caching headers - res.setHeader('ETag', etag); - res.setHeader( - 'Cache-Control', - `public, max-age=${config.s3CacheMaxAge}`, - ); - res.setHeader('Content-Length', stats.size); - // Determine content type from extension const ext = path.extname(privateUrl).toLowerCase(); const mimeTypes = { @@ -421,12 +446,84 @@ const downloadFile = async (req, res) => { res.setHeader('Content-Type', mimeTypes[ext]); } + // Handle Range requests for cached files + const rangeHeader = req.headers.range; + if (rangeHeader) { + const range = parseRangeHeader(rangeHeader, stats.size); + if (range) { + const { start, end } = range; + const chunkSize = end - start + 1; + + res.status(206); + res.setHeader('Content-Range', `bytes ${start}-${end}/${stats.size}`); + res.setHeader('Content-Length', chunkSize); + res.setHeader('ETag', etag); + res.setHeader('Cache-Control', `public, max-age=${config.s3CacheMaxAge}`); + + return fs.createReadStream(cachePath, { start, end }).pipe(res); + } + // Invalid range - return 416 + res.setHeader('Content-Range', `bytes */${stats.size}`); + return res.status(416).end(); + } + + // Set caching headers for full file + res.setHeader('ETag', etag); + res.setHeader( + 'Cache-Control', + `public, max-age=${config.s3CacheMaxAge}`, + ); + res.setHeader('Content-Length', stats.size); + // Stream from cache return fs.createReadStream(cachePath).pipe(res); } } - // Download from S3 + // Handle Range requests for S3 (bypass cache for partial requests) + const rangeHeader = req.headers.range; + if (rangeHeader) { + // For Range requests, we need to get file size first via headObject + const headResult = await s3.download(privateUrl, { signal, headOnly: true }); + const totalSize = headResult.contentLength; + + if (!totalSize) { + log.warn({ privateUrl }, 'Cannot determine file size for range request'); + return res.status(500).send(createErrorResponse('Cannot determine file size', 'SIZE_UNKNOWN')); + } + + const range = parseRangeHeader(rangeHeader, totalSize); + if (!range) { + res.setHeader('Content-Range', `bytes */${totalSize}`); + return res.status(416).end(); + } + + const { start, end } = range; + const chunkSize = end - start + 1; + + // Download range from S3 + const rangeResult = await s3.download(privateUrl, { + signal, + range: `bytes=${start}-${end}`, + }); + + res.status(206); + res.setHeader('Content-Range', `bytes ${start}-${end}/${totalSize}`); + res.setHeader('Content-Length', chunkSize); + if (rangeResult.contentType) res.setHeader('Content-Type', rangeResult.contentType); + res.setHeader('Cache-Control', `public, max-age=${config.s3CacheMaxAge}`); + + if (typeof rangeResult.body.pipe === 'function') { + return rangeResult.body.pipe(res); + } else if (typeof rangeResult.body.transformToByteArray === 'function') { + const bytes = await rangeResult.body.transformToByteArray(); + return res.send(Buffer.from(bytes)); + } else { + return res.send(rangeResult.body); + } + } + + // Download from S3 (full file) const result = await s3.download(privateUrl, { signal }); if (result.contentType) res.setHeader('Content-Type', result.contentType); @@ -537,7 +634,58 @@ const downloadFile = async (req, res) => { .send(createErrorResponse('File not found', 'NOT_FOUND')); } } else { - res.download(path.join(config.uploadDir, privateUrl)); + // Local storage - support Range requests for video streaming + const localFilePath = path.join(config.uploadDir, privateUrl); + + if (!fs.existsSync(localFilePath)) { + return res.status(404).send(createErrorResponse('File not found', 'NOT_FOUND')); + } + + const stats = fs.statSync(localFilePath); + const totalSize = stats.size; + + // Determine content type from extension + const ext = path.extname(privateUrl).toLowerCase(); + const mimeTypes = { + '.jpg': 'image/jpeg', + '.jpeg': 'image/jpeg', + '.png': 'image/png', + '.gif': 'image/gif', + '.webp': 'image/webp', + '.svg': 'image/svg+xml', + '.ico': 'image/x-icon', + '.mp4': 'video/mp4', + '.webm': 'video/webm', + '.mp3': 'audio/mpeg', + '.wav': 'audio/wav', + '.pdf': 'application/pdf', + }; + if (mimeTypes[ext]) { + res.setHeader('Content-Type', mimeTypes[ext]); + } + + // Handle Range requests + const rangeHeader = req.headers.range; + if (rangeHeader) { + const range = parseRangeHeader(rangeHeader, totalSize); + if (range) { + const { start, end } = range; + const chunkSize = end - start + 1; + + res.status(206); + res.setHeader('Content-Range', `bytes ${start}-${end}/${totalSize}`); + res.setHeader('Content-Length', chunkSize); + + return fs.createReadStream(localFilePath, { start, end }).pipe(res); + } + // Invalid range - return 416 + res.setHeader('Content-Range', `bytes */${totalSize}`); + return res.status(416).end(); + } + + // Full file download + res.setHeader('Content-Length', totalSize); + return fs.createReadStream(localFilePath).pipe(res); } } catch (error) { // Don't log abort errors as they're expected when client disconnects diff --git a/backend/src/services/file/S3StorageProvider.js b/backend/src/services/file/S3StorageProvider.js index 990cb47..20ed25e 100644 --- a/backend/src/services/file/S3StorageProvider.js +++ b/backend/src/services/file/S3StorageProvider.js @@ -227,18 +227,44 @@ class S3StorageProvider extends BaseStorageProvider { * @param {string} key - Storage key/path * @param {Object} [options] - Download options * @param {AbortSignal} [options.signal] - AbortController signal for cancellation + * @param {boolean} [options.headOnly] - Only get metadata (HEAD request) + * @param {string} [options.range] - HTTP Range header value (e.g., "bytes=0-1000") * @returns {Promise<{ body: ReadableStream, contentType?: string, contentLength?: number }>} */ async download(key, options = {}) { const fullKey = this.buildKey(key); - const { signal } = options; + const { signal, headOnly, range } = options; const sendOptions = signal ? { abortSignal: signal } : {}; + + // HEAD request for metadata only + if (headOnly) { + const output = await this.client.send( + new HeadObjectCommand({ + Bucket: this.bucket, + Key: fullKey, + }), + sendOptions, + ); + return { + body: null, + contentType: output.ContentType, + contentLength: output.ContentLength, + }; + } + + // Build GetObjectCommand with optional Range header + const commandParams = { + Bucket: this.bucket, + Key: fullKey, + }; + + if (range) { + commandParams.Range = range; + } + const output = await this.client.send( - new GetObjectCommand({ - Bucket: this.bucket, - Key: fullKey, - }), + new GetObjectCommand(commandParams), sendOptions, ); diff --git a/frontend/public/sw.js b/frontend/public/sw.js index 8ee50ae..dddd0b6 100644 --- a/frontend/public/sw.js +++ b/frontend/public/sw.js @@ -1,2 +1,2 @@ (()=>{"use strict";let e,t,a,s={googleAnalytics:"googleAnalytics",precache:"precache-v2",prefix:"serwist",runtime:"runtime",suffix:"undefined"!=typeof registration?registration.scope:""},r=e=>[s.prefix,e,s.suffix].filter(e=>e&&e.length>0).join("-"),n={updateDetails:e=>{var t=t=>{let a=e[t];"string"==typeof a&&(s[t]=a)};for(let e of Object.keys(s))t(e)},getGoogleAnalyticsName:e=>e||r(s.googleAnalytics),getPrecacheName:e=>e||r(s.precache),getRuntimeName:e=>e||r(s.runtime)};class i extends Error{details;constructor(e,t){super(((e,...t)=>{let a=e;return t.length>0&&(a+=` :: ${JSON.stringify(t)}`),a})(e,t)),this.name=e,this.details=t}}function c(e){return new Promise(t=>setTimeout(t,e))}let o=new Set;function l(e,t){let a=new URL(e);for(let e of t)a.searchParams.delete(e);return a.href}async function h(e,t,a,s){let r=l(t.url,a);if(t.url===r)return e.match(t,s);let n={...s,ignoreSearch:!0};for(let i of(await e.keys(t,n)))if(r===l(i.url,a))return e.match(i,s)}class u{promise;resolve;reject;constructor(){this.promise=new Promise((e,t)=>{this.resolve=e,this.reject=t})}}let d=async()=>{for(let e of o)await e()},m="-precache-",g=async(e,t=m)=>{let a=(await self.caches.keys()).filter(a=>a.includes(t)&&a.includes(self.registration.scope)&&a!==e);return await Promise.all(a.map(e=>self.caches.delete(e))),a},p=(e,t)=>{let a=t();return e.waitUntil(a),a},f=(e,t)=>t.some(t=>e instanceof t),w=new WeakMap,y=new WeakMap,_=new WeakMap,b={get(e,t,a){if(e instanceof IDBTransaction){if("done"===t)return w.get(e);if("store"===t)return a.objectStoreNames[1]?void 0:a.objectStore(a.objectStoreNames[0])}return R(e[t])},set:(e,t,a)=>(e[t]=a,!0),has:(e,t)=>e instanceof IDBTransaction&&("done"===t||"store"===t)||t in e};function R(e){if(e instanceof IDBRequest){let t=new Promise((t,a)=>{let s=()=>{e.removeEventListener("success",r),e.removeEventListener("error",n)},r=()=>{t(R(e.result)),s()},n=()=>{a(e.error),s()};e.addEventListener("success",r),e.addEventListener("error",n)});return _.set(t,e),t}if(y.has(e))return y.get(e);let s=function(e){if("function"==typeof e)return(a||(a=[IDBCursor.prototype.advance,IDBCursor.prototype.continue,IDBCursor.prototype.continuePrimaryKey])).includes(e)?function(...t){return e.apply(v(this),t),R(this.request)}:function(...t){return R(e.apply(v(this),t))};return(e instanceof IDBTransaction&&function(e){if(w.has(e))return;let t=new Promise((t,a)=>{let s=()=>{e.removeEventListener("complete",r),e.removeEventListener("error",n),e.removeEventListener("abort",n)},r=()=>{t(),s()},n=()=>{a(e.error||new DOMException("AbortError","AbortError")),s()};e.addEventListener("complete",r),e.addEventListener("error",n),e.addEventListener("abort",n)});w.set(e,t)}(e),f(e,t||(t=[IDBDatabase,IDBObjectStore,IDBIndex,IDBCursor,IDBTransaction])))?new Proxy(e,b):e}(e);return s!==e&&(y.set(e,s),_.set(s,e)),s}let v=e=>_.get(e);function x(e,t,{blocked:a,upgrade:s,blocking:r,terminated:n}={}){let i=indexedDB.open(e,t),c=R(i);return s&&i.addEventListener("upgradeneeded",e=>{s(R(i.result),e.oldVersion,e.newVersion,R(i.transaction),e)}),a&&i.addEventListener("blocked",e=>a(e.oldVersion,e.newVersion,e)),c.then(e=>{n&&e.addEventListener("close",()=>n()),r&&e.addEventListener("versionchange",e=>r(e.oldVersion,e.newVersion,e))}).catch(()=>{}),c}let E=["get","getKey","getAll","getAllKeys","count"],S=["put","add","delete","clear"],q=new Map;function C(e,t){if(!(e instanceof IDBDatabase&&!(t in e)&&"string"==typeof t))return;if(q.get(t))return q.get(t);let a=t.replace(/FromIndex$/,""),s=t!==a,r=S.includes(a);if(!(a in(s?IDBIndex:IDBObjectStore).prototype)||!(r||E.includes(a)))return;let n=async function(e,...t){let n=this.transaction(e,r?"readwrite":"readonly"),i=n.store;return s&&(i=i.index(t.shift())),(await Promise.all([i[a](...t),r&&n.done]))[0]};return q.set(t,n),n}b=(e=>({...e,get:(t,a,s)=>C(t,a)||e.get(t,a,s),has:(t,a)=>!!C(t,a)||e.has(t,a)}))(b);let N=["continue","continuePrimaryKey","advance"],D={},T=new WeakMap,P=new WeakMap,U={get(e,t){if(!N.includes(t))return e[t];let a=D[t];return a||(a=D[t]=function(...e){T.set(this,P.get(this)[t](...e))}),a}};async function*A(...e){let t=this;if(t instanceof IDBCursor||(t=await t.openCursor(...e)),!t)return;let a=new Proxy(t,U);for(P.set(a,t),_.set(a,v(t));t;)yield a,t=await (T.get(a)||t.continue()),T.delete(a)}function k(e,t){return t===Symbol.asyncIterator&&f(e,[IDBIndex,IDBObjectStore,IDBCursor])||"iterate"===t&&f(e,[IDBIndex,IDBObjectStore])}b=(e=>({...e,get:(t,a,s)=>k(t,a)?A:e.get(t,a,s),has:(t,a)=>k(t,a)||e.has(t,a)}))(b);let I=async(t,a)=>{let s=null;if(t.url&&(s=new URL(t.url).origin),s!==self.location.origin)throw new i("cross-origin-copy-response",{origin:s});let r=t.clone(),n={headers:new Headers(r.headers),status:r.status,statusText:r.statusText},c=a?a(n):n,o=!function(){if(void 0===e){let t=new Response("");if("body"in t)try{new Response(t.body),e=!0}catch{e=!1}e=!1}return e}()?await r.blob():r.body;return new Response(o,c)},L="requests",W="queueName";class F{_db=null;async addEntry(e){let t=(await this.getDb()).transaction(L,"readwrite",{durability:"relaxed"});await t.store.add(e),await t.done}async getFirstEntryId(){let e=await this.getDb(),t=await e.transaction(L).store.openCursor();return t?.value.id}async getAllEntriesByQueueName(e){let t=await this.getDb();return await t.getAllFromIndex(L,W,IDBKeyRange.only(e))||[]}async getEntryCountByQueueName(e){return(await this.getDb()).countFromIndex(L,W,IDBKeyRange.only(e))}async deleteEntry(e){let t=await this.getDb();await t.delete(L,e)}async getFirstEntryByQueueName(e){return await this.getEndEntryFromIndex(IDBKeyRange.only(e),"next")}async getLastEntryByQueueName(e){return await this.getEndEntryFromIndex(IDBKeyRange.only(e),"prev")}async getEndEntryFromIndex(e,t){let a=await this.getDb(),s=await a.transaction(L).store.index(W).openCursor(e,t);return s?.value}async getDb(){return this._db||(this._db=await x("serwist-background-sync",3,{upgrade:this._upgradeDb})),this._db}_upgradeDb(e,t){t>0&&t<3&&e.objectStoreNames.contains(L)&&e.deleteObjectStore(L),e.createObjectStore(L,{autoIncrement:!0,keyPath:"id"}).createIndex(W,W,{unique:!1})}}class K{_queueName;_queueDb;constructor(e){this._queueName=e,this._queueDb=new F}async pushEntry(e){delete e.id,e.queueName=this._queueName,await this._queueDb.addEntry(e)}async unshiftEntry(e){let t=await this._queueDb.getFirstEntryId();t?e.id=t-1:delete e.id,e.queueName=this._queueName,await this._queueDb.addEntry(e)}async popEntry(){return this._removeEntry(await this._queueDb.getLastEntryByQueueName(this._queueName))}async shiftEntry(){return this._removeEntry(await this._queueDb.getFirstEntryByQueueName(this._queueName))}async getAll(){return await this._queueDb.getAllEntriesByQueueName(this._queueName)}async size(){return await this._queueDb.getEntryCountByQueueName(this._queueName)}async deleteEntry(e){await this._queueDb.deleteEntry(e)}async _removeEntry(e){return e&&await this.deleteEntry(e.id),e}}let B=["method","referrer","referrerPolicy","mode","credentials","cache","redirect","integrity","keepalive"];class M{_requestData;static async fromRequest(e){let t={url:e.url,headers:{}};for(let a of("GET"!==e.method&&(t.body=await e.clone().arrayBuffer()),e.headers.forEach((e,a)=>{t.headers[a]=e}),B))void 0!==e[a]&&(t[a]=e[a]);return new M(t)}constructor(e){"navigate"===e.mode&&(e.mode="same-origin"),this._requestData=e}toObject(){let e=Object.assign({},this._requestData);return e.headers=Object.assign({},this._requestData.headers),e.body&&(e.body=e.body.slice(0)),e}toRequest(){return new Request(this._requestData.url,this._requestData)}clone(){return new M(this.toObject())}}let O="serwist-background-sync",j=new Set,H=e=>{let t={request:new M(e.requestData).toRequest(),timestamp:e.timestamp};return e.metadata&&(t.metadata=e.metadata),t};class ${_name;_onSync;_maxRetentionTime;_queueStore;_forceSyncFallback;_syncInProgress=!1;_requestsAddedDuringSync=!1;constructor(e,{forceSyncFallback:t,onSync:a,maxRetentionTime:s}={}){if(j.has(e))throw new i("duplicate-queue-name",{name:e});j.add(e),this._name=e,this._onSync=a||this.replayRequests,this._maxRetentionTime=s||10080,this._forceSyncFallback=!!t,this._queueStore=new K(this._name),this._addSyncListener()}get name(){return this._name}async pushRequest(e){await this._addRequest(e,"push")}async unshiftRequest(e){await this._addRequest(e,"unshift")}async popRequest(){return this._removeRequest("pop")}async shiftRequest(){return this._removeRequest("shift")}async getAll(){let e=await this._queueStore.getAll(),t=Date.now(),a=[];for(let s of e){let e=60*this._maxRetentionTime*1e3;t-s.timestamp>e?await this._queueStore.deleteEntry(s.id):a.push(H(s))}return a}async size(){return await this._queueStore.size()}async _addRequest({request:e,metadata:t,timestamp:a=Date.now()},s){let r={requestData:(await M.fromRequest(e.clone())).toObject(),timestamp:a};switch(t&&(r.metadata=t),s){case"push":await this._queueStore.pushEntry(r);break;case"unshift":await this._queueStore.unshiftEntry(r)}this._syncInProgress?this._requestsAddedDuringSync=!0:await this.registerSync()}async _removeRequest(e){let t,a=Date.now();switch(e){case"pop":t=await this._queueStore.popEntry();break;case"shift":t=await this._queueStore.shiftEntry()}if(t){let s=60*this._maxRetentionTime*1e3;return a-t.timestamp>s?this._removeRequest(e):H(t)}}async replayRequests(){let e;for(;e=await this.shiftRequest();)try{await fetch(e.request.clone())}catch{throw await this.unshiftRequest(e),new i("queue-replay-failed",{name:this._name})}}async registerSync(){if("sync"in self.registration&&!this._forceSyncFallback)try{await self.registration.sync.register(`${O}:${this._name}`)}catch(e){}}_addSyncListener(){"sync"in self.registration&&!this._forceSyncFallback?self.addEventListener("sync",e=>{if(e.tag===`${O}:${this._name}`){let t=async()=>{let t;this._syncInProgress=!0;try{await this._onSync({queue:this})}catch(e){if(e instanceof Error)throw e}finally{this._requestsAddedDuringSync&&!(t&&!e.lastChance)&&await this.registerSync(),this._syncInProgress=!1,this._requestsAddedDuringSync=!1}};e.waitUntil(t())}}):this._onSync({queue:this})}static get _queueNames(){return j}}class z{_queue;constructor(e,t){this._queue=new $(e,t)}async fetchDidFail({request:e}){await this._queue.pushRequest({request:e})}}let G={cacheWillUpdate:async({response:e})=>200===e.status||0===e.status?e:null};function V(e){return"string"==typeof e?new Request(e):e}class Q{event;request;url;params;_cacheKeys={};_strategy;_handlerDeferred;_extendLifetimePromises;_plugins;_pluginStateMap;constructor(e,t){for(let a of(this.event=t.event,this.request=t.request,t.url&&(this.url=t.url,this.params=t.params),this._strategy=e,this._handlerDeferred=new u,this._extendLifetimePromises=[],this._plugins=[...e.plugins],this._pluginStateMap=new Map,this._plugins))this._pluginStateMap.set(a,{});this.event.waitUntil(this._handlerDeferred.promise)}async fetch(e){let{event:t}=this,a=V(e),s=await this.getPreloadResponse();if(s)return s;let r=this.hasCallback("fetchDidFail")?a.clone():null;try{for(let e of this.iterateCallbacks("requestWillFetch"))a=await e({request:a.clone(),event:t})}catch(e){if(e instanceof Error)throw new i("plugin-error-request-will-fetch",{thrownErrorMessage:e.message})}let n=a.clone();try{let e;for(let s of(e=await fetch(a,"navigate"===a.mode?void 0:this._strategy.fetchOptions),this.iterateCallbacks("fetchDidSucceed")))e=await s({event:t,request:n,response:e});return e}catch(e){throw r&&await this.runCallbacks("fetchDidFail",{error:e,event:t,originalRequest:r.clone(),request:n.clone()}),e}}async fetchAndCachePut(e){let t=await this.fetch(e),a=t.clone();return this.waitUntil(this.cachePut(e,a)),t}async cacheMatch(e){let t,a=V(e),{cacheName:s,matchOptions:r}=this._strategy,n=await this.getCacheKey(a,"read"),i={...r,cacheName:s};for(let e of(t=await caches.match(n,i),this.iterateCallbacks("cachedResponseWillBeUsed")))t=await e({cacheName:s,matchOptions:r,cachedResponse:t,request:n,event:this.event})||void 0;return t}async cachePut(e,t){let a=V(e);await c(0);let s=await this.getCacheKey(a,"write");if(!t)throw new i("cache-put-with-no-response",{url:new URL(String(s.url),location.href).href.replace(RegExp(`^${location.origin}`),"")});let r=await this._ensureResponseSafeToCache(t);if(!r)return!1;let{cacheName:n,matchOptions:o}=this._strategy,l=await self.caches.open(n),u=this.hasCallback("cacheDidUpdate"),m=u?await h(l,s.clone(),["__WB_REVISION__"],o):null;try{await l.put(s,u?r.clone():r)}catch(e){if(e instanceof Error)throw"QuotaExceededError"===e.name&&await d(),e}for(let e of this.iterateCallbacks("cacheDidUpdate"))await e({cacheName:n,oldResponse:m,newResponse:r.clone(),request:s,event:this.event});return!0}async getCacheKey(e,t){let a=`${e.url} | ${t}`;if(!this._cacheKeys[a]){let s=e;for(let e of this.iterateCallbacks("cacheKeyWillBeUsed"))s=V(await e({mode:t,request:s,event:this.event,params:this.params}));this._cacheKeys[a]=s}return this._cacheKeys[a]}hasCallback(e){for(let t of this._strategy.plugins)if(e in t)return!0;return!1}async runCallbacks(e,t){for(let a of this.iterateCallbacks(e))await a(t)}*iterateCallbacks(e){for(let t of this._strategy.plugins)if("function"==typeof t[e]){let a=this._pluginStateMap.get(t),s=s=>{let r={...s,state:a};return t[e](r)};yield s}}waitUntil(e){return this._extendLifetimePromises.push(e),e}async doneWaiting(){let e;for(;e=this._extendLifetimePromises.shift();)await e}destroy(){this._handlerDeferred.resolve(null)}async getPreloadResponse(){if(this.event instanceof FetchEvent&&"navigate"===this.event.request.mode&&"preloadResponse"in this.event)try{let e=await this.event.preloadResponse;if(e)return e}catch(e){}}async _ensureResponseSafeToCache(e){let t=e,a=!1;for(let e of this.iterateCallbacks("cacheWillUpdate"))if(t=await e({request:this.request,response:t,event:this.event})||void 0,a=!0,!t)break;return!a&&t&&200!==t.status&&(t=void 0),t}}class J{cacheName;plugins;fetchOptions;matchOptions;constructor(e={}){this.cacheName=n.getRuntimeName(e.cacheName),this.plugins=e.plugins||[],this.fetchOptions=e.fetchOptions,this.matchOptions=e.matchOptions}handle(e){let[t]=this.handleAll(e);return t}handleAll(e){e instanceof FetchEvent&&(e={event:e,request:e.request});let t=e.event,a="string"==typeof e.request?new Request(e.request):e.request,s=new Q(this,e.url?{event:t,request:a,url:e.url,params:e.params}:{event:t,request:a}),r=this._getResponse(s,a,t),n=this._awaitComplete(r,s,a,t);return[r,n]}async _getResponse(e,t,a){let s;await e.runCallbacks("handlerWillStart",{event:a,request:t});try{if(s=await this._handle(t,e),void 0===s||"error"===s.type)throw new i("no-response",{url:t.url})}catch(r){if(r instanceof Error){for(let n of e.iterateCallbacks("handlerDidError"))if(void 0!==(s=await n({error:r,event:a,request:t})))break}if(!s)throw r}for(let r of e.iterateCallbacks("handlerWillRespond"))s=await r({event:a,request:t,response:s});return s}async _awaitComplete(e,t,a,s){let r,n;try{r=await e}catch{}try{await t.runCallbacks("handlerDidRespond",{event:s,request:a,response:r}),await t.doneWaiting()}catch(e){e instanceof Error&&(n=e)}if(await t.runCallbacks("handlerDidComplete",{event:s,request:a,response:r,error:n}),t.destroy(),n)throw n}}class X extends J{_networkTimeoutSeconds;constructor(e={}){super(e),this.plugins.some(e=>"cacheWillUpdate"in e)||this.plugins.unshift(G),this._networkTimeoutSeconds=e.networkTimeoutSeconds||0}async _handle(e,t){let a,s=[],r=[];if(this._networkTimeoutSeconds){let{id:n,promise:i}=this._getTimeoutPromise({request:e,logs:s,handler:t});a=n,r.push(i)}let n=this._getNetworkPromise({timeoutId:a,request:e,logs:s,handler:t});r.push(n);let c=await t.waitUntil((async()=>await t.waitUntil(Promise.race(r))||await n)());if(!c)throw new i("no-response",{url:e.url});return c}_getTimeoutPromise({request:e,logs:t,handler:a}){let s;return{promise:new Promise(t=>{s=setTimeout(async()=>{t(await a.cacheMatch(e))},1e3*this._networkTimeoutSeconds)}),id:s}}async _getNetworkPromise({timeoutId:e,request:t,logs:a,handler:s}){let r,n;try{n=await s.fetchAndCachePut(t)}catch(e){e instanceof Error&&(r=e)}return e&&clearTimeout(e),(r||!n)&&(n=await s.cacheMatch(t)),n}}class Y extends J{_networkTimeoutSeconds;constructor(e={}){super(e),this._networkTimeoutSeconds=e.networkTimeoutSeconds||0}async _handle(e,t){let a,s;try{let a=[t.fetch(e)];if(this._networkTimeoutSeconds){let e=c(1e3*this._networkTimeoutSeconds);a.push(e)}if(!(s=await Promise.race(a)))throw Error(`Timed out the network response after ${this._networkTimeoutSeconds} seconds.`)}catch(e){e instanceof Error&&(a=e)}if(!s)throw new i("no-response",{url:e.url,error:a});return s}}let Z=e=>e&&"object"==typeof e?e:{handle:e};class ee{handler;match;method;catchHandler;constructor(e,t,a="GET"){this.handler=Z(t),this.match=e,this.method=a}setCatchHandler(e){this.catchHandler=Z(e)}}class et extends J{_fallbackToNetwork;static defaultPrecacheCacheabilityPlugin={cacheWillUpdate:async({response:e})=>!e||e.status>=400?null:e};static copyRedirectedCacheableResponsesPlugin={cacheWillUpdate:async({response:e})=>e.redirected?await I(e):e};constructor(e={}){e.cacheName=n.getPrecacheName(e.cacheName),super(e),this._fallbackToNetwork=!1!==e.fallbackToNetwork,this.plugins.push(et.copyRedirectedCacheableResponsesPlugin)}async _handle(e,t){let a=await t.getPreloadResponse();if(a)return a;let s=await t.cacheMatch(e);return s||(t.event&&"install"===t.event.type?await this._handleInstall(e,t):await this._handleFetch(e,t))}async _handleFetch(e,t){let a,s=t.params||{};if(this._fallbackToNetwork){let r=s.integrity,n=e.integrity,i=!n||n===r;a=await t.fetch(new Request(e,{integrity:"no-cors"!==e.mode?n||r:void 0})),r&&i&&"no-cors"!==e.mode&&(this._useDefaultCacheabilityPluginIfNeeded(),await t.cachePut(e,a.clone()))}else throw new i("missing-precache-entry",{cacheName:this.cacheName,url:e.url});return a}async _handleInstall(e,t){this._useDefaultCacheabilityPluginIfNeeded();let a=await t.fetch(e);if(!await t.cachePut(e,a.clone()))throw new i("bad-precaching-response",{url:e.url,status:a.status});return a}_useDefaultCacheabilityPluginIfNeeded(){let e=null,t=0;for(let[a,s]of this.plugins.entries())s!==et.copyRedirectedCacheableResponsesPlugin&&(s===et.defaultPrecacheCacheabilityPlugin&&(e=a),s.cacheWillUpdate&&t++);0===t?this.plugins.push(et.defaultPrecacheCacheabilityPlugin):t>1&&null!==e&&this.plugins.splice(e,1)}}class ea extends ee{_allowlist;_denylist;constructor(e,{allowlist:t=[/./],denylist:a=[]}={}){super(e=>this._match(e),e),this._allowlist=t,this._denylist=a}_match({url:e,request:t}){if(t&&"navigate"!==t.mode)return!1;let a=e.pathname+e.search;for(let e of this._denylist)if(e.test(a))return!1;return!!this._allowlist.some(e=>e.test(a))}}class es extends ee{constructor(e,t,a){super(({url:t})=>{let a=e.exec(t.href);if(a)return t.origin!==location.origin&&0!==a.index?void 0:a.slice(1)},t,a)}}let er=e=>{if(!e)throw new i("add-to-cache-list-unexpected-type",{entry:e});if("string"==typeof e){let t=new URL(e,location.href);return{cacheKey:t.href,url:t.href}}let{revision:t,url:a}=e;if(!a)throw new i("add-to-cache-list-unexpected-type",{entry:e});if(!t){let e=new URL(a,location.href);return{cacheKey:e.href,url:e.href}}let s=new URL(a,location.href),r=new URL(a,location.href);return s.searchParams.set("__WB_REVISION__",t),{cacheKey:s.href,url:r.href}};class en{updatedURLs=[];notUpdatedURLs=[];handlerWillStart=async({request:e,state:t})=>{t&&(t.originalRequest=e)};cachedResponseWillBeUsed=async({event:e,state:t,cachedResponse:a})=>{if("install"===e.type&&t?.originalRequest&&t.originalRequest instanceof Request){let e=t.originalRequest.url;a?this.notUpdatedURLs.push(e):this.updatedURLs.push(e)}return a}}let ei=async(e,t,a)=>{let s=t.map((e,t)=>({index:t,item:e})),r=async e=>{let t=[];for(;;){let r=s.pop();if(!r)return e(t);let n=await a(r.item);t.push({result:n,index:r.index})}},n=Array.from({length:e},()=>new Promise(r));return(await Promise.all(n)).flat().sort((e,t)=>e.indexe.result)};"undefined"!=typeof navigator&&/^((?!chrome|android).)*safari/i.test(navigator.userAgent);let ec="cache-entries",eo=e=>{let t=new URL(e,location.href);return t.hash="",t.href};class el{_cacheName;_db=null;constructor(e){this._cacheName=e}_getId(e){return`${this._cacheName}|${eo(e)}`}_upgradeDb(e){let t=e.createObjectStore(ec,{keyPath:"id"});t.createIndex("cacheName","cacheName",{unique:!1}),t.createIndex("timestamp","timestamp",{unique:!1})}_upgradeDbAndDeleteOldDbs(e){this._upgradeDb(e),this._cacheName&&function(e,{blocked:t}={}){let a=indexedDB.deleteDatabase(e);t&&a.addEventListener("blocked",e=>t(e.oldVersion,e)),R(a).then(()=>void 0)}(this._cacheName)}async setTimestamp(e,t){e=eo(e);let a={id:this._getId(e),cacheName:this._cacheName,url:e,timestamp:t},s=(await this.getDb()).transaction(ec,"readwrite",{durability:"relaxed"});await s.store.put(a),await s.done}async getTimestamp(e){let t=await this.getDb(),a=await t.get(ec,this._getId(e));return a?.timestamp}async expireEntries(e,t){let a=await this.getDb(),s=await a.transaction(ec,"readwrite").store.index("timestamp").openCursor(null,"prev"),r=[],n=0;for(;s;){let a=s.value;a.cacheName===this._cacheName&&(e&&a.timestamp=t?(s.delete(),r.push(a.url)):n++),s=await s.continue()}return r}async getDb(){return this._db||(this._db=await x("serwist-expiration",1,{upgrade:this._upgradeDbAndDeleteOldDbs.bind(this)})),this._db}}class eh{_isRunning=!1;_rerunRequested=!1;_maxEntries;_maxAgeSeconds;_matchOptions;_cacheName;_timestampModel;constructor(e,t={}){this._maxEntries=t.maxEntries,this._maxAgeSeconds=t.maxAgeSeconds,this._matchOptions=t.matchOptions,this._cacheName=e,this._timestampModel=new el(e)}async expireEntries(){if(this._isRunning){this._rerunRequested=!0;return}this._isRunning=!0;let e=this._maxAgeSeconds?Date.now()-1e3*this._maxAgeSeconds:0,t=await this._timestampModel.expireEntries(e,this._maxEntries),a=await self.caches.open(this._cacheName);for(let e of t)await a.delete(e,this._matchOptions);this._isRunning=!1,this._rerunRequested&&(this._rerunRequested=!1,this.expireEntries())}async updateTimestamp(e){await this._timestampModel.setTimestamp(e,Date.now())}async isURLExpired(e){if(!this._maxAgeSeconds)return!1;let t=await this._timestampModel.getTimestamp(e),a=Date.now()-1e3*this._maxAgeSeconds;return void 0===t||tthis.deleteCacheAndMetadata(),o.add(t))}_getCacheExpiration(e){if(e===n.getRuntimeName())throw new i("expire-custom-caches-only");let t=this._cacheExpirations.get(e);return t||(t=new eh(e,this._config),this._cacheExpirations.set(e,t)),t}cachedResponseWillBeUsed({event:e,cacheName:t,request:a,cachedResponse:s}){if(!s)return null;let r=this._isResponseDateFresh(s),n=this._getCacheExpiration(t),i="last-used"===this._config.maxAgeFrom,c=(async()=>{i&&await n.updateTimestamp(a.url),await n.expireEntries()})();try{e.waitUntil(c)}catch{}return r?s:null}_isResponseDateFresh(e){if("last-used"===this._config.maxAgeFrom)return!0;let t=Date.now();if(!this._config.maxAgeSeconds)return!0;let a=this._getDateHeaderTimestamp(e);return null===a||a>=t-1e3*this._config.maxAgeSeconds}_getDateHeaderTimestamp(e){if(!e.headers.has("date"))return null;let t=new Date(e.headers.get("date")).getTime();return Number.isNaN(t)?null:t}async cacheDidUpdate({cacheName:e,request:t}){let a=this._getCacheExpiration(e);await a.updateTimestamp(t.url),await a.expireEntries()}async deleteCacheAndMetadata(){for(let[e,t]of this._cacheExpirations)await self.caches.delete(e),await t.delete();this._cacheExpirations=new Map}}let ed="www.google-analytics.com",em="www.googletagmanager.com",eg=/^\/(\w+\/)?collect/,ep=({serwist:e,cacheName:t,...a})=>{let s=n.getGoogleAnalyticsName(t),r=new z("serwist-google-analytics",{maxRetentionTime:2880,onSync:(e=>async({queue:t})=>{let a;for(;a=await t.shiftRequest();){let{request:s,timestamp:r}=a,n=new URL(s.url);try{let t="POST"===s.method?new URLSearchParams(await s.clone().text()):n.searchParams,a=r-(Number(t.get("qt"))||0),i=Date.now()-a;if(t.set("qt",String(i)),e.parameterOverrides)for(let a of Object.keys(e.parameterOverrides)){let s=e.parameterOverrides[a];t.set(a,s)}"function"==typeof e.hitFilter&&e.hitFilter.call(null,t),await fetch(new Request(n.origin+n.pathname,{body:t.toString(),method:"POST",mode:"cors",credentials:"omit",headers:{"Content-Type":"text/plain"}}))}catch(e){throw await t.unshiftRequest(a),e}}})(a)});for(let t of[new ee(({url:e})=>e.hostname===em&&"/gtm.js"===e.pathname,new X({cacheName:s}),"GET"),new ee(({url:e})=>e.hostname===ed&&"/analytics.js"===e.pathname,new X({cacheName:s}),"GET"),new ee(({url:e})=>e.hostname===em&&"/gtag/js"===e.pathname,new X({cacheName:s}),"GET"),...(e=>{let t=({url:e})=>e.hostname===ed&&eg.test(e.pathname),a=new Y({plugins:[e]});return[new ee(t,a,"GET"),new ee(t,a,"POST")]})(r)])e.registerRoute(t)};class ef{_fallbackUrls;_serwist;constructor({fallbackUrls:e,serwist:t}){this._fallbackUrls=e,this._serwist=t}async handlerDidError(e){for(let t of this._fallbackUrls)if("string"==typeof t){let e=await this._serwist.matchPrecache(t);if(void 0!==e)return e}else if(t.matcher(e)){let e=await this._serwist.matchPrecache(t.url);if(void 0!==e)return e}}}let ew=async(e,t)=>{try{if(206===t.status)return t;let a=e.headers.get("range");if(!a)throw new i("no-range-header");let s=(e=>{let t=e.trim().toLowerCase();if(!t.startsWith("bytes="))throw new i("unit-must-be-bytes",{normalizedRangeHeader:t});if(t.includes(","))throw new i("single-range-only",{normalizedRangeHeader:t});let a=/(\d*)-(\d*)/.exec(t);if(!a||!(a[1]||a[2]))throw new i("invalid-range-values",{normalizedRangeHeader:t});return{start:""===a[1]?void 0:Number(a[1]),end:""===a[2]?void 0:Number(a[2])}})(a),r=await t.blob(),n=((e,t,a)=>{let s,r,n=e.size;if(a&&a>n||t&&t<0)throw new i("range-not-satisfiable",{size:n,end:a,start:t});return void 0!==t&&void 0!==a?(s=t,r=a+1):void 0!==t&&void 0===a?(s=t,r=n):void 0!==a&&void 0===t&&(s=n-a,r=n),{start:s,end:r}})(r,s.start,s.end),c=r.slice(n.start,n.end),o=c.size,l=new Response(c,{status:206,statusText:"Partial Content",headers:t.headers});return l.headers.set("Content-Length",String(o)),l.headers.set("Content-Range",`bytes ${n.start}-${n.end-1}/${r.size}`),l}catch(e){return new Response("",{status:416,statusText:"Range Not Satisfiable"})}};class ey{cachedResponseWillBeUsed=async({request:e,cachedResponse:t})=>t&&e.headers.has("range")?await ew(e,t):t}class e_ extends J{async _handle(e,t){let a,s=await t.cacheMatch(e);if(!s)try{s=await t.fetchAndCachePut(e)}catch(e){e instanceof Error&&(a=e)}if(!s)throw new i("no-response",{url:e.url,error:a});return s}}class eb extends J{constructor(e={}){super(e),this.plugins.some(e=>"cacheWillUpdate"in e)||this.plugins.unshift(G)}async _handle(e,t){let a,s=t.fetchAndCachePut(e).catch(()=>{});t.waitUntil(s);let r=await t.cacheMatch(e);if(r);else try{r=await s}catch(e){e instanceof Error&&(a=e)}if(!r)throw new i("no-response",{url:e.url,error:a});return r}}class eR extends ee{constructor(e,t){super(({request:a})=>{let s=e.getUrlsToPrecacheKeys();for(let r of function*(e,{directoryIndex:t="index.html",ignoreURLParametersMatching:a=[/^utm_/,/^fbclid$/],cleanURLs:s=!0,urlManipulation:r}={}){let n=new URL(e,location.href);n.hash="",yield n.href;let i=((e,t=[])=>{for(let a of[...e.searchParams.keys()])t.some(e=>e.test(a))&&e.searchParams.delete(a);return e})(n,a);if(yield i.href,t&&i.pathname.endsWith("/")){let e=new URL(i.href);e.pathname+=t,yield e.href}if(s){let e=new URL(i.href);e.pathname+=".html",yield e.href}if(r)for(let e of r({url:n}))yield e.href}(a.url,t)){let t=s.get(r);if(t){let a=e.getIntegrityForPrecacheKey(t);return{cacheKey:t,integrity:a}}}},e.precacheStrategy)}}class ev{_precacheController;constructor({precacheController:e}){this._precacheController=e}cacheKeyWillBeUsed=async({request:e,params:t})=>{let a=t?.cacheKey||this._precacheController.getPrecacheKeyForUrl(e.url);return a?new Request(a,{headers:e.headers}):e}}class ex{_urlsToCacheKeys=new Map;_urlsToCacheModes=new Map;_cacheKeysToIntegrities=new Map;_concurrentPrecaching;_precacheStrategy;_routes;_defaultHandlerMap;_catchHandler;_requestRules;constructor({precacheEntries:e,precacheOptions:t,skipWaiting:a=!1,importScripts:s,navigationPreload:r=!1,cacheId:i,clientsClaim:c=!1,runtimeCaching:o,offlineAnalyticsConfig:l,disableDevLogs:h=!1,fallbacks:u,requestRules:d}={}){var m,p;let{precacheStrategyOptions:f,precacheRouteOptions:w,precacheMiscOptions:y}=((e,t={})=>{let{cacheName:a,plugins:s=[],fetchOptions:r,matchOptions:i,fallbackToNetwork:c,directoryIndex:o,ignoreURLParametersMatching:l,cleanURLs:h,urlManipulation:u,cleanupOutdatedCaches:d,concurrency:m=10,navigateFallback:g,navigateFallbackAllowlist:p,navigateFallbackDenylist:f}=t??{};return{precacheStrategyOptions:{cacheName:n.getPrecacheName(a),plugins:[...s,new ev({precacheController:e})],fetchOptions:r,matchOptions:i,fallbackToNetwork:c},precacheRouteOptions:{directoryIndex:o,ignoreURLParametersMatching:l,cleanURLs:h,urlManipulation:u},precacheMiscOptions:{cleanupOutdatedCaches:d,concurrency:m,navigateFallback:g,navigateFallbackAllowlist:p,navigateFallbackDenylist:f}}})(this,t);if(this._concurrentPrecaching=y.concurrency,this._precacheStrategy=new et(f),this._routes=new Map,this._defaultHandlerMap=new Map,this._requestRules=d,this.handleInstall=this.handleInstall.bind(this),this.handleActivate=this.handleActivate.bind(this),this.handleFetch=this.handleFetch.bind(this),this.handleCache=this.handleCache.bind(this),s&&s.length>0&&self.importScripts(...s),r&&self.registration?.navigationPreload&&self.addEventListener("activate",e=>{e.waitUntil(self.registration.navigationPreload.enable().then(()=>{}))}),void 0!==i&&(m={prefix:i},n.updateDetails(m)),a?self.skipWaiting():self.addEventListener("message",e=>{e.data&&"SKIP_WAITING"===e.data.type&&self.skipWaiting()}),c&&self.addEventListener("activate",()=>self.clients.claim()),e&&e.length>0&&this.addToPrecacheList(e),y.cleanupOutdatedCaches&&(p=f.cacheName,self.addEventListener("activate",e=>{e.waitUntil(g(n.getPrecacheName(p)).then(e=>{}))})),this.registerRoute(new eR(this,w)),y.navigateFallback&&this.registerRoute(new ea(this.createHandlerBoundToUrl(y.navigateFallback),{allowlist:y.navigateFallbackAllowlist,denylist:y.navigateFallbackDenylist})),void 0!==l&&("boolean"==typeof l?l&&ep({serwist:this}):ep({...l,serwist:this})),void 0!==o){if(void 0!==u){let e=new ef({fallbackUrls:u.entries,serwist:this});o.forEach(t=>{t.handler instanceof J&&!t.handler.plugins.some(e=>"handlerDidError"in e)&&t.handler.plugins.push(e)})}for(let e of o)this.registerCapture(e.matcher,e.handler,e.method)}h&&(self.__WB_DISABLE_DEV_LOGS=!0)}get precacheStrategy(){return this._precacheStrategy}get routes(){return this._routes}addEventListeners(){self.addEventListener("install",this.handleInstall),self.addEventListener("activate",this.handleActivate),self.addEventListener("fetch",this.handleFetch),self.addEventListener("message",this.handleCache)}addToPrecacheList(e){let t=[];for(let a of e){"string"==typeof a?t.push(a):a&&!a.integrity&&void 0===a.revision&&t.push(a.url);let{cacheKey:e,url:s}=er(a),r="string"!=typeof a&&a.revision?"reload":"default";if(this._urlsToCacheKeys.has(s)&&this._urlsToCacheKeys.get(s)!==e)throw new i("add-to-cache-list-conflicting-entries",{firstEntry:this._urlsToCacheKeys.get(s),secondEntry:e});if("string"!=typeof a&&a.integrity){if(this._cacheKeysToIntegrities.has(e)&&this._cacheKeysToIntegrities.get(e)!==a.integrity)throw new i("add-to-cache-list-conflicting-integrities",{url:s});this._cacheKeysToIntegrities.set(e,a.integrity)}this._urlsToCacheKeys.set(s,e),this._urlsToCacheModes.set(s,r)}t.length>0&&console.warn(`Serwist is precaching URLs without revision info: ${t.join(", ")} -This is generally NOT safe. Learn more at https://bit.ly/wb-precache`)}handleInstall(e){return this.registerRequestRules(e),p(e,async()=>{let t=new en;this.precacheStrategy.plugins.push(t),await ei(this._concurrentPrecaching,Array.from(this._urlsToCacheKeys.entries()),async([t,a])=>{let s=this._cacheKeysToIntegrities.get(a),r=this._urlsToCacheModes.get(t),n=new Request(t,{integrity:s,cache:r,credentials:"same-origin"});await Promise.all(this.precacheStrategy.handleAll({event:e,request:n,url:new URL(n.url),params:{cacheKey:a}}))});let{updatedURLs:a,notUpdatedURLs:s}=t;return{updatedURLs:a,notUpdatedURLs:s}})}async registerRequestRules(e){if(this._requestRules&&e?.addRoutes)try{await e.addRoutes(this._requestRules),this._requestRules=void 0}catch(e){throw e}}handleActivate(e){return p(e,async()=>{let e=await self.caches.open(this.precacheStrategy.cacheName),t=await e.keys(),a=new Set(this._urlsToCacheKeys.values()),s=[];for(let r of t)a.has(r.url)||(await e.delete(r),s.push(r.url));return{deletedCacheRequests:s}})}handleFetch(e){let{request:t}=e,a=this.handleRequest({request:t,event:e});a&&e.respondWith(a)}handleCache(e){if(e.data&&"CACHE_URLS"===e.data.type){let{payload:t}=e.data,a=Promise.all(t.urlsToCache.map(t=>{let a;return a="string"==typeof t?new Request(t):new Request(...t),this.handleRequest({request:a,event:e})}));e.waitUntil(a),e.ports?.[0]&&a.then(()=>e.ports[0].postMessage(!0))}}setDefaultHandler(e,t="GET"){this._defaultHandlerMap.set(t,Z(e))}setCatchHandler(e){this._catchHandler=Z(e)}registerCapture(e,t,a){let s=((e,t,a)=>{if("string"==typeof e){let s=new URL(e,location.href);return new ee(({url:e})=>e.href===s.href,t,a)}if(e instanceof RegExp)return new es(e,t,a);if("function"==typeof e)return new ee(e,t,a);if(e instanceof ee)return e;throw new i("unsupported-route-type",{moduleName:"serwist",funcName:"parseRoute",paramName:"capture"})})(e,t,a);return this.registerRoute(s),s}registerRoute(e){this._routes.has(e.method)||this._routes.set(e.method,[]),this._routes.get(e.method).push(e)}unregisterRoute(e){if(!this._routes.has(e.method))throw new i("unregister-route-but-not-found-with-method",{method:e.method});let t=this._routes.get(e.method).indexOf(e);if(t>-1)this._routes.get(e.method).splice(t,1);else throw new i("unregister-route-route-not-registered")}getUrlsToPrecacheKeys(){return this._urlsToCacheKeys}getPrecachedUrls(){return[...this._urlsToCacheKeys.keys()]}getPrecacheKeyForUrl(e){let t=new URL(e,location.href);return this._urlsToCacheKeys.get(t.href)}getIntegrityForPrecacheKey(e){return this._cacheKeysToIntegrities.get(e)}async matchPrecache(e){let t=e instanceof Request?e.url:e,a=this.getPrecacheKeyForUrl(t);if(a)return(await self.caches.open(this.precacheStrategy.cacheName)).match(a)}createHandlerBoundToUrl(e){let t=this.getPrecacheKeyForUrl(e);if(!t)throw new i("non-precached-url",{url:e});return a=>(a.request=new Request(e),a.params={cacheKey:t,...a.params},this.precacheStrategy.handle(a))}handleRequest({request:e,event:t}){let a,s=new URL(e.url,location.href);if(!s.protocol.startsWith("http"))return;let r=s.origin===location.origin,{params:n,route:i}=this.findMatchingRoute({event:t,request:e,sameOrigin:r,url:s}),c=i?.handler,o=e.method;if(!c&&this._defaultHandlerMap.has(o)&&(c=this._defaultHandlerMap.get(o)),!c)return;try{a=c.handle({url:s,request:e,event:t,params:n})}catch(e){a=Promise.reject(e)}let l=i?.catchHandler;return a instanceof Promise&&(this._catchHandler||l)&&(a=a.catch(async a=>{if(l)try{return await l.handle({url:s,request:e,event:t,params:n})}catch(e){e instanceof Error&&(a=e)}if(this._catchHandler)return this._catchHandler.handle({url:s,request:e,event:t});throw a})),a}findMatchingRoute({url:e,sameOrigin:t,request:a,event:s}){for(let r of this._routes.get(a.method)||[]){let n,i=r.match({url:e,sameOrigin:t,request:a,event:s});if(i)return Array.isArray(n=i)&&0===n.length||i.constructor===Object&&0===Object.keys(i).length?n=void 0:"boolean"==typeof i&&(n=void 0),{route:r,params:n}}return{}}}let eE={rscPrefetch:"pages-rsc-prefetch",rsc:"pages-rsc",html:"pages"},eS=[{matcher:/^https:\/\/fonts\.(?:gstatic)\.com\/.*/i,handler:new e_({cacheName:"google-fonts-webfonts",plugins:[new eu({maxEntries:4,maxAgeSeconds:31536e3,maxAgeFrom:"last-used"})]})},{matcher:/^https:\/\/fonts\.(?:googleapis)\.com\/.*/i,handler:new eb({cacheName:"google-fonts-stylesheets",plugins:[new eu({maxEntries:4,maxAgeSeconds:604800,maxAgeFrom:"last-used"})]})},{matcher:/\.(?:eot|otf|ttc|ttf|woff|woff2|font.css)$/i,handler:new eb({cacheName:"static-font-assets",plugins:[new eu({maxEntries:4,maxAgeSeconds:604800,maxAgeFrom:"last-used"})]})},{matcher:/\.(?:jpg|jpeg|gif|png|svg|ico|webp)$/i,handler:new eb({cacheName:"static-image-assets",plugins:[new eu({maxEntries:64,maxAgeSeconds:2592e3,maxAgeFrom:"last-used"})]})},{matcher:/\/_next\/static.+\.js$/i,handler:new e_({cacheName:"next-static-js-assets",plugins:[new eu({maxEntries:64,maxAgeSeconds:86400,maxAgeFrom:"last-used"})]})},{matcher:/\/_next\/image\?url=.+$/i,handler:new eb({cacheName:"next-image",plugins:[new eu({maxEntries:64,maxAgeSeconds:86400,maxAgeFrom:"last-used"})]})},{matcher:/\.(?:mp3|wav|ogg)$/i,handler:new e_({cacheName:"static-audio-assets",plugins:[new eu({maxEntries:32,maxAgeSeconds:86400,maxAgeFrom:"last-used"}),new ey]})},{matcher:/\.(?:mp4|webm)$/i,handler:new e_({cacheName:"static-video-assets",plugins:[new eu({maxEntries:32,maxAgeSeconds:86400,maxAgeFrom:"last-used"}),new ey]})},{matcher:/\.(?:js)$/i,handler:new eb({cacheName:"static-js-assets",plugins:[new eu({maxEntries:48,maxAgeSeconds:86400,maxAgeFrom:"last-used"})]})},{matcher:/\.(?:css|less)$/i,handler:new eb({cacheName:"static-style-assets",plugins:[new eu({maxEntries:32,maxAgeSeconds:86400,maxAgeFrom:"last-used"})]})},{matcher:/\/_next\/data\/.+\/.+\.json$/i,handler:new X({cacheName:"next-data",plugins:[new eu({maxEntries:32,maxAgeSeconds:86400,maxAgeFrom:"last-used"})]})},{matcher:/\.(?:json|xml|csv)$/i,handler:new X({cacheName:"static-data-assets",plugins:[new eu({maxEntries:32,maxAgeSeconds:86400,maxAgeFrom:"last-used"})]})},{matcher:/\/api\/auth\/.*/,handler:new Y({networkTimeoutSeconds:10})},{matcher:({sameOrigin:e,url:{pathname:t}})=>e&&t.startsWith("/api/"),method:"GET",handler:new X({cacheName:"apis",plugins:[new eu({maxEntries:16,maxAgeSeconds:86400,maxAgeFrom:"last-used"})],networkTimeoutSeconds:10})},{matcher:({request:e,url:{pathname:t},sameOrigin:a})=>"1"===e.headers.get("RSC")&&"1"===e.headers.get("Next-Router-Prefetch")&&a&&!t.startsWith("/api/"),handler:new X({cacheName:eE.rscPrefetch,plugins:[new eu({maxEntries:32,maxAgeSeconds:86400})]})},{matcher:({request:e,url:{pathname:t},sameOrigin:a})=>"1"===e.headers.get("RSC")&&a&&!t.startsWith("/api/"),handler:new X({cacheName:eE.rsc,plugins:[new eu({maxEntries:32,maxAgeSeconds:86400})]})},{matcher:({request:e,url:{pathname:t},sameOrigin:a})=>e.headers.get("Content-Type")?.includes("text/html")&&a&&!t.startsWith("/api/"),handler:new X({cacheName:eE.html,plugins:[new eu({maxEntries:32,maxAgeSeconds:86400})]})},{matcher:({url:{pathname:e},sameOrigin:t})=>t&&!e.startsWith("/api/"),handler:new X({cacheName:"others",plugins:[new eu({maxEntries:32,maxAgeSeconds:86400})]})},{matcher:({sameOrigin:e})=>!e,handler:new X({cacheName:"cross-origin",plugins:[new eu({maxEntries:32,maxAgeSeconds:3600})],networkTimeoutSeconds:10})},{matcher:/.*/i,method:"GET",handler:new Y}],eq={cacheNames:{static:"tour-builder-static-v1",dynamic:"tour-builder-dynamic-v1",assets:"tour-builder-assets-v1"}},eC=[".png",".jpg",".jpeg",".gif",".webp",".svg",".ico",".mp4",".webm",".mov",".mp3",".wav",".ogg",".m4a",".woff",".woff2",".ttf",".eot",".css",".js"],eN=e=>{let t=new URL(e.url);return[".mp4",".webm",".mov"].some(e=>t.pathname.toLowerCase().endsWith(e))},eD=e=>{let t=new URL(e.url);return[".mp3",".wav",".ogg",".m4a",".aac"].some(e=>t.pathname.toLowerCase().endsWith(e))},eT=e=>{try{if(e.includes("/file/download?privateUrl=")){let t=e.match(/privateUrl=([^&]+)/);if(t)return decodeURIComponent(t[1]).replace(/^\/+/,"")}if(e.includes("X-Amz-Signature=")||e.includes("x-amz-signature=")){let t=new URL(e).pathname.split("/").filter(Boolean),a=t.findIndex(e=>"assets"===e);if(-1!==a)return t.slice(a).join("/");if(t.length>1)return t.slice(1).join("/")}if(e.startsWith("assets/"))return e;let t=e.match(/^https?:\/\/[^/]+\.s3\.[^/]+\.amazonaws\.com\/[^/]+\/(assets\/.+)$/);if(t)return t[1].split("?")[0];return null}catch(e){return null}},eP=new Map;setInterval(()=>{eP.clear(),console.log("[SW] Cleared storage path mappings")},36e5);let eU=new ex({precacheEntries:[{'revision':'48e9a00892afb3b6ad229718e9bed021','url':'/_next/static/-PbEwkiFN4JCq5Uqr-1sT/_buildManifest.js'},{'revision':'b6652df95db52feb4daf4eca35380933','url':'/_next/static/-PbEwkiFN4JCq5Uqr-1sT/_ssgManifest.js'},{'revision':null,'url':'/_next/static/chunks/12142dde-27563c775e4f8cf0.js'},{'revision':null,'url':'/_next/static/chunks/1232-ee492d53bb7b1303.js'},{'revision':null,'url':'/_next/static/chunks/1818-b3660c83cef1d173.js'},{'revision':null,'url':'/_next/static/chunks/223-5e7611f0ad4bb7f3.js'},{'revision':null,'url':'/_next/static/chunks/2450-c991dd9dfbd7be31.js'},{'revision':null,'url':'/_next/static/chunks/3643-cd854ecad4d47645.js'},{'revision':null,'url':'/_next/static/chunks/400-ac29014eab06ed0b.js'},{'revision':null,'url':'/_next/static/chunks/4166-2355014a69094c93.js'},{'revision':null,'url':'/_next/static/chunks/4271-b6985491cfc2640a.js'},{'revision':null,'url':'/_next/static/chunks/4449-8c19077600e94603.js'},{'revision':null,'url':'/_next/static/chunks/4498-03cdfe804ff7affb.js'},{'revision':null,'url':'/_next/static/chunks/4587-c9e5910a896d025b.js'},{'revision':null,'url':'/_next/static/chunks/5371-1944e2cec48f4711.js'},{'revision':null,'url':'/_next/static/chunks/5541.e90be83844a6de15.js'},{'revision':null,'url':'/_next/static/chunks/5926-dd7d5959a02a865e.js'},{'revision':null,'url':'/_next/static/chunks/6d2b60a9-eb6c7fd9a57c4f19.js'},{'revision':null,'url':'/_next/static/chunks/7e42aecb-94f8c450c54b9556.js'},{'revision':null,'url':'/_next/static/chunks/8230-251e23b7a24ff307.js'},{'revision':null,'url':'/_next/static/chunks/8232-06043c4b3efac0d3.js'},{'revision':null,'url':'/_next/static/chunks/8317-4e9c090ccc089653.js'},{'revision':null,'url':'/_next/static/chunks/8628-4294e63d8a7907d5.js'},{'revision':null,'url':'/_next/static/chunks/8671-1ccce6aa57c8d062.js'},{'revision':null,'url':'/_next/static/chunks/9375.2c053880845a1e8b.js'},{'revision':null,'url':'/_next/static/chunks/9848-799d062feeef8c3c.js'},{'revision':null,'url':'/_next/static/chunks/fa3de7d5.7acfffddc0ff677e.js'},{'revision':null,'url':'/_next/static/chunks/framework-4dea986807dc9d02.js'},{'revision':null,'url':'/_next/static/chunks/main-f35af714f1aa0b25.js'},{'revision':null,'url':'/_next/static/chunks/pages/_error-530d8245847656c6.js'},{'revision':null,'url':'/_next/static/chunks/pages/access_logs/%5Baccess_logsId%5D-8d339e7e896d096c.js'},{'revision':null,'url':'/_next/static/chunks/pages/access_logs/access_logs-edit-35566af647779ab1.js'},{'revision':null,'url':'/_next/static/chunks/pages/access_logs/access_logs-list-b7b1e92379006764.js'},{'revision':null,'url':'/_next/static/chunks/pages/access_logs/access_logs-new-1c0b08a5d9c083df.js'},{'revision':null,'url':'/_next/static/chunks/pages/access_logs/access_logs-table-97bbef6e9c64cb6a.js'},{'revision':null,'url':'/_next/static/chunks/pages/access_logs/access_logs-view-a1df383934285298.js'},{'revision':null,'url':'/_next/static/chunks/pages/asset_variants/%5Basset_variantsId%5D-f30f413d0d49c7e4.js'},{'revision':null,'url':'/_next/static/chunks/pages/asset_variants/asset_variants-edit-72af829781a60809.js'},{'revision':null,'url':'/_next/static/chunks/pages/asset_variants/asset_variants-list-cf3da7bb076aa6f0.js'},{'revision':null,'url':'/_next/static/chunks/pages/asset_variants/asset_variants-new-3665f1d0e6e84dd4.js'},{'revision':null,'url':'/_next/static/chunks/pages/asset_variants/asset_variants-table-ee30957ff8d738c7.js'},{'revision':null,'url':'/_next/static/chunks/pages/asset_variants/asset_variants-view-cc2e3bbc0af9ae2c.js'},{'revision':null,'url':'/_next/static/chunks/pages/assets/%5BassetsId%5D-c76bec5d8cc7a727.js'},{'revision':null,'url':'/_next/static/chunks/pages/assets/assets-edit-244a4f0d1e658028.js'},{'revision':null,'url':'/_next/static/chunks/pages/assets/assets-list-7efe17782be4c508.js'},{'revision':null,'url':'/_next/static/chunks/pages/assets/assets-new-4befa34e01efba55.js'},{'revision':null,'url':'/_next/static/chunks/pages/assets/assets-table-5c322503011d3c30.js'},{'revision':null,'url':'/_next/static/chunks/pages/assets/assets-view-89c254b1127feb02.js'},{'revision':null,'url':'/_next/static/chunks/pages/constructor-facb32c487338a67.js'},{'revision':null,'url':'/_next/static/chunks/pages/dashboard-ef173dde9b60a5fc.js'},{'revision':null,'url':'/_next/static/chunks/pages/element-type-defaults-5ab16a411a1c84a0.js'},{'revision':null,'url':'/_next/static/chunks/pages/element-type-defaults/%5Bid%5D-bb582ba5e9158901.js'},{'revision':null,'url':'/_next/static/chunks/pages/error-e0bd1c24aeb3d601.js'},{'revision':null,'url':'/_next/static/chunks/pages/forgot-78b70d91742fd5b1.js'},{'revision':null,'url':'/_next/static/chunks/pages/index-53587e829512c859.js'},{'revision':null,'url':'/_next/static/chunks/pages/login-79ec11c8552cf796.js'},{'revision':null,'url':'/_next/static/chunks/pages/p/%5BprojectSlug%5D-b50eefa1596a8b74.js'},{'revision':null,'url':'/_next/static/chunks/pages/p/%5BprojectSlug%5D/stage-822175880ff64ec0.js'},{'revision':null,'url':'/_next/static/chunks/pages/password-reset-18f74e912f3914f5.js'},{'revision':null,'url':'/_next/static/chunks/pages/permissions/%5BpermissionsId%5D-03b8d6ce2746e5ac.js'},{'revision':null,'url':'/_next/static/chunks/pages/permissions/permissions-edit-35f3e6f87eb315d5.js'},{'revision':null,'url':'/_next/static/chunks/pages/permissions/permissions-list-155df5771dec31f4.js'},{'revision':null,'url':'/_next/static/chunks/pages/permissions/permissions-new-215d5a028e89340a.js'},{'revision':null,'url':'/_next/static/chunks/pages/permissions/permissions-table-f664d12fd7bc2c5d.js'},{'revision':null,'url':'/_next/static/chunks/pages/permissions/permissions-view-2f8c6ca80f30e4e7.js'},{'revision':null,'url':'/_next/static/chunks/pages/presigned_url_requests/%5Bpresigned_url_requestsId%5D-4d3473c7577b4b0d.js'},{'revision':null,'url':'/_next/static/chunks/pages/presigned_url_requests/presigned_url_requests-edit-52cb8b59b5fce4e7.js'},{'revision':null,'url':'/_next/static/chunks/pages/presigned_url_requests/presigned_url_requests-list-736e04880898639a.js'},{'revision':null,'url':'/_next/static/chunks/pages/presigned_url_requests/presigned_url_requests-new-baaf991bd748200f.js'},{'revision':null,'url':'/_next/static/chunks/pages/presigned_url_requests/presigned_url_requests-table-0147792af7b6e2ca.js'},{'revision':null,'url':'/_next/static/chunks/pages/presigned_url_requests/presigned_url_requests-view-39a5e0b77ea161c1.js'},{'revision':null,'url':'/_next/static/chunks/pages/privacy-policy-53ea2331c015449b.js'},{'revision':null,'url':'/_next/static/chunks/pages/profile-c1b2c9f67ee299cf.js'},{'revision':null,'url':'/_next/static/chunks/pages/project-element-defaults-41c52e13536a5e19.js'},{'revision':null,'url':'/_next/static/chunks/pages/project-element-defaults/%5Bid%5D-a5b938d6317d0f8a.js'},{'revision':null,'url':'/_next/static/chunks/pages/project_audio_tracks/%5Bproject_audio_tracksId%5D-21a2b75463c7ffdb.js'},{'revision':null,'url':'/_next/static/chunks/pages/project_audio_tracks/project_audio_tracks-edit-de76fb3c1d783efa.js'},{'revision':null,'url':'/_next/static/chunks/pages/project_audio_tracks/project_audio_tracks-list-7444bf0ff2adcf19.js'},{'revision':null,'url':'/_next/static/chunks/pages/project_audio_tracks/project_audio_tracks-new-ba6d6a3f8515b0d7.js'},{'revision':null,'url':'/_next/static/chunks/pages/project_audio_tracks/project_audio_tracks-table-b985f5c250b58ae0.js'},{'revision':null,'url':'/_next/static/chunks/pages/project_audio_tracks/project_audio_tracks-view-f2db64bfe6512eac.js'},{'revision':null,'url':'/_next/static/chunks/pages/project_memberships/%5Bproject_membershipsId%5D-e963682ce1f09381.js'},{'revision':null,'url':'/_next/static/chunks/pages/project_memberships/project_memberships-edit-58f19ec0e1dedb5a.js'},{'revision':null,'url':'/_next/static/chunks/pages/project_memberships/project_memberships-list-12773f6237f73890.js'},{'revision':null,'url':'/_next/static/chunks/pages/project_memberships/project_memberships-new-2e8763e625895fa4.js'},{'revision':null,'url':'/_next/static/chunks/pages/project_memberships/project_memberships-table-cfeb32ab7462cc52.js'},{'revision':null,'url':'/_next/static/chunks/pages/project_memberships/project_memberships-view-8b8a23b0cc93252e.js'},{'revision':null,'url':'/_next/static/chunks/pages/projects/%5BprojectsId%5D-86773b41c7e0155d.js'},{'revision':null,'url':'/_next/static/chunks/pages/projects/projects-edit-d696bd0d2f58534a.js'},{'revision':null,'url':'/_next/static/chunks/pages/projects/projects-list-6b45fca8d4e25a1a.js'},{'revision':null,'url':'/_next/static/chunks/pages/projects/projects-new-fa8a012e1d8c87f3.js'},{'revision':null,'url':'/_next/static/chunks/pages/projects/projects-table-d0b2ccf7259abe27.js'},{'revision':null,'url':'/_next/static/chunks/pages/projects/projects-view-7719886c9aa95720.js'},{'revision':null,'url':'/_next/static/chunks/pages/publish_events/%5Bpublish_eventsId%5D-91bb40dd49d451bd.js'},{'revision':null,'url':'/_next/static/chunks/pages/publish_events/publish_events-edit-5ee3360c4efe4ecc.js'},{'revision':null,'url':'/_next/static/chunks/pages/publish_events/publish_events-list-7dfc48227b634d01.js'},{'revision':null,'url':'/_next/static/chunks/pages/publish_events/publish_events-new-967699a276f0fc1f.js'},{'revision':null,'url':'/_next/static/chunks/pages/publish_events/publish_events-table-b7cb7261af5db570.js'},{'revision':null,'url':'/_next/static/chunks/pages/publish_events/publish_events-view-28f438459443352a.js'},{'revision':null,'url':'/_next/static/chunks/pages/pwa_caches/%5Bpwa_cachesId%5D-12230990c23e7fd8.js'},{'revision':null,'url':'/_next/static/chunks/pages/pwa_caches/pwa_caches-edit-e07b159d3fb8e05a.js'},{'revision':null,'url':'/_next/static/chunks/pages/pwa_caches/pwa_caches-list-ae78031d0263b1b8.js'},{'revision':null,'url':'/_next/static/chunks/pages/pwa_caches/pwa_caches-new-393d3dda89d132c0.js'},{'revision':null,'url':'/_next/static/chunks/pages/pwa_caches/pwa_caches-table-b9a9d80285f6ccf5.js'},{'revision':null,'url':'/_next/static/chunks/pages/pwa_caches/pwa_caches-view-0f23c3f267670577.js'},{'revision':null,'url':'/_next/static/chunks/pages/register-74317198a1107978.js'},{'revision':null,'url':'/_next/static/chunks/pages/roles/%5BrolesId%5D-c0bdb71711f9e8af.js'},{'revision':null,'url':'/_next/static/chunks/pages/roles/roles-edit-b7e1e9f9cdd69669.js'},{'revision':null,'url':'/_next/static/chunks/pages/roles/roles-list-bb2e8266272a34b3.js'},{'revision':null,'url':'/_next/static/chunks/pages/roles/roles-new-f88f533479cddf80.js'},{'revision':null,'url':'/_next/static/chunks/pages/roles/roles-table-2ae06818d79da9ec.js'},{'revision':null,'url':'/_next/static/chunks/pages/roles/roles-view-6f87b5ff9d68835f.js'},{'revision':null,'url':'/_next/static/chunks/pages/search-f2874b266789f090.js'},{'revision':null,'url':'/_next/static/chunks/pages/terms-of-use-7d00698d6183bde4.js'},{'revision':null,'url':'/_next/static/chunks/pages/tour_pages/%5Btour_pagesId%5D-3dc1e444688481a5.js'},{'revision':null,'url':'/_next/static/chunks/pages/tour_pages/tour_pages-edit-8063848d4e94fbe9.js'},{'revision':null,'url':'/_next/static/chunks/pages/tour_pages/tour_pages-list-c12f25262a5eec2c.js'},{'revision':null,'url':'/_next/static/chunks/pages/tour_pages/tour_pages-new-59a0c6ab6948843b.js'},{'revision':null,'url':'/_next/static/chunks/pages/tour_pages/tour_pages-table-20ae6c4ea35a9ffe.js'},{'revision':null,'url':'/_next/static/chunks/pages/tour_pages/tour_pages-view-554ee8551b6d8ad4.js'},{'revision':null,'url':'/_next/static/chunks/pages/users/%5BusersId%5D-29784dfa80b7733a.js'},{'revision':null,'url':'/_next/static/chunks/pages/users/users-edit-2be3c12290c2330b.js'},{'revision':null,'url':'/_next/static/chunks/pages/users/users-list-2185908a5a705b0e.js'},{'revision':null,'url':'/_next/static/chunks/pages/users/users-new-5b4a2f9ec6e25d41.js'},{'revision':null,'url':'/_next/static/chunks/pages/users/users-table-584067057f84845c.js'},{'revision':null,'url':'/_next/static/chunks/pages/users/users-view-6dc01b2679a81960.js'},{'revision':null,'url':'/_next/static/chunks/pages/verify-email-3c9a0ef6c6d5daff.js'},{'revision':'846118c33b2c0e922d7b3a7676f81f6f','url':'/_next/static/chunks/polyfills-42372ed130431b0a.js'},{'revision':null,'url':'/_next/static/chunks/webpack-16d698d3c4c52d85.js'},{'revision':null,'url':'/_next/static/css/715be398208dca58.css'},{'revision':null,'url':'/_next/static/css/ae94e3be50540262.css'},{'revision':null,'url':'/_next/static/css/de6fa09b8a0934d1.css'},{'revision':null,'url':'/_next/static/media/instrument-sans-latin-ext-wdth-normal.a718fc63.woff2'},{'revision':null,'url':'/_next/static/media/instrument-sans-latin-ext-wght-normal.7db92424.woff2'},{'revision':null,'url':'/_next/static/media/instrument-sans-latin-wdth-normal.68c3c527.woff2'},{'revision':null,'url':'/_next/static/media/instrument-sans-latin-wght-normal.ae05c57c.woff2'},{'revision':'43778b43fb039fdd5b0510561dc952ac','url':'/assets/vm-shot-2026-03-17T04-16-09-161Z.jpg'},{'revision':'3395d49fe2b96221471b4db0f5caf9e7','url':'/assets/vm-shot-2026-03-17T04-19-03-565Z.jpg'},{'revision':'62127cd8f85821f3e570a377a8a7e14b','url':'/assets/vm-shot-2026-03-17T04-36-56-252Z.jpg'},{'revision':'145c2d7e4ef298391258c6d8a8aaaece','url':'/assets/vm-shot-2026-03-17T04-45-14-111Z.jpg'},{'revision':'c6aae6521f08c847e370764d1c9c613d','url':'/assets/vm-shot-2026-03-17T04-50-58-546Z.jpg'},{'revision':'bf90fb959c1d418b01eb7e55de6672d5','url':'/assets/vm-shot-2026-03-19T06-12-36-229Z.jpg'},{'revision':'73de6cac0695249f4e59ce778a8e6e74','url':'/assets/vm-shot-2026-03-19T06-13-19-234Z.jpg'},{'revision':'bf90fb959c1d418b01eb7e55de6672d5','url':'/assets/vm-shot-2026-03-19T06-13-54-729Z.jpg'},{'revision':'ca6cbfcc74b52f00eef0f2adc8e65456','url':'/assets/vm-shot-2026-03-24T14-29-20-260Z.jpg'},{'revision':'0957c5365895c5ba31b10a84f4e45929','url':'/data-sources/clients.json'},{'revision':'5703d42f7838705ecf87510c4032b20c','url':'/data-sources/history.json'},{'revision':'5404a85badad8210a634ce41bb511545','url':'/favicon.svg'},{'revision':'508520242399a6b1fec65430901f4e6f','url':'/locales/de/common.json'},{'revision':'0a64739b954a93627749ffcb846fceaa','url':'/locales/en/common.json'},{'revision':'34715e25a3bcbab44f84232ce082d2ee','url':'/locales/es/common.json'},{'revision':'772a72f35589c06bdd8d9179c86459e6','url':'/locales/fr/common.json'},{'revision':'c67326edc61e0b5cf87e509ea7553466','url':'/manifest.json'},{'revision':'1254c9c72aa64724c203d26278712800','url':'/offline.html'}],skipWaiting:!0,clientsClaim:!0,navigationPreload:!0,runtimeCaching:[{matcher:e=>{let{request:t}=e,a=new URL(t.url);return"image"===t.destination||"font"===t.destination||[".css",".js",".woff",".woff2"].some(e=>a.pathname.endsWith(e))},handler:new e_({cacheName:eq.cacheNames.assets,plugins:[{cacheKeyWillBeUsed:async e=>{let{request:t,mode:a}=e,s=eT(t.url);return s?(console.log("[SW] Using storagePath for static asset ".concat(a,":"),s.slice(-40)),new Request(s)):t},cacheWillUpdate:async e=>{let{response:t}=e;return t&&200===t.status?t:null}}]})},{matcher:e=>{let{request:t}=e;return eN(t)},handler:new e_({cacheName:eq.cacheNames.assets,plugins:[{cacheKeyWillBeUsed:async e=>{let{request:t,mode:a}=e,s=eT(t.url);if(s){let e=eP.get(s);return e?(console.log("[SW] Using storageKey for video ".concat(a,":"),e.slice(-40)),new Request(e)):(console.log("[SW] Using storagePath for video ".concat(a,":"),s.slice(-40)),new Request(s))}return t},cachedResponseWillBeUsed:async e=>{let{cachedResponse:t,request:a}=e;if(!t)return null;let s=a.headers.get("range");if(!s)return t;let r=s.match(/bytes=(\d+)-(\d*)/);if(!r)return t;let n=parseInt(r[1],10),i=r[2]?parseInt(r[2],10):void 0,c=await t.blob(),o=void 0!==i?c.slice(n,i+1):c.slice(n);return new Response(o,{status:206,statusText:"Partial Content",headers:{"Content-Type":t.headers.get("Content-Type")||"video/mp4","Content-Length":String(o.size),"Content-Range":"bytes ".concat(n,"-").concat(void 0!==i?i:c.size-1,"/").concat(c.size),"Accept-Ranges":"bytes"}})},cacheWillUpdate:async e=>{let{response:t}=e;return t&&200===t.status?t:null}}]})},{matcher:e=>{let{request:t}=e;return eD(t)},handler:new e_({cacheName:eq.cacheNames.assets,plugins:[{cacheKeyWillBeUsed:async e=>{let{request:t,mode:a}=e,s=eT(t.url);if(s){let e=eP.get(s);return e?(console.log("[SW] Using storageKey for audio ".concat(a,":"),e.slice(-40)),new Request(e)):(console.log("[SW] Using storagePath for audio ".concat(a,":"),s.slice(-40)),new Request(s))}return t},cachedResponseWillBeUsed:async e=>{let{cachedResponse:t,request:a}=e;if(!t)return null;let s=a.headers.get("range");if(!s)return t;let r=s.match(/bytes=(\d+)-(\d*)/);if(!r)return t;let n=parseInt(r[1],10),i=r[2]?parseInt(r[2],10):void 0,c=await t.blob(),o=void 0!==i?c.slice(n,i+1):c.slice(n);return new Response(o,{status:206,statusText:"Partial Content",headers:{"Content-Type":t.headers.get("Content-Type")||"audio/mpeg","Content-Length":String(o.size),"Content-Range":"bytes ".concat(n,"-").concat(void 0!==i?i:c.size-1,"/").concat(c.size),"Accept-Ranges":"bytes"}})},cacheWillUpdate:async e=>{let{response:t}=e;return t&&200===t.status?t:null}}]})},{matcher:e=>{let{url:t}=e;return t.pathname.startsWith("/api/")},handler:new X({cacheName:"api-cache",networkTimeoutSeconds:10})},{matcher:e=>{let{request:t}=e;return(e=>{let t=new URL(e.url);return(!t.pathname.startsWith("/api/")||!!t.pathname.includes("/file/download"))&&!!(eC.some(e=>t.pathname.toLowerCase().endsWith(e))||t.pathname.includes("/file/download")||t.hostname.includes("amazonaws.com")||t.hostname.includes("cloudfront.net"))})(t)&&!eN(t)&&!eD(t)},handler:new e_({cacheName:eq.cacheNames.assets,plugins:[{cacheKeyWillBeUsed:async e=>{let{request:t,mode:a}=e,s=eT(t.url);return s?(console.log("[SW] Using storagePath for dynamic asset ".concat(a,":"),s.slice(-40)),new Request(s)):t},cacheWillUpdate:async e=>{let{response:t}=e;return t&&200===t.status?t:null}}]})},...eS]});self.addEventListener("message",e=>{let{type:t,payload:a}=e.data||{};switch(t){case"REGISTER_CACHE_URL":if(null==a?void 0:a.storageKey){let e=a.presignedUrl&&eT(a.presignedUrl)||a.storageKey;eP.set(e,a.storageKey),console.log("[SW] Registered storage path for caching",{storagePath:e.slice(-50),storageKey:a.storageKey.slice(-50)})}break;case"CLEAR_URL_MAPPINGS":eP.clear(),console.log("[SW] Storage path mappings cleared");break;case"CACHE_ASSETS":Array.isArray(null==a?void 0:a.urls)&&e.waitUntil(caches.open(eq.cacheNames.assets).then(e=>Promise.all(a.urls.map(t=>fetch(t).then(a=>{if(200===a.status)return e.put(t,a)}).catch(e=>{console.warn("[SW] Failed to cache asset:",t,e)})))));break;case"CACHE_VIDEO_CHUNK":(null==a?void 0:a.url)&&(null==a?void 0:a.chunk)&&e.waitUntil(caches.open(eq.cacheNames.assets).then(e=>{let t=new Response(a.chunk,{headers:{"Content-Type":a.contentType||"video/mp4","Content-Length":String(a.chunk.byteLength)}});return e.put(a.url,t)}));break;case"CLEAR_CACHE":eP.clear(),e.waitUntil(Promise.all([caches.delete(eq.cacheNames.dynamic),caches.delete(eq.cacheNames.assets)]).then(()=>{console.log("[SW] Caches and storage path mappings cleared")}));break;case"GET_CACHE_STATUS":e.waitUntil(caches.open(eq.cacheNames.assets).then(t=>t.keys().then(t=>{let a=e.source;null==a||a.postMessage({type:"CACHE_STATUS",payload:{cachedCount:t.length,urls:t.map(e=>e.url)}})})));break;case"SKIP_WAITING":self.skipWaiting()}}),eU.addEventListeners(),console.log("[SW] Serwist service worker loaded")})(); \ No newline at end of file +This is generally NOT safe. Learn more at https://bit.ly/wb-precache`)}handleInstall(e){return this.registerRequestRules(e),p(e,async()=>{let t=new en;this.precacheStrategy.plugins.push(t),await ei(this._concurrentPrecaching,Array.from(this._urlsToCacheKeys.entries()),async([t,a])=>{let s=this._cacheKeysToIntegrities.get(a),r=this._urlsToCacheModes.get(t),n=new Request(t,{integrity:s,cache:r,credentials:"same-origin"});await Promise.all(this.precacheStrategy.handleAll({event:e,request:n,url:new URL(n.url),params:{cacheKey:a}}))});let{updatedURLs:a,notUpdatedURLs:s}=t;return{updatedURLs:a,notUpdatedURLs:s}})}async registerRequestRules(e){if(this._requestRules&&e?.addRoutes)try{await e.addRoutes(this._requestRules),this._requestRules=void 0}catch(e){throw e}}handleActivate(e){return p(e,async()=>{let e=await self.caches.open(this.precacheStrategy.cacheName),t=await e.keys(),a=new Set(this._urlsToCacheKeys.values()),s=[];for(let r of t)a.has(r.url)||(await e.delete(r),s.push(r.url));return{deletedCacheRequests:s}})}handleFetch(e){let{request:t}=e,a=this.handleRequest({request:t,event:e});a&&e.respondWith(a)}handleCache(e){if(e.data&&"CACHE_URLS"===e.data.type){let{payload:t}=e.data,a=Promise.all(t.urlsToCache.map(t=>{let a;return a="string"==typeof t?new Request(t):new Request(...t),this.handleRequest({request:a,event:e})}));e.waitUntil(a),e.ports?.[0]&&a.then(()=>e.ports[0].postMessage(!0))}}setDefaultHandler(e,t="GET"){this._defaultHandlerMap.set(t,Z(e))}setCatchHandler(e){this._catchHandler=Z(e)}registerCapture(e,t,a){let s=((e,t,a)=>{if("string"==typeof e){let s=new URL(e,location.href);return new ee(({url:e})=>e.href===s.href,t,a)}if(e instanceof RegExp)return new es(e,t,a);if("function"==typeof e)return new ee(e,t,a);if(e instanceof ee)return e;throw new i("unsupported-route-type",{moduleName:"serwist",funcName:"parseRoute",paramName:"capture"})})(e,t,a);return this.registerRoute(s),s}registerRoute(e){this._routes.has(e.method)||this._routes.set(e.method,[]),this._routes.get(e.method).push(e)}unregisterRoute(e){if(!this._routes.has(e.method))throw new i("unregister-route-but-not-found-with-method",{method:e.method});let t=this._routes.get(e.method).indexOf(e);if(t>-1)this._routes.get(e.method).splice(t,1);else throw new i("unregister-route-route-not-registered")}getUrlsToPrecacheKeys(){return this._urlsToCacheKeys}getPrecachedUrls(){return[...this._urlsToCacheKeys.keys()]}getPrecacheKeyForUrl(e){let t=new URL(e,location.href);return this._urlsToCacheKeys.get(t.href)}getIntegrityForPrecacheKey(e){return this._cacheKeysToIntegrities.get(e)}async matchPrecache(e){let t=e instanceof Request?e.url:e,a=this.getPrecacheKeyForUrl(t);if(a)return(await self.caches.open(this.precacheStrategy.cacheName)).match(a)}createHandlerBoundToUrl(e){let t=this.getPrecacheKeyForUrl(e);if(!t)throw new i("non-precached-url",{url:e});return a=>(a.request=new Request(e),a.params={cacheKey:t,...a.params},this.precacheStrategy.handle(a))}handleRequest({request:e,event:t}){let a,s=new URL(e.url,location.href);if(!s.protocol.startsWith("http"))return;let r=s.origin===location.origin,{params:n,route:i}=this.findMatchingRoute({event:t,request:e,sameOrigin:r,url:s}),c=i?.handler,o=e.method;if(!c&&this._defaultHandlerMap.has(o)&&(c=this._defaultHandlerMap.get(o)),!c)return;try{a=c.handle({url:s,request:e,event:t,params:n})}catch(e){a=Promise.reject(e)}let l=i?.catchHandler;return a instanceof Promise&&(this._catchHandler||l)&&(a=a.catch(async a=>{if(l)try{return await l.handle({url:s,request:e,event:t,params:n})}catch(e){e instanceof Error&&(a=e)}if(this._catchHandler)return this._catchHandler.handle({url:s,request:e,event:t});throw a})),a}findMatchingRoute({url:e,sameOrigin:t,request:a,event:s}){for(let r of this._routes.get(a.method)||[]){let n,i=r.match({url:e,sameOrigin:t,request:a,event:s});if(i)return Array.isArray(n=i)&&0===n.length||i.constructor===Object&&0===Object.keys(i).length?n=void 0:"boolean"==typeof i&&(n=void 0),{route:r,params:n}}return{}}}let eE={rscPrefetch:"pages-rsc-prefetch",rsc:"pages-rsc",html:"pages"},eS=[{matcher:/^https:\/\/fonts\.(?:gstatic)\.com\/.*/i,handler:new e_({cacheName:"google-fonts-webfonts",plugins:[new eu({maxEntries:4,maxAgeSeconds:31536e3,maxAgeFrom:"last-used"})]})},{matcher:/^https:\/\/fonts\.(?:googleapis)\.com\/.*/i,handler:new eb({cacheName:"google-fonts-stylesheets",plugins:[new eu({maxEntries:4,maxAgeSeconds:604800,maxAgeFrom:"last-used"})]})},{matcher:/\.(?:eot|otf|ttc|ttf|woff|woff2|font.css)$/i,handler:new eb({cacheName:"static-font-assets",plugins:[new eu({maxEntries:4,maxAgeSeconds:604800,maxAgeFrom:"last-used"})]})},{matcher:/\.(?:jpg|jpeg|gif|png|svg|ico|webp)$/i,handler:new eb({cacheName:"static-image-assets",plugins:[new eu({maxEntries:64,maxAgeSeconds:2592e3,maxAgeFrom:"last-used"})]})},{matcher:/\/_next\/static.+\.js$/i,handler:new e_({cacheName:"next-static-js-assets",plugins:[new eu({maxEntries:64,maxAgeSeconds:86400,maxAgeFrom:"last-used"})]})},{matcher:/\/_next\/image\?url=.+$/i,handler:new eb({cacheName:"next-image",plugins:[new eu({maxEntries:64,maxAgeSeconds:86400,maxAgeFrom:"last-used"})]})},{matcher:/\.(?:mp3|wav|ogg)$/i,handler:new e_({cacheName:"static-audio-assets",plugins:[new eu({maxEntries:32,maxAgeSeconds:86400,maxAgeFrom:"last-used"}),new ey]})},{matcher:/\.(?:mp4|webm)$/i,handler:new e_({cacheName:"static-video-assets",plugins:[new eu({maxEntries:32,maxAgeSeconds:86400,maxAgeFrom:"last-used"}),new ey]})},{matcher:/\.(?:js)$/i,handler:new eb({cacheName:"static-js-assets",plugins:[new eu({maxEntries:48,maxAgeSeconds:86400,maxAgeFrom:"last-used"})]})},{matcher:/\.(?:css|less)$/i,handler:new eb({cacheName:"static-style-assets",plugins:[new eu({maxEntries:32,maxAgeSeconds:86400,maxAgeFrom:"last-used"})]})},{matcher:/\/_next\/data\/.+\/.+\.json$/i,handler:new X({cacheName:"next-data",plugins:[new eu({maxEntries:32,maxAgeSeconds:86400,maxAgeFrom:"last-used"})]})},{matcher:/\.(?:json|xml|csv)$/i,handler:new X({cacheName:"static-data-assets",plugins:[new eu({maxEntries:32,maxAgeSeconds:86400,maxAgeFrom:"last-used"})]})},{matcher:/\/api\/auth\/.*/,handler:new Y({networkTimeoutSeconds:10})},{matcher:({sameOrigin:e,url:{pathname:t}})=>e&&t.startsWith("/api/"),method:"GET",handler:new X({cacheName:"apis",plugins:[new eu({maxEntries:16,maxAgeSeconds:86400,maxAgeFrom:"last-used"})],networkTimeoutSeconds:10})},{matcher:({request:e,url:{pathname:t},sameOrigin:a})=>"1"===e.headers.get("RSC")&&"1"===e.headers.get("Next-Router-Prefetch")&&a&&!t.startsWith("/api/"),handler:new X({cacheName:eE.rscPrefetch,plugins:[new eu({maxEntries:32,maxAgeSeconds:86400})]})},{matcher:({request:e,url:{pathname:t},sameOrigin:a})=>"1"===e.headers.get("RSC")&&a&&!t.startsWith("/api/"),handler:new X({cacheName:eE.rsc,plugins:[new eu({maxEntries:32,maxAgeSeconds:86400})]})},{matcher:({request:e,url:{pathname:t},sameOrigin:a})=>e.headers.get("Content-Type")?.includes("text/html")&&a&&!t.startsWith("/api/"),handler:new X({cacheName:eE.html,plugins:[new eu({maxEntries:32,maxAgeSeconds:86400})]})},{matcher:({url:{pathname:e},sameOrigin:t})=>t&&!e.startsWith("/api/"),handler:new X({cacheName:"others",plugins:[new eu({maxEntries:32,maxAgeSeconds:86400})]})},{matcher:({sameOrigin:e})=>!e,handler:new X({cacheName:"cross-origin",plugins:[new eu({maxEntries:32,maxAgeSeconds:3600})],networkTimeoutSeconds:10})},{matcher:/.*/i,method:"GET",handler:new Y}],eq={cacheNames:{static:"tour-builder-static-v1",dynamic:"tour-builder-dynamic-v1",assets:"tour-builder-assets-v1"}},eC=[".png",".jpg",".jpeg",".gif",".webp",".svg",".ico",".mp4",".webm",".mov",".mp3",".wav",".ogg",".m4a",".woff",".woff2",".ttf",".eot",".css",".js"],eN=e=>{let t=new URL(e.url);return[".mp4",".webm",".mov"].some(e=>t.pathname.toLowerCase().endsWith(e))},eD=e=>{let t=new URL(e.url);return[".mp3",".wav",".ogg",".m4a",".aac"].some(e=>t.pathname.toLowerCase().endsWith(e))},eT=e=>{try{if(e.includes("/file/download?privateUrl=")){let t=e.match(/privateUrl=([^&]+)/);if(t)return decodeURIComponent(t[1]).replace(/^\/+/,"")}if(e.includes("X-Amz-Signature=")||e.includes("x-amz-signature=")){let t=new URL(e).pathname.split("/").filter(Boolean),a=t.findIndex(e=>"assets"===e);if(-1!==a)return t.slice(a).join("/");if(t.length>1)return t.slice(1).join("/")}if(e.startsWith("assets/"))return e;let t=e.match(/^https?:\/\/[^/]+\.s3\.[^/]+\.amazonaws\.com\/[^/]+\/(assets\/.+)$/);if(t)return t[1].split("?")[0];return null}catch(e){return null}},eP=new Map;setInterval(()=>{eP.clear(),console.log("[SW] Cleared storage path mappings")},36e5);let eU=new ex({precacheEntries:[{'revision':'8176e69ce8a57d1f2c88833db9695e7f','url':'/_next/static/Nx8WK1R_jk0ImwaPSFpEZ/_buildManifest.js'},{'revision':'b6652df95db52feb4daf4eca35380933','url':'/_next/static/Nx8WK1R_jk0ImwaPSFpEZ/_ssgManifest.js'},{'revision':null,'url':'/_next/static/chunks/12142dde-27563c775e4f8cf0.js'},{'revision':null,'url':'/_next/static/chunks/223-5e7611f0ad4bb7f3.js'},{'revision':null,'url':'/_next/static/chunks/3639-cd14c0b3222abc0f.js'},{'revision':null,'url':'/_next/static/chunks/3643-cd854ecad4d47645.js'},{'revision':null,'url':'/_next/static/chunks/3658-85ed1f5c943630b6.js'},{'revision':null,'url':'/_next/static/chunks/400-ac29014eab06ed0b.js'},{'revision':null,'url':'/_next/static/chunks/4166-fb783b1c70f3d744.js'},{'revision':null,'url':'/_next/static/chunks/4271-b6985491cfc2640a.js'},{'revision':null,'url':'/_next/static/chunks/4449-bdf4aeb088e38679.js'},{'revision':null,'url':'/_next/static/chunks/4498-33287c5f7f4f906f.js'},{'revision':null,'url':'/_next/static/chunks/4515-1830bd87b3c54e01.js'},{'revision':null,'url':'/_next/static/chunks/4587-c9e5910a896d025b.js'},{'revision':null,'url':'/_next/static/chunks/5371-1944e2cec48f4711.js'},{'revision':null,'url':'/_next/static/chunks/5541.e90be83844a6de15.js'},{'revision':null,'url':'/_next/static/chunks/6d2b60a9-eb6c7fd9a57c4f19.js'},{'revision':null,'url':'/_next/static/chunks/7e42aecb-94f8c450c54b9556.js'},{'revision':null,'url':'/_next/static/chunks/8180-d64b7d080d421773.js'},{'revision':null,'url':'/_next/static/chunks/8230-251e23b7a24ff307.js'},{'revision':null,'url':'/_next/static/chunks/8232-06043c4b3efac0d3.js'},{'revision':null,'url':'/_next/static/chunks/8317-4e9c090ccc089653.js'},{'revision':null,'url':'/_next/static/chunks/8366-6410354c1491f60a.js'},{'revision':null,'url':'/_next/static/chunks/8628-58bfe2d651454e89.js'},{'revision':null,'url':'/_next/static/chunks/9375.2c053880845a1e8b.js'},{'revision':null,'url':'/_next/static/chunks/9848-799d062feeef8c3c.js'},{'revision':null,'url':'/_next/static/chunks/fa3de7d5.7acfffddc0ff677e.js'},{'revision':null,'url':'/_next/static/chunks/framework-4dea986807dc9d02.js'},{'revision':null,'url':'/_next/static/chunks/main-f35af714f1aa0b25.js'},{'revision':null,'url':'/_next/static/chunks/pages/_error-530d8245847656c6.js'},{'revision':null,'url':'/_next/static/chunks/pages/access_logs/%5Baccess_logsId%5D-f28874507e6f5098.js'},{'revision':null,'url':'/_next/static/chunks/pages/access_logs/access_logs-edit-b460eaa195886f05.js'},{'revision':null,'url':'/_next/static/chunks/pages/access_logs/access_logs-list-374a36d094b2f91d.js'},{'revision':null,'url':'/_next/static/chunks/pages/access_logs/access_logs-new-316dae427bb1a425.js'},{'revision':null,'url':'/_next/static/chunks/pages/access_logs/access_logs-table-320341225528532a.js'},{'revision':null,'url':'/_next/static/chunks/pages/access_logs/access_logs-view-5caa4fc126fce326.js'},{'revision':null,'url':'/_next/static/chunks/pages/asset_variants/%5Basset_variantsId%5D-dfa121d326fe8623.js'},{'revision':null,'url':'/_next/static/chunks/pages/asset_variants/asset_variants-edit-9d551c7545acc058.js'},{'revision':null,'url':'/_next/static/chunks/pages/asset_variants/asset_variants-list-69fee1f7ee693caf.js'},{'revision':null,'url':'/_next/static/chunks/pages/asset_variants/asset_variants-new-4903a03e698a9694.js'},{'revision':null,'url':'/_next/static/chunks/pages/asset_variants/asset_variants-table-2ce1dd3f6569c112.js'},{'revision':null,'url':'/_next/static/chunks/pages/asset_variants/asset_variants-view-215e389fc57263d7.js'},{'revision':null,'url':'/_next/static/chunks/pages/assets/%5BassetsId%5D-9818cf1bfe706c30.js'},{'revision':null,'url':'/_next/static/chunks/pages/assets/assets-edit-31cc8b6d10fcc590.js'},{'revision':null,'url':'/_next/static/chunks/pages/assets/assets-list-827103fa1118a8f3.js'},{'revision':null,'url':'/_next/static/chunks/pages/assets/assets-new-1449e00ef4e9ea7a.js'},{'revision':null,'url':'/_next/static/chunks/pages/assets/assets-table-333b85d5a403eb30.js'},{'revision':null,'url':'/_next/static/chunks/pages/assets/assets-view-4a904823c447ae56.js'},{'revision':null,'url':'/_next/static/chunks/pages/constructor-cbb0fc807094ddb8.js'},{'revision':null,'url':'/_next/static/chunks/pages/dashboard-9c31c36b2aaac272.js'},{'revision':null,'url':'/_next/static/chunks/pages/element-type-defaults-b4b2a682b7110d4a.js'},{'revision':null,'url':'/_next/static/chunks/pages/element-type-defaults/%5Bid%5D-e1d59f81bfeb92d2.js'},{'revision':null,'url':'/_next/static/chunks/pages/error-078bf46bb4f4c577.js'},{'revision':null,'url':'/_next/static/chunks/pages/forgot-22af52cc664b2251.js'},{'revision':null,'url':'/_next/static/chunks/pages/index-f7e2d9108644a6f9.js'},{'revision':null,'url':'/_next/static/chunks/pages/login-a1d399f13c626ebd.js'},{'revision':null,'url':'/_next/static/chunks/pages/p/%5BprojectSlug%5D-f5626235effe723b.js'},{'revision':null,'url':'/_next/static/chunks/pages/p/%5BprojectSlug%5D/stage-a1d36cb5470f39e7.js'},{'revision':null,'url':'/_next/static/chunks/pages/password-reset-455af6aac51516fe.js'},{'revision':null,'url':'/_next/static/chunks/pages/permissions/%5BpermissionsId%5D-ba71d986ba5cc8fc.js'},{'revision':null,'url':'/_next/static/chunks/pages/permissions/permissions-edit-40c5c80e903296e5.js'},{'revision':null,'url':'/_next/static/chunks/pages/permissions/permissions-list-173a939b8f767fb0.js'},{'revision':null,'url':'/_next/static/chunks/pages/permissions/permissions-new-a5aaba465f5e06ba.js'},{'revision':null,'url':'/_next/static/chunks/pages/permissions/permissions-table-f2a0df70716b0284.js'},{'revision':null,'url':'/_next/static/chunks/pages/permissions/permissions-view-bc146f95287c4342.js'},{'revision':null,'url':'/_next/static/chunks/pages/presigned_url_requests/%5Bpresigned_url_requestsId%5D-d7074280b7a1269c.js'},{'revision':null,'url':'/_next/static/chunks/pages/presigned_url_requests/presigned_url_requests-edit-3a1654236738cc0a.js'},{'revision':null,'url':'/_next/static/chunks/pages/presigned_url_requests/presigned_url_requests-list-f20820d8be8c754a.js'},{'revision':null,'url':'/_next/static/chunks/pages/presigned_url_requests/presigned_url_requests-new-aecd81a5cb7a2ca9.js'},{'revision':null,'url':'/_next/static/chunks/pages/presigned_url_requests/presigned_url_requests-table-fc0554487470948c.js'},{'revision':null,'url':'/_next/static/chunks/pages/presigned_url_requests/presigned_url_requests-view-786d8c42729dfae4.js'},{'revision':null,'url':'/_next/static/chunks/pages/privacy-policy-53ea2331c015449b.js'},{'revision':null,'url':'/_next/static/chunks/pages/profile-cf3e76df64b95b2e.js'},{'revision':null,'url':'/_next/static/chunks/pages/project-element-defaults-b42e1eb7ad9baeec.js'},{'revision':null,'url':'/_next/static/chunks/pages/project-element-defaults/%5Bid%5D-dc40c92c88f5f468.js'},{'revision':null,'url':'/_next/static/chunks/pages/project_audio_tracks/%5Bproject_audio_tracksId%5D-18851bfd20223bd3.js'},{'revision':null,'url':'/_next/static/chunks/pages/project_audio_tracks/project_audio_tracks-edit-50440128faecb960.js'},{'revision':null,'url':'/_next/static/chunks/pages/project_audio_tracks/project_audio_tracks-list-17bf02d64388805e.js'},{'revision':null,'url':'/_next/static/chunks/pages/project_audio_tracks/project_audio_tracks-new-a9eb871b89fc9a7d.js'},{'revision':null,'url':'/_next/static/chunks/pages/project_audio_tracks/project_audio_tracks-table-8904e0e22909a1cd.js'},{'revision':null,'url':'/_next/static/chunks/pages/project_audio_tracks/project_audio_tracks-view-9a0e90abaa80f5c6.js'},{'revision':null,'url':'/_next/static/chunks/pages/project_memberships/%5Bproject_membershipsId%5D-d3231a2f7a8c4841.js'},{'revision':null,'url':'/_next/static/chunks/pages/project_memberships/project_memberships-edit-a5aac2ede6e10fa6.js'},{'revision':null,'url':'/_next/static/chunks/pages/project_memberships/project_memberships-list-f038d3a013deaef7.js'},{'revision':null,'url':'/_next/static/chunks/pages/project_memberships/project_memberships-new-635c81ed93a9a59b.js'},{'revision':null,'url':'/_next/static/chunks/pages/project_memberships/project_memberships-table-3b26e0796ef0631b.js'},{'revision':null,'url':'/_next/static/chunks/pages/project_memberships/project_memberships-view-a49b17115676b9d6.js'},{'revision':null,'url':'/_next/static/chunks/pages/projects/%5BprojectsId%5D-97fde22ccd383416.js'},{'revision':null,'url':'/_next/static/chunks/pages/projects/projects-edit-cdada24d860cdbc1.js'},{'revision':null,'url':'/_next/static/chunks/pages/projects/projects-list-4797bde4b728a4f7.js'},{'revision':null,'url':'/_next/static/chunks/pages/projects/projects-new-620fb97ab7b8c2ae.js'},{'revision':null,'url':'/_next/static/chunks/pages/projects/projects-table-4dab8b67c60241ee.js'},{'revision':null,'url':'/_next/static/chunks/pages/projects/projects-view-6ab88073d01e7234.js'},{'revision':null,'url':'/_next/static/chunks/pages/publish_events/%5Bpublish_eventsId%5D-baf397a319772a40.js'},{'revision':null,'url':'/_next/static/chunks/pages/publish_events/publish_events-edit-cb5fb3d2fe04da3e.js'},{'revision':null,'url':'/_next/static/chunks/pages/publish_events/publish_events-list-ed11277fbf030af1.js'},{'revision':null,'url':'/_next/static/chunks/pages/publish_events/publish_events-new-772e5dc7930d5413.js'},{'revision':null,'url':'/_next/static/chunks/pages/publish_events/publish_events-table-125100eb7ce83597.js'},{'revision':null,'url':'/_next/static/chunks/pages/publish_events/publish_events-view-989df6b2ae0f52ff.js'},{'revision':null,'url':'/_next/static/chunks/pages/pwa_caches/%5Bpwa_cachesId%5D-d15b2294a1fa5c3a.js'},{'revision':null,'url':'/_next/static/chunks/pages/pwa_caches/pwa_caches-edit-387462a9b31a5fd5.js'},{'revision':null,'url':'/_next/static/chunks/pages/pwa_caches/pwa_caches-list-893828e277a6fcd8.js'},{'revision':null,'url':'/_next/static/chunks/pages/pwa_caches/pwa_caches-new-2e051be868dfd532.js'},{'revision':null,'url':'/_next/static/chunks/pages/pwa_caches/pwa_caches-table-4b13eb9bd719ab77.js'},{'revision':null,'url':'/_next/static/chunks/pages/pwa_caches/pwa_caches-view-d4cd0ed163eeef15.js'},{'revision':null,'url':'/_next/static/chunks/pages/register-29076d71b547f51d.js'},{'revision':null,'url':'/_next/static/chunks/pages/roles/%5BrolesId%5D-530e349681b76250.js'},{'revision':null,'url':'/_next/static/chunks/pages/roles/roles-edit-3b414a4ab1be52f0.js'},{'revision':null,'url':'/_next/static/chunks/pages/roles/roles-list-11195c24a3a766b3.js'},{'revision':null,'url':'/_next/static/chunks/pages/roles/roles-new-372976d488ae4ffd.js'},{'revision':null,'url':'/_next/static/chunks/pages/roles/roles-table-fc38127ef72aa50c.js'},{'revision':null,'url':'/_next/static/chunks/pages/roles/roles-view-46eac6512efffec1.js'},{'revision':null,'url':'/_next/static/chunks/pages/search-e8adbb82adbdc228.js'},{'revision':null,'url':'/_next/static/chunks/pages/terms-of-use-7d00698d6183bde4.js'},{'revision':null,'url':'/_next/static/chunks/pages/tour_pages/%5Btour_pagesId%5D-24bac32931f40a4d.js'},{'revision':null,'url':'/_next/static/chunks/pages/tour_pages/tour_pages-edit-496f16c2780ded8c.js'},{'revision':null,'url':'/_next/static/chunks/pages/tour_pages/tour_pages-list-722b408231e0669a.js'},{'revision':null,'url':'/_next/static/chunks/pages/tour_pages/tour_pages-new-7b92fde20d8cb30c.js'},{'revision':null,'url':'/_next/static/chunks/pages/tour_pages/tour_pages-table-4e74acbf5f7ca7f5.js'},{'revision':null,'url':'/_next/static/chunks/pages/tour_pages/tour_pages-view-ddbb70192d378851.js'},{'revision':null,'url':'/_next/static/chunks/pages/users/%5BusersId%5D-f6a6505de81b0049.js'},{'revision':null,'url':'/_next/static/chunks/pages/users/users-edit-e55cf6d47da856f8.js'},{'revision':null,'url':'/_next/static/chunks/pages/users/users-list-10513f2e4f30f4ee.js'},{'revision':null,'url':'/_next/static/chunks/pages/users/users-new-dfdb731d8160e527.js'},{'revision':null,'url':'/_next/static/chunks/pages/users/users-table-0804ca961eb56a7f.js'},{'revision':null,'url':'/_next/static/chunks/pages/users/users-view-436d3bcdfad89528.js'},{'revision':null,'url':'/_next/static/chunks/pages/verify-email-3c9a0ef6c6d5daff.js'},{'revision':'846118c33b2c0e922d7b3a7676f81f6f','url':'/_next/static/chunks/polyfills-42372ed130431b0a.js'},{'revision':null,'url':'/_next/static/chunks/webpack-fb711922a0fad4b6.js'},{'revision':null,'url':'/_next/static/css/715be398208dca58.css'},{'revision':null,'url':'/_next/static/css/a4fad1b1a88475b2.css'},{'revision':null,'url':'/_next/static/css/de6fa09b8a0934d1.css'},{'revision':null,'url':'/_next/static/media/instrument-sans-latin-ext-wdth-normal.a718fc63.woff2'},{'revision':null,'url':'/_next/static/media/instrument-sans-latin-ext-wght-normal.7db92424.woff2'},{'revision':null,'url':'/_next/static/media/instrument-sans-latin-wdth-normal.68c3c527.woff2'},{'revision':null,'url':'/_next/static/media/instrument-sans-latin-wght-normal.ae05c57c.woff2'},{'revision':'43778b43fb039fdd5b0510561dc952ac','url':'/assets/vm-shot-2026-03-17T04-16-09-161Z.jpg'},{'revision':'3395d49fe2b96221471b4db0f5caf9e7','url':'/assets/vm-shot-2026-03-17T04-19-03-565Z.jpg'},{'revision':'62127cd8f85821f3e570a377a8a7e14b','url':'/assets/vm-shot-2026-03-17T04-36-56-252Z.jpg'},{'revision':'145c2d7e4ef298391258c6d8a8aaaece','url':'/assets/vm-shot-2026-03-17T04-45-14-111Z.jpg'},{'revision':'c6aae6521f08c847e370764d1c9c613d','url':'/assets/vm-shot-2026-03-17T04-50-58-546Z.jpg'},{'revision':'bf90fb959c1d418b01eb7e55de6672d5','url':'/assets/vm-shot-2026-03-19T06-12-36-229Z.jpg'},{'revision':'73de6cac0695249f4e59ce778a8e6e74','url':'/assets/vm-shot-2026-03-19T06-13-19-234Z.jpg'},{'revision':'bf90fb959c1d418b01eb7e55de6672d5','url':'/assets/vm-shot-2026-03-19T06-13-54-729Z.jpg'},{'revision':'ca6cbfcc74b52f00eef0f2adc8e65456','url':'/assets/vm-shot-2026-03-24T14-29-20-260Z.jpg'},{'revision':'0957c5365895c5ba31b10a84f4e45929','url':'/data-sources/clients.json'},{'revision':'5703d42f7838705ecf87510c4032b20c','url':'/data-sources/history.json'},{'revision':'5404a85badad8210a634ce41bb511545','url':'/favicon.svg'},{'revision':'fbe5fc5322bbd6ffda8bff4441b6bfa6','url':'/fonts/MapleMedium.otf'},{'revision':'508520242399a6b1fec65430901f4e6f','url':'/locales/de/common.json'},{'revision':'0a64739b954a93627749ffcb846fceaa','url':'/locales/en/common.json'},{'revision':'34715e25a3bcbab44f84232ce082d2ee','url':'/locales/es/common.json'},{'revision':'772a72f35589c06bdd8d9179c86459e6','url':'/locales/fr/common.json'},{'revision':'c67326edc61e0b5cf87e509ea7553466','url':'/manifest.json'},{'revision':'1254c9c72aa64724c203d26278712800','url':'/offline.html'}],skipWaiting:!0,clientsClaim:!0,navigationPreload:!0,runtimeCaching:[{matcher:e=>{let{request:t}=e;return"1"===new URL(t.url).searchParams.get("sw-bypass")},handler:new Y},{matcher:e=>{let{request:t}=e,a=new URL(t.url);return"image"===t.destination||"font"===t.destination||[".css",".js",".woff",".woff2"].some(e=>a.pathname.endsWith(e))},handler:new e_({cacheName:eq.cacheNames.assets,plugins:[{cacheKeyWillBeUsed:async e=>{let{request:t,mode:a}=e,s=eT(t.url);return s?(console.log("[SW] Using storagePath for static asset ".concat(a,":"),s.slice(-40)),new Request(s)):t},cacheWillUpdate:async e=>{let{response:t}=e;return t&&200===t.status?t:null}}]})},{matcher:e=>{let{request:t}=e;return eN(t)},handler:new e_({cacheName:eq.cacheNames.assets,plugins:[{cacheKeyWillBeUsed:async e=>{let{request:t,mode:a}=e,s=eT(t.url);if(s){let e=eP.get(s);return e?(console.log("[SW] Using storageKey for video ".concat(a,":"),e.slice(-40)),new Request(e)):(console.log("[SW] Using storagePath for video ".concat(a,":"),s.slice(-40)),new Request(s))}return t},cachedResponseWillBeUsed:async e=>{let{cachedResponse:t,request:a}=e;if(!t)return null;let s=a.headers.get("range");if(!s)return t;let r=s.match(/bytes=(\d+)-(\d*)/);if(!r)return t;let n=parseInt(r[1],10),i=r[2]?parseInt(r[2],10):void 0,c=await t.blob(),o=void 0!==i?c.slice(n,i+1):c.slice(n);return new Response(o,{status:206,statusText:"Partial Content",headers:{"Content-Type":t.headers.get("Content-Type")||"video/mp4","Content-Length":String(o.size),"Content-Range":"bytes ".concat(n,"-").concat(void 0!==i?i:c.size-1,"/").concat(c.size),"Accept-Ranges":"bytes"}})},cacheWillUpdate:async e=>{let{response:t}=e;return t&&200===t.status?t:null}}]})},{matcher:e=>{let{request:t}=e;return eD(t)},handler:new e_({cacheName:eq.cacheNames.assets,plugins:[{cacheKeyWillBeUsed:async e=>{let{request:t,mode:a}=e,s=eT(t.url);if(s){let e=eP.get(s);return e?(console.log("[SW] Using storageKey for audio ".concat(a,":"),e.slice(-40)),new Request(e)):(console.log("[SW] Using storagePath for audio ".concat(a,":"),s.slice(-40)),new Request(s))}return t},cachedResponseWillBeUsed:async e=>{let{cachedResponse:t,request:a}=e;if(!t)return null;let s=a.headers.get("range");if(!s)return t;let r=s.match(/bytes=(\d+)-(\d*)/);if(!r)return t;let n=parseInt(r[1],10),i=r[2]?parseInt(r[2],10):void 0,c=await t.blob(),o=void 0!==i?c.slice(n,i+1):c.slice(n);return new Response(o,{status:206,statusText:"Partial Content",headers:{"Content-Type":t.headers.get("Content-Type")||"audio/mpeg","Content-Length":String(o.size),"Content-Range":"bytes ".concat(n,"-").concat(void 0!==i?i:c.size-1,"/").concat(c.size),"Accept-Ranges":"bytes"}})},cacheWillUpdate:async e=>{let{response:t}=e;return t&&200===t.status?t:null}}]})},{matcher:e=>{let{url:t}=e;return t.pathname.startsWith("/api/")},handler:new X({cacheName:"api-cache",networkTimeoutSeconds:10})},{matcher:e=>{let{request:t}=e;return(e=>{let t=new URL(e.url);return(!t.pathname.startsWith("/api/")||!!t.pathname.includes("/file/download"))&&!!(eC.some(e=>t.pathname.toLowerCase().endsWith(e))||t.pathname.includes("/file/download")||t.hostname.includes("amazonaws.com")||t.hostname.includes("cloudfront.net"))})(t)&&!eN(t)&&!eD(t)},handler:new e_({cacheName:eq.cacheNames.assets,plugins:[{cacheKeyWillBeUsed:async e=>{let{request:t,mode:a}=e,s=eT(t.url);return s?(console.log("[SW] Using storagePath for dynamic asset ".concat(a,":"),s.slice(-40)),new Request(s)):t},cacheWillUpdate:async e=>{let{response:t}=e;return t&&200===t.status?t:null}}]})},...eS]});self.addEventListener("message",e=>{let{type:t,payload:a}=e.data||{};switch(t){case"REGISTER_CACHE_URL":if(null==a?void 0:a.storageKey){let e=a.presignedUrl&&eT(a.presignedUrl)||a.storageKey;eP.set(e,a.storageKey),console.log("[SW] Registered storage path for caching",{storagePath:e.slice(-50),storageKey:a.storageKey.slice(-50)})}break;case"CLEAR_URL_MAPPINGS":eP.clear(),console.log("[SW] Storage path mappings cleared");break;case"CACHE_ASSETS":Array.isArray(null==a?void 0:a.urls)&&e.waitUntil(caches.open(eq.cacheNames.assets).then(e=>Promise.all(a.urls.map(t=>fetch(t).then(a=>{if(200===a.status)return e.put(t,a)}).catch(e=>{console.warn("[SW] Failed to cache asset:",t,e)})))));break;case"CACHE_VIDEO_CHUNK":(null==a?void 0:a.url)&&(null==a?void 0:a.chunk)&&e.waitUntil(caches.open(eq.cacheNames.assets).then(e=>{let t=new Response(a.chunk,{headers:{"Content-Type":a.contentType||"video/mp4","Content-Length":String(a.chunk.byteLength)}});return e.put(a.url,t)}));break;case"CLEAR_CACHE":eP.clear(),e.waitUntil(Promise.all([caches.delete(eq.cacheNames.dynamic),caches.delete(eq.cacheNames.assets)]).then(()=>{console.log("[SW] Caches and storage path mappings cleared")}));break;case"GET_CACHE_STATUS":e.waitUntil(caches.open(eq.cacheNames.assets).then(t=>t.keys().then(t=>{let a=e.source;null==a||a.postMessage({type:"CACHE_STATUS",payload:{cachedCount:t.length,urls:t.map(e=>e.url)}})})));break;case"SKIP_WAITING":self.skipWaiting()}}),eU.addEventListeners(),console.log("[SW] Serwist service worker loaded")})(); \ No newline at end of file diff --git a/frontend/src/components/CanvasLoadingSpinner.tsx b/frontend/src/components/CanvasLoadingSpinner.tsx index c6c8785..ace5cad 100644 --- a/frontend/src/components/CanvasLoadingSpinner.tsx +++ b/frontend/src/components/CanvasLoadingSpinner.tsx @@ -24,7 +24,7 @@ interface CanvasLoadingSpinnerProps { const CanvasLoadingSpinner: React.FC = ({ isVisible, - message = 'Loading...', + message, size = 'md', progress, zIndex = 100, @@ -50,23 +50,20 @@ const CanvasLoadingSpinner: React.FC = ({ const sizeClasses = { sm: 'w-8 h-8 border-2', - md: 'w-12 h-12 border-3', + md: 'w-12 h-12 border-[3px]', lg: 'w-16 h-16 border-4', }; return (
-
+ {/* Spinner with subtle shadow for visibility on any background */} +
{/* Spinner ring */}
= ({ {/* Progress indicator (optional) */} {progress !== undefined && (
- + {Math.round(progress)}%
)}
{message && ( -

{message}

+

+ {message} +

)}
); diff --git a/frontend/src/components/Constructor/CanvasBackground.tsx b/frontend/src/components/Constructor/CanvasBackground.tsx index f9cdfba..e94f4b1 100644 --- a/frontend/src/components/Constructor/CanvasBackground.tsx +++ b/frontend/src/components/Constructor/CanvasBackground.tsx @@ -16,10 +16,19 @@ import React, { import NextImage from 'next/image'; import { useBackgroundVideoPlayback } from '../../hooks/useBackgroundVideoPlayback'; import PreviousBackgroundOverlay from '../PreviousBackgroundOverlay'; -import CanvasLoadingSpinner from '../CanvasLoadingSpinner'; -import { scheduleAfterPaint } from '../../lib/browserUtils'; import { baseURLApi } from '../../config'; +/** + * Schedule a callback to run after the next browser paint. + * Uses double rAF pattern: first rAF schedules for next frame, + * second rAF ensures the frame has actually been committed. + */ +const scheduleAfterPaint = (callback: () => void): void => { + requestAnimationFrame(() => { + requestAnimationFrame(callback); + }); +}; + // Type for requestVideoFrameCallback (Safari 15.4+, Chrome 83+) // The callback receives (now: DOMHighResTimeStamp, metadata: VideoFrameCallbackMetadata) // but we ignore them since we only need to know the frame was painted @@ -51,6 +60,8 @@ interface CanvasBackgroundProps { videoEndTime?: number | null; /** Original storage path for video - used for play-once tracking (not the resolved blob URL) */ videoStoragePath?: string; + /** Pause video playback (e.g., during navigation to show frozen frame) */ + pauseVideo?: boolean; } const CanvasBackground: React.FC = ({ @@ -69,21 +80,33 @@ const CanvasBackground: React.FC = ({ videoStartTime = null, videoEndTime = null, videoStoragePath, + pauseVideo = false, }) => { + // During page switching with video paused, keep showing the previous video URL. + // This prevents black flash when the video element would remount with a new URL. + // The old video element stays visible (paused at frozen frame) until new page is ready. + const activeVideoUrl = + isSwitching && !isNewBgReady && pauseVideo && previousBgVideoUrl + ? previousBgVideoUrl + : backgroundVideoUrl; + // Use background video playback hook for custom start/end time handling // Use storagePath for play-once tracking (falls back to videoUrl if not provided) + // Pass pauseVideo to hook for centralized playback control (fixes video playing during navigation) const { videoRef, shouldBlockAutoplay } = useBackgroundVideoPlayback({ - videoUrl: backgroundVideoUrl, + videoUrl: activeVideoUrl, videoStoragePath: videoStoragePath || backgroundVideoUrl, autoplay: videoAutoplay, loop: videoLoop, muted: videoMuted, startTime: videoStartTime, endTime: videoEndTime, + paused: pauseVideo, }); - // Block autoplay if video already played this session (when loop=false) - const effectiveAutoplay = videoAutoplay && !shouldBlockAutoplay; + // Block autoplay if: video already played this session OR externally paused + const effectiveAutoplay = + videoAutoplay && !shouldBlockAutoplay && !pauseVideo; // Video error state for fallback to proxy URL const [videoError, setVideoError] = useState(false); @@ -96,6 +119,9 @@ const CanvasBackground: React.FC = ({ const video = videoRef.current; if (!backgroundVideoUrl || !video) { setIsVideoBuffering(false); + // CRITICAL: Also notify parent that buffering is done when there's no video + // Without this, parent's isBackgroundVideoBuffering stays stuck at true from previous page + onVideoBufferStateChange?.(false); return; } @@ -124,13 +150,13 @@ const CanvasBackground: React.FC = ({ // Fallback to proxy URL if presigned URL fails (e.g., CORS, expiration) const videoSrc = useMemo(() => { - if (!backgroundVideoUrl) return undefined; + if (!activeVideoUrl) return undefined; if (videoError && videoStoragePath) { // Fallback to backend proxy (bypasses CORS issues) return `${baseURLApi}/file/download?privateUrl=${encodeURIComponent(videoStoragePath)}`; } - return backgroundVideoUrl; - }, [backgroundVideoUrl, videoStoragePath, videoError]); + return activeVideoUrl; + }, [activeVideoUrl, videoStoragePath, videoError]); // Reset error state when video URL changes useEffect(() => { @@ -145,25 +171,159 @@ const CanvasBackground: React.FC = ({ } }, [videoError, videoStoragePath]); - const handleLoad = () => { + // Track if we've already called onBackgroundReady to avoid double calls + const didReportImageReadyRef = useRef(false); + const imageRef = useRef(null); + // Ref for NextImage wrapper to detect its internal img element + const nextImageWrapperRef = useRef(null); + // Track previous URL to detect changes synchronously during render + const prevImageUrlRef = useRef(undefined); + // Track previous switching state to detect navigation start + const prevIsSwitchingRef = useRef(false); + + // CRITICAL: Reset ready flag SYNCHRONOUSLY during render, before onLoad can fire. + // Reset when: + // 1. URL changes - new image needs to report ready + // 2. isSwitching transitions from false to true - navigation started, even if URL is the same + // (handles case where two pages have the same background image) + // + // Using useEffect for this creates a race condition: + // 1. URL changes, component re-renders + // 2. For cached images, onLoad fires immediately (maybe even before React attaches handlers) + // 3. handleLoad checks didReportImageReadyRef which is still TRUE from previous image + // 4. Guard exits early, callback is skipped + // 5. useEffect runs AFTER render, resetting the flag too late + // By resetting synchronously here, we ensure the flag is false before any event handlers run. + const switchingStarted = isSwitching && !prevIsSwitchingRef.current; + if (prevImageUrlRef.current !== backgroundImageUrl || switchingStarted) { + didReportImageReadyRef.current = false; + prevImageUrlRef.current = backgroundImageUrl; + } + prevIsSwitchingRef.current = isSwitching; + + const handleLoad = useCallback(() => { + if (didReportImageReadyRef.current) { + return; + } + didReportImageReadyRef.current = true; // Wait for paint to ensure background is actually rendered before reporting ready. // This prevents the transition overlay from being removed before the background is visible. scheduleAfterPaint(() => { onBackgroundReady?.(); }); - }; + }, [onBackgroundReady, backgroundImageUrl]); - const handleError = () => { + const handleError = useCallback(() => { + if (didReportImageReadyRef.current) return; + didReportImageReadyRef.current = true; onBackgroundReady?.(); - }; + }, [onBackgroundReady]); - // Track if we've already called onBackgroundReady to avoid double calls - const didReportReadyRef = useRef(false); - - // Reset ready flag when video URL changes + // Handle already-loaded images (blob URLs from preload cache) + // The onLoad event may not fire for images that are already in memory useEffect(() => { + const img = imageRef.current; + if (!backgroundImageUrl || !img || didReportImageReadyRef.current) return; + + // Check if image is already loaded (common with blob URLs) + if (img.complete && img.naturalWidth > 0) { + // Use decode() to ensure image is fully decoded before reporting ready + if (typeof img.decode === 'function') { + img.decode().then(handleLoad).catch(handleLoad); + } else { + handleLoad(); + } + } + }, [backgroundImageUrl, handleLoad]); + + // Handle NextImage load detection (for non-blob URLs like presigned URLs) + // NextImage's onLoad may not fire for cached images, so we detect its internal img element + useEffect(() => { + // Only handle non-blob URLs (blob URLs use native img with imageRef) + if ( + !backgroundImageUrl || + backgroundImageUrl.startsWith('blob:') || + didReportImageReadyRef.current + ) + return; + + const wrapper = nextImageWrapperRef.current; + if (!wrapper) return; + + let loadCleanup: (() => void) | null = null; + + // Setup load listener on the internal img element + const setupLoadListener = (img: HTMLImageElement) => { + // Use decode() to ensure image is fully decoded before reporting ready + // This prevents flash on first load when image needs to be fetched and decoded + const decodeAndReport = () => { + if (typeof img.decode === 'function') { + img.decode().then(handleLoad).catch(handleLoad); + } else { + handleLoad(); + } + }; + + // If already loaded, decode and report + if (img.complete && img.naturalWidth > 0) { + decodeAndReport(); + return; + } + // Not loaded yet, attach load event listener + const onLoad = () => decodeAndReport(); + img.addEventListener('load', onLoad, { once: true }); + loadCleanup = () => img.removeEventListener('load', onLoad); + }; + + // Check if NextImage's internal img element already exists + const existingImg = wrapper.querySelector('img'); + if (existingImg) { + setupLoadListener(existingImg); + return () => loadCleanup?.(); + } + + // Wait for NextImage to render its internal img element using MutationObserver + const observer = new MutationObserver((mutations) => { + for (let i = 0; i < mutations.length; i++) { + const addedNodes = mutations[i].addedNodes; + for (let j = 0; j < addedNodes.length; j++) { + const node = addedNodes[j]; + if (node instanceof HTMLImageElement) { + setupLoadListener(node); + observer.disconnect(); + return; + } + if (node instanceof Element) { + const img = node.querySelector('img'); + if (img) { + setupLoadListener(img); + observer.disconnect(); + return; + } + } + } + } + }); + + observer.observe(wrapper, { childList: true, subtree: true }); + + return () => { + observer.disconnect(); + loadCleanup?.(); + }; + }, [backgroundImageUrl, handleLoad]); + + // Track if we've already called onBackgroundReady to avoid double calls (for video) + const didReportReadyRef = useRef(false); + // Track previous video URL to detect changes synchronously during render + const prevVideoUrlRef = useRef(undefined); + + // CRITICAL: Reset ready flag SYNCHRONOUSLY during render (same reason as image above). + // Also reset when switching starts, to handle pages with same video URL. + if (prevVideoUrlRef.current !== backgroundVideoUrl || switchingStarted) { didReportReadyRef.current = false; - }, [backgroundVideoUrl]); + prevVideoUrlRef.current = backgroundVideoUrl; + } // Handle video first frame ready using requestVideoFrameCallback // This ensures the video's first frame is actually painted before we report ready @@ -188,11 +348,15 @@ const CanvasBackground: React.FC = ({ }, 5000); // Use requestVideoFrameCallback for precise frame-level timing (Safari 15.4+, Chrome 83+) + // RVFC fires when frame is decoded, but compositor may not have painted yet. + // Wrap in scheduleAfterPaint for consistency with image handling. const videoWithRVFC = video as HTMLVideoElementWithRVFC; if (typeof videoWithRVFC.requestVideoFrameCallback === 'function') { videoWithRVFC.requestVideoFrameCallback(() => { clearTimeout(timeout); - reportVideoReady(); + scheduleAfterPaint(() => { + reportVideoReady(); + }); }); } else { // Fallback: use playing event + scheduleAfterPaint @@ -216,6 +380,10 @@ const CanvasBackground: React.FC = ({ // When endTime is set, we disable native loop and handle it via the hook const useNativeLoop = videoEndTime == null ? videoLoop : false; + // Note: pauseVideo is now handled by useBackgroundVideoPlayback hook directly. + // The hook centralizes all playback control, eliminating race conditions between + // separate effects competing to control the video element. + return ( <> {/* Background image - z-1 keeps it below backdrop blur layer (z-5). @@ -233,6 +401,7 @@ const CanvasBackground: React.FC = ({ {backgroundImageUrl.startsWith('blob:') ? ( // eslint-disable-next-line @next/next/no-img-element Background = ({ onError={handleError} /> ) : ( - +
+ +
)}
)} @@ -265,6 +439,7 @@ const CanvasBackground: React.FC = ({ videoUrl={previousBgVideoUrl} isSwitching={isSwitching} isNewBgReady={isNewBgReady} + paused={pauseVideo} /> {/* Background video - z-1 keeps it below backdrop blur layer (z-5) @@ -274,18 +449,13 @@ const CanvasBackground: React.FC = ({ webkit-playsinline is legacy attribute for older iOS versions. preload="metadata" is required for iOS Safari video initialization. Video fades in when ready (opacity transition from 0 to 1). */} - {backgroundVideoUrl && ( + {activeVideoUrl && (
); }; diff --git a/frontend/src/components/Constructor/TransitionPreviewOverlay.tsx b/frontend/src/components/Constructor/TransitionPreviewOverlay.tsx index 959c7cd..02b38fd 100644 --- a/frontend/src/components/Constructor/TransitionPreviewOverlay.tsx +++ b/frontend/src/components/Constructor/TransitionPreviewOverlay.tsx @@ -7,9 +7,14 @@ * * Supports letterbox mode to constrain transitions within canvas bounds, * matching the behavior of background images and UI elements. + * + * Hide behavior: + * - Waits one requestAnimationFrame after isFadingOut=true + * - This ensures the new background is painted before hiding + * - Then hides instantly (no CSS transition) since last video frame = new bg */ -import React from 'react'; +import React, { useState, useEffect } from 'react'; import CanvasLoadingSpinner from '../CanvasLoadingSpinner'; interface TransitionPreviewOverlayProps { @@ -17,12 +22,12 @@ interface TransitionPreviewOverlayProps { videoRef: React.RefObject; /** Whether the overlay is visible */ isActive: boolean; - /** Whether the video is currently buffering (used to hide video during load) */ + /** Whether the video is currently buffering (used to show spinner) */ isBuffering?: boolean; + /** Whether first video frame has been displayed (used to determine if video should be visible during buffering) */ + isVideoReady?: boolean; /** Show loading spinner during buffering (default: false for backward compat) */ showSpinner?: boolean; - /** Loading message for spinner */ - spinnerMessage?: string; /** Letterbox styles from useCanvasScale - positions overlay within canvas bounds */ letterboxStyles?: React.CSSProperties; /** Video object-fit mode (default: 'contain' to match backgrounds) */ @@ -31,45 +36,72 @@ interface TransitionPreviewOverlayProps { opacity?: number; /** Forces video element remount when changed - prevents decoder state issues with pre-created blob URLs */ videoKey?: string; + /** When true, overlay will hide after one paint frame (ensures bg is painted first) */ + isFadingOut?: boolean; + /** Fade-out duration in ms - kept for interface compat, not used for video transitions (instant hide) */ + fadeOutDuration?: number; } const TransitionPreviewOverlay: React.FC = ({ videoRef, isActive, isBuffering = false, + isVideoReady = false, showSpinner = false, - spinnerMessage = 'Preparing transition...', letterboxStyles, videoFit = 'contain', opacity, videoKey, + isFadingOut = false, + // fadeOutDuration - not used for video transitions (instant hide) }) => { + // Delay hide by one frame to ensure new background is painted + const [shouldHide, setShouldHide] = useState(false); + + useEffect(() => { + if (isFadingOut) { + // Wait one frame to ensure new background is painted + const rafId = requestAnimationFrame(() => { + setShouldHide(true); + }); + return () => cancelAnimationFrame(rafId); + } else { + setShouldHide(false); + } + }, [isFadingOut]); + if (!isActive) return null; - // Container opacity: 0 while buffering to prevent black flash - // Video first frame = old page background, so we hide everything until ready - const containerOpacity = isBuffering ? 0 : (opacity ?? 1); + // Container opacity: + // - 0 during initial buffering (before first frame displayed) + // - 0 when shouldHide (after one-frame delay, new bg is painted) + // - otherwise use provided opacity or 1 + const isInitialBuffering = isBuffering && !isVideoReady; + const containerOpacity = isInitialBuffering + ? 0 + : shouldHide + ? 0 + : (opacity ?? 1); + + // Only use transition for initial buffering fade-in (150ms) + // No transition when hiding - instant hide since last video frame = new bg + const useTransition = isInitialBuffering || (!shouldHide && !isFadingOut); return ( // Outer: full viewport, transparent background // Transparent ensures if Safari clears video frame when paused, // the new page background shows through instead of black flash
- {/* Loading spinner during buffering - provides user feedback */} + {/* Loading spinner during buffering */} {isBuffering && showSpinner && ( - + )} - {/* Video container - hidden while buffering */} + {/* Video container - hidden while buffering or fading out */}
{/* Inner: respects letterbox dimensions when provided */} diff --git a/frontend/src/components/Constructor/types.ts b/frontend/src/components/Constructor/types.ts index dbd964f..c88fadd 100644 --- a/frontend/src/components/Constructor/types.ts +++ b/frontend/src/components/Constructor/types.ts @@ -109,7 +109,6 @@ export interface CanvasElementProps { element: CanvasElement; isSelected: boolean; isEditMode: boolean; - isDisabled?: boolean; canvasElapsedSec: number; preloadedIconUrl: boolean; onClick: (element: CanvasElement) => void; diff --git a/frontend/src/components/PreviousBackgroundOverlay.tsx b/frontend/src/components/PreviousBackgroundOverlay.tsx index c0e2ad8..8edd7f6 100644 --- a/frontend/src/components/PreviousBackgroundOverlay.tsx +++ b/frontend/src/components/PreviousBackgroundOverlay.tsx @@ -1,10 +1,14 @@ /** * PreviousBackgroundOverlay Component * - * Shows the previous page background during page transitions + * Shows the previous page background IMAGE during page transitions * while the new background is loading. * - * Used by CanvasBackground component. + * Renders when: isSwitching=true AND isNewBgReady=false + * Hides instantly when new background is ready. + * + * Note: Video backgrounds are NOT rendered here. During transitions, + * video is covered by TransitionPreviewOverlay at z-50. */ import React from 'react'; @@ -12,54 +16,44 @@ import React from 'react'; interface PreviousBackgroundOverlayProps { /** Previous background image URL */ imageUrl?: string; - /** Previous background video URL */ + /** Previous background video URL (kept for interface compatibility, not rendered) */ videoUrl?: string; /** Whether page is currently switching */ isSwitching?: boolean; /** Whether new background is ready */ isNewBgReady?: boolean; + /** Whether to pause video playback (kept for interface compatibility) */ + paused?: boolean; /** Additional CSS classes */ className?: string; + /** Fade duration - DEPRECATED, kept for interface compat */ + fadeDuration?: number; } const PreviousBackgroundOverlay: React.FC = ({ imageUrl, - videoUrl, + // videoUrl - not used, see docstring isSwitching = false, isNewBgReady = false, + // paused - not used, see docstring className = '', + // fadeDuration - deprecated, not used }) => { - // Show previous background during loading (before new bg is ready) - const showPreviousBackground = isSwitching && !isNewBgReady; + // Simple render logic: show while switching AND new bg not ready + const shouldRender = isSwitching && !isNewBgReady && !!imageUrl; - if (!showPreviousBackground) return null; + if (!shouldRender) return null; return ( - <> - {/* Previous background image */} - {imageUrl && ( -
- )} - {/* Previous background video */} - {videoUrl && ( -