mirror of
https://github.com/movie-web/movie-web.git
synced 2025-01-11 22:49:10 +01:00
source selection
Co-authored-by: Jip Frijlink <JipFr@users.noreply.github.com>
This commit is contained in:
parent
056f837dcb
commit
e448c0b5a8
@ -163,8 +163,6 @@ registerProvider({
|
|||||||
|
|
||||||
const subtitleRes = (await get(subtitleApiQuery)).data;
|
const subtitleRes = (await get(subtitleApiQuery)).data;
|
||||||
|
|
||||||
console.log(subtitleRes);
|
|
||||||
|
|
||||||
const mappedCaptions = subtitleRes.list.map(
|
const mappedCaptions = subtitleRes.list.map(
|
||||||
(subtitle: any): MWCaption => {
|
(subtitle: any): MWCaption => {
|
||||||
return {
|
return {
|
||||||
|
@ -97,7 +97,13 @@ function MediaCardContent({
|
|||||||
<h1 className="mb-1 max-h-[4.5rem] text-ellipsis break-words font-bold text-white line-clamp-3">
|
<h1 className="mb-1 max-h-[4.5rem] text-ellipsis break-words font-bold text-white line-clamp-3">
|
||||||
<span>{media.title}</span>
|
<span>{media.title}</span>
|
||||||
</h1>
|
</h1>
|
||||||
<DotList className="text-xs" content={[media.type, media.year]} />
|
<DotList
|
||||||
|
className="text-xs"
|
||||||
|
content={[
|
||||||
|
media.type.slice(0, 1).toUpperCase() + media.type.slice(1),
|
||||||
|
media.year,
|
||||||
|
]}
|
||||||
|
/>
|
||||||
</article>
|
</article>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
|
@ -19,13 +19,11 @@ if (key) {
|
|||||||
initializeChromecast();
|
initializeChromecast();
|
||||||
|
|
||||||
// TODO video todos:
|
// TODO video todos:
|
||||||
// - finish captions
|
|
||||||
// - chrome cast support
|
// - chrome cast support
|
||||||
// - bug: mobile controls start showing when resizing
|
|
||||||
// - bug: popouts sometimes stop working when selecting different episode
|
|
||||||
// - bug: unmounting player throws errors in console
|
// - bug: unmounting player throws errors in console
|
||||||
// - bug: safari fullscreen will make video overlap player controls
|
// - bug: safari fullscreen will make video overlap player controls
|
||||||
// - bug: safari progress bar is fucked (video doesnt change time but video.currentTime does change)
|
// - improvement: make scrapers use fuzzy matching on normalized titles
|
||||||
|
// - bug: source selection doesnt work with HLS
|
||||||
|
|
||||||
// TODO stuff to test:
|
// TODO stuff to test:
|
||||||
// - browser: firefox, chrome, edge, safari desktop
|
// - browser: firefox, chrome, edge, safari desktop
|
||||||
|
@ -12,6 +12,7 @@ import { PauseAction } from "@/video/components/actions/PauseAction";
|
|||||||
import { ProgressAction } from "@/video/components/actions/ProgressAction";
|
import { ProgressAction } from "@/video/components/actions/ProgressAction";
|
||||||
import { QualityDisplayAction } from "@/video/components/actions/QualityDisplayAction";
|
import { QualityDisplayAction } from "@/video/components/actions/QualityDisplayAction";
|
||||||
import { SeriesSelectionAction } from "@/video/components/actions/SeriesSelectionAction";
|
import { SeriesSelectionAction } from "@/video/components/actions/SeriesSelectionAction";
|
||||||
|
import { SourceSelectionAction } from "@/video/components/actions/SourceSelectionAction";
|
||||||
import { CaptionsSelectionAction } from "@/video/components/actions/CaptionsSelectionAction";
|
import { CaptionsSelectionAction } from "@/video/components/actions/CaptionsSelectionAction";
|
||||||
import { ShowTitleAction } from "@/video/components/actions/ShowTitleAction";
|
import { ShowTitleAction } from "@/video/components/actions/ShowTitleAction";
|
||||||
import { KeyboardShortcutsAction } from "@/video/components/actions/KeyboardShortcutsAction";
|
import { KeyboardShortcutsAction } from "@/video/components/actions/KeyboardShortcutsAction";
|
||||||
@ -77,7 +78,6 @@ export function VideoPlayer(props: Props) {
|
|||||||
[setShow]
|
[setShow]
|
||||||
);
|
);
|
||||||
|
|
||||||
// TODO source selection
|
|
||||||
return (
|
return (
|
||||||
<VideoPlayerBase
|
<VideoPlayerBase
|
||||||
autoPlay={props.autoPlay}
|
autoPlay={props.autoPlay}
|
||||||
@ -148,6 +148,7 @@ export function VideoPlayer(props: Props) {
|
|||||||
<div className="flex-1" />
|
<div className="flex-1" />
|
||||||
<QualityDisplayAction />
|
<QualityDisplayAction />
|
||||||
<SeriesSelectionAction />
|
<SeriesSelectionAction />
|
||||||
|
<SourceSelectionAction />
|
||||||
{/* <SourceSelectionControl media={props.media} /> */}
|
{/* <SourceSelectionControl media={props.media} /> */}
|
||||||
<div className="mx-2 h-6 w-px bg-white opacity-50" />
|
<div className="mx-2 h-6 w-px bg-white opacity-50" />
|
||||||
{/* <ChromeCastControl /> */}
|
{/* <ChromeCastControl /> */}
|
||||||
|
@ -3,6 +3,7 @@ import { useVideoPlayerDescriptor } from "@/video/state/hooks";
|
|||||||
import { VideoPlayerIconButton } from "@/video/components/parts/VideoPlayerIconButton";
|
import { VideoPlayerIconButton } from "@/video/components/parts/VideoPlayerIconButton";
|
||||||
import { useControls } from "@/video/state/logic/controls";
|
import { useControls } from "@/video/state/logic/controls";
|
||||||
import { PopoutAnchor } from "@/video/components/popouts/PopoutAnchor";
|
import { PopoutAnchor } from "@/video/components/popouts/PopoutAnchor";
|
||||||
|
import { useIsMobile } from "@/hooks/useIsMobile";
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
className?: string;
|
className?: string;
|
||||||
@ -11,6 +12,7 @@ interface Props {
|
|||||||
export function CaptionsSelectionAction(props: Props) {
|
export function CaptionsSelectionAction(props: Props) {
|
||||||
const descriptor = useVideoPlayerDescriptor();
|
const descriptor = useVideoPlayerDescriptor();
|
||||||
const controls = useControls(descriptor);
|
const controls = useControls(descriptor);
|
||||||
|
const { isMobile } = useIsMobile();
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className={props.className}>
|
<div className={props.className}>
|
||||||
@ -18,6 +20,8 @@ export function CaptionsSelectionAction(props: Props) {
|
|||||||
<PopoutAnchor for="captions">
|
<PopoutAnchor for="captions">
|
||||||
<VideoPlayerIconButton
|
<VideoPlayerIconButton
|
||||||
className={props.className}
|
className={props.className}
|
||||||
|
text={isMobile ? "Captions" : ""}
|
||||||
|
wide={isMobile}
|
||||||
onClick={() => controls.openPopout("captions")}
|
onClick={() => controls.openPopout("captions")}
|
||||||
icon={Icons.CAPTIONS}
|
icon={Icons.CAPTIONS}
|
||||||
/>
|
/>
|
||||||
|
32
src/video/components/actions/SourceSelectionAction.tsx
Normal file
32
src/video/components/actions/SourceSelectionAction.tsx
Normal file
@ -0,0 +1,32 @@
|
|||||||
|
import { Icons } from "@/components/Icon";
|
||||||
|
import { useVideoPlayerDescriptor } from "@/video/state/hooks";
|
||||||
|
import { VideoPlayerIconButton } from "@/video/components/parts/VideoPlayerIconButton";
|
||||||
|
import { useControls } from "@/video/state/logic/controls";
|
||||||
|
import { PopoutAnchor } from "@/video/components/popouts/PopoutAnchor";
|
||||||
|
import { useInterface } from "@/video/state/logic/interface";
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
className?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function SourceSelectionAction(props: Props) {
|
||||||
|
const descriptor = useVideoPlayerDescriptor();
|
||||||
|
const videoInterface = useInterface(descriptor);
|
||||||
|
const controls = useControls(descriptor);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className={props.className}>
|
||||||
|
<div className="relative">
|
||||||
|
<PopoutAnchor for="source">
|
||||||
|
<VideoPlayerIconButton
|
||||||
|
active={videoInterface.popout === "source"}
|
||||||
|
icon={Icons.FILE}
|
||||||
|
text="Source"
|
||||||
|
wide
|
||||||
|
onClick={() => controls.openPopout("source")}
|
||||||
|
/>
|
||||||
|
</PopoutAnchor>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
@ -1,6 +1,7 @@
|
|||||||
import { Transition } from "@/components/Transition";
|
import { Transition } from "@/components/Transition";
|
||||||
import { useSyncPopouts } from "@/video/components/hooks/useSyncPopouts";
|
import { useSyncPopouts } from "@/video/components/hooks/useSyncPopouts";
|
||||||
import { EpisodeSelectionPopout } from "@/video/components/popouts/EpisodeSelectionPopout";
|
import { EpisodeSelectionPopout } from "@/video/components/popouts/EpisodeSelectionPopout";
|
||||||
|
import { SourceSelectionPopout } from "@/video/components/popouts/SourceSelectionPopout";
|
||||||
import { CaptionSelectionPopout } from "@/video/components/popouts/CaptionSelectionPopout";
|
import { CaptionSelectionPopout } from "@/video/components/popouts/CaptionSelectionPopout";
|
||||||
import { useVideoPlayerDescriptor } from "@/video/state/hooks";
|
import { useVideoPlayerDescriptor } from "@/video/state/hooks";
|
||||||
import { useControls } from "@/video/state/logic/controls";
|
import { useControls } from "@/video/state/logic/controls";
|
||||||
@ -21,8 +22,13 @@ function ShowPopout(props: { popoutId: string | null }) {
|
|||||||
}, [props]);
|
}, [props]);
|
||||||
|
|
||||||
if (popoutId === "episodes") return <EpisodeSelectionPopout />;
|
if (popoutId === "episodes") return <EpisodeSelectionPopout />;
|
||||||
|
if (popoutId === "source") return <SourceSelectionPopout />;
|
||||||
if (popoutId === "captions") return <CaptionSelectionPopout />;
|
if (popoutId === "captions") return <CaptionSelectionPopout />;
|
||||||
return null;
|
return (
|
||||||
|
<div className="flex w-full items-center justify-center p-10">
|
||||||
|
Unknown popout
|
||||||
|
</div>
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
function PopoutContainer(props: { videoInterface: VideoInterfaceEvent }) {
|
function PopoutContainer(props: { videoInterface: VideoInterfaceEvent }) {
|
||||||
|
205
src/video/components/popouts/SourceSelectionPopout.tsx
Normal file
205
src/video/components/popouts/SourceSelectionPopout.tsx
Normal file
@ -0,0 +1,205 @@
|
|||||||
|
import { useMemo, useRef, useState } from "react";
|
||||||
|
import { Icon, Icons } from "@/components/Icon";
|
||||||
|
import { useLoading } from "@/hooks/useLoading";
|
||||||
|
import { Loading } from "@/components/layout/Loading";
|
||||||
|
import { IconPatch } from "@/components/buttons/IconPatch";
|
||||||
|
import { useVideoPlayerDescriptor } from "@/video/state/hooks";
|
||||||
|
import { useMeta } from "@/video/state/logic/meta";
|
||||||
|
import { useControls } from "@/video/state/logic/controls";
|
||||||
|
import { MWStream } from "@/backend/helpers/streams";
|
||||||
|
import { getProviders } from "@/backend/helpers/register";
|
||||||
|
import { runProvider } from "@/backend/helpers/run";
|
||||||
|
import { MWProviderScrapeResult } from "@/backend/helpers/provider";
|
||||||
|
import { PopoutListEntry, PopoutSection } from "./PopoutUtils";
|
||||||
|
|
||||||
|
// TODO HLS does not work
|
||||||
|
export function SourceSelectionPopout() {
|
||||||
|
const descriptor = useVideoPlayerDescriptor();
|
||||||
|
const controls = useControls(descriptor);
|
||||||
|
const meta = useMeta(descriptor);
|
||||||
|
const providers = useMemo(
|
||||||
|
() =>
|
||||||
|
meta ? getProviders().filter((v) => v.type.includes(meta.meta.type)) : [],
|
||||||
|
[meta]
|
||||||
|
);
|
||||||
|
|
||||||
|
const [selectedProvider, setSelectedProvider] = useState<string | null>(null);
|
||||||
|
const [scrapeResult, setScrapeResult] =
|
||||||
|
useState<MWProviderScrapeResult | null>(null);
|
||||||
|
const showingProvider = !!selectedProvider;
|
||||||
|
const selectedProviderPopulated = useMemo(
|
||||||
|
() => providers.find((v) => v.id === selectedProvider) ?? null,
|
||||||
|
[providers, selectedProvider]
|
||||||
|
);
|
||||||
|
const [runScraper, loading, error] = useLoading(
|
||||||
|
async (providerId: string) => {
|
||||||
|
const theProvider = providers.find((v) => v.id === providerId);
|
||||||
|
if (!theProvider) throw new Error("Invalid provider");
|
||||||
|
if (!meta) throw new Error("need meta");
|
||||||
|
return runProvider(theProvider, {
|
||||||
|
media: {
|
||||||
|
imdbId: "", // TODO get actual ids
|
||||||
|
tmdbId: "",
|
||||||
|
meta: meta.meta,
|
||||||
|
},
|
||||||
|
progress: () => {},
|
||||||
|
type: meta.meta.type,
|
||||||
|
episode: meta.episode?.episodeId as any,
|
||||||
|
season: meta.episode?.seasonId as any,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
function selectSource(stream: MWStream) {
|
||||||
|
controls.setSource({
|
||||||
|
quality: stream.quality,
|
||||||
|
source: stream.streamUrl,
|
||||||
|
type: stream.type,
|
||||||
|
});
|
||||||
|
if (meta) {
|
||||||
|
controls.setMeta({
|
||||||
|
...meta,
|
||||||
|
captions: stream.captions,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
controls.closePopout();
|
||||||
|
}
|
||||||
|
|
||||||
|
const providerRef = useRef<string | null>(null);
|
||||||
|
const selectProvider = (providerId?: string) => {
|
||||||
|
if (!providerId) {
|
||||||
|
providerRef.current = null;
|
||||||
|
setSelectedProvider(null);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
runScraper(providerId).then((v) => {
|
||||||
|
if (!providerRef.current) return;
|
||||||
|
if (v) {
|
||||||
|
const len = v.embeds.length + (v.stream ? 1 : 0);
|
||||||
|
if (len === 1) {
|
||||||
|
const realStream = v.stream;
|
||||||
|
if (!realStream) {
|
||||||
|
// TODO scrape embed
|
||||||
|
throw new Error("no embed scraper configured");
|
||||||
|
}
|
||||||
|
selectSource(realStream);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
setScrapeResult(v ?? null);
|
||||||
|
});
|
||||||
|
providerRef.current = providerId;
|
||||||
|
setSelectedProvider(providerId);
|
||||||
|
};
|
||||||
|
|
||||||
|
const titlePositionClass = useMemo(() => {
|
||||||
|
const offset = !showingProvider ? "left-0" : "left-10";
|
||||||
|
return [
|
||||||
|
"absolute w-full transition-[left,opacity] duration-200",
|
||||||
|
offset,
|
||||||
|
].join(" ");
|
||||||
|
}, [showingProvider]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<PopoutSection className="bg-ash-100 font-bold text-white">
|
||||||
|
<div className="relative flex items-center">
|
||||||
|
<button
|
||||||
|
className={[
|
||||||
|
"-m-1.5 rounded-lg p-1.5 transition-opacity duration-100 hover:bg-ash-200",
|
||||||
|
!showingProvider ? "pointer-events-none opacity-0" : "opacity-1",
|
||||||
|
].join(" ")}
|
||||||
|
onClick={() => selectProvider()}
|
||||||
|
type="button"
|
||||||
|
>
|
||||||
|
<Icon icon={Icons.CHEVRON_LEFT} />
|
||||||
|
</button>
|
||||||
|
<span
|
||||||
|
className={[
|
||||||
|
titlePositionClass,
|
||||||
|
showingProvider ? "opacity-1" : "opacity-0",
|
||||||
|
].join(" ")}
|
||||||
|
>
|
||||||
|
{selectedProviderPopulated?.displayName ?? ""}
|
||||||
|
</span>
|
||||||
|
<span
|
||||||
|
className={[
|
||||||
|
titlePositionClass,
|
||||||
|
!showingProvider ? "opacity-1" : "opacity-0",
|
||||||
|
].join(" ")}
|
||||||
|
>
|
||||||
|
Sources
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</PopoutSection>
|
||||||
|
<div className="relative grid h-full grid-rows-[minmax(1px,1fr)]">
|
||||||
|
<PopoutSection
|
||||||
|
className={[
|
||||||
|
"absolute inset-0 z-30 overflow-y-auto border-ash-400 bg-ash-100 transition-[max-height,padding] duration-200",
|
||||||
|
showingProvider
|
||||||
|
? "max-h-full border-t"
|
||||||
|
: "max-h-0 overflow-hidden py-0",
|
||||||
|
].join(" ")}
|
||||||
|
>
|
||||||
|
{loading ? (
|
||||||
|
<div className="flex h-full w-full items-center justify-center">
|
||||||
|
<Loading />
|
||||||
|
</div>
|
||||||
|
) : error ? (
|
||||||
|
<div className="flex h-full w-full items-center justify-center">
|
||||||
|
<div className="flex flex-col flex-wrap items-center text-slate-400">
|
||||||
|
<IconPatch
|
||||||
|
icon={Icons.EYE_SLASH}
|
||||||
|
className="text-xl text-bink-600"
|
||||||
|
/>
|
||||||
|
<p className="mt-6 w-full text-center">
|
||||||
|
Something went wrong loading the embeds for this thing that
|
||||||
|
you like
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
) : (
|
||||||
|
<>
|
||||||
|
{scrapeResult?.stream ? (
|
||||||
|
<PopoutListEntry
|
||||||
|
isOnDarkBackground
|
||||||
|
onClick={() => {
|
||||||
|
if (scrapeResult.stream) selectSource(scrapeResult.stream);
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
Native source
|
||||||
|
</PopoutListEntry>
|
||||||
|
) : null}
|
||||||
|
{scrapeResult?.embeds.map((v) => (
|
||||||
|
<PopoutListEntry
|
||||||
|
isOnDarkBackground
|
||||||
|
key={v.url}
|
||||||
|
onClick={() => {
|
||||||
|
console.log("EMBED CHOSEN");
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{v.type}
|
||||||
|
</PopoutListEntry>
|
||||||
|
))}
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</PopoutSection>
|
||||||
|
<PopoutSection className="relative h-full overflow-y-auto">
|
||||||
|
<div>
|
||||||
|
{providers.map((v) => (
|
||||||
|
<PopoutListEntry
|
||||||
|
key={v.id}
|
||||||
|
onClick={() => {
|
||||||
|
selectProvider(v.id);
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{v.displayName}
|
||||||
|
</PopoutListEntry>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
</PopoutSection>
|
||||||
|
</div>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
@ -2,6 +2,27 @@ import { nanoid } from "nanoid";
|
|||||||
import { _players } from "./cache";
|
import { _players } from "./cache";
|
||||||
import { VideoPlayerState } from "./types";
|
import { VideoPlayerState } from "./types";
|
||||||
|
|
||||||
|
export function resetForSource(s: VideoPlayerState) {
|
||||||
|
const state = s;
|
||||||
|
state.mediaPlaying = {
|
||||||
|
isPlaying: false,
|
||||||
|
isPaused: true,
|
||||||
|
isLoading: false,
|
||||||
|
isSeeking: false,
|
||||||
|
isDragSeeking: false,
|
||||||
|
isFirstLoading: true,
|
||||||
|
hasPlayedOnce: false,
|
||||||
|
volume: 0,
|
||||||
|
};
|
||||||
|
state.progress = {
|
||||||
|
time: 0,
|
||||||
|
duration: 0,
|
||||||
|
buffered: 0,
|
||||||
|
draggingTime: 0,
|
||||||
|
};
|
||||||
|
state.initalized = false;
|
||||||
|
}
|
||||||
|
|
||||||
function initPlayer(): VideoPlayerState {
|
function initPlayer(): VideoPlayerState {
|
||||||
return {
|
return {
|
||||||
interface: {
|
interface: {
|
||||||
@ -38,6 +59,7 @@ function initPlayer(): VideoPlayerState {
|
|||||||
initalized: false,
|
initalized: false,
|
||||||
|
|
||||||
pausedWhenSeeking: false,
|
pausedWhenSeeking: false,
|
||||||
|
hlsInstance: null,
|
||||||
stateProvider: null,
|
stateProvider: null,
|
||||||
wrapperElement: null,
|
wrapperElement: null,
|
||||||
};
|
};
|
||||||
|
17
src/video/state/providers/helpers.ts
Normal file
17
src/video/state/providers/helpers.ts
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
import { resetForSource } from "@/video/state/init";
|
||||||
|
import { updateMediaPlaying } from "@/video/state/logic/mediaplaying";
|
||||||
|
import { updateMisc } from "@/video/state/logic/misc";
|
||||||
|
import { updateProgress } from "@/video/state/logic/progress";
|
||||||
|
import { VideoPlayerState } from "@/video/state/types";
|
||||||
|
|
||||||
|
export function resetStateForSource(descriptor: string, s: VideoPlayerState) {
|
||||||
|
const state = s;
|
||||||
|
if (state.hlsInstance) {
|
||||||
|
state.hlsInstance.destroy();
|
||||||
|
state.hlsInstance = null;
|
||||||
|
}
|
||||||
|
resetForSource(state);
|
||||||
|
updateMediaPlaying(descriptor, state);
|
||||||
|
updateProgress(descriptor, state);
|
||||||
|
updateMisc(descriptor, state);
|
||||||
|
}
|
@ -15,6 +15,7 @@ import {
|
|||||||
} from "@/video/components/hooks/volumeStore";
|
} from "@/video/components/hooks/volumeStore";
|
||||||
import { updateError } from "@/video/state/logic/error";
|
import { updateError } from "@/video/state/logic/error";
|
||||||
import { updateMisc } from "@/video/state/logic/misc";
|
import { updateMisc } from "@/video/state/logic/misc";
|
||||||
|
import { resetStateForSource } from "@/video/state/providers/helpers";
|
||||||
import { getPlayerState } from "../cache";
|
import { getPlayerState } from "../cache";
|
||||||
import { updateMediaPlaying } from "../logic/mediaplaying";
|
import { updateMediaPlaying } from "../logic/mediaplaying";
|
||||||
import { VideoPlayerStateProvider } from "./providerTypes";
|
import { VideoPlayerStateProvider } from "./providerTypes";
|
||||||
@ -130,6 +131,7 @@ export function createVideoStateProvider(
|
|||||||
if (!source) {
|
if (!source) {
|
||||||
player.src = "";
|
player.src = "";
|
||||||
state.source = null;
|
state.source = null;
|
||||||
|
resetStateForSource(descriptor, state);
|
||||||
updateSource(descriptor, state);
|
updateSource(descriptor, state);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -149,6 +151,7 @@ export function createVideoStateProvider(
|
|||||||
}
|
}
|
||||||
|
|
||||||
const hls = new Hls({ enableWorker: false });
|
const hls = new Hls({ enableWorker: false });
|
||||||
|
state.hlsInstance = hls;
|
||||||
|
|
||||||
hls.on(Hls.Events.ERROR, (event, data) => {
|
hls.on(Hls.Events.ERROR, (event, data) => {
|
||||||
if (data.fatal) {
|
if (data.fatal) {
|
||||||
@ -175,6 +178,7 @@ export function createVideoStateProvider(
|
|||||||
url: source.source,
|
url: source.source,
|
||||||
caption: null,
|
caption: null,
|
||||||
};
|
};
|
||||||
|
resetStateForSource(descriptor, state);
|
||||||
updateSource(descriptor, state);
|
updateSource(descriptor, state);
|
||||||
},
|
},
|
||||||
setCaption(id, url) {
|
setCaption(id, url) {
|
||||||
|
@ -4,6 +4,7 @@ import {
|
|||||||
MWStreamType,
|
MWStreamType,
|
||||||
} from "@/backend/helpers/streams";
|
} from "@/backend/helpers/streams";
|
||||||
import { MWMediaMeta } from "@/backend/metadata/types";
|
import { MWMediaMeta } from "@/backend/metadata/types";
|
||||||
|
import Hls from "hls.js";
|
||||||
import { VideoPlayerStateProvider } from "./providers/providerTypes";
|
import { VideoPlayerStateProvider } from "./providers/providerTypes";
|
||||||
|
|
||||||
export type VideoPlayerMeta = {
|
export type VideoPlayerMeta = {
|
||||||
@ -73,6 +74,7 @@ export type VideoPlayerState = {
|
|||||||
|
|
||||||
// backing fields
|
// backing fields
|
||||||
pausedWhenSeeking: boolean; // when seeking, used to store if paused when started to seek
|
pausedWhenSeeking: boolean; // when seeking, used to store if paused when started to seek
|
||||||
|
hlsInstance: null | Hls; // HLS video player instance storage
|
||||||
stateProvider: VideoPlayerStateProvider | null;
|
stateProvider: VideoPlayerStateProvider | null;
|
||||||
wrapperElement: HTMLDivElement | null;
|
wrapperElement: HTMLDivElement | null;
|
||||||
};
|
};
|
||||||
|
Loading…
x
Reference in New Issue
Block a user