Tone.js/Tone/core/context/ToneAudioNode.ts

415 lines
12 KiB
TypeScript
Raw Normal View History

2024-05-03 14:10:40 +00:00
import { isAudioNode, isAudioParam } from "../util/AdvancedTypeCheck.js";
import { isDefined } from "../util/TypeCheck.js";
import { Param } from "./Param.js";
import { ToneWithContext, ToneWithContextOptions } from "./ToneWithContext.js";
import { assert, warn } from "../util/Debug.js";
2019-04-12 14:37:47 +00:00
2019-09-16 03:32:40 +00:00
export type InputNode = ToneAudioNode | AudioNode | Param<any> | AudioParam;
2019-04-12 14:37:47 +00:00
export type OutputNode = ToneAudioNode | AudioNode;
interface ChannelProperties {
2019-04-12 14:37:47 +00:00
channelCount: number;
channelCountMode: ChannelCountMode;
channelInterpretation: ChannelInterpretation;
}
/**
* The possible options for this node
*/
export type ToneAudioNodeOptions = ToneWithContextOptions;
2019-04-12 14:37:47 +00:00
/**
2019-08-21 20:01:12 +00:00
* ToneAudioNode is the base class for classes which process audio.
2024-04-29 14:48:28 +00:00
* @category Core
2019-04-12 14:37:47 +00:00
*/
2024-05-03 15:09:28 +00:00
export abstract class ToneAudioNode<
Options extends ToneAudioNodeOptions = ToneAudioNodeOptions,
> extends ToneWithContext<Options> {
2019-09-14 20:39:18 +00:00
/**
* The name of the class
*/
2021-10-13 17:32:29 +00:00
abstract readonly name: string;
2019-04-12 14:37:47 +00:00
/**
* The input node or nodes. If the object is a source,
* it does not have any input and this.input is undefined.
*/
abstract input: InputNode | undefined;
2019-04-12 14:37:47 +00:00
/**
* The output nodes. If the object is a sink,
* it does not have any output and this.output is undefined.
*/
abstract output: OutputNode | undefined;
2019-04-12 14:37:47 +00:00
/**
2019-09-14 20:39:18 +00:00
* The number of inputs feeding into the AudioNode.
* For source nodes, this will be 0.
2020-03-22 01:58:48 +00:00
* @example
* const node = new Tone.Gain();
2020-03-22 01:58:48 +00:00
* console.log(node.numberOfInputs);
2019-04-12 14:37:47 +00:00
*/
get numberOfInputs(): number {
if (isDefined(this.input)) {
if (isAudioParam(this.input) || this.input instanceof Param) {
return 1;
} else {
return this.input.numberOfInputs;
}
} else {
return 0;
}
}
2019-04-12 14:37:47 +00:00
/**
2019-09-14 20:39:18 +00:00
* The number of outputs of the AudioNode.
* @example
* const node = new Tone.Gain();
2020-03-22 01:58:48 +00:00
* console.log(node.numberOfOutputs);
2019-04-12 14:37:47 +00:00
*/
get numberOfOutputs(): number {
if (isDefined(this.output)) {
return this.output.numberOfOutputs;
} else {
return 0;
}
}
2019-04-12 14:37:47 +00:00
/**
* List all of the node that must be set to match the ChannelProperties
*/
protected _internalChannels: OutputNode[] = [];
2019-04-12 14:37:47 +00:00
//-------------------------------------
2019-04-12 14:37:47 +00:00
// AUDIO PROPERTIES
//-------------------------------------
2019-04-12 14:37:47 +00:00
/**
* Used to decide which nodes to get/set properties on
*/
private _isAudioNode(node: any): node is AudioNode | ToneAudioNode {
2024-05-03 15:09:28 +00:00
return (
isDefined(node) &&
(node instanceof ToneAudioNode || isAudioNode(node))
);
}
/**
* Get all of the audio nodes (either internal or input/output) which together
* make up how the class node responds to channel input/output
*/
private _getInternalNodes(): OutputNode[] {
const nodeList = this._internalChannels.slice(0);
if (this._isAudioNode(this.input)) {
nodeList.push(this.input);
}
if (this._isAudioNode(this.output)) {
if (this.input !== this.output) {
nodeList.push(this.output);
}
}
return nodeList;
}
2019-04-12 14:37:47 +00:00
/**
* Set the audio options for this node such as channelInterpretation
* channelCount, etc.
* @param options
*/
private _setChannelProperties(options: ChannelProperties): void {
const nodeList = this._getInternalNodes();
2024-05-03 15:09:28 +00:00
nodeList.forEach((node) => {
node.channelCount = options.channelCount;
node.channelCountMode = options.channelCountMode;
node.channelInterpretation = options.channelInterpretation;
});
2019-04-12 14:37:47 +00:00
}
/**
* Get the current audio options for this node such as channelInterpretation
* channelCount, etc.
*/
private _getChannelProperties(): ChannelProperties {
const nodeList = this._getInternalNodes();
2024-05-03 15:09:28 +00:00
assert(
nodeList.length > 0,
"ToneAudioNode does not have any internal nodes"
);
// use the first node to get properties
// they should all be the same
const node = nodeList[0];
return {
channelCount: node.channelCount,
channelCountMode: node.channelCountMode,
channelInterpretation: node.channelInterpretation,
};
2019-04-12 14:37:47 +00:00
}
/**
2019-09-14 20:39:18 +00:00
* channelCount is the number of channels used when up-mixing and down-mixing
* connections to any inputs to the node. The default value is 2 except for
* specific nodes where its value is specially determined.
2019-04-12 14:37:47 +00:00
*/
get channelCount(): number {
return this._getChannelProperties().channelCount;
}
set channelCount(channelCount) {
2019-04-12 14:37:47 +00:00
const props = this._getChannelProperties();
// merge it with the other properties
this._setChannelProperties(Object.assign(props, { channelCount }));
}
/**
2019-09-14 20:39:18 +00:00
* channelCountMode determines how channels will be counted when up-mixing and
* down-mixing connections to any inputs to the node.
* The default value is "max". This attribute has no effect for nodes with no inputs.
2019-08-21 20:01:12 +00:00
* * "max" - computedNumberOfChannels is the maximum of the number of channels of all connections to an input. In this mode channelCount is ignored.
* * "clamped-max" - computedNumberOfChannels is determined as for "max" and then clamped to a maximum value of the given channelCount.
* * "explicit" - computedNumberOfChannels is the exact value as specified by the channelCount.
2019-04-12 14:37:47 +00:00
*/
get channelCountMode(): ChannelCountMode {
return this._getChannelProperties().channelCountMode;
}
set channelCountMode(channelCountMode) {
2019-04-12 14:37:47 +00:00
const props = this._getChannelProperties();
// merge it with the other properties
this._setChannelProperties(Object.assign(props, { channelCountMode }));
}
/**
2019-09-14 20:39:18 +00:00
* channelInterpretation determines how individual channels will be treated
* when up-mixing and down-mixing connections to any inputs to the node.
* The default value is "speakers".
2019-04-12 14:37:47 +00:00
*/
get channelInterpretation(): ChannelInterpretation {
return this._getChannelProperties().channelInterpretation;
}
set channelInterpretation(channelInterpretation) {
2019-04-12 14:37:47 +00:00
const props = this._getChannelProperties();
// merge it with the other properties
2024-05-03 15:09:28 +00:00
this._setChannelProperties(
Object.assign(props, { channelInterpretation })
);
2019-04-12 14:37:47 +00:00
}
//-------------------------------------
2019-04-12 14:37:47 +00:00
// CONNECTIONS
//-------------------------------------
2019-04-12 14:37:47 +00:00
/**
* connect the output of a ToneAudioNode to an AudioParam, AudioNode, or ToneAudioNode
2019-10-23 03:04:52 +00:00
* @param destination The output to connect to
2019-04-12 14:37:47 +00:00
* @param outputNum The output to connect from
* @param inputNum The input to connect to
*/
connect(destination: InputNode, outputNum = 0, inputNum = 0): this {
connect(this, destination, outputNum, inputNum);
return this;
}
/**
* Connect the output to the context's destination node.
2020-03-22 01:58:48 +00:00
* @example
2020-05-27 01:09:32 +00:00
* const osc = new Tone.Oscillator("C2").start();
2020-03-22 01:58:48 +00:00
* osc.toDestination();
2019-04-12 14:37:47 +00:00
*/
toDestination(): this {
2019-04-12 14:37:47 +00:00
this.connect(this.context.destination);
return this;
}
/**
* Connect the output to the context's destination node.
2024-04-29 16:59:49 +00:00
* @see {@link toDestination}
2019-08-12 04:15:11 +00:00
* @deprecated
*/
toMaster(): this {
warn("toMaster() has been renamed toDestination()");
return this.toDestination();
}
2019-04-12 14:37:47 +00:00
/**
2019-09-08 17:39:56 +00:00
* disconnect the output
2019-04-12 14:37:47 +00:00
*/
disconnect(destination?: InputNode, outputNum = 0, inputNum = 0): this {
disconnect(this, destination, outputNum, inputNum);
return this;
}
/**
2019-09-14 20:39:18 +00:00
* Connect the output of this node to the rest of the nodes in series.
* @example
2020-07-19 20:22:47 +00:00
* const player = new Tone.Player("https://tonejs.github.io/audio/drum-samples/handdrum-loop.mp3");
2020-05-27 01:09:32 +00:00
* player.autostart = true;
* const filter = new Tone.AutoFilter(4).start();
* const distortion = new Tone.Distortion(0.5);
* // connect the player to the filter, distortion and then to the master output
* player.chain(filter, distortion, Tone.Destination);
2019-04-12 14:37:47 +00:00
*/
chain(...nodes: InputNode[]): this {
2019-06-24 18:35:32 +00:00
connectSeries(this, ...nodes);
2019-04-12 14:37:47 +00:00
return this;
}
/**
2019-09-14 20:39:18 +00:00
* connect the output of this node to the rest of the nodes in parallel.
2020-05-27 01:09:32 +00:00
* @example
2020-07-19 20:22:47 +00:00
* const player = new Tone.Player("https://tonejs.github.io/audio/drum-samples/conga-rhythm.mp3");
2020-05-27 01:09:32 +00:00
* player.autostart = true;
* const pitchShift = new Tone.PitchShift(4).toDestination();
* const filter = new Tone.Filter("G5").toDestination();
* // connect a node to the pitch shift and filter in parallel
2020-07-18 15:30:00 +00:00
* player.fan(pitchShift, filter);
2019-04-12 14:37:47 +00:00
*/
fan(...nodes: InputNode[]): this {
2024-05-03 15:09:28 +00:00
nodes.forEach((node) => this.connect(node));
2019-04-12 14:37:47 +00:00
return this;
}
/**
* Dispose and disconnect
*/
dispose(): this {
super.dispose();
2019-04-12 14:37:47 +00:00
if (isDefined(this.input)) {
if (this.input instanceof ToneAudioNode) {
this.input.dispose();
} else if (isAudioNode(this.input)) {
this.input.disconnect();
2019-04-12 14:37:47 +00:00
}
}
if (isDefined(this.output)) {
if (this.output instanceof ToneAudioNode) {
this.output.dispose();
} else if (isAudioNode(this.output)) {
2019-04-12 14:37:47 +00:00
this.output.disconnect();
}
}
this._internalChannels = [];
return this;
}
}
2019-07-11 13:57:06 +00:00
2019-09-14 21:47:07 +00:00
//-------------------------------------
2019-07-11 13:57:06 +00:00
// CONNECTIONS
2019-09-14 21:47:07 +00:00
//-------------------------------------
2019-07-11 13:57:06 +00:00
/**
2019-09-14 20:39:18 +00:00
* connect together all of the arguments in series
* @param nodes
2019-07-11 13:57:06 +00:00
*/
export function connectSeries(...nodes: InputNode[]): void {
const first = nodes.shift();
nodes.reduce((prev, current) => {
if (prev instanceof ToneAudioNode) {
prev.connect(current);
} else if (isAudioNode(prev)) {
2019-07-11 13:57:06 +00:00
connect(prev, current);
}
return current;
}, first);
}
/**
* Connect two nodes together so that signal flows from the
* first node to the second. Optionally specify the input and output channels.
* @param srcNode The source node
* @param dstNode The destination node
* @param outputNumber The output channel of the srcNode
* @param inputNumber The input channel of the dstNode
*/
2024-05-03 15:09:28 +00:00
export function connect(
srcNode: OutputNode,
dstNode: InputNode,
outputNumber = 0,
inputNumber = 0
): void {
2019-08-10 22:07:10 +00:00
assert(isDefined(srcNode), "Cannot connect from undefined node");
assert(isDefined(dstNode), "Cannot connect to undefined node");
if (dstNode instanceof ToneAudioNode || isAudioNode(dstNode)) {
2024-05-03 15:09:28 +00:00
assert(
dstNode.numberOfInputs > 0,
"Cannot connect to node with no inputs"
);
2019-08-10 22:07:10 +00:00
}
2024-05-03 15:09:28 +00:00
assert(
srcNode.numberOfOutputs > 0,
"Cannot connect from node with no outputs"
);
2019-08-10 22:07:10 +00:00
2019-07-11 13:57:06 +00:00
// resolve the input of the dstNode
2024-05-03 15:09:28 +00:00
while (dstNode instanceof ToneAudioNode || dstNode instanceof Param) {
if (isDefined(dstNode.input)) {
2019-07-11 13:57:06 +00:00
dstNode = dstNode.input;
}
}
while (srcNode instanceof ToneAudioNode) {
if (isDefined(srcNode.output)) {
2019-07-11 13:57:06 +00:00
srcNode = srcNode.output;
}
}
// make the connection
if (isAudioParam(dstNode)) {
srcNode.connect(dstNode as AudioParam, outputNumber);
2019-07-11 13:57:06 +00:00
} else {
srcNode.connect(dstNode, outputNumber, inputNumber);
}
}
/**
* Disconnect a node from all nodes or optionally include a destination node and input/output channels.
* @param srcNode The source node
* @param dstNode The destination node
* @param outputNumber The output channel of the srcNode
* @param inputNumber The input channel of the dstNode
*/
export function disconnect(
srcNode: OutputNode,
dstNode?: InputNode,
outputNumber = 0,
2024-05-03 15:09:28 +00:00
inputNumber = 0
2019-07-11 13:57:06 +00:00
): void {
// resolve the destination node
if (isDefined(dstNode)) {
while (dstNode instanceof ToneAudioNode) {
2019-11-03 16:41:03 +00:00
dstNode = dstNode.input;
2019-07-11 13:57:06 +00:00
}
}
// resolve the src node
2024-05-03 15:09:28 +00:00
while (!isAudioNode(srcNode)) {
if (isDefined(srcNode.output)) {
2019-07-11 13:57:06 +00:00
srcNode = srcNode.output;
}
}
if (isAudioParam(dstNode)) {
2019-10-31 18:49:56 +00:00
srcNode.disconnect(dstNode, outputNumber);
} else if (isAudioNode(dstNode)) {
2019-07-11 13:57:06 +00:00
srcNode.disconnect(dstNode, outputNumber, inputNumber);
} else {
srcNode.disconnect();
}
}
2021-03-17 18:55:45 +00:00
/**
2021-04-15 15:43:49 +00:00
* Connect the output of one or more source nodes to a single destination node
2021-03-18 01:04:01 +00:00
* @param nodes One or more source nodes followed by one destination node
2021-03-17 18:55:45 +00:00
* @example
* const player = new Tone.Player("https://tonejs.github.io/audio/drum-samples/conga-rhythm.mp3");
* const player1 = new Tone.Player("https://tonejs.github.io/audio/drum-samples/conga-rhythm.mp3");
* const filter = new Tone.Filter("G5").toDestination();
* // connect nodes to a common destination
* Tone.fanIn(player, player1, filter);
2021-03-17 18:55:45 +00:00
*/
2021-04-15 15:43:49 +00:00
export function fanIn(...nodes: OutputNode[]): void {
2021-03-18 01:04:01 +00:00
const dstNode = nodes.pop();
2021-04-15 15:43:49 +00:00
if (isDefined(dstNode)) {
2024-05-03 15:09:28 +00:00
nodes.forEach((node) => connect(node, dstNode));
2021-04-15 15:43:49 +00:00
}
2021-03-17 18:55:45 +00:00
}