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