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

586 lines
19 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-04-02 00:38:30 +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-04-02 00:38:30 +03:00
import { attachAudioPlayback } from './audioPlaybackService'
2026-03-23 14:22:44 +03:00
import { fmt } from './utils'
2026-04-04 18:33:45 +03:00
import { Header } from './components/header'
2026-03-23 14:22:44 +03:00
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
2026-04-02 00:38:30 +03:00
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)
}
2026-03-19 17:31:09 +03:00
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`)
}
2026-04-02 00:38:30 +03:00
const playback = attachAudioPlayback(audio, {
onEnded: nextTrack,
onErrorSkip: nextTrack,
onToast: showToast,
})
2026-03-19 17:31:09 +03:00
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:38:30 +03:00
void playback.loadStreamForTrack(track.slug)
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))
2026-04-02 00:38:30 +03:00
if (wasPlaying) playback.pauseAndClearSource()
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
}
2026-04-02 00:13:30 +03:00
function clearQueuePlayback() {
dispatch(clearQueue())
2026-04-02 00:38:30 +03:00
playback.pauseAndClearSource()
2026-03-19 17:31:09 +03:00
}
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-04-02 00:38:30 +03:00
if (playback.rewindCurrentTrackIfPastThreshold()) 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:38:30 +03:00
function togglePlay() {
const q = store.getState().queue
playback.togglePlay(() => {
playIndex(q.currentIndex === -1 ? 0 : q.currentIndex)
})
2026-03-19 17:31:09 +03:00
}
2026-04-02 00:38:30 +03:00
queueActionsRef.current = {
playIndex,
removeFromQueue,
moveQueueItem,
2026-03-19 17:31:09 +03:00
}
2026-04-02 00:38:30 +03:00
function onToggleShuffle() {
dispatch(toggleShuffle())
2026-03-19 17:31:09 +03:00
}
2026-04-02 00:38:30 +03:00
function onToggleRepeat() {
dispatch(toggleRepeat())
2026-03-19 17:31:09 +03:00
}
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 toggleSidebar() {
const sidebar = document.getElementById('sidebar')
const overlay = document.getElementById('sidebarOverlay')
sidebar?.classList.toggle('open')
overlay?.classList.toggle('show')
}
2026-04-02 00:48:57 +03:00
const onMediaSeekTo = (d: { seekTime?: number }) => {
if (typeof d.seekTime === 'number') {
playback.seekToTime(d.seekTime)
}
}
2026-03-19 17:31:09 +03:00
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)
2026-04-02 00:48:57 +03:00
navigator.mediaSession.setActionHandler('seekto', onMediaSeekTo as (d: any) => void)
2026-03-19 17:31:09 +03:00
} catch {
// ignore
}
}
2026-04-02 00:48:57 +03:00
const onMenuClick = () => toggleSidebar()
2026-03-19 17:31:09 +03:00
const btnMenu = document.querySelector('.btn-menu')
2026-04-02 00:48:57 +03:00
btnMenu?.addEventListener('click', onMenuClick)
2026-03-19 17:31:09 +03:00
2026-04-02 00:48:57 +03:00
const onSidebarOverlayClick = () => toggleSidebar()
2026-03-19 17:31:09 +03:00
const sidebarOverlay = document.getElementById('sidebarOverlay')
2026-04-02 00:48:57 +03:00
sidebarOverlay?.addEventListener('click', onSidebarOverlayClick)
2026-03-19 17:31:09 +03:00
const searchInput = document.getElementById('searchInput') as HTMLInputElement | null
2026-04-02 00:48:57 +03:00
const onSearchInput = (e: Event) => {
onSearch((e.target as HTMLInputElement).value)
}
const onSearchKeydown = (e: KeyboardEvent) => {
if (e.key === 'Escape') closeSearch()
}
2026-03-19 17:31:09 +03:00
if (searchInput) {
2026-04-02 00:48:57 +03:00
searchInput.addEventListener('input', onSearchInput)
searchInput.addEventListener('keydown', onSearchKeydown)
2026-03-19 17:31:09 +03:00
}
2026-04-02 00:48:57 +03:00
const onShuffleClick = () => onToggleShuffle()
const onRepeatClick = () => onToggleRepeat()
const onClearClick = () => clearQueuePlayback()
const onPrevClick = () => prevTrack()
const onPlayClick = () => togglePlay()
const onNextClick = () => nextTrack()
2026-03-19 17:31:09 +03:00
const btnShuffle = document.getElementById('btnShuffle')
2026-04-02 00:48:57 +03:00
btnShuffle?.addEventListener('click', onShuffleClick)
2026-03-19 17:31:09 +03:00
const btnRepeat = document.getElementById('btnRepeat')
2026-04-02 00:48:57 +03:00
btnRepeat?.addEventListener('click', onRepeatClick)
2026-03-19 17:31:09 +03:00
const btnClear = document.getElementById('btnClearQueue')
2026-04-02 00:48:57 +03:00
btnClear?.addEventListener('click', onClearClick)
2026-03-19 17:31:09 +03:00
const btnPrev = document.getElementById('btnPrev')
2026-04-02 00:48:57 +03:00
btnPrev?.addEventListener('click', onPrevClick)
2026-03-19 17:31:09 +03:00
const btnPlay = document.getElementById('btnPlayPause')
2026-04-02 00:48:57 +03:00
btnPlay?.addEventListener('click', onPlayClick)
2026-03-19 17:31:09 +03:00
const btnNext = document.getElementById('btnNext')
2026-04-02 00:48:57 +03:00
btnNext?.addEventListener('click', onNextClick)
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-04-02 00:38:30 +03:00
playback.dispose()
2026-04-02 00:48:57 +03:00
btnMenu?.removeEventListener('click', onMenuClick)
sidebarOverlay?.removeEventListener('click', onSidebarOverlayClick)
searchInput?.removeEventListener('input', onSearchInput)
searchInput?.removeEventListener('keydown', onSearchKeydown)
btnShuffle?.removeEventListener('click', onShuffleClick)
btnRepeat?.removeEventListener('click', onRepeatClick)
btnClear?.removeEventListener('click', onClearClick)
btnPrev?.removeEventListener('click', onPrevClick)
btnPlay?.removeEventListener('click', onPlayClick)
btnNext?.removeEventListener('click', onNextClick)
if ('mediaSession' in navigator) {
try {
navigator.mediaSession.setActionHandler('play', null)
navigator.mediaSession.setActionHandler('pause', null)
navigator.mediaSession.setActionHandler('previoustrack', null)
navigator.mediaSession.setActionHandler('nexttrack', null)
navigator.mediaSession.setActionHandler('seekto', null)
} catch {
// ignore
}
}
2026-03-19 17:31:09 +03:00
}
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)
}
/>
2026-04-04 18:49:29 +03:00
<PlayerBar
track={nowPlayingTrack}
queue={queueItemsView}
order={queueOrderView}
playingOrigIdx={queuePlayingOrigIdxView}
scrollSignal={queueScrollSignal}
onQueuePlay={(origIdx) => queueActionsRef.current?.playIndex(origIdx)}
onQueueRemove={(origIdx) =>
queueActionsRef.current?.removeFromQueue(origIdx)
}
onQueueMove={(fromPos, toPos) =>
queueActionsRef.current?.moveQueueItem(fromPos, toPos)
}
/>
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>
)
}