import { Volume } from "../../component/channel/Volume.js"; import { Decibels } from "../type/Units.js"; import { optionsFromArguments } from "../util/Defaults.js"; import { onContextClose, onContextInit } from "./ContextInitialization.js"; import { Gain } from "./Gain.js"; import { Param } from "./Param.js"; import { connectSeries, ToneAudioNode, ToneAudioNodeOptions, } from "./ToneAudioNode.js"; interface DestinationOptions extends ToneAudioNodeOptions { volume: Decibels; mute: boolean; } /** * A single master output which is connected to the * AudioDestinationNode (aka your speakers). * It provides useful conveniences such as the ability * to set the volume and mute the entire application. * It also gives you the ability to apply master effects to your application. * * @example * const oscillator = new Tone.Oscillator().start(); * // the audio will go from the oscillator to the speakers * oscillator.connect(Tone.getDestination()); * // a convenience for connecting to the master output is also provided: * oscillator.toDestination(); * @category Core */ export class DestinationClass extends ToneAudioNode { readonly name: string = "Destination"; input: Volume = new Volume({ context: this.context }); output: Gain = new Gain({ context: this.context }); /** * The volume of the master output in decibels. -Infinity is silent, and 0 is no change. * @example * const osc = new Tone.Oscillator().toDestination(); * osc.start(); * // ramp the volume down to silent over 10 seconds * Tone.getDestination().volume.rampTo(-Infinity, 10); */ volume: Param<"decibels"> = this.input.volume; constructor(options: Partial); constructor() { super(optionsFromArguments(DestinationClass.getDefaults(), arguments)); const options = optionsFromArguments( DestinationClass.getDefaults(), arguments ); connectSeries( this.input, this.output, this.context.rawContext.destination ); this.mute = options.mute; this._internalChannels = [ this.input, this.context.rawContext.destination, this.output, ]; } static getDefaults(): DestinationOptions { return Object.assign(ToneAudioNode.getDefaults(), { mute: false, volume: 0, }); } /** * Mute the output. * @example * const oscillator = new Tone.Oscillator().start().toDestination(); * setTimeout(() => { * // mute the output * Tone.Destination.mute = true; * }, 1000); */ get mute(): boolean { return this.input.mute; } set mute(mute: boolean) { this.input.mute = mute; } /** * Add a master effects chain. NOTE: this will disconnect any nodes which were previously * chained in the master effects chain. * @param args All arguments will be connected in a row and the Master will be routed through it. * @example * // route all audio through a filter and compressor * const lowpass = new Tone.Filter(800, "lowpass"); * const compressor = new Tone.Compressor(-18); * Tone.Destination.chain(lowpass, compressor); */ chain(...args: Array): this { this.input.disconnect(); args.unshift(this.input); args.push(this.output); connectSeries(...args); return this; } /** * The maximum number of channels the system can output * @example * console.log(Tone.Destination.maxChannelCount); */ get maxChannelCount(): number { return this.context.rawContext.destination.maxChannelCount; } /** * Clean up */ dispose(): this { super.dispose(); this.volume.dispose(); return this; } } //------------------------------------- // INITIALIZATION //------------------------------------- onContextInit((context) => { context.destination = new DestinationClass({ context }); }); onContextClose((context) => { context.destination.dispose(); });