diff --git a/src/components/image-plus.js b/src/components/image-plus.js
index 6f0163d23bb52ac1974077573d99f2408ab72640..a7efa5508fdce28031d1f1171e7c5b19b5944538 100644
--- a/src/components/image-plus.js
+++ b/src/components/image-plus.js
@@ -32,8 +32,40 @@ class GIFTexture extends THREE.Texture {
   }
 }
 
+/**
+ * Create video element to be used as a texture.
+ *
+ * @param {string} src - Url to a video file.
+ * @returns {Element} Video element.
+ */
+function createVideoEl(src) {
+  const videoEl = document.createElement("video");
+  videoEl.setAttribute("playsinline", "");
+  videoEl.setAttribute("webkit-playsinline", "");
+  videoEl.autoplay = true;
+  videoEl.loop = true;
+  videoEl.crossOrigin = "anonymous";
+  videoEl.src = src;
+  return videoEl;
+}
+
+const textureLoader = new THREE.TextureLoader();
+textureLoader.setCrossOrigin("anonymous");
+
+const textureCache = new Map();
+
+const noop = function() {};
+
+const errorImage = new Image();
+errorImage.src =
+  "";
+const errorTexture = new THREE.Texture(errorImage);
+errorImage.onload = () => {
+  errorTexture.needsUpdate = true;
+};
+
 AFRAME.registerComponent("image-plus", {
-  dependencies: ["geometry", "material"],
+  dependencies: ["geometry"],
 
   schema: {
     src: { type: "string" },
@@ -68,15 +100,6 @@ AFRAME.registerComponent("image-plus", {
     });
   },
 
-  _onMaterialLoaded(e) {
-    const src = e.detail.src;
-    const w = src.videoWidth || src.width;
-    const h = src.videoHeight || src.height;
-    if (w || h) {
-      this._fit(w, h);
-    }
-  },
-
   _onGrab: (function() {
     const q = new THREE.Quaternion();
     return function() {
@@ -88,13 +111,15 @@ AFRAME.registerComponent("image-plus", {
   })(),
 
   init() {
-    this._onMaterialLoaded = this._onMaterialLoaded.bind(this);
     this._onGrab = this._onGrab.bind(this);
 
-    this.el.addEventListener("materialtextureloaded", this._onMaterialLoaded);
-    this.el.addEventListener("materialvideoloadeddata", this._onMaterialLoaded);
     this.el.addEventListener("grab-start", this._onGrab);
 
+    const material = new THREE.MeshBasicMaterial();
+    material.side = THREE.DoubleSide;
+    material.transparent = true;
+    this.el.getObject3D("mesh").material = material;
+
     const worldPos = new THREE.Vector3().copy(this.data.initialOffset);
     this.billboardTarget = document.querySelector("#player-camera").object3D;
     this.billboardTarget.localToWorld(worldPos);
@@ -102,54 +127,151 @@ AFRAME.registerComponent("image-plus", {
     this.billboardTarget.getWorldQuaternion(this.el.object3D.quaternion);
   },
 
-  async loadGIF(url) {
-    const worker = new GIFWorker();
-    worker.onmessage = e => {
-      const [success, frames, delays, disposals] = e.data;
-      if (!success) {
-        console.error("error loading gif", e.data[1]);
-        return;
+  remove() {
+    const material = this.el.getObject3D("mesh").material;
+    const texture = material.map;
+
+    if (texture === errorTexture) return;
+
+    const url = texture.image.src;
+    const cacheItem = textureCache.get(url);
+    cacheItem.count--;
+    if (cacheItem.count <= 0) {
+      console.log("removing from cache");
+      // Unload the video element to prevent it from continuing to play in the background
+      if (texture.image instanceof HTMLVideoElement) {
+        const video = texture.image;
+        video.pause();
+        video.src = "";
+        video.load();
       }
 
-      let loadCnt = 0;
-      for (let i = 0; i < frames.length; i++) {
-        const img = new Image();
-        img.onload = e => {
-          loadCnt++;
-          frames[i] = e.target;
-          if (loadCnt === frames.length) {
-            const material = this.el.components.material.material;
-            material.map = new GIFTexture(frames, delays, disposals);
-            material.needsUpdate = true;
-            this._fit(frames[0].width, frames[0].height);
-          }
-        };
-        img.src = frames[i];
+      texture.dispose();
+
+      // THREE never lets go of material refs, long running PR HERE https://github.com/mrdoob/three.js/pull/12464
+      // Mitigate the damage a bit by at least breaking the image ref so Image/Video elements can be freed
+      // TODO: If/when THREE gets fixed, we should be able to safely remove this
+      delete texture.image;
+
+      textureCache.delete(url);
+    }
+  },
+
+  async loadGIF(url) {
+    return new Promise((resolve, reject) => {
+      // TODO: pool workers
+      const worker = new GIFWorker();
+      worker.onmessage = e => {
+        const [success, frames, delays, disposals] = e.data;
+        if (!success) {
+          reject(`error loading gif: ${e.data[1]}`);
+          return;
+        }
+
+        let loadCnt = 0;
+        for (let i = 0; i < frames.length; i++) {
+          const img = new Image();
+          img.onload = e => {
+            loadCnt++;
+            frames[i] = e.target;
+            if (loadCnt === frames.length) {
+              const texture = new GIFTexture(frames, delays, disposals);
+              texture.image.src = url;
+              resolve(texture);
+            }
+          };
+          img.src = frames[i];
+        }
+      };
+      fetch(url, { mode: "cors" })
+        .then(r => r.arrayBuffer())
+        .then(rawImageData => {
+          worker.postMessage(rawImageData, [rawImageData]);
+        })
+        .catch(reject);
+    });
+  },
+
+  loadVideo(url) {
+    return new Promise((resolve, reject) => {
+      const videoEl = createVideoEl(url);
+
+      const texture = new THREE.VideoTexture(videoEl);
+      texture.minFilter = THREE.LinearFilter;
+      videoEl.addEventListener("loadedmetadata", () => resolve(texture));
+      videoEl.onerror = reject;
+
+      // If iOS and video is HLS, do some hacks.
+      if (
+        this.el.sceneEl.isIOS &&
+        AFRAME.utils.material.isHLS(
+          videoEl.src || videoEl.getAttribute("src"),
+          videoEl.type || videoEl.getAttribute("type")
+        )
+      ) {
+        // Actually BGRA. Tell shader to correct later.
+        texture.format = THREE.RGBAFormat;
+        texture.needsCorrectionBGRA = true;
+        // Apparently needed for HLS. Tell shader to correct later.
+        texture.flipY = false;
+        texture.needsCorrectionFlipY = true;
       }
-    };
-    const rawImageData = await fetch(url, { mode: "cors" }).then(r => r.arrayBuffer());
-    worker.postMessage(rawImageData, [rawImageData]);
+    });
+  },
+
+  loadImage(url) {
+    return new Promise((resolve, reject) => {
+      textureLoader.load(url, resolve, noop, function(xhr) {
+        reject(`'${url}' could not be fetched (Error code: ${xhr.status}; Response: ${xhr.statusText})`);
+      });
+    });
   },
 
   async update() {
-    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
+    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;
+
+      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") {
+          console.log("load gif", contentType);
+          texture = await this.loadGIF(url);
+        } else if (contentType.startsWith("image/")) {
+          console.log("load image", contentType);
+          texture = await this.loadImage(url);
+        } else if (contentType.startsWith("video")) {
+          console.log("load video", contentType);
+          texture = await this.loadVideo(url);
+        } else {
+          throw new Error(`Unknown centent type: ${contentType}`);
         }
-      })
-    }).then(r => r.json());
-    const imageUrl = mediaJson.images.raw;
-    const contentType = await fetch(imageUrl, { method: "HEAD" }).then(r => r.headers.get("content-type"));
-    if (contentType === "image/gif") {
-      return this.loadGIF(imageUrl);
-    } else {
-      this.el.setAttribute("material", "src", `url(${imageUrl})`);
-      return Promise.resolve();
+
+        textureCache.set(url, { count: 1, texture });
+      }
+    } catch (e) {
+      console.error("Error loading media", this.data.src, e);
+      texture = errorTexture;
     }
+
+    const material = this.el.getObject3D("mesh").material;
+    material.map = texture;
+    material.needsUpdate = true;
+    this._fit(texture.image.videoWidth || texture.image.width, texture.image.videoHeight || texture.image.height);
   }
 });
diff --git a/src/components/sticky-object.js b/src/components/sticky-object.js
index 9e9d8bff32b729edc341d8d7fb424abbfc7e166f..084004b8831c2626e0a8fd278c11767dc5008c62 100644
--- a/src/components/sticky-object.js
+++ b/src/components/sticky-object.js
@@ -12,11 +12,20 @@ AFRAME.registerComponent("sticky-object", {
     this._onGrab = this._onGrab.bind(this);
     this._onRelease = this._onRelease.bind(this);
     this._onBodyLoaded = this._onBodyLoaded.bind(this);
+  },
+
+  play() {
     this.el.addEventListener("grab-start", this._onGrab);
     this.el.addEventListener("grab-end", this._onRelease);
     this.el.addEventListener("body-loaded", this._onBodyLoaded);
   },
 
+  pause() {
+    this.el.removeEventListener("grab-start", this._onGrab);
+    this.el.removeEventListener("grab-end", this._onRelease);
+    this.el.removeEventListener("body-loaded", this._onBodyLoaded);
+  },
+
   setLocked(locked) {
     if (!NAF.utils.isMine(this.el)) return;
 
diff --git a/src/hub.html b/src/hub.html
index 1581794db7e7a26007a0ce9ee1c99daae1452835..a75e5fa915dd613e33c9a3f50be78f248326732b 100644
--- a/src/hub.html
+++ b/src/hub.html
@@ -202,7 +202,6 @@
                     stretchable="useWorldPosition: true; usePhysics: never"
                     hoverable
                     geometry="primitive: plane"
-                    material="shader: flat; side: double; transparent: true;"
                     image-plus
                     sticky-object="autoLockOnLoad: true; autoLockOnRelease: true;"
                     position-at-box-shape-border="target:.delete-button;dirs:forward,back"