Tone.js/Tone/source/oscillator/PWMOscillator.ts

201 lines
4.5 KiB
TypeScript
Raw Normal View History

import { Degrees, Frequency, Seconds, Time } from "../../core/type/Units.js";
import { optionsFromArguments } from "../../core/util/Defaults.js";
import { readOnly } from "../../core/util/Interface.js";
import { Multiply } from "../../signal/Multiply.js";
import { Signal } from "../../signal/Signal.js";
import { Source } from "../Source.js";
import { Oscillator } from "./Oscillator.js";
import {
generateWaveform,
PWMOscillatorOptions,
ToneOscillatorInterface,
} from "./OscillatorInterface.js";
import { PulseOscillator } from "./PulseOscillator.js";
2019-07-16 21:10:07 +00:00
export { PWMOscillatorOptions } from "./OscillatorInterface.js";
2019-09-04 22:34:42 +00:00
2019-07-16 21:10:07 +00:00
/**
* PWMOscillator modulates the width of a Tone.PulseOscillator
* at the modulationFrequency. This has the effect of continuously
* changing the timbre of the oscillator by altering the harmonics
* generated.
* @example
2020-07-26 20:55:06 +00:00
* return Tone.Offline(() => {
* const pwm = new Tone.PWMOscillator(60, 0.3).toDestination().start();
* }, 0.1, 1);
2019-09-16 14:15:23 +00:00
* @category Source
2019-07-16 21:10:07 +00:00
*/
export class PWMOscillator
extends Source<PWMOscillatorOptions>
implements ToneOscillatorInterface
{
2019-09-04 23:18:44 +00:00
readonly name: string = "PWMOscillator";
2019-07-16 21:10:07 +00:00
2019-07-17 16:55:34 +00:00
readonly sourceType = "pwm";
2019-07-16 21:10:07 +00:00
/**
2019-09-14 20:39:18 +00:00
* the pulse oscillator
2019-07-16 21:10:07 +00:00
*/
private _pulse: PulseOscillator;
2019-07-16 21:10:07 +00:00
/**
2019-09-14 20:39:18 +00:00
* the modulator
2019-07-16 21:10:07 +00:00
*/
private _modulator: Oscillator;
2019-07-16 21:10:07 +00:00
/**
2019-09-14 20:39:18 +00:00
* Scale the oscillator so it doesn't go silent
* at the extreme values.
2019-07-16 21:10:07 +00:00
*/
private _scale: Multiply = new Multiply({
context: this.context,
value: 2,
});
/**
2019-09-14 20:39:18 +00:00
* The frequency control.
2019-07-16 21:10:07 +00:00
*/
readonly frequency: Signal<"frequency">;
2019-07-16 21:10:07 +00:00
/**
2019-09-14 20:39:18 +00:00
* The detune of the oscillator.
2019-07-16 21:10:07 +00:00
*/
readonly detune: Signal<"cents">;
2019-07-16 21:10:07 +00:00
/**
2020-04-30 03:34:01 +00:00
* The width modulation rate of the oscillator.
* @example
2020-07-26 20:55:06 +00:00
* return Tone.Offline(() => {
* const osc = new Tone.PWMOscillator(20, 2).toDestination().start();
* }, 0.1, 1);
2019-07-16 21:10:07 +00:00
*/
readonly modulationFrequency: Signal<"frequency">;
2019-07-16 21:10:07 +00:00
2019-08-27 15:47:52 +00:00
/**
* @param {Frequency} frequency The starting frequency of the oscillator.
* @param {Frequency} modulationFrequency The modulation frequency of the width of the pulse.
*/
2019-07-17 16:55:34 +00:00
constructor(frequency?: Frequency, modulationFrequency?: Frequency);
2019-08-27 15:47:52 +00:00
constructor(options?: Partial<PWMOscillatorOptions>);
2019-07-16 21:10:07 +00:00
constructor() {
const options = optionsFromArguments(
PWMOscillator.getDefaults(),
arguments,
["frequency", "modulationFrequency"]
);
super(options);
2019-07-16 21:10:07 +00:00
this._pulse = new PulseOscillator({
context: this.context,
frequency: options.modulationFrequency,
});
2019-07-16 21:10:07 +00:00
// change the pulse oscillator type
this._pulse.carrierType = "sine";
2019-07-16 21:10:07 +00:00
2019-09-16 03:32:40 +00:00
this.modulationFrequency = this._pulse.frequency;
this._modulator = new Oscillator({
context: this.context,
detune: options.detune,
frequency: options.frequency,
2019-08-10 15:51:35 +00:00
onstop: () => this.onstop(this),
phase: options.phase,
});
this.frequency = this._modulator.frequency;
this.detune = this._modulator.detune;
2019-07-16 21:10:07 +00:00
// connections
this._modulator.chain(this._scale, this._pulse.width);
this._pulse.connect(this.output);
readOnly(this, ["modulationFrequency", "frequency", "detune"]);
}
static getDefaults(): PWMOscillatorOptions {
return Object.assign(Source.getDefaults(), {
detune: 0,
frequency: 440,
2019-07-16 21:10:07 +00:00
modulationFrequency: 0.4,
phase: 0,
2021-10-13 22:38:30 +00:00
type: "pwm" as const,
2019-07-16 21:10:07 +00:00
});
}
/**
2019-09-14 20:39:18 +00:00
* start the oscillator
2019-07-16 21:10:07 +00:00
*/
protected _start(time: Time): void {
time = this.toSeconds(time);
this._modulator.start(time);
this._pulse.start(time);
}
/**
2019-09-14 20:39:18 +00:00
* stop the oscillator
2019-07-16 21:10:07 +00:00
*/
protected _stop(time: Time): void {
time = this.toSeconds(time);
this._modulator.stop(time);
this._pulse.stop(time);
}
/**
2019-09-14 20:39:18 +00:00
* restart the oscillator
2019-07-16 21:10:07 +00:00
*/
protected _restart(time: Seconds): void {
2019-07-16 21:10:07 +00:00
this._modulator.restart(time);
this._pulse.restart(time);
}
/**
* The type of the oscillator. Always returns "pwm".
*/
get type(): "pwm" {
2019-07-16 21:10:07 +00:00
return "pwm";
}
/**
* The baseType of the oscillator. Always returns "pwm".
*/
get baseType(): "pwm" {
2019-07-16 21:10:07 +00:00
return "pwm";
}
/**
* The partials of the waveform. Cannot set partials for this waveform type
*/
get partials(): number[] {
return [];
}
/**
* No partials for this waveform type.
*/
get partialCount(): number {
return 0;
}
/**
* The phase of the oscillator in degrees.
*/
get phase(): Degrees {
return this._modulator.phase;
}
set phase(phase: Degrees) {
this._modulator.phase = phase;
}
2019-11-17 18:09:19 +00:00
async asArray(length = 1024): Promise<Float32Array> {
return generateWaveform(this, length);
}
2019-07-16 21:10:07 +00:00
/**
2019-09-14 20:39:18 +00:00
* Clean up.
2019-07-16 21:10:07 +00:00
*/
dispose(): this {
super.dispose();
this._pulse.dispose();
this._scale.dispose();
this._modulator.dispose();
return this;
}
}