Tone.js/Tone/core/clock/TransportRepeatEvent.ts

161 lines
4.1 KiB
TypeScript
Raw Normal View History

2024-05-03 14:10:40 +00:00
import { BaseContext } from "../context/BaseContext.js";
import { TicksClass } from "../type/Ticks.js";
import { Seconds, Ticks, Time } from "../type/Units.js";
import { TransportEvent, TransportEventOptions } from "./TransportEvent.js";
import { GT, LT } from "../util/Math.js";
2019-05-22 03:37:03 +00:00
type Transport = import("../clock/Transport").TransportClass;
2019-07-11 13:57:06 +00:00
2019-05-22 03:37:03 +00:00
interface TransportRepeatEventOptions extends TransportEventOptions {
interval: Ticks;
duration: Ticks;
}
/**
* TransportRepeatEvent is an internal class used by Tone.Transport
* to schedule repeat events. This class should not be instantiated directly.
*/
export class TransportRepeatEvent extends TransportEvent {
/**
* When the event should stop repeating
*/
private duration: Ticks;
/**
* The interval of the repeated event
*/
private _interval: Ticks;
/**
* The ID of the current timeline event
*/
2019-11-17 18:09:19 +00:00
private _currentId = -1;
2019-05-22 03:37:03 +00:00
/**
* The ID of the next timeline event
*/
2019-11-17 18:09:19 +00:00
private _nextId = -1;
2019-05-22 03:37:03 +00:00
/**
* The time of the next event
*/
private _nextTick = this.time;
/**
* a reference to the bound start method
*/
private _boundRestart = this._restart.bind(this);
/**
* The audio context belonging to this event
*/
protected context: BaseContext;
2019-05-22 03:37:03 +00:00
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
*/
2024-05-03 15:09:28 +00:00
constructor(
transport: Transport,
opts: Partial<TransportRepeatEventOptions>
) {
2019-05-22 03:37:03 +00:00
super(transport, opts);
const options = Object.assign(TransportRepeatEvent.getDefaults(), opts);
this.duration = options.duration;
this._interval = options.interval;
2019-05-22 03:37:03 +00:00
this._nextTick = options.time;
this.transport.on("start", this._boundRestart);
this.transport.on("loopStart", this._boundRestart);
this.transport.on("ticks", this._boundRestart);
2019-05-22 03:37:03 +00:00
this.context = this.transport.context;
this._restart();
}
static getDefaults(): TransportRepeatEventOptions {
return Object.assign({}, TransportEvent.getDefaults(), {
duration: Infinity,
interval: 1,
once: false,
});
}
/**
* Invoke the callback. Returns the tick time which
* the next event should be scheduled at.
* @param time The AudioContext time in seconds of the event
*/
invoke(time: Seconds): void {
// create more events if necessary
this._createEvents(time);
// call the super class
super.invoke(time);
}
/**
* Create an event on the transport on the nextTick
*/
private _createEvent(): number {
if (LT(this._nextTick, this.floatTime + this.duration)) {
2024-05-03 15:09:28 +00:00
return this.transport.scheduleOnce(
this.invoke.bind(this),
new TicksClass(this.context, this._nextTick).toSeconds()
);
}
return -1;
}
2019-05-22 03:37:03 +00:00
/**
* Push more events onto the timeline to keep up with the position of the timeline
*/
private _createEvents(time: Seconds): void {
// schedule the next event
// const ticks = this.transport.getTicksAtTime(time);
// if the next tick is within the bounds set by "duration"
2024-05-03 15:09:28 +00:00
if (
LT(this._nextTick + this._interval, this.floatTime + this.duration)
) {
2019-05-22 03:37:03 +00:00
this._nextTick += this._interval;
this._currentId = this._nextId;
2024-05-03 15:09:28 +00:00
this._nextId = this.transport.scheduleOnce(
this.invoke.bind(this),
new TicksClass(this.context, this._nextTick).toSeconds()
);
2019-05-22 03:37:03 +00:00
}
}
/**
* Re-compute the events when the transport time has changed from a start/ticks/loopStart event
2019-05-22 03:37:03 +00:00
*/
private _restart(time?: Time): void {
this.transport.clear(this._currentId);
this.transport.clear(this._nextId);
// start at the first event
this._nextTick = this.floatTime;
2019-05-22 03:37:03 +00:00
const ticks = this.transport.getTicksAtTime(time);
if (GT(ticks, this.time)) {
// the event is not being scheduled from the beginning and should be offset
2024-05-03 15:09:28 +00:00
this._nextTick =
this.floatTime +
Math.ceil((ticks - this.floatTime) / this._interval) *
this._interval;
2019-05-22 03:37:03 +00:00
}
this._currentId = this._createEvent();
2019-05-22 03:37:03 +00:00
this._nextTick += this._interval;
this._nextId = this._createEvent();
2019-05-22 03:37:03 +00:00
}
/**
* Clean up
*/
dispose(): this {
super.dispose();
this.transport.clear(this._currentId);
this.transport.clear(this._nextId);
this.transport.off("start", this._boundRestart);
this.transport.off("loopStart", this._boundRestart);
this.transport.off("ticks", this._boundRestart);
2019-05-22 03:37:03 +00:00
return this;
}
}