import classNames from "classnames"; import { useMemo } from "react"; import { captionIsVisible, makeQueId, parseSubtitles, sanitize, } from "@/components/player/utils/captions"; import { Transition } from "@/components/Transition"; import { usePlayerStore } from "@/stores/player/store"; import { SubtitleStyling, useSubtitleStore } from "@/stores/subtitles"; export function CaptionCue({ text, styling, overrideCasing, }: { text?: string; styling: SubtitleStyling; overrideCasing: boolean; }) { const wordOverrides: Record = { i: "I", }; let textToUse = text; if (overrideCasing && text) { textToUse = text.slice(0, 1) + text.slice(1).toLowerCase(); } const textWithNewlines = (textToUse || "") .split(" ") .map((word) => wordOverrides[word] ?? word) .join(" ") .replaceAll(/ i'/g, " I'") .replaceAll(/\r?\n/g, "
"); // https://www.w3.org/TR/webvtt1/#dom-construction-rules // added a
for newlines const html = sanitize(textWithNewlines, { ALLOWED_TAGS: ["c", "b", "i", "u", "span", "ruby", "rt", "br"], ADD_TAGS: ["v", "lang"], ALLOWED_ATTR: ["title", "lang"], }); return (

); } export function SubtitleRenderer() { const videoTime = usePlayerStore((s) => s.progress.time); const srtData = usePlayerStore((s) => s.caption.selected?.srtData); const styling = useSubtitleStore((s) => s.styling); const overrideCasing = useSubtitleStore((s) => s.overrideCasing); const delay = useSubtitleStore((s) => s.delay); const parsedCaptions = useMemo( () => (srtData ? parseSubtitles(srtData) : []), [srtData] ); const visibileCaptions = useMemo( () => parsedCaptions.filter(({ start, end }) => captionIsVisible(start, end, delay, videoTime) ), [parsedCaptions, videoTime, delay] ); return (
{visibileCaptions.map(({ start, end, content }, i) => ( ))}
); } export function SubtitleView(props: { controlsShown: boolean }) { const caption = usePlayerStore((s) => s.caption.selected); const captionAsTrack = usePlayerStore((s) => s.caption.asTrack); if (captionAsTrack || !caption) return null; return (
); }