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-26 15:43:00 +00:00
|
|
|
import { ScaledEnvelope, ScaledEnvelopeOptions } from "./ScaledEnvelope";
|
|
|
|
|
|
|
|
export interface FrequencyEnvelopeOptions extends ScaledEnvelopeOptions {
|
|
|
|
baseFrequency: Frequency;
|
|
|
|
octaves: number;
|
|
|
|
exponent: number;
|
|
|
|
}
|
|
|
|
/**
|
|
|
|
* FrequencyEnvelope is a ScaledEnvelope, but instead of `min` and `max`
|
|
|
|
* it's got a `baseFrequency` and `octaves` parameter. See [[ScaledEnvelope]]
|
|
|
|
* @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
|
|
|
*/
|
|
|
|
export class FrequencyEnvelope extends ScaledEnvelope {
|
|
|
|
|
|
|
|
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
|
|
|
|
|
|
|
/**
|
|
|
|
* @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-26 15:43:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static getDefaults(): FrequencyEnvelopeOptions {
|
|
|
|
return Object.assign(ScaledEnvelope.getDefaults(), {
|
|
|
|
baseFrequency: 200,
|
|
|
|
exponent: 1,
|
|
|
|
octaves: 4,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The envelope's mininum output value. This is the value which it
|
|
|
|
* 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-10-28 16:20:04 +00:00
|
|
|
this._baseFrequency = this.toFrequency(min);
|
|
|
|
this.min = this._baseFrequency;
|
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) {
|
|
|
|
this._octaves = octaves;
|
2019-10-28 16:20:04 +00:00
|
|
|
this.max = this._baseFrequency * Math.pow(2, octaves);
|
2019-10-26 15:43:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Clean up
|
|
|
|
*/
|
|
|
|
dispose(): this {
|
|
|
|
super.dispose();
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
}
|