react-voice-visualizer
Advanced tools
Comparing version
type UseLatestReturnType<T> = { | ||
readonly current: T; | ||
}; | ||
declare function useLatest<T>(value: T): UseLatestReturnType<T>; | ||
export default useLatest; | ||
export declare function useLatest<T>(value: T): UseLatestReturnType<T>; | ||
export {}; |
import { Controls, useVoiceVisualizerParams } from "../types/types.ts"; | ||
declare function useVoiceVisualizer({ onStartRecording, onStopRecording, onPausedRecording, onResumedRecording, onClearCanvas, onEndAudioPlayback, onStartAudioPlayback, onPausedAudioPlayback, onResumedAudioPlayback, }?: useVoiceVisualizerParams): Controls; | ||
declare function useVoiceVisualizer({ onStartRecording, onStopRecording, onPausedRecording, onResumedRecording, onClearCanvas, onEndAudioPlayback, onStartAudioPlayback, onPausedAudioPlayback, onResumedAudioPlayback, onErrorPlayingAudio, }?: useVoiceVisualizerParams): Controls; | ||
export default useVoiceVisualizer; |
(function(){"use strict";(e=>{try{if(typeof window>"u")return;var i=document.createElement("style");i.appendChild(document.createTextNode(e)),document.head.appendChild(i)}catch(o){console.error("vite-plugin-css-injected-by-js",o)}})(".voice-visualizer__buttons-container{display:flex;justify-content:center;align-items:center;column-gap:20px;row-gap:15px;flex-wrap:wrap;margin-bottom:40px}.voice-visualizer__btn-center{box-sizing:border-box;flex-shrink:0;width:60px;height:60px;padding:0;display:flex;justify-content:center;align-items:center;border-radius:50%;background-color:#fff;border:4px solid #c5c5c5;outline:none;cursor:pointer;transition:border-color .3s,background-color .3s}.voice-visualizer__btn-center:hover{background-color:#eaeaea}.voice-visualizer__btn-center>img{width:auto;height:50%;max-height:30px}.voice-visualizer__btn-center.voice-visualizer__btn-center-pause{background-color:#ff3030}.voice-visualizer__btn-center.voice-visualizer__btn-center-pause:hover{background-color:#ff4f4f}.voice-visualizer__btn-center.voice-visualizer__btn-center-pause>img{height:50%;max-height:16px}.voice-visualizer__btn-center:hover{border:4px solid #9f9f9f}.voice-visualizer__btn-left{box-sizing:border-box;flex-shrink:0;width:60px;height:60px;padding:0;display:flex;justify-content:center;align-items:center;border-radius:50%;background-color:#ff3030;border:4px solid #c5c5c5;outline:none;cursor:pointer;transition:border-color .3s,background-color .3s,opacity .3s}.voice-visualizer__btn-left:hover{background-color:#ff4f4f}.voice-visualizer__btn-left:disabled{opacity:.6;background-color:#ff3030}.voice-visualizer__btn-left.voice-visualizer__btn-left-microphone{background-color:#fff}.voice-visualizer__btn-left.voice-visualizer__btn-left-microphone>img{width:auto;height:50%;max-height:30px}.voice-visualizer__btn-left>img{width:auto;height:50%;max-height:16px}.voice-visualizer__btn-left:hover{border:4px solid #9f9f9f}.voice-visualizer__btn{box-sizing:border-box;min-width:100px;min-height:60px;padding:5px 20px;border-radius:40px;font-size:15px;background-color:#f0f0f0;transition:background-color .3s,opacity .3s}.voice-visualizer__btn:disabled{opacity:.8;background-color:#f0f0f0}.voice-visualizer__btn:hover{background-color:#bebebe}.voice-visualizer__canvas-container{position:relative;width:fit-content;margin:0 auto;overflow:hidden}.voice-visualizer__canvas-container canvas{display:block}.voice-visualizer__canvas-microphone-btn{position:absolute;top:50%;left:50%;width:auto;max-width:12%;min-width:24px;height:50%;max-height:100px;background-color:transparent;border:none;outline:none;transform:translate(-50%,-50%)}.voice-visualizer__canvas-microphone-icon{width:100%;height:100%;will-change:transform;transition:transform .3s}.voice-visualizer__canvas-microphone-btn:hover .voice-visualizer__canvas-microphone-icon{transform:scale(1.03)}.voice-visualizer__canvas-audio-wave-icon{position:absolute;top:50%;left:50%;width:auto;max-width:40%;height:40%;max-height:100px;transform:translate(-118%,-50%) scale(-1)}.voice-visualizer__canvas-audio-wave-icon2{transform:translate(18%,-50%)}.voice-visualizer__canvas-audio-processing{position:absolute;top:50%;left:50%;margin:0;transform:translate(-50%,-50%)}.voice-visualizer__progress-indicator-hovered{position:absolute;top:0;pointer-events:none;height:100%;width:1px;background-color:#85858599}.voice-visualizer__progress-indicator-hovered-time{position:absolute;top:3%;left:1px;width:fit-content;margin:0;padding:0 7px;opacity:.8;font-size:12px;border-radius:0 4px 4px 0;background-color:#575757;text-align:left}.voice-visualizer__progress-indicator-hovered-time.voice-visualizer__progress-indicator-hovered-time-left{left:unset;right:1px;border-radius:4px 0 0 4px}.voice-visualizer__progress-indicator{position:absolute;top:0;pointer-events:none;height:100%;width:1px;background-color:#efefef}.voice-visualizer__progress-indicator-time{position:absolute;top:3%;left:1px;width:fit-content;box-sizing:border-box;min-width:37px;margin:0;padding:0 7px;font-size:12px;border-radius:0 4px 4px 0;text-align:left;color:#000;font-weight:500;background-color:#efefef}.voice-visualizer__progress-indicator-time.voice-visualizer__progress-indicator-time-left{left:unset;right:1px;border-radius:4px 0 0 4px}.voice-visualizer__audio-info-container{box-sizing:border-box;height:55px;display:flex;align-items:center;justify-content:center;gap:30px}.voice-visualizer__audio-info-time{margin:15px 0;min-width:38px;text-align:left}.voice-visualizer__visually-hidden{position:absolute;width:1px;height:1px;margin:-1px;padding:0;border:4px solid #c5c5c5;white-space:nowrap;clip-path:inset(100%);clip:rect(0 0 0 0);overflow:hidden}")})(); | ||
import { jsx as a, jsxs as de, Fragment as Ue } from "react/jsx-runtime"; | ||
import { useState as l, useRef as y, useCallback as tt, useLayoutEffect as We, forwardRef as rt, useEffect as Z } from "react"; | ||
const He = ({ | ||
import { jsx as a, jsxs as ae, Fragment as Fe } from "react/jsx-runtime"; | ||
import { useState as l, useRef as N, useCallback as rt, useLayoutEffect as Be, forwardRef as nt, useEffect as Z } from "react"; | ||
const be = ({ | ||
canvas: e, | ||
backgroundColor: t | ||
}) => { | ||
const n = e.height, r = e.width, c = Math.round(r / 2), s = e.getContext("2d"); | ||
return s ? (s.clearRect(0, 0, r, n), t !== "transparent" && (s.fillStyle = t, s.fillRect(0, 0, r, n)), { context: s, height: n, width: r, halfWidth: c }) : null; | ||
}, De = ({ | ||
const n = e.height, r = e.width, c = Math.round(r / 2), u = e.getContext("2d"); | ||
return u ? (u.clearRect(0, 0, r, n), t !== "transparent" && (u.fillStyle = t, u.fillRect(0, 0, r, n)), { context: u, height: n, width: r, halfWidth: c }) : null; | ||
}, je = ({ | ||
context: e, | ||
@@ -16,7 +16,7 @@ color: t, | ||
y: c, | ||
w: s, | ||
h: g | ||
w: u, | ||
h: M | ||
}) => { | ||
e.fillStyle = t, e.beginPath(), e.roundRect ? (e.roundRect(r, c, s, g, n), e.fill()) : e.fillRect(r, c, s, g); | ||
}, nt = ({ | ||
e.fillStyle = t, e.beginPath(), e.roundRect ? (e.roundRect(r, c, u, M, n), e.fill()) : e.fillRect(r, c, u, M); | ||
}, it = ({ | ||
barsData: e, | ||
@@ -27,20 +27,20 @@ canvas: t, | ||
backgroundColor: c, | ||
mainBarColor: s, | ||
secondaryBarColor: g, | ||
currentAudioTime: v = 0, | ||
rounded: M, | ||
duration: u | ||
mainBarColor: u, | ||
secondaryBarColor: M, | ||
currentAudioTime: m = 0, | ||
rounded: I, | ||
duration: v | ||
}) => { | ||
const I = He({ canvas: t, backgroundColor: c }); | ||
if (!I) | ||
const d = be({ canvas: t, backgroundColor: c }); | ||
if (!d) | ||
return; | ||
const { context: f, height: w } = I, L = v / u; | ||
e.forEach((o, p) => { | ||
const H = p / e.length, m = L > H; | ||
De({ | ||
context: f, | ||
color: m ? g : s, | ||
rounded: M, | ||
x: p * (n + r * n), | ||
y: w / 2 - o.max, | ||
const { context: z, height: A } = d, S = m / v; | ||
e.forEach((o, g) => { | ||
const b = g / e.length, h = S > b; | ||
je({ | ||
context: z, | ||
color: h ? M : u, | ||
rounded: I, | ||
x: g * (n + r * n), | ||
y: A / 2 - o.max, | ||
h: o.max * 2, | ||
@@ -51,3 +51,3 @@ w: n | ||
}; | ||
function it({ | ||
function ct({ | ||
context: e, | ||
@@ -58,15 +58,15 @@ color: t, | ||
height: c, | ||
barWidth: s | ||
barWidth: u | ||
}) { | ||
De({ | ||
je({ | ||
context: e, | ||
color: t, | ||
rounded: n, | ||
x: r / 2 + s / 2, | ||
x: r / 2 + u / 2, | ||
y: c / 2 - 1, | ||
h: 2, | ||
w: r - (r / 2 + s / 2) | ||
w: r - (r / 2 + u / 2) | ||
}); | ||
} | ||
const ct = ({ | ||
const ot = ({ | ||
audioData: e, | ||
@@ -77,61 +77,61 @@ unit: t, | ||
canvas: c, | ||
isRecordingInProgress: s, | ||
isPausedRecording: g, | ||
picks: v, | ||
backgroundColor: M, | ||
barWidth: u, | ||
mainBarColor: I, | ||
secondaryBarColor: f, | ||
rounded: w, | ||
animateCurrentPick: L, | ||
isRecordingInProgress: u, | ||
isPausedRecording: M, | ||
picks: m, | ||
backgroundColor: I, | ||
barWidth: v, | ||
mainBarColor: d, | ||
secondaryBarColor: z, | ||
rounded: A, | ||
animateCurrentPick: S, | ||
fullscreen: o | ||
}) => { | ||
const p = He({ canvas: c, backgroundColor: M }); | ||
if (!p) | ||
const g = be({ canvas: c, backgroundColor: I }); | ||
if (!g) | ||
return; | ||
const { context: H, height: m, width: x, halfWidth: j } = p; | ||
if (e != null && e.length && s) { | ||
const F = Math.max(...e); | ||
if (!g) { | ||
if (r.current >= u) { | ||
const { context: b, height: h, width: H, halfWidth: _ } = g; | ||
if (e != null && e.length && u) { | ||
const $ = Math.max(...e); | ||
if (!M) { | ||
if (r.current >= v) { | ||
r.current = 0; | ||
const D = (m - F / 258 * m) / m * 100, U = (-m + F / 258 * m * 2) / m * 100, V = n.current === u ? { | ||
startY: D, | ||
barHeight: U | ||
const j = (h - $ / 258 * h) / h * 100, ee = (-h + $ / 258 * h * 2) / h * 100, F = n.current === v ? { | ||
startY: j, | ||
barHeight: ee | ||
} : null; | ||
n.current >= t ? n.current = u : n.current += u, v.length > (o ? x : j) / u && v.pop(), v.unshift(V); | ||
n.current >= t ? n.current = v : n.current += v, m.length > (o ? H : _) / v && m.pop(), m.unshift(F); | ||
} | ||
r.current += 1; | ||
} | ||
!o && Q(), L && De({ | ||
context: H, | ||
rounded: w, | ||
color: I, | ||
x: o ? x : j, | ||
y: m - F / 258 * m, | ||
h: -m + F / 258 * m * 2, | ||
w: u | ||
!o && ue(), S && je({ | ||
context: b, | ||
rounded: A, | ||
color: d, | ||
x: o ? H : _, | ||
y: h - $ / 258 * h, | ||
h: -h + $ / 258 * h * 2, | ||
w: v | ||
}); | ||
let B = (o ? x : j) - r.current; | ||
v.forEach((D) => { | ||
D && De({ | ||
context: H, | ||
color: I, | ||
rounded: w, | ||
x: B, | ||
y: D.startY * m / 100 > m / 2 - 1 ? m / 2 - 1 : D.startY * m / 100, | ||
h: D.barHeight * m / 100 > 2 ? D.barHeight * m / 100 : 2, | ||
w: u | ||
}), B -= u; | ||
let U = (o ? H : _) - r.current; | ||
m.forEach((j) => { | ||
j && je({ | ||
context: b, | ||
color: d, | ||
rounded: A, | ||
x: U, | ||
y: j.startY * h / 100 > h / 2 - 1 ? h / 2 - 1 : j.startY * h / 100, | ||
h: j.barHeight * h / 100 > 2 ? j.barHeight * h / 100 : 2, | ||
w: v | ||
}), U -= v; | ||
}); | ||
} else | ||
v.length = 0; | ||
function Q() { | ||
it({ | ||
context: H, | ||
color: f, | ||
rounded: w, | ||
width: x, | ||
height: m, | ||
barWidth: u | ||
m.length = 0; | ||
function ue() { | ||
ct({ | ||
context: b, | ||
color: z, | ||
rounded: A, | ||
width: H, | ||
height: h, | ||
barWidth: v | ||
}); | ||
@@ -153,3 +153,3 @@ } | ||
).charAt(0)}`; | ||
}, ot = (e) => { | ||
}, st = (e) => { | ||
const t = Math.floor(e / 1e3), n = Math.floor(t / 3600), r = Math.floor(t % 3600 / 60), c = t % 60; | ||
@@ -169,3 +169,3 @@ return n > 0 ? `${String(n).padStart(2, "0")}:${String(r).padStart( | ||
} | ||
const st = ({ | ||
const at = ({ | ||
bufferData: e, | ||
@@ -177,22 +177,22 @@ height: t, | ||
}) => { | ||
const s = n / (r + c * r), g = Math.floor(e.length / s), v = t / 2; | ||
let M = [], u = 0; | ||
for (let I = 0; I < s; I++) { | ||
const f = []; | ||
let w = 0; | ||
for (let o = 0; o < g && I * g + o < e.length; o++) { | ||
const p = e[I * g + o]; | ||
p > 0 && (f.push(p), w++); | ||
const u = n / (r + c * r), M = Math.floor(e.length / u), m = t / 2; | ||
let I = [], v = 0; | ||
for (let d = 0; d < u; d++) { | ||
const z = []; | ||
let A = 0; | ||
for (let o = 0; o < M && d * M + o < e.length; o++) { | ||
const g = e[d * M + o]; | ||
g > 0 && (z.push(g), A++); | ||
} | ||
const L = f.reduce((o, p) => o + p, 0) / w; | ||
L > u && (u = L), M.push({ max: L }); | ||
const S = z.reduce((o, g) => o + g, 0) / A; | ||
S > v && (v = S), I.push({ max: S }); | ||
} | ||
if (v * 0.95 > u * v) { | ||
const I = v * 0.95 / u; | ||
M = M.map((f) => ({ | ||
max: f.max > 0.01 ? f.max * I : 1 | ||
if (m * 0.95 > v * m) { | ||
const d = m * 0.95 / v; | ||
I = I.map((z) => ({ | ||
max: z.max > 0.01 ? z.max * d : 1 | ||
})); | ||
} | ||
return M; | ||
}, at = (e) => { | ||
return I; | ||
}, ut = (e) => { | ||
if (!e) | ||
@@ -202,3 +202,3 @@ return ""; | ||
return t && t.length >= 2 ? `.${t[1]}` : ""; | ||
}, ut = (e) => { | ||
}, ht = (e) => { | ||
const t = Math.floor(e / 3600), n = Math.floor(e % 3600 / 60), r = e % 60, c = Math.floor( | ||
@@ -215,3 +215,3 @@ (r - Math.floor(r)) * 1e3 | ||
).charAt(0)}${String(c).charAt(1)}s`; | ||
}, ht = (e) => { | ||
}, lt = (e) => { | ||
onmessage = (t) => { | ||
@@ -221,3 +221,3 @@ postMessage(e(t.data)); | ||
}; | ||
function lt({ | ||
function mt({ | ||
fn: e, | ||
@@ -231,18 +231,18 @@ initialValue: t, | ||
setResult: c, | ||
run: (g) => { | ||
const v = new Worker( | ||
run: (M) => { | ||
const m = new Worker( | ||
// eslint-disable-next-line @typescript-eslint/restrict-template-expressions | ||
URL.createObjectURL(new Blob([`(${ht})(${e})`])) | ||
URL.createObjectURL(new Blob([`(${lt})(${e})`])) | ||
); | ||
v.onmessage = (M) => { | ||
M.data && (c(M.data), n && n(), v.terminate()); | ||
}, v.onerror = (M) => { | ||
console.error(M.message), v.terminate(); | ||
}, v.postMessage(g); | ||
m.onmessage = (I) => { | ||
I.data && (c(I.data), n && n(), m.terminate()); | ||
}, m.onerror = (I) => { | ||
console.error(I.message), m.terminate(); | ||
}, m.postMessage(M); | ||
} | ||
}; | ||
} | ||
const mt = (e, t = 250) => { | ||
const n = y(); | ||
return tt( | ||
const vt = (e, t = 250) => { | ||
const n = N(); | ||
return rt( | ||
// eslint-disable-next-line @typescript-eslint/no-explicit-any | ||
@@ -258,3 +258,9 @@ (...r) => { | ||
}; | ||
const vt = ({ | ||
function dt(e) { | ||
const t = N(e); | ||
return Be(() => { | ||
t.current = e; | ||
}, [e]), t; | ||
} | ||
const ft = ({ | ||
color: e = "#000000", | ||
@@ -298,10 +304,3 @@ stroke: t = 2, | ||
} | ||
), Be = "", dt = "", Pe = "", ft = ""; | ||
function zt(e) { | ||
const t = y(e); | ||
return We(() => { | ||
t.current = e; | ||
}, [e]), t; | ||
} | ||
const It = rt( | ||
), Pe = "", zt = "", We = "", gt = "", pt = nt( | ||
({ | ||
@@ -313,110 +312,110 @@ controls: { | ||
duration: r, | ||
audioSrc: c, | ||
currentAudioTime: s, | ||
bufferFromRecordedBlob: g, | ||
togglePauseResume: v, | ||
startRecording: M, | ||
stopRecording: u, | ||
saveAudioFile: I, | ||
isAvailableRecordedAudio: f, | ||
isPausedRecordedAudio: w, | ||
isPausedRecording: L, | ||
currentAudioTime: c, | ||
audioSrc: u, | ||
bufferFromRecordedBlob: M, | ||
togglePauseResume: m, | ||
startRecording: I, | ||
stopRecording: v, | ||
saveAudioFile: d, | ||
isAvailableRecordedAudio: z, | ||
isPausedRecordedAudio: A, | ||
isPausedRecording: S, | ||
isProcessingRecordedAudio: o, | ||
isCleared: p, | ||
formattedDuration: H, | ||
formattedRecordingTime: m, | ||
formattedRecordedAudioCurrentTime: x, | ||
clearCanvas: j, | ||
setCurrentAudioTime: Q, | ||
_setIsProcessingAudioOnComplete: F, | ||
_setIsProcessingOnResize: B | ||
isCleared: g, | ||
formattedDuration: b, | ||
formattedRecordingTime: h, | ||
formattedRecordedAudioCurrentTime: H, | ||
clearCanvas: _, | ||
setCurrentAudioTime: ue, | ||
_setIsProcessingAudioOnComplete: $, | ||
_setIsProcessingOnResize: U | ||
}, | ||
width: D = "100%", | ||
height: U = 200, | ||
speed: V = 3, | ||
backgroundColor: $ = "transparent", | ||
mainBarColor: R = "#FFFFFF", | ||
secondaryBarColor: q = "#5e5e5e", | ||
barWidth: P = 2, | ||
gap: fe = 1, | ||
rounded: ne = 5, | ||
isControlPanelShown: ie = !0, | ||
isDownloadAudioButtonShown: ce = !1, | ||
animateCurrentPick: X = !0, | ||
fullscreen: oe = !1, | ||
onlyRecording: W = !1, | ||
isDefaultUIShown: se = !0, | ||
defaultMicrophoneIconColor: Te = R, | ||
defaultAudioWaveIconColor: Le = R, | ||
mainContainerClassName: be, | ||
canvasContainerClassName: ze, | ||
isProgressIndicatorShown: Y = !W, | ||
progressIndicatorClassName: d, | ||
isProgressIndicatorTimeShown: C = !0, | ||
progressIndicatorTimeClassName: K, | ||
isProgressIndicatorOnHoverShown: ae = !W, | ||
progressIndicatorOnHoverClassName: G, | ||
isProgressIndicatorTimeOnHoverShown: T = !0, | ||
progressIndicatorTimeOnHoverClassName: _, | ||
isAudioProcessingTextShown: h = !0, | ||
audioProcessingTextClassName: pe, | ||
controlButtonsClassName: we | ||
}, Se) => { | ||
const [ge, Ee] = l(0), [b, Ce] = l(0), [ee, ue] = l(0), [he, _e] = l(0), [le, te] = l(!1), [Ne, ye] = l(window.innerWidth), [re, Ae] = l(!1), i = Ne < 768, A = Math.trunc(V), S = Math.trunc(fe), N = Math.trunc( | ||
i && S > 0 ? P + 1 : P | ||
), O = N + S * N, z = y(null), je = y([]), me = y(A), ve = y(N), Je = y(N), Me = y(null), Qe = zt(Ne), { | ||
result: Ie, | ||
setResult: Ve, | ||
run: qe | ||
} = lt({ | ||
fn: st, | ||
width: j = "100%", | ||
height: ee = 200, | ||
speed: F = 3, | ||
backgroundColor: O = "transparent", | ||
mainBarColor: D = "#FFFFFF", | ||
secondaryBarColor: B = "#5e5e5e", | ||
barWidth: te = 2, | ||
gap: k = 1, | ||
rounded: J = 5, | ||
isControlPanelShown: ye = !0, | ||
isDownloadAudioButtonShown: re = !1, | ||
animateCurrentPick: ne = !0, | ||
fullscreen: Y = !1, | ||
onlyRecording: G = !1, | ||
isDefaultUIShown: he = !0, | ||
defaultMicrophoneIconColor: De = D, | ||
defaultAudioWaveIconColor: ge = D, | ||
mainContainerClassName: le, | ||
canvasContainerClassName: Q, | ||
isProgressIndicatorShown: p = !G, | ||
progressIndicatorClassName: R, | ||
isProgressIndicatorTimeShown: V = !0, | ||
progressIndicatorTimeClassName: ie, | ||
isProgressIndicatorOnHoverShown: q = !G, | ||
progressIndicatorOnHoverClassName: C, | ||
isProgressIndicatorTimeOnHoverShown: x = !0, | ||
progressIndicatorTimeOnHoverClassName: s, | ||
isAudioProcessingTextShown: Me = !0, | ||
audioProcessingTextClassName: Te, | ||
controlButtonsClassName: Ie | ||
}, Ee) => { | ||
const [me, pe] = l(0), [T, Ce] = l(0), [X, ve] = l(0), [ce, _e] = l(0), [oe, K] = l(!1), [we, Le] = l(window.innerWidth), [se, Se] = l(!1), i = we < 768, f = Math.trunc(F), E = Math.trunc(k), w = Math.trunc( | ||
i && E > 0 ? te + 1 : te | ||
), Ne = w + E * w, L = N(null), He = N([]), Ae = N(f), Je = N(w), Qe = N(w), de = N(null), P = Ee, Ve = dt(we), { | ||
result: fe, | ||
setResult: qe, | ||
run: Xe | ||
} = mt({ | ||
fn: at, | ||
initialValue: [], | ||
onMessageReceived: Ke | ||
}), Xe = mt(xe); | ||
onMessageReceived: et | ||
}), Ke = vt(xe); | ||
Z(() => { | ||
xe(); | ||
const E = () => { | ||
Qe.current !== window.innerWidth && (f ? (ye(window.innerWidth), B(!0), Ae(!0), Xe()) : (ye(window.innerWidth), xe())); | ||
const y = () => { | ||
Ve.current !== window.innerWidth && (z ? (Le(window.innerWidth), U(!0), Se(!0), Ke()) : (Le(window.innerWidth), xe())); | ||
}; | ||
return window.addEventListener("resize", E), () => { | ||
window.removeEventListener("resize", E); | ||
return window.addEventListener("resize", y), () => { | ||
window.removeEventListener("resize", y); | ||
}; | ||
}, [D, f]), We(() => { | ||
z.current && ((me.current >= A || !e.length) && (me.current = 0, ct({ | ||
}, [j, z]), Be(() => { | ||
L.current && ((Ae.current >= f || !e.length) && (Ae.current = 0, ot({ | ||
audioData: e, | ||
unit: O, | ||
index: ve, | ||
index2: Je, | ||
canvas: z.current, | ||
picks: je.current, | ||
unit: Ne, | ||
index: Je, | ||
index2: Qe, | ||
canvas: L.current, | ||
picks: He.current, | ||
isRecordingInProgress: t, | ||
isPausedRecording: L, | ||
backgroundColor: $, | ||
mainBarColor: R, | ||
secondaryBarColor: q, | ||
barWidth: N, | ||
rounded: ne, | ||
animateCurrentPick: X, | ||
fullscreen: oe | ||
})), me.current += 1); | ||
isPausedRecording: S, | ||
backgroundColor: O, | ||
mainBarColor: D, | ||
secondaryBarColor: B, | ||
barWidth: w, | ||
rounded: J, | ||
animateCurrentPick: ne, | ||
fullscreen: Y | ||
})), Ae.current += 1); | ||
}, [ | ||
z.current, | ||
L.current, | ||
e, | ||
N, | ||
$, | ||
R, | ||
q, | ||
ne, | ||
oe, | ||
se, | ||
he | ||
w, | ||
O, | ||
D, | ||
B, | ||
J, | ||
Y, | ||
he, | ||
ce | ||
]), Z(() => { | ||
var E, k; | ||
if (f) | ||
return le ? (E = z.current) == null || E.addEventListener("mouseleave", Re) : (k = z.current) == null || k.addEventListener("mouseenter", $e), () => { | ||
var J, Fe; | ||
le ? (J = z.current) == null || J.removeEventListener( | ||
var y, W; | ||
if (z) | ||
return oe ? (y = L.current) == null || y.addEventListener("mouseleave", Oe) : (W = L.current) == null || W.addEventListener("mouseenter", $e), () => { | ||
var ze, Ue; | ||
oe ? (ze = L.current) == null || ze.removeEventListener( | ||
"mouseleave", | ||
Re | ||
) : (Fe = z.current) == null || Fe.removeEventListener( | ||
Oe | ||
) : (Ue = L.current) == null || Ue.removeEventListener( | ||
"mouseenter", | ||
@@ -426,51 +425,51 @@ $e | ||
}; | ||
}, [le, f]), Z(() => { | ||
var k; | ||
if (!g || !z.current || t || re) | ||
}, [oe, z]), Z(() => { | ||
var W; | ||
if (!M || !L.current || t || se) | ||
return; | ||
if (W) { | ||
j(); | ||
if (G) { | ||
_(); | ||
return; | ||
} | ||
je.current = []; | ||
const E = g.getChannelData(0); | ||
return qe({ | ||
bufferData: E, | ||
height: ee, | ||
width: he, | ||
barWidth: N, | ||
gap: S | ||
}), (k = z.current) == null || k.addEventListener( | ||
He.current = []; | ||
const y = M.getChannelData(0); | ||
return Xe({ | ||
bufferData: y, | ||
height: X, | ||
width: ce, | ||
barWidth: w, | ||
gap: E | ||
}), (W = L.current) == null || W.addEventListener( | ||
"mousemove", | ||
Oe | ||
Re | ||
), () => { | ||
var J; | ||
(J = z.current) == null || J.removeEventListener( | ||
var ze; | ||
(ze = L.current) == null || ze.removeEventListener( | ||
"mousemove", | ||
Oe | ||
Re | ||
); | ||
}; | ||
}, [ | ||
g, | ||
b, | ||
ee, | ||
fe, | ||
P, | ||
re | ||
M, | ||
T, | ||
X, | ||
k, | ||
te, | ||
se | ||
]), Z(() => { | ||
if (!(W || !(Ie != null && Ie.length) || !z.current || o)) { | ||
if (p) { | ||
Ve([]); | ||
if (!(G || !(fe != null && fe.length) || !L.current || o)) { | ||
if (g) { | ||
qe([]); | ||
return; | ||
} | ||
nt({ | ||
barsData: Ie, | ||
canvas: z.current, | ||
barWidth: N, | ||
gap: S, | ||
backgroundColor: $, | ||
mainBarColor: R, | ||
secondaryBarColor: q, | ||
currentAudioTime: s, | ||
rounded: ne, | ||
it({ | ||
barsData: fe, | ||
canvas: L.current, | ||
barWidth: w, | ||
gap: E, | ||
backgroundColor: O, | ||
mainBarColor: D, | ||
secondaryBarColor: B, | ||
currentAudioTime: c, | ||
rounded: J, | ||
duration: r | ||
@@ -480,51 +479,50 @@ }); | ||
}, [ | ||
Ie, | ||
s, | ||
p, | ||
ne, | ||
$, | ||
R, | ||
q | ||
fe, | ||
c, | ||
g, | ||
J, | ||
O, | ||
D, | ||
B | ||
]), Z(() => { | ||
o && z.current && He({ | ||
canvas: z.current, | ||
backgroundColor: $ | ||
o && L.current && be({ | ||
canvas: L.current, | ||
backgroundColor: O | ||
}); | ||
}, [o]); | ||
function xe() { | ||
if (!Me.current || !z.current) | ||
if (!de.current || !L.current) | ||
return; | ||
me.current = A; | ||
const E = Math.trunc( | ||
Me.current.clientHeight * window.devicePixelRatio / 2 | ||
Ae.current = f; | ||
const y = Math.trunc( | ||
de.current.clientHeight * window.devicePixelRatio / 2 | ||
) * 2; | ||
Ce(Me.current.clientWidth), ue(E), _e( | ||
Ce(de.current.clientWidth), ve(y), _e( | ||
Math.round( | ||
Me.current.clientWidth * window.devicePixelRatio | ||
de.current.clientWidth * window.devicePixelRatio | ||
) | ||
), Ae(!1); | ||
), Se(!1); | ||
} | ||
function Ke() { | ||
B(!1), F(!1); | ||
function et() { | ||
U(!1), $(!1), P != null && P.current && (P.current.src = u); | ||
} | ||
const $e = () => { | ||
te(!0); | ||
}, Re = () => { | ||
te(!1); | ||
}, Oe = (E) => { | ||
Ee(E.offsetX); | ||
}, et = (E) => { | ||
const k = Se; | ||
if (k.current && z.current) { | ||
const J = r / b * (E.clientX - z.current.getBoundingClientRect().left); | ||
k.current.currentTime = J, Q(J); | ||
K(!0); | ||
}, Oe = () => { | ||
K(!1); | ||
}, Re = (y) => { | ||
pe(y.offsetX); | ||
}, tt = (y) => { | ||
if (P != null && P.current && L.current) { | ||
const W = r / T * (y.clientX - L.current.getBoundingClientRect().left); | ||
P.current.currentTime = W, ue(W); | ||
} | ||
}, Ze = s / r * b; | ||
return /* @__PURE__ */ de("div", { className: `voice-visualizer ${be ?? ""}`, children: [ | ||
/* @__PURE__ */ de( | ||
}, Ze = c / r * T; | ||
return /* @__PURE__ */ ae("div", { className: `voice-visualizer ${le ?? ""}`, children: [ | ||
/* @__PURE__ */ ae( | ||
"div", | ||
{ | ||
className: `voice-visualizer__canvas-container ${ze ?? ""}`, | ||
ref: Me, | ||
style: { width: Ye(D) }, | ||
className: `voice-visualizer__canvas-container ${Q ?? ""}`, | ||
ref: de, | ||
style: { width: Ye(j) }, | ||
children: [ | ||
@@ -534,9 +532,9 @@ /* @__PURE__ */ a( | ||
{ | ||
ref: z, | ||
width: he, | ||
height: ee, | ||
onClick: et, | ||
ref: L, | ||
width: ce, | ||
height: X, | ||
onClick: tt, | ||
style: { | ||
height: Ye(U), | ||
width: b | ||
height: Ye(ee), | ||
width: T | ||
}, | ||
@@ -546,14 +544,14 @@ children: "Your browser does not support HTML5 Canvas." | ||
), | ||
se && p && /* @__PURE__ */ de(Ue, { children: [ | ||
/* @__PURE__ */ a(Ge, { color: Le }), | ||
/* @__PURE__ */ a(Ge, { color: Le, reflect: !0 }), | ||
he && g && /* @__PURE__ */ ae(Fe, { children: [ | ||
/* @__PURE__ */ a(Ge, { color: ge }), | ||
/* @__PURE__ */ a(Ge, { color: ge, reflect: !0 }), | ||
/* @__PURE__ */ a( | ||
"button", | ||
{ | ||
onClick: M, | ||
onClick: I, | ||
className: "voice-visualizer__canvas-microphone-btn", | ||
children: /* @__PURE__ */ a( | ||
vt, | ||
ft, | ||
{ | ||
color: Te, | ||
color: De, | ||
stroke: 0.5, | ||
@@ -566,25 +564,25 @@ className: "voice-visualizer__canvas-microphone-icon" | ||
] }), | ||
h && o && /* @__PURE__ */ a( | ||
Me && o && /* @__PURE__ */ a( | ||
"p", | ||
{ | ||
className: `voice-visualizer__canvas-audio-processing ${pe ?? ""}`, | ||
style: { color: R }, | ||
className: `voice-visualizer__canvas-audio-processing ${Te ?? ""}`, | ||
style: { color: D }, | ||
children: "Processing Audio..." | ||
} | ||
), | ||
le && f && !o && !i && ae && /* @__PURE__ */ a( | ||
oe && z && !o && !i && q && /* @__PURE__ */ a( | ||
"div", | ||
{ | ||
className: `voice-visualizer__progress-indicator-hovered ${G ?? ""}`, | ||
className: `voice-visualizer__progress-indicator-hovered ${C ?? ""}`, | ||
style: { | ||
left: ge | ||
left: me | ||
}, | ||
children: T && /* @__PURE__ */ a( | ||
children: x && /* @__PURE__ */ a( | ||
"p", | ||
{ | ||
className: `voice-visualizer__progress-indicator-hovered-time | ||
${b - ge < 70 ? "voice-visualizer__progress-indicator-hovered-time-left" : ""} | ||
${_ ?? ""}`, | ||
${T - me < 70 ? "voice-visualizer__progress-indicator-hovered-time-left" : ""} | ||
${s ?? ""}`, | ||
children: ke( | ||
r / b * ge | ||
r / T * me | ||
) | ||
@@ -595,14 +593,14 @@ } | ||
), | ||
Y && f && !o && r ? /* @__PURE__ */ a( | ||
p && z && !o && r ? /* @__PURE__ */ a( | ||
"div", | ||
{ | ||
className: `voice-visualizer__progress-indicator ${d ?? ""}`, | ||
className: `voice-visualizer__progress-indicator ${R ?? ""}`, | ||
style: { | ||
left: Ze < b - 1 ? Ze : b - 1 | ||
left: Ze < T - 1 ? Ze : T - 1 | ||
}, | ||
children: C && /* @__PURE__ */ a( | ||
children: V && /* @__PURE__ */ a( | ||
"p", | ||
{ | ||
className: `voice-visualizer__progress-indicator-time ${b - s * b / r < 70 ? "voice-visualizer__progress-indicator-time-left" : ""} ${K ?? ""}`, | ||
children: x | ||
className: `voice-visualizer__progress-indicator-time ${T - c * T / r < 70 ? "voice-visualizer__progress-indicator-time-left" : ""} ${ie ?? ""}`, | ||
children: H | ||
} | ||
@@ -615,18 +613,18 @@ ) | ||
), | ||
ie && /* @__PURE__ */ de(Ue, { children: [ | ||
/* @__PURE__ */ de("div", { className: "voice-visualizer__audio-info-container", children: [ | ||
t && /* @__PURE__ */ a("p", { className: "voice-visualizer__audio-info-time", children: m }), | ||
r && !o ? /* @__PURE__ */ a("p", { children: H }) : null | ||
ye && /* @__PURE__ */ ae(Fe, { children: [ | ||
/* @__PURE__ */ ae("div", { className: "voice-visualizer__audio-info-container", children: [ | ||
t && /* @__PURE__ */ a("p", { className: "voice-visualizer__audio-info-time", children: h }), | ||
r && !o ? /* @__PURE__ */ a("p", { children: b }) : null | ||
] }), | ||
/* @__PURE__ */ de("div", { className: "voice-visualizer__buttons-container", children: [ | ||
/* @__PURE__ */ ae("div", { className: "voice-visualizer__buttons-container", children: [ | ||
t && /* @__PURE__ */ a( | ||
"button", | ||
{ | ||
className: `voice-visualizer__btn-left ${L ? "voice-visualizer__btn-left-microphone" : ""}`, | ||
onClick: v, | ||
className: `voice-visualizer__btn-left ${S ? "voice-visualizer__btn-left-microphone" : ""}`, | ||
onClick: m, | ||
children: /* @__PURE__ */ a( | ||
"img", | ||
{ | ||
src: L ? Be : Pe, | ||
alt: L ? "Play" : "Pause" | ||
src: S ? Pe : We, | ||
alt: S ? "Play" : "Pause" | ||
} | ||
@@ -636,7 +634,7 @@ ) | ||
), | ||
!p && /* @__PURE__ */ a( | ||
!g && /* @__PURE__ */ a( | ||
"button", | ||
{ | ||
className: `voice-visualizer__btn-left ${t ? "voice-visualizer__visually-hidden" : ""}`, | ||
onClick: v, | ||
onClick: m, | ||
disabled: o, | ||
@@ -646,4 +644,4 @@ children: /* @__PURE__ */ a( | ||
{ | ||
src: w ? dt : Pe, | ||
alt: w ? "Play" : "Pause" | ||
src: A ? zt : We, | ||
alt: A ? "Play" : "Pause" | ||
} | ||
@@ -653,8 +651,8 @@ ) | ||
), | ||
p && /* @__PURE__ */ a( | ||
g && /* @__PURE__ */ a( | ||
"button", | ||
{ | ||
className: "voice-visualizer__btn-center", | ||
onClick: M, | ||
children: /* @__PURE__ */ a("img", { src: Be, alt: "Microphone" }) | ||
onClick: I, | ||
children: /* @__PURE__ */ a("img", { src: Pe, alt: "Microphone" }) | ||
} | ||
@@ -666,11 +664,11 @@ ), | ||
className: `voice-visualizer__btn-center voice-visualizer__btn-center-pause ${t ? "" : "voice-visualizer__visually-hidden"}`, | ||
onClick: u, | ||
children: /* @__PURE__ */ a("img", { src: ft, alt: "Stop" }) | ||
onClick: v, | ||
children: /* @__PURE__ */ a("img", { src: gt, alt: "Stop" }) | ||
} | ||
), | ||
!p && /* @__PURE__ */ a( | ||
!g && /* @__PURE__ */ a( | ||
"button", | ||
{ | ||
onClick: j, | ||
className: `voice-visualizer__btn ${we ?? ""}`, | ||
onClick: _, | ||
className: `voice-visualizer__btn ${Ie ?? ""}`, | ||
disabled: o, | ||
@@ -680,7 +678,7 @@ children: "Clear" | ||
), | ||
ce && n && /* @__PURE__ */ a( | ||
re && n && /* @__PURE__ */ a( | ||
"button", | ||
{ | ||
onClick: I, | ||
className: `voice-visualizer__btn ${we ?? ""}`, | ||
onClick: d, | ||
className: `voice-visualizer__btn ${Ie ?? ""}`, | ||
disabled: o, | ||
@@ -691,16 +689,7 @@ children: "Download Audio" | ||
] }) | ||
] }), | ||
f && /* @__PURE__ */ a( | ||
"audio", | ||
{ | ||
ref: Se, | ||
src: c, | ||
controls: !0, | ||
style: { display: "none" } | ||
} | ||
) | ||
] }) | ||
] }); | ||
} | ||
); | ||
function Lt({ | ||
function wt({ | ||
onStartRecording: e, | ||
@@ -711,155 +700,143 @@ onStopRecording: t, | ||
onClearCanvas: c, | ||
onEndAudioPlayback: s, | ||
onStartAudioPlayback: g, | ||
onPausedAudioPlayback: v, | ||
onResumedAudioPlayback: M | ||
onEndAudioPlayback: u, | ||
onStartAudioPlayback: M, | ||
onPausedAudioPlayback: m, | ||
onResumedAudioPlayback: I, | ||
onErrorPlayingAudio: v | ||
} = {}) { | ||
const [u, I] = l(!1), [f, w] = l(!1), [L, o] = l(null), [p, H] = l(new Uint8Array(0)), [m, x] = l(!1), [j, Q] = l(null), [F, B] = l(null), [D, U] = l(0), [V, $] = l(0), [R, q] = l(0), [P, fe] = l(""), [ne, ie] = l(!0), [ce, X] = l(0), [oe, W] = l(!0), [se, Te] = l(!1), [Le, be] = l(!1), [ze, Y] = l(null), d = y(null), C = y(null), K = y(null), ae = y(null), G = y(null), T = y(null), _ = y(null), h = y(null), pe = !!(F && !m), we = ut(R), Se = ot(D), ge = ke(ce), Ee = Le || m; | ||
const [d, z] = l(!1), [A, S] = l(!1), [o, g] = l(null), [b, h] = l(new Uint8Array(0)), [H, _] = l(!1), [ue, $] = l(null), [U, j] = l(null), [ee, F] = l(0), [O, D] = l(0), [B, te] = l(0), [k, J] = l(""), [ye, re] = l(!0), [ne, Y] = l(0), [G, he] = l(!0), [De, ge] = l(!1), [le, Q] = l(null), p = N(null), R = N(null), V = N(null), ie = N(null), q = N(null), C = N(null), x = N(null), s = N(null), Me = !!(U && !H), Te = ht(B), Ie = st(ee), Ee = ke(ne), me = De || H; | ||
Z(() => { | ||
if (!u || f) | ||
if (!d || A) | ||
return; | ||
const A = setInterval(() => { | ||
const S = performance.now(); | ||
U((N) => N + (S - V)), $(S); | ||
const f = setInterval(() => { | ||
const E = performance.now(); | ||
F((w) => w + (E - O)), D(E); | ||
}, 1e3); | ||
return () => clearInterval(A); | ||
}, [V, f, u]), Z(() => { | ||
if (!j || j.size === 0) | ||
return () => clearInterval(f); | ||
}, [O, A, d]), Z(() => { | ||
if (le) { | ||
K(); | ||
return; | ||
(async () => { | ||
var A; | ||
} | ||
}, [le]), Z(() => () => { | ||
K(); | ||
}, []), Z(() => (G || window.addEventListener("beforeunload", pe), () => { | ||
window.removeEventListener("beforeunload", pe); | ||
}), [G]); | ||
const pe = (i) => { | ||
i.preventDefault(), i.returnValue = ""; | ||
}, T = async (i) => { | ||
if (i) | ||
try { | ||
Y(null); | ||
const S = new Blob([j], { | ||
type: (A = d.current) == null ? void 0 : A.mimeType | ||
}), N = URL.createObjectURL(S); | ||
N && fe(N); | ||
const O = await j.arrayBuffer(), z = new AudioContext(), je = (ve) => { | ||
B(ve), q(ve.duration - 0.06); | ||
}, me = (ve) => { | ||
Y(ve); | ||
}; | ||
z.decodeAudioData( | ||
O, | ||
je, | ||
me | ||
if (i.size === 0) | ||
throw new Error("Error: The audio blob is empty"); | ||
const f = URL.createObjectURL(i); | ||
J(f); | ||
const E = await i.arrayBuffer(), Ne = await new AudioContext().decodeAudioData(E); | ||
j(Ne), te(Ne.duration - 0.06), Q(null); | ||
} catch (f) { | ||
console.error("Error processing the audio blob:", f), Q( | ||
f instanceof Error ? f : new Error("Error processing the audio blob") | ||
); | ||
} catch (S) { | ||
if (console.error("Error processing the audio blob:", S), S instanceof Error) { | ||
Y(S); | ||
return; | ||
} | ||
Y(new Error("Error processing the audio blob")); | ||
} | ||
})(); | ||
}, [j]), Z(() => { | ||
if (ze) { | ||
te(); | ||
return; | ||
} | ||
}, [ze]), Z(() => () => { | ||
_.current && cancelAnimationFrame(_.current), G.current && G.current.disconnect(), C.current && C.current.state !== "closed" && C.current.close(), T.current && cancelAnimationFrame(T.current), h != null && h.current && h.current.removeEventListener("ended", re), d.current && d.current.removeEventListener( | ||
"dataavailable", | ||
ue | ||
); | ||
}, []), Z(() => (!oe && !se && window.addEventListener("beforeunload", b), () => { | ||
window.removeEventListener("beforeunload", b); | ||
}), [oe, se]); | ||
const b = (i) => { | ||
i.preventDefault(), i.returnValue = ""; | ||
}, Ce = () => { | ||
navigator.mediaDevices.getUserMedia({ audio: !0 }).then((i) => { | ||
te(), W(!1), $(performance.now()), I(!0), o(i), C.current = new window.AudioContext(), K.current = C.current.createAnalyser(), ae.current = new Uint8Array( | ||
K.current.frequencyBinCount | ||
), G.current = C.current.createMediaStreamSource(i), G.current.connect(K.current), d.current = new MediaRecorder(i), d.current.addEventListener( | ||
D(performance.now()), z(!0), g(i), R.current = new window.AudioContext(), V.current = R.current.createAnalyser(), ie.current = new Uint8Array( | ||
V.current.frequencyBinCount | ||
), q.current = R.current.createMediaStreamSource(i), q.current.connect(V.current), p.current = new MediaRecorder(i), p.current.addEventListener( | ||
"dataavailable", | ||
ue | ||
), d.current.start(), ee(); | ||
ve | ||
), p.current.start(), X(); | ||
}).catch((i) => { | ||
if (console.error("Error starting audio recording:", i), i instanceof Error) { | ||
Y(i); | ||
Q(i); | ||
return; | ||
} | ||
Y(new Error("Error starting audio recording")); | ||
Q(new Error("Error starting audio recording")); | ||
}); | ||
}, ee = () => { | ||
K.current.getByteTimeDomainData(ae.current), H(new Uint8Array(ae.current)), T.current = requestAnimationFrame(ee); | ||
}, ue = (i) => { | ||
d.current && Q(i.data); | ||
}, he = () => { | ||
_.current && cancelAnimationFrame(_.current), h.current && (X(h.current.currentTime), _.current = requestAnimationFrame(he)); | ||
}, X = () => { | ||
V.current.getByteTimeDomainData(ie.current), h(new Uint8Array(ie.current)), C.current = requestAnimationFrame(X); | ||
}, ve = (i) => { | ||
s.current = new Audio(), $(i.data), T(i.data); | ||
}, ce = () => { | ||
s.current && (Y(s.current.currentTime), x.current = requestAnimationFrame(ce)); | ||
}, _e = () => { | ||
u || (e && e(), Ce()); | ||
}, le = () => { | ||
u && (t && t(), x(!0), I(!1), U(0), w(!1), T.current && cancelAnimationFrame(T.current), G.current && G.current.disconnect(), C.current && C.current.state !== "closed" && C.current.close(), L == null || L.getTracks().forEach((i) => i.stop()), d.current && (d.current.stop(), d.current.removeEventListener( | ||
K(), he(!1), !d && (e && e(), Ce()); | ||
}, oe = () => { | ||
d && (p.current && (p.current.stop(), p.current.removeEventListener( | ||
"dataavailable", | ||
ue | ||
))); | ||
}, te = () => { | ||
T.current && cancelAnimationFrame(T.current), h != null && h.current && h.current.removeEventListener("ended", re), _.current && cancelAnimationFrame(_.current), d.current && (d.current.removeEventListener( | ||
ve | ||
), p.current = null), o == null || o.getTracks().forEach((i) => i.stop()), t && t(), C.current && cancelAnimationFrame(C.current), q.current && q.current.disconnect(), R.current && R.current.state !== "closed" && R.current.close(), _(!0), z(!1), F(0), S(!1)); | ||
}, K = () => { | ||
C.current && cancelAnimationFrame(C.current), x.current && cancelAnimationFrame(x.current), p.current && (p.current.removeEventListener( | ||
"dataavailable", | ||
ue | ||
), d.current.stop(), d.current = null), L == null || L.getTracks().forEach((i) => i.stop()), d.current = null, C.current = null, K.current = null, ae.current = null, G.current = null, T.current = null, _.current = null, c && c(), o(null), I(!1), x(!1), Q(null), B(null), U(0), $(0), q(0), fe(""), X(0), ie(!0), w(!1), H(new Uint8Array(0)), Y(null), W(!0); | ||
}, Ne = (i) => { | ||
i instanceof Blob && (te(), Te(!0), W(!1), x(!0), I(!1), U(0), w(!1), Q(i)); | ||
}, ye = () => { | ||
var i, A, S, N; | ||
if (u) { | ||
w((O) => !O), ((i = d.current) == null ? void 0 : i.state) === "recording" ? (n && n(), (A = d.current) == null || A.pause(), U((O) => O + (performance.now() - V)), T.current && cancelAnimationFrame(T.current)) : (r && r(), (S = d.current) == null || S.resume(), $(performance.now()), T.current = requestAnimationFrame(ee)); | ||
ve | ||
), p.current.stop(), p.current = null), o == null || o.getTracks().forEach((i) => i.stop()), s != null && s.current && (s.current.removeEventListener("ended", se), s.current.pause(), s.current.src = "", s.current = null), p.current = null, R.current = null, V.current = null, ie.current = null, q.current = null, C.current = null, x.current = null, c && c(), g(null), z(!1), _(!1), $(null), j(null), F(0), D(0), te(0), J(""), Y(0), re(!0), S(!1), h(new Uint8Array(0)), Q(null), he(!0); | ||
}, we = () => { | ||
if (s.current && s.current.paused) { | ||
const i = s.current.play(); | ||
i !== void 0 && i.catch((f) => { | ||
console.error(f), v && v( | ||
f instanceof Error ? f : new Error("Error playing audio") | ||
); | ||
}); | ||
} | ||
}, Le = () => { | ||
var i, f, E; | ||
if (d) { | ||
S((w) => !w), ((i = p.current) == null ? void 0 : i.state) === "recording" ? (n && n(), (f = p.current) == null || f.pause(), F((w) => w + (performance.now() - O)), C.current && cancelAnimationFrame(C.current)) : (r && r(), (E = p.current) == null || E.resume(), D(performance.now()), C.current = requestAnimationFrame(X)); | ||
return; | ||
} | ||
if (h.current && pe) | ||
if (_.current && cancelAnimationFrame(_.current), h.current.paused) | ||
g && ce === 0 && g(), M && ce !== 0 && M(), h.current.addEventListener("ended", re), he(), ie(!1), (N = h.current) == null || N.play(); | ||
if (s.current && Me) | ||
if (s.current.paused) | ||
M && ne === 0 && M(), I && ne !== 0 && I(), requestAnimationFrame(ce), s.current.addEventListener("ended", se), we(), re(!1); | ||
else { | ||
v && v(), h.current.removeEventListener("ended", re), h.current.pause(), ie(!0); | ||
const O = h.current.currentTime; | ||
X(O), h.current.currentTime = O; | ||
x.current && cancelAnimationFrame(x.current), m && m(), s.current.removeEventListener("ended", se), s.current.pause(), re(!0); | ||
const w = s.current.currentTime; | ||
Y(w), s.current.currentTime = w; | ||
} | ||
}, re = () => { | ||
ie(!0), s && s(), h != null && h.current && (h.current.currentTime = 0, X(0)); | ||
}, Ae = () => { | ||
var A; | ||
if (!P) | ||
}, se = () => { | ||
x.current && cancelAnimationFrame(x.current), re(!0), u && u(), s != null && s.current && (s.current.currentTime = 0, Y(0)); | ||
}, Se = () => { | ||
var f; | ||
if (!k) | ||
return; | ||
const i = document.createElement("a"); | ||
i.href = P, i.download = `recorded_audio${at( | ||
(A = d.current) == null ? void 0 : A.mimeType | ||
)}`, document.body.appendChild(i), i.click(), document.body.removeChild(i), URL.revokeObjectURL(P); | ||
i.href = k, i.download = `recorded_audio${ut( | ||
(f = p.current) == null ? void 0 : f.mimeType | ||
)}`, document.body.appendChild(i), i.click(), document.body.removeChild(i), URL.revokeObjectURL(k); | ||
}; | ||
return { | ||
isRecordingInProgress: u, | ||
isPausedRecording: f, | ||
audioData: p, | ||
recordingTime: D, | ||
isProcessingRecordedAudio: Ee, | ||
recordedBlob: j, | ||
mediaRecorder: d.current, | ||
duration: R, | ||
currentAudioTime: ce, | ||
audioSrc: P, | ||
isPausedRecordedAudio: ne, | ||
bufferFromRecordedBlob: F, | ||
isCleared: oe, | ||
isAvailableRecordedAudio: pe, | ||
isPreloadedBlob: se, | ||
formattedDuration: we, | ||
formattedRecordingTime: Se, | ||
formattedRecordedAudioCurrentTime: ge, | ||
setPreloadedAudioBlob: Ne, | ||
audioRef: s, | ||
isRecordingInProgress: d, | ||
isPausedRecording: A, | ||
audioData: b, | ||
recordingTime: ee, | ||
isProcessingRecordedAudio: me, | ||
recordedBlob: ue, | ||
mediaRecorder: p.current, | ||
duration: B, | ||
currentAudioTime: ne, | ||
audioSrc: k, | ||
isPausedRecordedAudio: ye, | ||
bufferFromRecordedBlob: U, | ||
isCleared: G, | ||
isAvailableRecordedAudio: Me, | ||
formattedDuration: Te, | ||
formattedRecordingTime: Ie, | ||
formattedRecordedAudioCurrentTime: Ee, | ||
startRecording: _e, | ||
togglePauseResume: ye, | ||
stopRecording: le, | ||
saveAudioFile: Ae, | ||
clearCanvas: te, | ||
setCurrentAudioTime: X, | ||
error: ze, | ||
_setIsProcessingAudioOnComplete: x, | ||
_setIsProcessingOnResize: be, | ||
audioRef: h | ||
togglePauseResume: Le, | ||
stopRecording: oe, | ||
saveAudioFile: Se, | ||
clearCanvas: K, | ||
setCurrentAudioTime: Y, | ||
error: le, | ||
_setIsProcessingAudioOnComplete: _, | ||
_setIsProcessingOnResize: ge | ||
}; | ||
} | ||
export { | ||
It as VoiceVisualizer, | ||
Lt as useVoiceVisualizer | ||
pt as VoiceVisualizer, | ||
wt as useVoiceVisualizer | ||
}; |
@@ -7,2 +7,3 @@ import { Dispatch, MutableRefObject, SetStateAction } from "react"; | ||
export interface Controls { | ||
audioRef: MutableRefObject<HTMLAudioElement | null>; | ||
isRecordingInProgress: boolean; | ||
@@ -20,4 +21,2 @@ isPausedRecording: boolean; | ||
isAvailableRecordedAudio: boolean; | ||
isPreloadedBlob: boolean; | ||
setPreloadedAudioBlob: (blob: unknown) => void; | ||
recordedBlob: Blob | null; | ||
@@ -37,3 +36,2 @@ bufferFromRecordedBlob: AudioBuffer | null; | ||
_setIsProcessingOnResize: Dispatch<SetStateAction<boolean>>; | ||
audioRef: MutableRefObject<HTMLAudioElement | null>; | ||
} | ||
@@ -110,2 +108,3 @@ export interface BarsData { | ||
onResumedAudioPlayback?: () => void; | ||
onErrorPlayingAudio?: (error: Error) => void; | ||
} | ||
@@ -112,0 +111,0 @@ export interface UseWebWorkerParams<T> { |
{ | ||
"name": "react-voice-visualizer", | ||
"private": false, | ||
"version": "1.3.8", | ||
"version": "1.7.4", | ||
"type": "module", | ||
@@ -6,0 +6,0 @@ "author": "Yurii Zarytskyi", |
148
README.md
@@ -47,3 +47,3 @@ # react-voice-visualizer [Demo App](https://react-voice-visualizer.vercel.app/) | ||
```jsx | ||
```typescript jsx | ||
import { useEffect } from "react"; | ||
@@ -73,7 +73,7 @@ import { useVoiceVisualizer, VoiceVisualizer } from "react-voice-visualizer"; | ||
console.log(error); | ||
console.error(error); | ||
}, [error]); | ||
return ( | ||
<VoiceVisualizer controls={recorderControls} ref={audioRef}/> | ||
<VoiceVisualizer ref={audioRef} controls={recorderControls} /> | ||
); | ||
@@ -85,49 +85,2 @@ }; | ||
Additionally, you can use the setPreloadedAudioBlob function to load any audio data. Pass your audio data in a `Blob` format to this function: | ||
``` | ||
setPreloadedAudioBlob(audioBlob); | ||
``` | ||
Example: | ||
```jsx | ||
import { useEffect } from 'react'; | ||
import { useVoiceVisualizer, VoiceVisualizer } from 'react-voice-visualizer'; | ||
const App = () => { | ||
const recorderControls = useVoiceVisualizer(); | ||
const { | ||
// ... (Extracted controls and states, if necessary) | ||
setPreloadedAudioBlob, | ||
isPreloadedBlob, | ||
error, | ||
audioRef | ||
} = recorderControls; | ||
useEffect(() => { | ||
// Set the preloaded audioBlob when the component mounts | ||
// Assuming 'audioBlob' is defined somewhere | ||
if (audioBlob) { | ||
setPreloadedAudioBlob(audioBlob); | ||
} | ||
}, [audioBlob]); | ||
// Get and log any error when it occurs | ||
useEffect(() => { | ||
if (!error) return; | ||
console.log(error); | ||
}, [error]); | ||
return ( | ||
<VoiceVisualizer | ||
isControlPanelShown={false} // Set to 'false' in most cases, but should be determined based on the specific user's use case. | ||
controls={recorderControls} | ||
ref={audioRef} | ||
/> | ||
); | ||
}; | ||
export default App; | ||
``` | ||
## Getting started | ||
@@ -157,56 +110,47 @@ | ||
| Parameter | Type | Description | | ||
|:-------------------------|:----------------------------------------------------|:--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------| | ||
| `onStartRecording` | `() => void` | Callback function triggered when recording starts. | | ||
| `onStopRecording` | `() => void` | Callback function triggered when recording stops. | | ||
| `onPausedRecording` | `() => void` | Callback function triggered when recording is paused. | | ||
| `onResumedRecording` | `() => void` | Callback function triggered when recording is resumed. | | ||
| `onClearCanvas` | `() => void` | Callback function triggered when the canvas is cleared. | | ||
| `onEndAudioPlayback` | `() => void` | Callback function triggered when audio playback ends. | | ||
| `onStartAudioPlayback` | `() => void` | Callback function triggered when audio playback starts. | | ||
| `onPausedAudioPlayback` | `() => void` | Callback function triggered when audio playback is paused. | | ||
| `onResumedAudioPlayback` | `() => void` | Callback function triggered when audio playback is resumed. | | ||
| Parameter | Type | Description | | ||
|:-------------------------|:-------------------------|:----------------------------------------------------------------------------------------------------------------------------------------------------------| | ||
| `onStartRecording` | `() => void` | Callback function triggered when recording starts. | | ||
| `onStopRecording` | `() => void` | Callback function triggered when recording stops. | | ||
| `onPausedRecording` | `() => void` | Callback function triggered when recording is paused. | | ||
| `onResumedRecording` | `() => void` | Callback function triggered when recording is resumed. | | ||
| `onClearCanvas` | `() => void` | Callback function triggered when the canvas is cleared. | | ||
| `onEndAudioPlayback` | `() => void` | Callback function triggered when audio playback ends. | | ||
| `onStartAudioPlayback` | `() => void` | Callback function triggered when audio playback starts. | | ||
| `onPausedAudioPlayback` | `() => void` | Callback function triggered when audio playback is paused. | | ||
| `onResumedAudioPlayback` | `() => void` | Callback function triggered when audio playback is resumed. | | ||
| `onErrorPlayingAudio` | `(error: Error) => void` | Callback function is invoked when an error occurs during the execution of `audio.play()`. It provides an opportunity to handle and respond to such error. | | ||
##### Returns | ||
| Returns | Type | Description | | ||
|:---------------------------------------------------------------|:----------------------------------------------------|:--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------| | ||
| `audioRef` | `MutableRefObject`<br/>`<HTMLAudioElement \| null>` | Reference to the audio element used for playback. | | ||
| `isRecordingInProgress` | `boolean` | Indicates if audio recording is currently in progress. | | ||
| `isPausedRecording` | `boolean` | Indicates if audio recording is currently paused. | | ||
| `audioData` | `Uint8Array` | Audio data for real-time visualization. | | ||
| `recordingTime` | `number` | Elapsed time during recording in miliseconds. | | ||
| `mediaRecorder` | `MediaRecorder \| null` | MediaRecorder instance used for recording audio. | | ||
| `duration` | `number` | Duration of the recorded audio in seconds. | | ||
| `currentAudioTime` | `number` | Current playback time of the recorded audio in seconds. | | ||
| `audioSrc` | `string` | Source URL of the recorded audio file for playback. | | ||
| `isPausedRecordedAudio` | `boolean` | Indicates if recorded audio playback is paused. | | ||
| `isProcessingRecordedAudio` | `boolean` | Indicates if the recorded audio is being processed and 'Processing Audio...' text shown. | | ||
| `isCleared` | `boolean` | Indicates if the canvas has been cleared. | | ||
| `isPreloadedBlob` | `boolean` | Indicates whether a blob of recorded audio data has been preloaded. | | ||
| `isAvailableRecordedAudio` | `boolean` | Indicates whether recorded audi is available and not currently being processed. This return value can be used to check if it's an appropriate time to work with recorded audio data in your application. | | ||
| `recordedBlob` | `Blob \| null` | Recorded audio data in Blob format. | | ||
| `bufferFromRecordedBlob` | `AudioBuffer \| null` | Audio buffer from the recorded Blob. | | ||
| `formattedDuration` | `string` | Formatted duration time in format 09:51m. | | ||
| `formattedRecordingTime` | `string` | Formatted recording current time in format 09:51. | | ||
| `formattedRecordedAudioCurrentTime` | `string` | Formatted recorded audio current time in format 09:51:1. | | ||
| `setPreloadedAudioBlob` | `(audioBlob: Blob) => void` | This function allows you to load an existing audio blob for further processing, playback and visualization. The `audioBlob` parameter represents the recorded audio data stored in a Blob format. | | ||
| `startRecording` | `() => void` | Function to start audio recording. | | ||
| `togglePauseResume` | `() => void` | Function to toggle pause/resume during recording and playback of recorded audio. | | ||
| `stopRecording` | `() => void` | Function to stop audio recording. | | ||
| `saveAudioFile` | `() => void` | This function allows you to save the recorded audio as a `webm` file format. Please note that it supports saving audio only in the webm format. If you need to save the audio in a different format, you can use external libraries like FFmpeg to convert the Blob to your desired format. This flexibility allows you to tailor the output format according to your specific needs. | | ||
| `clearCanvas` | `() => void` | Function to clear the visualization canvas. | | ||
| `setCurrentAudioTime` | `Dispatch<SetStateAction<number>>` | Internal function to handle current audio time updates during playback. | | ||
| `error` | `Error \| null` | Error object if any error occurred during recording or playback. | | ||
| `_setIsProcessingAudioOnComplete` | `Dispatch<SetStateAction<boolean>>` | Internal function to set IsProcessingAudioOnComplete state. | | ||
| `_setIsProcessingOnResize` | `Dispatch<SetStateAction<boolean>>` | Internal function to set IsProcessingOnResize state. | | ||
| Returns | Type | Description | | ||
|:----------------------------------------|:-------------------------------------------------------|:--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------| | ||
| `audioRef` | `MutableRefObject`<br/>`<HTMLAudioElement \| null>` | Reference to the audio element used for playback. | | ||
| `isRecordingInProgress` | `boolean` | Indicates if audio recording is currently in progress. | | ||
| `isPausedRecording` | `boolean` | Indicates if audio recording is currently paused. | | ||
| `audioData` | `Uint8Array` | Audio data for real-time visualization. | | ||
| `recordingTime` | `number` | Elapsed time during recording in milliseconds. | | ||
| `mediaRecorder` | `MediaRecorder \| null` | MediaRecorder instance used for recording audio. | | ||
| `duration` | `number` | Duration of the recorded audio in seconds. | | ||
| `currentAudioTime` | `number` | Current playback time of the recorded audio in seconds. | | ||
| `audioSrc` | `string` | Source URL of the recorded audio file for playback. | | ||
| `isPausedRecordedAudio` | `boolean` | Indicates if recorded audio playback is paused. | | ||
| `isProcessingRecordedAudio` | `boolean` | Indicates if the recorded audio is being processed and 'Processing Audio...' text shown. | | ||
| `isCleared` | `boolean` | Indicates if the canvas has been cleared. | | ||
| `isAvailableRecordedAudio` | `boolean` | Indicates whether recorded audi is available and not currently being processed. This return value can be used to check if it's an appropriate time to work with recorded audio data in your application. | | ||
| `recordedBlob` | `Blob \| null` | Recorded audio data in Blob format. | | ||
| `bufferFromRecordedBlob` | `AudioBuffer \| null` | Audio buffer from the recorded Blob. | | ||
| `formattedDuration` | `string` | Formatted duration time in format 09:51m. | | ||
| `formattedRecordingTime` | `string` | Formatted recording current time in format 09:51. | | ||
| `formattedRecordedAudioCurrentTime` | `string` | Formatted recorded audio current time in format 09:51:1. | | ||
| `startRecording` | `() => void` | Function to start audio recording. | | ||
| `togglePauseResume` | `() => void` | Function to toggle pause/resume during recording and playback of recorded audio. | | ||
| `stopRecording` | `() => void` | Function to stop audio recording. | | ||
| `saveAudioFile` | `() => void` | This function allows you to save the recorded audio as a `webm` file format. Please note that it supports saving audio only in the webm format. If you need to save the audio in a different format, you can use external libraries like `FFmpeg` to convert the Blob to your desired format. This flexibility allows you to tailor the output format according to your specific needs. | | ||
| `clearCanvas` | `() => void` | Function to clear the visualization canvas. | | ||
| `setCurrentAudioTime` | `Dispatch<SetStateAction<number>>` | Internal function to handle current audio time updates during playback. | | ||
| `error` | `Error \| null` | Error object if any error occurred during recording or playback. | | ||
| `_setIsProcessingAudioOnComplete` | `Dispatch<SetStateAction<boolean>>` | Internal function to set `isProcessingAudioOnComplete` state. | | ||
| `_setIsProcessingOnResize` | `Dispatch<SetStateAction<boolean>>` | Internal function to set `isProcessingOnResize` state. | | ||
#### Load and visualize any Audio | ||
You can use the setPreloadedAudioBlob function to load any audio data. Pass your audio data as a Blob to this function: | ||
``` | ||
setPreloadedAudioBlob(audioBlob); | ||
``` | ||
### `VoiceVisualizer` Component | ||
@@ -220,4 +164,4 @@ | ||
|:--------------------------------------------------|:--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|:--------------|:-----------------------------| | ||
| **`ref`** | A reference to the audio element - `audioRef` from the `useVoiceVisualizer` hook. | - | `React.RefObject` (Required) | | ||
| **`controls`** | Provides the audio recording controls and states required for visualization. | - | `Controls` (Required) | | ||
| **`ref`** | A reference to the audio element - `audioRef` from the `useVoiceVisualizer` hook. | - | `React.RefObject` (Required) | | ||
| **`height`** | The height of the visualization canvas. | `200` | `string \| number` (Optional) | | ||
@@ -273,3 +217,3 @@ | **`width`** | The width of the visualization canvas. | `100%` | `string \| number` (Optional) | | ||
This library was created by [Yurii Zarytskyi](https://github.com/YZarytskyi) | ||
This library was created by [Yurii Zarytskyi](https://github.com/YZarytskyi) | ||
@@ -276,0 +220,0 @@ <a href="https://www.linkedin.com/in/yurii-zarytskyi/" rel="nofollow noreferrer"> |
Sorry, the diff of this file is not supported yet
115736
-5.22%1172
-2.17%217
-20.51%