import { ToneAudioBuffer } from "../../core/context/ToneAudioBuffer"; import { Positive, Time } from "../../core/type/Units"; import { defaultArg, optionsFromArguments } from "../../core/util/Defaults"; import { noOp } from "../../core/util/Interface"; import { isUndef } from "../../core/util/TypeCheck"; import { Source, SourceOptions } from "../Source"; import { ToneBufferSource } from "./ToneBufferSource"; export interface PlayerOptions extends SourceOptions { onload: () => void; playbackRate: Positive; loop: boolean; autostart: boolean; loopStart: Time; loopEnd: Time; reverse: boolean; fadeIn: Time; fadeOut: Time; url?: ToneAudioBuffer | string | AudioBuffer; } /** * Player is an audio file player with start, loop, and stop functions. * @example * var player = new Player("./path/to/sample.mp3").toDestination(); * //play as soon as the buffer is loaded * player.autostart = true; */ export class Player extends Source { readonly name: string = "Player"; /** * If the file should play as soon * as the buffer is loaded. * @example * //will play as soon as it's loaded * var player = new Player({ * "url" : "./path/to/sample.mp3", * "autostart" : true, * }).toDestination(); */ autostart: boolean; /** * The buffer */ private _buffer: ToneAudioBuffer; /** * if the buffer should loop once it's over */ private _loop: boolean; /** * if 'loop' is true, the loop will start at this position */ private _loopStart: Time; /** * if 'loop' is true, the loop will end at this position */ private _loopEnd: Time; /** * the playback rate */ private _playbackRate: Positive; /** * All of the active buffer source nodes */ private _activeSources: Set = new Set(); /** * The fadeIn time of the amplitude envelope. */ fadeIn: Time; /** * The fadeOut time of the amplitude envelope. */ fadeOut: Time; /** * @param url Either the AudioBuffer or the url from which to load the AudioBuffer * @param onload The function to invoke when the buffer is loaded. */ constructor(url?: string | AudioBuffer | ToneAudioBuffer, onload?: () => void); constructor(options?: Partial); constructor() { super(optionsFromArguments(Player.getDefaults(), arguments, ["url", "onload"])); const options = optionsFromArguments(Player.getDefaults(), arguments, ["url", "onload"]); this._buffer = new ToneAudioBuffer({ onload: this._onload.bind(this, options.onload), reverse: options.reverse, url: options.url, }); this.autostart = options.autostart; this._loop = options.loop; this._loopStart = options.loopStart; this._loopEnd = options.loopEnd; this._playbackRate = options.playbackRate; this.fadeIn = options.fadeIn; this.fadeOut = options.fadeOut; } static getDefaults(): PlayerOptions { return Object.assign(Source.getDefaults(), { autostart : false, fadeIn : 0, fadeOut : 0, loop : false, loopEnd : 0, loopStart : 0, onload : noOp, playbackRate : 1, reverse : false, }); } /** * Load the audio file as an audio buffer. * Decodes the audio asynchronously and invokes * the callback once the audio buffer loads. * Note: this does not need to be called if a url * was passed in to the constructor. Only use this * if you want to manually load a new url. * @param url The url of the buffer to load. Filetype support depends on the browser. */ async load(url: string): Promise { await this._buffer.load(url); this._onload(); return this; } /** * Internal callback when the buffer is loaded. */ private _onload(callback: () => void = noOp): void { callback(); if (this.autostart) { this.start(); } } /** * Internal callback when the buffer is done playing. */ private _onSourceEnd(source: ToneBufferSource): void { // invoke the onstop function this.onstop(this); // delete the source from the active sources this._activeSources.delete(source); if (this._activeSources.size === 0 && !this._synced) { this._state.setStateAtTime("stopped", this.now()); } } /** * Play the buffer at the given startTime. Optionally add an offset * and/or duration which will play the buffer from a position * within the buffer for the given duration. * * @param time When the player should start. * @param offset The offset from the beginning of the sample * to start at. * @param duration How long the sample should play. If no duration is given, it will default to the full length * of the sample (minus any offset) */ start(time?: Time, offset?: Time, duration?: Time): this { super.start(time, offset, duration); return this; } /** * Internal start method */ protected _start(startTime?: Time, offset?: Time, duration?: Time): void { // if it's a loop the default offset is the loopstart point if (this._loop) { offset = defaultArg(offset, this._loopStart); } else { // otherwise the default offset is 0 offset = defaultArg(offset, 0); } // compute the values in seconds let comptuedOffset = this.toSeconds(offset); // if it's synced, it should factor in the playback rate for computing the offset if (this._synced) { comptuedOffset *= this._playbackRate; } // compute the duration which is either the passed in duration of the buffer.duration - offset const origDuration = duration; duration = defaultArg(duration, Math.max(this._buffer.duration - comptuedOffset, 0)); let computedDuration = this.toSeconds(duration); // scale it by the playback rate computedDuration = computedDuration / this._playbackRate; // get the start time startTime = this.toSeconds(startTime); // make the source const source = new ToneBufferSource({ buffer : this._buffer, context: this.context, fadeIn : this.fadeIn, fadeOut : this.fadeOut, loop : this._loop, loopEnd : this._loopEnd, loopStart : this._loopStart, onended : this._onSourceEnd.bind(this), playbackRate : this._playbackRate, }).connect(this.output); // set the looping properties if (!this._loop && !this._synced) { // if it's not looping, set the state change at the end of the sample this._state.setStateAtTime("stopped", startTime + computedDuration, { implicitEnd: true, }); } // add it to the array of active sources this._activeSources.add(source); // start it if (this._loop && isUndef(origDuration)) { source.start(startTime, comptuedOffset); } else { // subtract the fade out time source.start(startTime, comptuedOffset, computedDuration - this.toSeconds(this.fadeOut)); } } /** * Stop playback. */ protected _stop(time?: Time): void { const computedTime = this.toSeconds(time); this._activeSources.forEach(source => source.stop(computedTime)); } /** * Stop and then restart the player from the beginning (or offset) * @param time When the player should start. * @param offset The offset from the beginning of the sample to start at. * @param duration How long the sample should play. If no duration is given, * it will default to the full length of the sample (minus any offset) */ restart(time?: Time, offset?: Time, duration?: Time): this { this._stop(time); this._start(time, offset, duration); return this; } /** * Seek to a specific time in the player's buffer. If the * source is no longer playing at that time, it will stop. * If you seek to a time that * @param {Time} offset The time to seek to. * @param {Time=} time The time for the seek event to occur. * @return {Player} this * @example * source.start(0.2); * source.stop(0.4); */ seek(offset: Time, when?: Time): this { const computedTime = this.toSeconds(when); if (this._state.getValueAtTime(computedTime) === "started") { const comptuedOffset = this.toSeconds(offset); // if it's currently playing, stop it this._stop(computedTime); // restart it at the given time this._start(computedTime, comptuedOffset); } return this; } /** * Set the loop start and end. Will only loop if loop is set to true. * @param loopStart The loop end time * @param loopEnd The loop end time * @example * //loop 0.1 seconds of the file. * player.setLoopPoints(0.2, 0.3); * player.loop = true; */ setLoopPoints(loopStart: Time, loopEnd: Time): this { this.loopStart = loopStart; this.loopEnd = loopEnd; return this; } /** * If loop is true, the loop will start at this position. */ get loopStart(): Time { return this._loopStart; } set loopStart(loopStart) { this._loopStart = loopStart; // get the current source this._activeSources.forEach(source => { source.loopStart = loopStart; }); } /** * If loop is true, the loop will end at this position. */ get loopEnd(): Time { return this._loopEnd; } set loopEnd(loopEnd) { this._loopEnd = loopEnd; // get the current source this._activeSources.forEach(source => { source.loopEnd = loopEnd; }); } /** * The audio buffer belonging to the player. */ get buffer(): ToneAudioBuffer { return this._buffer; } set buffer(buffer) { this._buffer.set(buffer); } /** * If the buffer should loop once it's over. */ get loop(): boolean { return this._loop; } set loop(loop) { // if no change, do nothing if (this._loop === loop) { return; } this._loop = loop; // set the loop of all of the sources this._activeSources.forEach(source => { source.loop = loop; }); if (loop) { // remove the next stopEvent const stopEvent = this._state.getNextState("stopped", this.now()); if (stopEvent) { this._state.cancel(stopEvent.time); } } } /** * The playback speed. 1 is normal speed. This is not a signal because * Safari and iOS currently don't support playbackRate as a signal. */ get playbackRate(): Positive { return this._playbackRate; } set playbackRate(rate) { this._playbackRate = rate; const now = this.now(); // cancel the stop event since it's at a different time now const stopEvent = this._state.getNextState("stopped", now); if (stopEvent && stopEvent.implicitEnd) { this._state.cancel(stopEvent.time); this._activeSources.forEach(source => source.cancelStop()); } // set all the sources this._activeSources.forEach(source => { source.playbackRate.setValueAtTime(rate, now); }); } /** * The direction the buffer should play in */ get reverse(): boolean { return this._buffer.reverse; } set reverse(rev) { this._buffer.reverse = rev; } /** * If the buffer is loaded */ get loaded(): boolean { return this._buffer.loaded; } dispose(): this { super.dispose(); // disconnect all of the players this._activeSources.forEach(source => source.dispose()); this._activeSources.clear(); this._buffer.dispose(); return this; } }