232 lines
		
	
	
		
			8.9 KiB
		
	
	
	
		
			TypeScript
		
	
	
			
		
		
	
	
			232 lines
		
	
	
		
			8.9 KiB
		
	
	
	
		
			TypeScript
		
	
	
| /*
 | |
| Copyright 2015, 2016 OpenMarket Ltd
 | |
| Copyright 2019 New Vector Ltd
 | |
| 
 | |
| Licensed under the Apache License, Version 2.0 (the "License");
 | |
| you may not use this file except in compliance with the License.
 | |
| You may obtain a copy of the License at
 | |
| 
 | |
|     http://www.apache.org/licenses/LICENSE-2.0
 | |
| 
 | |
| Unless required by applicable law or agreed to in writing, software
 | |
| distributed under the License is distributed on an "AS IS" BASIS,
 | |
| WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | |
| See the License for the specific language governing permissions and
 | |
| limitations under the License.
 | |
| */
 | |
| 
 | |
| import dis from "./dispatcher/dispatcher";
 | |
| import Timer from "./utils/Timer";
 | |
| 
 | |
| // important these are larger than the timeouts of timers
 | |
| // used with UserActivity.timeWhileActive*,
 | |
| // such as READ_MARKER_INVIEW_THRESHOLD_MS (timeWhileActiveRecently),
 | |
| // READ_MARKER_OUTOFVIEW_THRESHOLD_MS (timeWhileActiveRecently),
 | |
| // READ_RECEIPT_INTERVAL_MS (timeWhileActiveNow) in TimelinePanel
 | |
| 
 | |
| // 'Under a few seconds'. Must be less than 'RECENTLY_ACTIVE_THRESHOLD_MS'
 | |
| const CURRENTLY_ACTIVE_THRESHOLD_MS = 700;
 | |
| 
 | |
| // 'Under a few minutes'.
 | |
| const RECENTLY_ACTIVE_THRESHOLD_MS = 2 * 60 * 1000;
 | |
| 
 | |
| /**
 | |
|  * This class watches for user activity (moving the mouse or pressing a key)
 | |
|  * and starts/stops attached timers while the user is active.
 | |
|  *
 | |
|  * There are two classes of 'active': 'active now' and 'active recently'
 | |
|  * see doc on the userActive* functions for what these mean.
 | |
|  */
 | |
