From 5297044ec112084ce32a54164fc70be6b905f118 Mon Sep 17 00:00:00 2001 From: Benjamin Kane Date: Wed, 2 Oct 2024 15:29:07 -0400 Subject: [PATCH 01/11] organizing --- .../core/src/components/Modal/ModalLooker.tsx | 8 +- .../core/src/components/Modal/VideoLooker.tsx | 82 +++++++++++++++++++ .../core/src/components/Modal/useLooker.ts | 70 ++++++++++++++++ app/packages/looker/package.json | 3 + .../looker/src/elements/common/looker.ts | 8 +- app/packages/looker/src/elements/video.ts | 10 ++- app/packages/looker/src/index.ts | 2 +- .../looker/src/lookers/imavid/index.ts | 2 + app/packages/looker/src/lookers/video.ts | 4 + app/packages/looker/src/state.ts | 2 +- .../playback/src/views/TimelineExamples.tsx | 43 ---------- .../state/src/hooks/useCreateLooker.ts | 2 +- app/yarn.lock | 2 + 13 files changed, 188 insertions(+), 50 deletions(-) create mode 100644 app/packages/core/src/components/Modal/VideoLooker.tsx create mode 100644 app/packages/core/src/components/Modal/useLooker.ts diff --git a/app/packages/core/src/components/Modal/ModalLooker.tsx b/app/packages/core/src/components/Modal/ModalLooker.tsx index c18eb5e048..58e5ff7db8 100644 --- a/app/packages/core/src/components/Modal/ModalLooker.tsx +++ b/app/packages/core/src/components/Modal/ModalLooker.tsx @@ -7,8 +7,9 @@ import React, { useEffect, useMemo, useRef, useState } from "react"; import { useErrorHandler } from "react-error-boundary"; import { useRecoilCallback, useRecoilValue, useSetRecoilState } from "recoil"; import { v4 as uuid } from "uuid"; -import { useModalContext } from "./hooks"; import { ImaVidLookerReact } from "./ImaVidLooker"; +import { VideoLookerReact } from "./VideoLooker"; +import { useModalContext } from "./hooks"; export const useLookerOptionsUpdate = () => { return useRecoilCallback( @@ -197,11 +198,16 @@ export const ModalLooker = React.memo( const shouldRenderImavid = useRecoilValue( fos.shouldRenderImaVidLooker(true) ); + const video = useRecoilValue(fos.isVideoDataset); if (shouldRenderImavid) { return ; } + if (video) { + return ; + } + return ; } ); diff --git a/app/packages/core/src/components/Modal/VideoLooker.tsx b/app/packages/core/src/components/Modal/VideoLooker.tsx new file mode 100644 index 0000000000..2816f927b5 --- /dev/null +++ b/app/packages/core/src/components/Modal/VideoLooker.tsx @@ -0,0 +1,82 @@ +import { useTheme } from "@fiftyone/components"; +import type { VideoLooker } from "@fiftyone/looker"; +import { getFrameNumber } from "@fiftyone/looker"; +import { + useCreateTimeline, + useDefaultTimelineNameImperative, + useTimeline, +} from "@fiftyone/playback"; +import * as fos from "@fiftyone/state"; +import React, { useEffect, useMemo, useState } from "react"; +import "./Tmp"; +import useLooker from "./useLooker"; + +interface VideoLookerReactProps { + sample: fos.ModalSample; +} + +export const VideoLookerReact = (props: VideoLookerReactProps) => { + const theme = useTheme(); + const { id, looker, sample } = useLooker(props); + const [totalFrames, setTotalFrames] = useState(); + const frameRate = useMemo(() => { + return sample.frameRate; + }, [sample]); + + useEffect(() => { + const load = () => { + const duration = looker.getVideo().duration; + setTotalFrames(getFrameNumber(duration, duration, frameRate)); + looker.removeEventListener("load", load); + }; + looker.addEventListener("load", load); + }, [frameRate, looker]); + + return ( + <> +
+ {totalFrames !== undefined && ( + + )} + + ); +}; + +const TimelineController = ({ + looker, + totalFrames, +}: { + looker: VideoLooker; + totalFrames: number; +}) => { + const { getName } = useDefaultTimelineNameImperative(); + const timelineName = React.useMemo(() => getName(), [getName]); + + useCreateTimeline({ + name: timelineName, + config: totalFrames + ? { + totalFrames, + loop: true, + } + : undefined, + optOutOfAnimation: true, + }); + + const { playHeadState, config, pause, play, setSpeed } = + useTimeline(timelineName); + + fos.useEventHandler(looker, "pause", pause); + fos.useEventHandler(looker, "play", play); + + return null; +}; diff --git a/app/packages/core/src/components/Modal/useLooker.ts b/app/packages/core/src/components/Modal/useLooker.ts new file mode 100644 index 0000000000..eb6e506b03 --- /dev/null +++ b/app/packages/core/src/components/Modal/useLooker.ts @@ -0,0 +1,70 @@ +import * as fos from "@fiftyone/state"; +import React, { useEffect, useRef, useState } from "react"; +import { useErrorHandler } from "react-error-boundary"; +import { useRecoilValue } from "recoil"; +import { v4 as uuid } from "uuid"; +import { + useClearSelectedLabels, + useLookerOptionsUpdate, + useShowOverlays, +} from "./ModalLooker"; + +function useLooker({ + sample, +}: { + sample: fos.ModalSample; +}) { + const [id] = useState(() => uuid()); + const initialRef = useRef(true); + const ref = useRef(null); + const [reset, setReset] = useState(false); + const lookerOptions = fos.useLookerOptions(true); + const createLooker = fos.useCreateLooker(true, false, lookerOptions); + const selectedMediaField = useRecoilValue(fos.selectedMediaField(true)); + const colorScheme = useRecoilValue(fos.colorScheme); + const looker = React.useMemo(() => { + reset; + selectedMediaField; + return createLooker.current(sample); + }, [createLooker, reset, sample, selectedMediaField]) as L; + const handleError = useErrorHandler(); + const updateLookerOptions = useLookerOptionsUpdate(); + + fos.useEventHandler(looker, "clear", useClearSelectedLabels()); + fos.useEventHandler(looker, "error", (event) => handleError(event.detail)); + fos.useEventHandler(looker, "options", (e) => updateLookerOptions(e.detail)); + fos.useEventHandler(looker, "reset", () => setReset((c) => !c)); + fos.useEventHandler(looker, "select", fos.useOnSelectLabel()); + fos.useEventHandler(looker, "showOverlays", useShowOverlays()); + + useEffect(() => { + !initialRef.current && looker.updateOptions(lookerOptions); + }, [looker, lookerOptions]); + + useEffect(() => { + colorScheme; + !initialRef.current && looker.updateSample(sample); + }, [colorScheme, looker, sample]); + + useEffect(() => { + initialRef.current = false; + }, []); + + useEffect(() => { + ref.current?.dispatchEvent( + new CustomEvent("looker-attached", { bubbles: true }) + ); + }, []); + + useEffect(() => { + return () => looker?.destroy(); + }, [looker]); + + useEffect(() => { + looker.attach(id); + }, [looker, id]); + + return { id, looker, ref, sample, updateLookerOptions }; +} + +export default useLooker; diff --git a/app/packages/looker/package.json b/app/packages/looker/package.json index b47bb84fd6..8ac5a0b945 100644 --- a/app/packages/looker/package.json +++ b/app/packages/looker/package.json @@ -39,5 +39,8 @@ "typescript": "^4.7.4", "typescript-plugin-css-modules": "^5.1.0", "vite": "^5.2.14" + }, + "peerDependencies": { + "jotai": "*" } } diff --git a/app/packages/looker/src/elements/common/looker.ts b/app/packages/looker/src/elements/common/looker.ts index 8910d90cd0..ad2f256438 100644 --- a/app/packages/looker/src/elements/common/looker.ts +++ b/app/packages/looker/src/elements/common/looker.ts @@ -24,7 +24,11 @@ export class LookerElement extends BaseElement< const e = event as KeyboardEvent; update((state) => { - const { SHORTCUTS, error, shouldHandleKeyEvents } = state; + const { + SHORTCUTS, + error, + config: { shouldHandleKeyEvents }, + } = state; if (!error && e.key in SHORTCUTS) { const matchedControl = SHORTCUTS[e.key] as Control; const enabled = @@ -43,7 +47,7 @@ export class LookerElement extends BaseElement< } const e = event as KeyboardEvent; - update(({ SHORTCUTS, error, shouldHandleKeyEvents }) => { + update(({ SHORTCUTS, error, config: { shouldHandleKeyEvents } }) => { if (!error && e.key in SHORTCUTS) { const matchedControl = SHORTCUTS[e.key] as Control; diff --git a/app/packages/looker/src/elements/video.ts b/app/packages/looker/src/elements/video.ts index 3ea8ab6466..d0f990f8bc 100644 --- a/app/packages/looker/src/elements/video.ts +++ b/app/packages/looker/src/elements/video.ts @@ -2,6 +2,8 @@ * Copyright 2017-2024, Voxel51, Inc. */ +import { setFrameNumberAtom } from "@fiftyone/playback"; +import { getDefaultStore } from "jotai"; import { playbackRate, volume as volumeIcon, volumeMuted } from "../icons"; import lockIcon from "../icons/lock.svg"; import lockOpenIcon from "../icons/lockOpen.svg"; @@ -457,7 +459,7 @@ export class VideoElement extends BaseElement { duration, playing, options: { loop }, - config: { frameRate, support }, + config: { thumbnail, frameRate, support, sampleId }, lockedToSupport, }) => { let newFrameNumber = getFrameNumber(time, duration, frameRate); @@ -473,6 +475,12 @@ export class VideoElement extends BaseElement { this.frameNumber = newFrameNumber; } + !thumbnail && + getDefaultStore().set(setFrameNumberAtom, { + name: `timeline-${sampleId}`, + newFrameNumber, + }); + return { frameNumber: newFrameNumber, playing, diff --git a/app/packages/looker/src/index.ts b/app/packages/looker/src/index.ts index 333140735d..889ec3ff73 100644 --- a/app/packages/looker/src/index.ts +++ b/app/packages/looker/src/index.ts @@ -3,7 +3,7 @@ */ export { createColorGenerator, getRGB } from "@fiftyone/utilities"; -export { freeVideos } from "./elements/util"; +export { freeVideos, getFrameNumber } from "./elements/util"; export * from "./lookers"; export type { PointInfo } from "./overlays"; export type { diff --git a/app/packages/looker/src/lookers/imavid/index.ts b/app/packages/looker/src/lookers/imavid/index.ts index ae3d60d1d2..cd302395e4 100644 --- a/app/packages/looker/src/lookers/imavid/index.ts +++ b/app/packages/looker/src/lookers/imavid/index.ts @@ -15,6 +15,8 @@ import { IMAVID_PLAYBACK_RATE_LOCAL_STORAGE_KEY, } from "./constants"; +export { BUFFERING_PAUSE_TIMEOUT } from "./constants"; + const DEFAULT_PAN = 0; const DEFAULT_SCALE = 1; const FIRST_FRAME = 1; diff --git a/app/packages/looker/src/lookers/video.ts b/app/packages/looker/src/lookers/video.ts index e07771fe7e..03f71f36b5 100644 --- a/app/packages/looker/src/lookers/video.ts +++ b/app/packages/looker/src/lookers/video.ts @@ -546,6 +546,10 @@ export class VideoLooker extends AbstractLooker { this.setReader(); } + getVideo() { + return this.lookerElement.children[0].element as HTMLVideoElement; + } + private hasFrame(frameNumber: number) { return ( this.frames.has(frameNumber) && diff --git a/app/packages/looker/src/state.ts b/app/packages/looker/src/state.ts index 64dfe21278..32939b34bb 100644 --- a/app/packages/looker/src/state.ts +++ b/app/packages/looker/src/state.ts @@ -208,6 +208,7 @@ export interface BaseConfig { id: string; name: string; }; + shouldHandleKeyEvents?: boolean; } export interface FrameConfig extends BaseConfig { @@ -297,7 +298,6 @@ export interface BaseState { showOptions: boolean; config: BaseConfig; options: BaseOptions; - shouldHandleKeyEvents: boolean; scale: number; pan: Coordinates; panning: boolean; diff --git a/app/packages/playback/src/views/TimelineExamples.tsx b/app/packages/playback/src/views/TimelineExamples.tsx index 9f3fb12178..3e85387191 100644 --- a/app/packages/playback/src/views/TimelineExamples.tsx +++ b/app/packages/playback/src/views/TimelineExamples.tsx @@ -106,49 +106,6 @@ export const TimelineCreator = () => { ); }; -export const TimelineSubscriber1 = () => { - const { getName } = useDefaultTimelineNameImperative(); - const timelineName = React.useMemo(() => getName(), [getName]); - - const [myLocalFrameNumber, setMyLocalFrameNumber] = - React.useState(DEFAULT_FRAME_NUMBER); - - const loadRange = React.useCallback(async (range: BufferRange) => { - // no-op for now, but maybe for testing, i can resolve a promise inside settimeout - }, []); - - const myRenderFrame = React.useCallback((frameNumber: number) => { - setMyLocalFrameNumber(frameNumber); - }, []); - - const { subscribe, isTimelineInitialized, getFrameNumber } = useTimeline(); - - React.useEffect(() => { - if (!isTimelineInitialized) { - return; - } - - subscribe({ - id: `sub1`, - loadRange, - renderFrame: myRenderFrame, - }); - }, [loadRange, myRenderFrame, subscribe, isTimelineInitialized]); - - if (!isTimelineInitialized) { - return
loading...
; - } - - return ( - <> -
- Subscriber 1 frame number {timelineName}: {myLocalFrameNumber} -
- - - ); -}; - export const TimelineSubscriber2 = () => { const { getName } = useDefaultTimelineNameImperative(); const timelineName = React.useMemo(() => getName(), [getName]); diff --git a/app/packages/state/src/hooks/useCreateLooker.ts b/app/packages/state/src/hooks/useCreateLooker.ts index 06e04eb652..adb854be48 100644 --- a/app/packages/state/src/hooks/useCreateLooker.ts +++ b/app/packages/state/src/hooks/useCreateLooker.ts @@ -9,7 +9,7 @@ import { } from "@fiftyone/looker"; import { ImaVidFramesController } from "@fiftyone/looker/src/lookers/imavid/controller"; import { ImaVidFramesControllerStore } from "@fiftyone/looker/src/lookers/imavid/store"; -import { BaseState, ImaVidConfig } from "@fiftyone/looker/src/state"; +import type { BaseState, ImaVidConfig } from "@fiftyone/looker/src/state"; import { EMBEDDED_DOCUMENT_FIELD, LIST_FIELD, diff --git a/app/yarn.lock b/app/yarn.lock index dc9917db4b..86e96c90f8 100644 --- a/app/yarn.lock +++ b/app/yarn.lock @@ -1885,6 +1885,8 @@ __metadata: typescript-plugin-css-modules: ^5.1.0 uuid: ^8.3.2 vite: ^5.2.14 + peerDependencies: + jotai: "*" languageName: unknown linkType: soft From dab4c0dca4325d07ada919246cf01f92d59a9ad1 Mon Sep 17 00:00:00 2001 From: Benjamin Kane Date: Wed, 2 Oct 2024 15:31:34 -0400 Subject: [PATCH 02/11] rm Tmp --- app/packages/core/src/components/Modal/VideoLooker.tsx | 1 - 1 file changed, 1 deletion(-) diff --git a/app/packages/core/src/components/Modal/VideoLooker.tsx b/app/packages/core/src/components/Modal/VideoLooker.tsx index 2816f927b5..80c81ccc59 100644 --- a/app/packages/core/src/components/Modal/VideoLooker.tsx +++ b/app/packages/core/src/components/Modal/VideoLooker.tsx @@ -8,7 +8,6 @@ import { } from "@fiftyone/playback"; import * as fos from "@fiftyone/state"; import React, { useEffect, useMemo, useState } from "react"; -import "./Tmp"; import useLooker from "./useLooker"; interface VideoLookerReactProps { From caa57bf5ed4209689e59a380b84cd9509868d5fd Mon Sep 17 00:00:00 2001 From: Benjamin Kane Date: Wed, 2 Oct 2024 17:18:38 -0400 Subject: [PATCH 03/11] cleanup --- .../src/components/Modal/ImaVidLooker.tsx | 8 +- .../core/src/components/Modal/ModalLooker.tsx | 195 +++--------------- .../core/src/components/Modal/VideoLooker.tsx | 5 +- .../core/src/components/Modal/hooks.ts | 21 ++ .../Modal/{useLooker.ts => use-looker.ts} | 54 ++++- .../core/src/components/Modal/utils.ts | 7 + app/packages/looker/src/elements/base.ts | 3 +- .../looker/src/elements/common/looker.ts | 11 +- app/packages/looker/src/elements/video.ts | 10 +- app/packages/looker/src/lookers/video.ts | 8 + app/packages/looker/src/state.ts | 2 +- .../state/src/hooks/useCreateLooker.ts | 1 + app/packages/state/src/recoil/modal.ts | 12 +- 13 files changed, 133 insertions(+), 204 deletions(-) rename app/packages/core/src/components/Modal/{useLooker.ts => use-looker.ts} (64%) create mode 100644 app/packages/core/src/components/Modal/utils.ts diff --git a/app/packages/core/src/components/Modal/ImaVidLooker.tsx b/app/packages/core/src/components/Modal/ImaVidLooker.tsx index 6707202a8a..9797631977 100644 --- a/app/packages/core/src/components/Modal/ImaVidLooker.tsx +++ b/app/packages/core/src/components/Modal/ImaVidLooker.tsx @@ -17,13 +17,9 @@ import React, { import { useErrorHandler } from "react-error-boundary"; import { useRecoilValue, useSetRecoilState } from "recoil"; import { v4 as uuid } from "uuid"; +import { useClearSelectedLabels, useShowOverlays } from "./ModalLooker"; import { useInitializeImaVidSubscriptions, useModalContext } from "./hooks"; -import { - shortcutToHelpItems, - useClearSelectedLabels, - useLookerOptionsUpdate, - useShowOverlays, -} from "./ModalLooker"; +import { shortcutToHelpItems } from "./utils"; interface ImaVidLookerReactProps { sample: fos.ModalSample; diff --git a/app/packages/core/src/components/Modal/ModalLooker.tsx b/app/packages/core/src/components/Modal/ModalLooker.tsx index 58e5ff7db8..f9155d49f8 100644 --- a/app/packages/core/src/components/Modal/ModalLooker.tsx +++ b/app/packages/core/src/components/Modal/ModalLooker.tsx @@ -1,36 +1,12 @@ import { useTheme } from "@fiftyone/components"; -import { AbstractLooker } from "@fiftyone/looker"; -import { BaseState } from "@fiftyone/looker/src/state"; +import type { ImageLooker } from "@fiftyone/looker"; import * as fos from "@fiftyone/state"; -import { useEventHandler, useOnSelectLabel } from "@fiftyone/state"; -import React, { useEffect, useMemo, useRef, useState } from "react"; -import { useErrorHandler } from "react-error-boundary"; +import React, { useEffect, useMemo } from "react"; import { useRecoilCallback, useRecoilValue, useSetRecoilState } from "recoil"; -import { v4 as uuid } from "uuid"; import { ImaVidLookerReact } from "./ImaVidLooker"; import { VideoLookerReact } from "./VideoLooker"; import { useModalContext } from "./hooks"; - -export const useLookerOptionsUpdate = () => { - return useRecoilCallback( - ({ snapshot, set }) => - async (update: object, updater?: (updated: {}) => void) => { - const currentOptions = await snapshot.getPromise( - fos.savedLookerOptions - ); - - const panels = await snapshot.getPromise(fos.lookerPanels); - const updated = { - ...currentOptions, - ...update, - showJSON: panels.json.isOpen, - showHelp: panels.help.isOpen, - }; - set(fos.savedLookerOptions, updated); - if (updater) updater(updated); - } - ); -}; +import useLooker from "./use-looker"; export const useShowOverlays = () => { return useRecoilCallback(({ set }) => async (event: CustomEvent) => { @@ -48,137 +24,40 @@ export const useClearSelectedLabels = () => { }; interface LookerProps { - sample?: fos.ModalSample; - onClick?: React.MouseEventHandler; + sample: fos.ModalSample; } -const ModalLookerNoTimeline = React.memo( - ({ sample: sampleDataWithExtraParams }: LookerProps) => { - const [id] = useState(() => uuid()); - const colorScheme = useRecoilValue(fos.colorScheme); - - const { sample } = sampleDataWithExtraParams; - - const theme = useTheme(); - const initialRef = useRef(true); - const lookerOptions = fos.useLookerOptions(true); - const [reset, setReset] = useState(false); - const selectedMediaField = useRecoilValue(fos.selectedMediaField(true)); - const setModalLooker = useSetRecoilState(fos.modalLooker); - - const createLooker = fos.useCreateLooker(true, false, { - ...lookerOptions, - }); - - const { setActiveLookerRef } = useModalContext(); - - const looker = React.useMemo( - () => createLooker.current(sampleDataWithExtraParams), - [reset, createLooker, selectedMediaField] - ) as AbstractLooker; - - useEffect(() => { - setModalLooker(looker); - }, [looker]); - - useEffect(() => { - if (looker) { - setActiveLookerRef(looker as fos.Lookers); - } - }, [looker]); - - useEffect(() => { - !initialRef.current && looker.updateOptions(lookerOptions); - }, [lookerOptions]); - - useEffect(() => { - !initialRef.current && looker.updateSample(sample); - }, [sample, colorScheme]); - - useEffect(() => { - return () => looker?.destroy(); - }, [looker]); - - const handleError = useErrorHandler(); +const ModalLookerNoTimeline = React.memo((props: LookerProps) => { + const { id, looker, ref, sample } = useLooker(props); + const theme = useTheme(); + const setModalLooker = useSetRecoilState(fos.modalLooker); - const updateLookerOptions = useLookerOptionsUpdate(); - useEventHandler(looker, "options", (e) => updateLookerOptions(e.detail)); - useEventHandler(looker, "showOverlays", useShowOverlays()); - useEventHandler(looker, "reset", () => { - setReset((c) => !c); - }); + const { setActiveLookerRef } = useModalContext(); - const jsonPanel = fos.useJSONPanel(); - const helpPanel = fos.useHelpPanel(); + useEffect(() => { + setModalLooker(looker); + }, [looker, setModalLooker]); - useEventHandler(looker, "select", useOnSelectLabel()); - useEventHandler(looker, "error", (event) => handleError(event.detail)); - useEventHandler( - looker, - "panels", - async ({ detail: { showJSON, showHelp, SHORTCUTS } }) => { - if (showJSON) { - jsonPanel[showJSON](sample); - } - if (showHelp) { - if (showHelp == "close") { - helpPanel.close(); - } else { - helpPanel[showHelp](shortcutToHelpItems(SHORTCUTS)); - } - } - - updateLookerOptions({}, (updatedOptions) => - looker.updateOptions(updatedOptions) - ); - } - ); - - useEffect(() => { - initialRef.current = false; - }, []); - - useEffect(() => { - looker.attach(id); - }, [looker, id]); - - useEventHandler(looker, "clear", useClearSelectedLabels()); - - const hoveredSample = useRecoilValue(fos.hoveredSample); - - useEffect(() => { - const hoveredSampleId = hoveredSample?._id; - looker.updater((state) => ({ - ...state, - shouldHandleKeyEvents: hoveredSampleId === sample._id, - options: { - ...state.options, - }, - })); - }, [hoveredSample, sample, looker]); - - const ref = useRef(null); - useEffect(() => { - ref.current?.dispatchEvent( - new CustomEvent(`looker-attached`, { bubbles: true }) - ); - }, [ref]); - - return ( -
- ); - } -); + useEffect(() => { + if (looker) { + setActiveLookerRef(looker as fos.Lookers); + } + }, [looker, setActiveLookerRef]); + + return ( +
+ ); +}); export const ModalLooker = React.memo( ({ sample: propsSampleData }: LookerProps) => { @@ -211,13 +90,3 @@ export const ModalLooker = React.memo( return ; } ); - -export function shortcutToHelpItems(SHORTCUTS) { - return Object.values( - Object.values(SHORTCUTS).reduce((acc, v) => { - acc[v.shortcut] = v; - - return acc; - }, {}) - ); -} diff --git a/app/packages/core/src/components/Modal/VideoLooker.tsx b/app/packages/core/src/components/Modal/VideoLooker.tsx index 80c81ccc59..9c8f9b0cd2 100644 --- a/app/packages/core/src/components/Modal/VideoLooker.tsx +++ b/app/packages/core/src/components/Modal/VideoLooker.tsx @@ -8,7 +8,7 @@ import { } from "@fiftyone/playback"; import * as fos from "@fiftyone/state"; import React, { useEffect, useMemo, useState } from "react"; -import useLooker from "./useLooker"; +import useLooker from "./use-looker"; interface VideoLookerReactProps { sample: fos.ModalSample; @@ -71,8 +71,7 @@ const TimelineController = ({ optOutOfAnimation: true, }); - const { playHeadState, config, pause, play, setSpeed } = - useTimeline(timelineName); + const { pause, play } = useTimeline(timelineName); fos.useEventHandler(looker, "pause", pause); fos.useEventHandler(looker, "play", play); diff --git a/app/packages/core/src/components/Modal/hooks.ts b/app/packages/core/src/components/Modal/hooks.ts index 700955dd47..2cdb6d6310 100644 --- a/app/packages/core/src/components/Modal/hooks.ts +++ b/app/packages/core/src/components/Modal/hooks.ts @@ -19,6 +19,27 @@ export const useLookerHelpers = () => { }; }; +export const useLookerOptionsUpdate = () => { + return useRecoilCallback( + ({ snapshot, set }) => + async (update: object, updater?: (updated: {}) => void) => { + const currentOptions = await snapshot.getPromise( + fos.savedLookerOptions + ); + + const panels = await snapshot.getPromise(fos.lookerPanels); + const updated = { + ...currentOptions, + ...update, + showJSON: panels.json.isOpen, + showHelp: panels.help.isOpen, + }; + set(fos.savedLookerOptions, updated); + if (updater) updater(updated); + } + ); +}; + export const useInitializeImaVidSubscriptions = () => { const subscribeToImaVidStateChanges = useRecoilCallback( ({ set }) => diff --git a/app/packages/core/src/components/Modal/useLooker.ts b/app/packages/core/src/components/Modal/use-looker.ts similarity index 64% rename from app/packages/core/src/components/Modal/useLooker.ts rename to app/packages/core/src/components/Modal/use-looker.ts index eb6e506b03..65366cbfaa 100644 --- a/app/packages/core/src/components/Modal/useLooker.ts +++ b/app/packages/core/src/components/Modal/use-looker.ts @@ -1,13 +1,20 @@ import * as fos from "@fiftyone/state"; import React, { useEffect, useRef, useState } from "react"; import { useErrorHandler } from "react-error-boundary"; -import { useRecoilValue } from "recoil"; +import { selector, useRecoilValue } from "recoil"; import { v4 as uuid } from "uuid"; -import { - useClearSelectedLabels, - useLookerOptionsUpdate, - useShowOverlays, -} from "./ModalLooker"; +import { useClearSelectedLabels, useShowOverlays } from "./ModalLooker"; +import { useLookerOptionsUpdate } from "./hooks"; +import { shortcutToHelpItems } from "./utils"; + +export const hoveredSampleId = selector({ + key: "hoveredSampleId", + get: ({ get }) => { + return get(fos.hoveredSample)?._id; + }, +}); + +const CLOSE = "close"; function useLooker({ sample, @@ -56,13 +63,44 @@ function useLooker({ ); }, []); + useEffect(() => { + looker.attach(id); + }, [looker, id]); + useEffect(() => { return () => looker?.destroy(); }, [looker]); + const jsonPanel = fos.useJSONPanel(); + const helpPanel = fos.useHelpPanel(); + + fos.useEventHandler( + looker, + "panels", + async ({ detail: { showJSON, showHelp, SHORTCUTS } }) => { + if (showJSON) { + jsonPanel[showJSON](sample); + } + if (showHelp) { + if (showHelp === CLOSE) { + helpPanel.close(); + } else { + helpPanel[showHelp](shortcutToHelpItems(SHORTCUTS)); + } + } + + updateLookerOptions({}, (updatedOptions) => + looker.updateOptions(updatedOptions) + ); + } + ); + const hoveredId = useRecoilValue(hoveredSampleId); + useEffect(() => { - looker.attach(id); - }, [looker, id]); + looker.updateOptions({ + shouldHandleKeyEvents: sample.sample._id === hoveredId, + }); + }, [hoveredId, sample, looker]); return { id, looker, ref, sample, updateLookerOptions }; } diff --git a/app/packages/core/src/components/Modal/utils.ts b/app/packages/core/src/components/Modal/utils.ts new file mode 100644 index 0000000000..2461570f9a --- /dev/null +++ b/app/packages/core/src/components/Modal/utils.ts @@ -0,0 +1,7 @@ +export function shortcutToHelpItems(SHORTCUTS) { + const result = {}; + for (const k of SHORTCUTS) { + result[SHORTCUTS[k].shortcut] = SHORTCUTS[k]; + } + return Object.values(result); +} diff --git a/app/packages/looker/src/elements/base.ts b/app/packages/looker/src/elements/base.ts index e872a800c5..5ce470ef87 100644 --- a/app/packages/looker/src/elements/base.ts +++ b/app/packages/looker/src/elements/base.ts @@ -64,8 +64,7 @@ export abstract class BaseElement< for (const [eventType, handler] of Object.entries(this.getEvents(config))) { this.events[eventType] = (event) => handler({ event, update, dispatchEvent }); - this.element && - this.element.addEventListener(eventType, this.events[eventType]); + this.element?.addEventListener(eventType, this.events[eventType]); } } applyChildren(children: BaseElement[]) { diff --git a/app/packages/looker/src/elements/common/looker.ts b/app/packages/looker/src/elements/common/looker.ts index ad2f256438..95e6eda918 100644 --- a/app/packages/looker/src/elements/common/looker.ts +++ b/app/packages/looker/src/elements/common/looker.ts @@ -3,8 +3,10 @@ */ import { SELECTION_TEXT } from "../../constants"; -import { BaseState, Control, ControlEventKeyType } from "../../state"; -import { BaseElement, Events } from "../base"; +import type { BaseState, Control } from "../../state"; +import { ControlEventKeyType } from "../../state"; +import type { Events } from "../base"; +import { BaseElement } from "../base"; import { looker, lookerError, lookerHighlight } from "./looker.module.css"; @@ -27,7 +29,8 @@ export class LookerElement extends BaseElement< const { SHORTCUTS, error, - config: { shouldHandleKeyEvents }, + + options: { shouldHandleKeyEvents }, } = state; if (!error && e.key in SHORTCUTS) { const matchedControl = SHORTCUTS[e.key] as Control; @@ -47,7 +50,7 @@ export class LookerElement extends BaseElement< } const e = event as KeyboardEvent; - update(({ SHORTCUTS, error, config: { shouldHandleKeyEvents } }) => { + update(({ SHORTCUTS, error, options: { shouldHandleKeyEvents } }) => { if (!error && e.key in SHORTCUTS) { const matchedControl = SHORTCUTS[e.key] as Control; diff --git a/app/packages/looker/src/elements/video.ts b/app/packages/looker/src/elements/video.ts index d0f990f8bc..3ea8ab6466 100644 --- a/app/packages/looker/src/elements/video.ts +++ b/app/packages/looker/src/elements/video.ts @@ -2,8 +2,6 @@ * Copyright 2017-2024, Voxel51, Inc. */ -import { setFrameNumberAtom } from "@fiftyone/playback"; -import { getDefaultStore } from "jotai"; import { playbackRate, volume as volumeIcon, volumeMuted } from "../icons"; import lockIcon from "../icons/lock.svg"; import lockOpenIcon from "../icons/lockOpen.svg"; @@ -459,7 +457,7 @@ export class VideoElement extends BaseElement { duration, playing, options: { loop }, - config: { thumbnail, frameRate, support, sampleId }, + config: { frameRate, support }, lockedToSupport, }) => { let newFrameNumber = getFrameNumber(time, duration, frameRate); @@ -475,12 +473,6 @@ export class VideoElement extends BaseElement { this.frameNumber = newFrameNumber; } - !thumbnail && - getDefaultStore().set(setFrameNumberAtom, { - name: `timeline-${sampleId}`, - newFrameNumber, - }); - return { frameNumber: newFrameNumber, playing, diff --git a/app/packages/looker/src/lookers/video.ts b/app/packages/looker/src/lookers/video.ts index 03f71f36b5..fa9a92af49 100644 --- a/app/packages/looker/src/lookers/video.ts +++ b/app/packages/looker/src/lookers/video.ts @@ -20,7 +20,9 @@ import { } from "../state"; import { addToBuffers, createWorker, removeFromBuffers } from "../util"; +import { setFrameNumberAtom } from "@fiftyone/playback"; import { Schema } from "@fiftyone/utilities"; +import { getDefaultStore } from "jotai"; import { LRUCache } from "lru-cache"; import { CHUNK_SIZE, MAX_FRAME_CACHE_SIZE_BYTES } from "../constants"; import { getFrameNumber } from "../elements/util"; @@ -525,6 +527,12 @@ export class VideoLooker extends AbstractLooker { this.state.setZoom = false; } + !this.state.config.thumbnail && + getDefaultStore().set(setFrameNumberAtom, { + name: `timeline-${this.state.config.sampleId}`, + newFrameNumber: this.state.frameNumber, + }); + return super.postProcess(); } diff --git a/app/packages/looker/src/state.ts b/app/packages/looker/src/state.ts index 32939b34bb..b4d1da5ad8 100644 --- a/app/packages/looker/src/state.ts +++ b/app/packages/looker/src/state.ts @@ -175,6 +175,7 @@ interface BaseOptions { smoothMasks: boolean; zoomPad: number; selected: boolean; + shouldHandleKeyEvents?: boolean; inSelectionMode: boolean; timeZone: string; mimetype: string; @@ -208,7 +209,6 @@ export interface BaseConfig { id: string; name: string; }; - shouldHandleKeyEvents?: boolean; } export interface FrameConfig extends BaseConfig { diff --git a/app/packages/state/src/hooks/useCreateLooker.ts b/app/packages/state/src/hooks/useCreateLooker.ts index adb854be48..ae0f776038 100644 --- a/app/packages/state/src/hooks/useCreateLooker.ts +++ b/app/packages/state/src/hooks/useCreateLooker.ts @@ -132,6 +132,7 @@ export default >( mediaField, thumbnail, view, + shouldHandleKeyEvents: isModal, }; let sampleMediaFilePath = urls[mediaField]; diff --git a/app/packages/state/src/recoil/modal.ts b/app/packages/state/src/recoil/modal.ts index 2a79313a37..52baab87c2 100644 --- a/app/packages/state/src/recoil/modal.ts +++ b/app/packages/state/src/recoil/modal.ts @@ -1,13 +1,9 @@ -import { - AbstractLooker, - BaseState, - PointInfo, - type Sample, -} from "@fiftyone/looker"; +import { PointInfo, type Sample } from "@fiftyone/looker"; import { mainSample, mainSampleQuery } from "@fiftyone/relay"; import { atom, selector } from "recoil"; import { graphQLSelector } from "recoil-relay"; import { VariablesOf } from "relay-runtime"; +import type { Lookers } from "../hooks"; import { ComputeCoordinatesReturnType } from "../hooks/useTooltip"; import { ModalSelector, sessionAtom } from "../session"; import { ResponseFrom } from "../utils"; @@ -27,7 +23,7 @@ import { datasetName } from "./selectors"; import { mapSampleResponse } from "./utils"; import { view } from "./view"; -export const modalLooker = atom | null>({ +export const modalLooker = atom({ key: "modalLooker", default: null, dangerouslyAllowMutability: true, @@ -73,7 +69,7 @@ export const currentSampleId = selector({ ? get(pinned3DSample).id : get(nullableModalSampleId); - if (id && id.endsWith("-modal")) { + if (id?.endsWith("-modal")) { return id.replace("-modal", ""); } return id; From a26550bc4897db5ba9ed48a2c5523dd611d1def6 Mon Sep 17 00:00:00 2001 From: Benjamin Kane Date: Wed, 2 Oct 2024 17:24:10 -0400 Subject: [PATCH 04/11] add example back --- .../playback/src/views/TimelineExamples.tsx | 43 +++++++++++++++++++ 1 file changed, 43 insertions(+) diff --git a/app/packages/playback/src/views/TimelineExamples.tsx b/app/packages/playback/src/views/TimelineExamples.tsx index 3e85387191..9f3fb12178 100644 --- a/app/packages/playback/src/views/TimelineExamples.tsx +++ b/app/packages/playback/src/views/TimelineExamples.tsx @@ -106,6 +106,49 @@ export const TimelineCreator = () => { ); }; +export const TimelineSubscriber1 = () => { + const { getName } = useDefaultTimelineNameImperative(); + const timelineName = React.useMemo(() => getName(), [getName]); + + const [myLocalFrameNumber, setMyLocalFrameNumber] = + React.useState(DEFAULT_FRAME_NUMBER); + + const loadRange = React.useCallback(async (range: BufferRange) => { + // no-op for now, but maybe for testing, i can resolve a promise inside settimeout + }, []); + + const myRenderFrame = React.useCallback((frameNumber: number) => { + setMyLocalFrameNumber(frameNumber); + }, []); + + const { subscribe, isTimelineInitialized, getFrameNumber } = useTimeline(); + + React.useEffect(() => { + if (!isTimelineInitialized) { + return; + } + + subscribe({ + id: `sub1`, + loadRange, + renderFrame: myRenderFrame, + }); + }, [loadRange, myRenderFrame, subscribe, isTimelineInitialized]); + + if (!isTimelineInitialized) { + return
loading...
; + } + + return ( + <> +
+ Subscriber 1 frame number {timelineName}: {myLocalFrameNumber} +
+ + + ); +}; + export const TimelineSubscriber2 = () => { const { getName } = useDefaultTimelineNameImperative(); const timelineName = React.useMemo(() => getName(), [getName]); From 7448302574fbf0ce9dbf4bd9724de30818ff810d Mon Sep 17 00:00:00 2001 From: Benjamin Kane Date: Thu, 3 Oct 2024 21:12:01 -0400 Subject: [PATCH 05/11] cleaning --- .../src/components/Modal/ImaVidLooker.tsx | 15 ++--------- .../core/src/components/Modal/ModalLooker.tsx | 2 +- .../src/components/Modal/use-key-events.ts | 27 +++++++++++++++++++ .../core/src/components/Modal/use-looker.ts | 17 +++--------- .../looker/src/elements/common/looker.ts | 1 - app/packages/looker/src/lookers/video.ts | 14 ++++++++++ 6 files changed, 47 insertions(+), 29 deletions(-) create mode 100644 app/packages/core/src/components/Modal/use-key-events.ts diff --git a/app/packages/core/src/components/Modal/ImaVidLooker.tsx b/app/packages/core/src/components/Modal/ImaVidLooker.tsx index 9797631977..09d152d442 100644 --- a/app/packages/core/src/components/Modal/ImaVidLooker.tsx +++ b/app/packages/core/src/components/Modal/ImaVidLooker.tsx @@ -19,6 +19,7 @@ import { useRecoilValue, useSetRecoilState } from "recoil"; import { v4 as uuid } from "uuid"; import { useClearSelectedLabels, useShowOverlays } from "./ModalLooker"; import { useInitializeImaVidSubscriptions, useModalContext } from "./hooks"; +import useKeyEvents from "./use-key-events"; import { shortcutToHelpItems } from "./utils"; interface ImaVidLookerReactProps { @@ -128,19 +129,7 @@ export const ImaVidLookerReact = React.memo( useEventHandler(looker, "clear", useClearSelectedLabels()); - const hoveredSample = useRecoilValue(fos.hoveredSample); - - useEffect(() => { - const hoveredSampleId = hoveredSample?._id; - looker.updater((state) => ({ - ...state, - // todo: always setting it to true might not be wise - shouldHandleKeyEvents: true, - options: { - ...state.options, - }, - })); - }, [hoveredSample, sample, looker]); + useKeyEvents(initialRef, sample._id, looker); const ref = useRef(null); useEffect(() => { diff --git a/app/packages/core/src/components/Modal/ModalLooker.tsx b/app/packages/core/src/components/Modal/ModalLooker.tsx index f9155d49f8..08b52c7ca0 100644 --- a/app/packages/core/src/components/Modal/ModalLooker.tsx +++ b/app/packages/core/src/components/Modal/ModalLooker.tsx @@ -28,7 +28,7 @@ interface LookerProps { } const ModalLookerNoTimeline = React.memo((props: LookerProps) => { - const { id, looker, ref, sample } = useLooker(props); + const { id, looker, ref } = useLooker(props); const theme = useTheme(); const setModalLooker = useSetRecoilState(fos.modalLooker); diff --git a/app/packages/core/src/components/Modal/use-key-events.ts b/app/packages/core/src/components/Modal/use-key-events.ts new file mode 100644 index 0000000000..7ca5a09efa --- /dev/null +++ b/app/packages/core/src/components/Modal/use-key-events.ts @@ -0,0 +1,27 @@ +import type { Lookers } from "@fiftyone/state"; +import { hoveredSample } from "@fiftyone/state"; +import type { MutableRefObject } from "react"; +import { useEffect } from "react"; +import { selector, useRecoilValue } from "recoil"; + +export const hoveredSampleId = selector({ + key: "hoveredSampleId", + get: ({ get }) => { + return get(hoveredSample)?._id; + }, +}); + +export default function ( + ref: MutableRefObject, + id: string, + looker: Lookers +) { + const hoveredId = useRecoilValue(hoveredSampleId); + + useEffect(() => { + !ref.current && + looker.updateOptions({ + shouldHandleKeyEvents: id === hoveredId, + }); + }, [hoveredId, id, looker, ref]); +} diff --git a/app/packages/core/src/components/Modal/use-looker.ts b/app/packages/core/src/components/Modal/use-looker.ts index 65366cbfaa..3b1176e608 100644 --- a/app/packages/core/src/components/Modal/use-looker.ts +++ b/app/packages/core/src/components/Modal/use-looker.ts @@ -1,19 +1,13 @@ import * as fos from "@fiftyone/state"; import React, { useEffect, useRef, useState } from "react"; import { useErrorHandler } from "react-error-boundary"; -import { selector, useRecoilValue } from "recoil"; +import { useRecoilValue } from "recoil"; import { v4 as uuid } from "uuid"; import { useClearSelectedLabels, useShowOverlays } from "./ModalLooker"; import { useLookerOptionsUpdate } from "./hooks"; +import useKeyEvents from "./use-key-events"; import { shortcutToHelpItems } from "./utils"; -export const hoveredSampleId = selector({ - key: "hoveredSampleId", - get: ({ get }) => { - return get(fos.hoveredSample)?._id; - }, -}); - const CLOSE = "close"; function useLooker({ @@ -94,13 +88,8 @@ function useLooker({ ); } ); - const hoveredId = useRecoilValue(hoveredSampleId); - useEffect(() => { - looker.updateOptions({ - shouldHandleKeyEvents: sample.sample._id === hoveredId, - }); - }, [hoveredId, sample, looker]); + useKeyEvents(initialRef, sample.sample._id, looker); return { id, looker, ref, sample, updateLookerOptions }; } diff --git a/app/packages/looker/src/elements/common/looker.ts b/app/packages/looker/src/elements/common/looker.ts index 95e6eda918..b483b178f2 100644 --- a/app/packages/looker/src/elements/common/looker.ts +++ b/app/packages/looker/src/elements/common/looker.ts @@ -29,7 +29,6 @@ export class LookerElement extends BaseElement< const { SHORTCUTS, error, - options: { shouldHandleKeyEvents }, } = state; if (!error && e.key in SHORTCUTS) { diff --git a/app/packages/looker/src/lookers/video.ts b/app/packages/looker/src/lookers/video.ts index fa9a92af49..878f7dee76 100644 --- a/app/packages/looker/src/lookers/video.ts +++ b/app/packages/looker/src/lookers/video.ts @@ -219,6 +219,20 @@ export class VideoLooker extends AbstractLooker { get waiting() { const video = this.lookerElement.children[0].element as HTMLVideoElement; + if ( + !this.state.config.thumbnail && + video && + (video.seeking || + video.readyState < 2 || + !this.hasFrame(this.state.frameNumber)) + ) { + console.log( + video, + video.seeking, + video.readyState < 2, + !this.hasFrame(this.state.frameNumber) + ); + } return ( video && (video.seeking || From 9f88fed08ad8122db6ec9dc92d4bb6c37a53057f Mon Sep 17 00:00:00 2001 From: Benjamin Kane Date: Fri, 4 Oct 2024 10:02:36 -0400 Subject: [PATCH 06/11] useKeyEvents hook --- .../core/src/components/Modal/ImaVidLooker.tsx | 2 +- .../core/src/components/Modal/use-key-events.ts | 14 ++++++-------- .../core/src/components/Modal/use-looker.ts | 2 +- 3 files changed, 8 insertions(+), 10 deletions(-) diff --git a/app/packages/core/src/components/Modal/ImaVidLooker.tsx b/app/packages/core/src/components/Modal/ImaVidLooker.tsx index 09d152d442..82d7147a90 100644 --- a/app/packages/core/src/components/Modal/ImaVidLooker.tsx +++ b/app/packages/core/src/components/Modal/ImaVidLooker.tsx @@ -129,7 +129,7 @@ export const ImaVidLookerReact = React.memo( useEventHandler(looker, "clear", useClearSelectedLabels()); - useKeyEvents(initialRef, sample._id, looker); + useKeyEvents(sample._id, looker); const ref = useRef(null); useEffect(() => { diff --git a/app/packages/core/src/components/Modal/use-key-events.ts b/app/packages/core/src/components/Modal/use-key-events.ts index 7ca5a09efa..e58a1f7053 100644 --- a/app/packages/core/src/components/Modal/use-key-events.ts +++ b/app/packages/core/src/components/Modal/use-key-events.ts @@ -1,6 +1,5 @@ import type { Lookers } from "@fiftyone/state"; import { hoveredSample } from "@fiftyone/state"; -import type { MutableRefObject } from "react"; import { useEffect } from "react"; import { selector, useRecoilValue } from "recoil"; @@ -11,17 +10,16 @@ export const hoveredSampleId = selector({ }, }); -export default function ( - ref: MutableRefObject, - id: string, - looker: Lookers -) { +export default function (id: string, looker: Lookers) { const hoveredId = useRecoilValue(hoveredSampleId); useEffect(() => { - !ref.current && + const load = () => { looker.updateOptions({ shouldHandleKeyEvents: id === hoveredId, }); - }, [hoveredId, id, looker, ref]); + looker.removeEventListener("load", load); + }; + looker.addEventListener("load", load); + }, [hoveredId, id, looker]); } diff --git a/app/packages/core/src/components/Modal/use-looker.ts b/app/packages/core/src/components/Modal/use-looker.ts index 3b1176e608..e469766a9a 100644 --- a/app/packages/core/src/components/Modal/use-looker.ts +++ b/app/packages/core/src/components/Modal/use-looker.ts @@ -89,7 +89,7 @@ function useLooker({ } ); - useKeyEvents(initialRef, sample.sample._id, looker); + useKeyEvents(sample.sample._id, looker); return { id, looker, ref, sample, updateLookerOptions }; } From 238bb8dfc41187c7017f7adeb59b075f48f9b7a7 Mon Sep 17 00:00:00 2001 From: Benjamin Kane Date: Fri, 4 Oct 2024 10:12:45 -0400 Subject: [PATCH 07/11] after load handling --- .../src/components/Modal/ImaVidLooker.tsx | 2 +- .../src/components/Modal/use-key-events.ts | 29 ++++++++++++++----- .../core/src/components/Modal/use-looker.ts | 2 +- 3 files changed, 24 insertions(+), 9 deletions(-) diff --git a/app/packages/core/src/components/Modal/ImaVidLooker.tsx b/app/packages/core/src/components/Modal/ImaVidLooker.tsx index 82d7147a90..09d152d442 100644 --- a/app/packages/core/src/components/Modal/ImaVidLooker.tsx +++ b/app/packages/core/src/components/Modal/ImaVidLooker.tsx @@ -129,7 +129,7 @@ export const ImaVidLookerReact = React.memo( useEventHandler(looker, "clear", useClearSelectedLabels()); - useKeyEvents(sample._id, looker); + useKeyEvents(initialRef, sample._id, looker); const ref = useRef(null); useEffect(() => { diff --git a/app/packages/core/src/components/Modal/use-key-events.ts b/app/packages/core/src/components/Modal/use-key-events.ts index e58a1f7053..49a4ce313b 100644 --- a/app/packages/core/src/components/Modal/use-key-events.ts +++ b/app/packages/core/src/components/Modal/use-key-events.ts @@ -1,6 +1,7 @@ import type { Lookers } from "@fiftyone/state"; import { hoveredSample } from "@fiftyone/state"; -import { useEffect } from "react"; +import type { MutableRefObject } from "react"; +import { useEffect, useRef } from "react"; import { selector, useRecoilValue } from "recoil"; export const hoveredSampleId = selector({ @@ -10,16 +11,30 @@ export const hoveredSampleId = selector({ }, }); -export default function (id: string, looker: Lookers) { +export default function ( + ref: MutableRefObject, + id: string, + looker: Lookers +) { const hoveredId = useRecoilValue(hoveredSampleId); + const ready = useRef(false); useEffect(() => { - const load = () => { + if (ref.current) { + // initial call should wait for load event + const update = () => { + looker.updateOptions({ + shouldHandleKeyEvents: id === hoveredId, + }); + ready.current = true; + + looker.removeEventListener("load", update); + }; + looker.addEventListener("load", update); + } else if (ready.current) { looker.updateOptions({ shouldHandleKeyEvents: id === hoveredId, }); - looker.removeEventListener("load", load); - }; - looker.addEventListener("load", load); - }, [hoveredId, id, looker]); + } + }, [hoveredId, id, looker, ref]); } diff --git a/app/packages/core/src/components/Modal/use-looker.ts b/app/packages/core/src/components/Modal/use-looker.ts index e469766a9a..3b1176e608 100644 --- a/app/packages/core/src/components/Modal/use-looker.ts +++ b/app/packages/core/src/components/Modal/use-looker.ts @@ -89,7 +89,7 @@ function useLooker({ } ); - useKeyEvents(sample.sample._id, looker); + useKeyEvents(initialRef, sample.sample._id, looker); return { id, looker, ref, sample, updateLookerOptions }; } From a751bb4650c951ee2f07c7254ede70b5d193d6c5 Mon Sep 17 00:00:00 2001 From: Benjamin Kane Date: Fri, 4 Oct 2024 11:11:43 -0400 Subject: [PATCH 08/11] fix import --- app/packages/core/src/components/Modal/ImaVidLooker.tsx | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/app/packages/core/src/components/Modal/ImaVidLooker.tsx b/app/packages/core/src/components/Modal/ImaVidLooker.tsx index 09d152d442..952d6a1179 100644 --- a/app/packages/core/src/components/Modal/ImaVidLooker.tsx +++ b/app/packages/core/src/components/Modal/ImaVidLooker.tsx @@ -18,7 +18,11 @@ import { useErrorHandler } from "react-error-boundary"; import { useRecoilValue, useSetRecoilState } from "recoil"; import { v4 as uuid } from "uuid"; import { useClearSelectedLabels, useShowOverlays } from "./ModalLooker"; -import { useInitializeImaVidSubscriptions, useModalContext } from "./hooks"; +import { + useInitializeImaVidSubscriptions, + useLookerOptionsUpdate, + useModalContext, +} from "./hooks"; import useKeyEvents from "./use-key-events"; import { shortcutToHelpItems } from "./utils"; From 9fda07abc2c13d3db680ff130e3f657bd8943baa Mon Sep 17 00:00:00 2001 From: Benjamin Kane Date: Fri, 4 Oct 2024 11:25:53 -0400 Subject: [PATCH 09/11] enable timeline --- .../core/src/components/Modal/use-looker.ts | 8 +++++++- app/packages/looker/src/lookers/video.ts | 17 ++--------------- app/packages/looker/src/state.ts | 1 + app/packages/state/src/hooks/useCreateLooker.ts | 4 +++- 4 files changed, 13 insertions(+), 17 deletions(-) diff --git a/app/packages/core/src/components/Modal/use-looker.ts b/app/packages/core/src/components/Modal/use-looker.ts index 3b1176e608..d0d72ec568 100644 --- a/app/packages/core/src/components/Modal/use-looker.ts +++ b/app/packages/core/src/components/Modal/use-looker.ts @@ -20,7 +20,13 @@ function useLooker({ const ref = useRef(null); const [reset, setReset] = useState(false); const lookerOptions = fos.useLookerOptions(true); - const createLooker = fos.useCreateLooker(true, false, lookerOptions); + const createLooker = fos.useCreateLooker( + true, + false, + lookerOptions, + undefined, + true + ); const selectedMediaField = useRecoilValue(fos.selectedMediaField(true)); const colorScheme = useRecoilValue(fos.colorScheme); const looker = React.useMemo(() => { diff --git a/app/packages/looker/src/lookers/video.ts b/app/packages/looker/src/lookers/video.ts index 878f7dee76..0307594409 100644 --- a/app/packages/looker/src/lookers/video.ts +++ b/app/packages/looker/src/lookers/video.ts @@ -219,20 +219,6 @@ export class VideoLooker extends AbstractLooker { get waiting() { const video = this.lookerElement.children[0].element as HTMLVideoElement; - if ( - !this.state.config.thumbnail && - video && - (video.seeking || - video.readyState < 2 || - !this.hasFrame(this.state.frameNumber)) - ) { - console.log( - video, - video.seeking, - video.readyState < 2, - !this.hasFrame(this.state.frameNumber) - ); - } return ( video && (video.seeking || @@ -541,11 +527,12 @@ export class VideoLooker extends AbstractLooker { this.state.setZoom = false; } - !this.state.config.thumbnail && + if (this.state.config.enableTimeline) { getDefaultStore().set(setFrameNumberAtom, { name: `timeline-${this.state.config.sampleId}`, newFrameNumber: this.state.frameNumber, }); + } return super.postProcess(); } diff --git a/app/packages/looker/src/state.ts b/app/packages/looker/src/state.ts index b4d1da5ad8..48cadc6c4f 100644 --- a/app/packages/looker/src/state.ts +++ b/app/packages/looker/src/state.ts @@ -219,6 +219,7 @@ export interface FrameConfig extends BaseConfig { export type ImageConfig = BaseConfig; export interface VideoConfig extends BaseConfig { + enableTimeline: boolean; frameRate: number; support?: [number, number]; } diff --git a/app/packages/state/src/hooks/useCreateLooker.ts b/app/packages/state/src/hooks/useCreateLooker.ts index ae0f776038..1fc1b748e3 100644 --- a/app/packages/state/src/hooks/useCreateLooker.ts +++ b/app/packages/state/src/hooks/useCreateLooker.ts @@ -36,7 +36,8 @@ export default >( isModal: boolean, thumbnail: boolean, options: Omit[0], "selected">, - highlight?: (sample: Sample) => boolean + highlight?: (sample: Sample) => boolean, + enableTimeline?: boolean ) => { const environment = useRelayEnvironment(); const selected = useRecoilValue(selectedSamples); @@ -112,6 +113,7 @@ export default >( } let config: ConstructorParameters[1] = { + enableTimeline, fieldSchema: { ...fieldSchema, frames: { From b2a47f856654a93a2a68cbea2ac6e7598ffb838d Mon Sep 17 00:00:00 2001 From: Benjamin Kane Date: Fri, 4 Oct 2024 16:59:38 -0400 Subject: [PATCH 10/11] default to true --- app/packages/looker/src/state.ts | 1 + 1 file changed, 1 insertion(+) diff --git a/app/packages/looker/src/state.ts b/app/packages/looker/src/state.ts index 48cadc6c4f..84281f154a 100644 --- a/app/packages/looker/src/state.ts +++ b/app/packages/looker/src/state.ts @@ -459,6 +459,7 @@ export const DEFAULT_BASE_OPTIONS: BaseOptions = { pointFilter: (path: string, point: Point) => true, attributeVisibility: {}, mediaFallback: false, + shouldHandleKeyEvents: true, }; export const DEFAULT_FRAME_OPTIONS: FrameOptions = { From ecd3c7c560a2334b3273703c6d53d42ab4808806 Mon Sep 17 00:00:00 2001 From: Benjamin Kane Date: Fri, 4 Oct 2024 17:32:03 -0400 Subject: [PATCH 11/11] refreshers --- app/packages/core/src/components/Modal/use-looker.ts | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/app/packages/core/src/components/Modal/use-looker.ts b/app/packages/core/src/components/Modal/use-looker.ts index d0d72ec568..fa44065f95 100644 --- a/app/packages/core/src/components/Modal/use-looker.ts +++ b/app/packages/core/src/components/Modal/use-looker.ts @@ -30,8 +30,11 @@ function useLooker({ const selectedMediaField = useRecoilValue(fos.selectedMediaField(true)); const colorScheme = useRecoilValue(fos.colorScheme); const looker = React.useMemo(() => { + /** start refreshers */ reset; selectedMediaField; + /** end refreshers */ + return createLooker.current(sample); }, [createLooker, reset, sample, selectedMediaField]) as L; const handleError = useErrorHandler(); @@ -49,7 +52,10 @@ function useLooker({ }, [looker, lookerOptions]); useEffect(() => { + /** start refreshers */ colorScheme; + /** end refreshers */ + !initialRef.current && looker.updateSample(sample); }, [colorScheme, looker, sample]);