mirror of
https://github.com/yeicor-3d/yet-another-cad-viewer.git
synced 2025-12-19 22:24:17 +01:00
150 lines
6.5 KiB
TypeScript
150 lines
6.5 KiB
TypeScript
// These are the default values for the settings, which are overridden below
|
|
import {ungzip} from "pako";
|
|
import {b66Decode} from "../tools/b66.ts";
|
|
|
|
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
|
|
'<auto>', // 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=="),
|
|
|
|
// Playground settings
|
|
pg_code: "", // Automatically loaded and executed code for the playground
|
|
pg_code_url: "", // URL to load the code from (overrides pg_code)
|
|
pg_opacity_loading: -1, // Opacity of the code during first load and run (< 0 is 0.0 if preload and 0.9 if not)
|
|
pg_opacity_loaded: 0.9, // Opacity of the code after it has been run for the first time
|
|
};
|
|
|
|
// Auto-override any settings from the URL (either GET parameters or hash)
|
|
const url = new URL(window.location.href);
|
|
url.searchParams.forEach((value, key) => {
|
|
if (key in settings) (settings as any)[key] = parseSetting(key, value, settings);
|
|
})
|
|
if (url.hash.length > 0) { // Hash has bigger limits as it is not sent to the server
|
|
const hash = url.hash.slice(1);
|
|
const hashParams = new URLSearchParams(hash);
|
|
hashParams.forEach((value, key) => {
|
|
if (key in settings) (settings as any)[key] = parseSetting(key, value, settings);
|
|
});
|
|
}
|
|
|
|
// Grab the code from the URL if it is set
|
|
if (settings.pg_code_url.length > 0) {
|
|
// If the code URL is set, override the code
|
|
try {
|
|
const response = await fetch(settings.pg_code_url);
|
|
if (response.ok) {
|
|
settings.pg_code = await response.text();
|
|
} else {
|
|
console.warn("Failed to load code from URL:", settings.pg_code_url);
|
|
}
|
|
} catch (error) {
|
|
console.error("Error fetching code from URL:", settings.pg_code_url, error);
|
|
}
|
|
}
|
|
|
|
// 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 === '<auto>') {
|
|
if (settings.pg_code != "") { // <auto> means no preload URL if code is set
|
|
settings.preload = settings.preload.slice(0, i).concat(settings.preload.slice(i + 1));
|
|
continue; // Skip this preload 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 === '<auto>') { // Fallback to the default preload URL of localhost
|
|
url = "dev+http://localhost:32323";
|
|
}
|
|
}
|
|
settings.preload[i] = url;
|
|
}
|
|
|
|
// Auto-decompress the code and other playground settings
|
|
if (settings.pg_code.length > 0) {
|
|
try {
|
|
settings.pg_code = ungzip(b66Decode(settings.pg_code), {to: 'string'});
|
|
} catch (error) {
|
|
console.warn("Failed to decompress code (assuming raw code):", error);
|
|
}
|
|
if (settings.pg_opacity_loading < 0) {
|
|
// If the opacity is not set, use 0.0 if preload is set, otherwise 0.9
|
|
settings.pg_opacity_loading = settings.preload.length > 0 ? 0.0 : 0.9;
|
|
}
|
|
}
|
|
|
|
settingsCache = settings;
|
|
return settings;
|
|
}
|
|
|
|
const firstTimeNames: Array<string> = []; // 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}`);
|
|
}
|
|
}
|