8f8eef53c9
- Optimize imported scene preview images (smart WebP/JPEG/PNG, preserve alpha, keep pixel size) - Update converter to re-encode existing image assets with same algorithm - Improve import/export progress overlay and reduce presentation slide stutter Made-with: Cursor
145 lines
4.6 KiB
TypeScript
145 lines
4.6 KiB
TypeScript
import React, { useEffect, useRef, useState } from 'react';
|
|
|
|
import { computeTimeSec } from '../../main/video/videoPlaybackStore';
|
|
import type { SessionState } from '../../shared/ipc/contracts';
|
|
|
|
import { PixiEffectsOverlay } from './effects/PxiEffectsOverlay';
|
|
import { useEffectsState } from './effects/useEffectsState';
|
|
import styles from './PresentationView.module.css';
|
|
import { RotatedImage } from './RotatedImage';
|
|
import { useAssetUrl } from './useAssetImageUrl';
|
|
import { useVideoPlaybackState } from './video/useVideoPlaybackState';
|
|
|
|
export type PresentationViewProps = {
|
|
session: SessionState | null;
|
|
/** Если true — показываем укороченный заголовок/оверлей для предпросмотра. */
|
|
compact?: boolean;
|
|
showTitle?: boolean;
|
|
showEffects?: boolean;
|
|
};
|
|
|
|
export function PresentationView({
|
|
session,
|
|
compact = false,
|
|
showTitle = true,
|
|
showEffects = true,
|
|
}: PresentationViewProps) {
|
|
const [fxState] = useEffectsState();
|
|
const [vp] = useVideoPlaybackState();
|
|
const videoElRef = useRef<HTMLVideoElement | null>(null);
|
|
const [contentRect, setContentRect] = React.useState<{ x: number; y: number; w: number; h: number } | null>(
|
|
null,
|
|
);
|
|
const scene =
|
|
session?.project && session.currentSceneId ? session.project.scenes[session.currentSceneId] : undefined;
|
|
const originalUrl = useAssetUrl(scene?.previewAssetId ?? null);
|
|
const thumbUrl = useAssetUrl(scene?.previewThumbAssetId ?? null);
|
|
const [shownImageUrl, setShownImageUrl] = useState<string | null>(null);
|
|
const rot = scene?.previewRotationDeg ?? 0;
|
|
|
|
useEffect(() => {
|
|
if (!scene) {
|
|
setShownImageUrl(null);
|
|
return;
|
|
}
|
|
if (scene.previewAssetType !== 'image') {
|
|
setShownImageUrl(null);
|
|
return;
|
|
}
|
|
// Show thumbnail instantly (if exists) to avoid stutter on slide switch, then swap to original when loaded.
|
|
setShownImageUrl(thumbUrl ?? originalUrl);
|
|
if (!thumbUrl || !originalUrl || thumbUrl === originalUrl) return;
|
|
let cancelled = false;
|
|
const img = new Image();
|
|
img.decoding = 'async';
|
|
img.onload = () => {
|
|
if (cancelled) return;
|
|
setShownImageUrl(originalUrl);
|
|
};
|
|
img.onerror = () => {
|
|
// keep thumbnail
|
|
};
|
|
img.src = originalUrl;
|
|
return () => {
|
|
cancelled = true;
|
|
};
|
|
}, [originalUrl, scene, thumbUrl]);
|
|
|
|
useEffect(() => {
|
|
const el = videoElRef.current;
|
|
if (!el) return;
|
|
if (!vp) return;
|
|
if (!scene?.previewAssetId) return;
|
|
if (scene.previewAssetType !== 'video') return;
|
|
if (vp.targetAssetId !== scene.previewAssetId) return;
|
|
|
|
el.playbackRate = vp.playbackRate;
|
|
const desired = computeTimeSec(vp, vp.serverNowMs);
|
|
if (Number.isFinite(desired) && Math.abs(el.currentTime - desired) > 0.35) {
|
|
el.currentTime = Math.max(0, desired);
|
|
}
|
|
if (vp.playing) {
|
|
void el.play().catch(() => undefined);
|
|
} else {
|
|
el.pause();
|
|
}
|
|
// eslint-disable-next-line react-hooks/exhaustive-deps -- avoid reruns on 500ms heartbeats (serverNowMs-only updates)
|
|
}, [
|
|
scene?.previewAssetId,
|
|
scene?.previewAssetType,
|
|
originalUrl,
|
|
vp?.revision,
|
|
vp?.targetAssetId,
|
|
vp?.playing,
|
|
vp?.playbackRate,
|
|
]);
|
|
|
|
return (
|
|
<div className={styles.root}>
|
|
{shownImageUrl && scene?.previewAssetType === 'image' ? (
|
|
<div className={styles.fill}>
|
|
<RotatedImage
|
|
url={shownImageUrl}
|
|
rotationDeg={rot}
|
|
mode="contain"
|
|
onContentRectChange={setContentRect}
|
|
/>
|
|
</div>
|
|
) : originalUrl && scene?.previewAssetType === 'video' ? (
|
|
<video
|
|
ref={videoElRef}
|
|
className={styles.video}
|
|
src={originalUrl}
|
|
muted
|
|
playsInline
|
|
loop={false}
|
|
preload="auto"
|
|
onError={() => {
|
|
// noop: status surfaced in control app; keep presentation clean
|
|
}}
|
|
/>
|
|
) : (
|
|
<div className={styles.placeholderBg} />
|
|
)}
|
|
<div className={styles.vignette} />
|
|
{showEffects && scene?.previewAssetType !== 'video' ? (
|
|
<PixiEffectsOverlay
|
|
state={fxState}
|
|
viewport={
|
|
contentRect
|
|
? { x: contentRect.x, y: contentRect.y, w: contentRect.w, h: contentRect.h }
|
|
: undefined
|
|
}
|
|
/>
|
|
) : null}
|
|
{showTitle ? (
|
|
<div className={styles.titleWrap}>
|
|
<div className={compact ? styles.titleCompact : styles.titleFull}>
|
|
{scene?.title ?? 'Выберите сцену в редакторе'}
|
|
</div>
|
|
</div>
|
|
) : null}
|
|
</div>
|
|
);
|
|
}
|