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";