mirror of
https://github.com/movie-web/movie-web.git
synced 2025-01-04 08:25:58 +00:00
Merge pull request #127 from destruc7i0n/v3
Show more episode info in the page title for a series
This commit is contained in:
commit
0ca751f1d2
|
@ -14,6 +14,7 @@ import { ShowTitleControl } from "./controls/ShowTitleControl";
|
||||||
import { SkipTime } from "./controls/SkipTime";
|
import { SkipTime } from "./controls/SkipTime";
|
||||||
import { TimeControl } from "./controls/TimeControl";
|
import { TimeControl } from "./controls/TimeControl";
|
||||||
import { VolumeControl } from "./controls/VolumeControl";
|
import { VolumeControl } from "./controls/VolumeControl";
|
||||||
|
import { PageTitleControl } from "./controls/PageTitleControl";
|
||||||
import { VideoPlayerError } from "./parts/VideoPlayerError";
|
import { VideoPlayerError } from "./parts/VideoPlayerError";
|
||||||
import { VideoPlayerHeader } from "./parts/VideoPlayerHeader";
|
import { VideoPlayerHeader } from "./parts/VideoPlayerHeader";
|
||||||
import { useVideoPlayerState } from "./VideoContext";
|
import { useVideoPlayerState } from "./VideoContext";
|
||||||
|
@ -67,6 +68,7 @@ export function DecoratedVideoPlayer(
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<VideoPlayer autoPlay={props.autoPlay}>
|
<VideoPlayer autoPlay={props.autoPlay}>
|
||||||
|
<PageTitleControl media={props.media} />
|
||||||
<VideoPlayerError media={props.media} onGoBack={props.onGoBack}>
|
<VideoPlayerError media={props.media} onGoBack={props.onGoBack}>
|
||||||
<BackdropControl onBackdropChange={onBackdropChange}>
|
<BackdropControl onBackdropChange={onBackdropChange}>
|
||||||
<div className="absolute inset-0 flex items-center justify-center">
|
<div className="absolute inset-0 flex items-center justify-center">
|
||||||
|
|
23
src/components/video/controls/PageTitleControl.tsx
Normal file
23
src/components/video/controls/PageTitleControl.tsx
Normal file
|
@ -0,0 +1,23 @@
|
||||||
|
import { MWMediaMeta } from "@/backend/metadata/types";
|
||||||
|
import { Helmet } from "react-helmet";
|
||||||
|
import { useCurrentSeriesEpisodeInfo } from "../hooks/useCurrentSeriesEpisodeInfo";
|
||||||
|
|
||||||
|
interface PageTitleControlProps {
|
||||||
|
media?: MWMediaMeta;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function PageTitleControl(props: PageTitleControlProps) {
|
||||||
|
const { isSeries, humanizedEpisodeId } = useCurrentSeriesEpisodeInfo();
|
||||||
|
|
||||||
|
if (!props.media) return null;
|
||||||
|
|
||||||
|
const title = isSeries
|
||||||
|
? `${props.media.title} - ${humanizedEpisodeId}`
|
||||||
|
: props.media.title;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Helmet>
|
||||||
|
<title>{title}</title>
|
||||||
|
</Helmet>
|
||||||
|
);
|
||||||
|
}
|
|
@ -1,29 +1,14 @@
|
||||||
import { useMemo } from "react";
|
import { useCurrentSeriesEpisodeInfo } from "../hooks/useCurrentSeriesEpisodeInfo";
|
||||||
import { useVideoPlayerState } from "../VideoContext";
|
|
||||||
|
|
||||||
export function ShowTitleControl() {
|
export function ShowTitleControl() {
|
||||||
const { videoState } = useVideoPlayerState();
|
const { isSeries, currentEpisodeInfo, humanizedEpisodeId } =
|
||||||
|
useCurrentSeriesEpisodeInfo();
|
||||||
|
|
||||||
const { current, seasons } = videoState.seasonData;
|
if (!isSeries) return null;
|
||||||
|
|
||||||
const currentSeasonInfo = useMemo(() => {
|
|
||||||
return seasons?.find((season) => season.id === current?.seasonId);
|
|
||||||
}, [seasons, current]);
|
|
||||||
|
|
||||||
const currentEpisodeInfo = useMemo(() => {
|
|
||||||
return currentSeasonInfo?.episodes?.find(
|
|
||||||
(episode) => episode.id === current?.episodeId
|
|
||||||
);
|
|
||||||
}, [currentSeasonInfo, current]);
|
|
||||||
|
|
||||||
if (!videoState.seasonData.isSeries) return null;
|
|
||||||
if (!videoState.seasonData.current) return null;
|
|
||||||
|
|
||||||
const selectedText = `S${currentSeasonInfo?.number} E${currentEpisodeInfo?.number}`;
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<p className="ml-8 select-none space-x-2 text-white">
|
<p className="ml-8 select-none space-x-2 text-white">
|
||||||
<span>{selectedText}</span>
|
<span>{humanizedEpisodeId}</span>
|
||||||
<span className="opacity-50">{currentEpisodeInfo?.title}</span>
|
<span className="opacity-50">{currentEpisodeInfo?.title}</span>
|
||||||
</p>
|
</p>
|
||||||
);
|
);
|
||||||
|
|
33
src/components/video/hooks/useCurrentSeriesEpisodeInfo.ts
Normal file
33
src/components/video/hooks/useCurrentSeriesEpisodeInfo.ts
Normal file
|
@ -0,0 +1,33 @@
|
||||||
|
import { useMemo } from "react";
|
||||||
|
import { useVideoPlayerState } from "../VideoContext";
|
||||||
|
|
||||||
|
export function useCurrentSeriesEpisodeInfo() {
|
||||||
|
const { videoState } = useVideoPlayerState();
|
||||||
|
|
||||||
|
const { current, seasons } = videoState.seasonData;
|
||||||
|
|
||||||
|
const currentSeasonInfo = useMemo(() => {
|
||||||
|
return seasons?.find((season) => season.id === current?.seasonId);
|
||||||
|
}, [seasons, current]);
|
||||||
|
|
||||||
|
const currentEpisodeInfo = useMemo(() => {
|
||||||
|
return currentSeasonInfo?.episodes?.find(
|
||||||
|
(episode) => episode.id === current?.episodeId
|
||||||
|
);
|
||||||
|
}, [currentSeasonInfo, current]);
|
||||||
|
|
||||||
|
const isSeries = Boolean(
|
||||||
|
videoState.seasonData.isSeries && videoState.seasonData.current
|
||||||
|
);
|
||||||
|
|
||||||
|
if (!isSeries) return { isSeries: false };
|
||||||
|
|
||||||
|
const humanizedEpisodeId = `S${currentSeasonInfo?.number} E${currentEpisodeInfo?.number}`;
|
||||||
|
|
||||||
|
return {
|
||||||
|
isSeries: true,
|
||||||
|
humanizedEpisodeId,
|
||||||
|
currentSeasonInfo,
|
||||||
|
currentEpisodeInfo,
|
||||||
|
};
|
||||||
|
}
|
|
@ -111,7 +111,6 @@ export function MediaViewPlayer(props: MediaViewPlayerProps) {
|
||||||
return (
|
return (
|
||||||
<div className="fixed top-0 left-0 h-[100dvh] w-screen">
|
<div className="fixed top-0 left-0 h-[100dvh] w-screen">
|
||||||
<Helmet>
|
<Helmet>
|
||||||
<title>{props.meta.meta.title}</title>
|
|
||||||
<html data-full="true" />
|
<html data-full="true" />
|
||||||
</Helmet>
|
</Helmet>
|
||||||
<DecoratedVideoPlayer media={props.meta.meta} onGoBack={goBack} autoPlay>
|
<DecoratedVideoPlayer media={props.meta.meta} onGoBack={goBack} autoPlay>
|
||||||
|
|
Loading…
Reference in a new issue