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); + } +});