import { BaseContext } from "../context/BaseContext"; import { TicksClass } from "../type/Ticks"; import { Seconds, Ticks, Time } from "../type/Units"; import { TransportEvent, TransportEventOptions } from "./TransportEvent"; import { GT, LT } from "../util/Math"; type Transport = import("../clock/Transport").Transport; 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 */ private _currentId = -1; /** * The ID of the next timeline event */ private _nextId = -1; /** * 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; /** * @param transport The transport object which the event belongs to */ constructor(transport: Transport, opts: Partial) { super(transport, opts); const options = Object.assign(TransportRepeatEvent.getDefaults(), opts); this.duration = options.duration; this._interval = options.interval; this._nextTick = options.time; this.transport.on("start", this._boundRestart); this.transport.on("loopStart", this._boundRestart); this.transport.on("ticks", this._boundRestart); 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)) { return this.transport.scheduleOnce(this.invoke.bind(this), new TicksClass(this.context, this._nextTick).toSeconds()); } return -1; } /** * 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" if (LT(this._nextTick + this._interval, this.floatTime + this.duration)) { this._nextTick += this._interval; this._currentId = this._nextId; this._nextId = this.transport.scheduleOnce(this.invoke.bind(this), new TicksClass(this.context, this._nextTick).toSeconds()); } } /** * Re-compute the events when the transport time has changed from a start/ticks/loopStart event */ private _restart(time?: Time): void { this.transport.clear(this._currentId); this.transport.clear(this._nextId); // start at the first event this._nextTick = this.floatTime; const ticks = this.transport.getTicksAtTime(time); if (GT(ticks, this.time)) { // the event is not being scheduled from the beginning and should be offset this._nextTick = this.floatTime + Math.ceil((ticks - this.floatTime) / this._interval) * this._interval; } this._currentId = this._createEvent(); this._nextTick += this._interval; this._nextId = this._createEvent(); } /** * 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); return this; } }