| export default class UserActivity {
 | |
|     private readonly activeNowTimeout: Timer;
 | |
|     private readonly activeRecentlyTimeout: Timer;
 | |
|     private attachedActiveNowTimers: Timer[] = [];
 | |
|     private attachedActiveRecentlyTimers: Timer[] = [];
 | |
|     private lastScreenX = 0;
 | |
|     private lastScreenY = 0;
 | |
| 
 | |
|     public constructor(private readonly window: Window, private readonly document: Document) {
 | |
|         this.activeNowTimeout = new Timer(CURRENTLY_ACTIVE_THRESHOLD_MS);
 | |
|         this.activeRecentlyTimeout = new Timer(RECENTLY_ACTIVE_THRESHOLD_MS);
 | |
|     }
 | |
| 
 | |
|     public static sharedInstance(): UserActivity {
 | |
|         if (window.mxUserActivity === undefined) {
 | |
|             window.mxUserActivity = new UserActivity(window, document);
 | |
|         }
 | |
|         return window.mxUserActivity;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Runs the given timer while the user is 'active now', aborting when the user is no longer
 | |
|      * considered currently active.
 | |
|      * See userActiveNow() for what it means for a user to be 'active'.
 | |
|      * Can be called multiple times with the same already running timer, which is a NO-OP.
 | |
|      * Can be called before the user becomes active, in which case it is only started
 | |
|      * later on when the user does become active.
 | |
|      * @param {Timer} timer the timer to use
 | |
|      */
 | |
|     public timeWhileActiveNow(timer: Timer): void {
 | |
|         this.timeWhile(timer, this.attachedActiveNowTimers);
 | |
|         if (this.userActiveNow()) {
 | |
|             timer.start();
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Runs the given timer while the user is 'active' now or recently,
 | |
|      * aborting when the user becomes inactive.
 | |
|      * See userActiveRecently() for what it means for a user to be 'active recently'.
 | |
|      * Can be called multiple times with the same already running timer, which is a NO-OP.
 | |
|      * Can be called before the user becomes active, in which case it is only started
 | |
|      * later on when the user does become active.
 | |
|      * @param {Timer} timer the timer to use
 | |
|      */
 | |
|     public timeWhileActiveRecently(timer: Timer): void {
 | |
|         this.timeWhile(timer, this.attachedActiveRecentlyTimers);
 | |
|         if (this.userActiveRecently()) {
 | |
|             timer.start();
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     private timeWhile(timer: Timer, attachedTimers: Timer[]): void {
 | |
|         // important this happens first
 | |
|         const index = attachedTimers.indexOf(timer);
 | |
|         if (index === -1) {
 | |
|             attachedTimers.push(timer);
 | |
|             // remove when done or aborted
 | |
|             timer
 | |
|                 .finished()
 | |
|                 .finally(() => {
 | |
|                     const index = attachedTimers.indexOf(timer);
 | |
|                     if (index !== -1) {
 | |
|                         // should never be -1
 | |
|                         attachedTimers.splice(index, 1);
 | |
|                     }
 | |
|                     // as we fork the promise here,
 | |
|                     // avoid unhandled rejection warnings
 | |
|                 })
 | |
|                 .catch((err) => {});
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Start listening to user activity
 | |
|      */
 | |
|     public start(): void {
 | |
|         this.document.addEventListener("mousedown", this.onUserActivity);
 | |
|         this.document.addEventListener("mousemove", this.onUserActivity);
 | |
|         this.document.addEventListener("keydown", this.onUserActivity);
 | |
|         this.document.addEventListener("visibilitychange", this.onPageVisibilityChanged);
 | |
|         this.window.addEventListener("blur", this.onWindowBlurred);
 | |
|         this.window.addEventListener("focus", this.onUserActivity);
 | |
|         // can't use document.scroll here because that's only the document
 | |
|         // itself being scrolled. Need to use addEventListener's useCapture.
 | |
|         // also this needs to be the wheel event, not scroll, as scroll is
 | |
|         // fired when the view scrolls down for a new message.
 | |
|         this.window.addEventListener("wheel", this.onUserActivity, {
 | |
|             passive: true,
 | |
|             capture: true,
 | |
|         });
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Stop tracking user activity
 | |
|      */
 | |
|     public stop(): void {
 | |
|         this.document.removeEventListener("mousedown", this.onUserActivity);
 | |
|         this.document.removeEventListener("mousemove", this.onUserActivity);
 | |
|         this.document.removeEventListener("keydown", this.onUserActivity);
 | |
|         this.window.removeEventListener("wheel", this.onUserActivity, {
 | |
|             capture: true,
 | |
|         });
 | |
|         this.document.removeEventListener("visibilitychange", this.onPageVisibilityChanged);
 | |
|         this.window.removeEventListener("blur", this.onWindowBlurred);
 | |
|         this.window.removeEventListener("focus", this.onUserActivity);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Return true if the user is currently 'active'
 | |
|      * A user is 'active' while they are interacting with the app and for a very short (<1s)
 | |
|      * time after that. This is intended to give a strong indication that the app has the
 | |
|      * user's attention at any given moment.
 | |
|      * @returns {boolean} true if user is currently 'active'
 | |
|      */
 | |
|     public userActiveNow(): boolean {
 | |
|         return this.activeNowTimeout.isRunning();
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Return true if the user is currently active or has been recently
 | |
|      * A user is 'active recently' for a longer period of time (~2 mins) after
 | |
|      * they have been 'active' and while the app still has the focus. This is
 | |
|      * intended to indicate when the app may still have the user's attention
 | |
|      * (or they may have gone to make tea and left the window focused).
 | |
|      * @returns {boolean} true if user has been active recently
 | |
|      */
 | |
|     public userActiveRecently(): boolean {
 | |
|         return this.activeRecentlyTimeout.isRunning();
 | |
|     }
 | |
| 
 | |
|     private onPageVisibilityChanged = (e: Event): void => {
 | |
|         if (this.document.visibilityState === "hidden") {
 | |
|             this.activeNowTimeout.abort();
 | |
|             this.activeRecentlyTimeout.abort();
 | |
|         } else {
 | |
|             this.onUserActivity(e);
 | |
|         }
 | |
|     };
 | |
| 
 | |
|     private onWindowBlurred = (): void => {
 | |
|         this.activeNowTimeout.abort();
 | |
|         this.activeRecentlyTimeout.abort();
 | |
|     };
 | |
| 
 | |
|     // XXX: exported for tests
 | |
|     public onUserActivity = (event: Event): void => {
 | |
|         // ignore anything if the window isn't focused
 | |
|         if (!this.document.hasFocus()) return;
 | |
| 
 | |
|         if (event.type === "mousemove" && this.isMouseEvent(event)) {
 | |
|             if (event.screenX === this.lastScreenX && event.screenY === this.lastScreenY) {
 | |
|                 // mouse hasn't actually moved
 | |
|                 return;
 | |
|             }
 | |
|             this.lastScreenX = event.screenX;
 | |
|             this.lastScreenY = event.screenY;
 | |
|         }
 | |
| 
 | |
|         dis.dispatch({ action: "user_activity" });
 | |
|         if (!this.activeNowTimeout.isRunning()) {
 | |
|             this.activeNowTimeout.start();
 | |
|             dis.dispatch({ action: "user_activity_start" });
 | |
| 
 | |
|             UserActivity.runTimersUntilTimeout(this.attachedActiveNowTimers, this.activeNowTimeout);
 | |
|         } else {
 | |
|             this.activeNowTimeout.restart();
 | |
|         }
 | |
| 
 | |
|         if (!this.activeRecentlyTimeout.isRunning()) {
 | |
|             this.activeRecentlyTimeout.start();
 | |
| 
 | |
|             UserActivity.runTimersUntilTimeout(this.attachedActiveRecentlyTimers, this.activeRecentlyTimeout);
 | |
|         } else {
 | |
|             this.activeRecentlyTimeout.restart();
 | |
|         }
 | |
|     };
 | |
| 
 | |
|     private static async runTimersUntilTimeout(attachedTimers: Timer[], timeout: Timer): Promise<void> {
 | |
|         attachedTimers.forEach((t) => t.start());
 | |
|         try {
 | |
|             await timeout.finished();
 | |
|         } catch (_e) {
 | |
|             /* aborted */
 | |
|         }
 | |
|         attachedTimers.forEach((t) => t.abort());
 | |
|     }
 | |
| 
 | |
|     private isMouseEvent(event: Event): event is MouseEvent {
 | |
|         return event.type.startsWith("mouse");
 | |
|     }
 | |
| }
 |