From c33d634d0d4ed6e1fe5782c86d0fff6f54ba6ea0 Mon Sep 17 00:00:00 2001
From: Greg Fodor <gfodor@gmail.com>
Date: Sat, 10 Nov 2018 23:38:32 +0000
Subject: [PATCH] Add commands

---
 src/assets/translations.data.json      |  1 +
 src/components/character-controller.js |  2 +-
 src/hub.js                             | 47 ++++++++-------
 src/message-dispatch.js                | 83 ++++++++++++++++++++++++++
 src/react-components/presence-log.js   |  9 ++-
 src/react-components/ui-root.js        |  8 +--
 6 files changed, 121 insertions(+), 29 deletions(-)
 create mode 100644 src/message-dispatch.js

diff --git a/src/assets/translations.data.json b/src/assets/translations.data.json
index 819a5a046..ac6e0dbec 100644
--- a/src/assets/translations.data.json
+++ b/src/assets/translations.data.json
@@ -51,6 +51,7 @@
     "audio.granted-next": "Next",
     "exit.subtitle.exited": "Your session has ended. Refresh your browser to start a new one.",
     "exit.subtitle.closed": "This room is no longer available.",
+    "exit.subtitle.left": "You have left the room.",
     "exit.subtitle.full": "This room is full, please try again later.",
     "exit.subtitle.connect_error": "Unable to connect to this room, please try again later.",
     "exit.subtitle.version_mismatch": "The version you deployed is not available yet. Your browser will refresh in 5 seconds.",
