diff --git a/src/assets/translations.data.json b/src/assets/translations.data.json
index 1757ec61cce5b97855750537a7812efe5349a57e..48bd0d1b4db58dfc385d902cd38b69e1ed2945a4 100644
--- a/src/assets/translations.data.json
+++ b/src/assets/translations.data.json
@@ -1,6 +1,5 @@
 {
-  "en":
-  {
+  "en": {
     "entry.screen-prefix": "Enter on ",
     "entry.desktop-screen": "Screen",
     "entry.mobile-screen": "Phone",
@@ -28,7 +27,7 @@
     "audio.granted-subtitle": "You can still mute yourself in-game",
     "audio.grant-next": "  ",
     "audio.granted-next": "NEXT",
-    "exit.subtitle": "Your session has ended.",
+    "exit.subtitle": "Your session has ended. Refresh your browser to start a new one.",
     "autoexit.title": "Auto-ending session in ",
     "autoexit.title_units": " seconds",
     "autoexit.subtitle": "You have started another session.",
diff --git a/src/hub.js b/src/hub.js
index 9dc3247e3a033ba8bb08ddad32a33a0c9a899e8b..1c68e22da8b5681860ba197f81ec64ce663158c5 100644
--- a/src/hub.js
+++ b/src/hub.js
@@ -53,6 +53,7 @@ import UIRoot from "./react-components/ui-root";
 
 import "./systems/personal-space-bubble";
 import "./systems/app-mode";
+import "./systems/exit-on-blur";
 
 import "./gltf-component-mappings";
 
@@ -113,6 +114,13 @@ concurrentLoadDetector.start();
 store.update({ profile: { ...generateDefaultProfile(), ...(store.state.profile || {}) } });
 
 async function exitScene() {
+  if (NAF.connection.adapter && NAF.connection.adapter.localMediaStream) {
+    const tracks = NAF.connection.adapter.localMediaStream.getTracks();
+
+    for (const track of tracks) {
+      track.stop();
+    }
+  }
   const scene = document.querySelector("a-scene");
   scene.renderer.animate(null); // Stop animation loop, TODO A-Frame should do this
   document.body.removeChild(scene);
diff --git a/src/react-components/ui-root.js b/src/react-components/ui-root.js
index 282f4b4572156fb4e757616a1e8c02e19df34727..39983ef6baa59f529ccc0c9ed985c1efad4b7955 100644
--- a/src/react-components/ui-root.js
+++ b/src/react-components/ui-root.js
@@ -102,10 +102,12 @@ class UIRoot extends Component {
     this.props.scene.addEventListener("loaded", this.onSceneLoaded);
     this.props.scene.addEventListener("stateadded", this.onAframeStateChanged);
     this.props.scene.addEventListener("stateremoved", this.onAframeStateChanged);
+    this.props.scene.addEventListener("exit", this.exit);
   }
 
   componentWillUnmount() {
     this.props.scene.removeEventListener("loaded", this.onSceneLoaded);
+    this.props.scene.removeEventListener("exit", this.exit);
   }
 
   onSceneLoaded = () => {
diff --git a/src/systems/exit-on-blur.js b/src/systems/exit-on-blur.js
new file mode 100644
index 0000000000000000000000000000000000000000..e6263a101f5a2fd1f71cd4b122a12431104e2457
--- /dev/null
+++ b/src/systems/exit-on-blur.js
@@ -0,0 +1,30 @@
+AFRAME.registerSystem("exit-on-blur", {
+  init() {
+    this.onBlur = this.onBlur.bind(this);
+    this.onFocus = this.onFocus.bind(this);
+
+    window.addEventListener("blur", this.onBlur);
+    window.addEventListener("focus", this.onFocus);
+
+    this.exitTimeout = null;
+  },
+
+  onBlur() {
+    if (this.el.isMobile) {
+      this.exitTimeout = setTimeout(() => {
+        this.el.dispatchEvent(new CustomEvent("exit"));
+      }, 10 * 1000);
+    }
+  },
+
+  onFocus() {
+    if (this.el.isMobile) {
+      clearTimeout(this.exitTimeout);
+    }
+  },
+
+  remove() {
+    clearTimeout(this.exitTimeout);
+    window.removeEventListener("blur", this.onBlur);
+  }
+});