@lucaismyname/ginger 0.0.30 → 0.0.31
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/README.md +140 -8
- package/dist/analyzer/liveAudioGraph.d.ts +14 -1
- package/dist/analyzer/liveAudioGraph.d.ts.map +1 -1
- package/dist/analyzer/useGingerLiveAnalyzer.d.ts +6 -0
- package/dist/analyzer/useGingerLiveAnalyzer.d.ts.map +1 -1
- package/dist/client.cjs +1 -1
- package/dist/client.js +36 -29
- package/dist/client.js.map +1 -1
- package/dist/context/GingerProvider.d.ts +1 -1
- package/dist/context/GingerProvider.d.ts.map +1 -1
- package/dist/core/playbackReducer.d.ts.map +1 -1
- package/dist/equalizer/index.cjs +2 -0
- package/dist/equalizer/index.cjs.map +1 -0
- package/dist/equalizer/index.d.ts +3 -0
- package/dist/equalizer/index.d.ts.map +1 -0
- package/dist/equalizer/index.js +51 -0
- package/dist/equalizer/index.js.map +1 -0
- package/dist/equalizer/useGingerEqualizer.d.ts +41 -0
- package/dist/equalizer/useGingerEqualizer.d.ts.map +1 -0
- package/dist/ginger-L2ZFgzH4.js +2223 -0
- package/dist/ginger-L2ZFgzH4.js.map +1 -0
- package/dist/ginger-NEcOSSJD.cjs +2 -0
- package/dist/ginger-NEcOSSJD.cjs.map +1 -0
- package/dist/hooks/useGingerChapterProgress.d.ts +14 -0
- package/dist/hooks/useGingerChapterProgress.d.ts.map +1 -0
- package/dist/hooks/useGingerKeyboardShortcuts.d.ts +6 -0
- package/dist/hooks/useGingerKeyboardShortcuts.d.ts.map +1 -1
- package/dist/hooks/useGingerPlaybackHistory.d.ts +26 -0
- package/dist/hooks/useGingerPlaybackHistory.d.ts.map +1 -0
- package/dist/hooks/useGingerSleepTimer.d.ts.map +1 -1
- package/dist/hooks/useGingerVolumeFade.d.ts +22 -0
- package/dist/hooks/useGingerVolumeFade.d.ts.map +1 -0
- package/dist/index.cjs +1 -1
- package/dist/index.d.ts +18 -2
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +36 -29
- package/dist/index.js.map +1 -1
- package/dist/liveAudioGraph-CmEsdLgZ.js +150 -0
- package/dist/liveAudioGraph-CmEsdLgZ.js.map +1 -0
- package/dist/liveAudioGraph-D1BXMv_u.cjs +2 -0
- package/dist/liveAudioGraph-D1BXMv_u.cjs.map +1 -0
- package/dist/selectors-BalBCc7X.js +127 -0
- package/dist/selectors-BalBCc7X.js.map +1 -0
- package/dist/selectors-YXnP8Y8g.cjs +2 -0
- package/dist/selectors-YXnP8Y8g.cjs.map +1 -0
- package/dist/store.d.ts +46 -0
- package/dist/store.d.ts.map +1 -0
- package/dist/testing/index.cjs +1 -1
- package/dist/testing/index.js +1 -1
- package/dist/testing/mockWebAudio.d.ts +15 -1
- package/dist/testing/mockWebAudio.d.ts.map +1 -1
- package/dist/types.d.ts +30 -8
- package/dist/types.d.ts.map +1 -1
- package/dist/useGingerChapterProgress-BOqUimE7.cjs +2 -0
- package/dist/useGingerChapterProgress-BOqUimE7.cjs.map +1 -0
- package/dist/useGingerChapterProgress-DLYdGytK.js +321 -0
- package/dist/useGingerChapterProgress-DLYdGytK.js.map +1 -0
- package/package.json +7 -2
- package/dist/ginger-DYlsaFGA.js +0 -2283
- package/dist/ginger-DYlsaFGA.js.map +0 -1
- package/dist/ginger-YDZhJqet.cjs +0 -2
- package/dist/ginger-YDZhJqet.cjs.map +0 -1
- package/dist/useNextTrackPrefetch-BZdly__j.js +0 -265
- package/dist/useNextTrackPrefetch-BZdly__j.js.map +0 -1
- package/dist/useNextTrackPrefetch-pBFiWDyx.cjs +0 -2
- package/dist/useNextTrackPrefetch-pBFiWDyx.cjs.map +0 -1
|
@@ -0,0 +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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"}
|
package/package.json
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@lucaismyname/ginger",
|
|
3
|
-
"version": "0.0.
|
|
4
|
-
"description": "A
|
|
3
|
+
"version": "0.0.31",
|
|
4
|
+
"description": "A headless & batteries-included React audio-player primitive",
|
|
5
5
|
"type": "module",
|
|
6
6
|
"sideEffects": false,
|
|
7
7
|
"files": [
|
|
@@ -37,6 +37,11 @@
|
|
|
37
37
|
"types": "./dist/experimental-gapless/index.d.ts",
|
|
38
38
|
"import": "./dist/experimental-gapless/index.js",
|
|
39
39
|
"require": "./dist/experimental-gapless/index.cjs"
|
|
40
|
+
},
|
|
41
|
+
"./equalizer": {
|
|
42
|
+
"types": "./dist/equalizer/index.d.ts",
|
|
43
|
+
"import": "./dist/equalizer/index.js",
|
|
44
|
+
"require": "./dist/equalizer/index.cjs"
|
|
40
45
|
}
|
|
41
46
|
},
|
|
42
47
|
"scripts": {
|