|
|
|
import React, { useState, useContext, useMemo } from 'react';
|
|
|
|
import {
|
|
|
|
defineMessages,
|
|
|
|
FormattedNumber,
|
|
|
|
FormattedDate,
|
|
|
|
useIntl,
|
|
|
|
} from 'react-intl';
|
|
|
|
import type { MovieDetails as MovieDetailsType } from '../../../server/models/Movie';
|
|
|
|
import useSWR from 'swr';
|
|
|
|
import { useRouter } from 'next/router';
|
|
|
|
import Button from '../Common/Button';
|
|
|
|
import Link from 'next/link';
|
|
|
|
import Slider from '../Slider';
|
|
|
|
import PersonCard from '../PersonCard';
|
|
|
|
import { LanguageContext } from '../../context/LanguageContext';
|
|
|
|
import LoadingSpinner from '../Common/LoadingSpinner';
|
|
|
|
import { useUser, Permission } from '../../hooks/useUser';
|
|
|
|
import { MediaStatus } from '../../../server/constants/media';
|
|
|
|
import axios from 'axios';
|
|
|
|
import SlideOver from '../Common/SlideOver';
|
|
|
|
import RequestBlock from '../RequestBlock';
|
|
|
|
import TmdbLogo from '../../assets/tmdb_logo.svg';
|
|
|
|
import RTFresh from '../../assets/rt_fresh.svg';
|
|
|
|
import RTRotten from '../../assets/rt_rotten.svg';
|
|
|
|
import RTAudFresh from '../../assets/rt_aud_fresh.svg';
|
|
|
|
import RTAudRotten from '../../assets/rt_aud_rotten.svg';
|
|
|
|
import type { RTRating } from '../../../server/api/rottentomatoes';
|
|
|
|
import Error from '../../pages/_error';
|
|
|
|
import ExternalLinkBlock from '../ExternalLinkBlock';
|
|
|
|
import { sortCrewPriority } from '../../utils/creditHelpers';
|
|
|
|
import StatusBadge from '../StatusBadge';
|
|
|
|
import RequestButton from '../RequestButton';
|
|
|
|
import MediaSlider from '../MediaSlider';
|
|
|
|
import ConfirmButton from '../Common/ConfirmButton';
|
|
|
|
import DownloadBlock from '../DownloadBlock';
|
|
|
|
import PageTitle from '../Common/PageTitle';
|
|
|
|
import useSettings from '../../hooks/useSettings';
|
|
|
|
import PlayButton, { PlayButtonLink } from '../Common/PlayButton';
|
|
|
|
|
|
|
|
const messages = defineMessages({
|
|
|
|
releasedate: 'Release Date',
|
|
|
|
userrating: 'User Rating',
|
|
|
|
status: 'Status',
|
|
|
|
revenue: 'Revenue',
|
|
|
|
budget: 'Budget',
|
|
|
|
watchtrailer: 'Watch Trailer',
|
|
|
|
originallanguage: 'Original Language',
|
|
|
|
overview: 'Overview',
|
|
|
|
runtime: '{minutes} minutes',
|
|
|
|
cast: 'Cast',
|
|
|
|
recommendations: 'Recommendations',
|
|
|
|
similar: 'Similar Titles',
|
|
|
|
cancelrequest: 'Cancel Request',
|
|
|
|
available: 'Available',
|
|
|
|
unavailable: 'Unavailable',
|
|
|
|
pending: 'Pending',
|
|
|
|
overviewunavailable: 'Overview unavailable.',
|
|
|
|
manageModalTitle: 'Manage Movie',
|
|
|
|
manageModalRequests: 'Requests',
|
|
|
|
manageModalNoRequests: 'No Requests',
|
|
|
|
manageModalClearMedia: 'Clear All Media Data',
|
|
|
|
manageModalClearMediaWarning:
|
|
|
|
'This will irreversibly remove all data for this movie, including any requests.\
|
|
|
|
If this item exists in your Plex library, the media information will be recreated during the next scan.',
|
|
|
|
approve: 'Approve',
|
|
|
|
decline: 'Decline',
|
|
|
|
studio: 'Studio',
|
|
|
|
viewfullcrew: 'View Full Crew',
|
|
|
|
view: 'View',
|
|
|
|
areyousure: 'Are you sure?',
|
|
|
|
openradarr: 'Open Movie in Radarr',
|
|
|
|
openradarr4k: 'Open Movie in 4K Radarr',
|
|
|
|
downloadstatus: 'Download Status',
|
|
|
|
playonplex: 'Play on Plex',
|
|
|
|
play4konplex: 'Play 4K on Plex',
|
|
|
|
markavailable: 'Mark as Available',
|
|
|
|
mark4kavailable: 'Mark 4K as Available',
|
|
|
|
});
|
|
|
|
|
|
|
|
interface MovieDetailsProps {
|
|
|
|
movie?: MovieDetailsType;
|
|
|
|
}
|
|
|
|
|
|
|
|
const MovieDetails: React.FC<MovieDetailsProps> = ({ movie }) => {
|
|
|
|
const settings = useSettings();
|
|
|
|
const { user, hasPermission } = useUser();
|
|
|
|
const router = useRouter();
|
|
|
|
const intl = useIntl();
|
|
|
|
const { locale } = useContext(LanguageContext);
|
|
|
|
const [showManager, setShowManager] = useState(false);
|
|
|
|
|
|
|
|
const { data, error, revalidate } = useSWR<MovieDetailsType>(
|
|
|
|
`/api/v1/movie/${router.query.movieId}?language=${locale}`,
|
|
|
|
{
|
|
|
|
initialData: movie,
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
|
|
|
const { data: ratingData } = useSWR<RTRating>(
|
|
|
|
`/api/v1/movie/${router.query.movieId}/ratings`
|
|
|
|
);
|
|
|
|
|
|
|
|
const sortedCrew = useMemo(() => sortCrewPriority(data?.credits.crew ?? []), [
|
|
|
|
data,
|
|
|
|
]);
|
|
|
|
|
|
|
|
if (!data && !error) {
|
|
|
|
return <LoadingSpinner />;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!data) {
|
|
|
|
return <Error statusCode={404} />;
|
|
|
|
}
|
|
|
|
|
|
|
|
const mediaLinks: PlayButtonLink[] = [];
|
|
|
|
|
|
|
|
if (data.mediaInfo?.plexUrl) {
|
|
|
|
mediaLinks.push({
|
|
|
|
text: intl.formatMessage(messages.playonplex),
|
|
|
|
url: data.mediaInfo?.plexUrl,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
if (
|
|
|
|
data.mediaInfo?.plexUrl4k &&
|
|
|
|
hasPermission([Permission.REQUEST_4K, Permission.REQUEST_4K_MOVIE], {
|
|
|
|
type: 'or',
|
|
|
|
})
|
|
|
|
) {
|
|
|
|
mediaLinks.push({
|
|
|
|
text: intl.formatMessage(messages.play4konplex),
|
|
|
|
url: data.mediaInfo?.plexUrl4k,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
const trailerUrl = data.relatedVideos
|
|
|
|
?.filter((r) => r.type === 'Trailer')
|
|
|
|
.sort((a, b) => a.size - b.size)
|
|
|
|
.pop()?.url;
|
|
|
|
|
|
|
|
if (trailerUrl) {
|
|
|
|
mediaLinks.push({
|
|
|
|
text: intl.formatMessage(messages.watchtrailer),
|
|
|
|
url: trailerUrl,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
const deleteMedia = async () => {
|
|
|
|
if (data?.mediaInfo?.id) {
|
|
|
|
await axios.delete(`/api/v1/media/${data?.mediaInfo?.id}`);
|
|
|
|
revalidate();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const markAvailable = async (is4k = false) => {
|
|
|
|
await axios.post(`/api/v1/media/${data?.mediaInfo?.id}/available`, {
|
|
|
|
is4k,
|
|
|
|
});
|
|
|
|
revalidate();
|
|
|
|
};
|
|
|
|
|
|
|
|
const region = user?.settings?.region
|
|
|
|
? user.settings.region
|
|
|
|
: settings.currentSettings.region
|
|
|
|
? settings.currentSettings.region
|
|
|
|
: 'US';
|
|
|
|
const movieAttributes: React.ReactNode[] = [];
|
|
|
|
|
|
|
|
if (
|
|
|
|
data.releases.results.length &&
|
|
|
|
(data.releases.results.find((r) => r.iso_3166_1 === region)
|
|
|
|
?.release_dates[0].certification ||
|
|
|
|
data.releases.results[0].release_dates[0].certification)
|
|
|
|
) {
|
|
|
|
movieAttributes.push(
|
|
|
|
<span className="p-0.5 py-0 border rounded-md">
|
|
|
|
{data.releases.results.find((r) => r.iso_3166_1 === region)
|
|
|
|
?.release_dates[0].certification ||
|
|
|
|
data.releases.results[0].release_dates[0].certification}
|
|
|
|
</span>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (data.runtime) {
|
|
|
|
movieAttributes.push(
|
|
|
|
intl.formatMessage(messages.runtime, { minutes: data.runtime })
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (data.genres.length) {
|
|
|
|
movieAttributes.push(
|
|
|
|
data.genres
|
|
|
|
.map((g) => (
|
|
|
|
<Link href={`/discover/movies/genre/${g.id}`} key={`genre-${g.id}`}>
|
|
|
|
<a className="hover:underline">{g.name}</a>
|
|
|
|
</Link>
|
|
|
|
))
|
|
|
|
.reduce((prev, curr) => (
|
|
|
|
<>
|
|
|
|
{prev}, {curr}
|
|
|
|
</>
|
|
|
|
))
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
return (
|
|
|
|
<div
|
|
|
|
className="px-4 pt-16 -mx-4 -mt-16 bg-center bg-cover"
|
|
|
|
style={{
|
|
|
|
height: 493,
|
|
|
|
backgroundImage: `linear-gradient(180deg, rgba(17, 24, 39, 0.47) 0%, rgba(17, 24, 39, 1) 100%), url(//image.tmdb.org/t/p/w1920_and_h800_multi_faces/${data.backdropPath})`,
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
<PageTitle title={data.title} />
|
|
|
|
<SlideOver
|
|
|
|
show={showManager}
|
|
|
|
title={intl.formatMessage(messages.manageModalTitle)}
|
|
|
|
onClose={() => setShowManager(false)}
|
|
|
|
subText={data.title}
|
|
|
|
>
|
|
|
|
{((data?.mediaInfo?.downloadStatus ?? []).length > 0 ||
|
|
|
|
(data?.mediaInfo?.downloadStatus4k ?? []).length > 0) && (
|
|
|
|
<>
|
|
|
|
<h3 className="mb-2 text-xl">
|
|
|
|
{intl.formatMessage(messages.downloadstatus)}
|
|
|
|
</h3>
|
|
|
|
<div className="mb-6 overflow-hidden bg-gray-600 rounded-md shadow">
|
|
|
|
<ul>
|
|
|
|
{data.mediaInfo?.downloadStatus?.map((status, index) => (
|
|
|
|
<li
|
|
|
|
key={`dl-status-${status.externalId}-${index}`}
|
|
|
|
className="border-b border-gray-700 last:border-b-0"
|
|
|
|
>
|
|
|
|
<DownloadBlock downloadItem={status} />
|
|
|
|
</li>
|
|
|
|
))}
|
|
|
|
{data.mediaInfo?.downloadStatus4k?.map((status, index) => (
|
|
|
|
<li
|
|
|
|
key={`dl-status-${status.externalId}-${index}`}
|
|
|
|
className="border-b border-gray-700 last:border-b-0"
|
|
|
|
>
|
|
|
|
<DownloadBlock downloadItem={status} is4k />
|
|
|
|
</li>
|
|
|
|
))}
|
|
|
|
</ul>
|
|
|
|
</div>
|
|
|
|
</>
|
|
|
|
)}
|
|
|
|
{data?.mediaInfo &&
|
|
|
|
(data.mediaInfo.status !== MediaStatus.AVAILABLE ||
|
|
|
|
(data.mediaInfo.status4k !== MediaStatus.AVAILABLE &&
|
|
|
|
settings.currentSettings.movie4kEnabled)) && (
|
|
|
|
<div className="mb-6">
|
|
|
|
{data?.mediaInfo &&
|
|
|
|
data?.mediaInfo.status !== MediaStatus.AVAILABLE && (
|
|
|
|
<div className="flex flex-col mb-2 sm:flex-row flex-nowrap">
|
|
|
|
<Button
|
|
|
|
onClick={() => markAvailable()}
|
|
|
|
className="w-full sm:mb-0"
|
|
|
|
buttonType="success"
|
|
|
|
>
|
|
|
|
<svg
|
|
|
|
className="w-5 h-5 mr-1"
|
|
|
|
fill="currentColor"
|
|
|
|
viewBox="0 0 20 20"
|
|
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
|
|
>
|
|
|
|
<path
|
|
|
|
fillRule="evenodd"
|
|
|
|
d="M10 18a8 8 0 100-16 8 8 0 000 16zm1-11a1 1 0 10-2 0v2H7a1 1 0 100 2h2v2a1 1 0 102 0v-2h2a1 1 0 100-2h-2V7z"
|
|
|
|
clipRule="evenodd"
|
|
|
|
/>
|
|
|
|
</svg>
|
|
|
|
<span>{intl.formatMessage(messages.markavailable)}</span>
|
|
|
|
</Button>
|
|
|
|
</div>
|
|
|
|
)}
|
|
|
|
{data?.mediaInfo &&
|
|
|
|
data?.mediaInfo.status4k !== MediaStatus.AVAILABLE &&
|
|
|
|
settings.currentSettings.movie4kEnabled && (
|
|
|
|
<div className="flex flex-col mb-2 sm:flex-row flex-nowrap">
|
|
|
|
<Button
|
|
|
|
onClick={() => markAvailable(true)}
|
|
|
|
className="w-full sm:mb-0"
|
|
|
|
buttonType="success"
|
|
|
|
>
|
|
|
|
<svg
|
|
|
|
className="w-5 h-5 mr-1"
|
|
|
|
fill="currentColor"
|
|
|
|
viewBox="0 0 20 20"
|
|
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
|
|
>
|
|
|
|
<path
|
|
|
|
fillRule="evenodd"
|
|
|
|
d="M10 18a8 8 0 100-16 8 8 0 000 16zm1-11a1 1 0 10-2 0v2H7a1 1 0 100 2h2v2a1 1 0 102 0v-2h2a1 1 0 100-2h-2V7z"
|
|
|
|
clipRule="evenodd"
|
|
|
|
/>
|
|
|
|
</svg>
|
|
|
|
<span>
|
|
|
|
{intl.formatMessage(messages.mark4kavailable)}
|
|
|
|
</span>
|
|
|
|
</Button>
|
|
|
|
</div>
|
|
|
|
)}
|
|
|
|
</div>
|
|
|
|
)}
|
|
|
|
<h3 className="mb-2 text-xl">
|
|
|
|
{intl.formatMessage(messages.manageModalRequests)}
|
|
|
|
</h3>
|
|
|
|
<div className="overflow-hidden bg-gray-600 rounded-md shadow">
|
|
|
|
<ul>
|
|
|
|
{data.mediaInfo?.requests?.map((request) => (
|
|
|
|
<li
|
|
|
|
key={`manage-request-${request.id}`}
|
|
|
|
className="border-b border-gray-700 last:border-b-0"
|
|
|
|
>
|
|
|
|
<RequestBlock request={request} onUpdate={() => revalidate()} />
|
|
|
|
</li>
|
|
|
|
))}
|
|
|
|
{(data.mediaInfo?.requests ?? []).length === 0 && (
|
|
|
|
<li className="py-4 text-center text-gray-400">
|
|
|
|
{intl.formatMessage(messages.manageModalNoRequests)}
|
|
|
|
</li>
|
|
|
|
)}
|
|
|
|
</ul>
|
|
|
|
</div>
|
|
|
|
{(data?.mediaInfo?.serviceUrl || data?.mediaInfo?.serviceUrl4k) && (
|
|
|
|
<div className="mt-8">
|
|
|
|
{data?.mediaInfo?.serviceUrl && (
|
|
|
|
<a
|
|
|
|
href={data?.mediaInfo?.serviceUrl}
|
|
|
|
target="_blank"
|
|
|
|
rel="noreferrer"
|
|
|
|
className="block mb-2 last:mb-0"
|
|
|
|
>
|
|
|
|
<Button buttonType="ghost" className="w-full">
|
|
|
|
<svg
|
|
|
|
className="w-5 h-5 mr-1"
|
|
|
|
fill="currentColor"
|
|
|
|
viewBox="0 0 20 20"
|
|
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
|
|
>
|
|
|
|
<path d="M11 3a1 1 0 100 2h2.586l-6.293 6.293a1 1 0 101.414 1.414L15 6.414V9a1 1 0 102 0V4a1 1 0 00-1-1h-5z" />
|
|
|
|
<path d="M5 5a2 2 0 00-2 2v8a2 2 0 002 2h8a2 2 0 002-2v-3a1 1 0 10-2 0v3H5V7h3a1 1 0 000-2H5z" />
|
|
|
|
</svg>
|
|
|
|
<span>{intl.formatMessage(messages.openradarr)}</span>
|
|
|
|
</Button>
|
|
|
|
</a>
|
|
|
|
)}
|
|
|
|
{data?.mediaInfo?.serviceUrl4k && (
|
|
|
|
<a
|
|
|
|
href={data?.mediaInfo?.serviceUrl4k}
|
|
|
|
target="_blank"
|
|
|
|
rel="noreferrer"
|
|
|
|
>
|
|
|
|
<Button buttonType="ghost" className="w-full">
|
|
|
|
<svg
|
|
|
|
className="w-5 h-5 mr-1"
|
|
|
|
fill="currentColor"
|
|
|
|
viewBox="0 0 20 20"
|
|
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
|
|
>
|
|
|
|
<path d="M11 3a1 1 0 100 2h2.586l-6.293 6.293a1 1 0 101.414 1.414L15 6.414V9a1 1 0 102 0V4a1 1 0 00-1-1h-5z" />
|
|
|
|
<path d="M5 5a2 2 0 00-2 2v8a2 2 0 002 2h8a2 2 0 002-2v-3a1 1 0 10-2 0v3H5V7h3a1 1 0 000-2H5z" />
|
|
|
|
</svg>
|
|
|
|
<span>{intl.formatMessage(messages.openradarr4k)}</span>
|
|
|
|
</Button>
|
|
|
|
</a>
|
|
|
|
)}
|
|
|
|
</div>
|
|
|
|
)}
|
|
|
|
{data?.mediaInfo && (
|
|
|
|
<div className="mt-8">
|
|
|
|
<ConfirmButton
|
|
|
|
onClick={() => deleteMedia()}
|
|
|
|
confirmText={intl.formatMessage(messages.areyousure)}
|
|
|
|
className="w-full"
|
|
|
|
>
|
|
|
|
{intl.formatMessage(messages.manageModalClearMedia)}
|
|
|
|
</ConfirmButton>
|
|
|
|
<div className="mt-2 text-sm text-gray-400">
|
|
|
|
{intl.formatMessage(messages.manageModalClearMediaWarning)}
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
)}
|
|
|
|
</SlideOver>
|
|
|
|
<div className="flex flex-col items-center pt-4 lg:flex-row lg:items-end">
|
|
|
|
<div className="lg:mr-4">
|
|
|
|
<img
|
|
|
|
src={
|
|
|
|
data.posterPath
|
|
|
|
? `//image.tmdb.org/t/p/w600_and_h900_bestv2${data.posterPath}`
|
|
|
|
: '/images/overseerr_poster_not_found.png'
|
|
|
|
}
|
|
|
|
alt=""
|
|
|
|
className="w-32 rounded shadow md:rounded-lg md:shadow-2xl md:w-44 lg:w-52"
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
<div className="flex flex-col flex-1 mt-4 text-center text-white lg:mr-4 lg:mt-0 lg:text-left">
|
|
|
|
<div className="mb-2 space-x-2">
|
|
|
|
<span className="ml-2 lg:ml-0">
|
|
|
|
<StatusBadge
|
|
|
|
status={data.mediaInfo?.status}
|
|
|
|
inProgress={(data.mediaInfo?.downloadStatus ?? []).length > 0}
|
|
|
|
plexUrl={data.mediaInfo?.plexUrl}
|
|
|
|
/>
|
|
|
|
</span>
|
|
|
|
{settings.currentSettings.movie4kEnabled &&
|
|
|
|
hasPermission(
|
|
|
|
[Permission.REQUEST_4K, Permission.REQUEST_4K_MOVIE],
|
|
|
|
{
|
|
|
|
type: 'or',
|
|
|
|
}
|
|
|
|
) && (
|
|
|
|
<span>
|
|
|
|
<StatusBadge
|
|
|
|
status={data.mediaInfo?.status4k}
|
|
|
|
is4k
|
|
|
|
inProgress={
|
|
|
|
(data.mediaInfo?.downloadStatus4k ?? []).length > 0
|
|
|
|
}
|
|
|
|
plexUrl4k={data.mediaInfo?.plexUrl4k}
|
|
|
|
/>
|
|
|
|
</span>
|
|
|
|
)}
|
|
|
|
</div>
|
|
|
|
<h1 className="text-2xl lg:text-4xl">
|
|
|
|
{data.title}{' '}
|
|
|
|
{data.releaseDate && (
|
|
|
|
<span className="text-2xl">({data.releaseDate.slice(0, 4)})</span>
|
|
|
|
)}
|
|
|
|
</h1>
|
|
|
|
<span className="mt-1 text-xs lg:text-base lg:mt-0">
|
|
|
|
{movieAttributes.length > 0 &&
|
|
|
|
movieAttributes
|
|
|
|
.map((t, k) => <span key={k}>{t}</span>)
|
|
|
|
.reduce((prev, curr) => (
|
|
|
|
<>
|
|
|
|
{prev} | {curr}
|
|
|
|
</>
|
|
|
|
))}
|
|
|
|
</span>
|
|
|
|
</div>
|
|
|
|
<div className="relative z-10 flex flex-wrap justify-center flex-shrink-0 mt-4 sm:justify-end sm:flex-nowrap lg:mt-0">
|
|
|
|
<div className="mb-3 sm:mb-0">
|
|
|
|
<PlayButton links={mediaLinks} />
|
|
|
|
</div>
|
|
|
|
<div className="mb-3 sm:mb-0">
|
|
|
|
<RequestButton
|
|
|
|
mediaType="movie"
|
|
|
|
media={data.mediaInfo}
|
|
|
|
tmdbId={data.id}
|
|
|
|
onUpdate={() => revalidate()}
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
{hasPermission(Permission.MANAGE_REQUESTS) && (
|
|
|
|
<Button
|
|
|
|
buttonType="default"
|
|
|
|
className="mb-3 ml-2 first:ml-0 sm:mb-0"
|
|
|
|
onClick={() => setShowManager(true)}
|
|
|
|
>
|
|
|
|
<svg
|
|
|
|
className="w-5"
|
|
|
|
style={{ height: 20 }}
|
|
|
|
fill="none"
|
|
|
|
stroke="currentColor"
|
|
|
|
viewBox="0 0 24 24"
|
|
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
|
|
>
|
|
|
|
<path
|
|
|
|
strokeLinecap="round"
|
|
|
|
strokeLinejoin="round"
|
|
|
|
strokeWidth={2}
|
|
|
|
d="M10.325 4.317c.426-1.756 2.924-1.756 3.35 0a1.724 1.724 0 002.573 1.066c1.543-.94 3.31.826 2.37 2.37a1.724 1.724 0 001.065 2.572c1.756.426 1.756 2.924 0 3.35a1.724 1.724 0 00-1.066 2.573c.94 1.543-.826 3.31-2.37 2.37a1.724 1.724 0 00-2.572 1.065c-.426 1.756-2.924 1.756-3.35 0a1.724 1.724 0 00-2.573-1.066c-1.543.94-3.31-.826-2.37-2.37a1.724 1.724 0 00-1.065-2.572c-1.756-.426-1.756-2.924 0-3.35a1.724 1.724 0 001.066-2.573c-.94-1.543.826-3.31 2.37-2.37.996.608 2.296.07 2.572-1.065z"
|
|
|
|
/>
|
|
|
|
<path
|
|
|
|
strokeLinecap="round"
|
|
|
|
strokeLinejoin="round"
|
|
|
|
strokeWidth={2}
|
|
|
|
d="M15 12a3 3 0 11-6 0 3 3 0 016 0z"
|
|
|
|
/>
|
|
|
|
</svg>
|
|
|
|
</Button>
|
|
|
|
)}
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
<div className="flex flex-col pt-8 pb-4 text-white md:flex-row">
|
|
|
|
<div className="flex-1 md:mr-8">
|
|
|
|
<h2 className="text-xl md:text-2xl">
|
|
|
|
{intl.formatMessage(messages.overview)}
|
|
|
|
</h2>
|
|
|
|
<p className="pt-2 text-sm md:text-base">
|
|
|
|
{data.overview
|
|
|
|
? data.overview
|
|
|
|
: intl.formatMessage(messages.overviewunavailable)}
|
|
|
|
</p>
|
|
|
|
<ul className="grid grid-cols-2 gap-6 mt-6 sm:grid-cols-3">
|
|
|
|
{sortedCrew.slice(0, 6).map((person) => (
|
|
|
|
<li
|
|
|
|
className="flex flex-col col-span-1"
|
|
|
|
key={`crew-${person.job}-${person.id}`}
|
|
|
|
>
|
|
|
|
<span className="font-bold">{person.job}</span>
|
|
|
|
<Link href={`/person/${person.id}`}>
|
|
|
|
<a className="text-gray-400 transition duration-300 hover:text-underline hover:text-gray-100">
|
|
|
|
{person.name}
|
|
|
|
</a>
|
|
|
|
</Link>
|
|
|
|
</li>
|
|
|
|
))}
|
|
|
|
</ul>
|
|
|
|
{sortedCrew.length > 0 && (
|
|
|
|
<div className="flex justify-end mt-4">
|
|
|
|
<Link href={`/movie/${data.id}/crew`}>
|
|
|
|
<a className="flex items-center text-gray-400 transition duration-300 hover:text-gray-100">
|
|
|
|
<span>{intl.formatMessage(messages.viewfullcrew)}</span>
|
|
|
|
<svg
|
|
|
|
className="inline-block w-5 h-5 ml-1"
|
|
|
|
fill="none"
|
|
|
|
stroke="currentColor"
|
|
|
|
viewBox="0 0 24 24"
|
|
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
|
|
>
|
|
|
|
<path
|
|
|
|
strokeLinecap="round"
|
|
|
|
strokeLinejoin="round"
|
|
|
|
strokeWidth={2}
|
|
|
|
d="M13 9l3 3m0 0l-3 3m3-3H8m13 0a9 9 0 11-18 0 9 9 0 0118 0z"
|
|
|
|
/>
|
|
|
|
</svg>
|
|
|
|
</a>
|
|
|
|
</Link>
|
|
|
|
</div>
|
|
|
|
)}
|
|
|
|
</div>
|
|
|
|
<div className="w-full mt-8 md:w-80 md:mt-0">
|
|
|
|
{data.collection && (
|
|
|
|
<div className="mb-6">
|
|
|
|
<Link href={`/collection/${data.collection.id}`}>
|
|
|
|
<a>
|
|
|
|
<div
|
|
|
|
className="relative z-0 transition duration-300 scale-100 bg-gray-800 bg-center bg-cover rounded-lg shadow-md cursor-pointer transform-gpu group hover:scale-105"
|
|
|
|
style={{
|
|
|
|
backgroundImage: `linear-gradient(180deg, rgba(31, 41, 55, 0.47) 0%, rgba(31, 41, 55, 0.80) 100%), url(//image.tmdb.org/t/p/w1440_and_h320_multi_faces/${data.collection.backdropPath})`,
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
<div className="flex items-center justify-between p-4 text-gray-200 transition duration-300 h-14 group-hover:text-white">
|
|
|
|
<div>{data.collection.name}</div>
|
|
|
|
<Button buttonSize="sm">
|
|
|
|
{intl.formatMessage(messages.view)}
|
|
|
|
</Button>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</a>
|
|
|
|
</Link>
|
|
|
|
</div>
|
|
|
|
)}
|
|
|
|
<div className="bg-gray-900 border border-gray-800 rounded-lg shadow">
|
|
|
|
{(!!data.voteCount ||
|
|
|
|
(ratingData?.criticsRating && !!ratingData?.criticsScore) ||
|
|
|
|
(ratingData?.audienceRating && !!ratingData?.audienceScore)) && (
|
|
|
|
<div className="flex items-center justify-center px-4 py-2 border-b border-gray-800 last:border-b-0">
|
|
|
|
{ratingData?.criticsRating && !!ratingData?.criticsScore && (
|
|
|
|
<>
|
|
|
|
<span className="text-sm">
|
|
|
|
{ratingData.criticsRating === 'Rotten' ? (
|
|
|
|
<RTRotten className="w-6 mr-1" />
|
|
|
|
) : (
|
|
|
|
<RTFresh className="w-6 mr-1" />
|
|
|
|
)}
|
|
|
|
</span>
|
|
|
|
<span className="mr-4 text-sm text-gray-400 last:mr-0">
|
|
|
|
{ratingData.criticsScore}%
|
|
|
|
</span>
|
|
|
|
</>
|
|
|
|
)}
|
|
|
|
{ratingData?.audienceRating && !!ratingData?.audienceScore && (
|
|
|
|
<>
|
|
|
|
<span className="text-sm">
|
|
|
|
{ratingData.audienceRating === 'Spilled' ? (
|
|
|
|
<RTAudRotten className="w-6 mr-1" />
|
|
|
|
) : (
|
|
|
|
<RTAudFresh className="w-6 mr-1" />
|
|
|
|
)}
|
|
|
|
</span>
|
|
|
|
<span className="mr-4 text-sm text-gray-400 last:mr-0">
|
|
|
|
{ratingData.audienceScore}%
|
|
|
|
</span>
|
|
|
|
</>
|
|
|
|
)}
|
|
|
|
{!!data.voteCount && (
|
|
|
|
<>
|
|
|
|
<span className="text-sm">
|
|
|
|
<TmdbLogo className="w-6 mr-2" />
|
|
|
|
</span>
|
|
|
|
<span className="text-sm text-gray-400">
|
|
|
|
{data.voteAverage}/10
|
|
|
|
</span>
|
|
|
|
</>
|
|
|
|
)}
|
|
|
|
</div>
|
|
|
|
)}
|
|
|
|
{data.releaseDate && (
|
|
|
|
<div className="flex px-4 py-2 border-b border-gray-800 last:border-b-0">
|
|
|
|
<span className="text-sm">
|
|
|
|
{intl.formatMessage(messages.releasedate)}
|
|
|
|
</span>
|
|
|
|
<span className="flex-1 text-sm text-right text-gray-400">
|
|
|
|
<FormattedDate
|
|
|
|
value={new Date(data.releaseDate)}
|
|
|
|
year="numeric"
|
|
|
|
month="long"
|
|
|
|
day="numeric"
|
|
|
|
/>
|
|
|
|
</span>
|
|
|
|
</div>
|
|
|
|
)}
|
|
|
|
<div className="flex px-4 py-2 border-b border-gray-800 last:border-b-0">
|
|
|
|
<span className="text-sm">
|
|
|
|
{intl.formatMessage(messages.status)}
|
|
|
|
</span>
|
|
|
|
<span className="flex-1 text-sm text-right text-gray-400">
|
|
|
|
{data.status}
|
|
|
|
</span>
|
|
|
|
</div>
|
|
|
|
{data.revenue > 0 && (
|
|
|
|
<div className="flex px-4 py-2 border-b border-gray-800 last:border-b-0">
|
|
|
|
<span className="text-sm">
|
|
|
|
{intl.formatMessage(messages.revenue)}
|
|
|
|
</span>
|
|
|
|
<span className="flex-1 text-sm text-right text-gray-400">
|
|
|
|
<FormattedNumber
|
|
|
|
currency="USD"
|
|
|
|
style="currency"
|
|
|
|
value={data.revenue}
|
|
|
|
/>
|
|
|
|
</span>
|
|
|
|
</div>
|
|
|
|
)}
|
|
|
|
{data.budget > 0 && (
|
|
|
|
<div className="flex px-4 py-2 border-b border-gray-800 last:border-b-0">
|
|
|
|
<span className="text-sm">
|
|
|
|
{intl.formatMessage(messages.budget)}
|
|
|
|
</span>
|
|
|
|
<span className="flex-1 text-sm text-right text-gray-400">
|
|
|
|
<FormattedNumber
|
|
|
|
currency="USD"
|
|
|
|
style="currency"
|
|
|
|
value={data.budget}
|
|
|
|
/>
|
|
|
|
</span>
|
|
|
|
</div>
|
|
|
|
)}
|
|
|
|
{data.spokenLanguages.some(
|
|
|
|
(lng) => lng.iso_639_1 === data.originalLanguage
|
|
|
|
) && (
|
|
|
|
<div className="flex px-4 py-2 border-b border-gray-800 last:border-b-0">
|
|
|
|
<span className="text-sm">
|
|
|
|
{intl.formatMessage(messages.originallanguage)}
|
|
|
|
</span>
|
|
|
|
<span className="flex-1 text-sm text-right text-gray-400">
|
|
|
|
{
|
|
|
|
data.spokenLanguages.find(
|
|
|
|
(lng) => lng.iso_639_1 === data.originalLanguage
|
|
|
|
)?.name
|
|
|
|
}
|
|
|
|
</span>
|
|
|
|
</div>
|
|
|
|
)}
|
|
|
|
{data.productionCompanies[0] && (
|
|
|
|
<div className="flex px-4 py-2 border-b border-gray-800 last:border-b-0">
|
|
|
|
<span className="text-sm">
|
|
|
|
{intl.formatMessage(messages.studio)}
|
|
|
|
</span>
|
|
|
|
<span className="flex-1 text-sm text-right text-gray-400">
|
|
|
|
<Link
|
|
|
|
href={`/discover/movies/studio/${data.productionCompanies[0].id}`}
|
|
|
|
>
|
|
|
|
<a className="hover:underline">
|
|
|
|
{data.productionCompanies[0].name}
|
|
|
|
</a>
|
|
|
|
</Link>
|
|
|
|
</span>
|
|
|
|
</div>
|
|
|
|
)}
|
|
|
|
</div>
|
|
|
|
<div className="mt-4">
|
|
|
|
<ExternalLinkBlock
|
|
|
|
mediaType="movie"
|
|
|
|
tmdbId={data.id}
|
|
|
|
tvdbId={data.externalIds.tvdbId}
|
|
|
|
imdbId={data.externalIds.imdbId}
|
|
|
|
rtUrl={ratingData?.url}
|
|
|
|
plexUrl={data.mediaInfo?.plexUrl ?? data.mediaInfo?.plexUrl4k}
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
{data.credits.cast.length > 0 && (
|
|
|
|
<>
|
|
|
|
<div className="mt-6 mb-4 md:flex md:items-center md:justify-between">
|
|
|
|
<div className="flex-1 min-w-0">
|
|
|
|
<Link href="/movie/[movieId]/cast" as={`/movie/${data.id}/cast`}>
|
|
|
|
<a className="inline-flex items-center text-xl leading-7 text-gray-300 hover:text-white sm:text-2xl sm:leading-9 sm:truncate">
|
|
|
|
<span>{intl.formatMessage(messages.cast)}</span>
|
|
|
|
<svg
|
|
|
|
className="w-6 h-6 ml-2"
|
|
|
|
fill="none"
|
|
|
|
stroke="currentColor"
|
|
|
|
viewBox="0 0 24 24"
|
|
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
|
|
>
|
|
|
|
<path
|
|
|
|
strokeLinecap="round"
|
|
|
|
strokeLinejoin="round"
|
|
|
|
strokeWidth={2}
|
|
|
|
d="M13 9l3 3m0 0l-3 3m3-3H8m13 0a9 9 0 11-18 0 9 9 0 0118 0z"
|
|
|
|
/>
|
|
|
|
</svg>
|
|
|
|
</a>
|
|
|
|
</Link>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
<Slider
|
|
|
|
sliderKey="cast"
|
|
|
|
isLoading={false}
|
|
|
|
isEmpty={false}
|
|
|
|
items={data.credits.cast.slice(0, 20).map((person) => (
|
|
|
|
<PersonCard
|
|
|
|
key={`cast-item-${person.id}`}
|
|
|
|
personId={person.id}
|
|
|
|
name={person.name}
|
|
|
|
subName={person.character}
|
|
|
|
profilePath={person.profilePath}
|
|
|
|
/>
|
|
|
|
))}
|
|
|
|
/>
|
|
|
|
</>
|
|
|
|
)}
|
|
|
|
<MediaSlider
|
|
|
|
sliderKey="recommendations"
|
|
|
|
title={intl.formatMessage(messages.recommendations)}
|
|
|
|
url={`/api/v1/movie/${router.query.movieId}/recommendations`}
|
|
|
|
linkUrl={`/movie/${data.id}/recommendations`}
|
|
|
|
hideWhenEmpty
|
|
|
|
/>
|
|
|
|
<MediaSlider
|
|
|
|
sliderKey="similar"
|
|
|
|
title={intl.formatMessage(messages.similar)}
|
|
|
|
url={`/api/v1/movie/${router.query.movieId}/similar`}
|
|
|
|
linkUrl={`/movie/${data.id}/similar`}
|
|
|
|
hideWhenEmpty
|
|
|
|
/>
|
|
|
|
<div className="pb-8" />
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
export default MovieDetails;
|