diff --git a/src/components/text-button.js b/src/components/text-button.js
new file mode 100644
index 0000000000000000000000000000000000000000..f858faaec7da928cc06ea9691b1584416709247b
--- /dev/null
+++ b/src/components/text-button.js
@@ -0,0 +1,55 @@
+AFRAME.registerComponent("text-button", {
+  schema: {
+    haptic: { type: "selector" },
+    textEl: { type: "string" },
+    textHoverColor: { type: "string" },
+    textColor: { type: "string" },
+    backgroundHoverColor: { type: "string" },
+    backgroundColor: { type: "string" }
+  },
+
+  init() {
+    this.onHover = () => {
+      this.hovering = true;
+      this.updateButtonState();
+      this.emitHapticPulse();
+    };
+    this.onHoverOut = () => {
+      this.hovering = false;
+      this.updateButtonState();
+    };
+    this.onClick = () => {
+      this.emitHapticPulse();
+    };
+    this.textEl = this.el.querySelector(this.data.textEl);
+  },
+
+  emitHapticPulse() {
+    if (this.data.haptic) {
+      this.data.haptic.emit("haptic_pulse", { intensity: "low" });
+    }
+  },
+
+  play() {
+    this.updateButtonState();
+    this.el.addEventListener("mouseover", this.onHover);
+    this.el.addEventListener("mouseout", this.onHoverOut);
+    this.el.addEventListener("click", this.onClick);
+  },
+
+  pause() {
+    this.el.removeEventListener("mouseover", this.onHover);
+    this.el.removeEventListener("mouseout", this.onHoverOut);
+    this.el.removeEventListener("click", this.onClick);
+  },
+
+  update() {
+    this.updateButtonState();
+  },
+
+  updateButtonState() {
+    const hovering = this.hovering;
+    this.el.setAttribute("slice9", "color", hovering ? this.data.backgroundHoverColor : this.data.backgroundColor);
+    this.textEl.setAttribute("text", "color", hovering ? this.data.textHoverColor : this.data.textColor);
+  }
+});
diff --git a/src/components/visible-while-frozen.js b/src/components/visible-while-frozen.js
new file mode 100644
index 0000000000000000000000000000000000000000..15724b8d94d2bda92f7e67a4aaa51c058816bce9
--- /dev/null
+++ b/src/components/visible-while-frozen.js
@@ -0,0 +1,19 @@
+AFRAME.registerComponent("visible-while-frozen", {
+  init() {
+    this.onStateChange = evt => {
+      if (!evt.detail === "frozen") return;
+      this.el.setAttribute("visible", this.el.sceneEl.is("frozen"));
+    };
+    this.el.setAttribute("visible", this.el.sceneEl.is("frozen"));
+  },
+
+  play() {
+    this.el.sceneEl.addEventListener("stateadded", this.onStateChange);
+    this.el.sceneEl.addEventListener("stateremoved", this.onStateChange);
+  },
+
+  pause() {
+    this.el.sceneEl.removeEventListener("stateadded", this.onStateChange);
+    this.el.sceneEl.removeEventListener("stateremoved", this.onStateChange);
+  }
+});
diff --git a/src/hub.html b/src/hub.html
index a80599f4fd9d6a125eed5c5bb33ab9094915de23..aff18e1d4e5cc8b54d69bdef2d2a2c7f79a78b9d 100644
--- a/src/hub.html
+++ b/src/hub.html
@@ -94,8 +94,29 @@
                             <a-entity>
                               <a-entity personal-space-invader="radius: 0.2; useMaterial: true;" bone-visibility> </a-entity>
                               <a-entity billboard>
-                                  <a-entity slice9="width: 0.45; height: 0.2; left: 53; top: 53; right: 10; bottom: 10; opacity: 1.3; src: #tooltip" position="0 0 .35">
-                                      <a-entity text="value:Block;width:3;align:center;" position="0 0 0.01"></a-entity>
+                                  <a-entity
+                                      visible-while-frozen
+                                      text-button="haptic:#player-right-controller;
+                                                   textEl:.text;
+                                                   textHoverColor: #f00;
+                                                   textColor: #fff;
+                                                   backgroundHoverColor: #f00;
+                                                   backgroundColor: #fff;
+                                                   "
+                                      slice9="width: 0.45;
+                                                   height: 0.2;
+                                                   left: 53;
+                                                   top: 53;
+                                                   right: 10;
+                                                   bottom: 10;
+                                                   opacity: 1.3;
+                                                   src: #tooltip"
+                                      position="0 0 .35">
+                                      <a-entity class="text"
+                                                text="value:Block;
+                                                       width:3;
+                                                       align:center;"
+                                                position="0 0 0.01"></a-entity>
                                   </a-entity>
                               </a-entity>
                             </a-entity>
diff --git a/src/hub.js b/src/hub.js
index f04c5ae7db83c8396186f1017202b5b4704c5167..9728c6217877071021544431fa0134697329a93d 100644
--- a/src/hub.js
+++ b/src/hub.js
@@ -52,6 +52,8 @@ import "./components/gltf-bundle";
 import "./components/hud-controller";
 import "./components/freeze-controller";
 import "./components/icon-button";
+import "./components/text-button";
+import "./components/visible-while-frozen";
 import "./components/stats-plus";
 
 import ReactDOM from "react-dom";