2019-10-26 15:43:00 +00:00
|
|
|
import { optionsFromArguments } from "../../core/util/Defaults";
|
2019-10-28 16:20:04 +00:00
|
|
|
import { Frequency, Hertz, NormalRange, Positive, Time } from "../../core/type/Units";
|
2019-10-29 21:49:22 +00:00
|
|
|
import { Envelope, EnvelopeOptions } from "./Envelope";
|
|
|
|
import { Scale } from "../../signal/Scale";
|
|
|
|
import { Pow } from "../../signal/Pow";
|
2019-12-17 17:42:40 +00:00
|
|
|
import { assertRange } from "../../core/util/Debug";
|
2019-10-26 15:43:00 +00:00
|
|
|
|
2019-10-29 21:49:22 +00:00
|
|
|
export interface FrequencyEnvelopeOptions extends EnvelopeOptions {
|
2019-10-26 15:43:00 +00:00
|
|
|
baseFrequency: Frequency;
|
|
|
|
octaves: number;
|
|
|
|
exponent: number;
|
|
|
|
}
|
|
|
|
/**
|
2019-10-30 17:22:53 +00:00
|
|
|
* FrequencyEnvelope is an [[Envelope]] which ramps between [[baseFrequency]]
|
|
|
|
* and [[octaves]]. It can also have an optional [[exponent]] to adjust the curve
|
|
|
|
* which it ramps.
|
2019-10-26 15:43:00 +00:00
|
|
|
* @example
|
|
|
|
* import { FrequencyEnvelope, Oscillator } from "tone";
|
|
|
|
* const oscillator = new Oscillator().toDestination().start();
|
|
|
|
* const freqEnv = new FrequencyEnvelope({
|
|
|
|
* attack: 0.2,
|
|
|
|
* baseFrequency: "C2",
|
|
|
|
* octaves: 4
|
|
|
|
* });
|
|
|
|
* freqEnv.connect(oscillator.frequency);
|
2019-10-28 16:20:04 +00:00
|
|
|
* freqEnv.triggerAttack();
|
2019-10-26 15:43:00 +00:00
|
|
|
*/
|
2019-10-29 21:49:22 +00:00
|
|
|
export class FrequencyEnvelope extends Envelope {
|
2019-10-26 15:43:00 +00:00
|
|
|
|
|
|
|
readonly name: string = "FrequencyEnvelope";
|
|
|
|
|
2019-10-28 16:20:04 +00:00
|
|
|
/**
|
|
|
|
* Private reference to the base frequency as a number
|
|
|
|
*/
|
|
|
|
private _baseFrequency: Hertz;
|
2019-10-26 15:43:00 +00:00
|
|
|
|
2019-10-28 16:20:04 +00:00
|
|
|
/**
|
|
|
|
* The number of octaves
|
|
|
|
*/
|
|
|
|
private _octaves: Positive;
|
2019-10-26 15:43:00 +00:00
|
|
|
|
2019-10-29 21:49:22 +00:00
|
|
|
/**
|
|
|
|
* Internal scaler from 0-1 to the final output range
|
|
|
|
*/
|
|
|
|
private _scale: Scale;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Apply a power curve to the output
|
|
|
|
*/
|
|
|
|
private _exponent: Pow;
|
|
|
|
|
2019-10-26 15:43:00 +00:00
|
|
|
/**
|
|
|
|
* @param attack the attack time in seconds
|
|
|
|
* @param decay the decay time in seconds
|
|
|
|
* @param sustain a percentage (0-1) of the full amplitude
|
|
|
|
* @param release the release time in seconds
|
|
|
|
*/
|
|
|
|
constructor(attack?: Time, decay?: Time, sustain?: NormalRange, release?: Time);
|
|
|
|
constructor(options?: Partial<FrequencyEnvelopeOptions>)
|
|
|
|
constructor() {
|
2019-10-28 16:20:04 +00:00
|
|
|
super(optionsFromArguments(FrequencyEnvelope.getDefaults(), arguments, ["attack", "decay", "sustain", "release"]));
|
|
|
|
const options = optionsFromArguments(FrequencyEnvelope.getDefaults(), arguments, ["attack", "decay", "sustain", "release"]);
|
2019-10-26 15:43:00 +00:00
|
|
|
|
|
|
|
this._octaves = options.octaves;
|
2019-10-28 16:20:04 +00:00
|
|
|
this._baseFrequency = this.toFrequency(options.baseFrequency);
|
2019-10-29 21:49:22 +00:00
|
|
|
|
|
|
|
this._exponent = this.input = new Pow({
|
|
|
|
context: this.context,
|
|
|
|
value: options.exponent
|
|
|
|
});
|
|
|
|
this._scale = this.output = new Scale({
|
|
|
|
context: this.context,
|
|
|
|
min: this._baseFrequency,
|
|
|
|
max: this._baseFrequency * Math.pow(2, this._octaves),
|
|
|
|
});
|
|
|
|
this._sig.chain(this._exponent, this._scale);
|
2019-10-26 15:43:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static getDefaults(): FrequencyEnvelopeOptions {
|
2019-10-29 21:49:22 +00:00
|
|
|
return Object.assign(Envelope.getDefaults(), {
|
2019-10-26 15:43:00 +00:00
|
|
|
baseFrequency: 200,
|
|
|
|
exponent: 1,
|
|
|
|
octaves: 4,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2019-10-30 17:22:53 +00:00
|
|
|
* The envelope's minimum output value. This is the value which it
|
2019-10-26 15:43:00 +00:00
|
|
|
* starts at.
|
|
|
|
*/
|
|
|
|
get baseFrequency(): Frequency {
|
2019-10-27 21:45:21 +00:00
|
|
|
return this._baseFrequency;
|
2019-10-26 15:43:00 +00:00
|
|
|
}
|
|
|
|
set baseFrequency(min) {
|
2019-12-17 17:42:40 +00:00
|
|
|
const freq = this.toFrequency(min);
|
|
|
|
assertRange(freq, 0);
|
|
|
|
this._baseFrequency = freq;
|
2019-10-29 21:49:22 +00:00
|
|
|
this._scale.min = this._baseFrequency;
|
|
|
|
// update the max value when the min changes
|
|
|
|
this.octaves = this._octaves;
|
2019-10-26 15:43:00 +00:00
|
|
|
}
|
2019-12-17 17:42:40 +00:00
|
|
|
|
2019-10-26 15:43:00 +00:00
|
|
|
/**
|
|
|
|
* The number of octaves above the baseFrequency that the
|
|
|
|
* envelope will scale to.
|
|
|
|
*/
|
|
|
|
get octaves(): Positive {
|
|
|
|
return this._octaves;
|
|
|
|
}
|
|
|
|
set octaves(octaves: Positive) {
|
2019-12-17 17:42:40 +00:00
|
|
|
assertRange(octaves, 0);
|
2019-10-26 15:43:00 +00:00
|
|
|
this._octaves = octaves;
|
2019-10-29 21:49:22 +00:00
|
|
|
this._scale.max = this._baseFrequency * Math.pow(2, octaves);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The envelope's exponent value.
|
|
|
|
*/
|
|
|
|
get exponent(): number {
|
|
|
|
return this._exponent.value;
|
|
|
|
}
|
|
|
|
set exponent(exponent) {
|
|
|
|
this._exponent.value = exponent;
|
2019-10-26 15:43:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Clean up
|
|
|
|
*/
|
|
|
|
dispose(): this {
|
|
|
|
super.dispose();
|
2019-10-29 21:49:22 +00:00
|
|
|
this._exponent.dispose();
|
|
|
|
this._scale.dispose();
|
2019-10-26 15:43:00 +00:00
|
|
|
return this;
|
|
|
|
}
|
|
|
|
}
|