You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
bazarr/frontend/src/components/modals/ManualSearchModal.tsx

331 lines
8.2 KiB

import { useEpisodesProvider, useMoviesProvider } from "@/apis/hooks";
import { usePayload } from "@/modules/redux/hooks/modal";
import { createAndDispatchTask } from "@/modules/task/utilities";
import { isMovie } from "@/utilities";
import {
faCaretDown,
faCheck,
faDownload,
faInfoCircle,
faTimes,
} from "@fortawesome/free-solid-svg-icons";
import { FontAwesomeIcon } from "@fortawesome/react-fontawesome";
import { FunctionComponent, useCallback, useMemo, useState } from "react";
import {
Badge,
Button,
Col,
Collapse,
Container,
OverlayTrigger,
Popover,
Row,
} from "react-bootstrap";
import { Column } from "react-table";
import { BaseModal, BaseModalProps, LoadingIndicator, PageTable } from "..";
import Language from "../bazarr/Language";
type SupportType = Item.Movie | Item.Episode;
interface Props<T extends SupportType> {
download: (item: T, result: SearchResultType) => Promise<void>;
}
export function ManualSearchModal<T extends SupportType>(
props: Props<T> & BaseModalProps
) {
const { download, ...modal } = props;
const item = usePayload<T>(modal.modalKey);
const [episodeId, setEpisodeId] = useState<number | undefined>(undefined);
const [radarrId, setRadarrId] = useState<number | undefined>(undefined);
const episodes = useEpisodesProvider(episodeId);
const movies = useMoviesProvider(radarrId);
const isInitial = episodeId === undefined && radarrId === undefined;
const isFetching = episodes.isFetching || movies.isFetching;
const results = useMemo(
() => [...(episodes.data ?? []), ...(movies.data ?? [])],
[episodes.data, movies.data]
);
const search = useCallback(() => {
setEpisodeId(undefined);
setRadarrId(undefined);
if (item) {
if (isMovie(item)) {
setRadarrId(item.radarrId);
movies.refetch();
} else {
setEpisodeId(item.sonarrEpisodeId);
episodes.refetch();
}
}
}, [episodes, item, movies]);
const columns = useMemo<Column<SearchResultType>[]>(
() => [
{
Header: "Score",
accessor: (d) => `${d.score}%`,
},
{
accessor: "language",
Cell: ({ row: { original }, value }) => {
const lang: Language.Info = {
code2: value,
hi: original.hearing_impaired === "True",
forced: original.forced === "True",
name: "",
};
return (
<Badge variant="secondary">
<Language.Text value={lang}></Language.Text>
</Badge>
);
},
},
{
Header: "Provider",
accessor: "provider",
Cell: (row) => {
const value = row.value;
const { url } = row.row.original;
if (url) {
return (
<a href={url} target="_blank" rel="noopener noreferrer">
{value}
</a>
);
} else {
return value;
}
},
},
{
Header: "Release",
accessor: "release_info",
className: "text-nowrap",
Cell: (row) => {
const value = row.value;
const [open, setOpen] = useState(false);
const items = useMemo(
() =>
value.slice(1).map((v, idx) => (
<span className="release-text hidden-item" key={idx}>
{v}
</span>
)),
[value]
);
if (value.length === 0) {
return <span className="text-muted">Cannot get release info</span>;
}
const cls = [
"release-container",
"d-flex",
"justify-content-between",
"align-items-center",
];
if (value.length > 1) {
cls.push("release-multi");
}
return (
<div className={cls.join(" ")} onClick={() => setOpen((o) => !o)}>
<div className="text-container">
<span className="release-text">{value[0]}</span>
<Collapse in={open}>
<div>{items}</div>
</Collapse>
</div>
{value.length > 1 && (
<FontAwesomeIcon
className="release-icon"
icon={faCaretDown}
rotation={open ? 180 : undefined}
></FontAwesomeIcon>
)}
</div>
);
},
},
{
Header: "Upload",
accessor: (d) => d.uploader ?? "-",
},
{
accessor: "matches",
Cell: (row) => {
const { matches, dont_matches } = row.row.original;
return <StateIcon matches={matches} dont={dont_matches}></StateIcon>;
},
},
{
accessor: "subtitle",
Cell: ({ row }) => {
const result = row.original;
return (
<Button
size="sm"
variant="light"
disabled={item === null}
onClick={() => {
if (!item) return;
createAndDispatchTask(
item.title,
"download-subtitles",
download,
item,
result
);
}}
>
<FontAwesomeIcon icon={faDownload}></FontAwesomeIcon>
</Button>
);
},
},
],
[download, item]
);
const content = () => {
if (isInitial) {
return (
<div className="px-4 py-5">
<p className="mb-3 small">{item?.path ?? ""}</p>
<Button variant="primary" block onClick={search}>
Start Search
</Button>
</div>
);
} else if (isFetching) {
return <LoadingIndicator animation="grow"></LoadingIndicator>;
} else {
return (
<>
<p className="mb-3 small">{item?.path ?? ""}</p>
<PageTable
emptyText="No Result"
columns={columns}
data={results}
></PageTable>
</>
);
}
};
const footer = (
<Button
variant="light"
hidden={isFetching === true || isInitial === true}
onClick={search}
>
Search Again
</Button>
);
const title = useMemo(() => {
let title = "Unknown";
if (item) {
if (item.sceneName) {
title = item.sceneName;
} else if (isMovie(item)) {
title = item.title;
} else {
title = item.title;
}
}
return `Search - ${title}`;
}, [item]);
return (
<BaseModal
closeable={isFetching === false}
size="xl"
title={title}
footer={footer}
{...modal}
>
{content()}
</BaseModal>
);
}
const StateIcon: FunctionComponent<{ matches: string[]; dont: string[] }> = ({
matches,
dont,
}) => {
let icon = faCheck;
let color = "var(--success)";
if (dont.length > 0) {
icon = faInfoCircle;
color = "var(--warning)";
}
const matchElements = useMemo(
() =>
matches.map((v, idx) => (
<p key={`match-${idx}`} className="text-nowrap m-0">
{v}
</p>
)),
[matches]
);
const dontElements = useMemo(
() =>
dont.map((v, idx) => (
<p key={`dont-${idx}`} className="text-nowrap m-0">
{v}
</p>
)),
[dont]
);
const popover = useMemo(
() => (
<Popover className="w-100" id="manual-search-matches-info">
<Popover.Content>
<Container fluid>
<Row>
<Col xs={6}>
<FontAwesomeIcon
color="var(--success)"
icon={faCheck}
></FontAwesomeIcon>
{matchElements}
</Col>
<Col xs={6}>
<FontAwesomeIcon
color="var(--danger)"
icon={faTimes}
></FontAwesomeIcon>
{dontElements}
</Col>
</Row>
</Container>
</Popover.Content>
</Popover>
),
[matchElements, dontElements]
);
return (
<OverlayTrigger overlay={popover} placement={"left"}>
<FontAwesomeIcon icon={icon} color={color}></FontAwesomeIcon>
</OverlayTrigger>
);
};