2024-05-03 14:10:40 +00:00
|
|
|
import { Ticker, TickerClockSource } from "../clock/Ticker.js";
|
|
|
|
import { Seconds } from "../type/Units.js";
|
|
|
|
import { isAudioContext } from "../util/AdvancedTypeCheck.js";
|
|
|
|
import { optionsFromArguments } from "../util/Defaults.js";
|
|
|
|
import { Timeline } from "../util/Timeline.js";
|
|
|
|
import { isDefined } from "../util/TypeCheck.js";
|
2021-01-01 01:53:38 +00:00
|
|
|
import {
|
|
|
|
AnyAudioContext,
|
|
|
|
createAudioContext,
|
|
|
|
createAudioWorkletNode,
|
2024-05-03 14:10:40 +00:00
|
|
|
} from "./AudioContext.js";
|
|
|
|
import { closeContext, initializeContext } from "./ContextInitialization.js";
|
|
|
|
import { BaseContext, ContextLatencyHint } from "./BaseContext.js";
|
|
|
|
import { assert } from "../util/Debug.js";
|
2019-07-11 13:57:06 +00:00
|
|
|
|
2024-04-28 17:05:26 +00:00
|
|
|
type Transport = import("../clock/Transport").TransportClass;
|
|
|
|
type Destination = import("./Destination").DestinationClass;
|
|
|
|
type Listener = import("./Listener").ListenerClass;
|
|
|
|
type Draw = import("../util/Draw").DrawClass;
|
2019-04-12 14:37:47 +00:00
|
|
|
|
2019-05-23 18:00:49 +00:00
|
|
|
export interface ContextOptions {
|
2019-04-12 14:37:47 +00:00
|
|
|
clockSource: TickerClockSource;
|
|
|
|
latencyHint: ContextLatencyHint;
|
|
|
|
lookAhead: Seconds;
|
|
|
|
updateInterval: Seconds;
|
2019-08-16 16:49:04 +00:00
|
|
|
context: AnyAudioContext;
|
2019-04-12 14:37:47 +00:00
|
|
|
}
|
|
|
|
|
2019-05-23 18:00:49 +00:00
|
|
|
export interface ContextTimeoutEvent {
|
2019-04-12 14:37:47 +00:00
|
|
|
callback: (...args: any[]) => void;
|
|
|
|
id: number;
|
|
|
|
time: Seconds;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Wrapper around the native AudioContext.
|
2019-08-26 17:44:43 +00:00
|
|
|
* @category Core
|
2019-04-12 14:37:47 +00:00
|
|
|
*/
|
2019-11-06 22:32:19 +00:00
|
|
|
export class Context extends BaseContext {
|
2019-09-04 23:18:44 +00:00
|
|
|
readonly name: string = "Context";
|
2019-04-12 14:37:47 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* private reference to the BaseAudioContext
|
|
|
|
*/
|
2019-08-16 16:49:04 +00:00
|
|
|
protected readonly _context: AnyAudioContext;
|
2019-04-12 14:37:47 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* A reliable callback method
|
|
|
|
*/
|
|
|
|
private readonly _ticker: Ticker;
|
|
|
|
|
|
|
|
/**
|
2019-09-14 20:39:18 +00:00
|
|
|
* The default latency hint
|
2019-04-12 14:37:47 +00:00
|
|
|
*/
|
2020-05-12 16:31:17 +00:00
|
|
|
private _latencyHint!: ContextLatencyHint | Seconds;
|
2019-04-12 14:37:47 +00:00
|
|
|
|
|
|
|
/**
|
2019-09-14 20:39:18 +00:00
|
|
|
* An object containing all of the constants AudioBufferSourceNodes
|
2019-04-12 14:37:47 +00:00
|
|
|
*/
|
|
|
|
private _constants = new Map<number, AudioBufferSourceNode>();
|
|
|
|
|
|
|
|
/**
|
2019-09-14 20:39:18 +00:00
|
|
|
* All of the setTimeout events.
|
2019-04-12 14:37:47 +00:00
|
|
|
*/
|
2019-05-23 18:00:49 +00:00
|
|
|
private _timeouts: Timeline<ContextTimeoutEvent> = new Timeline();
|
2019-04-12 14:37:47 +00:00
|
|
|
|
|
|
|
/**
|
2019-09-14 20:39:18 +00:00
|
|
|
* The timeout id counter
|
2019-04-12 14:37:47 +00:00
|
|
|
*/
|
|
|
|
private _timeoutIds = 0;
|
|
|
|
|
2019-05-23 18:00:49 +00:00
|
|
|
/**
|
|
|
|
* A reference the Transport singleton belonging to this context
|
|
|
|
*/
|
2019-06-23 18:59:27 +00:00
|
|
|
private _transport!: Transport;
|
2019-12-22 02:01:45 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* A reference the Listener singleton belonging to this context
|
|
|
|
*/
|
|
|
|
private _listener!: Listener;
|
2020-04-15 01:06:21 +00:00
|
|
|
|
2019-06-23 18:59:27 +00:00
|
|
|
/**
|
|
|
|
* A reference the Destination singleton belonging to this context
|
|
|
|
*/
|
|
|
|
private _destination!: Destination;
|
|
|
|
|
2019-10-28 16:12:27 +00:00
|
|
|
/**
|
|
|
|
* A reference the Transport singleton belonging to this context
|
|
|
|
*/
|
|
|
|
private _draw!: Draw;
|
|
|
|
|
2019-06-23 18:59:27 +00:00
|
|
|
/**
|
|
|
|
* Private indicator if the context has been initialized
|
|
|
|
*/
|
2019-11-17 18:09:19 +00:00
|
|
|
private _initialized = false;
|
2019-05-23 18:00:49 +00:00
|
|
|
|
2022-03-01 17:38:26 +00:00
|
|
|
/**
|
|
|
|
* Private indicator if a close() has been called on the context, since close is async
|
|
|
|
*/
|
|
|
|
private _closeStarted = false;
|
|
|
|
|
2019-10-04 15:51:52 +00:00
|
|
|
/**
|
|
|
|
* Indicates if the context is an OfflineAudioContext or an AudioContext
|
|
|
|
*/
|
|
|
|
readonly isOffline: boolean = false;
|
|
|
|
|
2019-08-16 16:49:04 +00:00
|
|
|
constructor(context?: AnyAudioContext);
|
2019-05-23 18:00:49 +00:00
|
|
|
constructor(options?: Partial<ContextOptions>);
|
|
|
|
constructor() {
|
2019-04-12 14:37:47 +00:00
|
|
|
super();
|
2021-01-01 01:53:38 +00:00
|
|
|
const options = optionsFromArguments(Context.getDefaults(), arguments, [
|
|
|
|
"context",
|
|
|
|
]);
|
2019-04-12 14:37:47 +00:00
|
|
|
|
2019-10-14 03:20:42 +00:00
|
|
|
if (options.context) {
|
|
|
|
this._context = options.context;
|
2021-11-29 16:14:14 +00:00
|
|
|
// custom context provided, latencyHint unknown (unless explicitly provided in options)
|
|
|
|
this._latencyHint = arguments[0]?.latencyHint || "";
|
2019-10-14 03:20:42 +00:00
|
|
|
} else {
|
2020-05-12 16:31:17 +00:00
|
|
|
this._context = createAudioContext({
|
|
|
|
latencyHint: options.latencyHint,
|
|
|
|
});
|
2021-11-29 16:14:14 +00:00
|
|
|
this._latencyHint = options.latencyHint;
|
2019-10-14 03:20:42 +00:00
|
|
|
}
|
2019-04-12 14:37:47 +00:00
|
|
|
|
2021-01-01 01:53:38 +00:00
|
|
|
this._ticker = new Ticker(
|
|
|
|
this.emit.bind(this, "tick"),
|
|
|
|
options.clockSource,
|
2021-11-29 16:14:14 +00:00
|
|
|
options.updateInterval,
|
|
|
|
this._context.sampleRate
|
2021-01-01 01:53:38 +00:00
|
|
|
);
|
2019-04-12 14:37:47 +00:00
|
|
|
this.on("tick", this._timeoutLoop.bind(this));
|
|
|
|
|
|
|
|
// fwd events from the context
|
2019-09-04 02:00:03 +00:00
|
|
|
this._context.onstatechange = () => {
|
2019-04-12 14:37:47 +00:00
|
|
|
this.emit("statechange", this.state);
|
2019-09-04 02:00:03 +00:00
|
|
|
};
|
2024-05-03 15:09:28 +00:00
|
|
|
|
2021-11-29 16:14:14 +00:00
|
|
|
// if no custom updateInterval provided, updateInterval will be derived by lookAhead setter
|
2024-05-03 15:09:28 +00:00
|
|
|
this[
|
|
|
|
arguments[0]?.hasOwnProperty("updateInterval")
|
|
|
|
? "_lookAhead"
|
|
|
|
: "lookAhead"
|
|
|
|
] = options.lookAhead;
|
2019-04-12 14:37:47 +00:00
|
|
|
}
|
|
|
|
|
2019-06-23 18:59:27 +00:00
|
|
|
static getDefaults(): ContextOptions {
|
|
|
|
return {
|
|
|
|
clockSource: "worker",
|
|
|
|
latencyHint: "interactive",
|
|
|
|
lookAhead: 0.1,
|
2019-10-30 22:05:28 +00:00
|
|
|
updateInterval: 0.05,
|
2019-10-14 03:20:42 +00:00
|
|
|
} as ContextOptions;
|
2019-06-23 18:59:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Finish setting up the context. **You usually do not need to do this manually.**
|
|
|
|
*/
|
2019-10-03 21:33:39 +00:00
|
|
|
private initialize(): this {
|
2019-06-23 18:59:27 +00:00
|
|
|
if (!this._initialized) {
|
|
|
|
// add any additional modules
|
2019-07-11 13:57:06 +00:00
|
|
|
initializeContext(this);
|
2019-06-23 18:59:27 +00:00
|
|
|
this._initialized = true;
|
|
|
|
}
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
2019-09-14 21:47:07 +00:00
|
|
|
//---------------------------
|
2019-04-12 14:37:47 +00:00
|
|
|
// BASE AUDIO CONTEXT METHODS
|
2019-09-14 21:47:07 +00:00
|
|
|
//---------------------------
|
2019-04-12 14:37:47 +00:00
|
|
|
|
|
|
|
createAnalyser(): AnalyserNode {
|
|
|
|
return this._context.createAnalyser();
|
|
|
|
}
|
|
|
|
createOscillator(): OscillatorNode {
|
|
|
|
return this._context.createOscillator();
|
|
|
|
}
|
|
|
|
createBufferSource(): AudioBufferSourceNode {
|
|
|
|
return this._context.createBufferSource();
|
|
|
|
}
|
|
|
|
createBiquadFilter(): BiquadFilterNode {
|
|
|
|
return this._context.createBiquadFilter();
|
|
|
|
}
|
2021-01-01 01:53:38 +00:00
|
|
|
createBuffer(
|
|
|
|
numberOfChannels: number,
|
|
|
|
length: number,
|
|
|
|
sampleRate: number
|
|
|
|
): AudioBuffer {
|
2019-04-12 14:37:47 +00:00
|
|
|
return this._context.createBuffer(numberOfChannels, length, sampleRate);
|
|
|
|
}
|
2021-01-01 01:53:38 +00:00
|
|
|
createChannelMerger(
|
|
|
|
numberOfInputs?: number | undefined
|
|
|
|
): ChannelMergerNode {
|
2019-04-12 14:37:47 +00:00
|
|
|
return this._context.createChannelMerger(numberOfInputs);
|
|
|
|
}
|
2021-01-01 01:53:38 +00:00
|
|
|
createChannelSplitter(
|
|
|
|
numberOfOutputs?: number | undefined
|
|
|
|
): ChannelSplitterNode {
|
2019-04-12 14:37:47 +00:00
|
|
|
return this._context.createChannelSplitter(numberOfOutputs);
|
|
|
|
}
|
|
|
|
createConstantSource(): ConstantSourceNode {
|
|
|
|
return this._context.createConstantSource();
|
|
|
|
}
|
|
|
|
createConvolver(): ConvolverNode {
|
|
|
|
return this._context.createConvolver();
|
|
|
|
}
|
|
|
|
createDelay(maxDelayTime?: number | undefined): DelayNode {
|
|
|
|
return this._context.createDelay(maxDelayTime);
|
|
|
|
}
|
|
|
|
createDynamicsCompressor(): DynamicsCompressorNode {
|
|
|
|
return this._context.createDynamicsCompressor();
|
|
|
|
}
|
|
|
|
createGain(): GainNode {
|
|
|
|
return this._context.createGain();
|
|
|
|
}
|
2021-01-01 01:53:38 +00:00
|
|
|
createIIRFilter(
|
|
|
|
feedForward: number[] | Float32Array,
|
|
|
|
feedback: number[] | Float32Array
|
|
|
|
): IIRFilterNode {
|
2019-09-23 14:00:43 +00:00
|
|
|
// @ts-ignore
|
2019-09-04 02:08:20 +00:00
|
|
|
return this._context.createIIRFilter(feedForward, feedback);
|
2019-04-12 14:37:47 +00:00
|
|
|
}
|
|
|
|
createPanner(): PannerNode {
|
|
|
|
return this._context.createPanner();
|
|
|
|
}
|
|
|
|
createPeriodicWave(
|
|
|
|
real: number[] | Float32Array,
|
|
|
|
imag: number[] | Float32Array,
|
2021-01-01 01:53:38 +00:00
|
|
|
constraints?: PeriodicWaveConstraints | undefined
|
2019-04-12 14:37:47 +00:00
|
|
|
): PeriodicWave {
|
|
|
|
return this._context.createPeriodicWave(real, imag, constraints);
|
|
|
|
}
|
|
|
|
createStereoPanner(): StereoPannerNode {
|
|
|
|
return this._context.createStereoPanner();
|
|
|
|
}
|
|
|
|
createWaveShaper(): WaveShaperNode {
|
|
|
|
return this._context.createWaveShaper();
|
|
|
|
}
|
2019-09-20 22:50:22 +00:00
|
|
|
createMediaStreamSource(stream: MediaStream): MediaStreamAudioSourceNode {
|
2021-01-01 01:53:38 +00:00
|
|
|
assert(
|
|
|
|
isAudioContext(this._context),
|
|
|
|
"Not available if OfflineAudioContext"
|
|
|
|
);
|
2020-04-15 01:06:21 +00:00
|
|
|
const context = this._context as AudioContext;
|
|
|
|
return context.createMediaStreamSource(stream);
|
|
|
|
}
|
2021-01-01 01:53:38 +00:00
|
|
|
createMediaElementSource(
|
|
|
|
element: HTMLMediaElement
|
|
|
|
): MediaElementAudioSourceNode {
|
|
|
|
assert(
|
|
|
|
isAudioContext(this._context),
|
|
|
|
"Not available if OfflineAudioContext"
|
|
|
|
);
|
2020-10-02 14:28:57 +00:00
|
|
|
const context = this._context as AudioContext;
|
|
|
|
return context.createMediaElementSource(element);
|
|
|
|
}
|
2020-04-15 01:06:21 +00:00
|
|
|
createMediaStreamDestination(): MediaStreamAudioDestinationNode {
|
2021-01-01 01:53:38 +00:00
|
|
|
assert(
|
|
|
|
isAudioContext(this._context),
|
|
|
|
"Not available if OfflineAudioContext"
|
|
|
|
);
|
2020-04-15 01:06:21 +00:00
|
|
|
const context = this._context as AudioContext;
|
|
|
|
return context.createMediaStreamDestination();
|
2019-09-20 22:50:22 +00:00
|
|
|
}
|
2019-04-12 14:37:47 +00:00
|
|
|
decodeAudioData(audioData: ArrayBuffer): Promise<AudioBuffer> {
|
|
|
|
return this._context.decodeAudioData(audioData);
|
|
|
|
}
|
2019-06-23 18:59:27 +00:00
|
|
|
|
2019-04-12 14:37:47 +00:00
|
|
|
/**
|
2019-09-14 20:39:18 +00:00
|
|
|
* The current time in seconds of the AudioContext.
|
2019-04-12 14:37:47 +00:00
|
|
|
*/
|
|
|
|
get currentTime(): Seconds {
|
|
|
|
return this._context.currentTime;
|
|
|
|
}
|
|
|
|
/**
|
2019-09-14 20:39:18 +00:00
|
|
|
* The current time in seconds of the AudioContext.
|
2019-04-12 14:37:47 +00:00
|
|
|
*/
|
|
|
|
get state(): AudioContextState {
|
|
|
|
return this._context.state;
|
|
|
|
}
|
|
|
|
/**
|
2019-09-14 20:39:18 +00:00
|
|
|
* The current time in seconds of the AudioContext.
|
2019-04-12 14:37:47 +00:00
|
|
|
*/
|
|
|
|
get sampleRate(): number {
|
|
|
|
return this._context.sampleRate;
|
|
|
|
}
|
2019-12-22 02:01:45 +00:00
|
|
|
|
2019-04-12 14:37:47 +00:00
|
|
|
/**
|
2019-09-14 20:39:18 +00:00
|
|
|
* The listener
|
2019-04-12 14:37:47 +00:00
|
|
|
*/
|
2019-12-22 02:01:45 +00:00
|
|
|
get listener(): Listener {
|
|
|
|
this.initialize();
|
|
|
|
return this._listener;
|
|
|
|
}
|
|
|
|
set listener(l) {
|
2021-01-01 01:53:38 +00:00
|
|
|
assert(
|
|
|
|
!this._initialized,
|
|
|
|
"The listener cannot be set after initialization."
|
|
|
|
);
|
2019-12-22 02:01:45 +00:00
|
|
|
this._listener = l;
|
2019-04-12 14:37:47 +00:00
|
|
|
}
|
|
|
|
|
2019-06-23 18:59:27 +00:00
|
|
|
/**
|
2019-09-14 20:39:18 +00:00
|
|
|
* There is only one Transport per Context. It is created on initialization.
|
2019-06-23 18:59:27 +00:00
|
|
|
*/
|
|
|
|
get transport(): Transport {
|
2019-10-03 21:33:39 +00:00
|
|
|
this.initialize();
|
2019-06-23 18:59:27 +00:00
|
|
|
return this._transport;
|
|
|
|
}
|
|
|
|
set transport(t: Transport) {
|
2021-01-01 01:53:38 +00:00
|
|
|
assert(
|
|
|
|
!this._initialized,
|
|
|
|
"The transport cannot be set after initialization."
|
|
|
|
);
|
2019-06-23 18:59:27 +00:00
|
|
|
this._transport = t;
|
|
|
|
}
|
|
|
|
|
2019-10-28 16:12:27 +00:00
|
|
|
/**
|
|
|
|
* This is the Draw object for the context which is useful for synchronizing the draw frame with the Tone.js clock.
|
|
|
|
*/
|
|
|
|
get draw(): Draw {
|
|
|
|
this.initialize();
|
|
|
|
return this._draw;
|
|
|
|
}
|
|
|
|
set draw(d) {
|
2019-12-16 20:58:31 +00:00
|
|
|
assert(!this._initialized, "Draw cannot be set after initialization.");
|
2019-10-28 16:12:27 +00:00
|
|
|
this._draw = d;
|
|
|
|
}
|
|
|
|
|
2019-06-23 18:59:27 +00:00
|
|
|
/**
|
2019-09-14 20:39:18 +00:00
|
|
|
* A reference to the Context's destination node.
|
2019-06-23 18:59:27 +00:00
|
|
|
*/
|
|
|
|
get destination(): Destination {
|
2019-10-03 21:33:39 +00:00
|
|
|
this.initialize();
|
2019-06-23 18:59:27 +00:00
|
|
|
return this._destination;
|
|
|
|
}
|
|
|
|
set destination(d: Destination) {
|
2021-01-01 01:53:38 +00:00
|
|
|
assert(
|
|
|
|
!this._initialized,
|
|
|
|
"The destination cannot be set after initialization."
|
|
|
|
);
|
2019-06-23 18:59:27 +00:00
|
|
|
this._destination = d;
|
|
|
|
}
|
|
|
|
|
2019-09-27 21:52:37 +00:00
|
|
|
//--------------------------------------------
|
|
|
|
// AUDIO WORKLET
|
|
|
|
//--------------------------------------------
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Maps a module name to promise of the addModule method
|
|
|
|
*/
|
2022-03-21 15:34:23 +00:00
|
|
|
private _workletPromise: null | Promise<void> = null;
|
2019-09-27 21:52:37 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Create an audio worklet node from a name and options. The module
|
2024-04-29 14:48:37 +00:00
|
|
|
* must first be loaded using {@link addAudioWorkletModule}.
|
2019-09-27 21:52:37 +00:00
|
|
|
*/
|
|
|
|
createAudioWorkletNode(
|
2020-04-15 01:06:21 +00:00
|
|
|
name: string,
|
2019-09-27 21:52:37 +00:00
|
|
|
options?: Partial<AudioWorkletNodeOptions>
|
|
|
|
): AudioWorkletNode {
|
|
|
|
return createAudioWorkletNode(this.rawContext, name, options);
|
|
|
|
}
|
2020-04-15 01:06:21 +00:00
|
|
|
|
2019-09-27 21:52:37 +00:00
|
|
|
/**
|
|
|
|
* Add an AudioWorkletProcessor module
|
|
|
|
* @param url The url of the module
|
|
|
|
*/
|
2022-03-21 15:34:23 +00:00
|
|
|
async addAudioWorkletModule(url: string): Promise<void> {
|
2021-01-01 01:53:38 +00:00
|
|
|
assert(
|
|
|
|
isDefined(this.rawContext.audioWorklet),
|
|
|
|
"AudioWorkletNode is only available in a secure context (https or localhost)"
|
|
|
|
);
|
2022-03-21 15:34:23 +00:00
|
|
|
if (!this._workletPromise) {
|
|
|
|
this._workletPromise = this.rawContext.audioWorklet.addModule(url);
|
2019-09-27 21:52:37 +00:00
|
|
|
}
|
2022-03-21 15:34:23 +00:00
|
|
|
await this._workletPromise;
|
2019-09-27 21:52:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns a promise which resolves when all of the worklets have been loaded on this context
|
|
|
|
*/
|
|
|
|
protected async workletsAreReady(): Promise<void> {
|
2024-05-03 15:09:28 +00:00
|
|
|
(await this._workletPromise) ? this._workletPromise : Promise.resolve();
|
2019-09-27 21:52:37 +00:00
|
|
|
}
|
|
|
|
|
2019-09-14 21:47:07 +00:00
|
|
|
//---------------------------
|
2019-04-12 14:37:47 +00:00
|
|
|
// TICKER
|
2019-09-14 21:47:07 +00:00
|
|
|
//---------------------------
|
2019-04-12 14:37:47 +00:00
|
|
|
|
|
|
|
/**
|
2019-09-14 20:39:18 +00:00
|
|
|
* How often the interval callback is invoked.
|
|
|
|
* This number corresponds to how responsive the scheduling
|
2021-11-29 16:14:14 +00:00
|
|
|
* can be. Setting to 0 will result in the lowest practial interval
|
|
|
|
* based on context properties. context.updateInterval + context.lookAhead
|
|
|
|
* gives you the total latency between scheduling an event and hearing it.
|
2019-04-12 14:37:47 +00:00
|
|
|
*/
|
|
|
|
get updateInterval(): Seconds {
|
|
|
|
return this._ticker.updateInterval;
|
|
|
|
}
|
|
|
|
set updateInterval(interval: Seconds) {
|
|
|
|
this._ticker.updateInterval = interval;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2019-09-14 20:39:18 +00:00
|
|
|
* What the source of the clock is, either "worker" (default),
|
|
|
|
* "timeout", or "offline" (none).
|
2019-04-12 14:37:47 +00:00
|
|
|
*/
|
|
|
|
get clockSource(): TickerClockSource {
|
|
|
|
return this._ticker.type;
|
|
|
|
}
|
|
|
|
set clockSource(type: TickerClockSource) {
|
|
|
|
this._ticker.type = type;
|
|
|
|
}
|
|
|
|
|
2021-11-29 16:14:14 +00:00
|
|
|
/**
|
|
|
|
* The amount of time into the future events are scheduled. Giving Web Audio
|
|
|
|
* a short amount of time into the future to schedule events can reduce clicks and
|
|
|
|
* improve performance. This value can be set to 0 to get the lowest latency.
|
2024-04-29 14:48:37 +00:00
|
|
|
* Adjusting this value also affects the {@link updateInterval}.
|
2021-11-29 16:14:14 +00:00
|
|
|
*/
|
|
|
|
get lookAhead(): Seconds {
|
|
|
|
return this._lookAhead;
|
|
|
|
}
|
|
|
|
set lookAhead(time: Seconds) {
|
|
|
|
this._lookAhead = time;
|
2021-12-01 22:44:53 +00:00
|
|
|
// if lookAhead is 0, default to .01 updateInterval
|
2024-05-03 15:09:28 +00:00
|
|
|
this.updateInterval = time ? time / 2 : 0.01;
|
|
|
|
}
|
2021-11-29 16:14:14 +00:00
|
|
|
private _lookAhead!: Seconds;
|
|
|
|
|
2019-04-12 14:37:47 +00:00
|
|
|
/**
|
2019-05-23 18:00:49 +00:00
|
|
|
* The type of playback, which affects tradeoffs between audio
|
|
|
|
* output latency and responsiveness.
|
|
|
|
* In addition to setting the value in seconds, the latencyHint also
|
|
|
|
* accepts the strings "interactive" (prioritizes low latency),
|
|
|
|
* "playback" (prioritizes sustained playback), "balanced" (balances
|
2020-05-27 01:09:32 +00:00
|
|
|
* latency and performance).
|
2019-05-23 18:00:49 +00:00
|
|
|
* @example
|
2020-05-12 16:31:17 +00:00
|
|
|
* // prioritize sustained playback
|
|
|
|
* const context = new Tone.Context({ latencyHint: "playback" });
|
|
|
|
* // set this context as the global Context
|
|
|
|
* Tone.setContext(context);
|
2020-08-04 01:21:17 +00:00
|
|
|
* // the global context is gettable with Tone.getContext()
|
|
|
|
* console.log(Tone.getContext().latencyHint);
|
2019-04-12 14:37:47 +00:00
|
|
|
*/
|
|
|
|
get latencyHint(): ContextLatencyHint | Seconds {
|
|
|
|
return this._latencyHint;
|
|
|
|
}
|
2020-05-12 16:31:17 +00:00
|
|
|
|
2019-04-12 14:37:47 +00:00
|
|
|
/**
|
2020-05-12 16:31:17 +00:00
|
|
|
* The unwrapped AudioContext or OfflineAudioContext
|
2019-04-12 14:37:47 +00:00
|
|
|
*/
|
2019-08-16 16:49:04 +00:00
|
|
|
get rawContext(): AnyAudioContext {
|
2019-04-12 14:37:47 +00:00
|
|
|
return this._context;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2024-04-29 14:48:37 +00:00
|
|
|
* The current audio context time plus a short {@link lookAhead}.
|
2021-10-13 17:11:41 +00:00
|
|
|
* @example
|
|
|
|
* setInterval(() => {
|
|
|
|
* console.log("now", Tone.now());
|
|
|
|
* }, 100);
|
2019-04-12 14:37:47 +00:00
|
|
|
*/
|
|
|
|
now(): Seconds {
|
2021-11-29 16:14:14 +00:00
|
|
|
return this._context.currentTime + this._lookAhead;
|
2019-04-12 14:37:47 +00:00
|
|
|
}
|
|
|
|
|
2019-10-29 18:29:52 +00:00
|
|
|
/**
|
2024-04-29 14:48:37 +00:00
|
|
|
* The current audio context time without the {@link lookAhead}.
|
|
|
|
* In most cases it is better to use {@link now} instead of {@link immediate} since
|
|
|
|
* with {@link now} the {@link lookAhead} is applied equally to _all_ components including internal components,
|
|
|
|
* to making sure that everything is scheduled in sync. Mixing {@link now} and {@link immediate}
|
|
|
|
* can cause some timing issues. If no lookAhead is desired, you can set the {@link lookAhead} to `0`.
|
2019-10-29 18:29:52 +00:00
|
|
|
*/
|
|
|
|
immediate(): Seconds {
|
|
|
|
return this._context.currentTime;
|
|
|
|
}
|
|
|
|
|
2019-04-12 14:37:47 +00:00
|
|
|
/**
|
2019-09-14 20:39:18 +00:00
|
|
|
* Starts the audio context from a suspended state. This is required
|
2024-05-03 15:09:28 +00:00
|
|
|
* to initially start the AudioContext.
|
2024-04-29 16:59:49 +00:00
|
|
|
* @see {@link start}
|
2019-04-12 14:37:47 +00:00
|
|
|
*/
|
|
|
|
resume(): Promise<void> {
|
2020-10-23 20:43:17 +00:00
|
|
|
if (isAudioContext(this._context)) {
|
2019-04-12 14:37:47 +00:00
|
|
|
return this._context.resume();
|
|
|
|
} else {
|
|
|
|
return Promise.resolve();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2020-05-19 01:13:22 +00:00
|
|
|
* Close the context. Once closed, the context can no longer be used and
|
2021-01-01 01:53:38 +00:00
|
|
|
* any AudioNodes created from the context will be silent.
|
2019-04-12 14:37:47 +00:00
|
|
|
*/
|
2019-08-16 16:49:04 +00:00
|
|
|
async close(): Promise<void> {
|
2024-05-03 15:09:28 +00:00
|
|
|
if (
|
|
|
|
isAudioContext(this._context) &&
|
|
|
|
this.state !== "closed" &&
|
|
|
|
!this._closeStarted
|
|
|
|
) {
|
2022-03-01 17:38:26 +00:00
|
|
|
this._closeStarted = true;
|
2019-04-12 14:37:47 +00:00
|
|
|
await this._context.close();
|
|
|
|
}
|
2019-07-16 19:41:59 +00:00
|
|
|
if (this._initialized) {
|
|
|
|
closeContext(this);
|
|
|
|
}
|
2019-04-12 14:37:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2021-01-01 01:53:38 +00:00
|
|
|
* **Internal** Generate a looped buffer at some constant value.
|
2019-04-12 14:37:47 +00:00
|
|
|
*/
|
|
|
|
getConstant(val: number): AudioBufferSourceNode {
|
|
|
|
if (this._constants.has(val)) {
|
|
|
|
return this._constants.get(val) as AudioBufferSourceNode;
|
|
|
|
} else {
|
2021-01-01 01:53:38 +00:00
|
|
|
const buffer = this._context.createBuffer(
|
|
|
|
1,
|
|
|
|
128,
|
|
|
|
this._context.sampleRate
|
|
|
|
);
|
2019-04-12 14:37:47 +00:00
|
|
|
const arr = buffer.getChannelData(0);
|
|
|
|
for (let i = 0; i < arr.length; i++) {
|
|
|
|
arr[i] = val;
|
|
|
|
}
|
|
|
|
const constant = this._context.createBufferSource();
|
|
|
|
constant.channelCount = 1;
|
|
|
|
constant.channelCountMode = "explicit";
|
|
|
|
constant.buffer = buffer;
|
|
|
|
constant.loop = true;
|
|
|
|
constant.start(0);
|
|
|
|
this._constants.set(val, constant);
|
|
|
|
return constant;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2019-09-14 20:39:18 +00:00
|
|
|
* Clean up. Also closes the audio context.
|
2019-04-12 14:37:47 +00:00
|
|
|
*/
|
2019-05-23 18:00:49 +00:00
|
|
|
dispose(): this {
|
2019-07-23 16:11:57 +00:00
|
|
|
super.dispose();
|
2019-04-12 14:37:47 +00:00
|
|
|
this._ticker.dispose();
|
|
|
|
this._timeouts.dispose();
|
2021-01-01 01:53:38 +00:00
|
|
|
Object.keys(this._constants).map((val) =>
|
|
|
|
this._constants[val].disconnect()
|
|
|
|
);
|
2022-02-22 22:22:50 +00:00
|
|
|
this.close();
|
2019-04-12 14:37:47 +00:00
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
2019-09-14 21:47:07 +00:00
|
|
|
//---------------------------
|
2019-04-12 14:37:47 +00:00
|
|
|
// TIMEOUTS
|
2019-09-14 21:47:07 +00:00
|
|
|
//---------------------------
|
2019-04-12 14:37:47 +00:00
|
|
|
|
|
|
|
/**
|
2019-09-14 20:39:18 +00:00
|
|
|
* The private loop which keeps track of the context scheduled timeouts
|
|
|
|
* Is invoked from the clock source
|
2019-04-12 14:37:47 +00:00
|
|
|
*/
|
2019-09-14 21:47:07 +00:00
|
|
|
private _timeoutLoop(): void {
|
2019-04-12 14:37:47 +00:00
|
|
|
const now = this.now();
|
|
|
|
let firstEvent = this._timeouts.peek();
|
|
|
|
while (this._timeouts.length && firstEvent && firstEvent.time <= now) {
|
|
|
|
// invoke the callback
|
|
|
|
firstEvent.callback();
|
|
|
|
// shift the first event off
|
|
|
|
this._timeouts.shift();
|
2019-05-23 18:00:49 +00:00
|
|
|
// get the next one
|
|
|
|
firstEvent = this._timeouts.peek();
|
2019-04-12 14:37:47 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2019-09-14 20:39:18 +00:00
|
|
|
* A setTimeout which is guaranteed by the clock source.
|
|
|
|
* Also runs in the offline context.
|
2019-08-30 16:06:38 +00:00
|
|
|
* @param fn The callback to invoke
|
|
|
|
* @param timeout The timeout in seconds
|
|
|
|
* @returns ID to use when invoking Context.clearTimeout
|
2019-04-12 14:37:47 +00:00
|
|
|
*/
|
|
|
|
setTimeout(fn: (...args: any[]) => void, timeout: Seconds): number {
|
|
|
|
this._timeoutIds++;
|
|
|
|
const now = this.now();
|
|
|
|
this._timeouts.add({
|
2019-09-14 21:47:07 +00:00
|
|
|
callback: fn,
|
|
|
|
id: this._timeoutIds,
|
|
|
|
time: now + timeout,
|
2019-04-12 14:37:47 +00:00
|
|
|
});
|
|
|
|
return this._timeoutIds;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2019-09-14 20:39:18 +00:00
|
|
|
* Clears a previously scheduled timeout with Tone.context.setTimeout
|
2019-08-30 16:06:38 +00:00
|
|
|
* @param id The ID returned from setTimeout
|
2019-04-12 14:37:47 +00:00
|
|
|
*/
|
2019-09-06 18:46:44 +00:00
|
|
|
clearTimeout(id: number): this {
|
2021-01-01 01:53:38 +00:00
|
|
|
this._timeouts.forEach((event) => {
|
2019-04-12 14:37:47 +00:00
|
|
|
if (event.id === id) {
|
|
|
|
this._timeouts.remove(event);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
return this;
|
|
|
|
}
|
2019-09-06 18:46:44 +00:00
|
|
|
|
|
|
|
/**
|
2024-04-29 14:48:37 +00:00
|
|
|
* Clear the function scheduled by {@link setInterval}
|
2019-09-06 18:46:44 +00:00
|
|
|
*/
|
|
|
|
clearInterval(id: number): this {
|
|
|
|
return this.clearTimeout(id);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Adds a repeating event to the context's callback clock
|
|
|
|
*/
|
|
|
|
setInterval(fn: (...args: any[]) => void, interval: Seconds): number {
|
|
|
|
const id = ++this._timeoutIds;
|
|
|
|
const intervalFn = () => {
|
|
|
|
const now = this.now();
|
|
|
|
this._timeouts.add({
|
2019-09-14 21:47:07 +00:00
|
|
|
callback: () => {
|
2019-09-06 18:46:44 +00:00
|
|
|
// invoke the callback
|
|
|
|
fn();
|
|
|
|
// invoke the event to repeat it
|
|
|
|
intervalFn();
|
|
|
|
},
|
|
|
|
id,
|
2019-09-14 21:47:07 +00:00
|
|
|
time: now + interval,
|
2019-09-06 18:46:44 +00:00
|
|
|
});
|
|
|
|
};
|
|
|
|
// kick it off
|
|
|
|
intervalFn();
|
|
|
|
return id;
|
|
|
|
}
|
2019-04-12 14:37:47 +00:00
|
|
|
}
|