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.
321 lines
9.1 KiB
321 lines
9.1 KiB
import { DownloadIcon } from '@heroicons/react/outline';
|
|
import { uniq } from 'lodash';
|
|
import Link from 'next/link';
|
|
import { useRouter } from 'next/router';
|
|
import React, { useState } from 'react';
|
|
import { defineMessages, useIntl } from 'react-intl';
|
|
import useSWR from 'swr';
|
|
import { MediaStatus } from '../../../server/constants/media';
|
|
import type { Collection } from '../../../server/models/Collection';
|
|
import useSettings from '../../hooks/useSettings';
|
|
import { Permission, useUser } from '../../hooks/useUser';
|
|
import globalMessages from '../../i18n/globalMessages';
|
|
import Error from '../../pages/_error';
|
|
import ButtonWithDropdown from '../Common/ButtonWithDropdown';
|
|
import CachedImage from '../Common/CachedImage';
|
|
import LoadingSpinner from '../Common/LoadingSpinner';
|
|
import PageTitle from '../Common/PageTitle';
|
|
import RequestModal from '../RequestModal';
|
|
import Slider from '../Slider';
|
|
import StatusBadge from '../StatusBadge';
|
|
import TitleCard from '../TitleCard';
|
|
|
|
const messages = defineMessages({
|
|
overview: 'Overview',
|
|
numberofmovies: '{count} Movies',
|
|
requestcollection: 'Request Collection',
|
|
requestcollection4k: 'Request Collection in 4K',
|
|
});
|
|
|
|
interface CollectionDetailsProps {
|
|
collection?: Collection;
|
|
}
|
|
|
|
const CollectionDetails: React.FC<CollectionDetailsProps> = ({
|
|
collection,
|
|
}) => {
|
|
const intl = useIntl();
|
|
const router = useRouter();
|
|
const settings = useSettings();
|
|
const { hasPermission } = useUser();
|
|
const [requestModal, setRequestModal] = useState(false);
|
|
const [is4k, setIs4k] = useState(false);
|
|
|
|
const {
|
|
data,
|
|
error,
|
|
mutate: revalidate,
|
|
} = useSWR<Collection>(`/api/v1/collection/${router.query.collectionId}`, {
|
|
fallbackData: collection,
|
|
revalidateOnMount: true,
|
|
});
|
|
|
|
const { data: genres } =
|
|
useSWR<{ id: number; name: string }[]>(`/api/v1/genres/movie`);
|
|
|
|
if (!data && !error) {
|
|
return <LoadingSpinner />;
|
|
}
|
|
|
|
if (!data) {
|
|
return <Error statusCode={404} />;
|
|
}
|
|
|
|
let collectionStatus = MediaStatus.UNKNOWN;
|
|
let collectionStatus4k = MediaStatus.UNKNOWN;
|
|
|
|
if (
|
|
data.parts.every(
|
|
(part) =>
|
|
part.mediaInfo && part.mediaInfo.status === MediaStatus.AVAILABLE
|
|
)
|
|
) {
|
|
collectionStatus = MediaStatus.AVAILABLE;
|
|
} else if (
|
|
data.parts.some(
|
|
(part) =>
|
|
part.mediaInfo && part.mediaInfo.status === MediaStatus.AVAILABLE
|
|
)
|
|
) {
|
|
collectionStatus = MediaStatus.PARTIALLY_AVAILABLE;
|
|
}
|
|
|
|
if (
|
|
data.parts.every(
|
|
(part) =>
|
|
part.mediaInfo && part.mediaInfo.status4k === MediaStatus.AVAILABLE
|
|
)
|
|
) {
|
|
collectionStatus4k = MediaStatus.AVAILABLE;
|
|
} else if (
|
|
data.parts.some(
|
|
(part) =>
|
|
part.mediaInfo && part.mediaInfo.status4k === MediaStatus.AVAILABLE
|
|
)
|
|
) {
|
|
collectionStatus4k = MediaStatus.PARTIALLY_AVAILABLE;
|
|
}
|
|
|
|
const hasRequestable =
|
|
hasPermission([Permission.REQUEST, Permission.REQUEST_MOVIE], {
|
|
type: 'or',
|
|
}) &&
|
|
data.parts.filter(
|
|
(part) => !part.mediaInfo || part.mediaInfo.status === MediaStatus.UNKNOWN
|
|
).length > 0;
|
|
|
|
const hasRequestable4k =
|
|
settings.currentSettings.movie4kEnabled &&
|
|
hasPermission([Permission.REQUEST_4K, Permission.REQUEST_4K_MOVIE], {
|
|
type: 'or',
|
|
}) &&
|
|
data.parts.filter(
|
|
(part) =>
|
|
!part.mediaInfo || part.mediaInfo.status4k === MediaStatus.UNKNOWN
|
|
).length > 0;
|
|
|
|
const collectionAttributes: React.ReactNode[] = [];
|
|
|
|
collectionAttributes.push(
|
|
intl.formatMessage(messages.numberofmovies, {
|
|
count: data.parts.length,
|
|
})
|
|
);
|
|
|
|
if (genres && data.parts.some((part) => part.genreIds.length)) {
|
|
collectionAttributes.push(
|
|
uniq(
|
|
data.parts.reduce(
|
|
(genresList: number[], curr) => genresList.concat(curr.genreIds),
|
|
[]
|
|
)
|
|
)
|
|
.map((genreId) => (
|
|
<Link
|
|
href={`/discover/movies/genre/${genreId}`}
|
|
key={`genre-${genreId}`}
|
|
>
|
|
<a className="hover:underline">
|
|
{genres.find((g) => g.id === genreId)?.name}
|
|
</a>
|
|
</Link>
|
|
))
|
|
.reduce((prev, curr) => (
|
|
<>
|
|
{intl.formatMessage(globalMessages.delimitedlist, {
|
|
a: prev,
|
|
b: curr,
|
|
})}
|
|
</>
|
|
))
|
|
);
|
|
}
|
|
|
|
return (
|
|
<div
|
|
className="media-page"
|
|
style={{
|
|
height: 493,
|
|
}}
|
|
>
|
|
{data.backdropPath && (
|
|
<div className="media-page-bg-image">
|
|
<CachedImage
|
|
alt=""
|
|
src={`https://image.tmdb.org/t/p/w1920_and_h800_multi_faces/${data.backdropPath}`}
|
|
layout="fill"
|
|
objectFit="cover"
|
|
priority
|
|
/>
|
|
<div
|
|
className="absolute inset-0"
|
|
style={{
|
|
backgroundImage:
|
|
'linear-gradient(180deg, rgba(17, 24, 39, 0.47) 0%, rgba(17, 24, 39, 1) 100%)',
|
|
}}
|
|
/>
|
|
</div>
|
|
)}
|
|
<PageTitle title={data.name} />
|
|
<RequestModal
|
|
tmdbId={data.id}
|
|
show={requestModal}
|
|
type="collection"
|
|
is4k={is4k}
|
|
onComplete={() => {
|
|
revalidate();
|
|
setRequestModal(false);
|
|
}}
|
|
onCancel={() => setRequestModal(false)}
|
|
/>
|
|
<div className="media-header">
|
|
<div className="media-poster">
|
|
<CachedImage
|
|
src={
|
|
data.posterPath
|
|
? `https://image.tmdb.org/t/p/w600_and_h900_bestv2${data.posterPath}`
|
|
: '/images/overseerr_poster_not_found.png'
|
|
}
|
|
alt=""
|
|
layout="responsive"
|
|
width={600}
|
|
height={900}
|
|
priority
|
|
/>
|
|
</div>
|
|
<div className="media-title">
|
|
<div className="media-status">
|
|
<StatusBadge
|
|
status={collectionStatus}
|
|
inProgress={data.parts.some(
|
|
(part) => (part.mediaInfo?.downloadStatus ?? []).length > 0
|
|
)}
|
|
/>
|
|
{settings.currentSettings.movie4kEnabled &&
|
|
hasPermission(
|
|
[Permission.REQUEST_4K, Permission.REQUEST_4K_MOVIE],
|
|
{
|
|
type: 'or',
|
|
}
|
|
) && (
|
|
<StatusBadge
|
|
status={collectionStatus4k}
|
|
is4k
|
|
inProgress={data.parts.some(
|
|
(part) =>
|
|
(part.mediaInfo?.downloadStatus4k ?? []).length > 0
|
|
)}
|
|
/>
|
|
)}
|
|
</div>
|
|
<h1>{data.name}</h1>
|
|
<span className="media-attributes">
|
|
{collectionAttributes.length > 0 &&
|
|
collectionAttributes
|
|
.map((t, k) => <span key={k}>{t}</span>)
|
|
.reduce((prev, curr) => (
|
|
<>
|
|
{prev}
|
|
<span>|</span>
|
|
{curr}
|
|
</>
|
|
))}
|
|
</span>
|
|
</div>
|
|
<div className="media-actions">
|
|
{(hasRequestable || hasRequestable4k) && (
|
|
<ButtonWithDropdown
|
|
buttonType="primary"
|
|
onClick={() => {
|
|
setRequestModal(true);
|
|
setIs4k(!hasRequestable);
|
|
}}
|
|
text={
|
|
<>
|
|
<DownloadIcon />
|
|
<span>
|
|
{intl.formatMessage(
|
|
hasRequestable
|
|
? messages.requestcollection
|
|
: messages.requestcollection4k
|
|
)}
|
|
</span>
|
|
</>
|
|
}
|
|
>
|
|
{hasRequestable && hasRequestable4k && (
|
|
<ButtonWithDropdown.Item
|
|
buttonType="primary"
|
|
onClick={() => {
|
|
setRequestModal(true);
|
|
setIs4k(true);
|
|
}}
|
|
>
|
|
<DownloadIcon />
|
|
<span>
|
|
{intl.formatMessage(messages.requestcollection4k)}
|
|
</span>
|
|
</ButtonWithDropdown.Item>
|
|
)}
|
|
</ButtonWithDropdown>
|
|
)}
|
|
</div>
|
|
</div>
|
|
{data.overview && (
|
|
<div className="media-overview">
|
|
<div className="flex-1">
|
|
<h2>{intl.formatMessage(messages.overview)}</h2>
|
|
<p>{data.overview}</p>
|
|
</div>
|
|
</div>
|
|
)}
|
|
<div className="slider-header">
|
|
<div className="slider-title">
|
|
<span>{intl.formatMessage(globalMessages.movies)}</span>
|
|
</div>
|
|
</div>
|
|
<Slider
|
|
sliderKey="collection-movies"
|
|
isLoading={false}
|
|
isEmpty={data.parts.length === 0}
|
|
items={data.parts.map((title) => (
|
|
<TitleCard
|
|
key={`collection-movie-${title.id}`}
|
|
id={title.id}
|
|
image={title.posterPath}
|
|
status={title.mediaInfo?.status}
|
|
summary={title.overview}
|
|
title={title.title}
|
|
userScore={title.voteAverage}
|
|
year={title.releaseDate}
|
|
mediaType={title.mediaType}
|
|
/>
|
|
))}
|
|
/>
|
|
<div className="pb-8" />
|
|
</div>
|
|
);
|
|
};
|
|
|
|
export default CollectionDetails;
|