jellytau/src/lib/stores/player.ts

255 lines
7.1 KiB
TypeScript

/**
* Player state store - Thin wrapper over Rust PlayerController
*
* This store is display-only for most fields, receiving updates from
* backend events via playerEvents.ts. User actions are sent as commands
* to the Rust backend, which drives state changes.
*
* @req: UR-005 - Control media playback (pause, play, skip, scrub)
* @req: DR-001 - Player state machine (idle, loading, playing, paused, seeking, error)
* @req: DR-009 - Audio player UI (mini player, full screen)
*/
import { writable, derived } from "svelte/store";
import type { MediaItem } from "$lib/api/types";
import { isRemoteMode } from "./playbackMode";
import { selectedSession } from "./sessions";
import { ticksToSeconds } from "$lib/utils/playbackUnits";
// Merged media item from backend (matches Rust MergedMediaItem)
export interface MergedMediaItem {
id: string;
title: string;
artist: string | null;
album: string | null;
albumId: string | null;
duration: number | null;
primaryImageTag: string | null;
mediaType: "audio" | "video";
}
export type PlayerState =
| { kind: "idle" }
| { kind: "loading"; media: MediaItem }
| { kind: "playing"; media: MediaItem; position: number; duration: number }
| { kind: "paused"; media: MediaItem; position: number; duration: number }
| { kind: "seeking"; media: MediaItem; target: number }
| { kind: "error"; media: MediaItem | null; error: string };
export type RepeatMode = "off" | "all" | "one";
interface PlayerStore {
state: PlayerState;
volume: number;
muted: boolean;
}
function createPlayerStore() {
const initialState: PlayerStore = {
state: { kind: "idle" },
volume: 1.0,
muted: false,
};
const { subscribe, set, update } = writable<PlayerStore>(initialState);
function setIdle() {
update((s) => ({ ...s, state: { kind: "idle" } }));
}
function setLoading(media: MediaItem) {
update((s) => ({ ...s, state: { kind: "loading", media } }));
}
function setPlaying(media: MediaItem, position: number, duration: number) {
update((s) => ({
...s,
state: { kind: "playing", media, position, duration },
}));
}
function setPaused(media: MediaItem, position: number, duration: number) {
update((s) => ({
...s,
state: { kind: "paused", media, position, duration },
}));
}
function setSeeking(media: MediaItem, target: number) {
update((s) => ({ ...s, state: { kind: "seeking", media, target } }));
}
function setError(error: string, media: MediaItem | null = null) {
update((s) => ({ ...s, state: { kind: "error", media, error } }));
}
function updatePosition(position: number, duration?: number) {
update((s) => {
if (s.state.kind === "playing" || s.state.kind === "paused") {
return {
...s,
state: {
...s.state,
position,
// Update duration if provided and valid
duration: duration !== undefined && duration > 0 ? duration : s.state.duration
},
};
}
return s;
});
}
function setVolume(volume: number) {
update((s) => ({ ...s, volume: Math.max(0, Math.min(1, volume)) }));
}
function setMuted(muted: boolean) {
update((s) => ({ ...s, muted }));
}
function toggleMute() {
update((s) => ({ ...s, muted: !s.muted }));
}
return {
subscribe,
setIdle,
setLoading,
setPlaying,
setPaused,
setSeeking,
setError,
updatePosition,
setVolume,
setMuted,
toggleMute,
};
}
export const player = createPlayerStore();
// Derived stores
export const playerState = derived(player, ($p) => $p.state);
export const currentMedia = derived(player, ($p) => {
const state = $p.state;
if (state.kind === "idle") return null;
return state.media;
});
export const isPlaying = derived(player, ($p) => $p.state.kind === "playing");
export const isPaused = derived(player, ($p) => $p.state.kind === "paused");
export const isLoading = derived(player, ($p) => $p.state.kind === "loading");
export const playbackPosition = derived(player, ($p) => {
const state = $p.state;
if (state.kind === "playing" || state.kind === "paused") {
return state.position;
}
return 0;
});
export const playbackDuration = derived(player, ($p) => {
const state = $p.state;
if (state.kind === "playing" || state.kind === "paused") {
return state.duration;
}
return 0;
});
export const volume = derived(player, ($p) => $p.volume);
export const isMuted = derived(player, ($p) => $p.muted);
// Merged playback state (combines local and remote based on playback mode)
// These stores replace the mergedPlaybackState.ts helper functions
/**
* Merged media item - prefers remote session when in remote mode
*/
export const mergedMedia = derived(
[isRemoteMode, selectedSession, currentMedia],
([$isRemote, $session, $local]) => {
if ($isRemote && $session?.nowPlayingItem) {
return $session.nowPlayingItem;
}
return $local;
}
);
/**
* Merged isPlaying state - prefers remote session when in remote mode
*/
export const mergedIsPlaying = derived(
[isRemoteMode, selectedSession, isPlaying],
([$isRemote, $session, $localIsPlaying]) => {
if ($isRemote && $session?.playState) {
return !$session.playState.isPaused;
}
return $localIsPlaying;
}
);
/**
* Merged position - prefers remote session when in remote mode
*/
export const mergedPosition = derived(
[isRemoteMode, selectedSession, playbackPosition],
([$isRemote, $session, $localPosition]) => {
if ($isRemote && $session?.playState) {
return ticksToSeconds($session.playState.positionTicks ?? 0);
}
return $localPosition;
}
);
/**
* Merged duration - prefers remote session when in remote mode
*/
export const mergedDuration = derived(
[isRemoteMode, selectedSession, playbackDuration],
([$isRemote, $session, $localDuration]) => {
if ($isRemote && $session?.nowPlayingItem?.runTimeTicks) {
return ticksToSeconds($session.nowPlayingItem.runTimeTicks);
}
return $localDuration;
}
);
/**
* Merged volume - prefers remote session when in remote mode
* Both local and remote use 0-1 normalized range
*/
export const mergedVolume = derived(
[isRemoteMode, selectedSession, volume],
([$isRemote, $session, $localVolume]) => {
if ($isRemote && $session?.playState) {
// Convert remote 0-100 to normalized 0-1
return ($session.playState.volumeLevel ?? 100) / 100;
}
return $localVolume;
}
);
/**
* Should show audio miniplayer - state machine gated
* Only true when:
* 1. Player is in playing or paused state (not idle, loading, error)
* 2. Current media is audio (not video: Movie or Episode)
*/
export const shouldShowAudioMiniPlayer = derived(
[player, currentMedia],
([$player, $media]) => {
const state = $player.state;
// Only show when actively playing or paused
if (state.kind !== "playing" && state.kind !== "paused") {
return false;
}
// Don't show for video content
const mediaType = $media?.type;
if (mediaType === "Movie" || mediaType === "Episode") {
return false;
}
// Show for audio content
return true;
}
);