2024-05-03 18:31:14 +00:00
|
|
|
import { Seconds, Ticks } from "../type/Units.js";
|
|
|
|
import { noOp } from "../util/Interface.js";
|
|
|
|
import type { TransportClass as Transport } from "./Transport.js";
|
2019-05-22 03:37:03 +00:00
|
|
|
|
|
|
|
export interface TransportEventOptions {
|
|
|
|
callback: (time: number) => void;
|
|
|
|
once: boolean;
|
|
|
|
time: Ticks;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2024-04-29 14:48:37 +00:00
|
|
|
* TransportEvent is an internal class used by {@link TransportClass}
|
2019-05-22 03:37:03 +00:00
|
|
|
* to schedule events. Do no invoke this class directly, it is
|
|
|
|
* handled from within Tone.Transport.
|
|
|
|
*/
|
|
|
|
export class TransportEvent {
|
|
|
|
/**
|
|
|
|
* Reference to the Transport that created it
|
|
|
|
*/
|
|
|
|
protected transport: Transport;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The unique id of the event
|
|
|
|
*/
|
|
|
|
id: number = TransportEvent._eventId++;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The time the event starts
|
|
|
|
*/
|
|
|
|
time: Ticks;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The callback to invoke
|
|
|
|
*/
|
|
|
|
private callback?: (time: Seconds) => void;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* If the event should be removed after being invoked.
|
|
|
|
*/
|
|
|
|
private _once: boolean;
|
|
|
|
|
2021-01-13 03:54:45 +00:00
|
|
|
/**
|
|
|
|
* The remaining value between the passed in time, and Math.floor(time).
|
2024-05-03 18:31:14 +00:00
|
|
|
* This value is later added back when scheduling to get sub-tick precision.
|
2021-01-13 03:54:45 +00:00
|
|
|
*/
|
|
|
|
protected _remainderTime = 0;
|
|
|
|
|
2019-08-21 20:59:01 +00:00
|
|
|
/**
|
2019-10-23 03:04:52 +00:00
|
|
|
* @param transport The transport object which the event belongs to
|
2019-08-21 20:59:01 +00:00
|
|
|
*/
|
2019-05-22 03:37:03 +00:00
|
|
|
constructor(transport: Transport, opts: Partial<TransportEventOptions>) {
|
2024-05-03 18:31:14 +00:00
|
|
|
const options: TransportEventOptions = Object.assign(
|
|
|
|
TransportEvent.getDefaults(),
|
|
|
|
opts
|
|
|
|
);
|
2019-05-22 03:37:03 +00:00
|
|
|
|
|
|
|
this.transport = transport;
|
|
|
|
this.callback = options.callback;
|
|
|
|
this._once = options.once;
|
2021-01-13 03:54:45 +00:00
|
|
|
this.time = Math.floor(options.time);
|
|
|
|
this._remainderTime = options.time - this.time;
|
2019-05-22 03:37:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static getDefaults(): TransportEventOptions {
|
|
|
|
return {
|
|
|
|
callback: noOp,
|
|
|
|
once: false,
|
|
|
|
time: 0,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Current ID counter
|
|
|
|
*/
|
2019-11-17 18:09:19 +00:00
|
|
|
private static _eventId = 0;
|
2019-05-22 03:37:03 +00:00
|
|
|
|
2021-01-13 03:54:45 +00:00
|
|
|
/**
|
|
|
|
* Get the time and remainder time.
|
|
|
|
*/
|
|
|
|
protected get floatTime(): number {
|
|
|
|
return this.time + this._remainderTime;
|
|
|
|
}
|
|
|
|
|
2019-05-22 03:37:03 +00:00
|
|
|
/**
|
|
|
|
* Invoke the event callback.
|
|
|
|
* @param time The AudioContext time in seconds of the event
|
|
|
|
*/
|
|
|
|
invoke(time: Seconds): void {
|
|
|
|
if (this.callback) {
|
2021-01-13 03:54:45 +00:00
|
|
|
const tickDuration = this.transport.bpm.getDurationOfTicks(1, time);
|
|
|
|
this.callback(time + this._remainderTime * tickDuration);
|
2019-05-22 03:37:03 +00:00
|
|
|
if (this._once) {
|
|
|
|
this.transport.clear(this.id);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Clean up
|
|
|
|
*/
|
|
|
|
dispose(): this {
|
|
|
|
this.callback = undefined;
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
}
|