123 lines
3.6 KiB
JavaScript
123 lines
3.6 KiB
JavaScript
|
|
/**
|
|
* Provides a simplified access point to the frequency bins in the form of a visualization update listener.
|
|
*/
|
|
export default class Visualizer {
|
|
#source;
|
|
#stream;
|
|
#analyzing = false;
|
|
#updateListeners = [];
|
|
#audioCtx;
|
|
#analyzer;
|
|
#buffer;
|
|
#lastUpdate;
|
|
#fftSize;
|
|
|
|
/**
|
|
* @callback Visualizer~visualizerUpdateListener
|
|
* @param {number} delta elapsed time since last update.
|
|
* @param {Uint8Array} bins the bins with varying frequency values.
|
|
* @param {number} sigBin The bin with the greatest amplitude.
|
|
*/
|
|
|
|
/**
|
|
*
|
|
* @param {MediaSource|HTMLMediaElement} stream a media source to analyze.
|
|
* @param {number} [fftSize = 1024] the size of the fft window.
|
|
*/
|
|
constructor(stream, fftSize = 1024) {
|
|
this.#fftSize = fftSize;
|
|
this.#stream = stream;
|
|
this.#analyzing = false;
|
|
this.#updateListeners = [];
|
|
this.#lastUpdate = null;
|
|
}
|
|
|
|
#initialize() {
|
|
this.#audioCtx = new window.AudioContext();
|
|
if (this.#stream instanceof HTMLMediaElement) {
|
|
this.#source = this.#audioCtx.createMediaElementSource(this.#stream);
|
|
} else {
|
|
this.#source = this.#audioCtx.createMediaStreamSource(this.#stream);
|
|
}
|
|
this.#analyzer = this.#audioCtx.createAnalyser();
|
|
this.#analyzer.fftSize = this.fftSize;
|
|
this.#buffer = new Uint8Array(this.#analyzer.frequencyBinCount);
|
|
this.#source.connect(this.#analyzer);
|
|
this.#analyzer.connect(this.#audioCtx.destination);
|
|
}
|
|
|
|
/**
|
|
* Begins analyzing and sending out update pings.
|
|
*/
|
|
analyze() {
|
|
if (this.#analyzing) {
|
|
return;
|
|
}
|
|
if (!this.#audioCtx) this.#initialize();
|
|
this.#analyzing = true;
|
|
let self = this; // since calling from requestAnimationFrame means "this" is no longer set to produced object.
|
|
const update = (timestamp) => {
|
|
if (!self.#analyzing) return;
|
|
|
|
if (!self.#lastUpdate) {
|
|
self.#lastUpdate = timestamp;
|
|
}
|
|
let delta = timestamp - self.#lastUpdate;
|
|
self.#analyzer.getByteFrequencyData(self.#buffer);
|
|
|
|
self.#updateListeners.forEach(listener => {
|
|
listener(delta, self.#buffer, self.#buffer.indexOf(Math.max(self.#buffer)));
|
|
});
|
|
requestAnimationFrame(update);
|
|
};
|
|
requestAnimationFrame(update);
|
|
}
|
|
|
|
/**
|
|
* Stops the analysis. Listeners will stop receiving bins.
|
|
*/
|
|
stop() {
|
|
this.#analyzing = false;
|
|
}
|
|
|
|
/**
|
|
*
|
|
* @param {Visualizer~visualizerUpdateListener} listener the visualizer update listener to be registered.
|
|
* @returns {boolean} true if and only if the listener was successfully added.
|
|
*/
|
|
addUpdateListener(listener) {
|
|
if (this.#updateListeners.includes(listener));
|
|
this.#updateListeners.push(listener);
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
*
|
|
* @param {Visualizer~visualizerUpdateListener} listener the visualizer update listener to remove.
|
|
* @returns {boolean} true if and only if the removal of the listener was a success.
|
|
*/
|
|
removeUpdateListener(listener) {
|
|
const removeIndex = this.#updateListeners.indexOf(listener);
|
|
if (removeIndex < 0) return false;
|
|
|
|
this.#updateListeners.splice(removeIndex, 1);
|
|
return true;
|
|
}
|
|
|
|
|
|
/**
|
|
* @returns {number} The number of bins based on the size of the FFT window.
|
|
*/
|
|
get numberOfBins() {
|
|
return this.#fftSize / 2;
|
|
}
|
|
|
|
/**
|
|
*
|
|
* @returns {number} The fft window size.
|
|
*/
|
|
get fftSize() {
|
|
return this.#fftSize;
|
|
}
|
|
} |