2024-05-03 15:09:28 +00:00
|
|
|
import {
|
|
|
|
Cents,
|
|
|
|
Degrees,
|
|
|
|
Frequency,
|
|
|
|
Seconds,
|
|
|
|
Time,
|
|
|
|
} from "../../core/type/Units.js";
|
2024-05-03 14:10:40 +00:00
|
|
|
import { optionsFromArguments } from "../../core/util/Defaults.js";
|
|
|
|
import { noOp, readOnly } from "../../core/util/Interface.js";
|
|
|
|
import { Signal } from "../../signal/Signal.js";
|
|
|
|
import { Source } from "../Source.js";
|
|
|
|
import { Oscillator } from "./Oscillator.js";
|
2020-04-17 02:24:18 +00:00
|
|
|
import {
|
2024-05-03 15:09:28 +00:00
|
|
|
FatConstructorOptions,
|
|
|
|
FatOscillatorOptions,
|
|
|
|
generateWaveform,
|
|
|
|
NonCustomOscillatorType,
|
|
|
|
ToneOscillatorInterface,
|
|
|
|
ToneOscillatorType,
|
2024-05-03 14:10:40 +00:00
|
|
|
} from "./OscillatorInterface.js";
|
|
|
|
import { assertRange } from "../../core/util/Debug.js";
|
2019-07-16 20:30:09 +00:00
|
|
|
|
2024-05-03 14:10:40 +00:00
|
|
|
export { FatOscillatorOptions } from "./OscillatorInterface.js";
|
2019-09-04 22:34:42 +00:00
|
|
|
|
2019-07-16 20:30:09 +00:00
|
|
|
/**
|
2019-09-14 20:39:18 +00:00
|
|
|
* FatOscillator is an array of oscillators with detune spread between the oscillators
|
2019-08-30 16:06:38 +00:00
|
|
|
* @example
|
2020-04-17 02:24:18 +00:00
|
|
|
* const fatOsc = new Tone.FatOscillator("Ab3", "sawtooth", 40).toDestination().start();
|
2019-09-16 14:15:23 +00:00
|
|
|
* @category Source
|
2019-07-16 20:30:09 +00:00
|
|
|
*/
|
2024-05-03 15:09:28 +00:00
|
|
|
export class FatOscillator
|
|
|
|
extends Source<FatOscillatorOptions>
|
|
|
|
implements ToneOscillatorInterface
|
|
|
|
{
|
2019-09-04 23:18:44 +00:00
|
|
|
readonly name: string = "FatOscillator";
|
2019-07-16 20:30:09 +00:00
|
|
|
|
2019-10-28 15:37:53 +00:00
|
|
|
readonly frequency: Signal<"frequency">;
|
|
|
|
readonly detune: Signal<"cents">;
|
2019-07-16 20:30:09 +00:00
|
|
|
|
|
|
|
/**
|
2019-09-14 20:39:18 +00:00
|
|
|
* The array of oscillators
|
2019-07-16 20:30:09 +00:00
|
|
|
*/
|
|
|
|
private _oscillators: Oscillator[] = [];
|
|
|
|
|
|
|
|
/**
|
2019-09-14 20:39:18 +00:00
|
|
|
* The total spread of the oscillators
|
2019-07-16 20:30:09 +00:00
|
|
|
*/
|
|
|
|
private _spread: Cents;
|
|
|
|
|
|
|
|
/**
|
2019-09-14 20:39:18 +00:00
|
|
|
* The type of the oscillator
|
2019-07-16 20:30:09 +00:00
|
|
|
*/
|
|
|
|
private _type: ToneOscillatorType;
|
|
|
|
|
|
|
|
/**
|
2019-09-14 20:39:18 +00:00
|
|
|
* The phase of the oscillators
|
2019-07-16 20:30:09 +00:00
|
|
|
*/
|
|
|
|
private _phase: Degrees;
|
|
|
|
|
|
|
|
/**
|
2019-09-14 20:39:18 +00:00
|
|
|
* The partials array
|
2019-07-16 20:30:09 +00:00
|
|
|
*/
|
|
|
|
private _partials: number[];
|
|
|
|
|
|
|
|
/**
|
2019-09-14 20:39:18 +00:00
|
|
|
* The number of partials to use
|
2019-07-16 20:30:09 +00:00
|
|
|
*/
|
|
|
|
private _partialCount: number;
|
|
|
|
|
2019-10-25 20:54:33 +00:00
|
|
|
/**
|
|
|
|
* @param frequency The oscillator's frequency.
|
|
|
|
* @param type The type of the oscillator.
|
|
|
|
* @param spread The detune spread between the oscillators.
|
|
|
|
*/
|
2024-05-03 15:09:28 +00:00
|
|
|
constructor(
|
|
|
|
frequency?: Frequency,
|
|
|
|
type?: ToneOscillatorType,
|
|
|
|
spread?: Cents
|
|
|
|
);
|
2019-07-19 16:32:17 +00:00
|
|
|
constructor(options?: Partial<FatConstructorOptions>);
|
2019-07-16 20:30:09 +00:00
|
|
|
constructor() {
|
2024-05-03 15:09:28 +00:00
|
|
|
super(
|
|
|
|
optionsFromArguments(FatOscillator.getDefaults(), arguments, [
|
|
|
|
"frequency",
|
|
|
|
"type",
|
|
|
|
"spread",
|
|
|
|
])
|
|
|
|
);
|
|
|
|
const options = optionsFromArguments(
|
|
|
|
FatOscillator.getDefaults(),
|
|
|
|
arguments,
|
|
|
|
["frequency", "type", "spread"]
|
|
|
|
);
|
2019-07-16 20:30:09 +00:00
|
|
|
|
2019-08-08 18:15:56 +00:00
|
|
|
this.frequency = new Signal({
|
|
|
|
context: this.context,
|
|
|
|
units: "frequency",
|
|
|
|
value: options.frequency,
|
|
|
|
});
|
|
|
|
this.detune = new Signal({
|
|
|
|
context: this.context,
|
|
|
|
units: "cents",
|
|
|
|
value: options.detune,
|
|
|
|
});
|
2019-07-16 20:30:09 +00:00
|
|
|
|
|
|
|
this._spread = options.spread;
|
|
|
|
this._type = options.type;
|
|
|
|
this._phase = options.phase;
|
|
|
|
this._partials = options.partials;
|
|
|
|
this._partialCount = options.partialCount;
|
|
|
|
|
|
|
|
// set the count initially
|
|
|
|
this.count = options.count;
|
|
|
|
|
|
|
|
readOnly(this, ["frequency", "detune"]);
|
|
|
|
}
|
|
|
|
|
|
|
|
static getDefaults(): FatOscillatorOptions {
|
|
|
|
return Object.assign(Oscillator.getDefaults(), {
|
2019-09-16 03:32:40 +00:00
|
|
|
count: 3,
|
|
|
|
spread: 20,
|
|
|
|
type: "sawtooth",
|
2019-07-16 20:30:09 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2019-09-14 20:39:18 +00:00
|
|
|
* start the oscillator
|
2019-07-16 20:30:09 +00:00
|
|
|
*/
|
|
|
|
protected _start(time: Time): void {
|
|
|
|
time = this.toSeconds(time);
|
2024-05-03 15:09:28 +00:00
|
|
|
this._forEach((osc) => osc.start(time));
|
2019-07-16 20:30:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2019-09-14 20:39:18 +00:00
|
|
|
* stop the oscillator
|
2019-07-16 20:30:09 +00:00
|
|
|
*/
|
|
|
|
protected _stop(time: Time): void {
|
|
|
|
time = this.toSeconds(time);
|
2024-05-03 15:09:28 +00:00
|
|
|
this._forEach((osc) => osc.stop(time));
|
2019-07-16 20:30:09 +00:00
|
|
|
}
|
|
|
|
|
2019-12-14 21:09:24 +00:00
|
|
|
protected _restart(time: Seconds): void {
|
2024-05-03 15:09:28 +00:00
|
|
|
this._forEach((osc) => osc.restart(time));
|
2019-07-16 20:30:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2019-09-14 20:39:18 +00:00
|
|
|
* Iterate over all of the oscillators
|
2019-07-16 20:30:09 +00:00
|
|
|
*/
|
|
|
|
private _forEach(iterator: (osc: Oscillator, index: number) => void): void {
|
|
|
|
for (let i = 0; i < this._oscillators.length; i++) {
|
|
|
|
iterator(this._oscillators[i], i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The type of the oscillator
|
|
|
|
*/
|
|
|
|
get type(): ToneOscillatorType {
|
|
|
|
return this._type;
|
|
|
|
}
|
|
|
|
set type(type: ToneOscillatorType) {
|
|
|
|
this._type = type;
|
2024-05-03 15:09:28 +00:00
|
|
|
this._forEach((osc) => (osc.type = type));
|
2019-07-16 20:30:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The detune spread between the oscillators. If "count" is
|
|
|
|
* set to 3 oscillators and the "spread" is set to 40,
|
|
|
|
* the three oscillators would be detuned like this: [-20, 0, 20]
|
|
|
|
* for a total detune spread of 40 cents.
|
2019-10-25 20:54:33 +00:00
|
|
|
* @example
|
2020-04-17 02:24:18 +00:00
|
|
|
* const fatOsc = new Tone.FatOscillator().toDestination().start();
|
2019-10-25 20:54:33 +00:00
|
|
|
* fatOsc.spread = 70;
|
2019-07-16 20:30:09 +00:00
|
|
|
*/
|
|
|
|
get spread(): Cents {
|
|
|
|
return this._spread;
|
|
|
|
}
|
|
|
|
set spread(spread: Cents) {
|
|
|
|
this._spread = spread;
|
|
|
|
if (this._oscillators.length > 1) {
|
|
|
|
const start = -spread / 2;
|
|
|
|
const step = spread / (this._oscillators.length - 1);
|
2024-05-03 15:09:28 +00:00
|
|
|
this._forEach((osc, i) => (osc.detune.value = start + step * i));
|
2019-07-16 20:30:09 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2019-10-25 20:54:33 +00:00
|
|
|
* The number of detuned oscillators. Must be an integer greater than 1.
|
|
|
|
* @example
|
2020-04-17 02:24:18 +00:00
|
|
|
* const fatOsc = new Tone.FatOscillator("C#3", "sawtooth").toDestination().start();
|
2019-10-25 20:54:33 +00:00
|
|
|
* // use 4 sawtooth oscillators
|
|
|
|
* fatOsc.count = 4;
|
2019-07-16 20:30:09 +00:00
|
|
|
*/
|
|
|
|
get count(): number {
|
|
|
|
return this._oscillators.length;
|
|
|
|
}
|
|
|
|
set count(count: number) {
|
2019-10-25 20:54:33 +00:00
|
|
|
assertRange(count, 1);
|
2019-07-16 20:30:09 +00:00
|
|
|
if (this._oscillators.length !== count) {
|
|
|
|
// dispose the previous oscillators
|
2024-05-03 15:09:28 +00:00
|
|
|
this._forEach((osc) => osc.dispose());
|
2019-07-16 20:30:09 +00:00
|
|
|
this._oscillators = [];
|
|
|
|
for (let i = 0; i < count; i++) {
|
|
|
|
const osc = new Oscillator({
|
2019-09-16 03:32:40 +00:00
|
|
|
context: this.context,
|
2019-11-22 18:26:16 +00:00
|
|
|
volume: -6 - count * 1.1,
|
|
|
|
type: this._type as NonCustomOscillatorType,
|
|
|
|
phase: this._phase + (i / count) * 360,
|
|
|
|
partialCount: this._partialCount,
|
2019-08-10 15:51:35 +00:00
|
|
|
onstop: i === 0 ? () => this.onstop(this) : noOp,
|
2019-07-16 20:30:09 +00:00
|
|
|
});
|
|
|
|
if (this.type === "custom") {
|
|
|
|
osc.partials = this._partials;
|
|
|
|
}
|
|
|
|
this.frequency.connect(osc.frequency);
|
|
|
|
this.detune.connect(osc.detune);
|
2019-12-06 19:10:33 +00:00
|
|
|
osc.detune.overridden = false;
|
2019-07-16 20:30:09 +00:00
|
|
|
osc.connect(this.output);
|
|
|
|
this._oscillators[i] = osc;
|
|
|
|
}
|
|
|
|
// set the spread
|
|
|
|
this.spread = this._spread;
|
|
|
|
if (this.state === "started") {
|
2024-05-03 15:09:28 +00:00
|
|
|
this._forEach((osc) => osc.start());
|
2019-07-16 20:30:09 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
get phase(): Degrees {
|
|
|
|
return this._phase;
|
|
|
|
}
|
|
|
|
set phase(phase: Degrees) {
|
|
|
|
this._phase = phase;
|
2024-05-03 15:09:28 +00:00
|
|
|
this._forEach(
|
|
|
|
(osc, i) => (osc.phase = this._phase + (i / this.count) * 360)
|
|
|
|
);
|
2019-07-16 20:30:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
get baseType(): OscillatorType {
|
|
|
|
return this._oscillators[0].baseType;
|
|
|
|
}
|
|
|
|
set baseType(baseType: OscillatorType) {
|
2024-05-03 15:09:28 +00:00
|
|
|
this._forEach((osc) => (osc.baseType = baseType));
|
2019-07-16 20:30:09 +00:00
|
|
|
this._type = this._oscillators[0].type;
|
|
|
|
}
|
|
|
|
|
|
|
|
get partials(): number[] {
|
|
|
|
return this._oscillators[0].partials;
|
|
|
|
}
|
|
|
|
set partials(partials: number[]) {
|
|
|
|
this._partials = partials;
|
2019-10-09 21:32:37 +00:00
|
|
|
this._partialCount = this._partials.length;
|
2019-07-17 16:55:34 +00:00
|
|
|
if (partials.length) {
|
|
|
|
this._type = "custom";
|
2024-05-03 15:09:28 +00:00
|
|
|
this._forEach((osc) => (osc.partials = partials));
|
2019-07-17 16:55:34 +00:00
|
|
|
}
|
2019-07-16 20:30:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
get partialCount(): number {
|
|
|
|
return this._oscillators[0].partialCount;
|
|
|
|
}
|
|
|
|
set partialCount(partialCount: number) {
|
|
|
|
this._partialCount = partialCount;
|
2024-05-03 15:09:28 +00:00
|
|
|
this._forEach((osc) => (osc.partialCount = partialCount));
|
2019-07-16 20:30:09 +00:00
|
|
|
this._type = this._oscillators[0].type;
|
|
|
|
}
|
|
|
|
|
2019-11-17 18:09:19 +00:00
|
|
|
async asArray(length = 1024): Promise<Float32Array> {
|
2019-10-08 23:22:59 +00:00
|
|
|
return generateWaveform(this, length);
|
|
|
|
}
|
|
|
|
|
2019-07-16 20:30:09 +00:00
|
|
|
/**
|
2019-09-14 20:39:18 +00:00
|
|
|
* Clean up.
|
2019-07-16 20:30:09 +00:00
|
|
|
*/
|
|
|
|
dispose(): this {
|
|
|
|
super.dispose();
|
|
|
|
this.frequency.dispose();
|
|
|
|
this.detune.dispose();
|
2024-05-03 15:09:28 +00:00
|
|
|
this._forEach((osc) => osc.dispose());
|
2019-07-16 20:30:09 +00:00
|
|
|
return this;
|
|
|
|
}
|
|
|
|
}
|