From 9e5c1d48015ee98a1e4fbe0f16500a7e5f3baaa9 Mon Sep 17 00:00:00 2001 From: lukasIO Date: Mon, 5 Aug 2024 17:54:09 +0200 Subject: [PATCH 1/3] Logarithmic binning of multiband track volumes --- packages/react/src/hooks/useTrackVolume.ts | 67 ++++++++++++++++++---- 1 file changed, 57 insertions(+), 10 deletions(-) diff --git a/packages/react/src/hooks/useTrackVolume.ts b/packages/react/src/hooks/useTrackVolume.ts index a1aa45371..2446a28a7 100644 --- a/packages/react/src/hooks/useTrackVolume.ts +++ b/packages/react/src/hooks/useTrackVolume.ts @@ -113,6 +113,7 @@ export function useMultibandTrackVolume( ? trackOrTrackReference : trackOrTrackReference?.publication?.track; const [frequencyBands, setFrequencyBands] = React.useState>([]); + options.analyserOptions = { ...multibandDefaults.analyserOptions, ...options.analyserOptions }; const opts = { ...multibandDefaults, ...options }; React.useEffect(() => { if (!track || !track?.mediaStream) { @@ -132,16 +133,14 @@ export function useMultibandTrackVolume( frequencies = frequencies.slice(options.loPass, options.hiPass); const normalizedFrequencies = normalizeFrequencies(frequencies); - const chunkSize = Math.ceil(normalizedFrequencies.length / opts.bands); - const chunks: Array = []; - for (let i = 0; i < opts.bands; i++) { - const summedVolumes = normalizedFrequencies - .slice(i * chunkSize, (i + 1) * chunkSize) - .reduce((acc, val) => (acc += val), 0); - chunks.push(summedVolumes / chunkSize); - } - - setFrequencyBands(chunks); + const binVolumes = fftToLogBins( + normalizedFrequencies, + analyser.context.sampleRate, + opts.bands, + opts.analyserOptions.fftSize!, + ); + + setFrequencyBands(binVolumes); }; const interval = setInterval(updateVolume, opts.updateInterval); @@ -154,3 +153,51 @@ export function useMultibandTrackVolume( return frequencyBands; } + +function calculateLogBinEdges(minFreq: number, maxFreq: number, numBins: number): number[] { + const logBinEdges: number[] = []; + const logMinFreq = Math.log(minFreq); + const logMaxFreq = Math.log(maxFreq); + const binWidth = (logMaxFreq - logMinFreq) / numBins; + + for (let i = 0; i <= numBins; i++) { + logBinEdges.push(Math.exp(logMinFreq + i * binWidth)); + } + + return logBinEdges; +} + +function fftToLogBins( + rawFFTValues: Float32Array, + sampleRate: number, + numBins: number, + fftSize: number, +): number[] { + const deltaF = sampleRate / fftSize; + + // Define the min and max frequencies of interest + const minFreq = deltaF; // Starting from the first FFT bin frequency + const maxFreq = sampleRate / 2; // Nyquist frequency + + const logBinEdges = calculateLogBinEdges(minFreq, maxFreq, numBins); + const logBinValues = new Array(numBins).fill(0); + + for (let i = 0; i < numBins; i++) { + const startFreq = logBinEdges[i]; + const endFreq = logBinEdges[i + 1]; + + const startBin = Math.ceil(startFreq / deltaF); + const endBin = Math.floor(endFreq / deltaF); + + if (startBin >= rawFFTValues.length) { + break; + } + + for (let j = startBin; j <= endBin && j < rawFFTValues.length; j++) { + logBinValues[i] += rawFFTValues[j]; + } + logBinValues[i] /= endBin - startBin + 1; + } + + return logBinValues; +} From 3ff50be85aee237848f0a77f27fc3fd65f1f9bb4 Mon Sep 17 00:00:00 2001 From: lukasIO Date: Tue, 6 Aug 2024 15:08:28 +0200 Subject: [PATCH 2/3] Create lemon-dancers-work.md --- .changeset/lemon-dancers-work.md | 5 +++++ 1 file changed, 5 insertions(+) create mode 100644 .changeset/lemon-dancers-work.md diff --git a/.changeset/lemon-dancers-work.md b/.changeset/lemon-dancers-work.md new file mode 100644 index 000000000..94a238de5 --- /dev/null +++ b/.changeset/lemon-dancers-work.md @@ -0,0 +1,5 @@ +--- +"@livekit/components-react": patch +--- + +Logarithmic binning of multiband track volumes From ab5955c94bb28853ab67bec7cbe48f7a4262b640 Mon Sep 17 00:00:00 2001 From: lukasIO Date: Fri, 9 Aug 2024 13:59:49 +0200 Subject: [PATCH 3/3] some more visual mapping --- .../src/components/participant/AudioVisualizer.tsx | 10 +++++++++- packages/react/src/hooks/useTrackVolume.ts | 1 + 2 files changed, 10 insertions(+), 1 deletion(-) diff --git a/packages/react/src/components/participant/AudioVisualizer.tsx b/packages/react/src/components/participant/AudioVisualizer.tsx index b9201e7e7..43759fc46 100644 --- a/packages/react/src/components/participant/AudioVisualizer.tsx +++ b/packages/react/src/components/participant/AudioVisualizer.tsx @@ -30,7 +30,15 @@ export const AudioVisualizer: ( const barCount = 7; const trackReference = useEnsureTrackRef(trackRef); - const volumes = useMultibandTrackVolume(trackReference, { bands: 7, loPass: 300 }); + const volumes = useMultibandTrackVolume(trackReference, { + bands: 7, + loPass: 60, + hiPass: 1800, + analyserOptions: { + minDecibels: -60, + maxDecibels: -20, + }, + }); return (