feishin/src/renderer/features/player/hooks/use-handle-playqueue-add.ts

227 lines
9 KiB
TypeScript
Raw Normal View History

2022-12-20 04:11:06 -08:00
import { useQueryClient } from '@tanstack/react-query';
2022-12-25 01:25:46 -08:00
import isElectron from 'is-electron';
2022-12-28 15:31:04 -08:00
import { nanoid } from 'nanoid/non-secure';
import { useCallback, useRef } from 'react';
import { useTranslation } from 'react-i18next';
import { queryKeys } from '/@/renderer/api/query-keys';
import { PlayersRef } from '/@/renderer/features/player/ref/players-ref';
import { updateSong } from '/@/renderer/features/player/update-remote-song';
import {
getAlbumArtistSongsById,
getAlbumSongsById,
getArtistSongsById,
getGenreSongsById,
2023-07-01 19:10:05 -07:00
getPlaylistSongsById,
getSongById,
getSongsByQuery,
} from '/@/renderer/features/player/utils';
import { useCurrentServer, usePlayerControls, usePlayerStore } from '/@/renderer/store';
import { useGeneralSettings, usePlaybackType } from '/@/renderer/store/settings.store';
2024-09-01 08:26:30 -07:00
import { setQueue, setQueueNext } from '/@/renderer/utils/set-transcoded-queue-data';
import { toast } from '/@/shared/components/toast/toast';
2025-05-20 19:23:36 -07:00
import {
instanceOfCancellationError,
LibraryItem,
QueueSong,
Song,
SongListResponse,
} from '/@/shared/types/domain-types';
import { Play, PlaybackType, PlayQueueAddOptions } from '/@/shared/types/types';
const getRootQueryKey = (itemType: LibraryItem, serverId: string) => {
2023-07-01 19:10:05 -07:00
let queryKey;
switch (itemType) {
case LibraryItem.ALBUM:
queryKey = queryKeys.songs.list(serverId);
break;
case LibraryItem.ALBUM_ARTIST:
queryKey = queryKeys.songs.list(serverId);
break;
2023-08-04 02:27:04 -07:00
case LibraryItem.GENRE:
queryKey = queryKeys.songs.list(serverId);
break;
2023-07-01 19:10:05 -07:00
case LibraryItem.PLAYLIST:
queryKey = queryKeys.playlists.songList(serverId);
break;
case LibraryItem.SONG:
queryKey = queryKeys.songs.list(serverId);
break;
default:
queryKey = queryKeys.songs.list(serverId);
break;
}
return queryKey;
};
2022-12-20 04:11:06 -08:00
const mpvPlayer = isElectron() ? window.api.mpvPlayer : null;
const addToQueue = usePlayerStore.getState().actions.addToQueue;
2022-12-20 04:11:06 -08:00
export const useHandlePlayQueueAdd = () => {
const { t } = useTranslation();
2023-07-01 19:10:05 -07:00
const queryClient = useQueryClient();
const playbackType = usePlaybackType();
2023-07-01 19:10:05 -07:00
const server = useCurrentServer();
const { play } = usePlayerControls();
const timeoutIds = useRef<null | Record<string, ReturnType<typeof setTimeout>>>({});
2023-07-01 19:10:05 -07:00
const { doubleClickQueueAll } = useGeneralSettings();
2023-07-01 19:10:05 -07:00
const handlePlayQueueAdd = useCallback(
async (options: PlayQueueAddOptions) => {
if (!server) return toast.error({ message: 'No server selected', type: 'error' });
const { byData, byItemType, initialIndex, initialSongId, playType, query } = options;
let songs: null | QueueSong[] = null;
// Allow this to be undefined for "play shuffled". If undefined, default to 0,
// otherwise, choose the selected item in the queue
let initialSongIndex: number | undefined;
let toastId: null | string = null;
2023-07-01 19:10:05 -07:00
if (byItemType) {
let songList: SongListResponse | undefined;
const { id, type: itemType } = byItemType;
2023-07-01 19:10:05 -07:00
const fetchId = nanoid();
timeoutIds.current = {
...timeoutIds.current,
[fetchId]: setTimeout(() => {
toastId = toast.info({
2023-07-01 19:10:05 -07:00
autoClose: false,
message: t('player.playbackFetchCancel', {
postProcess: 'sentenceCase',
}),
2023-07-01 19:10:05 -07:00
onClose: () => {
queryClient.cancelQueries({
exact: false,
queryKey: getRootQueryKey(itemType, server?.id),
});
},
title: t('player.playbackFetchInProgress', {
postProcess: 'sentenceCase',
}),
2023-07-01 19:10:05 -07:00
});
}, 2000),
};
try {
if (itemType === LibraryItem.PLAYLIST) {
songList = await getPlaylistSongsById({
id: id?.[0],
query,
queryClient,
server,
});
} else if (itemType === LibraryItem.ALBUM) {
songList = await getAlbumSongsById({ id, query, queryClient, server });
} else if (itemType === LibraryItem.ALBUM_ARTIST) {
songList = await getAlbumArtistSongsById({
id,
query,
queryClient,
server,
});
2025-05-06 14:43:42 -07:00
} else if (itemType === LibraryItem.ARTIST) {
songList = await getArtistSongsById({
id,
query,
queryClient,
server,
});
2023-08-04 02:27:04 -07:00
} else if (itemType === LibraryItem.GENRE) {
songList = await getGenreSongsById({ id, query, queryClient, server });
2023-07-01 19:10:05 -07:00
} else if (itemType === LibraryItem.SONG) {
if (id?.length === 1) {
songList = await getSongById({ id: id?.[0], queryClient, server });
} else if (!doubleClickQueueAll && initialSongId) {
songList = await getSongById({
id: initialSongId,
queryClient,
server,
});
2023-07-01 19:10:05 -07:00
} else {
songList = await getSongsByQuery({ query, queryClient, server });
}
}
clearTimeout(timeoutIds.current[fetchId] as ReturnType<typeof setTimeout>);
delete timeoutIds.current[fetchId];
if (toastId) {
toast.hide(toastId);
}
2023-07-01 19:10:05 -07:00
} catch (err: any) {
if (instanceOfCancellationError(err)) {
return null;
}
clearTimeout(timeoutIds.current[fetchId] as ReturnType<typeof setTimeout>);
delete timeoutIds.current[fetchId];
toast.hide(fetchId);
return toast.error({
message: err.message,
title: t('error.genericError', { postProcess: 'sentenceCase' }) as string,
2023-07-01 19:10:05 -07:00
});
}
songs =
songList?.items?.map((song: Song) => ({ ...song, uniqueId: nanoid() })) || null;
} else if (byData) {
songs = byData.map((song) => ({ ...song, uniqueId: nanoid() })) || null;
}
if (!songs || songs?.length === 0)
return toast.warn({
message: t('common.noResultsFromQuery', { postProcess: 'sentenceCase' }),
2025-05-06 13:23:29 -07:00
title: t('player.playbackFetchNoResults', { postProcess: 'sentenceCase' }),
});
2023-07-01 19:10:05 -07:00
if (initialIndex) {
initialSongIndex = initialIndex;
} else if (initialSongId) {
initialSongIndex = songs.findIndex((song) => song.id === initialSongId);
2023-05-21 08:13:48 -07:00
}
const hadSong = usePlayerStore.getState().queue.default.length > 0;
2023-07-01 19:10:05 -07:00
const playerData = addToQueue({ initialIndex: initialSongIndex, playType, songs });
2024-07-03 08:47:26 +00:00
updateSong(playerData.current.song);
2024-09-11 20:36:46 -07:00
const replacesQueue = playType === Play.NOW || playType === Play.SHUFFLE;
if (playbackType === PlaybackType.LOCAL) {
mpvPlayer!.volume(usePlayerStore.getState().volume);
2023-07-01 19:10:05 -07:00
2024-09-11 20:36:46 -07:00
if (replacesQueue || !hadSong) {
2023-08-07 14:48:02 -07:00
mpvPlayer!.pause();
2024-09-01 08:26:30 -07:00
setQueue(playerData, false);
} else {
2024-09-01 08:26:30 -07:00
setQueueNext(playerData);
2023-07-01 19:10:05 -07:00
}
} else {
const player =
playerData.current.player === 1
? PlayersRef.current?.player1
: PlayersRef.current?.player2;
const underlying = player?.getInternalPlayer();
2024-09-11 20:36:46 -07:00
if (underlying && replacesQueue) {
underlying.currentTime = 0;
}
2023-07-01 19:10:05 -07:00
}
// We should only play if the queue was empty, or we are doing play NOW
// (override the queue).
2024-09-11 20:36:46 -07:00
if (replacesQueue || !hadSong) {
play();
}
2023-07-01 19:10:05 -07:00
return null;
2023-07-01 19:10:05 -07:00
},
[doubleClickQueueAll, play, playbackType, queryClient, server, t],
2023-07-01 19:10:05 -07:00
);
return handlePlayQueueAdd;
2022-12-20 04:11:06 -08:00
};