Move all lyrics fetching logic into query

This commit is contained in:
jeffvli 2023-06-05 02:50:01 -07:00 committed by Jeff
parent f10912d930
commit 3f78c3f420
2 changed files with 120 additions and 131 deletions

View file

@ -1,132 +1,33 @@
import { useCallback, useEffect, useRef, useState } from 'react';
import { Center, Group } from '@mantine/core'; import { Center, Group } from '@mantine/core';
import isElectron from 'is-electron';
import { ErrorBoundary } from 'react-error-boundary'; import { ErrorBoundary } from 'react-error-boundary';
import { ErrorFallback } from '/@/renderer/features/action-required';
import { getServerById, useCurrentSong } from '/@/renderer/store';
import { UnsynchronizedLyrics } from '/@/renderer/features/lyrics/unsynchronized-lyrics';
import { RiInformationFill } from 'react-icons/ri'; import { RiInformationFill } from 'react-icons/ri';
import { TextTitle } from '/@/renderer/components'; import { useSongLyrics } from './queries/lyric-query';
import {
InternetProviderLyricResponse,
LyricOverride,
LyricsResponse,
SynchronizedLyricsArray,
} from '/@/renderer/api/types';
import { useSongLyrics } from '/@/renderer/features/lyrics/queries/lyric-query';
import { SynchronizedLyrics } from './synchronized-lyrics'; import { SynchronizedLyrics } from './synchronized-lyrics';
import { Spinner, TextTitle } from '/@/renderer/components';
const lyrics = isElectron() ? window.electron.lyrics : null; import { ErrorFallback } from '/@/renderer/features/action-required';
import { UnsynchronizedLyrics } from '/@/renderer/features/lyrics/unsynchronized-lyrics';
const ipc = isElectron() ? window.electron.ipc : null; import { getServerById, useCurrentSong } from '/@/renderer/store';
// use by https://github.com/ustbhuangyi/lyric-parser
const timeExp = /\[(\d{2,}):(\d{2})(?:\.(\d{2,3}))?]([^\n]+)\n/g;
export const Lyrics = () => { export const Lyrics = () => {
const currentSong = useCurrentSong(); const currentSong = useCurrentSong();
const currentServer = getServerById(currentSong?.serverId); const currentServer = getServerById(currentSong?.serverId);
const [overrideLyrics, setOverrideLyrics] = useState<string | null>(null); const { data, isLoading } = useSongLyrics(
const [overrideData, setOverrideData] = useState<LyricOverride | null>(null); {
const [source, setSource] = useState<string | null>(null); query: { songId: currentSong?.id || '' },
const [songLyrics, setSongLyrics] = useState<LyricsResponse | null>(null);
const remoteLyrics = useSongLyrics({
options: { enabled: !!currentSong },
query: { songId: currentSong?.id ?? '' },
serverId: currentServer?.id, serverId: currentServer?.id,
});
const songRef = useRef<string | null>(null);
useEffect(() => {
lyrics?.remoteLyricsListener(
(
_event: any,
songName: string,
lyricSource: string,
lyric: InternetProviderLyricResponse,
) => {
if (songName === songRef.current) {
const { lyrics, ...rest } = lyric;
setSource(lyricSource);
setOverrideData(rest);
setOverrideLyrics(lyrics);
}
}, },
currentSong,
); );
return () => {
ipc?.removeAllListeners('lyric-get');
};
}, []);
useEffect(() => {
const hasTaggedLyrics = currentSong && currentSong.lyrics;
const hasLyricsResponse =
!remoteLyrics.isLoading && remoteLyrics?.isSuccess && remoteLyrics?.data !== null;
if (!hasTaggedLyrics && !hasLyricsResponse) {
lyrics?.fetchRemoteLyrics(currentSong);
}
songRef.current = currentSong?.name ?? null;
setOverrideData(null);
setOverrideLyrics(null);
setSource(null);
}, [currentSong, remoteLyrics.isLoading, remoteLyrics?.data, remoteLyrics?.isSuccess]);
useEffect(() => {
let lyrics: string | null = null;
if (currentSong?.lyrics) {
lyrics = currentSong.lyrics;
setSource(currentServer?.name ?? 'music server');
} else if (overrideLyrics) {
lyrics = overrideLyrics;
} else if (remoteLyrics.data) {
setSource(currentServer?.name ?? 'music server');
setSongLyrics(remoteLyrics.data);
return;
}
if (lyrics) {
const synchronizedLines = lyrics.matchAll(timeExp);
const synchronizedTimes: SynchronizedLyricsArray = [];
for (const line of synchronizedLines) {
const [, minute, sec, ms, text] = line;
const minutes = parseInt(minute, 10);
const seconds = parseInt(sec, 10);
const milis = ms?.length === 3 ? parseInt(ms, 10) : parseInt(ms, 10) * 10;
const timeInMilis = (minutes * 60 + seconds) * 1000 + milis;
synchronizedTimes.push([timeInMilis, text]);
}
if (synchronizedTimes.length === 0) {
setSongLyrics(lyrics);
} else {
setSongLyrics(synchronizedTimes);
}
} else {
setSongLyrics(null);
}
}, [currentServer?.name, currentSong, overrideLyrics, remoteLyrics.data]);
const clearOverride = useCallback(() => {
setOverrideData(null);
setOverrideLyrics(null);
}, []);
return ( return (
<ErrorBoundary FallbackComponent={ErrorFallback}> <ErrorBoundary FallbackComponent={ErrorFallback}>
{!songLyrics ? ( {isLoading ? (
<Spinner
container
size={25}
/>
) : !data?.lyrics ? (
<Center p="2rem"> <Center p="2rem">
<Group> <Group>
<RiInformationFill size="2rem" /> <RiInformationFill size="2rem" />
@ -140,19 +41,19 @@ export const Lyrics = () => {
</Center> </Center>
) : ( ) : (
<> <>
{Array.isArray(songLyrics) ? ( {Array.isArray(data.lyrics) ? (
<SynchronizedLyrics <SynchronizedLyrics
lyrics={songLyrics} lyrics={data.lyrics}
override={overrideData} override={null}
source={source} source={data.source}
onRemoveLyric={clearOverride} onRemoveLyric={() => {}}
/> />
) : ( ) : (
<UnsynchronizedLyrics <UnsynchronizedLyrics
lyrics={songLyrics} lyrics={data.lyrics}
override={overrideData} override={null}
source={source} source={data.source}
onRemoveLyric={clearOverride} onRemoveLyric={() => {}}
/> />
)} )}
</> </>

View file

@ -1,12 +1,43 @@
import { useQuery } from '@tanstack/react-query'; import { useQuery } from '@tanstack/react-query';
import { LyricsQuery } from '/@/renderer/api/types'; import {
LyricsQuery,
QueueSong,
SynchronizedLyricsArray,
InternetProviderLyricResponse,
} from '/@/renderer/api/types';
import { QueryHookArgs } from '/@/renderer/lib/react-query'; import { QueryHookArgs } from '/@/renderer/lib/react-query';
import { getServerById } from '/@/renderer/store'; import { getServerById, useLyricsSettings } from '/@/renderer/store';
import { controller } from '/@/renderer/api/controller';
import { queryKeys } from '/@/renderer/api/query-keys'; import { queryKeys } from '/@/renderer/api/query-keys';
import { ServerType } from '/@/renderer/types'; import { ServerType } from '/@/renderer/types';
import { api } from '/@/renderer/api';
import isElectron from 'is-electron';
export const useSongLyrics = (args: QueryHookArgs<LyricsQuery>) => { const lyricsIpc = isElectron() ? window.electron.lyrics : null;
// use by https://github.com/ustbhuangyi/lyric-parser
const timeExp = /\[(\d{2,}):(\d{2})(?:\.(\d{2,3}))?]([^\n]+)\n/g;
const formatLyrics = (lyrics: string) => {
const synchronizedLines = lyrics.matchAll(timeExp);
const formattedLyrics: SynchronizedLyricsArray = [];
for (const line of synchronizedLines) {
const [, minute, sec, ms, text] = line;
const minutes = parseInt(minute, 10);
const seconds = parseInt(sec, 10);
const milis = ms?.length === 3 ? parseInt(ms, 10) : parseInt(ms, 10) * 10;
const timeInMilis = (minutes * 60 + seconds) * 1000 + milis;
formattedLyrics.push([timeInMilis, text]);
}
// If no synchronized lyrics were found, return the original lyrics
if (formattedLyrics.length === 0) return lyrics;
return formattedLyrics;
};
export const useServerLyrics = (args: QueryHookArgs<LyricsQuery>) => {
const { query, serverId } = args; const { query, serverId } = args;
const server = getServerById(serverId); const server = getServerById(serverId);
@ -18,8 +49,65 @@ export const useSongLyrics = (args: QueryHookArgs<LyricsQuery>) => {
if (!server) throw new Error('Server not found'); if (!server) throw new Error('Server not found');
// This should only be called for Jellyfin. Return null to ignore errors // This should only be called for Jellyfin. Return null to ignore errors
if (server.type !== ServerType.JELLYFIN) return null; if (server.type !== ServerType.JELLYFIN) return null;
return controller.getLyrics({ apiClientProps: { server, signal }, query }); return api.controller.getLyrics({ apiClientProps: { server, signal }, query });
}, },
queryKey: queryKeys.songs.lyrics(server?.id || '', query), queryKey: queryKeys.songs.lyrics(server?.id || '', query),
}); });
}; };
export const useSongLyrics = (args: QueryHookArgs<LyricsQuery>, song: QueueSong | undefined) => {
const { query } = args;
const { fetch } = useLyricsSettings();
const server = getServerById(song?.serverId);
return useQuery({
cacheTime: 1000 * 60 * 10,
enabled: !!song && !!server,
onError: () => {},
queryFn: async ({ signal }) => {
if (!server) throw new Error('Server not found');
if (!song) return null;
if (song.lyrics) {
return {
artist: song.artists?.[0]?.name,
lyrics: formatLyrics(song.lyrics),
name: song.name,
source: server?.name ?? 'music server',
};
}
if (server.type === ServerType.JELLYFIN) {
const jfLyrics = await api.controller.getLyrics({
apiClientProps: { server, signal },
query: { songId: song.id },
});
if (jfLyrics) {
return {
artist: song.artists?.[0]?.name,
lyrics: jfLyrics,
name: song.name,
source: server?.name ?? 'music server',
};
}
}
if (fetch) {
const remoteLyricsResult: InternetProviderLyricResponse | null =
await lyricsIpc?.fetchRemoteLyrics(song);
if (remoteLyricsResult) {
return {
...remoteLyricsResult,
lyrics: formatLyrics(remoteLyricsResult.lyrics),
};
}
}
return null;
},
queryKey: queryKeys.songs.lyrics(server?.id || '', query),
staleTime: 1000 * 60 * 2,
});
};