Files
furumi-ng/furumi-node-player/client/src/FurumiPlayer.tsx
T

639 lines
21 KiB
TypeScript
Raw Normal View History

2026-03-19 17:31:09 +03:00
import { useEffect, useRef, useState, type MouseEvent as ReactMouseEvent } from 'react'
import './furumi-player.css'
2026-03-23 12:45:24 +03:00
import {
API_ROOT,
searchTracks,
preloadStream,
} from './furumiApi'
2026-04-02 00:13:30 +03:00
import { store, useAppDispatch, useAppSelector } from './store'
2026-03-23 15:51:50 +03:00
import { fetchArtists } from './store/slices/artistsSlice'
import { fetchArtistAlbums } from './store/slices/albumsSlice'
import { fetchArtistTracks } from './store/slices/artistTracksSlice'
import { fetchAlbumTracks } from './store/slices/albumTracksSlice'
import { fetchTrackDetail } from './store/slices/trackDetailSlice'
2026-04-02 00:13:30 +03:00
import {
addTrack,
addTracksBatch,
replaceQueue,
clearQueue,
playAtIndex,
removeFromQueueAt,
moveQueueItemInOrder,
toggleShuffle,
toggleRepeat,
rebuildShuffleOrder,
selectQueueOrder,
selectPlayingOrigIdx,
selectQueueScrollSignal,
selectNowPlayingTrack,
selectQueueItems,
} from './store/slices/queueSlice'
2026-03-23 14:22:44 +03:00
import { fmt } from './utils'
import { Header } from './components/Header'
import { MainPanel, type Crumb } from './components/MainPanel'
import { PlayerBar } from './components/PlayerBar'
2026-03-23 15:51:50 +03:00
import type { Track } from './types'
2026-03-19 17:31:09 +03:00
2026-03-23 12:34:27 +03:00
export function FurumiPlayer() {
2026-03-23 15:51:50 +03:00
const dispatch = useAppDispatch()
const artistsLoading = useAppSelector((s) => s.artists.loading)
const artistsError = useAppSelector((s) => s.artists.error)
const albumsLoading = useAppSelector((s) => s.albums.loading)
const albumsError = useAppSelector((s) => s.albums.error)
const albumTracksLoading = useAppSelector((s) => s.albumTracks.loading)
const albumTracksError = useAppSelector((s) => s.albumTracks.error)
2026-04-02 00:13:30 +03:00
const queueItemsView = useAppSelector(selectQueueItems)
const queueOrderView = useAppSelector(selectQueueOrder)
const queuePlayingOrigIdxView = useAppSelector(selectPlayingOrigIdx)
const queueScrollSignal = useAppSelector(selectQueueScrollSignal)
const nowPlayingTrack = useAppSelector(selectNowPlayingTrack)
2026-03-23 14:22:44 +03:00
const [breadcrumbs, setBreadcrumbs] = useState<Crumb[]>(
2026-03-19 17:31:09 +03:00
[],
)
const [libraryItems, setLibraryItems] = useState<
Array<{
key: string
className: string
icon: string
name: string
detail?: string
nameClassName?: string
onClick: () => void
button?: { title: string; onClick: (ev: ReactMouseEvent<HTMLButtonElement>) => void }
}>
>([])
const [searchResults, setSearchResults] = useState<
Array<{ result_type: string; slug: string; name: string; detail?: string }>
>([])
const [searchOpen, setSearchOpen] = useState(false)
2026-04-02 00:29:21 +03:00
const searchSelectRef = useRef<(type: string, slug: string) => void>(() => { })
2026-03-19 17:31:09 +03:00
2026-03-19 18:04:13 +03:00
const queueActionsRef = useRef<{
playIndex: (i: number) => void
removeFromQueue: (idx: number) => void
moveQueueItem: (fromPos: number, toPos: number) => void
} | null>(null)
2026-03-23 14:22:44 +03:00
const audioRef = useRef<HTMLAudioElement>(null)
2026-03-19 17:31:09 +03:00
useEffect(() => {
2026-04-02 00:13:30 +03:00
if (!nowPlayingTrack) {
document.title = 'Furumi Player'
return
}
document.title = `${nowPlayingTrack.title} — Furumi`
const coverUrl = `${API_ROOT}/tracks/${nowPlayingTrack.slug}/cover`
if ('mediaSession' in navigator) {
try {
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
navigator.mediaSession.metadata = new window.MediaMetadata({
title: nowPlayingTrack.title,
artist: nowPlayingTrack.artist || '',
album: '',
artwork: [{ src: coverUrl, sizes: '512x512' }],
})
} catch {
// ignore
}
}
}, [nowPlayingTrack])
const shuffle = useAppSelector((s) => s.queue.shuffle)
const repeatAll = useAppSelector((s) => s.queue.repeatAll)
useEffect(() => {
const btnShuffle = document.getElementById('btnShuffle')
const btnRepeat = document.getElementById('btnRepeat')
btnShuffle?.classList.toggle('active', shuffle)
btnRepeat?.classList.toggle('active', repeatAll)
}, [shuffle, repeatAll])
useEffect(() => {
2026-03-23 14:22:44 +03:00
const audioEl = audioRef.current
if (!audioEl) return
const audio = audioEl
2026-03-19 17:31:09 +03:00
let searchTimer: number | null = null
let toastTimer: number | null = null
let muted = false
2026-04-02 00:13:30 +03:00
let playbackErrorSkips = 0
const MAX_PLAYBACK_ERROR_SKIPS = 5
2026-03-19 17:31:09 +03:00
try {
const v = window.localStorage.getItem('furumi_vol')
const volSlider = document.getElementById('volSlider') as HTMLInputElement | null
if (v !== null && volSlider) {
audio.volume = Number(v) / 100
volSlider.value = v
}
} catch {
// ignore
}
audio.addEventListener('timeupdate', () => {
if (audio.duration) {
const fill = document.getElementById('progressFill')
const timeElapsed = document.getElementById('timeElapsed')
const timeDuration = document.getElementById('timeDuration')
if (fill) fill.style.width = `${(audio.currentTime / audio.duration) * 100}%`
if (timeElapsed) timeElapsed.textContent = fmt(audio.currentTime)
if (timeDuration) timeDuration.textContent = fmt(audio.duration)
}
})
audio.addEventListener('ended', () => nextTrack())
2026-04-02 00:13:30 +03:00
audio.addEventListener('playing', () => {
playbackErrorSkips = 0
})
2026-03-19 17:31:09 +03:00
audio.addEventListener('play', () => {
const btn = document.getElementById('btnPlayPause')
if (btn) btn.innerHTML = '&#9208;'
})
audio.addEventListener('pause', () => {
const btn = document.getElementById('btnPlayPause')
if (btn) btn.innerHTML = '&#9654;'
})
audio.addEventListener('error', () => {
showToast('Playback error')
2026-04-02 00:13:30 +03:00
if (playbackErrorSkips >= MAX_PLAYBACK_ERROR_SKIPS) return
playbackErrorSkips += 1
2026-03-19 17:31:09 +03:00
nextTrack()
})
async function showArtists() {
setBreadcrumb([{ label: 'Artists', action: showArtists }])
2026-03-23 15:51:50 +03:00
try {
const artists = await dispatch(fetchArtists()).unwrap()
setLibraryItems(
artists.map((a) => ({
key: `artist:${a.slug}`,
className: 'file-item dir',
icon: '👤',
name: a.name,
detail: `${a.album_count} albums`,
onClick: () => void showArtistAlbums(a.slug, a.name),
})),
)
} catch {
// Error is stored in artists.error
2026-03-19 17:31:09 +03:00
}
}
async function showArtistAlbums(artistSlug: string, artistName: string) {
setBreadcrumb([
{ label: 'Artists', action: showArtists },
{ label: artistName, action: () => showArtistAlbums(artistSlug, artistName) },
])
2026-03-23 15:51:50 +03:00
try {
const { albums } = await dispatch(fetchArtistAlbums(artistSlug)).unwrap()
const allTracksItem = {
key: `artist-all:${artistSlug}`,
className: 'file-item',
icon: '▶',
name: 'Play all tracks',
nameClassName: 'name',
onClick: () => void playAllArtistTracks(artistSlug),
2026-03-19 17:31:09 +03:00
}
2026-03-23 15:51:50 +03:00
const albumItems = albums.map((a) => {
const year = a.year ? ` (${a.year})` : ''
return {
key: `album:${a.slug}`,
className: 'file-item dir',
icon: '💿',
name: `${a.name}${year}`,
detail: `${a.track_count} tracks`,
onClick: () => void showAlbumTracks(a.slug, a.name, artistSlug, artistName),
button: {
title: 'Add album to queue',
onClick: (ev: ReactMouseEvent<HTMLButtonElement>) => {
ev.stopPropagation()
void addAlbumToQueue(a.slug)
},
},
}
})
setLibraryItems([allTracksItem, ...albumItems])
} catch {
// Error is stored in albums.error
}
2026-03-19 17:31:09 +03:00
}
async function showAlbumTracks(
albumSlug: string,
albumName: string,
artistSlug: string,
artistName: string,
) {
setBreadcrumb([
{ label: 'Artists', action: showArtists },
{ label: artistName, action: () => showArtistAlbums(artistSlug, artistName) },
{ label: albumName },
])
2026-03-23 15:51:50 +03:00
const result = await dispatch(fetchAlbumTracks(albumSlug))
if (result.meta.requestStatus === 'rejected') return
const { tracks } = result.payload as { albumSlug: string; tracks: Track[] }
2026-03-19 17:31:09 +03:00
const playAlbumItem = {
key: `album-play:${albumSlug}`,
className: 'file-item',
icon: '▶',
name: 'Play album',
onClick: () => {
void addAlbumToQueue(albumSlug, true)
},
}
2026-03-23 15:51:50 +03:00
const trackItems = tracks.map((t) => {
2026-03-19 17:31:09 +03:00
const num = t.track_number ? `${t.track_number}. ` : ''
const dur = t.duration_secs ? fmt(t.duration_secs) : ''
return {
key: `track:${t.slug}`,
className: 'file-item',
icon: '🎵',
name: `${num}${t.title}`,
detail: dur,
onClick: () => {
addTrackToQueue(
{
slug: t.slug,
title: t.title,
artist: t.artist_name,
album_slug: albumSlug,
duration: t.duration_secs,
},
true,
)
},
}
})
setLibraryItems([playAlbumItem, ...trackItems])
}
function setBreadcrumb(parts: Crumb[]) {
setBreadcrumbs(parts)
}
function addTrackToQueue(
track: {
slug: string
title: string
artist: string
album_slug: string | null
duration: number | null
},
playNow?: boolean,
) {
2026-04-02 00:13:30 +03:00
const prevIdx = store.getState().queue.currentIndex
dispatch(addTrack({ track, playNow }))
const q = store.getState().queue
if (q.currentIndex !== -1 && (playNow || q.currentIndex !== prevIdx)) {
playIndex(q.currentIndex)
2026-03-19 17:31:09 +03:00
}
}
async function addAlbumToQueue(albumSlug: string, playFirst?: boolean) {
2026-03-23 15:51:50 +03:00
const result = await dispatch(fetchAlbumTracks(albumSlug))
if (result.meta.requestStatus === 'rejected') return
const { tracks } = result.payload as { albumSlug: string; tracks: Track[] }
if (!tracks || !tracks.length) return
2026-04-02 00:13:30 +03:00
const list = tracks.map((t) => ({
slug: t.slug,
title: t.title,
artist: t.artist_name,
album_slug: t.album_slug,
duration: t.duration_secs,
}))
const prevIdx = store.getState().queue.currentIndex
dispatch(addTracksBatch({ tracks: list, playFirst }))
const q = store.getState().queue
if (q.currentIndex !== -1 && q.currentIndex !== prevIdx) {
playIndex(q.currentIndex)
}
2026-03-19 17:31:09 +03:00
showToast(`Added ${list.length} tracks`)
}
async function playAllArtistTracks(artistSlug: string) {
2026-03-23 15:51:50 +03:00
const result = await dispatch(fetchArtistTracks(artistSlug))
if (result.meta.requestStatus === 'rejected') return
const { tracks } = result.payload as { artistSlug: string; tracks: Track[] }
if (!tracks || !tracks.length) return
2026-04-02 00:13:30 +03:00
const list = tracks.map((t) => ({
2026-03-23 15:51:50 +03:00
slug: t.slug,
title: t.title,
artist: t.artist_name,
album_slug: t.album_slug,
duration: t.duration_secs,
2026-04-02 00:13:30 +03:00
}))
dispatch(replaceQueue({ items: list, playFromIndex: 0 }))
2026-03-19 17:31:09 +03:00
playIndex(0)
showToast(`Added ${list.length} tracks`)
}
function playIndex(i: number) {
2026-04-02 00:13:30 +03:00
const q = store.getState().queue
if (i < 0 || i >= q.items.length) return
dispatch(playAtIndex(i))
const track = store.getState().queue.items[i]
2026-04-02 00:29:21 +03:00
// TODO remove after auth refactor
preloadStream(track.slug).then(response => {
console.log('response', response)
audio.src = URL.createObjectURL(response?.data)
void audio.play().catch(() => { })
// Optionally revoke old object URL if needed to avoid memory leaks
// if (oldSrc?.startsWith('blob:')) {
// URL.revokeObjectURL(oldSrc)
// }
})
// audio.src = `${API_ROOT}/stream/${track.slug}`
// void audio.play().catch(() => {})
2026-03-19 17:31:09 +03:00
if (window.history && window.history.replaceState) {
const url = new URL(window.location.href)
url.searchParams.set('t', track.slug)
window.history.replaceState(null, '', url.toString())
}
}
function removeFromQueue(idx: number) {
2026-04-02 00:13:30 +03:00
const wasPlaying = store.getState().queue.currentIndex === idx
dispatch(removeFromQueueAt(idx))
if (wasPlaying) {
2026-03-19 17:31:09 +03:00
audio.pause()
audio.src = ''
}
2026-04-02 00:13:30 +03:00
}
2026-03-23 14:22:44 +03:00
2026-04-02 00:13:30 +03:00
function moveQueueItem(fromPos: number, toPos: number) {
dispatch(moveQueueItemInOrder({ fromPos, toPos }))
2026-03-19 18:04:13 +03:00
}
queueActionsRef.current = {
playIndex,
removeFromQueue,
moveQueueItem,
2026-03-19 17:31:09 +03:00
}
2026-04-02 00:13:30 +03:00
function clearQueuePlayback() {
dispatch(clearQueue())
2026-03-19 17:31:09 +03:00
audio.pause()
audio.src = ''
}
function togglePlay() {
2026-04-02 00:13:30 +03:00
const q = store.getState().queue
if (!audio.src && q.items.length) {
playIndex(q.currentIndex === -1 ? 0 : q.currentIndex)
2026-03-19 17:31:09 +03:00
return
}
if (audio.paused) void audio.play()
else audio.pause()
}
function nextTrack() {
2026-04-02 00:13:30 +03:00
const q = store.getState().queue
if (!q.items.length) return
const order = selectQueueOrder(store.getState())
const pos = order.indexOf(q.currentIndex)
2026-03-19 17:31:09 +03:00
if (pos < order.length - 1) playIndex(order[pos + 1])
2026-04-02 00:13:30 +03:00
else if (q.repeatAll) {
if (q.shuffle) dispatch(rebuildShuffleOrder())
const first = selectQueueOrder(store.getState())[0]
if (first !== undefined) playIndex(first)
2026-03-19 17:31:09 +03:00
}
}
function prevTrack() {
2026-04-02 00:13:30 +03:00
const q = store.getState().queue
if (!q.items.length) return
2026-03-19 17:31:09 +03:00
if (audio.currentTime > 3) {
audio.currentTime = 0
return
}
2026-04-02 00:13:30 +03:00
const order = selectQueueOrder(store.getState())
const pos = order.indexOf(q.currentIndex)
2026-03-19 17:31:09 +03:00
if (pos > 0) playIndex(order[pos - 1])
2026-04-02 00:13:30 +03:00
else if (q.repeatAll) playIndex(order[order.length - 1])
2026-03-19 17:31:09 +03:00
}
2026-04-02 00:13:30 +03:00
function onToggleShuffle() {
dispatch(toggleShuffle())
2026-03-19 17:31:09 +03:00
}
2026-04-02 00:13:30 +03:00
function onToggleRepeat() {
dispatch(toggleRepeat())
2026-03-19 17:31:09 +03:00
}
function seekTo(e: MouseEvent) {
if (!audio.duration) return
const bar = document.getElementById('progressBar') as HTMLDivElement | null
if (!bar) return
const rect = bar.getBoundingClientRect()
const pct = (e.clientX - rect.left) / rect.width
audio.currentTime = pct * audio.duration
}
function toggleMute() {
muted = !muted
audio.muted = muted
const volIcon = document.getElementById('volIcon')
if (volIcon) volIcon.innerHTML = muted ? '&#128263;' : '&#128266;'
}
function setVolume(v: number) {
audio.volume = v / 100
const volIcon = document.getElementById('volIcon')
if (volIcon) volIcon.innerHTML = v === 0 ? '&#128263;' : '&#128266;'
window.localStorage.setItem('furumi_vol', String(v))
}
function onSearch(q: string) {
if (searchTimer) {
window.clearTimeout(searchTimer)
}
if (q.length < 2) {
closeSearch()
return
}
searchTimer = window.setTimeout(async () => {
2026-03-23 12:45:24 +03:00
const results = await searchTracks(q)
2026-03-19 17:31:09 +03:00
if (!results || !(results as any[]).length) {
closeSearch()
return
}
setSearchResults(results as any[])
setSearchOpen(true)
}, 250)
}
function closeSearch() {
setSearchOpen(false)
setSearchResults([])
}
function onSearchSelect(type: string, slug: string) {
closeSearch()
if (type === 'artist') void showArtistAlbums(slug, '')
else if (type === 'album') void addAlbumToQueue(slug, true)
else if (type === 'track') {
addTrackToQueue(
{ slug, title: '', artist: '', album_slug: null, duration: null },
true,
)
2026-03-23 12:45:24 +03:00
void preloadStream(slug)
2026-03-19 17:31:09 +03:00
}
}
searchSelectRef.current = onSearchSelect
function showToast(msg: string) {
const t = document.getElementById('toast')
if (!t) return
t.textContent = msg
t.classList.add('show')
if (toastTimer) window.clearTimeout(toastTimer)
toastTimer = window.setTimeout(() => t.classList.remove('show'), 2500)
}
function toggleSidebar() {
const sidebar = document.getElementById('sidebar')
const overlay = document.getElementById('sidebarOverlay')
sidebar?.classList.toggle('open')
overlay?.classList.toggle('show')
}
if ('mediaSession' in navigator) {
try {
navigator.mediaSession.setActionHandler('play', togglePlay)
navigator.mediaSession.setActionHandler('pause', togglePlay)
navigator.mediaSession.setActionHandler('previoustrack', prevTrack)
navigator.mediaSession.setActionHandler('nexttrack', nextTrack)
navigator.mediaSession.setActionHandler('seekto', (d: any) => {
if (typeof d.seekTime === 'number') {
audio.currentTime = d.seekTime
}
})
} catch {
// ignore
}
}
const btnMenu = document.querySelector('.btn-menu')
btnMenu?.addEventListener('click', () => toggleSidebar())
const sidebarOverlay = document.getElementById('sidebarOverlay')
sidebarOverlay?.addEventListener('click', () => toggleSidebar())
const searchInput = document.getElementById('searchInput') as HTMLInputElement | null
if (searchInput) {
searchInput.addEventListener('input', (e) => {
onSearch((e.target as HTMLInputElement).value)
})
searchInput.addEventListener('keydown', (e: KeyboardEvent) => {
if (e.key === 'Escape') closeSearch()
})
}
const btnShuffle = document.getElementById('btnShuffle')
2026-04-02 00:13:30 +03:00
btnShuffle?.addEventListener('click', () => onToggleShuffle())
2026-03-19 17:31:09 +03:00
const btnRepeat = document.getElementById('btnRepeat')
2026-04-02 00:13:30 +03:00
btnRepeat?.addEventListener('click', () => onToggleRepeat())
2026-03-19 17:31:09 +03:00
const btnClear = document.getElementById('btnClearQueue')
2026-04-02 00:13:30 +03:00
btnClear?.addEventListener('click', () => clearQueuePlayback())
2026-03-19 17:31:09 +03:00
const btnPrev = document.getElementById('btnPrev')
btnPrev?.addEventListener('click', () => prevTrack())
const btnPlay = document.getElementById('btnPlayPause')
btnPlay?.addEventListener('click', () => togglePlay())
const btnNext = document.getElementById('btnNext')
btnNext?.addEventListener('click', () => nextTrack())
const progressBar = document.getElementById('progressBar')
progressBar?.addEventListener('click', (e) => seekTo(e as MouseEvent))
const volIcon = document.getElementById('volIcon')
volIcon?.addEventListener('click', () => toggleMute())
const volSlider = document.getElementById('volSlider') as HTMLInputElement | null
if (volSlider) {
volSlider.addEventListener('input', (e) => {
const v = Number((e.target as HTMLInputElement).value)
setVolume(v)
})
}
const clearQueueBtn = document.getElementById('btnClearQueue')
2026-04-02 00:13:30 +03:00
clearQueueBtn?.addEventListener('click', () => clearQueuePlayback())
2026-03-19 17:31:09 +03:00
2026-04-02 00:29:21 +03:00
; (async () => {
const url = new URL(window.location.href)
const urlSlug = url.searchParams.get('t')
if (urlSlug) {
try {
const { detail } = await dispatch(fetchTrackDetail(urlSlug)).unwrap()
addTrackToQueue(
{
slug: detail.slug,
title: detail.title,
artist: detail.artist_name,
album_slug: detail.album_slug,
duration: detail.duration_secs,
},
true,
)
} catch {
// fetchTrackDetail rejected — track not found or error
}
2026-03-19 17:31:09 +03:00
}
2026-04-02 00:29:21 +03:00
void showArtists()
})()
2026-03-19 17:31:09 +03:00
return () => {
2026-03-19 18:04:13 +03:00
queueActionsRef.current = null
2026-03-19 17:31:09 +03:00
audio.pause()
}
2026-04-02 00:13:30 +03:00
}, [dispatch])
2026-03-19 17:31:09 +03:00
2026-03-23 15:51:50 +03:00
const libraryLoading =
breadcrumbs.length === 1
? artistsLoading
: breadcrumbs.length === 2
? albumsLoading
: albumTracksLoading
const libraryError =
breadcrumbs.length === 1
? artistsError
: breadcrumbs.length === 2
? albumsError
: albumTracksError
2026-03-19 17:31:09 +03:00
return (
<div className="furumi-root">
2026-03-23 14:22:44 +03:00
<Header
searchOpen={searchOpen}
searchResults={searchResults}
onSearchSelect={(type, slug) => searchSelectRef.current(type, slug)}
/>
<MainPanel
breadcrumbs={breadcrumbs}
libraryLoading={libraryLoading}
libraryError={libraryError}
libraryItems={libraryItems}
queueItemsView={queueItemsView}
queueOrderView={queueOrderView}
queuePlayingOrigIdxView={queuePlayingOrigIdxView}
queueScrollSignal={queueScrollSignal}
onQueuePlay={(origIdx) => queueActionsRef.current?.playIndex(origIdx)}
onQueueRemove={(origIdx) =>
queueActionsRef.current?.removeFromQueue(origIdx)
}
onQueueMove={(fromPos, toPos) =>
queueActionsRef.current?.moveQueueItem(fromPos, toPos)
}
/>
<PlayerBar track={nowPlayingTrack} />
2026-03-19 17:31:09 +03:00
<div className="toast" id="toast" />
2026-03-23 14:22:44 +03:00
<audio ref={audioRef} />
2026-03-19 17:31:09 +03:00
</div>
)
}