import { Ticker, TickerClockSource } from "../clock/Ticker"; import { Transport } from "../clock/Transport"; import { optionsFromArguments } from "../util/Defaults"; import { Emitter } from "../util/Emitter"; import { Omit } from "../util/Interface"; import { Timeline } from "../util/Timeline"; import { isString } from "../util/TypeCheck"; import { getAudioContext } from "./AudioContext"; import { Destination } from "./Destination"; export type ContextLatencyHint = AudioContextLatencyCategory | "fastest"; // these are either not used in Tone.js or deprecated and not implemented. export type ExcludedFromBaseAudioContext = "createScriptProcessor" | "onstatechange" | "addEventListener" | "removeEventListener" | "listener" | "dispatchEvent" | "audioWorklet" | "destination"; // the subset of the BaseAudioContext which Tone.Context implements. export type BaseAudioContextSubset = Omit; export interface ContextOptions { clockSource: TickerClockSource; latencyHint: ContextLatencyHint; lookAhead: Seconds; updateInterval: Seconds; context: BaseAudioContext; } export interface ContextTimeoutEvent { callback: (...args: any[]) => void; id: number; time: Seconds; } /** * Wrapper around the native AudioContext. */ export class Context extends Emitter<"statechange" | "tick"> implements BaseAudioContextSubset { name = "Context"; /** * The amount of time into the future events are scheduled */ lookAhead: Seconds; /** * private reference to the BaseAudioContext */ protected readonly _context: BaseAudioContext; /** * A reliable callback method */ private readonly _ticker: Ticker; /** * The default latency hint */ private _latencyHint: ContextLatencyHint | Seconds; /** * An object containing all of the constants AudioBufferSourceNodes */ private _constants = new Map(); /** * All of the setTimeout events. */ private _timeouts: Timeline = new Timeline(); /** * The timeout id counter */ private _timeoutIds = 0; /** * A reference the Transport singleton belonging to this context */ private _transport!: Transport; /** * A reference the Destination singleton belonging to this context */ private _destination!: Destination; /** * Private indicator if the context has been initialized */ private _initialized: boolean = false; constructor(context?: BaseAudioContext); // tslint:disable-next-line: unified-signatures constructor(options?: Partial); constructor() { super(); const options = optionsFromArguments(Context.getDefaults(), arguments, ["context"]); this._context = options.context; this._latencyHint = options.latencyHint; this.lookAhead = options.lookAhead; this._ticker = new Ticker(this.emit.bind(this, "tick"), options.clockSource, options.updateInterval); this.on("tick", this._timeoutLoop.bind(this)); // fwd events from the context this._context.addEventListener("statechange", () => { this.emit("statechange", this.state); }); } static getDefaults(): ContextOptions { return { clockSource: "worker", context: getAudioContext(), latencyHint: "interactive", lookAhead: 0.1, updateInterval: 0.03, }; } /** * Finish setting up the context. **You usually do not need to do this manually.** */ initialize(): this { if (!this._initialized) { // add any additional modules Context._notifyNewContext.forEach(cb => cb(this)); this._initialized = true; } return this; } /////////////////////////////////////////////////////////////////////// // BASE AUDIO CONTEXT METHODS /////////////////////////////////////////////////////////////////////// createAnalyser(): AnalyserNode { return this._context.createAnalyser(); } createOscillator(): OscillatorNode { return this._context.createOscillator(); } createBufferSource(): AudioBufferSourceNode { return this._context.createBufferSource(); } createBiquadFilter(): BiquadFilterNode { return this._context.createBiquadFilter(); } createBuffer(numberOfChannels: number, length: number, sampleRate: number): AudioBuffer { return this._context.createBuffer(numberOfChannels, length, sampleRate); } createChannelMerger(numberOfInputs?: number | undefined): ChannelMergerNode { return this._context.createChannelMerger(numberOfInputs); } createChannelSplitter(numberOfOutputs?: number | undefined): ChannelSplitterNode { 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(); } createIIRFilter(feedforward: number[], feedback: number[]): IIRFilterNode { return this._context.createIIRFilter(feedforward, feedback); } createPanner(): PannerNode { return this._context.createPanner(); } createPeriodicWave( real: number[] | Float32Array, imag: number[] | Float32Array, constraints?: PeriodicWaveConstraints | undefined, ): PeriodicWave { return this._context.createPeriodicWave(real, imag, constraints); } createStereoPanner(): StereoPannerNode { return this._context.createStereoPanner(); } createWaveShaper(): WaveShaperNode { return this._context.createWaveShaper(); } decodeAudioData(audioData: ArrayBuffer): Promise { return this._context.decodeAudioData(audioData); } /** * The current time in seconds of the AudioContext. */ get currentTime(): Seconds { return this._context.currentTime; } /** * The current time in seconds of the AudioContext. */ get state(): AudioContextState { return this._context.state; } /** * The current time in seconds of the AudioContext. */ get sampleRate(): number { return this._context.sampleRate; } /** * The listener */ get listener(): AudioListener { return this._context.listener; } /** * There is only one Transport per Context. It is created on initialization. */ get transport(): Transport { this.assert(this._initialized, "The context must be initialized before being used by invoking context.initialize()"); return this._transport; } set transport(t: Transport) { this.assert(!this._initialized, "The transport cannot be set after initialization."); this._transport = t; } /** * A reference to the Context's destination node. */ get destination(): Destination { this.assert(this._initialized, "The context must be initialized before being used by invoking context.initialize()"); return this._destination; } set destination(d: Destination) { this.assert(!this._initialized, "The transport cannot be set after initialization."); this._destination = d; } /////////////////////////////////////////////////////////////////////// // TICKER /////////////////////////////////////////////////////////////////////// /** * How often the interval callback is invoked. * This number corresponds to how responsive the scheduling * can be. context.updateInterval + context.lookAhead gives you the * total latency between scheduling an event and hearing it. */ get updateInterval(): Seconds { return this._ticker.updateInterval; } set updateInterval(interval: Seconds) { this._ticker.updateInterval = interval; } /** * What the source of the clock is, either "worker" (default), * "timeout", or "offline" (none). */ get clockSource(): TickerClockSource { return this._ticker.type; } set clockSource(type: TickerClockSource) { this._ticker.type = type; } /** * 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 * latency and performance), and "fastest" (lowest latency, might glitch more often). * @example * //set the lookAhead to 0.3 seconds * Tone.context.latencyHint = 0.3; */ get latencyHint(): ContextLatencyHint | Seconds { return this._latencyHint; } set latencyHint(hint: ContextLatencyHint | Seconds) { let lookAheadValue = 0; this._latencyHint = hint; if (isString(hint)) { switch (hint) { case "interactive": lookAheadValue = 0.1; break; case "playback": lookAheadValue = 0.8; break; case "balanced": lookAheadValue = 0.25; break; case "fastest": lookAheadValue = 0.01; break; } } this.lookAhead = lookAheadValue; this.updateInterval = lookAheadValue / 3; } /** * The unwrapped AudioContext. */ get rawContext(): BaseAudioContext { return this._context; } /** * The current audio context time */ now(): Seconds { return this._context.currentTime + this.lookAhead; } /** * Starts the audio context from a suspended state. This is required * to initially start the AudioContext. */ resume(): Promise { if (this._context.state === "suspended" && this._context instanceof AudioContext) { return this._context.resume(); } else { return Promise.resolve(); } } /** * Promise which is invoked when the context is running. * Tries to resume the context if it's not started. */ async close(): Promise { if (this._context instanceof AudioContext) { await this._context.close(); } return this; } /** * Generate a looped buffer at some constant value. */ getConstant(val: number): AudioBufferSourceNode { if (this._constants.has(val)) { return this._constants.get(val) as AudioBufferSourceNode; } else { const buffer = this._context.createBuffer(1, 128, this._context.sampleRate); 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; } } /** * Clean up. Also closes the audio context. */ dispose(): this { this._ticker.dispose(); this._timeouts.dispose(); Object.keys(this._constants).map(val => this._constants[val].disconnect()); return this; } /////////////////////////////////////////////////////////////////////// // TIMEOUTS /////////////////////////////////////////////////////////////////////// /** * The private loop which keeps track of the context scheduled timeouts * Is invoked from the clock source */ private _timeoutLoop(): void { 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(); // get the next one firstEvent = this._timeouts.peek(); } } /** * A setTimeout which is guarented by the clock source. * Also runs in the offline context. * @param fn The callback to invoke * @param timeout The timeout in seconds * @returns ID to use when invoking Context.clearTimeout */ setTimeout(fn: (...args: any[]) => void, timeout: Seconds): number { this._timeoutIds++; const now = this.now(); this._timeouts.add({ callback : fn, id : this._timeoutIds, time : now + timeout, }); return this._timeoutIds; } /** * Clears a previously scheduled timeout with Tone.context.setTimeout * @param id The ID returned from setTimeout */ clearTimeout(id: number): Context { this._timeouts.forEach(event => { if (event.id === id) { this._timeouts.remove(event); } }); return this; } /////////////////////////////////////////////////////////////////////////// // INITIALIZING NEW CONTEXT /////////////////////////////////////////////////////////////////////////// /** * Array of callbacks to invoke when a new context is created */ private static _notifyNewContext: Array<(ctx: Context) => void> = []; /** * Used internally to setup a new Context */ static onInit(cb: (ctx: Context) => void): void { Context._notifyNewContext.push(cb); } }