From 846e90ddf371e9ae97769472072e35befb604e0a Mon Sep 17 00:00:00 2001
From: netpro2k <netpro2k@gmail.com>
Date: Wed, 25 Jul 2018 18:01:52 -0700
Subject: [PATCH] Move zip handling into gltf-model-plus so that blob urls can
 be cleaned up

---
 src/components/gltf-model-plus.js | 38 +++++++++++++++++++++++++++----
 src/components/media-loader.js    | 23 ++-----------------
 2 files changed, 36 insertions(+), 25 deletions(-)

diff --git a/src/components/gltf-model-plus.js b/src/components/gltf-model-plus.js
index 0b9755a4a..b32e05a95 100644
--- a/src/components/gltf-model-plus.js
+++ b/src/components/gltf-model-plus.js
@@ -1,3 +1,5 @@
+import JSZip from "jszip";
+
 const GLTFCache = {};
 
 AFRAME.GLTFModelPlus = {
@@ -181,14 +183,41 @@ function nextTick() {
   });
 }
 
-function cachedLoadGLTF(src, basePath, preferredTechnique, onProgress) {
+function cachedLoadGLTF(src, basePath, contentType, preferredTechnique, onProgress) {
   // Load the gltf model from the cache if it exists.
   if (!GLTFCache[src]) {
-    GLTFCache[src] = new Promise((resolve, reject) => {
+    GLTFCache[src] = new Promise(async (resolve, reject) => {
+      let gltfUrl = src;
+      let onLoad = resolve;
+      if (contentType === "model/gltf+zip") {
+        const zip = await fetch(src)
+          .then(r => r.blob())
+          .then(JSZip.loadAsync);
+
+        // Rewrite any url refferences in the GLTF to blob urls
+        const gltfJson = JSON.parse(await zip.file("scene.gltf").async("text"));
+        const fileMap = await Object.values(zip.files).reduce(async (prev, file) => {
+          if (file.name === "scene.gltf") return prev;
+          const out = await prev;
+          out[file.name] = URL.createObjectURL(await file.async("blob"));
+          return out;
+        }, Promise.resolve({}));
+        gltfJson.buffers && gltfJson.buffers.forEach(b => (b.uri = fileMap[b.uri]));
+        gltfJson.images && gltfJson.images.forEach(i => (i.uri = fileMap[i.uri]));
+
+        gltfUrl = fileMap["scene.gtlf"] = URL.createObjectURL(
+          new Blob([JSON.stringify(gltfJson, null, 2)], { type: "text/plain" })
+        );
+
+        onLoad = model => {
+          Object.keys(fileMap).forEach(URL.revokeObjectURL);
+          resolve(model);
+        };
+      }
       const gltfLoader = new THREE.GLTFLoader();
       gltfLoader.path = basePath;
       gltfLoader.preferredTechnique = preferredTechnique;
-      gltfLoader.load(src, resolve, onProgress, reject);
+      gltfLoader.load(gltfUrl, onLoad, onProgress, reject);
     });
   }
   return GLTFCache[src].then(cloneGltf);
@@ -203,6 +232,7 @@ function cachedLoadGLTF(src, basePath, preferredTechnique, onProgress) {
 AFRAME.registerComponent("gltf-model-plus", {
   schema: {
     src: { type: "string" },
+    contentType: { type: "string" },
     basePath: { type: "string", default: undefined },
     inflate: { default: false }
   },
@@ -244,7 +274,7 @@ AFRAME.registerComponent("gltf-model-plus", {
       }
 
       const gltfPath = THREE.LoaderUtils.extractUrlBase(src);
-      const model = await cachedLoadGLTF(src, this.data.basePath, this.preferredTechnique);
+      const model = await cachedLoadGLTF(src, this.data.basePath, this.data.contentType, this.preferredTechnique);
 
       // If we started loading something else already
       // TODO: there should be a way to cancel loading instead
diff --git a/src/components/media-loader.js b/src/components/media-loader.js
index d760eedc4..4cee88ae2 100644
--- a/src/components/media-loader.js
+++ b/src/components/media-loader.js
@@ -1,5 +1,3 @@
-import JSZip from "jszip";
-
 import { getBox, getScaleCoefficient } from "../utils/auto-box-collider";
 import { resolveFarsparkUrl } from "../utils/media-utils";
 
@@ -70,24 +68,6 @@ AFRAME.registerComponent("media-loader", {
         this.el.setAttribute("image-plus", { src: raw, contentType });
         this.el.setAttribute("position-at-box-shape-border", { target: ".delete-button", dirs: ["forward", "back"] });
       } else if (contentType.startsWith("model/gltf") || url.endsWith(".gltf") || url.endsWith(".glb")) {
-        let gltfUrl = raw;
-        if (contentType === "model/gltf+zip") {
-          gltfUrl = await fetch(raw)
-            .then(r => r.blob())
-            .then(JSZip.loadAsync)
-            .then(async zip => {
-              const gltf = JSON.parse(await zip.file("scene.gltf").async("text"));
-              const fileMap = await Object.values(zip.files).reduce(async (prev, file) => {
-                if (file.name === "scene.gltf") return prev;
-                const out = await prev;
-                out[file.name] = URL.createObjectURL(await file.async("blob"));
-                return out;
-              }, Promise.resolve({}));
-              gltf.buffers && gltf.buffers.forEach(b => (b.uri = fileMap[b.uri]));
-              gltf.images && gltf.images.forEach(i => (i.uri = fileMap[i.uri]));
-              return URL.createObjectURL(new Blob([JSON.stringify(gltf, null, 2)], { type: "text/plain" }));
-            });
-        }
         this.el.addEventListener(
           "model-loaded",
           () => {
@@ -98,7 +78,8 @@ AFRAME.registerComponent("media-loader", {
         );
         this.el.addEventListener("model-error", this.onError, { once: true });
         this.el.setAttribute("gltf-model-plus", {
-          src: gltfUrl,
+          src: raw,
+          contentType,
           basePath: THREE.LoaderUtils.extractUrlBase(origin),
           inflate: true
         });
-- 
GitLab