mirror of
https://github.com/yeicor-3d/yet-another-cad-viewer.git
synced 2025-12-20 06:27:04 +01:00
Somewhat more conventional visual presentation: + Use a neutral gray background a-la most other CAD/modelling software for better contrast with the geometry (which defaults to a bright yellow). This is done with the "skybox-environment" image in model-viewer, so add a new setting value in settings.ts for this (it can be overriden in the URL just like other settings) + But using a skybox will cause that image to be used for lighting too, which is clearly not desired. So fetch a nice professional HDRI image from Polyhaven for lighting. This is much better (more directional, higher contrast) than the default light environment anyway. + The checkerboard texture isn't really a good default. Use a 1x1 white pixel instead, essentially presenting the model materials unchanged. Also collect the default color in gltf.py out of the code and put it next to the texture for clarity. This should probably be wired through to a setting at some point. Signed-off-by: Andy Ross <andy@plausible.org>
97 lines
4.1 KiB
TypeScript
97 lines
4.1 KiB
TypeScript
// 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
|
|
'<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
|
|
background: "https://dl.polyhaven.org/file/ph-assets/HDRIs/hdr/1k/qwantani_afternoon_1k.hdr",
|
|
// Uniform (1x1 pixel) medium gray background for visibility
|
|
skybox: "",
|
|
|
|
};
|
|
|
|
// 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 === '<auto>') {
|
|
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;
|
|
}
|
|
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}`);
|
|
}
|
|
}
|