@quantlab/handsontable
Version:
Spreadsheet-like data grid editor that provides copy/paste functionality compatible with Excel/Google Docs
113 lines (100 loc) • 2.21 kB
JavaScript
import {requestAnimationFrame, cancelAnimationFrame} from './../helpers/feature';
/**
* @class Interval
* @util
*/
class Interval {
static create(func, delay) {
return new Interval(func, delay);
}
constructor(func, delay) {
/**
* Animation frame request id.
*
* @type {Number}
*/
this.timer = null;
/**
* Function to invoke repeatedly.
*
* @type {Function}
*/
this.func = func;
/**
* Number of milliseconds that function should wait before next call.
*/
this.delay = parseDelay(delay);
/**
* Flag which indicates if interval object was stopped.
*
* @type {Boolean}
* @default true
*/
this.stopped = true;
/**
* Interval time (in milliseconds) of the last callback call.
*
* @private
* @type {Number}
*/
this._then = null;
/**
* Bounded function `func`.
*
* @private
* @type {Function}
*/
this._callback = () => this.__callback();
}
/**
* Start loop.
*
* @returns {Interval}
*/
start() {
if (this.stopped) {
this._then = Date.now();
this.stopped = false;
this.timer = requestAnimationFrame(this._callback);
}
return this;
}
/**
* Stop looping.
*
* @returns {Interval}
*/
stop() {
if (!this.stopped) {
this.stopped = true;
cancelAnimationFrame(this.timer);
this.timer = null;
}
return this;
}
/**
* Loop callback, fired on every animation frame.
*
* @private
*/
__callback() {
this.timer = requestAnimationFrame(this._callback);
if (this.delay) {
const now = Date.now();
const elapsed = now - this._then;
if (elapsed > this.delay) {
this._then = now - (elapsed % this.delay);
this.func();
}
} else {
this.func();
}
}
}
export default Interval;
export function parseDelay(delay) {
if (typeof delay === 'string' && /fps$/.test(delay)) {
delay = 1000 / parseInt(delay.replace('fps', '') || 0, 10);
}
return delay;
}