diff --git a/src/assets/translations.data.json b/src/assets/translations.data.json index 1b9844c8abf2dc7d9ed7c089f84923faa601fd96..9948dfdc528fbdd8fdc3881ef59e0d2543ae18ca 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", @@ -33,7 +32,7 @@ "audio.granted-title": "Mic permissions granted", "audio.granted-subtitle": "You can still mute yourself in-game", "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 d8c357d2512452b58dbde4784f4130caf516235a..cf096150c926e89d4999d87cd0ff34210dc73cd3 100644 --- a/src/hub.js +++ b/src/hub.js @@ -58,6 +58,7 @@ import HubChannel from "./utils/hub-channel"; import "./systems/personal-space-bubble"; import "./systems/app-mode"; +import "./systems/exit-on-blur"; import "./gltf-component-mappings"; @@ -124,6 +125,9 @@ if (!store.state.profile.has_changed_name) { } async function exitScene() { + if (NAF.connection.adapter && NAF.connection.adapter.localMediaStream) { + NAF.connection.adapter.localMediaStream.getTracks().forEach(t => t.stop()); + } hubChannel.disconnect(); const scene = document.querySelector("a-scene"); scene.renderer.animate(null); // Stop animation loop, TODO A-Frame should do this diff --git a/src/react-components/ui-root.js b/src/react-components/ui-root.js index 3e22208dacc6219b59cc9276562c71d43f129f03..2883d23f066498e941d561b6722b9fc5b434590c 100644 --- a/src/react-components/ui-root.js +++ b/src/react-components/ui-root.js @@ -107,6 +107,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); } componentDidUpdate(prevProps) { 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); + } +});