From 723bbb0eace6dfc759f4df3b512a26254917ada9 Mon Sep 17 00:00:00 2001 From: netpro2k <netpro2k@gmail.com> Date: Tue, 3 Jul 2018 16:07:07 -0700 Subject: [PATCH] Move farspark resolving and content type resolution out of image-plus --- src/components/image-plus.js | 25 +++++--------- src/hub.js | 26 +-------------- src/utils/media-utils.js | 64 ++++++++++++++++++++++++++++++++++++ src/vendor/GLTFLoader.js | 19 +++-------- 4 files changed, 77 insertions(+), 57 deletions(-) create mode 100644 src/utils/media-utils.js diff --git a/src/components/image-plus.js b/src/components/image-plus.js index a7efa5508..150552bf1 100644 --- a/src/components/image-plus.js +++ b/src/components/image-plus.js @@ -54,8 +54,6 @@ textureLoader.setCrossOrigin("anonymous"); const textureCache = new Map(); -const noop = function() {}; - const errorImage = new Image(); errorImage.src = ""; @@ -69,6 +67,7 @@ AFRAME.registerComponent("image-plus", { schema: { src: { type: "string" }, + contentType: { type: "string" }, initialOffset: { default: { x: 0, y: 0, z: -1.5 } }, reorientOnGrab: { default: false } @@ -92,6 +91,7 @@ AFRAME.registerComponent("image-plus", { } this.el.setAttribute("geometry", { width, height }); this.el.setAttribute("shape", { + shape: "box", halfExtents: { x: width / 2, y: height / 2, @@ -221,7 +221,7 @@ AFRAME.registerComponent("image-plus", { loadImage(url) { return new Promise((resolve, reject) => { - textureLoader.load(url, resolve, noop, function(xhr) { + textureLoader.load(url, resolve, null, function(xhr) { reject(`'${url}' could not be fetched (Error code: ${xhr.status}; Response: ${xhr.statusText})`); }); }); @@ -230,26 +230,17 @@ AFRAME.registerComponent("image-plus", { async update() { let texture; try { - const mediaJson = await fetch("https://smoke-dev.reticulum.io/api/v1/media", { - method: "POST", - headers: { - "Content-Type": "application/json" - }, - body: JSON.stringify({ - media: { - url: this.data.src - } - }) - }).then(r => r.json()); - const url = mediaJson.images.raw; + const url = this.data.src; + const contentType = this.data.contentType; if (textureCache.has(url)) { const cacheItem = textureCache.get(url); texture = cacheItem.texture; cacheItem.count++; } else { - const contentType = await fetch(url, { method: "HEAD" }).then(r => r.headers.get("content-type")); - if (contentType === "image/gif") { + if (url === "error") { + texture = errorTexture; + } else if (contentType === "image/gif") { console.log("load gif", contentType); texture = await this.loadGIF(url); } else if (contentType.startsWith("image/")) { diff --git a/src/hub.js b/src/hub.js index 12f6ccff2..9bddbbabb 100644 --- a/src/hub.js +++ b/src/hub.js @@ -81,6 +81,7 @@ import HubChannel from "./utils/hub-channel"; import LinkChannel from "./utils/link-channel"; import { connectToReticulum } from "./utils/phoenix-utils"; import { disableiOSZoom } from "./utils/disable-ios-zoom"; +import { addMedia } from "./utils/media-utils"; import "./systems/personal-space-bubble"; import "./systems/app-mode"; @@ -302,31 +303,6 @@ const onReady = async () => { NAF.connection.entities.completeSync(ev.detail.clientId); }); - const offset = { x: 0, y: 0, z: -1.5 }; - const addMedia = url => { - const scene = AFRAME.scenes[0]; - if (url.endsWith(".gltf") || url.endsWith(".glb")) { - const model = document.createElement("a-entity"); - model.id = "interactable-model-" + Date.now(); - model.setAttribute("offset-relative-to", { - on: "model-loaded", - target: "#player-camera", - offset: offset, - selfDestruct: true - }); - model.setAttribute("gltf-model-plus", "src", url); - model.setAttribute("auto-box-collider", { resize: true }); - model.setAttribute("networked", { template: "#interactable-model" }); - scene.appendChild(model); - } else { - const image = document.createElement("a-entity"); - image.id = "interactable-image-" + Date.now(); - image.setAttribute("image-plus", "src", url); - image.setAttribute("networked", { template: "#interactable-image" }); - scene.appendChild(image); - } - }; - scene.addEventListener("add_media", e => { addMedia(e.detail); }); diff --git a/src/utils/media-utils.js b/src/utils/media-utils.js new file mode 100644 index 000000000..b94ad6b88 --- /dev/null +++ b/src/utils/media-utils.js @@ -0,0 +1,64 @@ +const whitelistedHosts = [/^.*\.reticulum\.io$/, /^.*hubs\.mozilla\.com$/, /^hubs\.local$/]; +const isHostWhitelisted = hostname => !!whitelistedHosts.filter(r => r.test(hostname)).length; +export const resolveFarsparkUrl = async url => { + const parsedUrl = new URL(url); + if ((parsedUrl.protocol !== "http:" && parsedUrl.protocol !== "https:") || isHostWhitelisted(parsedUrl.hostname)) + return url; + + return (await fetch("https://dev.reticulum.io/api/v1/media", { + method: "POST", + headers: { "Content-Type": "application/json" }, + body: JSON.stringify({ media: { url } }) + }).then(r => r.json())).images.raw; +}; + +const fetchContentType = async url => fetch(url, { method: "HEAD" }).then(r => r.headers.get("content-type")); +let interactableId = 0; + +export const spawnNetworkedImage = (src, contentType) => { + const scene = AFRAME.scenes[0]; + const image = document.createElement("a-entity"); + image.id = "interactable-image-" + interactableId++; + image.setAttribute("networked", { template: "#interactable-image" }); + image.setAttribute("image-plus", { src, contentType }); + scene.appendChild(image); + return image; +}; + +const offset = { x: 0, y: 0, z: -1.5 }; +export const spawnNetworkedInteractable = src => { + const scene = AFRAME.scenes[0]; + const model = document.createElement("a-entity"); + model.id = "interactable-model-" + interactableId++; + model.setAttribute("networked", { template: "#interactable-model" }); + model.setAttribute("offset-relative-to", { + on: "model-loaded", + target: "#player-camera", + offset: offset, + selfDestruct: true + }); + model.setAttribute("gltf-model-plus", "src", src); + model.setAttribute("auto-box-collider", { resize: true }); + scene.appendChild(model); + return model; +}; + +export const addMedia = async url => { + try { + const farsparkUrl = await resolveFarsparkUrl(url); + console.log("resolved", url, farsparkUrl); + + const contentType = await fetchContentType(farsparkUrl); + + if (contentType.startsWith("image/") || contentType.startsWith("video/")) { + spawnNetworkedImage(farsparkUrl, contentType); + } else if (contentType.startsWith("model/gltf") || url.endsWith(".gltf") || url.endsWith(".glb")) { + spawnNetworkedInteractable(farsparkUrl); + } else { + throw new Error(`Unsupported content type: ${contentType}`); + } + } catch (e) { + console.error("Error adding media", e); + spawnNetworkedImage("error"); + } +}; diff --git a/src/vendor/GLTFLoader.js b/src/vendor/GLTFLoader.js index 1df68673f..906bb9465 100644 --- a/src/vendor/GLTFLoader.js +++ b/src/vendor/GLTFLoader.js @@ -9,18 +9,7 @@ * @author Don McCurdy / https://www.donmccurdy.com */ -async function toFarsparkURL(url){ - var mediaJson = await fetch("https://smoke-dev.reticulum.io/api/v1/media", { - method: "POST", - headers: { - "Content-Type": "application/json" - }, - body: JSON.stringify({ - media: {url} - }) - }).then(r => r.json()); - return mediaJson.images.raw; -} + import { resolveFarsparkUrl } from "../utils/media-utils" THREE.GLTFLoader = ( function () { @@ -50,7 +39,7 @@ THREE.GLTFLoader = ( function () { loader.setResponseType( 'arraybuffer' ); - var farsparkURL = await toFarsparkURL(url); + var farsparkURL = await resolveFarsparkUrl(url); loader.load( farsparkURL, function ( data ) { @@ -1633,7 +1622,7 @@ THREE.GLTFLoader = ( function () { var options = this.options; - var farsparkURL = await toFarsparkURL(resolveURL(bufferDef.uri, options.path)); + var farsparkURL = await resolveFarsparkUrl(resolveURL(bufferDef.uri, options.path)); return new Promise( function ( resolve, reject ) { @@ -1833,7 +1822,7 @@ THREE.GLTFLoader = ( function () { var urlToLoad = resolveURL(sourceURI, options.path); if (!hasBufferView){ - urlToLoad = await toFarsparkURL(urlToLoad); + urlToLoad = await resolveFarsparkUrl(urlToLoad); } return Promise.resolve( sourceURI ).then( function ( sourceURI ) { -- GitLab