Tone.js/Tone/event/Sequence.ts

312 lines
7.5 KiB
TypeScript
Raw Normal View History

2019-07-25 01:53:35 +00:00
import { TicksClass } from "../core/type/Ticks";
import { omitFromObject, optionsFromArguments } from "../core/util/Defaults";
import { isArray, isString } from "../core/util/TypeCheck";
import { Part } from "./Part";
import { ToneEvent, ToneEventCallback, ToneEventOptions } from "./ToneEvent";
type SequenceEventDescription = Array<any | any[]>;
interface SequenceOptions extends Omit<ToneEventOptions, "value"> {
loopStart: number;
loopEnd: number;
subdivision: Time;
events: SequenceEventDescription;
}
/**
* A sequence is an alternate notation of a part. Instead
* of passing in an array of [time, event] pairs, pass
* in an array of events which will be spaced at the
* given subdivision. Sub-arrays will subdivide that beat
* by the number of items are in the array.
* Sequence notation inspiration from [Tidal](http://yaxu.org/tidal/)
* @param callback The callback to invoke with every note
* @param sequence The sequence
* @param subdivision The subdivision between which events are placed.
* @example
* var seq = new Sequence(function(time, note){
* console.log(note);
* //straight quater notes
* }, ["C4", "E4", "G4", "A4"], "4n");
* @example
* var seq = new Sequence(function(time, note){
* console.log(note);
* //subdivisions are given as subarrays
* }, ["C4", ["E4", "D4", "E4"], "G4", ["A4", "G4"]]);
*/
export class Sequence extends ToneEvent {
name = "Sequence";
/**
* The subdivison of each note
*/
private _subdivision: Ticks;
/**
* The object responsible for scheduling all of the events
*/
private _part: Part = new Part({
callback: this._seqCallback.bind(this),
context: this.context,
});
/**
* private reference to all of the sequence proxies
*/
private _events: any[] = [];
/**
* The proxied array
*/
private _eventsArray: any[] = [];
constructor(
callback?: ToneEventCallback,
events?: SequenceEventDescription,
subdivision?: Time,
);
constructor(options?: Partial<SequenceOptions>);
constructor() {
super(optionsFromArguments(Sequence.getDefaults(), arguments, ["callback", "events", "subdivision"]));
const options = optionsFromArguments(Sequence.getDefaults(), arguments, ["callback", "events", "subdivision"]);
this._subdivision = this.toTicks(options.subdivision);
this.events = options.events;
// set all of the values
this.loop = options.loop;
this.loopStart = options.loopStart;
this.loopEnd = options.loopEnd;
this.playbackRate = options.playbackRate;
this.probability = options.probability;
this.humanize = options.humanize;
this.mute = options.mute;
this.playbackRate = options.playbackRate;
}
static getDefaults(): SequenceOptions {
return Object.assign( omitFromObject(ToneEvent.getDefaults(), ["value"]), {
events: [],
loop: true,
loopEnd : 0,
loopStart: 0,
subdivision: "8n",
});
}
/**
* The internal callback for when an event is invoked
*/
private _seqCallback(time: Seconds, value: any): void {
if (value !== null) {
this.callback(time, value);
}
}
/**
* The sequence
*/
get events(): any[] {
return this._events;
}
set events(s) {
this.clear();
this._eventsArray = s;
this._events = this._createSequence(this._eventsArray);
this._eventsUpdated();
}
/**
* Start the part at the given time.
* @param time When to start the part.
* @param offset The offset index to start at
*/
start(time?: TransportTime, offset?: number): this {
this._part.start(time, offset ? this._indexTime(offset) : offset);
return this;
}
/**
* Stop the part at the given time.
* @param time When to stop the part.
*/
stop(time?: TransportTime): this {
this._part.stop(time);
return this;
}
/**
* The subdivision of the sequence. This can only be
* set in the constructor. The subdivision is the
* interval between successive steps.
*/
get subdivision(): Seconds {
return new TicksClass(this.context, this._subdivision).toSeconds();
}
/**
* Create a sequence proxy which can be monitored to create subsequences
*/
private _createSequence(array: any[]): any[] {
return new Proxy(array, {
get: (target: any[], property: PropertyKey): any => {
// property is index in this case
return target[property];
},
set: (target: any[], property: PropertyKey, value: any): boolean => {
if (isString(property) && isFinite(parseInt(property, 10))) {
const index = parseInt(property, 10);
if (isArray(value)) {
target[property] = this._createSequence(value);
} else {
target[property] = value;
}
} else {
target[property] = value;
}
this._eventsUpdated();
// return true to accept the changes
return true;
},
});
}
/**
* When the sequence has changed, all of the events need to be recreated
*/
private _eventsUpdated(): void {
this._part.clear();
this._rescheduleSequence(this._eventsArray, this._subdivision, this.startOffset);
// update the loopEnd
this.loopEnd = this.loopEnd;
}
/**
* reschedule all of the events that need to be rescheduled
*/
private _rescheduleSequence(sequence: any[], subdivision: Ticks, startOffset: Ticks): void {
sequence.forEach((value, index) => {
const eventOffset = index * (subdivision) + startOffset;
if (isArray(value)) {
this._rescheduleSequence(value, subdivision / value.length, eventOffset);
} else {
const startTime = new TicksClass(this.context, eventOffset, "i").toSeconds();
this._part.add(startTime, value);
}
});
}
/**
* Get the time of the index given the Sequence's subdivision
* @param index
* @return The time of that index
* @private
*/
private _indexTime(index: number): Seconds {
return new TicksClass(this.context, index * (this._subdivision) + this.startOffset).toSeconds();
}
/**
* Clear all of the events
*/
clear(): this {
this._part.clear();
return this;
}
dispose(): this {
super.dispose();
this._part.dispose();
return this;
}
///////////////////////////////////////////////////////////////////////////
// PROXY CALLS
///////////////////////////////////////////////////////////////////////////
get loop(): boolean | number {
return this._part.loop;
}
set loop(l) {
if (this._part) {
this._part.loop = l;
}
}
/**
* The index at which the sequence should start looping
*/
get loopStart(): number {
return this._loopStart;
}
set loopStart(index) {
this._loopStart = index;
if (this._part) {
this._part.loopStart = this._indexTime(index);
}
}
/**
* The index at which the sequence should end looping
*/
get loopEnd(): number {
return this._loopEnd;
}
set loopEnd(index) {
this._loopEnd = index;
if (this._part) {
if (index === 0) {
this._part.loopEnd = this._indexTime(this._eventsArray.length);
} else {
this._part.loopEnd = this._indexTime(index);
}
}
}
get startOffset(): Ticks {
return this._part.startOffset;
}
set startOffset(start) {
if (this._part) {
this._part.startOffset = start;
}
}
get playbackRate(): Positive {
return this._part.playbackRate;
}
set playbackRate(rate) {
if (this._part) {
this._part.playbackRate = rate;
}
}
get probability(): NormalRange {
return this._part.probability;
}
set probability(prob) {
if (this._part) {
this._part.probability = prob;
}
}
get humanize(): boolean | Time {
return this._part.humanize;
}
set humanize(variation) {
if (this._part) {
this._part.humanize = variation;
}
}
/**
* The number of scheduled events
*/
get length(): number {
return this._part.length;
}
}