2024-05-03 18:31:14 +00:00
|
|
|
import { optionsFromArguments } from "../../core/util/Defaults.js";
|
|
|
|
import { Frequency, Hertz, NormalRange, Time } from "../../core/type/Units.js";
|
|
|
|
import { Envelope, EnvelopeOptions } from "./Envelope.js";
|
|
|
|
import { Scale } from "../../signal/Scale.js";
|
|
|
|
import { Pow } from "../../signal/Pow.js";
|
|
|
|
import { assertRange } from "../../core/util/Debug.js";
|
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;
|
|
|
|
}
|
|
|
|
/**
|
2024-04-29 14:48:37 +00:00
|
|
|
* FrequencyEnvelope is an {@link Envelope} which ramps between {@link baseFrequency}
|
|
|
|
* and {@link octaves}. It can also have an optional {@link exponent} to adjust the curve
|
2024-05-03 18:31:14 +00:00
|
|
|
* which it ramps.
|
2019-10-26 15:43:00 +00:00
|
|
|
* @example
|
2020-04-17 02:24:18 +00:00
|
|
|
* const oscillator = new Tone.Oscillator().toDestination().start();
|
|
|
|
* const freqEnv = new Tone.FrequencyEnvelope({
|
2019-10-26 15:43:00 +00:00
|
|
|
* attack: 0.2,
|
|
|
|
* baseFrequency: "C2",
|
|
|
|
* octaves: 4
|
|
|
|
* });
|
|
|
|
* freqEnv.connect(oscillator.frequency);
|
2019-10-28 16:20:04 +00:00
|
|
|
* freqEnv.triggerAttack();
|
2020-09-02 20:53:38 +00:00
|
|
|
* @category Component
|
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
|
|
|
|
*/
|
2020-11-02 18:47:24 +00:00
|
|
|
private _octaves: number;
|
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
|
|
|
|
*/
|
2024-05-03 18:31:14 +00:00
|
|
|
constructor(
|
|
|
|
attack?: Time,
|
|
|
|
decay?: Time,
|
|
|
|
sustain?: NormalRange,
|
|
|
|
release?: Time
|
|
|
|
);
|
|
|
|
constructor(options?: Partial<FrequencyEnvelopeOptions>);
|
2019-10-26 15:43:00 +00:00
|
|
|
constructor() {
|
2024-05-03 18:31:14 +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,
|
2024-05-03 18:31:14 +00:00
|
|
|
value: options.exponent,
|
2019-10-29 21:49:22 +00:00
|
|
|
});
|
|
|
|
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
|
|
|
}
|
2020-04-17 02:24:18 +00:00
|
|
|
|
2019-10-26 15:43:00 +00:00
|
|
|
/**
|
|
|
|
* The number of octaves above the baseFrequency that the
|
|
|
|
* envelope will scale to.
|
|
|
|
*/
|
2020-11-02 18:47:24 +00:00
|
|
|
get octaves(): number {
|
2019-10-26 15:43:00 +00:00
|
|
|
return this._octaves;
|
|
|
|
}
|
2020-11-02 18:47:24 +00:00
|
|
|
set octaves(octaves: number) {
|
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;
|
|
|
|
}
|
|
|
|
}
|