Tone.js/Tone/core/clock/Ticker.ts

163 lines
3.3 KiB
TypeScript
Raw Normal View History

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
*/
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;
constructor(
callback: () => void,
type: TickerClockSource,
updateInterval: Seconds,
contextSampleRate?: number
) {
2019-05-22 03:37:03 +00:00
this._callback = callback;
this._type = type;
this._minimumUpdateInterval = Math.max(
128 / (contextSampleRate || 44100),
0.001
);
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 {
const blob = new Blob(
[
/* javascript */ `
2019-05-22 03:37:03 +00:00
// the initial timeout time
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
tick();
`,
],
{ 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) {
this._updateInterval = Math.max(interval, this._minimumUpdateInterval);
2019-05-22 03:37:03 +00:00
if (this._type === "worker") {
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();
}
}