Signed-off-by: 0xsysr3ll <0xsysr3ll@pm.me> Co-authored-by: fallenbagel <98979876+Fallenbagel@users.noreply.github.com> Co-authored-by: 0xsysr3ll <0xsysr3ll@pm.me> Co-authored-by: gauthier-th <mail@gauthierth.fr>
1151 lines
40 KiB
TypeScript
1151 lines
40 KiB
TypeScript
import RTAudFresh from '@app/assets/rt_aud_fresh.svg';
|
||
import RTAudRotten from '@app/assets/rt_aud_rotten.svg';
|
||
import RTFresh from '@app/assets/rt_fresh.svg';
|
||
import RTRotten from '@app/assets/rt_rotten.svg';
|
||
import ImdbLogo from '@app/assets/services/imdb.svg';
|
||
import Spinner from '@app/assets/spinner.svg';
|
||
import TmdbLogo from '@app/assets/tmdb_logo.svg';
|
||
import BlocklistModal from '@app/components/BlocklistModal';
|
||
import Button from '@app/components/Common/Button';
|
||
import CachedImage from '@app/components/Common/CachedImage';
|
||
import LoadingSpinner from '@app/components/Common/LoadingSpinner';
|
||
import PageTitle from '@app/components/Common/PageTitle';
|
||
import type { PlayButtonLink } from '@app/components/Common/PlayButton';
|
||
import PlayButton from '@app/components/Common/PlayButton';
|
||
import Tag from '@app/components/Common/Tag';
|
||
import Tooltip from '@app/components/Common/Tooltip';
|
||
import ExternalLinkBlock from '@app/components/ExternalLinkBlock';
|
||
import IssueModal from '@app/components/IssueModal';
|
||
import ManageSlideOver from '@app/components/ManageSlideOver';
|
||
import MediaSlider from '@app/components/MediaSlider';
|
||
import PersonCard from '@app/components/PersonCard';
|
||
import RequestButton from '@app/components/RequestButton';
|
||
import Slider from '@app/components/Slider';
|
||
import StatusBadge from '@app/components/StatusBadge';
|
||
import useDeepLinks from '@app/hooks/useDeepLinks';
|
||
import useLocale from '@app/hooks/useLocale';
|
||
import useSettings from '@app/hooks/useSettings';
|
||
import { Permission, UserType, useUser } from '@app/hooks/useUser';
|
||
import globalMessages from '@app/i18n/globalMessages';
|
||
import ErrorPage from '@app/pages/_error';
|
||
import { sortCrewPriority } from '@app/utils/creditHelpers';
|
||
import defineMessages from '@app/utils/defineMessages';
|
||
import { refreshIntervalHelper } from '@app/utils/refreshIntervalHelper';
|
||
import {
|
||
ArrowRightCircleIcon,
|
||
CloudIcon,
|
||
CogIcon,
|
||
ExclamationTriangleIcon,
|
||
EyeSlashIcon,
|
||
FilmIcon,
|
||
MinusCircleIcon,
|
||
PlayIcon,
|
||
StarIcon,
|
||
TicketIcon,
|
||
} from '@heroicons/react/24/outline';
|
||
import {
|
||
ChevronDoubleDownIcon,
|
||
ChevronDoubleUpIcon,
|
||
} from '@heroicons/react/24/solid';
|
||
import { type RatingResponse } from '@server/api/ratings';
|
||
import { IssueStatus } from '@server/constants/issue';
|
||
import { MediaStatus, MediaType } from '@server/constants/media';
|
||
import { MediaServerType } from '@server/constants/server';
|
||
import type { MovieDetails as MovieDetailsType } from '@server/models/Movie';
|
||
import axios from 'axios';
|
||
import { countries } from 'country-flag-icons';
|
||
import 'country-flag-icons/3x2/flags.css';
|
||
import { uniqBy } from 'lodash';
|
||
import Link from 'next/link';
|
||
import { useRouter } from 'next/router';
|
||
import { useCallback, useEffect, useMemo, useState } from 'react';
|
||
import { useIntl } from 'react-intl';
|
||
import { useToasts } from 'react-toast-notifications';
|
||
import useSWR from 'swr';
|
||
|
||
const messages = defineMessages('components.MovieDetails', {
|
||
originaltitle: 'Original Title',
|
||
releasedate:
|
||
'{releaseCount, plural, one {Release Date} other {Release Dates}}',
|
||
revenue: 'Revenue',
|
||
budget: 'Budget',
|
||
watchtrailer: 'Watch Trailer',
|
||
originallanguage: 'Original Language',
|
||
overview: 'Overview',
|
||
runtime: '{minutes} minutes',
|
||
cast: 'Cast',
|
||
recommendations: 'Recommendations',
|
||
similar: 'Similar Titles',
|
||
overviewunavailable: 'Overview unavailable.',
|
||
studio: '{studioCount, plural, one {Studio} other {Studios}}',
|
||
viewfullcrew: 'View Full Crew',
|
||
openradarr: 'Open Movie in Radarr',
|
||
openradarr4k: 'Open Movie in 4K Radarr',
|
||
downloadstatus: 'Download Status',
|
||
play: 'Play on {mediaServerName}',
|
||
play4k: 'Play 4K on {mediaServerName}',
|
||
markavailable: 'Mark as Available',
|
||
mark4kavailable: 'Mark as Available in 4K',
|
||
showmore: 'Show More',
|
||
showless: 'Show Less',
|
||
streamingproviders: 'Currently Streaming On',
|
||
productioncountries:
|
||
'Production {countryCount, plural, one {Country} other {Countries}}',
|
||
theatricalrelease: 'Theatrical Release',
|
||
digitalrelease: 'Digital Release',
|
||
physicalrelease: 'Physical Release',
|
||
reportissue: 'Report an Issue',
|
||
managemovie: 'Manage Movie',
|
||
rtcriticsscore: 'Rotten Tomatoes Tomatometer',
|
||
rtaudiencescore: 'Rotten Tomatoes Audience Score',
|
||
tmdbuserscore: 'TMDB User Score',
|
||
imdbuserscore: 'IMDB User Score – votes: {formattedCount}',
|
||
watchlistSuccess: '<strong>{title}</strong> added to watchlist successfully!',
|
||
watchlistDeleted:
|
||
'<strong>{title}</strong> Removed from watchlist successfully!',
|
||
watchlistError: 'Something went wrong. Please try again.',
|
||
removefromwatchlist: 'Remove From Watchlist',
|
||
addtowatchlist: 'Add To Watchlist',
|
||
});
|
||
|
||
interface MovieDetailsProps {
|
||
movie?: MovieDetailsType;
|
||
}
|
||
|
||
const MovieDetails = ({ movie }: MovieDetailsProps) => {
|
||
const settings = useSettings();
|
||
const { user, hasPermission } = useUser();
|
||
const router = useRouter();
|
||
const intl = useIntl();
|
||
const { locale } = useLocale();
|
||
const [showManager, setShowManager] = useState(
|
||
router.query.manage == '1' ? true : false
|
||
);
|
||
const minStudios = 3;
|
||
const [showMoreStudios, setShowMoreStudios] = useState(false);
|
||
const [showIssueModal, setShowIssueModal] = useState(false);
|
||
const [isUpdating, setIsUpdating] = useState<boolean>(false);
|
||
const [toggleWatchlist, setToggleWatchlist] = useState<boolean>(
|
||
!movie?.onUserWatchlist
|
||
);
|
||
const [isBlocklistUpdating, setIsBlocklistUpdating] =
|
||
useState<boolean>(false);
|
||
const [showBlocklistModal, setShowBlocklistModal] = useState(false);
|
||
const { addToast } = useToasts();
|
||
|
||
const {
|
||
data,
|
||
error,
|
||
mutate: revalidate,
|
||
} = useSWR<MovieDetailsType>(`/api/v1/movie/${router.query.movieId}`, {
|
||
fallbackData: movie,
|
||
refreshInterval: refreshIntervalHelper(
|
||
{
|
||
downloadStatus: movie?.mediaInfo?.downloadStatus,
|
||
downloadStatus4k: movie?.mediaInfo?.downloadStatus4k,
|
||
},
|
||
15000
|
||
),
|
||
});
|
||
|
||
const { data: ratingData } = useSWR<RatingResponse>(
|
||
`/api/v1/movie/${router.query.movieId}/ratingscombined`
|
||
);
|
||
|
||
const sortedCrew = useMemo(
|
||
() => sortCrewPriority(data?.credits.crew ?? []),
|
||
[data]
|
||
);
|
||
|
||
useEffect(() => {
|
||
setShowManager(router.query.manage == '1' ? true : false);
|
||
}, [router.query.manage]);
|
||
|
||
const closeBlocklistModal = useCallback(
|
||
() => setShowBlocklistModal(false),
|
||
[]
|
||
);
|
||
|
||
const { mediaUrl: plexUrl, mediaUrl4k: plexUrl4k } = useDeepLinks({
|
||
mediaUrl: data?.mediaInfo?.mediaUrl,
|
||
mediaUrl4k: data?.mediaInfo?.mediaUrl4k,
|
||
iOSPlexUrl: data?.mediaInfo?.iOSPlexUrl,
|
||
iOSPlexUrl4k: data?.mediaInfo?.iOSPlexUrl4k,
|
||
});
|
||
|
||
if (!data && !error) {
|
||
return <LoadingSpinner />;
|
||
}
|
||
|
||
if (!data) {
|
||
return <ErrorPage statusCode={404} />;
|
||
}
|
||
|
||
const showAllStudios = data.productionCompanies.length <= minStudios + 1;
|
||
const mediaLinks: PlayButtonLink[] = [];
|
||
|
||
if (
|
||
plexUrl &&
|
||
hasPermission([Permission.REQUEST, Permission.REQUEST_MOVIE], {
|
||
type: 'or',
|
||
})
|
||
) {
|
||
mediaLinks.push({
|
||
text: getAvailableMediaServerName(),
|
||
url: plexUrl,
|
||
svg: <PlayIcon />,
|
||
});
|
||
}
|
||
|
||
if (
|
||
settings.currentSettings.movie4kEnabled &&
|
||
plexUrl4k &&
|
||
hasPermission([Permission.REQUEST_4K, Permission.REQUEST_4K_MOVIE], {
|
||
type: 'or',
|
||
})
|
||
) {
|
||
mediaLinks.push({
|
||
text: getAvailable4kMediaServerName(),
|
||
url: plexUrl4k,
|
||
svg: <PlayIcon />,
|
||
});
|
||
}
|
||
|
||
const trailerVideo = data.relatedVideos
|
||
?.filter((r) => r.type === 'Trailer')
|
||
.sort((a, b) => a.size - b.size)
|
||
.pop();
|
||
const trailerUrl =
|
||
trailerVideo?.site === 'YouTube' &&
|
||
settings.currentSettings.youtubeUrl != ''
|
||
? `${settings.currentSettings.youtubeUrl}${trailerVideo?.key}`
|
||
: trailerVideo?.url;
|
||
|
||
if (trailerUrl) {
|
||
mediaLinks.push({
|
||
text: intl.formatMessage(messages.watchtrailer),
|
||
url: trailerUrl,
|
||
svg: <FilmIcon />,
|
||
});
|
||
}
|
||
|
||
const discoverRegion = user?.settings?.discoverRegion
|
||
? user.settings.discoverRegion
|
||
: settings.currentSettings.discoverRegion
|
||
? settings.currentSettings.discoverRegion
|
||
: 'US';
|
||
|
||
const releases = data.releases.results.find(
|
||
(r) => r.iso_3166_1 === discoverRegion
|
||
)?.release_dates;
|
||
|
||
// Release date types:
|
||
// 1. Premiere
|
||
// 2. Theatrical (limited)
|
||
// 3. Theatrical
|
||
// 4. Digital
|
||
// 5. Physical
|
||
// 6. TV
|
||
const filteredReleases = uniqBy(
|
||
releases?.filter((r) => r.type > 2 && r.type < 6),
|
||
'type'
|
||
);
|
||
|
||
const movieAttributes: React.ReactNode[] = [];
|
||
|
||
const certification = releases?.find((r) => r.certification)?.certification;
|
||
if (certification) {
|
||
movieAttributes.push(
|
||
<span className="rounded-md border p-0.5 py-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}`}
|
||
className="hover:underline"
|
||
>
|
||
{g.name}
|
||
</Link>
|
||
))
|
||
.reduce((prev, curr) => (
|
||
<>
|
||
{intl.formatMessage(globalMessages.delimitedlist, {
|
||
a: prev,
|
||
b: curr,
|
||
})}
|
||
</>
|
||
))
|
||
);
|
||
}
|
||
|
||
const streamingRegion = user?.settings?.streamingRegion
|
||
? user.settings.streamingRegion
|
||
: settings.currentSettings.streamingRegion
|
||
? settings.currentSettings.streamingRegion
|
||
: 'US';
|
||
const streamingProviders =
|
||
data?.watchProviders?.find(
|
||
(provider) => provider.iso_3166_1 === streamingRegion
|
||
)?.flatrate ?? [];
|
||
|
||
function getAvailableMediaServerName() {
|
||
if (settings.currentSettings.mediaServerType === MediaServerType.EMBY) {
|
||
return intl.formatMessage(messages.play, { mediaServerName: 'Emby' });
|
||
}
|
||
|
||
if (settings.currentSettings.mediaServerType === MediaServerType.PLEX) {
|
||
return intl.formatMessage(messages.play, { mediaServerName: 'Plex' });
|
||
}
|
||
|
||
return intl.formatMessage(messages.play, { mediaServerName: 'Jellyfin' });
|
||
}
|
||
|
||
function getAvailable4kMediaServerName() {
|
||
if (settings.currentSettings.mediaServerType === MediaServerType.EMBY) {
|
||
return intl.formatMessage(messages.play, { mediaServerName: 'Emby' });
|
||
}
|
||
|
||
if (settings.currentSettings.mediaServerType === MediaServerType.PLEX) {
|
||
return intl.formatMessage(messages.play4k, { mediaServerName: 'Plex' });
|
||
}
|
||
|
||
return intl.formatMessage(messages.play4k, { mediaServerName: 'Jellyfin' });
|
||
}
|
||
|
||
const onClickWatchlistBtn = async (): Promise<void> => {
|
||
setIsUpdating(true);
|
||
|
||
try {
|
||
const response = await axios.post('/api/v1/watchlist', {
|
||
tmdbId: movie?.id,
|
||
mediaType: MediaType.MOVIE,
|
||
title: movie?.title,
|
||
});
|
||
|
||
if (response.data) {
|
||
addToast(
|
||
<span>
|
||
{intl.formatMessage(messages.watchlistSuccess, {
|
||
title: movie?.title,
|
||
strong: (msg: React.ReactNode) => <strong>{msg}</strong>,
|
||
})}
|
||
</span>,
|
||
{ appearance: 'success', autoDismiss: true }
|
||
);
|
||
}
|
||
} catch (e) {
|
||
addToast(intl.formatMessage(messages.watchlistError), {
|
||
appearance: 'error',
|
||
autoDismiss: true,
|
||
});
|
||
}
|
||
|
||
setIsUpdating(false);
|
||
setToggleWatchlist((prevState) => !prevState);
|
||
};
|
||
|
||
const onClickDeleteWatchlistBtn = async (): Promise<void> => {
|
||
setIsUpdating(true);
|
||
try {
|
||
await axios.delete(`/api/v1/watchlist/${movie?.id}`);
|
||
|
||
addToast(
|
||
<span>
|
||
{intl.formatMessage(messages.watchlistDeleted, {
|
||
title: movie?.title,
|
||
strong: (msg: React.ReactNode) => <strong>{msg}</strong>,
|
||
})}
|
||
</span>,
|
||
{ appearance: 'info', autoDismiss: true }
|
||
);
|
||
} catch (e) {
|
||
addToast(intl.formatMessage(messages.watchlistError), {
|
||
appearance: 'error',
|
||
autoDismiss: true,
|
||
});
|
||
} finally {
|
||
setIsUpdating(false);
|
||
setToggleWatchlist((prevState) => !prevState);
|
||
}
|
||
};
|
||
|
||
const onClickHideItemBtn = async (): Promise<void> => {
|
||
setIsBlocklistUpdating(true);
|
||
|
||
try {
|
||
await axios.post('/api/v1/blocklist', {
|
||
tmdbId: movie?.id,
|
||
mediaType: 'movie',
|
||
title: movie?.title,
|
||
user: user?.id,
|
||
});
|
||
|
||
addToast(
|
||
<span>
|
||
{intl.formatMessage(globalMessages.blocklistSuccess, {
|
||
title: movie?.title,
|
||
strong: (msg: React.ReactNode) => <strong>{msg}</strong>,
|
||
})}
|
||
</span>,
|
||
{ appearance: 'success', autoDismiss: true }
|
||
);
|
||
|
||
revalidate();
|
||
} catch (e) {
|
||
if (e?.response?.status === 412) {
|
||
addToast(
|
||
<span>
|
||
{intl.formatMessage(globalMessages.blocklistDuplicateError, {
|
||
title: movie?.title,
|
||
strong: (msg: React.ReactNode) => <strong>{msg}</strong>,
|
||
})}
|
||
</span>,
|
||
{ appearance: 'info', autoDismiss: true }
|
||
);
|
||
} else {
|
||
addToast(intl.formatMessage(globalMessages.blocklistError), {
|
||
appearance: 'error',
|
||
autoDismiss: true,
|
||
});
|
||
}
|
||
}
|
||
|
||
setIsBlocklistUpdating(false);
|
||
closeBlocklistModal();
|
||
};
|
||
|
||
const showHideButton = hasPermission([Permission.MANAGE_BLOCKLIST], {
|
||
type: 'or',
|
||
});
|
||
|
||
return (
|
||
<div
|
||
className="media-page"
|
||
style={{
|
||
height: 493,
|
||
}}
|
||
>
|
||
{data.backdropPath && (
|
||
<div className="media-page-bg-image">
|
||
<CachedImage
|
||
type="tmdb"
|
||
alt=""
|
||
src={`https://image.tmdb.org/t/p/w1920_and_h800_multi_faces/${data.backdropPath}`}
|
||
style={{ width: '100%', height: '100%', objectFit: 'cover' }}
|
||
fill
|
||
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.title} />
|
||
<IssueModal
|
||
onCancel={() => setShowIssueModal(false)}
|
||
show={showIssueModal}
|
||
mediaType="movie"
|
||
tmdbId={data.id}
|
||
/>
|
||
<ManageSlideOver
|
||
data={data}
|
||
mediaType="movie"
|
||
onClose={() => {
|
||
setShowManager(false);
|
||
router.push({
|
||
pathname: router.pathname,
|
||
query: { movieId: router.query.movieId },
|
||
});
|
||
}}
|
||
revalidate={() => revalidate()}
|
||
show={showManager}
|
||
/>
|
||
<BlocklistModal
|
||
tmdbId={data.id}
|
||
type="movie"
|
||
show={showBlocklistModal}
|
||
onCancel={closeBlocklistModal}
|
||
onComplete={onClickHideItemBtn}
|
||
isUpdating={isBlocklistUpdating}
|
||
/>
|
||
<div className="media-header">
|
||
<div className="media-poster">
|
||
<CachedImage
|
||
type="tmdb"
|
||
src={
|
||
data.posterPath
|
||
? `https://image.tmdb.org/t/p/w600_and_h900_bestv2${data.posterPath}`
|
||
: '/images/seerr_poster_not_found.png'
|
||
}
|
||
alt=""
|
||
sizes="100vw"
|
||
style={{ width: '100%', height: 'auto' }}
|
||
width={600}
|
||
height={900}
|
||
priority
|
||
/>
|
||
</div>
|
||
<div className="media-title">
|
||
<div className="media-status">
|
||
<StatusBadge
|
||
status={data.mediaInfo?.status}
|
||
downloadItem={data.mediaInfo?.downloadStatus}
|
||
title={data.title}
|
||
inProgress={(data.mediaInfo?.downloadStatus ?? []).length > 0}
|
||
tmdbId={data.mediaInfo?.tmdbId}
|
||
mediaType="movie"
|
||
plexUrl={plexUrl}
|
||
serviceUrl={data.mediaInfo?.serviceUrl}
|
||
/>
|
||
{settings.currentSettings.movie4kEnabled &&
|
||
hasPermission(
|
||
[
|
||
Permission.MANAGE_REQUESTS,
|
||
Permission.REQUEST_4K,
|
||
Permission.REQUEST_4K_MOVIE,
|
||
],
|
||
{
|
||
type: 'or',
|
||
}
|
||
) && (
|
||
<StatusBadge
|
||
status={data.mediaInfo?.status4k}
|
||
downloadItem={data.mediaInfo?.downloadStatus4k}
|
||
title={data.title}
|
||
is4k
|
||
inProgress={
|
||
(data.mediaInfo?.downloadStatus4k ?? []).length > 0
|
||
}
|
||
tmdbId={data.mediaInfo?.tmdbId}
|
||
mediaType="movie"
|
||
plexUrl={plexUrl4k}
|
||
serviceUrl={data.mediaInfo?.serviceUrl4k}
|
||
/>
|
||
)}
|
||
</div>
|
||
<h1 data-testid="media-title">
|
||
{data.title}{' '}
|
||
{data.releaseDate && (
|
||
<span className="media-year">
|
||
({data.releaseDate.slice(0, 4)})
|
||
</span>
|
||
)}
|
||
</h1>
|
||
<span className="media-attributes">
|
||
{movieAttributes.length > 0 &&
|
||
movieAttributes
|
||
.map((t, k) => <span key={k}>{t}</span>)
|
||
.reduce((prev, curr) => (
|
||
<>
|
||
{prev}
|
||
<span>|</span>
|
||
{curr}
|
||
</>
|
||
))}
|
||
</span>
|
||
</div>
|
||
<div className="media-actions">
|
||
{showHideButton &&
|
||
data?.mediaInfo?.status !== MediaStatus.PROCESSING &&
|
||
data?.mediaInfo?.status !== MediaStatus.AVAILABLE &&
|
||
data?.mediaInfo?.status !== MediaStatus.PARTIALLY_AVAILABLE &&
|
||
data?.mediaInfo?.status !== MediaStatus.PENDING &&
|
||
data?.mediaInfo?.status !== MediaStatus.BLOCKLISTED && (
|
||
<Tooltip
|
||
content={intl.formatMessage(globalMessages.addToBlocklist)}
|
||
>
|
||
<Button
|
||
buttonType={'ghost'}
|
||
className="z-40 mr-2"
|
||
buttonSize={'md'}
|
||
onClick={() => setShowBlocklistModal(true)}
|
||
>
|
||
<EyeSlashIcon />
|
||
</Button>
|
||
</Tooltip>
|
||
)}
|
||
{data?.mediaInfo?.status !== MediaStatus.BLOCKLISTED &&
|
||
user?.userType !== UserType.PLEX && (
|
||
<>
|
||
{toggleWatchlist ? (
|
||
<Tooltip
|
||
content={intl.formatMessage(messages.addtowatchlist)}
|
||
>
|
||
<Button
|
||
buttonType={'ghost'}
|
||
className="z-40 mr-2"
|
||
buttonSize={'md'}
|
||
onClick={onClickWatchlistBtn}
|
||
>
|
||
{isUpdating ? (
|
||
<Spinner />
|
||
) : (
|
||
<StarIcon className={'text-amber-300'} />
|
||
)}
|
||
</Button>
|
||
</Tooltip>
|
||
) : (
|
||
<Tooltip
|
||
content={intl.formatMessage(messages.removefromwatchlist)}
|
||
>
|
||
<Button
|
||
className="z-40 mr-2"
|
||
buttonSize={'md'}
|
||
onClick={onClickDeleteWatchlistBtn}
|
||
>
|
||
{isUpdating ? <Spinner /> : <MinusCircleIcon />}
|
||
</Button>
|
||
</Tooltip>
|
||
)}
|
||
</>
|
||
)}
|
||
<div className="z-20">
|
||
<PlayButton links={mediaLinks} />
|
||
</div>
|
||
<RequestButton
|
||
mediaType="movie"
|
||
media={data.mediaInfo}
|
||
tmdbId={data.id}
|
||
onUpdate={() => revalidate()}
|
||
/>
|
||
{(data.mediaInfo?.status === MediaStatus.AVAILABLE ||
|
||
(settings.currentSettings.movie4kEnabled &&
|
||
hasPermission(
|
||
[Permission.REQUEST_4K, Permission.REQUEST_4K_MOVIE],
|
||
{
|
||
type: 'or',
|
||
}
|
||
) &&
|
||
data.mediaInfo?.status4k === MediaStatus.AVAILABLE)) &&
|
||
hasPermission(
|
||
[Permission.CREATE_ISSUES, Permission.MANAGE_ISSUES],
|
||
{
|
||
type: 'or',
|
||
}
|
||
) && (
|
||
<Tooltip content={intl.formatMessage(messages.reportissue)}>
|
||
<Button
|
||
buttonType="warning"
|
||
onClick={() => setShowIssueModal(true)}
|
||
className="ml-2 first:ml-0"
|
||
>
|
||
<ExclamationTriangleIcon />
|
||
</Button>
|
||
</Tooltip>
|
||
)}
|
||
{hasPermission(Permission.MANAGE_REQUESTS) &&
|
||
data.mediaInfo &&
|
||
(data.mediaInfo.jellyfinMediaId ||
|
||
data.mediaInfo.jellyfinMediaId4k ||
|
||
data.mediaInfo.status !== MediaStatus.UNKNOWN ||
|
||
data.mediaInfo.status4k !== MediaStatus.UNKNOWN) && (
|
||
<Tooltip content={intl.formatMessage(messages.managemovie)}>
|
||
<Button
|
||
buttonType="ghost"
|
||
onClick={() => setShowManager(true)}
|
||
className="relative ml-2 first:ml-0"
|
||
>
|
||
<CogIcon className="!mr-0" />
|
||
{hasPermission(
|
||
[Permission.MANAGE_ISSUES, Permission.VIEW_ISSUES],
|
||
{
|
||
type: 'or',
|
||
}
|
||
) &&
|
||
(
|
||
data.mediaInfo?.issues.filter(
|
||
(issue) => issue.status === IssueStatus.OPEN
|
||
) ?? []
|
||
).length > 0 && (
|
||
<>
|
||
<div className="absolute -right-1 -top-1 h-3 w-3 rounded-full bg-red-600" />
|
||
<div className="absolute -right-1 -top-1 h-3 w-3 animate-ping rounded-full bg-red-600" />
|
||
</>
|
||
)}
|
||
</Button>
|
||
</Tooltip>
|
||
)}
|
||
</div>
|
||
</div>
|
||
<div className="media-overview">
|
||
<div className="media-overview-left">
|
||
{data.tagline && <div className="tagline">{data.tagline}</div>}
|
||
<h2>{intl.formatMessage(messages.overview)}</h2>
|
||
<p>
|
||
{data.overview
|
||
? data.overview
|
||
: intl.formatMessage(messages.overviewunavailable)}
|
||
</p>
|
||
{sortedCrew.length > 0 && (
|
||
<>
|
||
<ul className="media-crew">
|
||
{sortedCrew.slice(0, 6).map((person) => (
|
||
<li key={`crew-${person.job}-${person.id}`}>
|
||
<span>{person.job}</span>
|
||
<Link href={`/person/${person.id}`} className="crew-name">
|
||
{person.name}
|
||
</Link>
|
||
</li>
|
||
))}
|
||
</ul>
|
||
<div className="mt-4 flex justify-end">
|
||
<Link
|
||
href={`/movie/${data.id}/crew`}
|
||
className="flex items-center text-gray-400 transition duration-300 hover:text-gray-100"
|
||
>
|
||
<span>{intl.formatMessage(messages.viewfullcrew)}</span>
|
||
<ArrowRightCircleIcon className="ml-1.5 inline-block h-5 w-5" />
|
||
</Link>
|
||
</div>
|
||
</>
|
||
)}
|
||
{data.keywords.length > 0 && (
|
||
<div className="mt-6">
|
||
{data.keywords.map((keyword) => (
|
||
<Link
|
||
href={`/discover/movies?keywords=${keyword.id}`}
|
||
key={`keyword-id-${keyword.id}`}
|
||
className="mb-2 mr-2 inline-flex last:mr-0"
|
||
>
|
||
<Tag>{keyword.name}</Tag>
|
||
</Link>
|
||
))}
|
||
</div>
|
||
)}
|
||
</div>
|
||
<div className="media-overview-right">
|
||
{data.collection && (
|
||
<div className="mb-6">
|
||
<Link href={`/collection/${data.collection.id}`}>
|
||
<div className="group relative z-0 scale-100 transform-gpu cursor-pointer overflow-hidden rounded-lg bg-gray-800 bg-cover bg-center shadow-md ring-1 ring-gray-700 transition duration-300 hover:scale-105 hover:ring-gray-500">
|
||
<div className="absolute inset-0 z-0">
|
||
<CachedImage
|
||
type="tmdb"
|
||
src={`https://image.tmdb.org/t/p/w1440_and_h320_multi_faces/${data.collection.backdropPath}`}
|
||
alt=""
|
||
style={{
|
||
width: '100%',
|
||
height: '100%',
|
||
objectFit: 'cover',
|
||
}}
|
||
fill
|
||
/>
|
||
<div
|
||
className="absolute inset-0"
|
||
style={{
|
||
backgroundImage:
|
||
'linear-gradient(180deg, rgba(31, 41, 55, 0.47) 0%, rgba(31, 41, 55, 0.80) 100%)',
|
||
}}
|
||
/>
|
||
</div>
|
||
<div className="relative z-10 flex h-full items-center justify-between p-4 text-gray-200 transition duration-300 group-hover:text-white">
|
||
<div>{data.collection.name}</div>
|
||
<Button buttonSize="sm">
|
||
{intl.formatMessage(globalMessages.view)}
|
||
</Button>
|
||
</div>
|
||
</div>
|
||
</Link>
|
||
</div>
|
||
)}
|
||
<div className="media-facts">
|
||
{(!!data.voteCount ||
|
||
(ratingData?.rt?.criticsRating &&
|
||
typeof ratingData?.rt?.criticsScore === 'number') ||
|
||
(ratingData?.rt?.audienceRating &&
|
||
!!ratingData?.rt?.audienceScore) ||
|
||
ratingData?.imdb?.criticsScore) && (
|
||
<div className="media-ratings">
|
||
{ratingData?.rt?.criticsRating &&
|
||
typeof ratingData?.rt?.criticsScore === 'number' && (
|
||
<Tooltip
|
||
content={intl.formatMessage(messages.rtcriticsscore)}
|
||
>
|
||
<a
|
||
href={ratingData.rt.url}
|
||
className="media-rating"
|
||
target="_blank"
|
||
rel="noreferrer"
|
||
>
|
||
{ratingData.rt.criticsRating === 'Rotten' ? (
|
||
<RTRotten className="w-6" />
|
||
) : (
|
||
<RTFresh className="w-6" />
|
||
)}
|
||
<span>{ratingData.rt.criticsScore}%</span>
|
||
</a>
|
||
</Tooltip>
|
||
)}
|
||
{ratingData?.rt?.audienceRating &&
|
||
!!ratingData?.rt?.audienceScore && (
|
||
<Tooltip
|
||
content={intl.formatMessage(messages.rtaudiencescore)}
|
||
>
|
||
<a
|
||
href={ratingData.rt.url}
|
||
className="media-rating"
|
||
target="_blank"
|
||
rel="noreferrer"
|
||
>
|
||
{ratingData.rt.audienceRating === 'Spilled' ? (
|
||
<RTAudRotten className="w-6" />
|
||
) : (
|
||
<RTAudFresh className="w-6" />
|
||
)}
|
||
<span>{ratingData.rt.audienceScore}%</span>
|
||
</a>
|
||
</Tooltip>
|
||
)}
|
||
{ratingData?.imdb?.criticsScore && (
|
||
<Tooltip
|
||
content={intl.formatMessage(messages.imdbuserscore, {
|
||
formattedCount: intl.formatNumber(
|
||
ratingData.imdb.criticsScoreCount,
|
||
{
|
||
notation: 'compact',
|
||
compactDisplay: 'short',
|
||
maximumFractionDigits: 1,
|
||
}
|
||
),
|
||
})}
|
||
>
|
||
<a
|
||
href={ratingData.imdb.url}
|
||
className="media-rating"
|
||
target="_blank"
|
||
rel="noreferrer"
|
||
>
|
||
<ImdbLogo className="mr-1 w-6" />
|
||
<span>{ratingData.imdb.criticsScore}</span>
|
||
</a>
|
||
</Tooltip>
|
||
)}
|
||
{!!data.voteCount && (
|
||
<Tooltip content={intl.formatMessage(messages.tmdbuserscore)}>
|
||
<a
|
||
href={`https://www.themoviedb.org/movie/${data.id}?language=${locale}`}
|
||
className="media-rating"
|
||
target="_blank"
|
||
rel="noreferrer"
|
||
>
|
||
<TmdbLogo className="mr-1 w-6" />
|
||
<span>{Math.round(data.voteAverage * 10)}%</span>
|
||
</a>
|
||
</Tooltip>
|
||
)}
|
||
</div>
|
||
)}
|
||
{data.originalTitle &&
|
||
data.originalLanguage !== locale.slice(0, 2) && (
|
||
<div className="media-fact">
|
||
<span>{intl.formatMessage(messages.originaltitle)}</span>
|
||
<span className="media-fact-value">{data.originalTitle}</span>
|
||
</div>
|
||
)}
|
||
<div className="media-fact">
|
||
<span>{intl.formatMessage(globalMessages.status)}</span>
|
||
<span className="media-fact-value">{data.status}</span>
|
||
</div>
|
||
{filteredReleases && filteredReleases.length > 0 ? (
|
||
<div className="media-fact">
|
||
<span>
|
||
{intl.formatMessage(messages.releasedate, {
|
||
releaseCount: filteredReleases.length,
|
||
})}
|
||
</span>
|
||
<span className="media-fact-value">
|
||
{filteredReleases.map((r, i) => (
|
||
<span
|
||
className="flex items-center justify-end"
|
||
key={`release-date-${i}`}
|
||
>
|
||
{r.type === 3 ? (
|
||
// Theatrical
|
||
<Tooltip
|
||
content={intl.formatMessage(
|
||
messages.theatricalrelease
|
||
)}
|
||
>
|
||
<TicketIcon className="h-4 w-4" />
|
||
</Tooltip>
|
||
) : r.type === 4 ? (
|
||
// Digital
|
||
<Tooltip
|
||
content={intl.formatMessage(messages.digitalrelease)}
|
||
>
|
||
<CloudIcon className="h-4 w-4" />
|
||
</Tooltip>
|
||
) : (
|
||
// Physical
|
||
<Tooltip
|
||
content={intl.formatMessage(messages.physicalrelease)}
|
||
>
|
||
<svg
|
||
className="h-4 w-4"
|
||
viewBox="0 0 24 24"
|
||
xmlns="http://www.w3.org/2000/svg"
|
||
>
|
||
<path
|
||
d="m12 2c-5.5242 0-10 4.4758-10 10 0 5.5242 4.4758 10 10 10 5.5242 0 10-4.4758 10-10 0-5.5242-4.4758-10-10-10zm0 18.065c-4.4476 0-8.0645-3.6169-8.0645-8.0645 0-4.4476 3.6169-8.0645 8.0645-8.0645 4.4476 0 8.0645 3.6169 8.0645 8.0645 0 4.4476-3.6169 8.0645-8.0645 8.0645zm0-14.516c-3.5565 0-6.4516 2.8952-6.4516 6.4516h1.2903c0-2.8468 2.3145-5.1613 5.1613-5.1613zm0 2.9032c-1.9597 0-3.5484 1.5887-3.5484 3.5484s1.5887 3.5484 3.5484 3.5484 3.5484-1.5887 3.5484-3.5484-1.5887-3.5484-3.5484-3.5484zm0 4.8387c-0.71371 0-1.2903-0.57661-1.2903-1.2903s0.57661-1.2903 1.2903-1.2903 1.2903 0.57661 1.2903 1.2903-0.57661 1.2903-1.2903 1.2903z"
|
||
fill="currentColor"
|
||
/>
|
||
</svg>
|
||
</Tooltip>
|
||
)}
|
||
<span className="ml-1.5">
|
||
{intl.formatDate(r.release_date, {
|
||
year: 'numeric',
|
||
month: 'long',
|
||
day: 'numeric',
|
||
timeZone: 'UTC',
|
||
})}
|
||
</span>
|
||
</span>
|
||
))}
|
||
</span>
|
||
</div>
|
||
) : (
|
||
data.releaseDate && (
|
||
<div className="media-fact">
|
||
<span>
|
||
{intl.formatMessage(messages.releasedate, {
|
||
releaseCount: 1,
|
||
})}
|
||
</span>
|
||
<span className="media-fact-value">
|
||
{intl.formatDate(data.releaseDate, {
|
||
year: 'numeric',
|
||
month: 'long',
|
||
day: 'numeric',
|
||
timeZone: 'UTC',
|
||
})}
|
||
</span>
|
||
</div>
|
||
)
|
||
)}
|
||
{data.revenue > 0 && (
|
||
<div className="media-fact">
|
||
<span>{intl.formatMessage(messages.revenue)}</span>
|
||
<span className="media-fact-value">
|
||
{intl.formatNumber(data.revenue, {
|
||
currency: 'USD',
|
||
style: 'currency',
|
||
})}
|
||
</span>
|
||
</div>
|
||
)}
|
||
{data.budget > 0 && (
|
||
<div className="media-fact">
|
||
<span>{intl.formatMessage(messages.budget)}</span>
|
||
<span className="media-fact-value">
|
||
{intl.formatNumber(data.budget, {
|
||
currency: 'USD',
|
||
style: 'currency',
|
||
})}
|
||
</span>
|
||
</div>
|
||
)}
|
||
{data.originalLanguage && (
|
||
<div className="media-fact">
|
||
<span>{intl.formatMessage(messages.originallanguage)}</span>
|
||
<span className="media-fact-value">
|
||
<Link
|
||
href={`/discover/movies/language/${data.originalLanguage}`}
|
||
>
|
||
{intl.formatDisplayName(data.originalLanguage, {
|
||
type: 'language',
|
||
fallback: 'none',
|
||
}) ??
|
||
data.spokenLanguages.find(
|
||
(lng) => lng.iso_639_1 === data.originalLanguage
|
||
)?.name}
|
||
</Link>
|
||
</span>
|
||
</div>
|
||
)}
|
||
{data.productionCountries.length > 0 && (
|
||
<div className="media-fact">
|
||
<span>
|
||
{intl.formatMessage(messages.productioncountries, {
|
||
countryCount: data.productionCountries.length,
|
||
})}
|
||
</span>
|
||
<span className="media-fact-value">
|
||
{data.productionCountries.map((c) => {
|
||
return (
|
||
<span
|
||
className="flex items-center justify-end"
|
||
key={`prodcountry-${c.iso_3166_1}`}
|
||
>
|
||
{countries.includes(c.iso_3166_1) && (
|
||
<span
|
||
className={`mr-1.5 text-xs leading-5 flag:${c.iso_3166_1}`}
|
||
/>
|
||
)}
|
||
<span>
|
||
{intl.formatDisplayName(c.iso_3166_1, {
|
||
type: 'region',
|
||
fallback: 'none',
|
||
}) ?? c.name}
|
||
</span>
|
||
</span>
|
||
);
|
||
})}
|
||
</span>
|
||
</div>
|
||
)}
|
||
{data.productionCompanies.length > 0 && (
|
||
<div className="media-fact">
|
||
<span>
|
||
{intl.formatMessage(messages.studio, {
|
||
studioCount: data.productionCompanies.length,
|
||
})}
|
||
</span>
|
||
<span className="media-fact-value">
|
||
{data.productionCompanies
|
||
.slice(
|
||
0,
|
||
showAllStudios || showMoreStudios
|
||
? data.productionCompanies.length
|
||
: minStudios
|
||
)
|
||
.map((s) => {
|
||
return (
|
||
<Link
|
||
href={`/discover/movies/studio/${s.id}`}
|
||
key={`studio-${s.id}`}
|
||
className="block"
|
||
>
|
||
{s.name}
|
||
</Link>
|
||
);
|
||
})}
|
||
{!showAllStudios && (
|
||
<button
|
||
type="button"
|
||
onClick={() => {
|
||
setShowMoreStudios(!showMoreStudios);
|
||
}}
|
||
>
|
||
<span className="flex items-center">
|
||
{intl.formatMessage(
|
||
!showMoreStudios
|
||
? messages.showmore
|
||
: messages.showless
|
||
)}
|
||
{!showMoreStudios ? (
|
||
<ChevronDoubleDownIcon className="ml-1 h-4 w-4" />
|
||
) : (
|
||
<ChevronDoubleUpIcon className="ml-1 h-4 w-4" />
|
||
)}
|
||
</span>
|
||
</button>
|
||
)}
|
||
</span>
|
||
</div>
|
||
)}
|
||
{!!streamingProviders.length && (
|
||
<div className="media-fact flex-col gap-1">
|
||
<span>{intl.formatMessage(messages.streamingproviders)}</span>
|
||
<span className="media-fact-value flex flex-row flex-wrap gap-5">
|
||
{streamingProviders.map((p) => {
|
||
return (
|
||
<Tooltip content={p.name}>
|
||
<span
|
||
className="opacity-50 transition duration-300 hover:opacity-100"
|
||
key={`provider-${p.id}`}
|
||
>
|
||
<CachedImage
|
||
type="tmdb"
|
||
src={'https://image.tmdb.org/t/p/w45/' + p.logoPath}
|
||
alt={p.name}
|
||
width={32}
|
||
height={32}
|
||
className="rounded-md"
|
||
/>
|
||
</span>
|
||
</Tooltip>
|
||
);
|
||
})}
|
||
</span>
|
||
</div>
|
||
)}
|
||
<div className="media-fact">
|
||
<ExternalLinkBlock
|
||
mediaType="movie"
|
||
tmdbId={data.id}
|
||
tvdbId={data.externalIds.tvdbId}
|
||
imdbId={data.externalIds.imdbId}
|
||
rtUrl={ratingData?.rt?.url}
|
||
mediaUrl={
|
||
data.mediaInfo?.mediaUrl ?? data.mediaInfo?.mediaUrl4k
|
||
}
|
||
/>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
{data.credits.cast.length > 0 && (
|
||
<>
|
||
<div className="slider-header">
|
||
<Link
|
||
href="/movie/[movieId]/cast"
|
||
as={`/movie/${data.id}/cast`}
|
||
className="slider-title"
|
||
>
|
||
<span>{intl.formatMessage(messages.cast)}</span>
|
||
<ArrowRightCircleIcon />
|
||
</Link>
|
||
</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="extra-bottom-space relative" />
|
||
</div>
|
||
);
|
||
};
|
||
|
||
export default MovieDetails;
|