@twilio/voice-sdk
Version:
Twilio's JavaScript Voice SDK
262 lines (259 loc) • 17.8 kB
JavaScript
import { __awaiter } from 'tslib';
import Deferred from './deferred.js';
import EventTarget from './eventtarget.js';
/**
* An {@link AudioPlayer} is an HTMLAudioElement-like object that uses AudioContext
* to circumvent browser limitations.
* @private
*/
class AudioPlayer extends EventTarget {
get destination() { return this._destination; }
get loop() { return this._loop; }
set loop(shouldLoop) {
const self = this;
function pauseAfterPlaythrough() {
self._audioNode.removeEventListener('ended', pauseAfterPlaythrough);
self.pause();
}
// If a sound is already looping, it should continue playing
// the current playthrough and then stop.
if (!shouldLoop && this.loop && !this.paused) {
this._audioNode.addEventListener('ended', pauseAfterPlaythrough);
}
this._loop = shouldLoop;
}
/**
* Whether the audio element is muted.
*/
get muted() { return this._gainNode.gain.value === 0; }
set muted(shouldBeMuted) {
this._gainNode.gain.value = shouldBeMuted ? 0 : 1;
}
/**
* Whether the sound is paused. this._audioNode only exists when sound is playing;
* otherwise AudioPlayer is considered paused.
*/
get paused() { return this._audioNode === null; }
get src() { return this._src; }
set src(src) {
this._load(src);
}
/**
* The srcObject of the HTMLMediaElement
*/
get srcObject() {
return this._audioElement.srcObject;
}
set srcObject(srcObject) {
this._audioElement.srcObject = srcObject;
}
get sinkId() { return this._sinkId; }
/**
* @private
*/
constructor(audioContext, srcOrOptions = {}, options = {}) {
super();
/**
* The AudioBufferSourceNode of the actively loaded sound. Null if a sound
* has not been loaded yet. This is re-used for each time the sound is
* played.
*/
this._audioNode = null;
/**
* Whether or not the audio element should loop. If disabled during playback,
* playing continues until the sound ends and then stops looping.
*/
this._loop = false;
/**
* An Array of deferred-like objects for each pending `play` Promise. When
* .pause() is called or .src is set, all pending play Promises are
* immediately rejected.
*/
this._pendingPlayDeferreds = [];
/**
* The current sinkId of the device audio is being played through.
*/
this._sinkId = 'default';
/**
* The source URL of the sound to play. When set, the currently playing sound will stop.
*/
this._src = '';
if (typeof srcOrOptions !== 'string') {
options = srcOrOptions;
}
this._audioContext = audioContext;
this._audioElement = new (options.AudioFactory || Audio)();
this._bufferPromise = this._createPlayDeferred().promise;
this._destination = this._audioContext.destination;
this._gainNode = this._audioContext.createGain();
this._gainNode.connect(this._destination);
this._XMLHttpRequest = options.XMLHttpRequestFactory || XMLHttpRequest;
this.addEventListener('canplaythrough', () => {
this._resolvePlayDeferreds();
});
if (typeof srcOrOptions === 'string') {
this.src = srcOrOptions;
}
}
/**
* Stop any ongoing playback and reload the source file.
*/
load() {
this._load(this._src);
}
/**
* Pause the audio coming from this AudioPlayer. This will reject any pending
* play Promises.
*/
pause() {
if (this.paused) {
return;
}
this._audioElement.pause();
this._audioNode.stop();
this._audioNode.disconnect(this._gainNode);
this._audioNode = null;
this._rejectPlayDeferreds(new Error('The play() request was interrupted by a call to pause().'));
}
/**
* Play the sound. If the buffer hasn't loaded yet, wait for the buffer to load. If
* the source URL is not set yet, this Promise will remain pending until a source
* URL is set.
*/
play() {
return __awaiter(this, void 0, void 0, function* () {
if (!this.paused) {
yield this._bufferPromise;
if (!this.paused) {
return;
}
throw new Error('The play() request was interrupted by a call to pause().');
}
this._audioNode = this._audioContext.createBufferSource();
this._audioNode.loop = this.loop;
this._audioNode.addEventListener('ended', () => {
if (this._audioNode && this._audioNode.loop) {
return;
}
this.dispatchEvent('ended');
});
const buffer = yield this._bufferPromise;
if (this.paused) {
throw new Error('The play() request was interrupted by a call to pause().');
}
this._audioNode.buffer = buffer;
this._audioNode.connect(this._gainNode);
this._audioNode.start();
if (this._audioElement.srcObject) {
return this._audioElement.play();
}
});
}
/**
* Change which device the sound should play through.
* @param sinkId - The sink of the device to play sound through.
*/
setSinkId(sinkId) {
return __awaiter(this, void 0, void 0, function* () {
if (typeof this._audioElement.setSinkId !== 'function') {
throw new Error('This browser does not support setSinkId.');
}
if (sinkId === this.sinkId) {
return;
}
if (sinkId === 'default') {
if (!this.paused) {
this._gainNode.disconnect(this._destination);
}
this._audioElement.srcObject = null;
this._destination = this._audioContext.destination;
this._gainNode.connect(this._destination);
this._sinkId = sinkId;
return;
}
yield this._audioElement.setSinkId(sinkId);
if (this._audioElement.srcObject) {
return;
}
this._gainNode.disconnect(this._audioContext.destination);
this._destination = this._audioContext.createMediaStreamDestination();
this._audioElement.srcObject = this._destination.stream;
this._sinkId = sinkId;
this._gainNode.connect(this._destination);
});
}
/**
* Create a Deferred for a Promise that will be resolved when .src is set or rejected
* when .pause is called.
*/
_createPlayDeferred() {
const deferred = new Deferred();
this._pendingPlayDeferreds.push(deferred);
return deferred;
}
/**
* Stop current playback and load a sound file.
* @param src - The source URL of the file to load
*/
_load(src) {
if (this._src && this._src !== src) {
this.pause();
}
this._src = src;
this._bufferPromise = new Promise((resolve, reject) => __awaiter(this, void 0, void 0, function* () {
if (!src) {
return this._createPlayDeferred().promise;
}
const buffer = yield bufferSound(this._audioContext, this._XMLHttpRequest, src);
this.dispatchEvent('canplaythrough');
resolve(buffer);
}));
}
/**
* Reject all deferreds for the Play promise.
* @param reason
*/
_rejectPlayDeferreds(reason) {
const deferreds = this._pendingPlayDeferreds;
deferreds.splice(0, deferreds.length).forEach(({ reject }) => reject(reason));
}
/**
* Resolve all deferreds for the Play promise.
* @param result
*/
_resolvePlayDeferreds(result) {
const deferreds = this._pendingPlayDeferreds;
deferreds.splice(0, deferreds.length).forEach(({ resolve }) => resolve(result));
}
}
/**
* Use XMLHttpRequest to load the AudioBuffer of a remote audio asset.
* @private
* @param context - The AudioContext to use to decode the audio data
* @param RequestFactory - The XMLHttpRequest factory to build
* @param src - The URL of the audio asset to load.
* @returns A Promise containing the decoded AudioBuffer.
*/
// tslint:disable-next-line:variable-name
function bufferSound(context, RequestFactory, src) {
return __awaiter(this, void 0, void 0, function* () {
const request = new RequestFactory();
request.open('GET', src, true);
request.responseType = 'arraybuffer';
const event = yield new Promise(resolve => {
request.addEventListener('load', resolve);
request.send();
});
// Safari uses a callback here instead of a Promise.
try {
return context.decodeAudioData(event.target.response);
}
catch (e) {
return new Promise(resolve => {
context.decodeAudioData(event.target.response, resolve);
});
}
});
}
export { AudioPlayer as default };
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"audioplayer.js","sources":["../../../lib/twilio/audioplayer/audioplayer.ts"],"sourcesContent":[null],"names":[],"mappings":";;;;AAqBA;;;;AAIG;AACH,MAAM,WAAY,SAAQ,WAAW,CAAA;IAmEnC,IAAI,WAAW,KAAsC,OAAO,IAAI,CAAC,YAAY,CAAC,CAAC;IAC/E,IAAI,IAAI,KAAc,OAAO,IAAI,CAAC,KAAK,CAAC,CAAC;IACzC,IAAI,IAAI,CAAC,UAAmB,EAAA;QAC1B,MAAM,IAAI,GAAG,IAAI;AACjB,QAAA,SAAS,qBAAqB,GAAA;YAC5B,IAAI,CAAC,UAAU,CAAC,mBAAmB,CAAC,OAAO,EAAE,qBAAqB,CAAC;YACnE,IAAI,CAAC,KAAK,EAAE;QACd;;;AAGA,QAAA,IAAI,CAAC,UAAU,IAAI,IAAI,CAAC,IAAI,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE;YAC5C,IAAI,CAAC,UAAU,CAAC,gBAAgB,CAAC,OAAO,EAAE,qBAAqB,CAAC;QAClE;AAEA,QAAA,IAAI,CAAC,KAAK,GAAG,UAAU;IACzB;AAEA;;AAEG;AACH,IAAA,IAAI,KAAK,GAAA,EAAc,OAAO,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,KAAK,KAAK,CAAC,CAAC,CAAC;IAC/D,IAAI,KAAK,CAAC,aAAsB,EAAA;AAC9B,QAAA,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,KAAK,GAAG,aAAa,GAAG,CAAC,GAAG,CAAC;IACnD;AAEA;;;AAGG;IACH,IAAI,MAAM,GAAA,EAAc,OAAO,IAAI,CAAC,UAAU,KAAK,IAAI,CAAC,CAAC;IACzD,IAAI,GAAG,KAAa,OAAO,IAAI,CAAC,IAAI,CAAC,CAAC;IACtC,IAAI,GAAG,CAAC,GAAW,EAAA;AACjB,QAAA,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC;IACjB;AAEA;;AAEG;AACH,IAAA,IAAI,SAAS,GAAA;AACX,QAAA,OAAO,IAAI,CAAC,aAAa,CAAC,SAAS;IACrC;IACA,IAAI,SAAS,CAAC,SAAuD,EAAA;AACnE,QAAA,IAAI,CAAC,aAAa,CAAC,SAAS,GAAG,SAAS;IAC1C;IACA,IAAI,MAAM,KAAa,OAAO,IAAI,CAAC,OAAO,CAAC,CAAC;AAkB5C;;AAEG;AACH,IAAA,WAAA,CAAY,YAAiB,EACjB,YAAA,GAA2C,EAA0B,EACrE,UAA+B,EAA0B,EAAA;AACnE,QAAA,KAAK,EAAE;AA1HT;;;;AAIG;QACK,IAAA,CAAA,UAAU,GAA+B,IAAI;AAqBrD;;;AAGG;QACK,IAAA,CAAA,KAAK,GAAY,KAAK;AAE9B;;;;AAIG;QACK,IAAA,CAAA,qBAAqB,GAAiC,EAAE;AAEhE;;AAEG;QACK,IAAA,CAAA,OAAO,GAAW,SAAS;AAEnC;;AAEG;QACK,IAAA,CAAA,IAAI,GAAW,EAAE;AA6EvB,QAAA,IAAI,OAAO,YAAY,KAAK,QAAQ,EAAE;YACpC,OAAO,GAAG,YAAY;QACxB;AAEA,QAAA,IAAI,CAAC,aAAa,GAAG,YAAkC;AACvD,QAAA,IAAI,CAAC,aAAa,GAAG,KAAK,OAAO,CAAC,YAAY,IAAI,KAAK,GAAG;QAC1D,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,mBAAmB,EAAE,CAAC,OAAO;QACxD,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,aAAa,CAAC,WAAW;QAClD,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC,aAAa,CAAC,UAAU,EAAE;QAChD,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,IAAI,CAAC,YAAY,CAAC;QACzC,IAAI,CAAC,eAAe,GAAG,OAAO,CAAC,qBAAqB,IAAI,cAAc;AAEtE,QAAA,IAAI,CAAC,gBAAgB,CAAC,gBAAgB,EAAE,MAAK;YAC3C,IAAI,CAAC,qBAAqB,EAAE;AAC9B,QAAA,CAAC,CAAC;AAEF,QAAA,IAAI,OAAO,YAAY,KAAK,QAAQ,EAAE;AACpC,YAAA,IAAI,CAAC,GAAG,GAAG,YAAY;QACzB;IACF;AAEA;;AAEG;IACH,IAAI,GAAA;AACF,QAAA,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC;IACvB;AAEA;;;AAGG;IACH,KAAK,GAAA;AACH,QAAA,IAAI,IAAI,CAAC,MAAM,EAAE;YAAE;QAAQ;AAE3B,QAAA,IAAI,CAAC,aAAa,CAAC,KAAK,EAAE;AAE1B,QAAA,IAAI,CAAC,UAAU,CAAC,IAAI,EAAE;QACtB,IAAI,CAAC,UAAU,CAAC,UAAU,CAAC,IAAI,CAAC,SAAS,CAAC;AAC1C,QAAA,IAAI,CAAC,UAAU,GAAG,IAAI;QAEtB,IAAI,CAAC,oBAAoB,CAAC,IAAI,KAAK,CAAC,0DAA0D,CAAC,CAAC;IAClG;AAEA;;;;AAIG;IACG,IAAI,GAAA;;AACR,YAAA,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE;gBAChB,MAAM,IAAI,CAAC,cAAc;AACzB,gBAAA,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE;oBAAE;gBAAQ;AAC5B,gBAAA,MAAM,IAAI,KAAK,CAAC,0DAA0D,CAAC;YAC7E;YAEA,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,aAAa,CAAC,kBAAkB,EAAE;YACzD,IAAI,CAAC,UAAU,CAAC,IAAI,GAAG,IAAI,CAAC,IAAI;YAEhC,IAAI,CAAC,UAAU,CAAC,gBAAgB,CAAC,OAAO,EAAE,MAAK;gBAC7C,IAAI,IAAI,CAAC,UAAU,IAAI,IAAI,CAAC,UAAU,CAAC,IAAI,EAAE;oBAAE;gBAAQ;AACvD,gBAAA,IAAI,CAAC,aAAa,CAAC,OAAO,CAAC;AAC7B,YAAA,CAAC,CAAC;AAEF,YAAA,MAAM,MAAM,GAAgB,MAAM,IAAI,CAAC,cAAc;AAErD,YAAA,IAAI,IAAI,CAAC,MAAM,EAAE;AACf,gBAAA,MAAM,IAAI,KAAK,CAAC,0DAA0D,CAAC;YAC7E;AAEA,YAAA,IAAI,CAAC,UAAU,CAAC,MAAM,GAAG,MAAM;YAC/B,IAAI,CAAC,UAAU,CAAC,OAAO,CAAC,IAAI,CAAC,SAAS,CAAC;AACvC,YAAA,IAAI,CAAC,UAAU,CAAC,KAAK,EAAE;AAEvB,YAAA,IAAI,IAAI,CAAC,aAAa,CAAC,SAAS,EAAE;AAChC,gBAAA,OAAO,IAAI,CAAC,aAAa,CAAC,IAAI,EAAE;YAClC;QACF,CAAC,CAAA;AAAA,IAAA;AAED;;;AAGG;AACG,IAAA,SAAS,CAAC,MAAc,EAAA;;YAC5B,IAAI,OAAO,IAAI,CAAC,aAAa,CAAC,SAAS,KAAK,UAAU,EAAE;AACtD,gBAAA,MAAM,IAAI,KAAK,CAAC,0CAA0C,CAAC;YAC7D;AAEA,YAAA,IAAI,MAAM,KAAK,IAAI,CAAC,MAAM,EAAE;gBAC1B;YACF;AAEA,YAAA,IAAI,MAAM,KAAK,SAAS,EAAE;AACxB,gBAAA,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE;oBAChB,IAAI,CAAC,SAAS,CAAC,UAAU,CAAC,IAAI,CAAC,YAAY,CAAC;gBAC9C;AAEA,gBAAA,IAAI,CAAC,aAAa,CAAC,SAAS,GAAG,IAAI;gBACnC,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,aAAa,CAAC,WAAW;gBAClD,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,IAAI,CAAC,YAAY,CAAC;AACzC,gBAAA,IAAI,CAAC,OAAO,GAAG,MAAM;gBACrB;YACF;YAEA,MAAM,IAAI,CAAC,aAAa,CAAC,SAAS,CAAC,MAAM,CAAC;AAC1C,YAAA,IAAI,IAAI,CAAC,aAAa,CAAC,SAAS,EAAE;gBAAE;YAAQ;YAE5C,IAAI,CAAC,SAAS,CAAC,UAAU,CAAC,IAAI,CAAC,aAAa,CAAC,WAAW,CAAC;YACzD,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,aAAa,CAAC,4BAA4B,EAAE;YACrE,IAAI,CAAC,aAAa,CAAC,SAAS,GAAG,IAAI,CAAC,YAAY,CAAC,MAAM;AACvD,YAAA,IAAI,CAAC,OAAO,GAAG,MAAM;YAErB,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,IAAI,CAAC,YAAY,CAAC;QAC3C,CAAC,CAAA;AAAA,IAAA;AAED;;;AAGG;IACK,mBAAmB,GAAA;AACzB,QAAA,MAAM,QAAQ,GAAG,IAAI,QAAQ,EAAE;AAC/B,QAAA,IAAI,CAAC,qBAAqB,CAAC,IAAI,CAAC,QAAiC,CAAC;AAClE,QAAA,OAAO,QAAiC;IAC1C;AAEA;;;AAGG;AACK,IAAA,KAAK,CAAC,GAAW,EAAA;QACvB,IAAI,IAAI,CAAC,IAAI,IAAI,IAAI,CAAC,IAAI,KAAK,GAAG,EAAE;YAClC,IAAI,CAAC,KAAK,EAAE;QACd;AAEA,QAAA,IAAI,CAAC,IAAI,GAAG,GAAG;QACf,IAAI,CAAC,cAAc,GAAG,IAAI,OAAO,CAAC,CAAO,OAAO,EAAE,MAAM,KAAI,SAAA,CAAA,IAAA,EAAA,MAAA,EAAA,MAAA,EAAA,aAAA;YAC1D,IAAI,CAAC,GAAG,EAAE;AACR,gBAAA,OAAO,IAAI,CAAC,mBAAmB,EAAE,CAAC,OAAO;YAC3C;AAEA,YAAA,MAAM,MAAM,GAAG,MAAM,WAAW,CAAC,IAAI,CAAC,aAAa,EAAE,IAAI,CAAC,eAAe,EAAE,GAAG,CAAC;AAC/E,YAAA,IAAI,CAAC,aAAa,CAAC,gBAAgB,CAAC;YACpC,OAAO,CAAC,MAAM,CAAC;QACjB,CAAC,CAAA,CAAC;IACJ;AAEA;;;AAGG;AACK,IAAA,oBAAoB,CAAC,MAAY,EAAA;AACvC,QAAA,MAAM,SAAS,GAAG,IAAI,CAAC,qBAAqB;QAC5C,SAAS,CAAC,MAAM,CAAC,CAAC,EAAE,SAAS,CAAC,MAAM,CAAC,CAAC,OAAO,CAAC,CAAC,EAAE,MAAM,EAAE,KAAK,MAAM,CAAC,MAAM,CAAC,CAAC;IAC/E;AAEA;;;AAGG;AACK,IAAA,qBAAqB,CAAC,MAAY,EAAA;AACxC,QAAA,MAAM,SAAS,GAAG,IAAI,CAAC,qBAAqB;QAC5C,SAAS,CAAC,MAAM,CAAC,CAAC,EAAE,SAAS,CAAC,MAAM,CAAC,CAAC,OAAO,CAAC,CAAC,EAAE,OAAO,EAAE,KAAK,OAAO,CAAC,MAAM,CAAC,CAAC;IACjF;AACD;AAED;;;;;;;AAOG;AACH;AACA,SAAe,WAAW,CAAC,OAAY,EAAE,cAAmB,EAAE,GAAW,EAAA;;AACvE,QAAA,MAAM,OAAO,GAAmB,IAAI,cAAc,EAAE;QACpD,OAAO,CAAC,IAAI,CAAC,KAAK,EAAE,GAAG,EAAE,IAAI,CAAC;AAC9B,QAAA,OAAO,CAAC,YAAY,GAAG,aAAa;QAEpC,MAAM,KAAK,GAAQ,MAAM,IAAI,OAAO,CAAC,OAAO,IAAG;AAC7C,YAAA,OAAO,CAAC,gBAAgB,CAAC,MAAM,EAAE,OAAO,CAAC;YACzC,OAAO,CAAC,IAAI,EAAE;AAChB,QAAA,CAAC,CAAC;;AAGF,QAAA,IAAI;YACF,OAAO,OAAO,CAAC,eAAe,CAAC,KAAK,CAAC,MAAM,CAAC,QAAQ,CAAC;QACvD;QAAE,OAAO,CAAC,EAAE;AACV,YAAA,OAAO,IAAI,OAAO,CAAC,OAAO,IAAG;gBAC3B,OAAO,CAAC,eAAe,CAAC,KAAK,CAAC,MAAM,CAAC,QAAQ,EAAE,OAAO,CAAC;AACzD,YAAA,CAAC,CAAyB;QAC5B;IACF,CAAC,CAAA;AAAA;;;;"}