2024-05-03 14:10:40 +00:00
|
|
|
import { Seconds } from "../type/Units.js";
|
2019-05-22 03:37:03 +00:00
|
|
|
|
|
|
|
export type TickerClockSource = "worker" | "timeout" | "offline";
|
|
|
|
|
|
|
|
/**
|
|
|
|
* A class which provides a reliable callback using either
|
|
|
|
* a Web Worker, or if that isn't supported, falls back to setTimeout.
|
|
|
|
*/
|
|
|
|
export class Ticker {
|
|
|
|
/**
|
|
|
|
* Either "worker" or "timeout" or "offline"
|
|
|
|
*/
|
|
|
|
private _type: TickerClockSource;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The update interval of the worker
|
|
|
|
*/
|
2021-11-29 16:14:14 +00:00
|
|
|
private _updateInterval!: Seconds;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The lowest allowable interval, preferably calculated from context sampleRate
|
|
|
|
*/
|
|
|
|
private _minimumUpdateInterval: Seconds;
|
2019-05-22 03:37:03 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* The callback to invoke at regular intervals
|
|
|
|
*/
|
|
|
|
private _callback: () => void;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* track the callback interval
|
|
|
|
*/
|
2020-09-24 01:24:53 +00:00
|
|
|
private _timeout!: ReturnType<typeof setTimeout>;
|
2019-05-22 03:37:03 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* private reference to the worker
|
|
|
|
*/
|
|
|
|
private _worker!: Worker;
|
|
|
|
|
2024-05-03 15:09:28 +00:00
|
|
|
constructor(
|
|
|
|
callback: () => void,
|
|
|
|
type: TickerClockSource,
|
|
|
|
updateInterval: Seconds,
|
|
|
|
contextSampleRate?: number
|
|
|
|
) {
|
2019-05-22 03:37:03 +00:00
|
|
|
this._callback = callback;
|
|
|
|
this._type = type;
|
2024-05-03 15:09:28 +00:00
|
|
|
this._minimumUpdateInterval = Math.max(
|
|
|
|
128 / (contextSampleRate || 44100),
|
|
|
|
0.001
|
|
|
|
);
|
2021-11-29 16:14:14 +00:00
|
|
|
this.updateInterval = updateInterval;
|
2019-05-22 03:37:03 +00:00
|
|
|
|
|
|
|
// create the clock source for the first time
|
|
|
|
this._createClock();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2019-09-14 20:39:18 +00:00
|
|
|
* Generate a web worker
|
2019-05-22 03:37:03 +00:00
|
|
|
*/
|
|
|
|
private _createWorker(): void {
|
2024-05-03 15:09:28 +00:00
|
|
|
const blob = new Blob(
|
|
|
|
[
|
|
|
|
/* javascript */ `
|
2019-05-22 03:37:03 +00:00
|
|
|
// the initial timeout time
|
2019-09-27 21:50:49 +00:00
|
|
|
let timeoutTime = ${(this._updateInterval * 1000).toFixed(1)};
|
2019-05-22 03:37:03 +00:00
|
|
|
// onmessage callback
|
|
|
|
self.onmessage = function(msg){
|
|
|
|
timeoutTime = parseInt(msg.data);
|
|
|
|
};
|
|
|
|
// the tick function which posts a message
|
|
|
|
// and schedules a new tick
|
|
|
|
function tick(){
|
|
|
|
setTimeout(tick, timeoutTime);
|
|
|
|
self.postMessage('tick');
|
|
|
|
}
|
|
|
|
// call tick initially
|
2019-09-27 21:50:49 +00:00
|
|
|
tick();
|
2024-05-03 15:09:28 +00:00
|
|
|
`,
|
|
|
|
],
|
|
|
|
{ type: "text/javascript" }
|
|
|
|
);
|
2019-07-26 15:50:59 +00:00
|
|
|
const blobUrl = URL.createObjectURL(blob);
|
2019-05-22 03:37:03 +00:00
|
|
|
const worker = new Worker(blobUrl);
|
|
|
|
|
|
|
|
worker.onmessage = this._callback.bind(this);
|
|
|
|
|
|
|
|
this._worker = worker;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Create a timeout loop
|
|
|
|
*/
|
|
|
|
private _createTimeout(): void {
|
2019-09-16 03:32:40 +00:00
|
|
|
this._timeout = setTimeout(() => {
|
2019-05-22 03:37:03 +00:00
|
|
|
this._createTimeout();
|
|
|
|
this._callback();
|
2020-09-24 01:24:53 +00:00
|
|
|
}, this._updateInterval * 1000);
|
2019-05-22 03:37:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Create the clock source.
|
|
|
|
*/
|
|
|
|
private _createClock(): void {
|
|
|
|
if (this._type === "worker") {
|
|
|
|
try {
|
|
|
|
this._createWorker();
|
|
|
|
} catch (e) {
|
|
|
|
// workers not supported, fallback to timeout
|
|
|
|
this._type = "timeout";
|
|
|
|
this._createClock();
|
|
|
|
}
|
|
|
|
} else if (this._type === "timeout") {
|
|
|
|
this._createTimeout();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Clean up the current clock source
|
|
|
|
*/
|
|
|
|
private _disposeClock(): void {
|
|
|
|
if (this._timeout) {
|
|
|
|
clearTimeout(this._timeout);
|
|
|
|
}
|
|
|
|
if (this._worker) {
|
|
|
|
this._worker.terminate();
|
|
|
|
this._worker.onmessage = null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The rate in seconds the ticker will update
|
|
|
|
*/
|
|
|
|
get updateInterval(): Seconds {
|
|
|
|
return this._updateInterval;
|
|
|
|
}
|
|
|
|
set updateInterval(interval: Seconds) {
|
2021-11-29 16:14:14 +00:00
|
|
|
this._updateInterval = Math.max(interval, this._minimumUpdateInterval);
|
2019-05-22 03:37:03 +00:00
|
|
|
if (this._type === "worker") {
|
2021-11-29 16:14:14 +00:00
|
|
|
this._worker?.postMessage(this._updateInterval * 1000);
|
2019-05-22 03:37:03 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The type of the ticker, either a worker or a timeout
|
|
|
|
*/
|
|
|
|
get type(): TickerClockSource {
|
|
|
|
return this._type;
|
|
|
|
}
|
|
|
|
set type(type: TickerClockSource) {
|
|
|
|
this._disposeClock();
|
|
|
|
this._type = type;
|
|
|
|
this._createClock();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Clean up
|
|
|
|
*/
|
|
|
|
dispose(): void {
|
|
|
|
this._disposeClock();
|
|
|
|
}
|
|
|
|
}
|