@lucaismyname/ginger 0.0.31 → 0.0.32
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/client.cjs +1 -1
- package/dist/client.js +37 -36
- package/dist/client.js.map +1 -1
- package/dist/equalizer/index.cjs +1 -1
- package/dist/equalizer/index.cjs.map +1 -1
- package/dist/equalizer/index.js +16 -15
- package/dist/equalizer/index.js.map +1 -1
- package/dist/index.cjs +1 -1
- package/dist/index.js +37 -36
- package/dist/index.js.map +1 -1
- package/dist/liveAudioGraph-0cpHD_Ic.cjs +2 -0
- package/dist/liveAudioGraph-0cpHD_Ic.cjs.map +1 -0
- package/dist/liveAudioGraph-DvPaxBCP.js +105 -0
- package/dist/liveAudioGraph-DvPaxBCP.js.map +1 -0
- package/dist/remote/index.cjs +2 -0
- package/dist/remote/index.cjs.map +1 -0
- package/dist/remote/index.d.ts +5 -0
- package/dist/remote/index.d.ts.map +1 -0
- package/dist/remote/index.js +149 -0
- package/dist/remote/index.js.map +1 -0
- package/dist/remote/remoteProtocol.d.ts +28 -0
- package/dist/remote/remoteProtocol.d.ts.map +1 -0
- package/dist/remote/useGingerRemote.d.ts +35 -0
- package/dist/remote/useGingerRemote.d.ts.map +1 -0
- package/dist/spatial/index.cjs +2 -0
- package/dist/spatial/index.cjs.map +1 -0
- package/dist/spatial/index.d.ts +3 -0
- package/dist/spatial/index.d.ts.map +1 -0
- package/dist/spatial/index.js +59 -0
- package/dist/spatial/index.js.map +1 -0
- package/dist/spatial/useGingerSpatialAudio.d.ts +34 -0
- package/dist/spatial/useGingerSpatialAudio.d.ts.map +1 -0
- package/dist/spatial/useGingerSpatialAudio.test.d.ts +2 -0
- package/dist/spatial/useGingerSpatialAudio.test.d.ts.map +1 -0
- package/dist/testing/mockWebAudio.d.ts +14 -0
- package/dist/testing/mockWebAudio.d.ts.map +1 -1
- package/dist/transcript/index.cjs +8 -0
- package/dist/transcript/index.cjs.map +1 -0
- package/dist/transcript/index.d.ts +5 -0
- package/dist/transcript/index.d.ts.map +1 -0
- package/dist/transcript/index.js +99 -0
- package/dist/transcript/index.js.map +1 -0
- package/dist/transcript/parseTranscript.d.ts +27 -0
- package/dist/transcript/parseTranscript.d.ts.map +1 -0
- package/dist/transcript/parseTranscript.test.d.ts +2 -0
- package/dist/transcript/parseTranscript.test.d.ts.map +1 -0
- package/dist/transcript/useGingerTranscriptSync.d.ts +23 -0
- package/dist/transcript/useGingerTranscriptSync.d.ts.map +1 -0
- package/dist/useGinger-BXgia32v.cjs +2 -0
- package/dist/useGinger-BXgia32v.cjs.map +1 -0
- package/dist/useGinger-hpp2pAGY.js +48 -0
- package/dist/useGinger-hpp2pAGY.js.map +1 -0
- package/dist/useGingerChapterProgress-BdaalJvX.cjs +2 -0
- package/dist/{useGingerChapterProgress-BOqUimE7.cjs.map → useGingerChapterProgress-BdaalJvX.cjs.map} +1 -1
- package/dist/{useGingerChapterProgress-DLYdGytK.js → useGingerChapterProgress-CZdv-HiI.js} +23 -22
- package/dist/{useGingerChapterProgress-DLYdGytK.js.map → useGingerChapterProgress-CZdv-HiI.js.map} +1 -1
- package/package.json +17 -2
- package/dist/liveAudioGraph-CmEsdLgZ.js +0 -150
- package/dist/liveAudioGraph-CmEsdLgZ.js.map +0 -1
- package/dist/liveAudioGraph-D1BXMv_u.cjs +0 -2
- package/dist/liveAudioGraph-D1BXMv_u.cjs.map +0 -1
- package/dist/useGingerChapterProgress-BOqUimE7.cjs +0 -2
package/dist/{useGingerChapterProgress-DLYdGytK.js.map → useGingerChapterProgress-CZdv-HiI.js.map}
RENAMED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"useGingerChapterProgress-DLYdGytK.js","sources":["../src/analyzer/useGingerLiveAnalyzer.ts","../src/hooks/useGingerKeyboardShortcuts.ts","../src/hooks/useGingerSleepTimer.ts","../src/hooks/useGingerDebugLog.ts","../src/hooks/useSeekDrag.ts","../src/hooks/useNextTrackPrefetch.ts","../src/store.ts","../src/hooks/useGingerPlaybackHistory.ts","../src/hooks/useGingerVolumeFade.ts","../src/hooks/useGingerChapterProgress.ts"],"sourcesContent":["import { useCallback, useLayoutEffect, useMemo, useRef, useState } from \"react\";\nimport { useGinger } from \"../hooks/useGinger\";\nimport { type LiveAnalyserOptions, attachLiveAnalyser, detachLiveAnalyser } from \"./liveAudioGraph\";\n\nexport type UseGingerLiveAnalyzerOptions = {\n /** When false, the analyser is detached and no frames are read. Default true. */\n enabled?: boolean;\n fftSize?: number;\n smoothingTimeConstant?: number;\n minDecibels?: number;\n maxDecibels?: number;\n};\n\nexport type UseGingerLiveAnalyzerResult = {\n /** Byte frequency data (0–255); length equals `frequencyBinCount`. Updated each animation frame while enabled. */\n frequencyData: Uint8Array;\n /** Byte time-domain data (0–255); length equals `fftSize`. */\n timeDomainData: Uint8Array;\n frequencyBinCount: number;\n sampleRate: number;\n isSuspended: boolean;\n error: string | null;\n resume: () => Promise<void>;\n /**\n * Monotonically increasing counter incremented each animation frame.\n * Use as a `useMemo` / `useEffect` dependency to respond to new audio data, since\n * `frequencyData` and `timeDomainData` are mutated in-place (their reference never changes).\n */\n frame: number;\n};\n\nconst emptyFreq = new Uint8Array(0);\nconst emptyTime = new Uint8Array(0);\n\nexport function useGingerLiveAnalyzer(\n options: UseGingerLiveAnalyzerOptions = {},\n): UseGingerLiveAnalyzerResult {\n const {\n enabled = true,\n fftSize = 2048,\n smoothingTimeConstant = 0.8,\n minDecibels = -100,\n maxDecibels = -30,\n } = options;\n\n const { audioRef, state } = useGinger();\n const opts = useMemo<LiveAnalyserOptions>(\n () => ({\n fftSize,\n smoothingTimeConstant,\n minDecibels,\n maxDecibels,\n }),\n [fftSize, smoothingTimeConstant, minDecibels, maxDecibels],\n );\n\n const [frame, setFrame] = useState(0);\n const [error, setError] = useState<string | null>(null);\n const [isSuspended, setIsSuspended] = useState(false);\n const [meta, setMeta] = useState({ frequencyBinCount: 0, sampleRate: 0 });\n\n const freqRef = useRef<Uint8Array>(emptyFreq);\n const timeRef = useRef<Uint8Array>(emptyTime);\n\n const resume = useCallback(async () => {\n const ctx = contextHolderRef.current;\n if (ctx && ctx.state === \"suspended\") {\n await ctx.resume();\n }\n }, []);\n\n const contextHolderRef = useRef<AudioContext | null>(null);\n const analyserHolderRef = useRef<AnalyserNode | null>(null);\n\n useLayoutEffect(() => {\n if (!enabled || typeof window === \"undefined\") {\n return;\n }\n\n let cancelled = false;\n let consumerId: number | null = null;\n let element: HTMLAudioElement | null = null;\n let rafId = 0;\n\n const onStateChange = () => {\n const ctx = contextHolderRef.current;\n if (ctx) setIsSuspended(ctx.state === \"suspended\");\n };\n\n const runLoop = () => {\n if (cancelled) return;\n const a = analyserHolderRef.current;\n const fq = freqRef.current;\n const td = timeRef.current;\n if (a && fq.length > 0 && td.length > 0) {\n a.getByteFrequencyData(fq as Uint8Array<ArrayBuffer>);\n a.getByteTimeDomainData(td as Uint8Array<ArrayBuffer>);\n setFrame((n) => n + 1);\n }\n rafId = requestAnimationFrame(runLoop);\n };\n\n type AttachOutcome = \"ok\" | \"no-element\" | \"error\";\n\n const attach = (): AttachOutcome => {\n const el = audioRef.current;\n if (!el || cancelled) return \"no-element\";\n try {\n const { id, context, analyser } = attachLiveAnalyser(el, opts);\n consumerId = id;\n element = el;\n contextHolderRef.current = context;\n analyserHolderRef.current = analyser;\n setIsSuspended(context.state === \"suspended\");\n setError(null);\n\n context.addEventListener(\"statechange\", onStateChange);\n\n const n = analyser.frequencyBinCount;\n const fft = analyser.fftSize;\n freqRef.current = new Uint8Array(n);\n timeRef.current = new Uint8Array(fft);\n setMeta({ frequencyBinCount: n, sampleRate: context.sampleRate });\n\n rafId = requestAnimationFrame(runLoop);\n return \"ok\";\n } catch (e) {\n const msg = e instanceof Error ? e.message : \"Failed to attach live analyser\";\n setError(msg);\n contextHolderRef.current = null;\n analyserHolderRef.current = null;\n freqRef.current = emptyFreq;\n timeRef.current = emptyTime;\n setMeta({ frequencyBinCount: 0, sampleRate: 0 });\n return \"error\";\n }\n };\n\n const first = attach();\n if (first !== \"ok\") {\n let retryRaf = 0;\n const maxAttempts = 120;\n let attempts = 0;\n\n const retryLoop = () => {\n if (cancelled) return;\n const out = attach();\n if (out === \"ok\" || out === \"error\") return;\n attempts += 1;\n if (attempts >= maxAttempts) return;\n retryRaf = requestAnimationFrame(retryLoop);\n };\n\n if (first === \"no-element\") {\n retryRaf = requestAnimationFrame(retryLoop);\n }\n\n return () => {\n cancelled = true;\n cancelAnimationFrame(retryRaf);\n cancelAnimationFrame(rafId);\n if (consumerId != null && element) {\n detachLiveAnalyser(element, consumerId);\n }\n contextHolderRef.current?.removeEventListener(\"statechange\", onStateChange);\n contextHolderRef.current = null;\n analyserHolderRef.current = null;\n freqRef.current = emptyFreq;\n timeRef.current = emptyTime;\n };\n }\n\n return () => {\n cancelled = true;\n cancelAnimationFrame(rafId);\n if (consumerId != null && element) {\n detachLiveAnalyser(element, consumerId);\n }\n contextHolderRef.current?.removeEventListener(\"statechange\", onStateChange);\n contextHolderRef.current = null;\n analyserHolderRef.current = null;\n freqRef.current = emptyFreq;\n timeRef.current = emptyTime;\n setMeta({ frequencyBinCount: 0, sampleRate: 0 });\n };\n }, [enabled, audioRef, opts, state.currentIndex]);\n\n return {\n frequencyData: freqRef.current,\n timeDomainData: timeRef.current,\n frequencyBinCount: meta.frequencyBinCount,\n sampleRate: meta.sampleRate,\n isSuspended,\n error,\n resume,\n frame,\n };\n}\n","import { useEffect } from \"react\";\nimport { useGingerMedia, useGingerPlayback } from \"../context/GingerSplitContexts\";\n\nexport type GingerKeyboardShortcutBindings = {\n playPause?: string;\n next?: string;\n previous?: string;\n mute?: string;\n /** Key to seek forward by `seekSeconds`. Default: undefined (disabled). */\n seekForward?: string;\n /** Key to seek backward by `seekSeconds`. Default: undefined (disabled). */\n seekBackward?: string;\n /** Seconds to seek per `seekForward` / `seekBackward` keypress. Default: 5. */\n seekSeconds?: number;\n};\n\nexport function useGingerKeyboardShortcuts(\n enabled = true,\n bindings: GingerKeyboardShortcutBindings = {},\n): void {\n const { togglePlayPause, next, prev } = useGingerPlayback();\n const { toggleMute, seek, currentTime, duration } = useGingerMedia();\n\n const {\n mute: muteBinding,\n seekForward: seekForwardBinding,\n seekBackward: seekBackwardBinding,\n } = bindings;\n\n useEffect(() => {\n if (!enabled || typeof window === \"undefined\") return;\n const playPause = (bindings.playPause ?? \" \").toLowerCase();\n const nextKey = (bindings.next ?? \"ArrowRight\").toLowerCase();\n const prevKey = (bindings.previous ?? \"ArrowLeft\").toLowerCase();\n const muteKey = muteBinding?.toLowerCase();\n const seekFwdKey = seekForwardBinding?.toLowerCase();\n const seekBwdKey = seekBackwardBinding?.toLowerCase();\n const seekSecs = bindings.seekSeconds ?? 5;\n\n const onKeyDown = (event: KeyboardEvent) => {\n const target = event.target as HTMLElement | null;\n if (\n target &&\n ([\"INPUT\", \"TEXTAREA\", \"SELECT\"].includes(target.tagName) || target.isContentEditable)\n )\n return;\n const key = event.key.toLowerCase();\n if (key === playPause) {\n event.preventDefault();\n togglePlayPause();\n } else if (key === nextKey) {\n event.preventDefault();\n next();\n } else if (key === prevKey) {\n event.preventDefault();\n prev();\n } else if (muteKey && key === muteKey) {\n event.preventDefault();\n toggleMute();\n } else if (seekFwdKey && key === seekFwdKey) {\n event.preventDefault();\n const dur = duration > 0 ? duration : Number.POSITIVE_INFINITY;\n seek(Math.min(dur, currentTime + seekSecs));\n } else if (seekBwdKey && key === seekBwdKey) {\n event.preventDefault();\n seek(Math.max(0, currentTime - seekSecs));\n }\n };\n window.addEventListener(\"keydown\", onKeyDown);\n return () => window.removeEventListener(\"keydown\", onKeyDown);\n }, [\n bindings.next,\n bindings.playPause,\n bindings.previous,\n bindings.seekSeconds,\n currentTime,\n duration,\n enabled,\n muteBinding,\n next,\n prev,\n seek,\n seekBackwardBinding,\n seekForwardBinding,\n toggleMute,\n togglePlayPause,\n ]);\n}\n","import { useEffect, useRef } from \"react\";\nimport { useGingerPlayback } from \"../context/GingerSplitContexts\";\n\nexport type GingerSleepTimerOptions = {\n durationMs?: number;\n stopAfterTracks?: number;\n respectPause?: boolean;\n enabled?: boolean;\n onFire?: () => void;\n};\n\nexport function useGingerSleepTimer(options: GingerSleepTimerOptions): void {\n const { durationMs, stopAfterTracks, respectPause = true, enabled = true, onFire } = options;\n const { currentIndex, pause, isPaused } = useGingerPlayback();\n const remainingTracksRef = useRef(stopAfterTracks ?? 0);\n const prevIndexRef = useRef(currentIndex);\n\n // Remaining milliseconds for the duration-based timer; carried across pause/resume cycles.\n const remainingMsRef = useRef(durationMs ?? 0);\n // Timestamp when the current timer segment started (set when playback resumes).\n const segmentStartRef = useRef<number | null>(null);\n\n useEffect(() => {\n remainingTracksRef.current = stopAfterTracks ?? 0;\n }, [stopAfterTracks]);\n\n // Keep remainingMsRef in sync when durationMs changes while the timer is inactive.\n const prevDurationMsRef = useRef(durationMs);\n useEffect(() => {\n if (prevDurationMsRef.current !== durationMs) {\n remainingMsRef.current = durationMs ?? 0;\n prevDurationMsRef.current = durationMs;\n }\n }, [durationMs]);\n\n useEffect(() => {\n if (!enabled || !durationMs || durationMs <= 0) {\n // Reset remaining when disabled or no duration\n remainingMsRef.current = durationMs ?? 0;\n segmentStartRef.current = null;\n return;\n }\n\n if (respectPause && isPaused) {\n // Snapshot how much time is left before pausing\n if (segmentStartRef.current !== null) {\n const elapsed = Date.now() - segmentStartRef.current;\n remainingMsRef.current = Math.max(0, remainingMsRef.current - elapsed);\n segmentStartRef.current = null;\n }\n return;\n }\n\n // Playing: start (or continue) the countdown from remainingMsRef\n segmentStartRef.current = Date.now();\n const id = setTimeout(() => {\n remainingMsRef.current = 0;\n segmentStartRef.current = null;\n pause();\n onFire?.();\n }, remainingMsRef.current);\n\n return () => {\n clearTimeout(id);\n // Snapshot remaining when effect cleans up (e.g. isPaused or deps changed)\n if (segmentStartRef.current !== null) {\n const elapsed = Date.now() - segmentStartRef.current;\n remainingMsRef.current = Math.max(0, remainingMsRef.current - elapsed);\n segmentStartRef.current = null;\n }\n };\n }, [durationMs, enabled, isPaused, onFire, pause, respectPause]);\n\n useEffect(() => {\n if (!enabled || !stopAfterTracks || stopAfterTracks <= 0) return;\n const prev = prevIndexRef.current;\n prevIndexRef.current = currentIndex;\n if (currentIndex === prev) return;\n remainingTracksRef.current -= 1;\n if (remainingTracksRef.current <= 0) {\n pause();\n onFire?.();\n }\n }, [currentIndex, enabled, onFire, pause, stopAfterTracks]);\n}\n","import { useEffect, useRef } from \"react\";\nimport { useGingerState } from \"../context/GingerSplitContexts\";\n\nexport function useGingerDebugLog(enabled = false): void {\n const state = useGingerState();\n const prevRef = useRef(state);\n\n useEffect(() => {\n if (!enabled || typeof console === \"undefined\") return;\n const prev = prevRef.current;\n if (prev !== state) {\n console.debug(\"[ginger]\", {\n from: {\n currentIndex: prev.currentIndex,\n isPaused: prev.isPaused,\n currentTime: prev.currentTime,\n repeatMode: prev.repeatMode,\n },\n to: {\n currentIndex: state.currentIndex,\n isPaused: state.isPaused,\n currentTime: state.currentTime,\n repeatMode: state.repeatMode,\n },\n });\n }\n prevRef.current = state;\n }, [enabled, state]);\n}\n","import { useCallback, useState } from \"react\";\nimport type { PointerEvent as ReactPointerEvent } from \"react\";\nimport {\n gingerStateFromContextValues,\n useGingerMedia,\n useGingerPlayback,\n} from \"../context/GingerSplitContexts\";\nimport { progressFraction } from \"../internal/selectors\";\n\nexport type SeekDragState = {\n /** Raw drag fraction — only updated during an active drag gesture. */\n fraction: number;\n /** Blended fraction: follows live playback when idle, drag position when dragging. */\n displayFraction: number;\n isDragging: boolean;\n onPointerDown: (event: ReactPointerEvent<HTMLElement>) => void;\n};\n\nfunction clamp01(value: number): number {\n return Math.max(0, Math.min(1, value));\n}\n\nexport function useSeekDrag(duration: number): SeekDragState {\n const media = useGingerMedia();\n const playback = useGingerPlayback();\n const { seek } = media;\n const [fraction, setFraction] = useState(0);\n const [isDragging, setIsDragging] = useState(false);\n\n const liveFraction = progressFraction(gingerStateFromContextValues(playback, media));\n const displayFraction = isDragging ? fraction : liveFraction;\n\n const onPointerDown = useCallback(\n (event: ReactPointerEvent<HTMLElement>) => {\n if (!(duration > 0)) return;\n const target = event.currentTarget;\n const rect = target.getBoundingClientRect();\n const update = (clientX: number) => {\n const ratio = clamp01((clientX - rect.left) / rect.width);\n setFraction(ratio);\n seek(ratio * duration);\n };\n setIsDragging(true);\n target.setPointerCapture(event.pointerId);\n update(event.clientX);\n const onMove = (moveEvent: PointerEvent) => update(moveEvent.clientX);\n const onUp = (upEvent: PointerEvent) => {\n update(upEvent.clientX);\n setIsDragging(false);\n target.releasePointerCapture(event.pointerId);\n target.removeEventListener(\"pointermove\", onMove);\n target.removeEventListener(\"pointerup\", onUp);\n target.removeEventListener(\"pointercancel\", onUp);\n };\n target.addEventListener(\"pointermove\", onMove);\n target.addEventListener(\"pointerup\", onUp);\n target.addEventListener(\"pointercancel\", onUp);\n },\n [duration, seek],\n );\n\n return { fraction, displayFraction, isDragging, onPointerDown };\n}\n","import { useEffect } from \"react\";\nimport { useGingerPlayback } from \"../context/GingerSplitContexts\";\nimport { computeNextIndex } from \"../core/transitions\";\n\nexport type UseNextTrackPrefetchOptions = {\n /** When false, no prefetch runs. Default true. */\n enabled?: boolean;\n /**\n * Match `crossOrigin` on `Ginger.Player` when `fileUrl` is cross-origin so the browser\n * can reuse cached media consistently.\n */\n crossOrigin?: \"\" | \"anonymous\" | \"use-credentials\" | undefined;\n};\n\n/**\n * Warms the browser cache for the **logical** next track (same rules as the Next control:\n * `computeNextIndex` from queue, repeat, and playback mode) using a detached `HTMLAudioElement`\n * with `preload=\"auto\"`. Safe to call alongside `Ginger.Player`; it does not replace main playback.\n */\nexport function useNextTrackPrefetch(options: UseNextTrackPrefetchOptions = {}): void {\n const { enabled = true, crossOrigin } = options;\n const { tracks, currentIndex, repeatMode, playbackMode } = useGingerPlayback();\n\n useEffect(() => {\n if (!enabled || typeof document === \"undefined\") return;\n const nextIndex = computeNextIndex({ tracks, currentIndex, repeatMode, playbackMode });\n if (nextIndex === currentIndex) return;\n const nextUrl = tracks[nextIndex]?.fileUrl ?? \"\";\n if (!nextUrl) return;\n\n const audio = document.createElement(\"audio\");\n audio.preload = \"auto\";\n if (crossOrigin) audio.crossOrigin = crossOrigin;\n audio.src = nextUrl;\n audio.load();\n\n return () => {\n audio.removeAttribute(\"src\");\n audio.load();\n };\n }, [enabled, crossOrigin, tracks, currentIndex, repeatMode, playbackMode]);\n}\n","import { clampPlaybackRate, clampVolume, createInitialState, gingerReducer } from \"./core/playbackReducer\";\nimport type { GingerAction, GingerInitPayload, GingerState, PlaylistMeta, RepeatMode, Track } from \"./types\";\n\nexport type GingerStoreOptions = {\n tracks?: Track[];\n currentIndex?: number;\n playlistMeta?: PlaylistMeta | null;\n isPaused?: boolean;\n isShuffled?: boolean;\n repeatMode?: RepeatMode;\n playbackMode?: GingerState[\"playbackMode\"];\n volume?: number;\n muted?: boolean;\n playbackRate?: number;\n};\n\nexport type GingerStore = {\n /** Returns the current state snapshot. */\n getState: () => GingerState;\n /** Dispatch an action to update state. Synchronously updates state and notifies listeners. */\n dispatch: (action: GingerAction) => void;\n /**\n * Subscribe to state changes. The listener is called after every `dispatch` that produces\n * a new state object. Returns an unsubscribe function.\n */\n subscribe: (listener: (state: GingerState) => void) => () => void;\n /** Convenience: re-initialise with a new set of init options (equivalent to `dispatch({ type: \"INIT\", ... })`). */\n init: (payload: GingerInitPayload) => void;\n /** Clamp helpers re-exported for convenience. */\n clampVolume: typeof clampVolume;\n clampPlaybackRate: typeof clampPlaybackRate;\n};\n\n/**\n * Framework-agnostic store wrapping `gingerReducer`.\n * Usable outside React — e.g. in Svelte, Vue, Node.js testing, or server-side rendering contexts.\n *\n * @example\n * ```ts\n * import { createGingerStore } from \"@lucaismyname/ginger\";\n *\n * const store = createGingerStore({ tracks: myTracks });\n * const unsub = store.subscribe((state) => console.log(state.currentIndex));\n * store.dispatch({ type: \"NEXT\" });\n * unsub();\n * ```\n */\nexport function createGingerStore(options: GingerStoreOptions = {}): GingerStore {\n let state = createInitialState({\n tracks: options.tracks ?? [],\n currentIndex: options.currentIndex,\n playlistMeta: options.playlistMeta,\n isPaused: options.isPaused,\n isShuffled: options.isShuffled,\n repeatMode: options.repeatMode,\n playbackMode: options.playbackMode,\n volume: options.volume,\n muted: options.muted,\n playbackRate: options.playbackRate,\n });\n\n const listeners = new Set<(state: GingerState) => void>();\n\n const dispatch = (action: GingerAction): void => {\n const next = gingerReducer(state, action);\n if (next !== state) {\n state = next;\n for (const listener of listeners) {\n listener(state);\n }\n }\n };\n\n const subscribe = (listener: (state: GingerState) => void): (() => void) => {\n listeners.add(listener);\n return () => listeners.delete(listener);\n };\n\n const init = (payload: GingerInitPayload): void => {\n dispatch({ type: \"INIT\", payload });\n };\n\n return {\n getState: () => state,\n dispatch,\n subscribe,\n init,\n clampVolume,\n clampPlaybackRate,\n };\n}\n","import { useCallback, useEffect, useRef, useState } from \"react\";\nimport { useGingerPlayback } from \"../context/GingerSplitContexts\";\nimport type { Track } from \"../types\";\n\nexport type GingerPlaybackHistoryEntry = {\n track: Track;\n /** The track's index in the queue at the time it was played. */\n index: number;\n /** Unix timestamp (ms) when the track started playing. */\n playedAt: number;\n};\n\nexport type UseGingerPlaybackHistoryOptions = {\n /** Maximum number of entries to keep. Oldest entries are dropped first. Default: 50. */\n maxLength?: number;\n};\n\nexport type UseGingerPlaybackHistoryResult = {\n /** Chronological list of played tracks; most recent entry is last. */\n history: GingerPlaybackHistoryEntry[];\n clearHistory: () => void;\n};\n\n/**\n * Records a history of played tracks in chronological order.\n * Useful for \"what was playing before\" in shuffle mode or for analytics.\n *\n * The history is stored in component state and does not survive remounts.\n * In shuffle mode, `index` reflects the position in the current shuffled queue.\n */\nexport function useGingerPlaybackHistory(\n options: UseGingerPlaybackHistoryOptions = {},\n): UseGingerPlaybackHistoryResult {\n const { maxLength = 50 } = options;\n const { tracks, currentIndex } = useGingerPlayback();\n\n const [history, setHistory] = useState<GingerPlaybackHistoryEntry[]>([]);\n const prevIndexRef = useRef<number | null>(null);\n const prevTracksRef = useRef(tracks);\n prevTracksRef.current = tracks;\n\n useEffect(() => {\n const track = tracks[currentIndex];\n if (!track) return;\n\n if (prevIndexRef.current === currentIndex) return;\n prevIndexRef.current = currentIndex;\n\n const entry: GingerPlaybackHistoryEntry = {\n track,\n index: currentIndex,\n playedAt: Date.now(),\n };\n\n setHistory((prev) => {\n const next = [...prev, entry];\n return next.length > maxLength ? next.slice(next.length - maxLength) : next;\n });\n }, [currentIndex, tracks, maxLength]);\n\n const clearHistory = useCallback(() => setHistory([]), []);\n\n return { history, clearHistory };\n}\n","import { useCallback, useRef, useState } from \"react\";\nimport { useGingerMedia } from \"../context/GingerSplitContexts\";\n\nexport type UseGingerVolumeFadeOptions = {\n /** Target volume to fade to (0–1). */\n targetVolume: number;\n /** Duration of the fade in milliseconds. */\n durationMs: number;\n /** Called when the fade completes normally (not when cancelled). */\n onComplete?: () => void;\n};\n\nexport type UseGingerVolumeFadeResult = {\n /** Start a volume fade. Cancels any in-progress fade. */\n fadeVolumeTo: (options: UseGingerVolumeFadeOptions) => void;\n /** Cancel the current fade and hold at the current volume. */\n cancelFade: () => void;\n /** True while a fade is in progress. */\n isFading: boolean;\n};\n\n/**\n * Smoothly interpolates volume over a given duration using `requestAnimationFrame`.\n * Useful for fade-in on track start, fade-out before sleep timer fires, or crossfade prep.\n */\nexport function useGingerVolumeFade(): UseGingerVolumeFadeResult {\n const { setVolume, volume } = useGingerMedia();\n const [isFading, setIsFading] = useState(false);\n\n const rafRef = useRef<number>(0);\n const cancelledRef = useRef(false);\n\n const cancelFade = useCallback(() => {\n cancelAnimationFrame(rafRef.current);\n cancelledRef.current = true;\n setIsFading(false);\n }, []);\n\n const fadeVolumeTo = useCallback(\n ({ targetVolume, durationMs, onComplete }: UseGingerVolumeFadeOptions) => {\n cancelAnimationFrame(rafRef.current);\n cancelledRef.current = false;\n\n const clamp = (v: number) => Math.min(1, Math.max(0, v));\n const target = clamp(targetVolume);\n const startTime = performance.now();\n\n // Capture start volume at the moment the fade begins\n let startVolume = volume;\n\n setIsFading(true);\n\n const tick = (now: number) => {\n if (cancelledRef.current) return;\n const elapsed = now - startTime;\n const progress = Math.min(1, elapsed / Math.max(1, durationMs));\n const current = startVolume + (target - startVolume) * progress;\n setVolume(clamp(current));\n\n if (progress < 1) {\n rafRef.current = requestAnimationFrame(tick);\n } else {\n setIsFading(false);\n onComplete?.();\n }\n };\n\n // Use a small delay so `volume` from closure is the current value\n rafRef.current = requestAnimationFrame((now) => {\n startVolume = volume;\n tick(now);\n });\n },\n [setVolume, volume],\n );\n\n return { fadeVolumeTo, cancelFade, isFading };\n}\n","import { useMemo } from \"react\";\nimport { useGingerMedia, useGingerPlayback } from \"../context/GingerSplitContexts\";\n\nexport type GingerChapterProgress = {\n /** Fraction (0–1) of the way through the active chapter. 0 when no chapter is active. */\n progress: number;\n /** Seconds elapsed since the start of the active chapter. */\n elapsed: number;\n /** Seconds remaining until the end of the active chapter (or until end of track if last chapter). */\n remaining: number;\n};\n\n/**\n * Returns detailed progress information for the currently active chapter.\n * Complements `useGingerChapters` with per-chapter playback fractions.\n */\nexport function useGingerChapterProgress(): GingerChapterProgress {\n const { tracks, currentIndex } = useGingerPlayback();\n const { currentTime, duration } = useGingerMedia();\n\n const chapters = useMemo(() => {\n const raw = tracks[currentIndex]?.chapters ?? [];\n return [...raw]\n .filter((c) => c && Number.isFinite(c.startSeconds) && c.startSeconds >= 0)\n .sort((a, b) => a.startSeconds - b.startSeconds);\n }, [tracks, currentIndex]);\n\n return useMemo<GingerChapterProgress>(() => {\n if (chapters.length === 0) return { progress: 0, elapsed: 0, remaining: 0 };\n\n // Find active chapter (last one whose start is <= currentTime)\n let activeIdx = -1;\n for (let i = chapters.length - 1; i >= 0; i--) {\n if (currentTime >= chapters[i]!.startSeconds) {\n activeIdx = i;\n break;\n }\n }\n\n if (activeIdx === -1) return { progress: 0, elapsed: 0, remaining: 0 };\n\n const chapter = chapters[activeIdx]!;\n const nextChapter = chapters[activeIdx + 1];\n const chapterEnd = nextChapter?.startSeconds ?? (duration > 0 ? duration : currentTime);\n const chapterDuration = Math.max(0, chapterEnd - chapter.startSeconds);\n const elapsed = Math.max(0, currentTime - chapter.startSeconds);\n const remaining = Math.max(0, chapterEnd - currentTime);\n const progress = chapterDuration > 0 ? Math.min(1, elapsed / chapterDuration) : 0;\n\n return { progress, elapsed, remaining };\n }, [chapters, currentTime, duration]);\n}\n"],"names":["emptyFreq","emptyTime","useGingerLiveAnalyzer","options","enabled","fftSize","smoothingTimeConstant","minDecibels","maxDecibels","audioRef","state","useGinger","opts","useMemo","frame","setFrame","useState","error","setError","isSuspended","setIsSuspended","meta","setMeta","freqRef","useRef","timeRef","resume","useCallback","ctx","contextHolderRef","analyserHolderRef","useLayoutEffect","cancelled","consumerId","element","rafId","onStateChange","runLoop","a","fq","td","n","attach","el","id","context","analyser","attachLiveAnalyser","fft","e","msg","first","retryRaf","maxAttempts","attempts","retryLoop","out","detachLiveAnalyser","_a","useGingerKeyboardShortcuts","bindings","togglePlayPause","next","prev","useGingerPlayback","toggleMute","seek","currentTime","duration","useGingerMedia","muteBinding","seekForwardBinding","seekBackwardBinding","useEffect","playPause","nextKey","prevKey","muteKey","seekFwdKey","seekBwdKey","seekSecs","onKeyDown","event","target","key","dur","useGingerSleepTimer","durationMs","stopAfterTracks","respectPause","onFire","currentIndex","pause","isPaused","remainingTracksRef","prevIndexRef","remainingMsRef","segmentStartRef","prevDurationMsRef","elapsed","useGingerDebugLog","useGingerState","prevRef","clamp01","value","useSeekDrag","media","playback","fraction","setFraction","isDragging","setIsDragging","liveFraction","progressFraction","gingerStateFromContextValues","displayFraction","onPointerDown","rect","update","clientX","ratio","onMove","moveEvent","onUp","upEvent","useNextTrackPrefetch","crossOrigin","tracks","repeatMode","playbackMode","nextIndex","computeNextIndex","nextUrl","audio","createGingerStore","createInitialState","listeners","dispatch","action","gingerReducer","listener","payload","clampVolume","clampPlaybackRate","useGingerPlaybackHistory","maxLength","history","setHistory","prevTracksRef","track","entry","clearHistory","useGingerVolumeFade","setVolume","volume","isFading","setIsFading","rafRef","cancelledRef","cancelFade","targetVolume","onComplete","clamp","v","startTime","startVolume","tick","now","progress","current","useGingerChapterProgress","chapters","c","b","activeIdx","i","chapter","nextChapter","chapterEnd","chapterDuration","remaining"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"useGingerChapterProgress-CZdv-HiI.js","sources":["../src/analyzer/useGingerLiveAnalyzer.ts","../src/hooks/useGingerKeyboardShortcuts.ts","../src/hooks/useGingerSleepTimer.ts","../src/hooks/useGingerDebugLog.ts","../src/hooks/useSeekDrag.ts","../src/hooks/useNextTrackPrefetch.ts","../src/store.ts","../src/hooks/useGingerPlaybackHistory.ts","../src/hooks/useGingerVolumeFade.ts","../src/hooks/useGingerChapterProgress.ts"],"sourcesContent":["import { useCallback, useLayoutEffect, useMemo, useRef, useState } from \"react\";\nimport { useGinger } from \"../hooks/useGinger\";\nimport { type LiveAnalyserOptions, attachLiveAnalyser, detachLiveAnalyser } from \"./liveAudioGraph\";\n\nexport type UseGingerLiveAnalyzerOptions = {\n /** When false, the analyser is detached and no frames are read. Default true. */\n enabled?: boolean;\n fftSize?: number;\n smoothingTimeConstant?: number;\n minDecibels?: number;\n maxDecibels?: number;\n};\n\nexport type UseGingerLiveAnalyzerResult = {\n /** Byte frequency data (0–255); length equals `frequencyBinCount`. Updated each animation frame while enabled. */\n frequencyData: Uint8Array;\n /** Byte time-domain data (0–255); length equals `fftSize`. */\n timeDomainData: Uint8Array;\n frequencyBinCount: number;\n sampleRate: number;\n isSuspended: boolean;\n error: string | null;\n resume: () => Promise<void>;\n /**\n * Monotonically increasing counter incremented each animation frame.\n * Use as a `useMemo` / `useEffect` dependency to respond to new audio data, since\n * `frequencyData` and `timeDomainData` are mutated in-place (their reference never changes).\n */\n frame: number;\n};\n\nconst emptyFreq = new Uint8Array(0);\nconst emptyTime = new Uint8Array(0);\n\nexport function useGingerLiveAnalyzer(\n options: UseGingerLiveAnalyzerOptions = {},\n): UseGingerLiveAnalyzerResult {\n const {\n enabled = true,\n fftSize = 2048,\n smoothingTimeConstant = 0.8,\n minDecibels = -100,\n maxDecibels = -30,\n } = options;\n\n const { audioRef, state } = useGinger();\n const opts = useMemo<LiveAnalyserOptions>(\n () => ({\n fftSize,\n smoothingTimeConstant,\n minDecibels,\n maxDecibels,\n }),\n [fftSize, smoothingTimeConstant, minDecibels, maxDecibels],\n );\n\n const [frame, setFrame] = useState(0);\n const [error, setError] = useState<string | null>(null);\n const [isSuspended, setIsSuspended] = useState(false);\n const [meta, setMeta] = useState({ frequencyBinCount: 0, sampleRate: 0 });\n\n const freqRef = useRef<Uint8Array>(emptyFreq);\n const timeRef = useRef<Uint8Array>(emptyTime);\n\n const resume = useCallback(async () => {\n const ctx = contextHolderRef.current;\n if (ctx && ctx.state === \"suspended\") {\n await ctx.resume();\n }\n }, []);\n\n const contextHolderRef = useRef<AudioContext | null>(null);\n const analyserHolderRef = useRef<AnalyserNode | null>(null);\n\n useLayoutEffect(() => {\n if (!enabled || typeof window === \"undefined\") {\n return;\n }\n\n let cancelled = false;\n let consumerId: number | null = null;\n let element: HTMLAudioElement | null = null;\n let rafId = 0;\n\n const onStateChange = () => {\n const ctx = contextHolderRef.current;\n if (ctx) setIsSuspended(ctx.state === \"suspended\");\n };\n\n const runLoop = () => {\n if (cancelled) return;\n const a = analyserHolderRef.current;\n const fq = freqRef.current;\n const td = timeRef.current;\n if (a && fq.length > 0 && td.length > 0) {\n a.getByteFrequencyData(fq as Uint8Array<ArrayBuffer>);\n a.getByteTimeDomainData(td as Uint8Array<ArrayBuffer>);\n setFrame((n) => n + 1);\n }\n rafId = requestAnimationFrame(runLoop);\n };\n\n type AttachOutcome = \"ok\" | \"no-element\" | \"error\";\n\n const attach = (): AttachOutcome => {\n const el = audioRef.current;\n if (!el || cancelled) return \"no-element\";\n try {\n const { id, context, analyser } = attachLiveAnalyser(el, opts);\n consumerId = id;\n element = el;\n contextHolderRef.current = context;\n analyserHolderRef.current = analyser;\n setIsSuspended(context.state === \"suspended\");\n setError(null);\n\n context.addEventListener(\"statechange\", onStateChange);\n\n const n = analyser.frequencyBinCount;\n const fft = analyser.fftSize;\n freqRef.current = new Uint8Array(n);\n timeRef.current = new Uint8Array(fft);\n setMeta({ frequencyBinCount: n, sampleRate: context.sampleRate });\n\n rafId = requestAnimationFrame(runLoop);\n return \"ok\";\n } catch (e) {\n const msg = e instanceof Error ? e.message : \"Failed to attach live analyser\";\n setError(msg);\n contextHolderRef.current = null;\n analyserHolderRef.current = null;\n freqRef.current = emptyFreq;\n timeRef.current = emptyTime;\n setMeta({ frequencyBinCount: 0, sampleRate: 0 });\n return \"error\";\n }\n };\n\n const first = attach();\n if (first !== \"ok\") {\n let retryRaf = 0;\n const maxAttempts = 120;\n let attempts = 0;\n\n const retryLoop = () => {\n if (cancelled) return;\n const out = attach();\n if (out === \"ok\" || out === \"error\") return;\n attempts += 1;\n if (attempts >= maxAttempts) return;\n retryRaf = requestAnimationFrame(retryLoop);\n };\n\n if (first === \"no-element\") {\n retryRaf = requestAnimationFrame(retryLoop);\n }\n\n return () => {\n cancelled = true;\n cancelAnimationFrame(retryRaf);\n cancelAnimationFrame(rafId);\n if (consumerId != null && element) {\n detachLiveAnalyser(element, consumerId);\n }\n contextHolderRef.current?.removeEventListener(\"statechange\", onStateChange);\n contextHolderRef.current = null;\n analyserHolderRef.current = null;\n freqRef.current = emptyFreq;\n timeRef.current = emptyTime;\n };\n }\n\n return () => {\n cancelled = true;\n cancelAnimationFrame(rafId);\n if (consumerId != null && element) {\n detachLiveAnalyser(element, consumerId);\n }\n contextHolderRef.current?.removeEventListener(\"statechange\", onStateChange);\n contextHolderRef.current = null;\n analyserHolderRef.current = null;\n freqRef.current = emptyFreq;\n timeRef.current = emptyTime;\n setMeta({ frequencyBinCount: 0, sampleRate: 0 });\n };\n }, [enabled, audioRef, opts, state.currentIndex]);\n\n return {\n frequencyData: freqRef.current,\n timeDomainData: timeRef.current,\n frequencyBinCount: meta.frequencyBinCount,\n sampleRate: meta.sampleRate,\n isSuspended,\n error,\n resume,\n frame,\n };\n}\n","import { useEffect } from \"react\";\nimport { useGingerMedia, useGingerPlayback } from \"../context/GingerSplitContexts\";\n\nexport type GingerKeyboardShortcutBindings = {\n playPause?: string;\n next?: string;\n previous?: string;\n mute?: string;\n /** Key to seek forward by `seekSeconds`. Default: undefined (disabled). */\n seekForward?: string;\n /** Key to seek backward by `seekSeconds`. Default: undefined (disabled). */\n seekBackward?: string;\n /** Seconds to seek per `seekForward` / `seekBackward` keypress. Default: 5. */\n seekSeconds?: number;\n};\n\nexport function useGingerKeyboardShortcuts(\n enabled = true,\n bindings: GingerKeyboardShortcutBindings = {},\n): void {\n const { togglePlayPause, next, prev } = useGingerPlayback();\n const { toggleMute, seek, currentTime, duration } = useGingerMedia();\n\n const {\n mute: muteBinding,\n seekForward: seekForwardBinding,\n seekBackward: seekBackwardBinding,\n } = bindings;\n\n useEffect(() => {\n if (!enabled || typeof window === \"undefined\") return;\n const playPause = (bindings.playPause ?? \" \").toLowerCase();\n const nextKey = (bindings.next ?? \"ArrowRight\").toLowerCase();\n const prevKey = (bindings.previous ?? \"ArrowLeft\").toLowerCase();\n const muteKey = muteBinding?.toLowerCase();\n const seekFwdKey = seekForwardBinding?.toLowerCase();\n const seekBwdKey = seekBackwardBinding?.toLowerCase();\n const seekSecs = bindings.seekSeconds ?? 5;\n\n const onKeyDown = (event: KeyboardEvent) => {\n const target = event.target as HTMLElement | null;\n if (\n target &&\n ([\"INPUT\", \"TEXTAREA\", \"SELECT\"].includes(target.tagName) || target.isContentEditable)\n )\n return;\n const key = event.key.toLowerCase();\n if (key === playPause) {\n event.preventDefault();\n togglePlayPause();\n } else if (key === nextKey) {\n event.preventDefault();\n next();\n } else if (key === prevKey) {\n event.preventDefault();\n prev();\n } else if (muteKey && key === muteKey) {\n event.preventDefault();\n toggleMute();\n } else if (seekFwdKey && key === seekFwdKey) {\n event.preventDefault();\n const dur = duration > 0 ? duration : Number.POSITIVE_INFINITY;\n seek(Math.min(dur, currentTime + seekSecs));\n } else if (seekBwdKey && key === seekBwdKey) {\n event.preventDefault();\n seek(Math.max(0, currentTime - seekSecs));\n }\n };\n window.addEventListener(\"keydown\", onKeyDown);\n return () => window.removeEventListener(\"keydown\", onKeyDown);\n }, [\n bindings.next,\n bindings.playPause,\n bindings.previous,\n bindings.seekSeconds,\n currentTime,\n duration,\n enabled,\n muteBinding,\n next,\n prev,\n seek,\n seekBackwardBinding,\n seekForwardBinding,\n toggleMute,\n togglePlayPause,\n ]);\n}\n","import { useEffect, useRef } from \"react\";\nimport { useGingerPlayback } from \"../context/GingerSplitContexts\";\n\nexport type GingerSleepTimerOptions = {\n durationMs?: number;\n stopAfterTracks?: number;\n respectPause?: boolean;\n enabled?: boolean;\n onFire?: () => void;\n};\n\nexport function useGingerSleepTimer(options: GingerSleepTimerOptions): void {\n const { durationMs, stopAfterTracks, respectPause = true, enabled = true, onFire } = options;\n const { currentIndex, pause, isPaused } = useGingerPlayback();\n const remainingTracksRef = useRef(stopAfterTracks ?? 0);\n const prevIndexRef = useRef(currentIndex);\n\n // Remaining milliseconds for the duration-based timer; carried across pause/resume cycles.\n const remainingMsRef = useRef(durationMs ?? 0);\n // Timestamp when the current timer segment started (set when playback resumes).\n const segmentStartRef = useRef<number | null>(null);\n\n useEffect(() => {\n remainingTracksRef.current = stopAfterTracks ?? 0;\n }, [stopAfterTracks]);\n\n // Keep remainingMsRef in sync when durationMs changes while the timer is inactive.\n const prevDurationMsRef = useRef(durationMs);\n useEffect(() => {\n if (prevDurationMsRef.current !== durationMs) {\n remainingMsRef.current = durationMs ?? 0;\n prevDurationMsRef.current = durationMs;\n }\n }, [durationMs]);\n\n useEffect(() => {\n if (!enabled || !durationMs || durationMs <= 0) {\n // Reset remaining when disabled or no duration\n remainingMsRef.current = durationMs ?? 0;\n segmentStartRef.current = null;\n return;\n }\n\n if (respectPause && isPaused) {\n // Snapshot how much time is left before pausing\n if (segmentStartRef.current !== null) {\n const elapsed = Date.now() - segmentStartRef.current;\n remainingMsRef.current = Math.max(0, remainingMsRef.current - elapsed);\n segmentStartRef.current = null;\n }\n return;\n }\n\n // Playing: start (or continue) the countdown from remainingMsRef\n segmentStartRef.current = Date.now();\n const id = setTimeout(() => {\n remainingMsRef.current = 0;\n segmentStartRef.current = null;\n pause();\n onFire?.();\n }, remainingMsRef.current);\n\n return () => {\n clearTimeout(id);\n // Snapshot remaining when effect cleans up (e.g. isPaused or deps changed)\n if (segmentStartRef.current !== null) {\n const elapsed = Date.now() - segmentStartRef.current;\n remainingMsRef.current = Math.max(0, remainingMsRef.current - elapsed);\n segmentStartRef.current = null;\n }\n };\n }, [durationMs, enabled, isPaused, onFire, pause, respectPause]);\n\n useEffect(() => {\n if (!enabled || !stopAfterTracks || stopAfterTracks <= 0) return;\n const prev = prevIndexRef.current;\n prevIndexRef.current = currentIndex;\n if (currentIndex === prev) return;\n remainingTracksRef.current -= 1;\n if (remainingTracksRef.current <= 0) {\n pause();\n onFire?.();\n }\n }, [currentIndex, enabled, onFire, pause, stopAfterTracks]);\n}\n","import { useEffect, useRef } from \"react\";\nimport { useGingerState } from \"../context/GingerSplitContexts\";\n\nexport function useGingerDebugLog(enabled = false): void {\n const state = useGingerState();\n const prevRef = useRef(state);\n\n useEffect(() => {\n if (!enabled || typeof console === \"undefined\") return;\n const prev = prevRef.current;\n if (prev !== state) {\n console.debug(\"[ginger]\", {\n from: {\n currentIndex: prev.currentIndex,\n isPaused: prev.isPaused,\n currentTime: prev.currentTime,\n repeatMode: prev.repeatMode,\n },\n to: {\n currentIndex: state.currentIndex,\n isPaused: state.isPaused,\n currentTime: state.currentTime,\n repeatMode: state.repeatMode,\n },\n });\n }\n prevRef.current = state;\n }, [enabled, state]);\n}\n","import { useCallback, useState } from \"react\";\nimport type { PointerEvent as ReactPointerEvent } from \"react\";\nimport {\n gingerStateFromContextValues,\n useGingerMedia,\n useGingerPlayback,\n} from \"../context/GingerSplitContexts\";\nimport { progressFraction } from \"../internal/selectors\";\n\nexport type SeekDragState = {\n /** Raw drag fraction — only updated during an active drag gesture. */\n fraction: number;\n /** Blended fraction: follows live playback when idle, drag position when dragging. */\n displayFraction: number;\n isDragging: boolean;\n onPointerDown: (event: ReactPointerEvent<HTMLElement>) => void;\n};\n\nfunction clamp01(value: number): number {\n return Math.max(0, Math.min(1, value));\n}\n\nexport function useSeekDrag(duration: number): SeekDragState {\n const media = useGingerMedia();\n const playback = useGingerPlayback();\n const { seek } = media;\n const [fraction, setFraction] = useState(0);\n const [isDragging, setIsDragging] = useState(false);\n\n const liveFraction = progressFraction(gingerStateFromContextValues(playback, media));\n const displayFraction = isDragging ? fraction : liveFraction;\n\n const onPointerDown = useCallback(\n (event: ReactPointerEvent<HTMLElement>) => {\n if (!(duration > 0)) return;\n const target = event.currentTarget;\n const rect = target.getBoundingClientRect();\n const update = (clientX: number) => {\n const ratio = clamp01((clientX - rect.left) / rect.width);\n setFraction(ratio);\n seek(ratio * duration);\n };\n setIsDragging(true);\n target.setPointerCapture(event.pointerId);\n update(event.clientX);\n const onMove = (moveEvent: PointerEvent) => update(moveEvent.clientX);\n const onUp = (upEvent: PointerEvent) => {\n update(upEvent.clientX);\n setIsDragging(false);\n target.releasePointerCapture(event.pointerId);\n target.removeEventListener(\"pointermove\", onMove);\n target.removeEventListener(\"pointerup\", onUp);\n target.removeEventListener(\"pointercancel\", onUp);\n };\n target.addEventListener(\"pointermove\", onMove);\n target.addEventListener(\"pointerup\", onUp);\n target.addEventListener(\"pointercancel\", onUp);\n },\n [duration, seek],\n );\n\n return { fraction, displayFraction, isDragging, onPointerDown };\n}\n","import { useEffect } from \"react\";\nimport { useGingerPlayback } from \"../context/GingerSplitContexts\";\nimport { computeNextIndex } from \"../core/transitions\";\n\nexport type UseNextTrackPrefetchOptions = {\n /** When false, no prefetch runs. Default true. */\n enabled?: boolean;\n /**\n * Match `crossOrigin` on `Ginger.Player` when `fileUrl` is cross-origin so the browser\n * can reuse cached media consistently.\n */\n crossOrigin?: \"\" | \"anonymous\" | \"use-credentials\" | undefined;\n};\n\n/**\n * Warms the browser cache for the **logical** next track (same rules as the Next control:\n * `computeNextIndex` from queue, repeat, and playback mode) using a detached `HTMLAudioElement`\n * with `preload=\"auto\"`. Safe to call alongside `Ginger.Player`; it does not replace main playback.\n */\nexport function useNextTrackPrefetch(options: UseNextTrackPrefetchOptions = {}): void {\n const { enabled = true, crossOrigin } = options;\n const { tracks, currentIndex, repeatMode, playbackMode } = useGingerPlayback();\n\n useEffect(() => {\n if (!enabled || typeof document === \"undefined\") return;\n const nextIndex = computeNextIndex({ tracks, currentIndex, repeatMode, playbackMode });\n if (nextIndex === currentIndex) return;\n const nextUrl = tracks[nextIndex]?.fileUrl ?? \"\";\n if (!nextUrl) return;\n\n const audio = document.createElement(\"audio\");\n audio.preload = \"auto\";\n if (crossOrigin) audio.crossOrigin = crossOrigin;\n audio.src = nextUrl;\n audio.load();\n\n return () => {\n audio.removeAttribute(\"src\");\n audio.load();\n };\n }, [enabled, crossOrigin, tracks, currentIndex, repeatMode, playbackMode]);\n}\n","import { clampPlaybackRate, clampVolume, createInitialState, gingerReducer } from \"./core/playbackReducer\";\nimport type { GingerAction, GingerInitPayload, GingerState, PlaylistMeta, RepeatMode, Track } from \"./types\";\n\nexport type GingerStoreOptions = {\n tracks?: Track[];\n currentIndex?: number;\n playlistMeta?: PlaylistMeta | null;\n isPaused?: boolean;\n isShuffled?: boolean;\n repeatMode?: RepeatMode;\n playbackMode?: GingerState[\"playbackMode\"];\n volume?: number;\n muted?: boolean;\n playbackRate?: number;\n};\n\nexport type GingerStore = {\n /** Returns the current state snapshot. */\n getState: () => GingerState;\n /** Dispatch an action to update state. Synchronously updates state and notifies listeners. */\n dispatch: (action: GingerAction) => void;\n /**\n * Subscribe to state changes. The listener is called after every `dispatch` that produces\n * a new state object. Returns an unsubscribe function.\n */\n subscribe: (listener: (state: GingerState) => void) => () => void;\n /** Convenience: re-initialise with a new set of init options (equivalent to `dispatch({ type: \"INIT\", ... })`). */\n init: (payload: GingerInitPayload) => void;\n /** Clamp helpers re-exported for convenience. */\n clampVolume: typeof clampVolume;\n clampPlaybackRate: typeof clampPlaybackRate;\n};\n\n/**\n * Framework-agnostic store wrapping `gingerReducer`.\n * Usable outside React — e.g. in Svelte, Vue, Node.js testing, or server-side rendering contexts.\n *\n * @example\n * ```ts\n * import { createGingerStore } from \"@lucaismyname/ginger\";\n *\n * const store = createGingerStore({ tracks: myTracks });\n * const unsub = store.subscribe((state) => console.log(state.currentIndex));\n * store.dispatch({ type: \"NEXT\" });\n * unsub();\n * ```\n */\nexport function createGingerStore(options: GingerStoreOptions = {}): GingerStore {\n let state = createInitialState({\n tracks: options.tracks ?? [],\n currentIndex: options.currentIndex,\n playlistMeta: options.playlistMeta,\n isPaused: options.isPaused,\n isShuffled: options.isShuffled,\n repeatMode: options.repeatMode,\n playbackMode: options.playbackMode,\n volume: options.volume,\n muted: options.muted,\n playbackRate: options.playbackRate,\n });\n\n const listeners = new Set<(state: GingerState) => void>();\n\n const dispatch = (action: GingerAction): void => {\n const next = gingerReducer(state, action);\n if (next !== state) {\n state = next;\n for (const listener of listeners) {\n listener(state);\n }\n }\n };\n\n const subscribe = (listener: (state: GingerState) => void): (() => void) => {\n listeners.add(listener);\n return () => listeners.delete(listener);\n };\n\n const init = (payload: GingerInitPayload): void => {\n dispatch({ type: \"INIT\", payload });\n };\n\n return {\n getState: () => state,\n dispatch,\n subscribe,\n init,\n clampVolume,\n clampPlaybackRate,\n };\n}\n","import { useCallback, useEffect, useRef, useState } from \"react\";\nimport { useGingerPlayback } from \"../context/GingerSplitContexts\";\nimport type { Track } from \"../types\";\n\nexport type GingerPlaybackHistoryEntry = {\n track: Track;\n /** The track's index in the queue at the time it was played. */\n index: number;\n /** Unix timestamp (ms) when the track started playing. */\n playedAt: number;\n};\n\nexport type UseGingerPlaybackHistoryOptions = {\n /** Maximum number of entries to keep. Oldest entries are dropped first. Default: 50. */\n maxLength?: number;\n};\n\nexport type UseGingerPlaybackHistoryResult = {\n /** Chronological list of played tracks; most recent entry is last. */\n history: GingerPlaybackHistoryEntry[];\n clearHistory: () => void;\n};\n\n/**\n * Records a history of played tracks in chronological order.\n * Useful for \"what was playing before\" in shuffle mode or for analytics.\n *\n * The history is stored in component state and does not survive remounts.\n * In shuffle mode, `index` reflects the position in the current shuffled queue.\n */\nexport function useGingerPlaybackHistory(\n options: UseGingerPlaybackHistoryOptions = {},\n): UseGingerPlaybackHistoryResult {\n const { maxLength = 50 } = options;\n const { tracks, currentIndex } = useGingerPlayback();\n\n const [history, setHistory] = useState<GingerPlaybackHistoryEntry[]>([]);\n const prevIndexRef = useRef<number | null>(null);\n const prevTracksRef = useRef(tracks);\n prevTracksRef.current = tracks;\n\n useEffect(() => {\n const track = tracks[currentIndex];\n if (!track) return;\n\n if (prevIndexRef.current === currentIndex) return;\n prevIndexRef.current = currentIndex;\n\n const entry: GingerPlaybackHistoryEntry = {\n track,\n index: currentIndex,\n playedAt: Date.now(),\n };\n\n setHistory((prev) => {\n const next = [...prev, entry];\n return next.length > maxLength ? next.slice(next.length - maxLength) : next;\n });\n }, [currentIndex, tracks, maxLength]);\n\n const clearHistory = useCallback(() => setHistory([]), []);\n\n return { history, clearHistory };\n}\n","import { useCallback, useRef, useState } from \"react\";\nimport { useGingerMedia } from \"../context/GingerSplitContexts\";\n\nexport type UseGingerVolumeFadeOptions = {\n /** Target volume to fade to (0–1). */\n targetVolume: number;\n /** Duration of the fade in milliseconds. */\n durationMs: number;\n /** Called when the fade completes normally (not when cancelled). */\n onComplete?: () => void;\n};\n\nexport type UseGingerVolumeFadeResult = {\n /** Start a volume fade. Cancels any in-progress fade. */\n fadeVolumeTo: (options: UseGingerVolumeFadeOptions) => void;\n /** Cancel the current fade and hold at the current volume. */\n cancelFade: () => void;\n /** True while a fade is in progress. */\n isFading: boolean;\n};\n\n/**\n * Smoothly interpolates volume over a given duration using `requestAnimationFrame`.\n * Useful for fade-in on track start, fade-out before sleep timer fires, or crossfade prep.\n */\nexport function useGingerVolumeFade(): UseGingerVolumeFadeResult {\n const { setVolume, volume } = useGingerMedia();\n const [isFading, setIsFading] = useState(false);\n\n const rafRef = useRef<number>(0);\n const cancelledRef = useRef(false);\n\n const cancelFade = useCallback(() => {\n cancelAnimationFrame(rafRef.current);\n cancelledRef.current = true;\n setIsFading(false);\n }, []);\n\n const fadeVolumeTo = useCallback(\n ({ targetVolume, durationMs, onComplete }: UseGingerVolumeFadeOptions) => {\n cancelAnimationFrame(rafRef.current);\n cancelledRef.current = false;\n\n const clamp = (v: number) => Math.min(1, Math.max(0, v));\n const target = clamp(targetVolume);\n const startTime = performance.now();\n\n // Capture start volume at the moment the fade begins\n let startVolume = volume;\n\n setIsFading(true);\n\n const tick = (now: number) => {\n if (cancelledRef.current) return;\n const elapsed = now - startTime;\n const progress = Math.min(1, elapsed / Math.max(1, durationMs));\n const current = startVolume + (target - startVolume) * progress;\n setVolume(clamp(current));\n\n if (progress < 1) {\n rafRef.current = requestAnimationFrame(tick);\n } else {\n setIsFading(false);\n onComplete?.();\n }\n };\n\n // Use a small delay so `volume` from closure is the current value\n rafRef.current = requestAnimationFrame((now) => {\n startVolume = volume;\n tick(now);\n });\n },\n [setVolume, volume],\n );\n\n return { fadeVolumeTo, cancelFade, isFading };\n}\n","import { useMemo } from \"react\";\nimport { useGingerMedia, useGingerPlayback } from \"../context/GingerSplitContexts\";\n\nexport type GingerChapterProgress = {\n /** Fraction (0–1) of the way through the active chapter. 0 when no chapter is active. */\n progress: number;\n /** Seconds elapsed since the start of the active chapter. */\n elapsed: number;\n /** Seconds remaining until the end of the active chapter (or until end of track if last chapter). */\n remaining: number;\n};\n\n/**\n * Returns detailed progress information for the currently active chapter.\n * Complements `useGingerChapters` with per-chapter playback fractions.\n */\nexport function useGingerChapterProgress(): GingerChapterProgress {\n const { tracks, currentIndex } = useGingerPlayback();\n const { currentTime, duration } = useGingerMedia();\n\n const chapters = useMemo(() => {\n const raw = tracks[currentIndex]?.chapters ?? [];\n return [...raw]\n .filter((c) => c && Number.isFinite(c.startSeconds) && c.startSeconds >= 0)\n .sort((a, b) => a.startSeconds - b.startSeconds);\n }, [tracks, currentIndex]);\n\n return useMemo<GingerChapterProgress>(() => {\n if (chapters.length === 0) return { progress: 0, elapsed: 0, remaining: 0 };\n\n // Find active chapter (last one whose start is <= currentTime)\n let activeIdx = -1;\n for (let i = chapters.length - 1; i >= 0; i--) {\n if (currentTime >= chapters[i]!.startSeconds) {\n activeIdx = i;\n break;\n }\n }\n\n if (activeIdx === -1) return { progress: 0, elapsed: 0, remaining: 0 };\n\n const chapter = chapters[activeIdx]!;\n const nextChapter = chapters[activeIdx + 1];\n const chapterEnd = nextChapter?.startSeconds ?? (duration > 0 ? duration : currentTime);\n const chapterDuration = Math.max(0, chapterEnd - chapter.startSeconds);\n const elapsed = Math.max(0, currentTime - chapter.startSeconds);\n const remaining = Math.max(0, chapterEnd - currentTime);\n const progress = chapterDuration > 0 ? Math.min(1, elapsed / chapterDuration) : 0;\n\n return { progress, elapsed, remaining };\n }, [chapters, currentTime, duration]);\n}\n"],"names":["emptyFreq","emptyTime","useGingerLiveAnalyzer","options","enabled","fftSize","smoothingTimeConstant","minDecibels","maxDecibels","audioRef","state","useGinger","opts","useMemo","frame","setFrame","useState","error","setError","isSuspended","setIsSuspended","meta","setMeta","freqRef","useRef","timeRef","resume","useCallback","ctx","contextHolderRef","analyserHolderRef","useLayoutEffect","cancelled","consumerId","element","rafId","onStateChange","runLoop","a","fq","td","n","attach","el","id","context","analyser","attachLiveAnalyser","fft","e","msg","first","retryRaf","maxAttempts","attempts","retryLoop","out","detachLiveAnalyser","_a","useGingerKeyboardShortcuts","bindings","togglePlayPause","next","prev","useGingerPlayback","toggleMute","seek","currentTime","duration","useGingerMedia","muteBinding","seekForwardBinding","seekBackwardBinding","useEffect","playPause","nextKey","prevKey","muteKey","seekFwdKey","seekBwdKey","seekSecs","onKeyDown","event","target","key","dur","useGingerSleepTimer","durationMs","stopAfterTracks","respectPause","onFire","currentIndex","pause","isPaused","remainingTracksRef","prevIndexRef","remainingMsRef","segmentStartRef","prevDurationMsRef","elapsed","useGingerDebugLog","useGingerState","prevRef","clamp01","value","useSeekDrag","media","playback","fraction","setFraction","isDragging","setIsDragging","liveFraction","progressFraction","gingerStateFromContextValues","displayFraction","onPointerDown","rect","update","clientX","ratio","onMove","moveEvent","onUp","upEvent","useNextTrackPrefetch","crossOrigin","tracks","repeatMode","playbackMode","nextIndex","computeNextIndex","nextUrl","audio","createGingerStore","createInitialState","listeners","dispatch","action","gingerReducer","listener","payload","clampVolume","clampPlaybackRate","useGingerPlaybackHistory","maxLength","history","setHistory","prevTracksRef","track","entry","clearHistory","useGingerVolumeFade","setVolume","volume","isFading","setIsFading","rafRef","cancelledRef","cancelFade","targetVolume","onComplete","clamp","v","startTime","startVolume","tick","now","progress","current","useGingerChapterProgress","chapters","c","b","activeIdx","i","chapter","nextChapter","chapterEnd","chapterDuration","remaining"],"mappings":"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"}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@lucaismyname/ginger",
|
|
3
|
-
"version": "0.0.
|
|
3
|
+
"version": "0.0.32",
|
|
4
4
|
"description": "A headless & batteries-included React audio-player primitive",
|
|
5
5
|
"type": "module",
|
|
6
6
|
"sideEffects": false,
|
|
@@ -9,7 +9,7 @@
|
|
|
9
9
|
"LICENSE",
|
|
10
10
|
"README.md"
|
|
11
11
|
],
|
|
12
|
-
"main": "./dist/index.cjs",
|
|
12
|
+
"main": "./dist/index.cjs",
|
|
13
13
|
"module": "./dist/index.js",
|
|
14
14
|
"types": "./dist/index.d.ts",
|
|
15
15
|
"exports": {
|
|
@@ -42,6 +42,21 @@
|
|
|
42
42
|
"types": "./dist/equalizer/index.d.ts",
|
|
43
43
|
"import": "./dist/equalizer/index.js",
|
|
44
44
|
"require": "./dist/equalizer/index.cjs"
|
|
45
|
+
},
|
|
46
|
+
"./spatial": {
|
|
47
|
+
"types": "./dist/spatial/index.d.ts",
|
|
48
|
+
"import": "./dist/spatial/index.js",
|
|
49
|
+
"require": "./dist/spatial/index.cjs"
|
|
50
|
+
},
|
|
51
|
+
"./transcript": {
|
|
52
|
+
"types": "./dist/transcript/index.d.ts",
|
|
53
|
+
"import": "./dist/transcript/index.js",
|
|
54
|
+
"require": "./dist/transcript/index.cjs"
|
|
55
|
+
},
|
|
56
|
+
"./remote": {
|
|
57
|
+
"types": "./dist/remote/index.d.ts",
|
|
58
|
+
"import": "./dist/remote/index.js",
|
|
59
|
+
"require": "./dist/remote/index.cjs"
|
|
45
60
|
}
|
|
46
61
|
},
|
|
47
62
|
"scripts": {
|
|
@@ -1,150 +0,0 @@
|
|
|
1
|
-
import { useMemo as g } from "react";
|
|
2
|
-
import { b as d, u as m, g as p } from "./GingerSplitContexts-BzBExb95.js";
|
|
3
|
-
import { r as y, a as h, p as k, e as v, b as A, d as b, g as x } from "./selectors-BalBCc7X.js";
|
|
4
|
-
function R() {
|
|
5
|
-
const e = d(), t = m();
|
|
6
|
-
return g(() => {
|
|
7
|
-
const n = p(e, t);
|
|
8
|
-
return {
|
|
9
|
-
state: n,
|
|
10
|
-
currentTrack: x(n),
|
|
11
|
-
playbackUi: b(n),
|
|
12
|
-
duration: A(n),
|
|
13
|
-
remaining: v(n),
|
|
14
|
-
progress: k(n),
|
|
15
|
-
artworkUrl: h(n),
|
|
16
|
-
albumLine: y(n),
|
|
17
|
-
play: e.play,
|
|
18
|
-
pause: e.pause,
|
|
19
|
-
togglePlayPause: e.togglePlayPause,
|
|
20
|
-
seek: t.seek,
|
|
21
|
-
setVolume: t.setVolume,
|
|
22
|
-
setMuted: t.setMuted,
|
|
23
|
-
toggleMute: t.toggleMute,
|
|
24
|
-
setPlaybackRate: t.setPlaybackRate,
|
|
25
|
-
next: e.next,
|
|
26
|
-
prev: e.prev,
|
|
27
|
-
setRepeatMode: e.setRepeatMode,
|
|
28
|
-
cycleRepeat: e.cycleRepeat,
|
|
29
|
-
toggleShuffle: e.toggleShuffle,
|
|
30
|
-
setQueue: e.setQueue,
|
|
31
|
-
insertTrackAt: e.insertTrackAt,
|
|
32
|
-
removeTrackAt: e.removeTrackAt,
|
|
33
|
-
moveTrack: e.moveTrack,
|
|
34
|
-
enqueueNext: e.enqueueNext,
|
|
35
|
-
playTrackAt: e.playTrackAt,
|
|
36
|
-
selectTrackAt: e.selectTrackAt,
|
|
37
|
-
setPlaylistMeta: e.setPlaylistMeta,
|
|
38
|
-
setPlaybackMode: e.setPlaybackMode,
|
|
39
|
-
init: e.init,
|
|
40
|
-
audioRef: t.audioRef,
|
|
41
|
-
dispatch: e.dispatch
|
|
42
|
-
};
|
|
43
|
-
}, [e, t]);
|
|
44
|
-
}
|
|
45
|
-
const r = /* @__PURE__ */ new WeakMap();
|
|
46
|
-
function M(e) {
|
|
47
|
-
const t = 2 ** Math.round(Math.log2(e));
|
|
48
|
-
return Math.min(32768, Math.max(32, t));
|
|
49
|
-
}
|
|
50
|
-
function C(e) {
|
|
51
|
-
const { processingChain: t } = e;
|
|
52
|
-
return t.length > 0 ? t[t.length - 1] : e.source;
|
|
53
|
-
}
|
|
54
|
-
function i(e) {
|
|
55
|
-
const { source: t, processingChain: n, consumers: s, context: a } = e;
|
|
56
|
-
try {
|
|
57
|
-
t.disconnect();
|
|
58
|
-
} catch {
|
|
59
|
-
}
|
|
60
|
-
for (const c of n)
|
|
61
|
-
try {
|
|
62
|
-
c.disconnect();
|
|
63
|
-
} catch {
|
|
64
|
-
}
|
|
65
|
-
for (const { analyser: c } of s.values())
|
|
66
|
-
try {
|
|
67
|
-
c.disconnect(a.destination);
|
|
68
|
-
} catch {
|
|
69
|
-
}
|
|
70
|
-
if (n.length > 0) {
|
|
71
|
-
t.connect(n[0]);
|
|
72
|
-
for (let c = 0; c < n.length - 1; c++)
|
|
73
|
-
n[c].connect(n[c + 1]);
|
|
74
|
-
}
|
|
75
|
-
const o = C(e);
|
|
76
|
-
if (s.size === 0)
|
|
77
|
-
o.connect(a.destination);
|
|
78
|
-
else {
|
|
79
|
-
let c = !0;
|
|
80
|
-
for (const { analyser: u } of s.values())
|
|
81
|
-
o.connect(u), c && (u.connect(a.destination), c = !1);
|
|
82
|
-
}
|
|
83
|
-
}
|
|
84
|
-
function l(e) {
|
|
85
|
-
let t = r.get(e);
|
|
86
|
-
if (!t) {
|
|
87
|
-
const n = window.AudioContext ?? window.webkitAudioContext;
|
|
88
|
-
if (!n)
|
|
89
|
-
throw new Error("Web Audio API is not available");
|
|
90
|
-
const s = new n(), a = s.createMediaElementSource(e);
|
|
91
|
-
t = {
|
|
92
|
-
context: s,
|
|
93
|
-
source: a,
|
|
94
|
-
consumers: /* @__PURE__ */ new Map(),
|
|
95
|
-
nextId: 0,
|
|
96
|
-
processingChain: [],
|
|
97
|
-
processingActive: !1
|
|
98
|
-
}, r.set(e, t);
|
|
99
|
-
}
|
|
100
|
-
return t;
|
|
101
|
-
}
|
|
102
|
-
function f(e, t) {
|
|
103
|
-
if (t.consumers.size === 0 && !t.processingActive) {
|
|
104
|
-
try {
|
|
105
|
-
t.source.disconnect();
|
|
106
|
-
} catch {
|
|
107
|
-
}
|
|
108
|
-
t.context.close(), r.delete(e);
|
|
109
|
-
}
|
|
110
|
-
}
|
|
111
|
-
function S(e, t) {
|
|
112
|
-
const n = l(e), { context: s } = n, a = s.createAnalyser();
|
|
113
|
-
a.fftSize = M(t.fftSize), a.smoothingTimeConstant = t.smoothingTimeConstant, a.minDecibels = t.minDecibels, a.maxDecibels = t.maxDecibels;
|
|
114
|
-
const o = n.nextId;
|
|
115
|
-
return n.nextId += 1, n.consumers.set(o, { analyser: a }), i(n), { id: o, context: s, analyser: a };
|
|
116
|
-
}
|
|
117
|
-
function z(e, t) {
|
|
118
|
-
const n = r.get(e);
|
|
119
|
-
if (!n) return;
|
|
120
|
-
const s = n.consumers.get(t);
|
|
121
|
-
if (s) {
|
|
122
|
-
try {
|
|
123
|
-
s.analyser.disconnect();
|
|
124
|
-
} catch {
|
|
125
|
-
}
|
|
126
|
-
if (n.consumers.delete(t), n.consumers.size === 0 && !n.processingActive) {
|
|
127
|
-
f(e, n);
|
|
128
|
-
return;
|
|
129
|
-
}
|
|
130
|
-
i(n);
|
|
131
|
-
}
|
|
132
|
-
}
|
|
133
|
-
function D(e, t) {
|
|
134
|
-
if (typeof window > "u") return;
|
|
135
|
-
if (t.length === 0) {
|
|
136
|
-
const s = r.get(e);
|
|
137
|
-
if (!s) return;
|
|
138
|
-
s.processingChain = [], s.processingActive = !1, s.consumers.size === 0 ? f(e, s) : i(s);
|
|
139
|
-
return;
|
|
140
|
-
}
|
|
141
|
-
const n = l(e);
|
|
142
|
-
n.processingChain = t, n.processingActive = !0, i(n);
|
|
143
|
-
}
|
|
144
|
-
export {
|
|
145
|
-
S as a,
|
|
146
|
-
z as d,
|
|
147
|
-
D as s,
|
|
148
|
-
R as u
|
|
149
|
-
};
|
|
150
|
-
//# sourceMappingURL=liveAudioGraph-CmEsdLgZ.js.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"liveAudioGraph-CmEsdLgZ.js","sources":["../src/hooks/useGinger.ts","../src/analyzer/liveAudioGraph.ts"],"sourcesContent":["import { useMemo } from \"react\";\nimport {\n gingerStateFromContextValues,\n useGingerMedia,\n useGingerPlayback,\n} from \"../context/GingerSplitContexts\";\nimport {\n derivePlaybackUiState,\n effectiveDuration,\n effectiveRemaining,\n getCurrentTrack,\n progressFraction,\n resolvedAlbumLine,\n resolvedArtwork,\n} from \"../internal/selectors\";\n\nexport function useGinger() {\n const pb = useGingerPlayback();\n const md = useGingerMedia();\n\n return useMemo(() => {\n const state = gingerStateFromContextValues(pb, md);\n return {\n state,\n currentTrack: getCurrentTrack(state),\n playbackUi: derivePlaybackUiState(state),\n duration: effectiveDuration(state),\n remaining: effectiveRemaining(state),\n progress: progressFraction(state),\n artworkUrl: resolvedArtwork(state),\n albumLine: resolvedAlbumLine(state),\n play: pb.play,\n pause: pb.pause,\n togglePlayPause: pb.togglePlayPause,\n seek: md.seek,\n setVolume: md.setVolume,\n setMuted: md.setMuted,\n toggleMute: md.toggleMute,\n setPlaybackRate: md.setPlaybackRate,\n next: pb.next,\n prev: pb.prev,\n setRepeatMode: pb.setRepeatMode,\n cycleRepeat: pb.cycleRepeat,\n toggleShuffle: pb.toggleShuffle,\n setQueue: pb.setQueue,\n insertTrackAt: pb.insertTrackAt,\n removeTrackAt: pb.removeTrackAt,\n moveTrack: pb.moveTrack,\n enqueueNext: pb.enqueueNext,\n playTrackAt: pb.playTrackAt,\n selectTrackAt: pb.selectTrackAt,\n setPlaylistMeta: pb.setPlaylistMeta,\n setPlaybackMode: pb.setPlaybackMode,\n init: pb.init,\n audioRef: md.audioRef,\n dispatch: pb.dispatch,\n };\n }, [pb, md]);\n}\n","/**\n * One MediaElementAudioSourceNode per HTMLAudioElement; multiple AnalyserNodes may tap the source.\n * An optional processing chain (e.g. EQ filters) can be inserted between the source and the\n * analysers via `setProcessingChain`.\n */\n\nexport type LiveAnalyserOptions = {\n fftSize: number;\n smoothingTimeConstant: number;\n minDecibels: number;\n maxDecibels: number;\n};\n\ntype Consumer = {\n analyser: AnalyserNode;\n};\n\ntype ElementEntry = {\n context: AudioContext;\n source: MediaElementAudioSourceNode;\n consumers: Map<number, Consumer>;\n nextId: number;\n /** Ordered processing nodes (e.g. BiquadFilterNode[]) inserted between source and analysers. */\n processingChain: AudioNode[];\n /** True while a processing chain is installed; prevents context teardown when no consumers exist. */\n processingActive: boolean;\n};\n\nconst entries = new WeakMap<HTMLAudioElement, ElementEntry>();\n\nfunction clampFftSize(n: number): number {\n const p = 2 ** Math.round(Math.log2(n));\n return Math.min(32768, Math.max(32, p));\n}\n\nfunction getChainTail(entry: ElementEntry): AudioNode {\n const { processingChain } = entry;\n return processingChain.length > 0 ? processingChain[processingChain.length - 1]! : entry.source;\n}\n\n/**\n * Rebuild all graph connections from scratch.\n * Call after any structural change (add/remove consumer, change processing chain).\n */\nfunction rebuildGraph(entry: ElementEntry): void {\n const { source, processingChain, consumers, context } = entry;\n\n // Disconnect source outputs\n try {\n source.disconnect();\n } catch {\n // ignore\n }\n\n // Disconnect processing chain node outputs\n for (const node of processingChain) {\n try {\n node.disconnect();\n } catch {\n // ignore\n }\n }\n\n // Disconnect all analysers from destination (we will reconnect selectively below)\n for (const { analyser } of consumers.values()) {\n try {\n analyser.disconnect(context.destination);\n } catch {\n // ignore\n }\n }\n\n // Build processing chain: source → node[0] → ... → node[N]\n if (processingChain.length > 0) {\n source.connect(processingChain[0]!);\n for (let i = 0; i < processingChain.length - 1; i++) {\n processingChain[i]!.connect(processingChain[i + 1]!);\n }\n }\n\n const tail = getChainTail(entry);\n\n if (consumers.size === 0) {\n // No analyser consumers: route tail directly to destination so audio is audible\n tail.connect(context.destination);\n } else {\n // Connect tail to all analysers; first one routes to destination (playback sink)\n let isFirst = true;\n for (const { analyser } of consumers.values()) {\n tail.connect(analyser);\n if (isFirst) {\n analyser.connect(context.destination);\n isFirst = false;\n }\n }\n }\n}\n\nfunction getOrCreateEntry(element: HTMLAudioElement): ElementEntry {\n let entry = entries.get(element);\n if (!entry) {\n const Context =\n window.AudioContext ??\n (window as unknown as { webkitAudioContext?: typeof AudioContext }).webkitAudioContext;\n if (!Context) {\n throw new Error(\"Web Audio API is not available\");\n }\n const context = new Context();\n const source = context.createMediaElementSource(element);\n entry = {\n context,\n source,\n consumers: new Map(),\n nextId: 0,\n processingChain: [],\n processingActive: false,\n };\n entries.set(element, entry);\n }\n return entry;\n}\n\nfunction maybeCloseEntry(element: HTMLAudioElement, entry: ElementEntry): void {\n if (entry.consumers.size === 0 && !entry.processingActive) {\n try {\n entry.source.disconnect();\n } catch {\n // ignore\n }\n void entry.context.close();\n entries.delete(element);\n }\n}\n\nexport function attachLiveAnalyser(\n element: HTMLAudioElement,\n options: LiveAnalyserOptions,\n): { id: number; context: AudioContext; analyser: AnalyserNode } {\n const entry = getOrCreateEntry(element);\n const { context } = entry;\n\n const analyser = context.createAnalyser();\n analyser.fftSize = clampFftSize(options.fftSize);\n analyser.smoothingTimeConstant = options.smoothingTimeConstant;\n analyser.minDecibels = options.minDecibels;\n analyser.maxDecibels = options.maxDecibels;\n\n const id = entry.nextId;\n entry.nextId += 1;\n entry.consumers.set(id, { analyser });\n\n rebuildGraph(entry);\n\n return { id, context, analyser };\n}\n\nexport function detachLiveAnalyser(element: HTMLAudioElement, id: number): void {\n const entry = entries.get(element);\n if (!entry) return;\n\n const consumer = entry.consumers.get(id);\n if (!consumer) return;\n\n try {\n consumer.analyser.disconnect();\n } catch {\n // ignore\n }\n entry.consumers.delete(id);\n\n if (entry.consumers.size === 0 && !entry.processingActive) {\n maybeCloseEntry(element, entry);\n return;\n }\n\n rebuildGraph(entry);\n}\n\n/**\n * Insert an ordered processing chain (e.g. BiquadFilterNode[]) between the audio source and the\n * analyser consumers. Pass an empty array to clear the chain.\n *\n * Safe to call while analyser consumers are active; the graph is rebuilt immediately.\n * Note: because `createMediaElementSource` can only be called once per element, the EQ and live\n * analyser share the same AudioContext. Calling both for the same element is supported.\n */\nexport function setProcessingChain(element: HTMLAudioElement, nodes: AudioNode[]): void {\n if (typeof window === \"undefined\") return;\n\n if (nodes.length === 0) {\n const entry = entries.get(element);\n if (!entry) return;\n entry.processingChain = [];\n entry.processingActive = false;\n if (entry.consumers.size === 0) {\n maybeCloseEntry(element, entry);\n } else {\n rebuildGraph(entry);\n }\n return;\n }\n\n const entry = getOrCreateEntry(element);\n entry.processingChain = nodes;\n entry.processingActive = true;\n rebuildGraph(entry);\n}\n"],"names":["useGinger","pb","useGingerPlayback","md","useGingerMedia","useMemo","state","gingerStateFromContextValues","getCurrentTrack","derivePlaybackUiState","effectiveDuration","effectiveRemaining","progressFraction","resolvedArtwork","resolvedAlbumLine","entries","clampFftSize","n","p","getChainTail","entry","processingChain","rebuildGraph","source","consumers","context","node","analyser","i","tail","isFirst","getOrCreateEntry","element","Context","maybeCloseEntry","attachLiveAnalyser","options","id","detachLiveAnalyser","consumer","setProcessingChain","nodes"],"mappings":"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"}
|
|
@@ -1,2 +0,0 @@
|
|
|
1
|
-
"use strict";const y=require("react"),l=require("./GingerSplitContexts-C7puo0M7.cjs"),i=require("./selectors-YXnP8Y8g.cjs");function h(){const e=l.useGingerPlayback(),t=l.useGingerMedia();return y.useMemo(()=>{const n=l.gingerStateFromContextValues(e,t);return{state:n,currentTrack:i.getCurrentTrack(n),playbackUi:i.derivePlaybackUiState(n),duration:i.effectiveDuration(n),remaining:i.effectiveRemaining(n),progress:i.progressFraction(n),artworkUrl:i.resolvedArtwork(n),albumLine:i.resolvedAlbumLine(n),play:e.play,pause:e.pause,togglePlayPause:e.togglePlayPause,seek:t.seek,setVolume:t.setVolume,setMuted:t.setMuted,toggleMute:t.toggleMute,setPlaybackRate:t.setPlaybackRate,next:e.next,prev:e.prev,setRepeatMode:e.setRepeatMode,cycleRepeat:e.cycleRepeat,toggleShuffle:e.toggleShuffle,setQueue:e.setQueue,insertTrackAt:e.insertTrackAt,removeTrackAt:e.removeTrackAt,moveTrack:e.moveTrack,enqueueNext:e.enqueueNext,playTrackAt:e.playTrackAt,selectTrackAt:e.selectTrackAt,setPlaylistMeta:e.setPlaylistMeta,setPlaybackMode:e.setPlaybackMode,init:e.init,audioRef:t.audioRef,dispatch:e.dispatch}},[e,t])}const a=new WeakMap;function p(e){const t=2**Math.round(Math.log2(e));return Math.min(32768,Math.max(32,t))}function m(e){const{processingChain:t}=e;return t.length>0?t[t.length-1]:e.source}function u(e){const{source:t,processingChain:n,consumers:s,context:c}=e;try{t.disconnect()}catch{}for(const r of n)try{r.disconnect()}catch{}for(const{analyser:r}of s.values())try{r.disconnect(c.destination)}catch{}if(n.length>0){t.connect(n[0]);for(let r=0;r<n.length-1;r++)n[r].connect(n[r+1])}const o=m(e);if(s.size===0)o.connect(c.destination);else{let r=!0;for(const{analyser:g}of s.values())o.connect(g),r&&(g.connect(c.destination),r=!1)}}function f(e){let t=a.get(e);if(!t){const n=window.AudioContext??window.webkitAudioContext;if(!n)throw new Error("Web Audio API is not available");const s=new n,c=s.createMediaElementSource(e);t={context:s,source:c,consumers:new Map,nextId:0,processingChain:[],processingActive:!1},a.set(e,t)}return t}function d(e,t){if(t.consumers.size===0&&!t.processingActive){try{t.source.disconnect()}catch{}t.context.close(),a.delete(e)}}function k(e,t){const n=f(e),{context:s}=n,c=s.createAnalyser();c.fftSize=p(t.fftSize),c.smoothingTimeConstant=t.smoothingTimeConstant,c.minDecibels=t.minDecibels,c.maxDecibels=t.maxDecibels;const o=n.nextId;return n.nextId+=1,n.consumers.set(o,{analyser:c}),u(n),{id:o,context:s,analyser:c}}function v(e,t){const n=a.get(e);if(!n)return;const s=n.consumers.get(t);if(s){try{s.analyser.disconnect()}catch{}if(n.consumers.delete(t),n.consumers.size===0&&!n.processingActive){d(e,n);return}u(n)}}function A(e,t){if(typeof window>"u")return;if(t.length===0){const s=a.get(e);if(!s)return;s.processingChain=[],s.processingActive=!1,s.consumers.size===0?d(e,s):u(s);return}const n=f(e);n.processingChain=t,n.processingActive=!0,u(n)}exports.attachLiveAnalyser=k;exports.detachLiveAnalyser=v;exports.setProcessingChain=A;exports.useGinger=h;
|
|
2
|
-
//# sourceMappingURL=liveAudioGraph-D1BXMv_u.cjs.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"liveAudioGraph-D1BXMv_u.cjs","sources":["../src/hooks/useGinger.ts","../src/analyzer/liveAudioGraph.ts"],"sourcesContent":["import { useMemo } from \"react\";\nimport {\n gingerStateFromContextValues,\n useGingerMedia,\n useGingerPlayback,\n} from \"../context/GingerSplitContexts\";\nimport {\n derivePlaybackUiState,\n effectiveDuration,\n effectiveRemaining,\n getCurrentTrack,\n progressFraction,\n resolvedAlbumLine,\n resolvedArtwork,\n} from \"../internal/selectors\";\n\nexport function useGinger() {\n const pb = useGingerPlayback();\n const md = useGingerMedia();\n\n return useMemo(() => {\n const state = gingerStateFromContextValues(pb, md);\n return {\n state,\n currentTrack: getCurrentTrack(state),\n playbackUi: derivePlaybackUiState(state),\n duration: effectiveDuration(state),\n remaining: effectiveRemaining(state),\n progress: progressFraction(state),\n artworkUrl: resolvedArtwork(state),\n albumLine: resolvedAlbumLine(state),\n play: pb.play,\n pause: pb.pause,\n togglePlayPause: pb.togglePlayPause,\n seek: md.seek,\n setVolume: md.setVolume,\n setMuted: md.setMuted,\n toggleMute: md.toggleMute,\n setPlaybackRate: md.setPlaybackRate,\n next: pb.next,\n prev: pb.prev,\n setRepeatMode: pb.setRepeatMode,\n cycleRepeat: pb.cycleRepeat,\n toggleShuffle: pb.toggleShuffle,\n setQueue: pb.setQueue,\n insertTrackAt: pb.insertTrackAt,\n removeTrackAt: pb.removeTrackAt,\n moveTrack: pb.moveTrack,\n enqueueNext: pb.enqueueNext,\n playTrackAt: pb.playTrackAt,\n selectTrackAt: pb.selectTrackAt,\n setPlaylistMeta: pb.setPlaylistMeta,\n setPlaybackMode: pb.setPlaybackMode,\n init: pb.init,\n audioRef: md.audioRef,\n dispatch: pb.dispatch,\n };\n }, [pb, md]);\n}\n","/**\n * One MediaElementAudioSourceNode per HTMLAudioElement; multiple AnalyserNodes may tap the source.\n * An optional processing chain (e.g. EQ filters) can be inserted between the source and the\n * analysers via `setProcessingChain`.\n */\n\nexport type LiveAnalyserOptions = {\n fftSize: number;\n smoothingTimeConstant: number;\n minDecibels: number;\n maxDecibels: number;\n};\n\ntype Consumer = {\n analyser: AnalyserNode;\n};\n\ntype ElementEntry = {\n context: AudioContext;\n source: MediaElementAudioSourceNode;\n consumers: Map<number, Consumer>;\n nextId: number;\n /** Ordered processing nodes (e.g. BiquadFilterNode[]) inserted between source and analysers. */\n processingChain: AudioNode[];\n /** True while a processing chain is installed; prevents context teardown when no consumers exist. */\n processingActive: boolean;\n};\n\nconst entries = new WeakMap<HTMLAudioElement, ElementEntry>();\n\nfunction clampFftSize(n: number): number {\n const p = 2 ** Math.round(Math.log2(n));\n return Math.min(32768, Math.max(32, p));\n}\n\nfunction getChainTail(entry: ElementEntry): AudioNode {\n const { processingChain } = entry;\n return processingChain.length > 0 ? processingChain[processingChain.length - 1]! : entry.source;\n}\n\n/**\n * Rebuild all graph connections from scratch.\n * Call after any structural change (add/remove consumer, change processing chain).\n */\nfunction rebuildGraph(entry: ElementEntry): void {\n const { source, processingChain, consumers, context } = entry;\n\n // Disconnect source outputs\n try {\n source.disconnect();\n } catch {\n // ignore\n }\n\n // Disconnect processing chain node outputs\n for (const node of processingChain) {\n try {\n node.disconnect();\n } catch {\n // ignore\n }\n }\n\n // Disconnect all analysers from destination (we will reconnect selectively below)\n for (const { analyser } of consumers.values()) {\n try {\n analyser.disconnect(context.destination);\n } catch {\n // ignore\n }\n }\n\n // Build processing chain: source → node[0] → ... → node[N]\n if (processingChain.length > 0) {\n source.connect(processingChain[0]!);\n for (let i = 0; i < processingChain.length - 1; i++) {\n processingChain[i]!.connect(processingChain[i + 1]!);\n }\n }\n\n const tail = getChainTail(entry);\n\n if (consumers.size === 0) {\n // No analyser consumers: route tail directly to destination so audio is audible\n tail.connect(context.destination);\n } else {\n // Connect tail to all analysers; first one routes to destination (playback sink)\n let isFirst = true;\n for (const { analyser } of consumers.values()) {\n tail.connect(analyser);\n if (isFirst) {\n analyser.connect(context.destination);\n isFirst = false;\n }\n }\n }\n}\n\nfunction getOrCreateEntry(element: HTMLAudioElement): ElementEntry {\n let entry = entries.get(element);\n if (!entry) {\n const Context =\n window.AudioContext ??\n (window as unknown as { webkitAudioContext?: typeof AudioContext }).webkitAudioContext;\n if (!Context) {\n throw new Error(\"Web Audio API is not available\");\n }\n const context = new Context();\n const source = context.createMediaElementSource(element);\n entry = {\n context,\n source,\n consumers: new Map(),\n nextId: 0,\n processingChain: [],\n processingActive: false,\n };\n entries.set(element, entry);\n }\n return entry;\n}\n\nfunction maybeCloseEntry(element: HTMLAudioElement, entry: ElementEntry): void {\n if (entry.consumers.size === 0 && !entry.processingActive) {\n try {\n entry.source.disconnect();\n } catch {\n // ignore\n }\n void entry.context.close();\n entries.delete(element);\n }\n}\n\nexport function attachLiveAnalyser(\n element: HTMLAudioElement,\n options: LiveAnalyserOptions,\n): { id: number; context: AudioContext; analyser: AnalyserNode } {\n const entry = getOrCreateEntry(element);\n const { context } = entry;\n\n const analyser = context.createAnalyser();\n analyser.fftSize = clampFftSize(options.fftSize);\n analyser.smoothingTimeConstant = options.smoothingTimeConstant;\n analyser.minDecibels = options.minDecibels;\n analyser.maxDecibels = options.maxDecibels;\n\n const id = entry.nextId;\n entry.nextId += 1;\n entry.consumers.set(id, { analyser });\n\n rebuildGraph(entry);\n\n return { id, context, analyser };\n}\n\nexport function detachLiveAnalyser(element: HTMLAudioElement, id: number): void {\n const entry = entries.get(element);\n if (!entry) return;\n\n const consumer = entry.consumers.get(id);\n if (!consumer) return;\n\n try {\n consumer.analyser.disconnect();\n } catch {\n // ignore\n }\n entry.consumers.delete(id);\n\n if (entry.consumers.size === 0 && !entry.processingActive) {\n maybeCloseEntry(element, entry);\n return;\n }\n\n rebuildGraph(entry);\n}\n\n/**\n * Insert an ordered processing chain (e.g. BiquadFilterNode[]) between the audio source and the\n * analyser consumers. Pass an empty array to clear the chain.\n *\n * Safe to call while analyser consumers are active; the graph is rebuilt immediately.\n * Note: because `createMediaElementSource` can only be called once per element, the EQ and live\n * analyser share the same AudioContext. Calling both for the same element is supported.\n */\nexport function setProcessingChain(element: HTMLAudioElement, nodes: AudioNode[]): void {\n if (typeof window === \"undefined\") return;\n\n if (nodes.length === 0) {\n const entry = entries.get(element);\n if (!entry) return;\n entry.processingChain = [];\n entry.processingActive = false;\n if (entry.consumers.size === 0) {\n maybeCloseEntry(element, entry);\n } else {\n rebuildGraph(entry);\n }\n return;\n }\n\n const entry = getOrCreateEntry(element);\n entry.processingChain = nodes;\n entry.processingActive = true;\n rebuildGraph(entry);\n}\n"],"names":["useGinger","pb","useGingerPlayback","md","useGingerMedia","useMemo","state","gingerStateFromContextValues","getCurrentTrack","derivePlaybackUiState","effectiveDuration","effectiveRemaining","progressFraction","resolvedArtwork","resolvedAlbumLine","entries","clampFftSize","n","p","getChainTail","entry","processingChain","rebuildGraph","source","consumers","context","node","analyser","i","tail","isFirst","getOrCreateEntry","element","Context","maybeCloseEntry","attachLiveAnalyser","options","id","detachLiveAnalyser","consumer","setProcessingChain","nodes"],"mappings":"4HAgBO,SAASA,GAAY,CAC1B,MAAMC,EAAKC,EAAAA,kBAAA,EACLC,EAAKC,EAAAA,eAAA,EAEX,OAAOC,EAAAA,QAAQ,IAAM,CACnB,MAAMC,EAAQC,EAAAA,6BAA6BN,EAAIE,CAAE,EACjD,MAAO,CACL,MAAAG,EACA,aAAcE,EAAAA,gBAAgBF,CAAK,EACnC,WAAYG,EAAAA,sBAAsBH,CAAK,EACvC,SAAUI,EAAAA,kBAAkBJ,CAAK,EACjC,UAAWK,EAAAA,mBAAmBL,CAAK,EACnC,SAAUM,EAAAA,iBAAiBN,CAAK,EAChC,WAAYO,EAAAA,gBAAgBP,CAAK,EACjC,UAAWQ,EAAAA,kBAAkBR,CAAK,EAClC,KAAML,EAAG,KACT,MAAOA,EAAG,MACV,gBAAiBA,EAAG,gBACpB,KAAME,EAAG,KACT,UAAWA,EAAG,UACd,SAAUA,EAAG,SACb,WAAYA,EAAG,WACf,gBAAiBA,EAAG,gBACpB,KAAMF,EAAG,KACT,KAAMA,EAAG,KACT,cAAeA,EAAG,cAClB,YAAaA,EAAG,YAChB,cAAeA,EAAG,cAClB,SAAUA,EAAG,SACb,cAAeA,EAAG,cAClB,cAAeA,EAAG,cAClB,UAAWA,EAAG,UACd,YAAaA,EAAG,YAChB,YAAaA,EAAG,YAChB,cAAeA,EAAG,cAClB,gBAAiBA,EAAG,gBACpB,gBAAiBA,EAAG,gBACpB,KAAMA,EAAG,KACT,SAAUE,EAAG,SACb,SAAUF,EAAG,QAAA,CAEjB,EAAG,CAACA,EAAIE,CAAE,CAAC,CACb,CC9BA,MAAMY,MAAc,QAEpB,SAASC,EAAaC,EAAmB,CACvC,MAAMC,EAAI,GAAK,KAAK,MAAM,KAAK,KAAKD,CAAC,CAAC,EACtC,OAAO,KAAK,IAAI,MAAO,KAAK,IAAI,GAAIC,CAAC,CAAC,CACxC,CAEA,SAASC,EAAaC,EAAgC,CACpD,KAAM,CAAE,gBAAAC,GAAoBD,EAC5B,OAAOC,EAAgB,OAAS,EAAIA,EAAgBA,EAAgB,OAAS,CAAC,EAAKD,EAAM,MAC3F,CAMA,SAASE,EAAaF,EAA2B,CAC/C,KAAM,CAAE,OAAAG,EAAQ,gBAAAF,EAAiB,UAAAG,EAAW,QAAAC,GAAYL,EAGxD,GAAI,CACFG,EAAO,WAAA,CACT,MAAQ,CAER,CAGA,UAAWG,KAAQL,EACjB,GAAI,CACFK,EAAK,WAAA,CACP,MAAQ,CAER,CAIF,SAAW,CAAE,SAAAC,CAAA,IAAcH,EAAU,SACnC,GAAI,CACFG,EAAS,WAAWF,EAAQ,WAAW,CACzC,MAAQ,CAER,CAIF,GAAIJ,EAAgB,OAAS,EAAG,CAC9BE,EAAO,QAAQF,EAAgB,CAAC,CAAE,EAClC,QAASO,EAAI,EAAGA,EAAIP,EAAgB,OAAS,EAAGO,IAC9CP,EAAgBO,CAAC,EAAG,QAAQP,EAAgBO,EAAI,CAAC,CAAE,CAEvD,CAEA,MAAMC,EAAOV,EAAaC,CAAK,EAE/B,GAAII,EAAU,OAAS,EAErBK,EAAK,QAAQJ,EAAQ,WAAW,MAC3B,CAEL,IAAIK,EAAU,GACd,SAAW,CAAE,SAAAH,CAAA,IAAcH,EAAU,SACnCK,EAAK,QAAQF,CAAQ,EACjBG,IACFH,EAAS,QAAQF,EAAQ,WAAW,EACpCK,EAAU,GAGhB,CACF,CAEA,SAASC,EAAiBC,EAAyC,CACjE,IAAIZ,EAAQL,EAAQ,IAAIiB,CAAO,EAC/B,GAAI,CAACZ,EAAO,CACV,MAAMa,EACJ,OAAO,cACN,OAAmE,mBACtE,GAAI,CAACA,EACH,MAAM,IAAI,MAAM,gCAAgC,EAElD,MAAMR,EAAU,IAAIQ,EACdV,EAASE,EAAQ,yBAAyBO,CAAO,EACvDZ,EAAQ,CACN,QAAAK,EACA,OAAAF,EACA,cAAe,IACf,OAAQ,EACR,gBAAiB,CAAA,EACjB,iBAAkB,EAAA,EAEpBR,EAAQ,IAAIiB,EAASZ,CAAK,CAC5B,CACA,OAAOA,CACT,CAEA,SAASc,EAAgBF,EAA2BZ,EAA2B,CAC7E,GAAIA,EAAM,UAAU,OAAS,GAAK,CAACA,EAAM,iBAAkB,CACzD,GAAI,CACFA,EAAM,OAAO,WAAA,CACf,MAAQ,CAER,CACKA,EAAM,QAAQ,MAAA,EACnBL,EAAQ,OAAOiB,CAAO,CACxB,CACF,CAEO,SAASG,EACdH,EACAI,EAC+D,CAC/D,MAAMhB,EAAQW,EAAiBC,CAAO,EAChC,CAAE,QAAAP,GAAYL,EAEdO,EAAWF,EAAQ,eAAA,EACzBE,EAAS,QAAUX,EAAaoB,EAAQ,OAAO,EAC/CT,EAAS,sBAAwBS,EAAQ,sBACzCT,EAAS,YAAcS,EAAQ,YAC/BT,EAAS,YAAcS,EAAQ,YAE/B,MAAMC,EAAKjB,EAAM,OACjB,OAAAA,EAAM,QAAU,EAChBA,EAAM,UAAU,IAAIiB,EAAI,CAAE,SAAAV,EAAU,EAEpCL,EAAaF,CAAK,EAEX,CAAE,GAAAiB,EAAI,QAAAZ,EAAS,SAAAE,CAAA,CACxB,CAEO,SAASW,EAAmBN,EAA2BK,EAAkB,CAC9E,MAAMjB,EAAQL,EAAQ,IAAIiB,CAAO,EACjC,GAAI,CAACZ,EAAO,OAEZ,MAAMmB,EAAWnB,EAAM,UAAU,IAAIiB,CAAE,EACvC,GAAKE,EAEL,IAAI,CACFA,EAAS,SAAS,WAAA,CACpB,MAAQ,CAER,CAGA,GAFAnB,EAAM,UAAU,OAAOiB,CAAE,EAErBjB,EAAM,UAAU,OAAS,GAAK,CAACA,EAAM,iBAAkB,CACzDc,EAAgBF,EAASZ,CAAK,EAC9B,MACF,CAEAE,EAAaF,CAAK,EACpB,CAUO,SAASoB,EAAmBR,EAA2BS,EAA0B,CACtF,GAAI,OAAO,OAAW,IAAa,OAEnC,GAAIA,EAAM,SAAW,EAAG,CACtB,MAAMrB,EAAQL,EAAQ,IAAIiB,CAAO,EACjC,GAAI,CAACZ,EAAO,OACZA,EAAM,gBAAkB,CAAA,EACxBA,EAAM,iBAAmB,GACrBA,EAAM,UAAU,OAAS,EAC3Bc,EAAgBF,EAASZ,CAAK,EAE9BE,EAAaF,CAAK,EAEpB,MACF,CAEA,MAAMA,EAAQW,EAAiBC,CAAO,EACtCZ,EAAM,gBAAkBqB,EACxBrB,EAAM,iBAAmB,GACzBE,EAAaF,CAAK,CACpB"}
|
|
@@ -1,2 +0,0 @@
|
|
|
1
|
-
"use strict";const t=require("react"),D=require("./liveAudioGraph-D1BXMv_u.cjs"),w=require("./GingerSplitContexts-C7puo0M7.cjs"),U=require("./selectors-YXnP8Y8g.cjs"),C=require("./ginger-NEcOSSJD.cjs"),F=new Uint8Array(0),q=new Uint8Array(0);function z(s={}){const{enabled:e=!0,fftSize:r=2048,smoothingTimeConstant:n=.8,minDecibels:u=-100,maxDecibels:c=-30}=s,{audioRef:a,state:o}=D.useGinger(),d=t.useMemo(()=>({fftSize:r,smoothingTimeConstant:n,minDecibels:u,maxDecibels:c}),[r,n,u,c]),[l,m]=t.useState(0),[f,i]=t.useState(null),[h,R]=t.useState(!1),[S,p]=t.useState({frequencyBinCount:0,sampleRate:0}),g=t.useRef(F),v=t.useRef(q),G=t.useCallback(async()=>{const x=y.current;x&&x.state==="suspended"&&await x.resume()},[]),y=t.useRef(null),I=t.useRef(null);return t.useLayoutEffect(()=>{if(!e||typeof window>"u")return;let x=!1,E=null,A=null,T=0;const N=()=>{const k=y.current;k&&R(k.state==="suspended")},V=()=>{if(x)return;const k=I.current,P=g.current,M=v.current;k&&P.length>0&&M.length>0&&(k.getByteFrequencyData(P),k.getByteTimeDomainData(M),m(L=>L+1)),T=requestAnimationFrame(V)},B=()=>{const k=a.current;if(!k||x)return"no-element";try{const{id:P,context:M,analyser:L}=D.attachLiveAnalyser(k,d);E=P,A=k,y.current=M,I.current=L,R(M.state==="suspended"),i(null),M.addEventListener("statechange",N);const b=L.frequencyBinCount,H=L.fftSize;return g.current=new Uint8Array(b),v.current=new Uint8Array(H),p({frequencyBinCount:b,sampleRate:M.sampleRate}),T=requestAnimationFrame(V),"ok"}catch(P){const M=P instanceof Error?P.message:"Failed to attach live analyser";return i(M),y.current=null,I.current=null,g.current=F,v.current=q,p({frequencyBinCount:0,sampleRate:0}),"error"}},K=B();if(K!=="ok"){let k=0;const P=120;let M=0;const L=()=>{if(x)return;const b=B();b==="ok"||b==="error"||(M+=1,!(M>=P)&&(k=requestAnimationFrame(L)))};return K==="no-element"&&(k=requestAnimationFrame(L)),()=>{var b;x=!0,cancelAnimationFrame(k),cancelAnimationFrame(T),E!=null&&A&&D.detachLiveAnalyser(A,E),(b=y.current)==null||b.removeEventListener("statechange",N),y.current=null,I.current=null,g.current=F,v.current=q}}return()=>{var k;x=!0,cancelAnimationFrame(T),E!=null&&A&&D.detachLiveAnalyser(A,E),(k=y.current)==null||k.removeEventListener("statechange",N),y.current=null,I.current=null,g.current=F,v.current=q,p({frequencyBinCount:0,sampleRate:0})}},[e,a,d,o.currentIndex]),{frequencyData:g.current,timeDomainData:v.current,frequencyBinCount:S.frequencyBinCount,sampleRate:S.sampleRate,isSuspended:h,error:f,resume:G,frame:l}}function X(s=!0,e={}){const{togglePlayPause:r,next:n,prev:u}=w.useGingerPlayback(),{toggleMute:c,seek:a,currentTime:o,duration:d}=w.useGingerMedia(),{mute:l,seekForward:m,seekBackward:f}=e;t.useEffect(()=>{if(!s||typeof window>"u")return;const i=(e.playPause??" ").toLowerCase(),h=(e.next??"ArrowRight").toLowerCase(),R=(e.previous??"ArrowLeft").toLowerCase(),S=l==null?void 0:l.toLowerCase(),p=m==null?void 0:m.toLowerCase(),g=f==null?void 0:f.toLowerCase(),v=e.seekSeconds??5,G=y=>{const I=y.target;if(I&&(["INPUT","TEXTAREA","SELECT"].includes(I.tagName)||I.isContentEditable))return;const x=y.key.toLowerCase();if(x===i)y.preventDefault(),r();else if(x===h)y.preventDefault(),n();else if(x===R)y.preventDefault(),u();else if(S&&x===S)y.preventDefault(),c();else if(p&&x===p){y.preventDefault();const E=d>0?d:Number.POSITIVE_INFINITY;a(Math.min(E,o+v))}else g&&x===g&&(y.preventDefault(),a(Math.max(0,o-v)))};return window.addEventListener("keydown",G),()=>window.removeEventListener("keydown",G)},[e.next,e.playPause,e.previous,e.seekSeconds,o,d,s,l,n,u,a,f,m,c,r])}function O(s){const{durationMs:e,stopAfterTracks:r,respectPause:n=!0,enabled:u=!0,onFire:c}=s,{currentIndex:a,pause:o,isPaused:d}=w.useGingerPlayback(),l=t.useRef(r??0),m=t.useRef(a),f=t.useRef(e??0),i=t.useRef(null);t.useEffect(()=>{l.current=r??0},[r]);const h=t.useRef(e);t.useEffect(()=>{h.current!==e&&(f.current=e??0,h.current=e)},[e]),t.useEffect(()=>{if(!u||!e||e<=0){f.current=e??0,i.current=null;return}if(n&&d){if(i.current!==null){const S=Date.now()-i.current;f.current=Math.max(0,f.current-S),i.current=null}return}i.current=Date.now();const R=setTimeout(()=>{f.current=0,i.current=null,o(),c==null||c()},f.current);return()=>{if(clearTimeout(R),i.current!==null){const S=Date.now()-i.current;f.current=Math.max(0,f.current-S),i.current=null}}},[e,u,d,c,o,n]),t.useEffect(()=>{if(!u||!r||r<=0)return;const R=m.current;m.current=a,a!==R&&(l.current-=1,l.current<=0&&(o(),c==null||c()))},[a,u,c,o,r])}function Y(s=!1){const e=w.useGingerState(),r=t.useRef(e);t.useEffect(()=>{if(!s||typeof console>"u")return;const n=r.current;n!==e&&console.debug("[ginger]",{from:{currentIndex:n.currentIndex,isPaused:n.isPaused,currentTime:n.currentTime,repeatMode:n.repeatMode},to:{currentIndex:e.currentIndex,isPaused:e.isPaused,currentTime:e.currentTime,repeatMode:e.repeatMode}}),r.current=e},[s,e])}function _(s){return Math.max(0,Math.min(1,s))}function j(s){const e=w.useGingerMedia(),r=w.useGingerPlayback(),{seek:n}=e,[u,c]=t.useState(0),[a,o]=t.useState(!1),d=U.progressFraction(w.gingerStateFromContextValues(r,e)),l=a?u:d,m=t.useCallback(f=>{if(!(s>0))return;const i=f.currentTarget,h=i.getBoundingClientRect(),R=g=>{const v=_((g-h.left)/h.width);c(v),n(v*s)};o(!0),i.setPointerCapture(f.pointerId),R(f.clientX);const S=g=>R(g.clientX),p=g=>{R(g.clientX),o(!1),i.releasePointerCapture(f.pointerId),i.removeEventListener("pointermove",S),i.removeEventListener("pointerup",p),i.removeEventListener("pointercancel",p)};i.addEventListener("pointermove",S),i.addEventListener("pointerup",p),i.addEventListener("pointercancel",p)},[s,n]);return{fraction:u,displayFraction:l,isDragging:a,onPointerDown:m}}function J(s={}){const{enabled:e=!0,crossOrigin:r}=s,{tracks:n,currentIndex:u,repeatMode:c,playbackMode:a}=w.useGingerPlayback();t.useEffect(()=>{var m;if(!e||typeof document>"u")return;const o=U.computeNextIndex({tracks:n,currentIndex:u,repeatMode:c,playbackMode:a});if(o===u)return;const d=((m=n[o])==null?void 0:m.fileUrl)??"";if(!d)return;const l=document.createElement("audio");return l.preload="auto",r&&(l.crossOrigin=r),l.src=d,l.load(),()=>{l.removeAttribute("src"),l.load()}},[e,r,n,u,c,a])}function Q(s={}){let e=C.createInitialState({tracks:s.tracks??[],currentIndex:s.currentIndex,playlistMeta:s.playlistMeta,isPaused:s.isPaused,isShuffled:s.isShuffled,repeatMode:s.repeatMode,playbackMode:s.playbackMode,volume:s.volume,muted:s.muted,playbackRate:s.playbackRate});const r=new Set,n=a=>{const o=C.gingerReducer(e,a);if(o!==e){e=o;for(const d of r)d(e)}};return{getState:()=>e,dispatch:n,subscribe:a=>(r.add(a),()=>r.delete(a)),init:a=>{n({type:"INIT",payload:a})},clampVolume:C.clampVolume,clampPlaybackRate:C.clampPlaybackRate}}function W(s={}){const{maxLength:e=50}=s,{tracks:r,currentIndex:n}=w.useGingerPlayback(),[u,c]=t.useState([]),a=t.useRef(null),o=t.useRef(r);o.current=r,t.useEffect(()=>{const l=r[n];if(!l||a.current===n)return;a.current=n;const m={track:l,index:n,playedAt:Date.now()};c(f=>{const i=[...f,m];return i.length>e?i.slice(i.length-e):i})},[n,r,e]);const d=t.useCallback(()=>c([]),[]);return{history:u,clearHistory:d}}function Z(){const{setVolume:s,volume:e}=w.useGingerMedia(),[r,n]=t.useState(!1),u=t.useRef(0),c=t.useRef(!1),a=t.useCallback(()=>{cancelAnimationFrame(u.current),c.current=!0,n(!1)},[]);return{fadeVolumeTo:t.useCallback(({targetVolume:d,durationMs:l,onComplete:m})=>{cancelAnimationFrame(u.current),c.current=!1;const f=p=>Math.min(1,Math.max(0,p)),i=f(d),h=performance.now();let R=e;n(!0);const S=p=>{if(c.current)return;const g=p-h,v=Math.min(1,g/Math.max(1,l)),G=R+(i-R)*v;s(f(G)),v<1?u.current=requestAnimationFrame(S):(n(!1),m==null||m())};u.current=requestAnimationFrame(p=>{R=e,S(p)})},[s,e]),cancelFade:a,isFading:r}}function $(){const{tracks:s,currentIndex:e}=w.useGingerPlayback(),{currentTime:r,duration:n}=w.useGingerMedia(),u=t.useMemo(()=>{var a;return[...((a=s[e])==null?void 0:a.chapters)??[]].filter(o=>o&&Number.isFinite(o.startSeconds)&&o.startSeconds>=0).sort((o,d)=>o.startSeconds-d.startSeconds)},[s,e]);return t.useMemo(()=>{if(u.length===0)return{progress:0,elapsed:0,remaining:0};let c=-1;for(let h=u.length-1;h>=0;h--)if(r>=u[h].startSeconds){c=h;break}if(c===-1)return{progress:0,elapsed:0,remaining:0};const a=u[c],o=u[c+1],d=(o==null?void 0:o.startSeconds)??(n>0?n:r),l=Math.max(0,d-a.startSeconds),m=Math.max(0,r-a.startSeconds),f=Math.max(0,d-r);return{progress:l>0?Math.min(1,m/l):0,elapsed:m,remaining:f}},[u,r,n])}exports.createGingerStore=Q;exports.useGingerChapterProgress=$;exports.useGingerDebugLog=Y;exports.useGingerKeyboardShortcuts=X;exports.useGingerLiveAnalyzer=z;exports.useGingerPlaybackHistory=W;exports.useGingerSleepTimer=O;exports.useGingerVolumeFade=Z;exports.useNextTrackPrefetch=J;exports.useSeekDrag=j;
|
|
2
|
-
//# sourceMappingURL=useGingerChapterProgress-BOqUimE7.cjs.map
|