129 lines
		
	
	
		
			1.9 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
		
		
			
		
	
	
			129 lines
		
	
	
		
			1.9 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
|  | class Timer { | ||
|  | 
 | ||
|  | 	constructor() { | ||
|  | 
 | ||
|  | 		this._previousTime = 0; | ||
|  | 		this._currentTime = 0; | ||
|  | 		this._startTime = now(); | ||
|  | 
 | ||
|  | 		this._delta = 0; | ||
|  | 		this._elapsed = 0; | ||
|  | 
 | ||
|  | 		this._timescale = 1; | ||
|  | 
 | ||
|  | 		// use Page Visibility API to avoid large time delta values
 | ||
|  | 
 | ||
|  | 		this._usePageVisibilityAPI = ( typeof document !== 'undefined' && document.hidden !== undefined ); | ||
|  | 
 | ||
|  | 		if ( this._usePageVisibilityAPI === true ) { | ||
|  | 
 | ||
|  | 			this._pageVisibilityHandler = handleVisibilityChange.bind( this ); | ||
|  | 
 | ||
|  | 			document.addEventListener( 'visibilitychange', this._pageVisibilityHandler, false ); | ||
|  | 
 | ||
|  | 		} | ||
|  | 
 | ||
|  | 	} | ||
|  | 
 | ||
|  | 	getDelta() { | ||
|  | 
 | ||
|  | 		return this._delta / 1000; | ||
|  | 
 | ||
|  | 	} | ||
|  | 
 | ||
|  | 	getElapsed() { | ||
|  | 
 | ||
|  | 		return this._elapsed / 1000; | ||
|  | 
 | ||
|  | 	} | ||
|  | 
 | ||
|  | 	getTimescale() { | ||
|  | 
 | ||
|  | 		return this._timescale; | ||
|  | 
 | ||
|  | 	} | ||
|  | 
 | ||
|  | 	setTimescale( timescale ) { | ||
|  | 
 | ||
|  | 		this._timescale = timescale; | ||
|  | 
 | ||
|  | 		return this; | ||
|  | 
 | ||
|  | 	} | ||
|  | 
 | ||
|  | 	reset() { | ||
|  | 
 | ||
|  | 		this._currentTime = now() - this._startTime; | ||
|  | 
 | ||
|  | 		return this; | ||
|  | 
 | ||
|  | 	} | ||
|  | 
 | ||
|  | 	dispose() { | ||
|  | 
 | ||
|  | 		if ( this._usePageVisibilityAPI === true ) { | ||
|  | 
 | ||
|  | 			document.removeEventListener( 'visibilitychange', this._pageVisibilityHandler ); | ||
|  | 
 | ||
|  | 		} | ||
|  | 
 | ||
|  | 		return this; | ||
|  | 
 | ||
|  | 	} | ||
|  | 
 | ||
|  | 	update( timestamp ) { | ||
|  | 
 | ||
|  | 
 | ||
|  | 		if ( this._usePageVisibilityAPI === true && document.hidden === true ) { | ||
|  | 
 | ||
|  | 			this._delta = 0; | ||
|  | 
 | ||
|  | 		} else { | ||
|  | 
 | ||
|  | 			this._previousTime = this._currentTime; | ||
|  | 			this._currentTime = ( timestamp !== undefined ? timestamp : now() ) - this._startTime; | ||
|  | 
 | ||
|  | 			this._delta = ( this._currentTime - this._previousTime ) * this._timescale; | ||
|  | 			this._elapsed += this._delta; // _elapsed is the accumulation of all previous deltas
 | ||
|  | 
 | ||
|  | 		} | ||
|  | 
 | ||
|  | 		return this; | ||
|  | 
 | ||
|  | 	} | ||
|  | 
 | ||
|  | } | ||
|  | 
 | ||
|  | class FixedTimer extends Timer { | ||
|  | 
 | ||
|  | 	constructor( fps = 60 ) { | ||
|  | 
 | ||
|  | 		super(); | ||
|  | 		this._delta = ( 1 / fps ) * 1000; | ||
|  | 
 | ||
|  | 	} | ||
|  | 
 | ||
|  | 	update() { | ||
|  | 
 | ||
|  | 		this._elapsed += ( this._delta * this._timescale ); // _elapsed is the accumulation of all previous deltas
 | ||
|  | 
 | ||
|  | 		return this; | ||
|  | 
 | ||
|  | 	} | ||
|  | 
 | ||
|  | } | ||
|  | 
 | ||
|  | function now() { | ||
|  | 
 | ||
|  | 	return ( typeof performance === 'undefined' ? Date : performance ).now(); | ||
|  | 
 | ||
|  | } | ||
|  | 
 | ||
|  | function handleVisibilityChange() { | ||
|  | 
 | ||
|  | 	if ( document.hidden === false ) this.reset(); | ||
|  | 
 | ||
|  | } | ||
|  | 
 | ||
|  | export { Timer, FixedTimer }; |