// These are the default values for the settings, which are overridden below let settingsCache: any = null; export async function settings() { if (settingsCache !== null) return settingsCache; let settings = { preload: [ // @ts-ignore // new URL('../../assets/fox.glb', import.meta.url).href, // @ts-ignore // new URL('../../assets/logo_build/base.glb', import.meta.url).href, // @ts-ignore // new URL('../../assets/logo_build/location.glb', import.meta.url).href, // @ts-ignore // new URL('../../assets/logo_build/img.jpg.glb', import.meta.url).href, // Websocket URLs automatically listen for new models from the python backend '', // Get the default preload URL if not overridden ], loadHelpers: true, edgeWidth: 0, /* The default line size for edges, set to 0 to use basic gl.LINEs */ displayLoadingEveryMs: 1000, /* How often to display partially loaded models */ monitorEveryMs: 100, monitorOpenTimeoutMs: 1000, // ModelViewer settings autoplay: true, // Global animation toggle arModes: 'webxr scene-viewer quick-look', zoomSensitivity: 0.25, orbitSensitivity: 1, panSensitivity: 1, exposure: 1, shadowIntensity: 0, // Nice low-res outdoor/high-contrast HDRI image (CC0 licensed) for lighting environment: new URL('../../assets/qwantani_afternoon_1k_hdr.jpg', import.meta.url).href, environmentIntensity: 1.0, // Uniform (1x1 pixel) medium gray background for visibility (following dark/light mode) skybox: (window.matchMedia("(prefers-color-scheme: dark)").matches ? "" + "12NgAAAAAgAB4iG8MwAAAABJRU5ErkJggg==" : "" + "12NgAAAAAgAB4iG8MwAAAABJRU5ErkJggg=="), }; // settings.skybox = settings.background = ''; // Disable both skybox and background // Auto-override any settings from the URL const url = new URL(window.location.href); url.searchParams.forEach((value, key) => { if (key in settings) (settings as any)[key] = parseSetting(key, value, settings); }) // Get the default preload URL if not overridden (requires a fetch that is avoided if possible) for (let i = 0; i < settings.preload.length; i++) { let url = settings.preload[i]; if (url === '') { const possibleBackend = new URL("./?api_updates=true", window.location.href) await fetch(possibleBackend, {method: "HEAD"}).then((response) => { if (response.ok && response.headers.get("Content-Type") === "text/event-stream") { // Frontend served by the backend: default to this URL for updates url = "dev+" + possibleBackend.href; } }).catch((error) => console.error("Failed to check for backend:", error)); if (url === '') { // Fallback to the default preload URL of localhost url = "dev+http://localhost:32323"; } } settings.preload[i] = url; } settingsCache = settings; return settings; } const firstTimeNames: Array = []; // Needed for array values, which clear the array when overridden function parseSetting(name: string, value: string, settings: any): any { let arrayElem = name.endsWith(".0") if (arrayElem) name = name.slice(0, -2); let prevValue = (settings as any)[name]; if (prevValue === undefined) throw new Error(`Unknown setting: ${name}`); if (Array.isArray(prevValue)) { if (!arrayElem) { let toExtend = [] if (!firstTimeNames.includes(name)) { firstTimeNames.push(name); } else { toExtend = prevValue; } toExtend.push(parseSetting(name + ".0", value, settings)); return toExtend; } else { prevValue = prevValue[0]; } } switch (typeof prevValue) { case 'boolean': return value === 'true'; case 'number': return Number(value); case 'string': return value; default: throw new Error(`Unknown setting type: ${typeof prevValue} -- ${prevValue}`); } }