Tone.js/Tone/component/envelope/Envelope.ts

635 lines
17 KiB
TypeScript
Raw Normal View History

2019-07-25 17:32:34 +00:00
import { InputNode, OutputNode } from "../../core/context/ToneAudioNode";
2019-07-18 14:21:27 +00:00
import { ToneAudioNode, ToneAudioNodeOptions } from "../../core/context/ToneAudioNode";
import { NormalRange, Time } from "../../core/type/Units";
2019-07-18 14:21:27 +00:00
import { optionsFromArguments } from "../../core/util/Defaults";
import { isArray, isObject, isString } from "../../core/util/TypeCheck";
import { connectSignal, Signal } from "../../signal/Signal";
import { OfflineContext } from "../../core/context/OfflineContext";
2019-10-09 16:46:00 +00:00
import { assertRange } from "../../core/util/Debug";
2019-07-18 14:21:27 +00:00
type BasicEnvelopeCurve = "linear" | "exponential";
type InternalEnvelopeCurve = BasicEnvelopeCurve | number[];
export type EnvelopeCurve = EnvelopeCurveName | number[];
export interface EnvelopeOptions extends ToneAudioNodeOptions {
attack: Time;
decay: Time;
sustain: NormalRange;
release: Time;
attackCurve: EnvelopeCurve;
releaseCurve: EnvelopeCurve;
decayCurve: BasicEnvelopeCurve;
}
/**
2019-09-14 20:39:18 +00:00
* Envelope is an [ADSR](https://en.wikipedia.org/wiki/Synthesizer#ADSR_envelope)
* envelope generator. Envelope outputs a signal which
* can be connected to an AudioParam or Tone.Signal.
2019-07-25 19:14:54 +00:00
* ```
* /\
* / \
* / \
* / \
* / \___________
* / \
* / \
* / \
* / \
* ```
2019-07-18 14:21:27 +00:00
*
* @example
2019-07-18 14:21:27 +00:00
* //an amplitude envelope
* var gainNode = Tone.context.createGain();
* var env = new Envelope({
* "attack" : 0.1,
* "decay" : 0.2,
* "sustain" : 1,
* "release" : 0.8,
* });
* env.connect(gainNode.gain);
2019-09-16 14:15:23 +00:00
* @category Component
2019-07-18 14:21:27 +00:00
*/
export class Envelope extends ToneAudioNode<EnvelopeOptions> {
2019-08-03 01:09:35 +00:00
readonly name: string = "Envelope";
2019-07-18 14:21:27 +00:00
/**
* Private container for the attack value
2019-07-18 14:21:27 +00:00
*/
private _attack!: Time;
2019-07-18 14:21:27 +00:00
/**
* Private holder of the decay time
2019-07-18 14:21:27 +00:00
*/
private _decay!: Time;
2019-07-18 14:21:27 +00:00
/**
* private holder for the sustain value
2019-07-18 14:21:27 +00:00
*/
private _sustain!: NormalRange;
2019-07-18 14:21:27 +00:00
/**
* private holder for the release value
2019-07-18 14:21:27 +00:00
*/
private _release!: Time;
2019-07-18 14:21:27 +00:00
/**
2019-09-14 20:39:18 +00:00
* The automation curve type for the attack
2019-07-18 14:21:27 +00:00
*/
private _attackCurve!: InternalEnvelopeCurve;
/**
2019-09-14 20:39:18 +00:00
* The automation curve type for the decay
2019-07-18 14:21:27 +00:00
*/
private _decayCurve!: BasicEnvelopeCurve;
/**
2019-09-14 20:39:18 +00:00
* The automation curve type for the release
2019-07-18 14:21:27 +00:00
*/
private _releaseCurve!: InternalEnvelopeCurve;
/**
2019-09-14 20:39:18 +00:00
* the signal which is output.
2019-07-18 14:21:27 +00:00
*/
2019-07-18 15:24:32 +00:00
protected _sig: Signal<NormalRange> = new Signal({
2019-07-18 14:21:27 +00:00
context: this.context,
value: 0,
});
/**
* The output signal of the envelope
*/
2019-07-18 15:24:32 +00:00
output: OutputNode = this._sig;
2019-07-18 14:21:27 +00:00
/**
* Envelope has no input
*/
2019-07-18 15:24:32 +00:00
input: InputNode | undefined = undefined;
2019-07-18 14:21:27 +00:00
2019-08-27 17:02:31 +00:00
/**
* @param attack The amount of time it takes for the envelope to go from
2019-09-14 20:39:18 +00:00
* 0 to it's maximum value.
* @param decay The period of time after the attack that it takes for the envelope
2019-09-14 20:39:18 +00:00
* to fall to the sustain value. Value must be greater than 0.
* @param sustain The percent of the maximum value that the envelope rests at until
2019-09-14 20:39:18 +00:00
* the release is triggered.
* @param release The amount of time after the release is triggered it takes to reach 0.
2019-09-14 20:39:18 +00:00
* Value must be greater than 0.
2019-08-27 17:02:31 +00:00
*/
2019-07-18 14:21:27 +00:00
constructor(attack?: Time, decay?: Time, sustain?: NormalRange, release?: Time);
constructor(options?: Partial<EnvelopeOptions>)
constructor() {
super(optionsFromArguments(Envelope.getDefaults(), arguments, ["attack", "decay", "sustain", "release"]));
const options = optionsFromArguments(Envelope.getDefaults(), arguments, ["attack", "decay", "sustain", "release"]);
this.attack = options.attack;
this.decay = options.decay;
this.sustain = options.sustain;
this.release = options.release;
this.attackCurve = options.attackCurve;
this.releaseCurve = options.releaseCurve;
this.decayCurve = options.decayCurve;
}
static getDefaults(): EnvelopeOptions {
return Object.assign(ToneAudioNode.getDefaults(), {
2019-09-16 03:32:40 +00:00
attack: 0.01,
attackCurve: "linear" as EnvelopeCurveName,
decay: 0.1,
decayCurve: "exponential" as BasicEnvelopeCurve,
release: 1,
releaseCurve: "exponential" as EnvelopeCurveName,
sustain: 0.5,
2019-07-18 14:21:27 +00:00
});
}
/**
* Read the current value of the envelope. Useful for
* synchronizing visual output to the envelope.
2019-07-18 14:21:27 +00:00
*/
get value(): NormalRange {
return this.getValueAtTime(this.now());
}
/**
* When triggerAttack is called, the attack time is the amount of
* time it takes for the envelope to reach it's maximum value.
* ```
* /\
* /X \
* /XX \
* /XXX \
* /XXXX \___________
* /XXXXX \
* /XXXXXX \
* /XXXXXXX \
* /XXXXXXXX \
* ```
* @min 0
* @max 2
*/
get attack(): Time {
return this._attack;
}
set attack(time) {
assertRange(this.toSeconds(time), 0);
this._attack = time;
}
/**
* After the attack portion of the envelope, the value will fall
* over the duration of the decay time to it's sustain value.
* ```
* /\
* / X\
* / XX\
* / XXX\
* / XXXX\___________
* / XXXXX \
* / XXXXX \
* / XXXXX \
* / XXXXX \
* ```
* @min 0
* @max 2
*/
get decay(): Time {
return this._decay;
}
set decay(time) {
assertRange(this.toSeconds(time), 0);
this._decay = time;
}
/**
* The sustain value is the value
* which the envelope rests at after triggerAttack is
* called, but before triggerRelease is invoked.
* ```
* /\
* / \
* / \
* / \
* / \___________
* / XXXXXXXXXXX\
* / XXXXXXXXXXX \
* / XXXXXXXXXXX \
* / XXXXXXXXXXX \
* ```
*/
get sustain(): NormalRange {
return this._sustain;
}
set sustain(val) {
assertRange(this.toSeconds(val), 0, 1);
this._sustain = val;
}
/**
* After triggerRelease is called, the envelope's
* value will fall to it's miminum value over the
* duration of the release time.
* ```
* /\
* / \
* / \
* / \
* / \___________
* / X\
* / XX\
* / XXX\
* / XXXX\
* ```
* @min 0
* @max 5
*/
get release(): Time {
return this._release;
}
set release(time) {
assertRange(this.toSeconds(time), 0);
this._release = time;
}
2019-07-18 14:21:27 +00:00
/**
2019-09-14 20:39:18 +00:00
* Get the curve
* @param curve
* @param direction In/Out
* @return The curve name
2019-07-18 14:21:27 +00:00
*/
private _getCurve(curve: InternalEnvelopeCurve, direction: EnvelopeDirection): EnvelopeCurve {
if (isString(curve)) {
return curve;
} else {
// look up the name in the curves array
let curveName: EnvelopeCurveName;
for (curveName in EnvelopeCurves) {
if (EnvelopeCurves[curveName][direction] === curve) {
return curveName;
}
}
// return the custom curve
return curve;
}
}
/**
2019-09-14 20:39:18 +00:00
* Assign a the curve to the given name using the direction
* @param name
* @param direction In/Out
* @param curve
2019-07-18 14:21:27 +00:00
*/
private _setCurve(
name: "_attackCurve" | "_decayCurve" | "_releaseCurve",
direction: EnvelopeDirection,
curve: EnvelopeCurve,
): void {
// check if it's a valid type
if (isString(curve) && Reflect.has(EnvelopeCurves, curve)) {
const curveDef = EnvelopeCurves[curve];
if (isObject(curveDef)) {
if (name !== "_decayCurve") {
this[name] = curveDef[direction];
}
} else {
this[name] = curveDef;
}
} else if (isArray(curve) && name !== "_decayCurve") {
this[name] = curve;
} else {
throw new Error("Envelope: invalid curve: " + curve);
}
}
/**
* The shape of the attack.
* Can be any of these strings:
2019-07-26 15:56:33 +00:00
* * "linear"
* * "exponential"
* * "sine"
* * "cosine"
* * "bounce"
* * "ripple"
* * "step"
*
2019-07-18 14:21:27 +00:00
* Can also be an array which describes the curve. Values
* in the array are evenly subdivided and linearly
* interpolated over the duration of the attack.
* @example
* env.attackCurve = "linear";
* @example
* //can also be an array
* env.attackCurve = [0, 0.2, 0.3, 0.4, 1]
*/
get attackCurve(): EnvelopeCurve {
return this._getCurve(this._attackCurve, "In");
}
set attackCurve(curve) {
this._setCurve("_attackCurve", "In", curve);
}
/**
* The shape of the release. See the attack curve types.
* @example
* env.releaseCurve = "linear";
*/
get releaseCurve(): EnvelopeCurve {
return this._getCurve(this._releaseCurve, "Out");
}
set releaseCurve(curve) {
this._setCurve("_releaseCurve", "Out", curve);
}
/**
* The shape of the decay either "linear" or "exponential"
* @example
* env.decayCurve = "linear";
*/
get decayCurve(): BasicEnvelopeCurve {
return this._decayCurve;
}
set decayCurve(curve) {
this.assert(["linear", "exponential"].some(c => c === curve), `Invalid envelope curve: ${curve}`);
this._decayCurve = curve;
}
/**
* Trigger the attack/decay portion of the ADSR envelope.
* @param time When the attack should start.
* @param velocity The velocity of the envelope scales the vales.
2019-09-14 20:39:18 +00:00
* number between 0-1
2019-07-18 14:21:27 +00:00
* @example
* //trigger the attack 0.5 seconds from now with a velocity of 0.2
* env.triggerAttack("+0.5", 0.2);
*/
triggerAttack(time?: Time, velocity: NormalRange = 1): this {
this.log("triggerAttack", time, velocity);
time = this.toSeconds(time);
const originalAttack = this.toSeconds(this.attack);
let attack = originalAttack;
const decay = this.toSeconds(this.decay);
// check if it's not a complete attack
const currentValue = this.getValueAtTime(time);
if (currentValue > 0) {
// subtract the current value from the attack time
const attackRate = 1 / attack;
const remainingDistance = 1 - currentValue;
// the attack is now the remaining time
attack = remainingDistance / attackRate;
}
// attack
if (attack === 0) {
// case where the attack time is 0 should set instantly
this._sig.setValueAtTime(velocity, time);
} else if (this._attackCurve === "linear") {
this._sig.linearRampTo(velocity, attack, time);
} else if (this._attackCurve === "exponential") {
this._sig.targetRampTo(velocity, attack, time);
2019-08-12 14:15:55 +00:00
} else {
2019-07-18 14:21:27 +00:00
this._sig.cancelAndHoldAtTime(time);
let curve = this._attackCurve;
// find the starting position in the curve
for (let i = 1; i < curve.length; i++) {
// the starting index is between the two values
if (curve[i - 1] <= currentValue && currentValue <= curve[i]) {
curve = this._attackCurve.slice(i);
// the first index is the current value
curve[0] = currentValue;
break;
}
}
this._sig.setValueCurveAtTime(curve, time, attack, velocity);
}
// decay
if (decay) {
const decayValue = velocity * this.sustain;
const decayStart = time + attack;
this.log("decay", decayStart);
if (this._decayCurve === "linear") {
this._sig.linearRampTo(decayValue, decay, decayStart + this.sampleTime);
2019-08-12 14:15:55 +00:00
} else {
this.assert(this._decayCurve === "exponential",
`decayCurve can only be "linear" or "exponential", got ${this._decayCurve}`);
2019-07-18 14:21:27 +00:00
this._sig.exponentialApproachValueAtTime(decayValue, decayStart, decay);
}
}
return this;
}
/**
2019-09-14 20:39:18 +00:00
* Triggers the release of the envelope.
* @param time When the release portion of the envelope should start.
* @example
2019-09-14 20:39:18 +00:00
* //trigger release immediately
* env.triggerRelease();
2019-07-18 14:21:27 +00:00
*/
triggerRelease(time?: Time): this {
this.log("triggerRelease", time);
time = this.toSeconds(time);
const currentValue = this.getValueAtTime(time);
if (currentValue > 0) {
const release = this.toSeconds(this.release);
if (release === 0) {
this._sig.setValueAtTime(0, time);
} else if (this._releaseCurve === "linear") {
2019-07-18 14:21:27 +00:00
this._sig.linearRampTo(0, release, time);
} else if (this._releaseCurve === "exponential") {
this._sig.targetRampTo(0, release, time);
} else {
2019-08-12 14:15:55 +00:00
this.assert(isArray(this._releaseCurve), "releaseCurve must be either 'linear', 'exponential' or an array");
this._sig.cancelAndHoldAtTime(time);
this._sig.setValueCurveAtTime(this._releaseCurve, time, release, currentValue);
2019-07-18 14:21:27 +00:00
}
}
return this;
}
/**
2019-09-14 20:39:18 +00:00
* Get the scheduled value at the given time. This will
* return the unconverted (raw) value.
2019-07-18 14:21:27 +00:00
*/
getValueAtTime(time: Time): NormalRange {
return this._sig.getValueAtTime(time);
}
/**
2019-09-14 20:39:18 +00:00
* triggerAttackRelease is shorthand for triggerAttack, then waiting
* some duration, then triggerRelease.
* @param duration The duration of the sustain.
* @param time When the attack should be triggered.
* @param velocity The velocity of the envelope.
* @example
2019-07-18 14:21:27 +00:00
* //trigger the attack and then the release after 0.6 seconds.
* env.triggerAttackRelease(0.6);
*/
triggerAttackRelease(duration: Time, time?: Time, velocity: NormalRange = 1): this {
time = this.toSeconds(time);
this.triggerAttack(time, velocity);
this.triggerRelease(time + this.toSeconds(duration));
return this;
}
/**
2019-09-14 20:39:18 +00:00
* Cancels all scheduled envelope changes after the given time.
2019-07-18 14:21:27 +00:00
*/
cancel(after?: Time): this {
2019-07-30 14:25:17 +00:00
this._sig.cancelScheduledValues(this.toSeconds(after));
2019-07-18 14:21:27 +00:00
return this;
}
/**
* Connect the envelope to a destination node.
*/
connect(destination: InputNode, outputNumber: number = 0, inputNumber: number = 0): this {
2019-07-18 15:24:32 +00:00
connectSignal(this, destination, outputNumber, inputNumber);
2019-07-18 14:21:27 +00:00
return this;
}
/**
* Render the envelope curve to an array of the given length.
* Good for visualizing the envelope curve
*/
async asArray(length: number = 1024): Promise<Float32Array> {
const duration = length / this.context.sampleRate;
const context = new OfflineContext(1, duration, this.context.sampleRate);
// normalize the ADSR for the given duration with 20% sustain time
const attackPortion = this.toSeconds(this.attack) + this.toSeconds(this.decay);
const envelopeDuration = attackPortion + this.toSeconds(this.release);
const sustainTime = envelopeDuration * 0.1;
const totalDuration = envelopeDuration + sustainTime;
// @ts-ignore
const clone = new this.constructor(Object.assign(this.get(), {
attack: duration * this.toSeconds(this.attack) / totalDuration,
decay: duration * this.toSeconds(this.decay) / totalDuration,
release: duration * this.toSeconds(this.release) / totalDuration,
context
})) as Envelope;
clone._sig.toDestination();
clone.triggerAttackRelease(duration * (attackPortion + sustainTime) / totalDuration, 0);
const buffer = await context.render();
return buffer.getChannelData(0);
}
2019-07-18 14:21:27 +00:00
dispose(): this {
super.dispose();
2019-07-18 14:21:27 +00:00
this._sig.dispose();
return this;
}
}
interface EnvelopeCurveObject {
In: number[];
Out: number[];
}
type EnvelopeDirection = keyof EnvelopeCurveObject;
interface EnvelopeCurveMap {
linear: "linear";
exponential: "exponential";
bounce: EnvelopeCurveObject;
cosine: EnvelopeCurveObject;
sine: EnvelopeCurveObject;
ripple: EnvelopeCurveObject;
step: EnvelopeCurveObject;
}
2019-09-16 03:32:40 +00:00
type EnvelopeCurveName = keyof EnvelopeCurveMap;
2019-07-18 14:21:27 +00:00
/**
2019-09-14 20:39:18 +00:00
* Generate some complex envelope curves.
2019-07-18 14:21:27 +00:00
*/
const EnvelopeCurves: EnvelopeCurveMap = (() => {
const curveLen = 128;
let i: number;
let k: number;
// cosine curve
const cosineCurve: number[] = [];
for (i = 0; i < curveLen; i++) {
cosineCurve[i] = Math.sin((i / (curveLen - 1)) * (Math.PI / 2));
}
// ripple curve
const rippleCurve: number[] = [];
const rippleCurveFreq = 6.4;
for (i = 0; i < curveLen - 1; i++) {
k = (i / (curveLen - 1));
const sineWave = Math.sin(k * (Math.PI * 2) * rippleCurveFreq - Math.PI / 2) + 1;
rippleCurve[i] = sineWave / 10 + k * 0.83;
}
rippleCurve[curveLen - 1] = 1;
// stairs curve
const stairsCurve: number[] = [];
const steps = 5;
for (i = 0; i < curveLen; i++) {
stairsCurve[i] = Math.ceil((i / (curveLen - 1)) * steps) / steps;
}
// in-out easing curve
const sineCurve: number[] = [];
for (i = 0; i < curveLen; i++) {
k = i / (curveLen - 1);
sineCurve[i] = 0.5 * (1 - Math.cos(Math.PI * k));
}
// a bounce curve
const bounceCurve: number[] = [];
for (i = 0; i < curveLen; i++) {
k = i / (curveLen - 1);
const freq = Math.pow(k, 3) * 4 + 0.2;
const val = Math.cos(freq * Math.PI * 2 * k);
bounceCurve[i] = Math.abs(val * (1 - k));
}
/**
2019-09-14 20:39:18 +00:00
* Invert a value curve to make it work for the release
2019-07-18 14:21:27 +00:00
*/
function invertCurve(curve: number[]): number[] {
const out = new Array(curve.length);
for (let j = 0; j < curve.length; j++) {
out[j] = 1 - curve[j];
}
return out;
}
/**
2019-09-14 20:39:18 +00:00
* reverse the curve
2019-07-18 14:21:27 +00:00
*/
function reverseCurve(curve: number[]): number[] {
return curve.slice(0).reverse();
}
/**
2019-09-14 20:39:18 +00:00
* attack and release curve arrays
2019-07-18 14:21:27 +00:00
*/
return {
2019-09-16 03:32:40 +00:00
bounce: {
In: invertCurve(bounceCurve),
Out: bounceCurve,
2019-07-18 14:21:27 +00:00
},
2019-09-16 03:32:40 +00:00
cosine: {
In: cosineCurve,
Out: reverseCurve(cosineCurve),
2019-07-18 14:21:27 +00:00
},
2019-09-16 03:32:40 +00:00
exponential: "exponential" as "exponential",
linear: "linear" as "linear",
ripple: {
In: rippleCurve,
Out: invertCurve(rippleCurve),
2019-07-18 14:21:27 +00:00
},
2019-09-16 03:32:40 +00:00
sine: {
In: sineCurve,
Out: invertCurve(sineCurve),
2019-07-18 14:21:27 +00:00
},
2019-09-16 03:32:40 +00:00
step: {
In: stairsCurve,
Out: invertCurve(stairsCurve),
2019-07-18 14:21:27 +00:00
},
};
})();