import { FrequencyClass } from "../core/type/Frequency";
import { Frequency, Positive, Time } from "../core/type/Units";
import { deepMerge, optionsFromArguments } from "../core/util/Defaults";
import { readOnly, RecursivePartial } from "../core/util/Interface";
import { Monophonic } from "./Monophonic";
import { Synth, SynthOptions } from "./Synth";
export interface MembraneSynthOptions extends SynthOptions {
pitchDecay: Time;
octaves: Positive;
}
/**
* MembraneSynth makes kick and tom sounds using a single oscillator
* with an amplitude envelope and frequency ramp. A Tone.OmniOscillator
* is routed through a Tone.AmplitudeEnvelope to the output. The drum
* quality of the sound comes from the frequency envelope applied
* during MembraneSynth.triggerAttack(note). The frequency envelope
* starts at note * .octaves
and ramps to note
* over the duration of .pitchDecay
.
* @example
* var synth = new MembraneSynth().toMaster();
* synth.triggerAttackRelease("C2", "8n");
* @category Instrument
*/
export class MembraneSynth extends Synth {
readonly name: string = "MembraneSynth";
/**
* The number of octaves the pitch envelope ramps.
* @min 0.5
* @max 8
*/
octaves: Positive;
/**
* The amount of time the frequency envelope takes.
* @min 0
* @max 0.5
*/
pitchDecay: Time;
/**
* Portamento is ignored in this synth. use pitch decay instead.
*/
readonly portamento = 0;
/**
* @param options the options available for the synth see defaults
*/
constructor(options?: RecursivePartial)
constructor() {
super(optionsFromArguments(MembraneSynth.getDefaults(), arguments));
const options = optionsFromArguments(MembraneSynth.getDefaults(), arguments);
this.pitchDecay = options.pitchDecay;
this.octaves = options.octaves;
readOnly(this, ["oscillator", "envelope"]);
}
static getDefaults(): MembraneSynthOptions {
return deepMerge(Monophonic.getDefaults(), Synth.getDefaults(), {
envelope: {
attack: 0.001,
attackCurve: "exponential",
decay: 0.4,
release: 1.4,
sustain: 0.01,
},
octaves: 10,
oscillator: {
type: "sine",
},
pitchDecay: 0.05,
});
}
setNote(note: Frequency | FrequencyClass, time?: Time): this {
const seconds = this.toSeconds(time);
const hertz = this.toFrequency(note instanceof FrequencyClass ? note.toFrequency() : note);
const maxNote = hertz * this.octaves;
this.oscillator.frequency.setValueAtTime(maxNote, seconds);
this.oscillator.frequency.exponentialRampToValueAtTime(hertz, seconds + this.toSeconds(this.pitchDecay));
return this;
}
dispose(): this {
super.dispose();
return this;
}
}