AniX/app/components/ReleasePlayer/VoiceoverSelectorMenu.tsx

109 lines
4.1 KiB
TypeScript
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

"use client";
import { ENDPOINTS } from "#/api/config";
import { useEffect } from "react";
import { _fetchAPI } from "./PlayerParsing";
import { useUserPlayerPreferencesStore } from "#/store/player";
import { numberDeclension } from "#/api/utils";
export interface Voiceover {
id: number;
name: string;
icon: string;
episodes_count: number;
view_count: number;
pinned: boolean;
}
interface VoiceoverSelectorMenuProps {
release_id: number;
token: string | null;
setVoiceover: (state) => void;
voiceover: Voiceover;
voiceoverList: Voiceover[];
setPlayerError: (state) => void;
}
export const VoiceoverSelectorMenu = ({
release_id,
token,
setVoiceover,
voiceover,
voiceoverList,
setPlayerError,
}: VoiceoverSelectorMenuProps) => {
const playerPreferenceStore = useUserPlayerPreferencesStore();
const preferredVO = playerPreferenceStore.getPreferredVoiceover(release_id);
useEffect(() => {
const __getInfo = async () => {
let url = `${ENDPOINTS.release.episode}/${release_id}`;
if (token) {
url += `?token=${token}`;
}
const vo = await _fetchAPI(
url,
"Не удалось получить информацию о озвучках",
setPlayerError,
{ 1: "Просмотр запрещён" }
);
if (vo) {
const selectedVO =
vo.types.find((voiceover: Voiceover) => voiceover.name === preferredVO) ||
vo.types[0];
setVoiceover({
selected: selectedVO,
available: vo.types,
});
}
};
__getInfo();
// eslint-disable-next-line react-hooks/exhaustive-deps
}, [release_id, token]);
if (!voiceover) return <></>
return (
<div className="flex flex-col items-start justify-start gap-4">
<p className="text-[20px] px-2 pt-2 pb-1 font-bold">Озвучка</p>
<div className="max-h-full flex flex-col gap-4 items-start justify-start overflow-x-hidden overflow-y-auto px-2 pb-2 scrollbar-thin scrollbar-thumb-[rgb(60_60_60_/_.8)] scrollbar-track-[rgb(30_30_30_/_.8)]">
{voiceoverList && voiceoverList.length > 0 ?
voiceoverList.map((vo: Voiceover) => {
return (
<button key={`release-${release_id}-voiceover-${vo.id}`}
className={`h-fit px-2 ${voiceover.id == vo.id ? "text-white" : "text-gray-300 hover:text-gray-100"} transition-colors`}
onClick={() => {
setVoiceover({
selected: vo,
available: voiceoverList
});
playerPreferenceStore.setPreferredVoiceover(
release_id,
vo.name
);
}
}
>
<div className="flex flex-col w-full gap-1">
<div className="flex items-center gap-2">
{/* eslint-disable-next-line @next/next/no-img-element */}
{vo.icon ? <img alt="" className="w-6 h-6 rounded-full" src={vo.icon}></img> : ""}
<span className="text-[16px] leading-none whitespace-nowrap">{vo.name}</span>
{vo.pinned && (
<span className={`h-4 iconify material-symbols--push-pin ${voiceover.id == vo.id ? "bg-white" : "bg-gray-300 hover:bg-gray-100"} transition-colors`}></span>
)}
</div>
<div className="flex gap-2">
{/* eslint-disable-next-line @next/next/no-img-element */}
<span>{vo.episodes_count} {numberDeclension(vo.episodes_count, "серия", "серии", "серий")}</span>
<span>{vo.view_count} {numberDeclension(vo.view_count, "просмотр", "просмотра", "просмотров")}</span>
</div>
</div>
</button>
);
})
: ""}
</div>
</div>
);
};