From e4cf493f9171614d2e7a9f5ffd0c2d31620da084 Mon Sep 17 00:00:00 2001 From: Dennis Benz Date: Fri, 26 Apr 2024 11:13:28 +0200 Subject: [PATCH] Fix jump controls Allow to jump when video player is playing. If current time is in the middle of the active segment, jump back to the beginning of this segment. --- src/main/SubtitleVideoArea.tsx | 12 ++++++++- src/main/VideoPlayers.tsx | 12 +++++++++ src/redux/videoSlice.ts | 46 +++++++++++++++++++++++++++++++--- 3 files changed, 65 insertions(+), 5 deletions(-) diff --git a/src/main/SubtitleVideoArea.tsx b/src/main/SubtitleVideoArea.tsx index 846139de3..63c0ce010 100644 --- a/src/main/SubtitleVideoArea.tsx +++ b/src/main/SubtitleVideoArea.tsx @@ -17,7 +17,15 @@ import { setIsPlayPreview, setCurrentlyAtAndTriggerPreview, } from "../redux/subtitleSlice"; -import { selectIsMuted, selectVideos, selectVolume, setIsMuted, setVolume } from "../redux/videoSlice"; +import { + selectIsMuted, + selectVideos, + selectVolume, + selectJumpTriggered, + setIsMuted, + setVolume, + setJumpTriggered, +} from "../redux/videoSlice"; import { Flavor } from "../types"; import { settings } from "../config"; import { useTranslation } from "react-i18next"; @@ -128,11 +136,13 @@ const SubtitleVideoArea: React.FC = () => { selectCurrentlyAtInSeconds={selectCurrentlyAtInSeconds} selectPreviewTriggered={selectPreviewTriggered} selectClickTriggered={selectClickTriggered} + selectJumpTriggered={selectJumpTriggered} selectAspectRatio={selectAspectRatio} setIsPlaying={setIsPlaying} selectVolume={selectVolume} setPreviewTriggered={setPreviewTriggered} setClickTriggered={setClickTriggered} + setJumpTriggered={setJumpTriggered} setCurrentlyAt={setCurrentlyAtAndTriggerPreview} setAspectRatio={setAspectRatio} /> diff --git a/src/main/VideoPlayers.tsx b/src/main/VideoPlayers.tsx index 5af37414b..b0c4262b9 100644 --- a/src/main/VideoPlayers.tsx +++ b/src/main/VideoPlayers.tsx @@ -18,6 +18,8 @@ import { selectAspectRatio, setClickTriggered, selectClickTriggered, + setJumpTriggered, + selectJumpTriggered, setCurrentlyAt, } from "../redux/videoSlice"; @@ -80,10 +82,12 @@ const VideoPlayers: React.FC<{ selectCurrentlyAtInSeconds={selectCurrentlyAtInSeconds} selectPreviewTriggered={selectPreviewTriggered} selectClickTriggered={selectClickTriggered} + selectJumpTriggered={selectJumpTriggered} selectAspectRatio={selectAspectRatio} setIsPlaying={setIsPlaying} setPreviewTriggered={setPreviewTriggered} setClickTriggered={setClickTriggered} + setJumpTriggered={setJumpTriggered} setCurrentlyAt={setCurrentlyAt} setAspectRatio={setAspectRatio} ref={el => { @@ -119,10 +123,12 @@ interface VideoPlayerProps { selectCurrentlyAtInSeconds: (state: RootState) => number, selectPreviewTriggered: (state: RootState) => boolean, selectClickTriggered: (state: RootState) => boolean, + selectJumpTriggered: (state: RootState) => boolean, selectAspectRatio: (state: RootState) => number, setIsPlaying: ActionCreatorWithPayload, setPreviewTriggered: ActionCreatorWithPayload, setClickTriggered: ActionCreatorWithPayload, + setJumpTriggered: ActionCreatorWithPayload, setCurrentlyAt: ActionCreatorWithPayload | AsyncThunk, setAspectRatio: ActionCreatorWithPayload<{ dataKey: number; } & { width: number, height: number; }, string>, } @@ -147,6 +153,7 @@ export const VideoPlayer = React.forwardRef { diff --git a/src/redux/videoSlice.ts b/src/redux/videoSlice.ts index f244f4916..db04e6ac8 100644 --- a/src/redux/videoSlice.ts +++ b/src/redux/videoSlice.ts @@ -12,6 +12,7 @@ export interface video { volume: number, // Video playback volume previewTriggered: boolean, // Basically acts as a callback for the video players. clickTriggered: boolean, // Another video player callback + jumpTriggered: boolean, // Another video player callback currentlyAt: number, // Position in the video in milliseconds segments: Segment[], tracks: Track[], @@ -54,6 +55,7 @@ export const initialState: video & httpRequestState = { selectedWorkflowId: "", previewTriggered: false, clickTriggered: false, + jumpTriggered: false, aspectRatios: [], hasChanges: false, waveformImages: [], @@ -136,6 +138,9 @@ const videoSlice = createSlice({ setClickTriggered: (state, action: PayloadAction) => { state.clickTriggered = action.payload; }, + setJumpTriggered: (state, action) => { + state.jumpTriggered = action.payload; + }, setCurrentlyAt: (state, action: PayloadAction) => { updateCurrentlyAt(state, action.payload); }, @@ -145,12 +150,18 @@ const videoSlice = createSlice({ jumpToPreviousSegment: state => { let previousSegmentIndex = state.activeSegmentIndex - 1; + // Jump to start of active segment if current time is in interval [start + 3s, end) + if (state.currentlyAt >= state.segments[state.activeSegmentIndex].start + 3000) { + previousSegmentIndex = state.activeSegmentIndex; + } + if (state.activeSegmentIndex == 0) { // Jump to start of first segment previousSegmentIndex = state.activeSegmentIndex; } updateCurrentlyAt(state, state.segments[previousSegmentIndex].start); + state.jumpTriggered = true; }, jumpToNextSegment: state => { let nextSegmentIndex = state.activeSegmentIndex + 1; @@ -161,6 +172,7 @@ const videoSlice = createSlice({ } updateCurrentlyAt(state, state.segments[nextSegmentIndex].start); + state.jumpTriggered = true; }, addSegment: (state, action: PayloadAction) => { state.segments.push(action.payload); @@ -296,6 +308,7 @@ const videoSlice = createSlice({ selectVolume: state => state.volume, selectPreviewTriggered: state => state.previewTriggered, selectClickTriggered: state => state.clickTriggered, + selectJumpTriggered: state => state.jumpTriggered, selectCurrentlyAt: state => state.currentlyAt, selectCurrentlyAtInSeconds: state => state.currentlyAt / 1000, selectSegments: state => state.segments, @@ -441,10 +454,34 @@ const setThumbnailHelper = (state: video, id: Track["id"], uri: Track["thumbnail } }; -export const { setTrackEnabled, setIsPlaying, setIsPlayPreview, setIsMuted, setVolume, setCurrentlyAt, - setCurrentlyAtInSeconds, addSegment, setAspectRatio, setHasChanges, setWaveformImages, setThumbnails, setThumbnail, - removeThumbnail, setLock, cut, markAsDeletedOrAlive, setSelectedWorkflowIndex, mergeLeft, mergeRight, mergeAll, - setPreviewTriggered, setClickTriggered, jumpToPreviousSegment, jumpToNextSegment } = videoSlice.actions; +export const { + setTrackEnabled, + setIsPlaying, + setIsPlayPreview, + setIsMuted, + setVolume, + setCurrentlyAt, + setCurrentlyAtInSeconds, + addSegment, + setAspectRatio, + setHasChanges, + setWaveformImages, + setThumbnails, + setThumbnail, + removeThumbnail, + setLock, + cut, + markAsDeletedOrAlive, + setSelectedWorkflowIndex, + mergeLeft, + mergeRight, + mergeAll, + setPreviewTriggered, + setClickTriggered, + setJumpTriggered, + jumpToPreviousSegment, + jumpToNextSegment +} = videoSlice.actions; export const selectVideos = createSelector( [(state: { videoState: { tracks: video["tracks"]; }; }) => state.videoState.tracks], @@ -459,6 +496,7 @@ export const { selectVolume, selectPreviewTriggered, selectClickTriggered, + selectJumpTriggered, selectCurrentlyAt, selectCurrentlyAtInSeconds, selectSegments,