diff --git a/src/components/character-controller.js b/src/components/character-controller.js
index e2a1aadbc..6eb743ec7 100644
--- a/src/components/character-controller.js
+++ b/src/components/character-controller.js
@@ -18,7 +18,7 @@ AFRAME.registerComponent("character-controller", {
     pivot: { type: "selector" },
     snapRotationDegrees: { default: THREE.Math.DEG2RAD * 45 },
     rotationSpeed: { default: -3 },
-    fly: { default: true }
+    fly: { default: false }
   },
 
   init: function() {
diff --git a/src/hub.js b/src/hub.js
index 553f2a3a1..b647cb9d3 100644
--- a/src/hub.js
+++ b/src/hub.js
@@ -76,6 +76,7 @@ import LinkChannel from "./utils/link-channel";
 import { connectToReticulum } from "./utils/phoenix-utils";
 import { disableiOSZoom } from "./utils/disable-ios-zoom";
 import { proxiedUrlFor } from "./utils/media-utils";
+import MessageDispatch from "./message-dispatch";
 import SceneEntryManager from "./scene-entry-manager";
 import Subscriptions from "./subscriptions";
 
@@ -213,7 +214,7 @@ function remountUI(props) {
   mountUI(uiProps);
 }
 
-async function handleHubChannelJoined(entryManager, hubChannel, data) {
+async function handleHubChannelJoined(entryManager, hubChannel, messageDispatch, data) {
   const scene = document.querySelector("a-scene");
 
   if (NAF.connection.isConnected()) {
@@ -251,7 +252,7 @@ async function handleHubChannelJoined(entryManager, hubChannel, data) {
     hubId: hub.hub_id,
     hubName: hub.name,
     hubEntryCode: hub.entry_code,
-    onSendMessage: hubChannel.sendMessage
+    onSendMessage: messageDispatch.dispatch
   });
 
   document
@@ -430,27 +431,7 @@ document.addEventListener("DOMContentLoaded", async () => {
   const joinPayload = { profile: store.state.profile, push_subscription_endpoint: pushSubscriptionEndpoint, context };
   const hubPhxChannel = socket.channel(`hub:${hubId}`, joinPayload);
 
-  hubPhxChannel
-    .join()
-    .receive("ok", async data => {
-      hubChannel.setPhoenixChannel(hubPhxChannel);
-      subscriptions.setHubChannel(hubChannel);
-      subscriptions.setSubscribed(data.subscriptions.web_push);
-      remountUI({ initialIsSubscribed: subscriptions.isSubscribed() });
-      await handleHubChannelJoined(entryManager, hubChannel, data);
-    })
-    .receive("error", res => {
-      if (res.reason === "closed") {
-        entryManager.exitScene();
-        remountUI({ roomUnavailableReason: "closed" });
-      }
-
-      console.error(res);
-    });
-
-  const hubPhxPresence = new Presence(hubPhxChannel);
   const presenceLogEntries = [];
-
   const addToPresenceLog = entry => {
     entry.key = Date.now().toString();
 
@@ -469,6 +450,28 @@ document.addEventListener("DOMContentLoaded", async () => {
     }, 20000);
   };
 
+  const messageDispatch = new MessageDispatch(scene, entryManager, hubChannel, addToPresenceLog, remountUI);
+
+  hubPhxChannel
+    .join()
+    .receive("ok", async data => {
+      hubChannel.setPhoenixChannel(hubPhxChannel);
+      subscriptions.setHubChannel(hubChannel);
+      subscriptions.setSubscribed(data.subscriptions.web_push);
+      remountUI({ initialIsSubscribed: subscriptions.isSubscribed() });
+      await handleHubChannelJoined(entryManager, hubChannel, messageDispatch, data);
+    })
+    .receive("error", res => {
+      if (res.reason === "closed") {
+        entryManager.exitScene();
+        remountUI({ roomUnavailableReason: "closed" });
+      }
+
+      console.error(res);
+    });
+
+  const hubPhxPresence = new Presence(hubPhxChannel);
+
   let isInitialSync = true;
 
   hubPhxPresence.onSync(() => {
diff --git a/src/message-dispatch.js b/src/message-dispatch.js
new file mode 100644
index 000000000..c165bb30d
--- /dev/null
+++ b/src/message-dispatch.js
@@ -0,0 +1,83 @@
+import { spawnChatMessage } from "./react-components/chat-message";
+const DUCK_URL = "https://asset-bundles-prod.reticulum.io/interactables/Ducky/DuckyMesh-438ff8e022.gltf";
+
+// Handles user-entered messages
+export default class MessageDispatch {
+  constructor(scene, entryManager, hubChannel, addToPresenceLog, remountUI) {
+    this.scene = scene;
+    this.entryManager = entryManager;
+    this.hubChannel = hubChannel;
+    this.addToPresenceLog = addToPresenceLog;
+    this.remountUI = remountUI;
+  }
+
+  dispatch = message => {
+    if (message.startsWith("/")) {
+      this.dispatchCommand(message.substring(1));
+      document.activeElement.blur(); // Commands should blur
+    } else {
+      this.hubChannel.sendMessage(message);
+    }
+  };
+
+  dispatchCommand = command => {
+    const entered = this.scene.is("entered");
+
+    switch (command) {
+      case "help":
+        // HACK for now, non-trivial to properly send this into React
+        document.querySelector(".help-button").click();
+        return;
+    }
+
+    if (!entered) {
+      this.addToPresenceLog({ type: "log", body: "You must enter the room to use this command." });
+      return;
+    }
+
+    const playerRig = document.querySelector("#player-rig");
+    const scales = [0.0625, 0.125, 0.25, 0.5, 1.0, 1.5, 3, 5, 7.5, 12.5];
+    const curScale = playerRig.object3D.scale;
+
+    switch (command) {
+      case "fly":
+        if (playerRig.getAttribute("character-controller").fly !== true) {
+          playerRig.setAttribute("character-controller", "fly", true);
+          this.addToPresenceLog({ type: "log", body: "Fly mode enabled." });
+        } else {
+          playerRig.setAttribute("character-controller", "fly", false);
+          this.addToPresenceLog({ type: "log", body: "Fly mode disabled." });
+        }
+        break;
+      case "bigger":
+        for (let i = 0; i < scales.length; i++) {
+          if (scales[i] > curScale.x) {
+            playerRig.object3D.scale.set(scales[i], scales[i], scales[i]);
+            break;
+          }
+        }
+
+        break;
+      case "smaller":
+        for (let i = scales.length - 1; i >= 0; i--) {
+          if (curScale.x > scales[i]) {
+            playerRig.object3D.scale.set(scales[i], scales[i], scales[i]);
+            break;
+          }
+        }
+
+        break;
+      case "leave":
+        this.entryManager.exitScene();
+        this.remountUI({ roomUnavailableReason: "left" });
+        break;
+      case "help":
+        // HACK for now, non-trivial to properly send this into React
+        document.querySelector(".help-button").click();
+        break;
+      case "duck":
+        spawnChatMessage(DUCK_URL);
+        break;
+    }
+  };
+}
diff --git a/src/react-components/presence-log.js b/src/react-components/presence-log.js
index 4426ac46e..7eabb323a 100644
--- a/src/react-components/presence-log.js
+++ b/src/react-components/presence-log.js
@@ -56,7 +56,7 @@ export default class PresenceLog extends Component {
             maySpawn={e.maySpawn}
           />
         );
-      case "spawn": {
+      case "spawn":
         return (
           <PhotoMessage
             key={e.key}
@@ -67,7 +67,12 @@ export default class PresenceLog extends Component {
             hubId={this.props.hubId}
           />
         );
-      }
+      case "log":
+        return (
+          <div key={e.key} className={classNames(entryClasses)}>
+            {e.body}
+          </div>
+        );
     }
   };
 
diff --git a/src/react-components/ui-root.js b/src/react-components/ui-root.js
index 77b0dfba4..0511b8edf 100644
--- a/src/react-components/ui-root.js
+++ b/src/react-components/ui-root.js
@@ -679,12 +679,12 @@ class UIRoot extends Component {
         <div>
           <FormattedMessage id={exitSubtitleId} />
           <p />
-          {this.props.roomUnavailableReason && (
+          {this.props.roomUnavailableReason !== "left" && (
             <div>
               You can also{" "}
               <WithHoverSound>
-                <a href="/">create a new room</a>.
-              </WithHoverSound>
+                <a href="/">create a new room</a>
+              </WithHoverSound>.
             </div>
           )}
         </div>
@@ -1209,7 +1209,7 @@ class UIRoot extends Component {
             )}
 
             <WithHoverSound>
-              <button onClick={() => this.showHelpDialog()} className={styles.helpIcon}>
+              <button onClick={() => this.showHelpDialog()} className={classNames([styles.helpIcon, "help-button"])}>
                 <i>
                   <FontAwesomeIcon icon={faQuestion} />
                 </i>
-- 
GitLab