diff --git a/src/components/gltf-model-plus.js b/src/components/gltf-model-plus.js index b1195922467eb57dd6b536e688b147c164e502ea..fad11f667fb6c879fbc1505e056d9b7f39130684 100644 --- a/src/components/gltf-model-plus.js +++ b/src/components/gltf-model-plus.js @@ -1,3 +1,4 @@ +import { isFarsparkUrl, decodeFarsparkUrl } from "../utils/media-utils"; const GLTFCache = {}; AFRAME.GLTFModelPlus = { @@ -186,6 +187,10 @@ function cachedLoadGLTF(src, preferredTechnique, onProgress) { if (!GLTFCache[src]) { GLTFCache[src] = new Promise((resolve, reject) => { const gltfLoader = new THREE.GLTFLoader(); + // Farspark urls can't handle relative paths. Use the original url as the base path. + if (isFarsparkUrl(src)) { + gltfLoader.path = THREE.LoaderUtils.extractUrlBase(decodeFarsparkUrl(src)); + } gltfLoader.preferredTechnique = preferredTechnique; gltfLoader.load(src, resolve, onProgress, reject); }); diff --git a/src/utils/media-utils.js b/src/utils/media-utils.js index 9c17813b926e5541eef96b586f8efb133e696f6c..65c13bb0149ce9e6c6d2b4fd853958efd523f236 100644 --- a/src/utils/media-utils.js +++ b/src/utils/media-utils.js @@ -1,6 +1,5 @@ const whitelistedHosts = [/^.*\.reticulum\.io$/, /^.*hubs\.mozilla\.com$/, /^hubs\.local$/]; const isHostWhitelisted = hostname => !!whitelistedHosts.filter(r => r.test(hostname)).length; - let resolveMediaUrl = "/api/v1/media"; if (process.env.NODE_ENV === "development") { resolveMediaUrl = `https://${process.env.DEV_RETICULUM_SERVER}${resolveMediaUrl}`; @@ -18,9 +17,24 @@ export const resolveFarsparkUrl = async url => { }).then(r => r.json())).raw; }; -const fetchContentType = async url => fetch(url, { method: "HEAD" }).then(r => r.headers.get("content-type")); -let interactableId = 0; +const farsparkRegex = /^farspark.*\.reticulum\.io$/; +export const isFarsparkUrl = url => farsparkRegex.test(new URL(url).hostname); +export const decodeFarsparkUrl = url => { + const parts = url.split("/"); + return atob(parts[parts.length - 1]); +}; +const staticContentMappings = { + "poly.googleapis.com": "model/gltf" +}; +const fetchContentType = async url => { + const staticContentType = staticContentMappings[new URL(decodeFarsparkUrl(url)).hostname]; + return staticContentType + ? Promise.resolve(staticContentType) + : fetch(url, { method: "HEAD" }).then(r => r.headers.get("content-type")); +}; + +let interactableId = 0; const offset = { x: 0, y: 0, z: -1.5 }; export const spawnNetworkedImage = (src, contentType) => { const scene = AFRAME.scenes[0];