diff --git a/app/packages/core/src/components/ColorModal/ColorFooter.tsx b/app/packages/core/src/components/ColorModal/ColorFooter.tsx index 04509601a2..f7a7515616 100644 --- a/app/packages/core/src/components/ColorModal/ColorFooter.tsx +++ b/app/packages/core/src/components/ColorModal/ColorFooter.tsx @@ -3,7 +3,7 @@ import * as foq from "@fiftyone/relay"; import * as fos from "@fiftyone/state"; import React, { useEffect } from "react"; import { useMutation } from "react-relay"; -import { useRecoilState, useRecoilValue } from "recoil"; +import { useRecoilValue } from "recoil"; import { ButtonGroup, ModalActionButtonContainer } from "./ShareStyledDiv"; import { activeColorEntry } from "./state"; @@ -14,8 +14,7 @@ const ColorFooter: React.FC = () => { ? canEditCustomColors.message : "Save to dataset app config"; const setColorScheme = fos.useSetSessionColorScheme(); - const [activeColorModalField, setActiveColorModalField] = - useRecoilState(activeColorEntry); + const activeColorModalField = useRecoilValue(activeColorEntry); const [setDatasetColorScheme] = useMutation(foq.setDatasetColorScheme); const colorScheme = useRecoilValue(fos.colorScheme); @@ -26,8 +25,8 @@ const ColorFooter: React.FC = () => { const subscription = useRecoilValue(fos.stateSubscription); useEffect( - () => foq.subscribe(() => setActiveColorModalField(null)), - [setActiveColorModalField] + () => foq.subscribe((_, { set }) => set(activeColorEntry, null)), + [] ); if (!activeColorModalField) return null; diff --git a/app/packages/core/src/components/Modal/ImaVidLooker.tsx b/app/packages/core/src/components/Modal/ImaVidLooker.tsx index 6707202a8a..952d6a1179 100644 --- a/app/packages/core/src/components/Modal/ImaVidLooker.tsx +++ b/app/packages/core/src/components/Modal/ImaVidLooker.tsx @@ -17,13 +17,14 @@ import React, { import { useErrorHandler } from "react-error-boundary"; import { useRecoilValue, useSetRecoilState } from "recoil"; import { v4 as uuid } from "uuid"; -import { useInitializeImaVidSubscriptions, useModalContext } from "./hooks"; +import { useClearSelectedLabels, useShowOverlays } from "./ModalLooker"; import { - shortcutToHelpItems, - useClearSelectedLabels, + useInitializeImaVidSubscriptions, useLookerOptionsUpdate, - useShowOverlays, -} from "./ModalLooker"; + useModalContext, +} from "./hooks"; +import useKeyEvents from "./use-key-events"; +import { shortcutToHelpItems } from "./utils"; interface ImaVidLookerReactProps { sample: fos.ModalSample; @@ -132,19 +133,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 c18eb5e048..6e0313ab6d 100644 --- a/app/packages/core/src/components/Modal/ModalLooker.tsx +++ b/app/packages/core/src/components/Modal/ModalLooker.tsx @@ -1,35 +1,11 @@ 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 { useRecoilCallback, useRecoilValue, useSetRecoilState } from "recoil"; -import { v4 as uuid } from "uuid"; -import { useModalContext } from "./hooks"; +import React, { useMemo } from "react"; +import { useRecoilCallback, useRecoilValue } from "recoil"; import { ImaVidLookerReact } from "./ImaVidLooker"; - -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 { VideoLookerReact } from "./VideoLooker"; +import useLooker from "./use-looker"; export const useShowOverlays = () => { return useRecoilCallback(({ set }) => async (event: CustomEvent) => { @@ -47,137 +23,27 @@ 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 updateLookerOptions = useLookerOptionsUpdate(); - useEventHandler(looker, "options", (e) => updateLookerOptions(e.detail)); - useEventHandler(looker, "showOverlays", useShowOverlays()); - useEventHandler(looker, "reset", () => { - setReset((c) => !c); - }); - - const jsonPanel = fos.useJSONPanel(); - const helpPanel = fos.useHelpPanel(); - - 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 ( -
- ); - } -); +const ModalLookerNoTimeline = React.memo((props: LookerProps) => { + const { id, ref } = useLooker(props); + const theme = useTheme(); + + return ( +
+ ); +}); export const ModalLooker = React.memo( ({ sample: propsSampleData }: LookerProps) => { @@ -197,21 +63,16 @@ export const ModalLooker = React.memo( const shouldRenderImavid = useRecoilValue( fos.shouldRenderImaVidLooker(true) ); + const video = useRecoilValue(fos.isVideoDataset); if (shouldRenderImavid) { return ; } + if (video) { + return ; + } + 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 new file mode 100644 index 0000000000..9c8f9b0cd2 --- /dev/null +++ b/app/packages/core/src/components/Modal/VideoLooker.tsx @@ -0,0 +1,80 @@ +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 useLooker from "./use-looker"; + +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 { pause, play } = useTimeline(timelineName); + + fos.useEventHandler(looker, "pause", pause); + fos.useEventHandler(looker, "play", play); + + return null; +}; 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/use-key-events.ts b/app/packages/core/src/components/Modal/use-key-events.ts new file mode 100644 index 0000000000..49a4ce313b --- /dev/null +++ b/app/packages/core/src/components/Modal/use-key-events.ts @@ -0,0 +1,40 @@ +import type { Lookers } from "@fiftyone/state"; +import { hoveredSample } from "@fiftyone/state"; +import type { MutableRefObject } from "react"; +import { useEffect, useRef } 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); + const ready = useRef(false); + + useEffect(() => { + 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, + }); + } + }, [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 new file mode 100644 index 0000000000..ed6b46d8ee --- /dev/null +++ b/app/packages/core/src/components/Modal/use-looker.ts @@ -0,0 +1,123 @@ +import * as fos from "@fiftyone/state"; +import React, { useEffect, useRef, useState } from "react"; +import { useErrorHandler } from "react-error-boundary"; +import { useRecoilValue, useSetRecoilState } from "recoil"; +import { v4 as uuid } from "uuid"; +import { useClearSelectedLabels, useShowOverlays } from "./ModalLooker"; +import { useLookerOptionsUpdate, useModalContext } from "./hooks"; +import useKeyEvents from "./use-key-events"; +import { shortcutToHelpItems } from "./utils"; + +const CLOSE = "close"; + +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, + undefined, + true + ); + 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(); + 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(() => { + /** start refreshers */ + colorScheme; + /** end refreshers */ + + !initialRef.current && looker.updateSample(sample.sample); + }, [colorScheme, looker, sample]); + + useEffect(() => { + initialRef.current = false; + }, []); + + useEffect(() => { + ref.current?.dispatchEvent( + new CustomEvent("looker-attached", { bubbles: true }) + ); + }, []); + + 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) + ); + } + ); + + useKeyEvents(initialRef, sample.sample._id, looker); + + const setModalLooker = useSetRecoilState(fos.modalLooker); + + const { setActiveLookerRef } = useModalContext(); + + useEffect(() => { + setModalLooker(looker); + }, [looker, setModalLooker]); + + useEffect(() => { + if (looker) { + setActiveLookerRef(looker as fos.Lookers); + } + }, [looker, setActiveLookerRef]); + + return { id, looker, ref, sample, updateLookerOptions }; +} + +export default useLooker; 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/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/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/bubbles.test.ts b/app/packages/looker/src/elements/common/bubbles.test.ts index 40b5dd654f..edac34b194 100644 --- a/app/packages/looker/src/elements/common/bubbles.test.ts +++ b/app/packages/looker/src/elements/common/bubbles.test.ts @@ -1,9 +1,11 @@ import type { Schema } from "@fiftyone/utilities"; import { + CLASSIFICATIONS, DYNAMIC_EMBEDDED_DOCUMENT_PATH, EMBEDDED_DOCUMENT_FIELD, LIST_FIELD, STRING_FIELD, + TEMPORAL_DETECTIONS, } from "@fiftyone/utilities"; import { describe, expect, it } from "vitest"; import { getBubbles, getField, unwind } from "./bubbles"; @@ -195,6 +197,38 @@ describe("text bubble tests", () => { } ) ).toStrictEqual([field.fields.value, ["value"]]); + + const classifications = { + ...FIELD_DATA, + dbField: "classes", + ftype: EMBEDDED_DOCUMENT_FIELD, + embeddedDocType: CLASSIFICATIONS, + fields: {}, + }; + + expect( + getBubbles( + "classes", + { classes: { classifications: [{ label: "label" }] } }, + { classes: classifications } + ) + ); + + const temporalDetections = { + ...FIELD_DATA, + dbField: "temporal", + ftype: EMBEDDED_DOCUMENT_FIELD, + embeddedDocType: TEMPORAL_DETECTIONS, + fields: {}, + }; + + expect( + getBubbles( + "temporal", + { temporal: { detections: [{ label: "label" }] } }, + { temporal: temporalDetections } + ) + ); }); it("getField gets field from a path keys", () => { diff --git a/app/packages/looker/src/elements/common/bubbles.ts b/app/packages/looker/src/elements/common/bubbles.ts index 9b649e8adf..feb9195c93 100644 --- a/app/packages/looker/src/elements/common/bubbles.ts +++ b/app/packages/looker/src/elements/common/bubbles.ts @@ -54,14 +54,14 @@ export const getBubbles = ( } if (field.embeddedDocType === withPath(LABELS_PATH, CLASSIFICATIONS)) { - out.values = out.values.flatMap( + out.values = unwind(field.dbField, out.values).flatMap( (value) => value.classifications || [] ) as Sample[]; break; } if (field.embeddedDocType === withPath(LABELS_PATH, TEMPORAL_DETECTIONS)) { - out.values = out.values.flatMap( + out.values = unwind(field.dbField, out.values).flatMap( (value) => value.detections || [] ) as Sample[]; break; diff --git a/app/packages/looker/src/elements/common/looker.ts b/app/packages/looker/src/elements/common/looker.ts index 8910d90cd0..b483b178f2 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"; @@ -24,7 +26,11 @@ export class LookerElement extends BaseElement< const e = event as KeyboardEvent; update((state) => { - const { SHORTCUTS, error, shouldHandleKeyEvents } = state; + const { + SHORTCUTS, + error, + options: { shouldHandleKeyEvents }, + } = state; if (!error && e.key in SHORTCUTS) { const matchedControl = SHORTCUTS[e.key] as Control; const enabled = @@ -43,7 +49,7 @@ export class LookerElement extends BaseElement< } const e = event as KeyboardEvent; - update(({ SHORTCUTS, error, 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/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..305bb3ce83 100644 --- a/app/packages/looker/src/lookers/video.ts +++ b/app/packages/looker/src/lookers/video.ts @@ -15,12 +15,15 @@ import { FrameSample, LabelData, StateUpdate, + VideoConfig, VideoSample, VideoState, } 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"; @@ -316,7 +319,7 @@ export class VideoLooker extends AbstractLooker { ...this.getDefaultOptions(), ...options, }, - buffers: [[firstFrame, firstFrame]] as Buffers, + buffers: this.initialBuffers(config), seekBarHovering: false, SHORTCUTS: VIDEO_SHORTCUTS, hasPoster: false, @@ -326,8 +329,8 @@ export class VideoLooker extends AbstractLooker { } hasDefaultZoom(state: VideoState, overlays: Overlay[]): boolean { - let pan = [0, 0]; - let scale = 1; + const pan = [0, 0]; + const scale = 1; return ( scale === state.scale && @@ -400,10 +403,9 @@ export class VideoLooker extends AbstractLooker { lookerWithReader !== this && frameCount !== null ) { - lookerWithReader && lookerWithReader.pause(); - this.setReader(); + lookerWithReader?.pause(); lookerWithReader = this; - this.state.buffers = [[1, 1]]; + this.setReader(); } else if (lookerWithReader !== this && frameCount) { this.state.buffering && this.dispatchEvent("buffering", false); this.state.playing = false; @@ -525,6 +527,13 @@ export class VideoLooker extends AbstractLooker { this.state.setZoom = false; } + if (this.state.config.enableTimeline) { + getDefaultStore().set(setFrameNumberAtom, { + name: `timeline-${this.state.config.sampleId}`, + newFrameNumber: this.state.frameNumber, + }); + } + return super.postProcess(); } @@ -540,10 +549,18 @@ export class VideoLooker extends AbstractLooker { } updateSample(sample: VideoSample) { - this.state.buffers = [[1, 1]]; - this.frames.clear(); + if (lookerWithReader === this) { + lookerWithReader?.pause(); + lookerWithReader = null; + } + + this.frames = new Map(); + this.state.buffers = this.initialBuffers(this.state.config); super.updateSample(sample); - this.setReader(); + } + + getVideo() { + return this.lookerElement.children[0].element as HTMLVideoElement; } private hasFrame(frameNumber: number) { @@ -552,6 +569,11 @@ export class VideoLooker extends AbstractLooker { this.frames.get(frameNumber)?.deref() !== undefined ); } + + private initialBuffers(config: VideoConfig) { + const firstFrame = config.support ? config.support[0] : 1; + return [[firstFrame, firstFrame]] as Buffers; + } } const withFrames = (obj: T): T => diff --git a/app/packages/looker/src/state.ts b/app/packages/looker/src/state.ts index 64dfe21278..0b46900802 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; @@ -218,6 +219,7 @@ export interface FrameConfig extends BaseConfig { export type ImageConfig = BaseConfig; export interface VideoConfig extends BaseConfig { + enableTimeline: boolean; frameRate: number; support?: [number, number]; } @@ -297,7 +299,6 @@ export interface BaseState { showOptions: boolean; config: BaseConfig; options: BaseOptions; - shouldHandleKeyEvents: boolean; scale: number; pan: Coordinates; panning: boolean; @@ -340,6 +341,7 @@ export interface ImageState extends BaseState { } export interface VideoState extends BaseState { + buffers: Buffers; config: VideoConfig; options: VideoOptions; seeking: boolean; @@ -458,6 +460,7 @@ export const DEFAULT_BASE_OPTIONS: BaseOptions = { pointFilter: (path: string, point: Point) => true, attributeVisibility: {}, mediaFallback: false, + shouldHandleKeyEvents: true, }; export const DEFAULT_FRAME_OPTIONS: FrameOptions = { diff --git a/app/packages/plugins/src/externalize.ts b/app/packages/plugins/src/externalize.ts index 923070fe30..f8a0fd2691 100644 --- a/app/packages/plugins/src/externalize.ts +++ b/app/packages/plugins/src/externalize.ts @@ -3,6 +3,7 @@ import * as foo from "@fiftyone/operators"; import * as fos from "@fiftyone/state"; import * as fou from "@fiftyone/utilities"; import * as fosp from "@fiftyone/spaces"; +import * as fop from "@fiftyone/plugins"; import * as mui from "@mui/material"; import React from "react"; import ReactDOM from "react-dom"; @@ -19,6 +20,7 @@ declare global { __fou__: typeof fou; __foo__: typeof foo; __fosp__: typeof fosp; + __fop__: typeof fop; __mui__: typeof mui; __styled__: typeof styled; } @@ -36,5 +38,6 @@ if (typeof window !== "undefined") { window.__foo__ = foo; window.__fosp__ = fosp; window.__mui__ = mui; + window.__fop__ = fop; window.__styled__ = styled; } diff --git a/app/packages/state/src/hooks/useCreateLooker.ts b/app/packages/state/src/hooks/useCreateLooker.ts index 06e04eb652..1fc1b748e3 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, @@ -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: { @@ -132,6 +134,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; diff --git a/app/packages/state/src/utils.ts b/app/packages/state/src/utils.ts index 1bb9e0c99a..4b61bc53f4 100644 --- a/app/packages/state/src/utils.ts +++ b/app/packages/state/src/utils.ts @@ -18,7 +18,7 @@ import { RecordSource, Store, } from "relay-runtime"; -import { State } from "./recoil"; +import { ModalSample, State } from "./recoil"; export const deferrer = (initialized: MutableRefObject) => @@ -105,6 +105,7 @@ export const getStandardizedUrls = ( urls: | readonly { readonly field: string; readonly url: string }[] | { [field: string]: string } + | ModalSample["urls"] ) => { if (!Array.isArray(urls)) { return urls; diff --git a/app/packages/utilities/src/schema.test.ts b/app/packages/utilities/src/schema.test.ts index f1d9624663..506c05ee97 100644 --- a/app/packages/utilities/src/schema.test.ts +++ b/app/packages/utilities/src/schema.test.ts @@ -19,7 +19,7 @@ const SCHEMA: schema.Schema = { "fiftyone.core.odm.embedded_document.DynamicEmbeddedDocument", ftype: "fiftyone.core.fields.EmbeddedDocumentField", info: {}, - name: "top", + name: "embedded", path: "embedded", subfield: null, fields: { @@ -30,7 +30,7 @@ const SCHEMA: schema.Schema = { ftype: "fiftyone.core.fields.EmbeddedDocumentField", info: {}, name: "field", - path: "field", + path: "embedded.field", subfield: null, }, }, @@ -42,7 +42,7 @@ const SCHEMA: schema.Schema = { "fiftyone.core.odm.embedded_document.DynamicEmbeddedDocument", ftype: "fiftyone.core.fields.EmbeddedDocumentField", info: {}, - name: "top", + name: "embeddedWithDbFields", path: "embeddedWithDbFields", subfield: null, fields: { @@ -54,7 +54,7 @@ const SCHEMA: schema.Schema = { ftype: "fiftyone.core.fields.EmbeddedDocumentField", info: {}, name: "sample_id", - path: "sample_id", + path: "embeddedWithDbFields.sample_id", subfield: null, }, }, @@ -62,7 +62,7 @@ const SCHEMA: schema.Schema = { }; describe("schema", () => { - describe("getCls ", () => { + describe("getCls", () => { it("should get top level cls", () => { expect(schema.getCls("top", SCHEMA)).toBe("TopLabel"); }); @@ -79,7 +79,7 @@ describe("schema", () => { }); }); - describe("getFieldInfo ", () => { + describe("getFieldInfo", () => { it("should get top level field info", () => { expect(schema.getFieldInfo("top", SCHEMA)).toEqual({ ...SCHEMA.top, @@ -89,7 +89,7 @@ describe("schema", () => { it("should get embedded field info", () => { expect(schema.getFieldInfo("embedded.field", SCHEMA)).toEqual({ - ...SCHEMA.embedded.fields.field, + ...SCHEMA.embedded.fields!.field, pathWithDbField: "", }); }); @@ -109,4 +109,69 @@ describe("schema", () => { expect(field?.pathWithDbField).toBe("embeddedWithDbFields._sample_id"); }); }); + + describe("getFieldsWithEmbeddedDocType", () => { + it("should get all fields with embeddedDocType at top level", () => { + expect( + schema.getFieldsWithEmbeddedDocType( + SCHEMA, + "fiftyone.core.labels.TopLabel" + ) + ).toEqual([SCHEMA.top]); + }); + + it("should get all fields with embeddedDocType in nested fields", () => { + expect( + schema.getFieldsWithEmbeddedDocType( + SCHEMA, + "fiftyone.core.labels.EmbeddedLabel" + ) + ).toEqual([ + SCHEMA.embedded.fields!.field, + SCHEMA.embeddedWithDbFields.fields!.sample_id, + ]); + }); + + it("should return empty array if embeddedDocType does not exist", () => { + expect( + schema.getFieldsWithEmbeddedDocType(SCHEMA, "nonexistentDocType") + ).toEqual([]); + }); + + it("should return empty array for empty schema", () => { + expect(schema.getFieldsWithEmbeddedDocType({}, "anyDocType")).toEqual([]); + }); + }); + + describe("doesSchemaContainEmbeddedDocType", () => { + it("should return true if embeddedDocType exists at top level", () => { + expect( + schema.doesSchemaContainEmbeddedDocType( + SCHEMA, + "fiftyone.core.labels.TopLabel" + ) + ).toBe(true); + }); + + it("should return true if embeddedDocType exists in nested fields", () => { + expect( + schema.doesSchemaContainEmbeddedDocType( + SCHEMA, + "fiftyone.core.labels.EmbeddedLabel" + ) + ).toBe(true); + }); + + it("should return false if embeddedDocType does not exist", () => { + expect( + schema.doesSchemaContainEmbeddedDocType(SCHEMA, "nonexistentDocType") + ).toBe(false); + }); + + it("should return false for empty schema", () => { + expect(schema.doesSchemaContainEmbeddedDocType({}, "anyDocType")).toBe( + false + ); + }); + }); }); diff --git a/app/packages/utilities/src/schema.ts b/app/packages/utilities/src/schema.ts index d65d2d9c7f..188ab1eb37 100644 --- a/app/packages/utilities/src/schema.ts +++ b/app/packages/utilities/src/schema.ts @@ -51,3 +51,43 @@ export function getCls(fieldPath: string, schema: Schema): string | undefined { return field.embeddedDocType.split(".").slice(-1)[0]; } + +export function getFieldsWithEmbeddedDocType( + schema: Schema, + embeddedDocType: string +): Field[] { + const result: Field[] = []; + + function recurse(schema: Schema) { + for (const field of Object.values(schema ?? {})) { + if (field.embeddedDocType === embeddedDocType) { + result.push(field); + } + if (field.fields) { + recurse(field.fields); + } + } + } + + recurse(schema); + return result; +} + +export function doesSchemaContainEmbeddedDocType( + schema: Schema, + embeddedDocType: string +): boolean { + function recurse(schema: Schema): boolean { + return Object.values(schema ?? {}).some((field) => { + if (field.embeddedDocType === embeddedDocType) { + return true; + } + if (field.fields) { + return recurse(field.fields); + } + return false; + }); + } + + return recurse(schema); +} 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 diff --git a/docs/source/integrations/coco.rst b/docs/source/integrations/coco.rst index 7ca3b0e890..2b910952ef 100644 --- a/docs/source/integrations/coco.rst +++ b/docs/source/integrations/coco.rst @@ -192,18 +192,14 @@ file containing COCO-formatted labels to work with: dataset = foz.load_zoo_dataset("quickstart") - # Classes list - classes = dataset.distinct("ground_truth.detections.label") - # The directory in which the dataset's images are stored IMAGES_DIR = os.path.dirname(dataset.first().filepath) # Export some labels in COCO format - dataset.take(5).export( + dataset.take(5, seed=51).export( dataset_type=fo.types.COCODetectionDataset, label_field="ground_truth", labels_path="/tmp/coco.json", - classes=classes, ) Now we have a ``/tmp/coco.json`` file on disk containing COCO labels @@ -220,7 +216,7 @@ corresponding to the images in ``IMAGES_DIR``: "licenses": [], "categories": [ { - "id": 0, + "id": 1, "name": "airplane", "supercategory": null }, @@ -229,9 +225,9 @@ corresponding to the images in ``IMAGES_DIR``: "images": [ { "id": 1, - "file_name": "001631.jpg", - "height": 612, - "width": 612, + "file_name": "003486.jpg", + "height": 427, + "width": 640, "license": null, "coco_url": null }, @@ -241,14 +237,14 @@ corresponding to the images in ``IMAGES_DIR``: { "id": 1, "image_id": 1, - "category_id": 9, + "category_id": 1, "bbox": [ - 92.14, - 220.04, - 519.86, - 61.89000000000001 + 34.34, + 147.46, + 492.69, + 192.36 ], - "area": 32174.135400000006, + "area": 94773.8484, "iscrowd": 0 }, ... @@ -271,8 +267,9 @@ dataset: include_id=True, ) - # Verify that the class list for our dataset was imported - print(coco_dataset.default_classes) # ['airplane', 'apple', ...] + # COCO categories are also imported + print(coco_dataset.info["categories"]) + # [{'id': 1, 'name': 'airplane', 'supercategory': None}, ...] print(coco_dataset) @@ -319,16 +316,16 @@ to add them to your dataset as follows: # # Mock COCO predictions, where: # - `image_id` corresponds to the `coco_id` field of `coco_dataset` - # - `category_id` corresponds to classes in `coco_dataset.default_classes` + # - `category_id` corresponds to `coco_dataset.info["categories"]` # predictions = [ - {"image_id": 1, "category_id": 18, "bbox": [258, 41, 348, 243], "score": 0.87}, - {"image_id": 2, "category_id": 11, "bbox": [61, 22, 504, 609], "score": 0.95}, + {"image_id": 1, "category_id": 2, "bbox": [258, 41, 348, 243], "score": 0.87}, + {"image_id": 2, "category_id": 4, "bbox": [61, 22, 504, 609], "score": 0.95}, ] + categories = coco_dataset.info["categories"] # Add COCO predictions to `predictions` field of dataset - classes = coco_dataset.default_classes - fouc.add_coco_labels(coco_dataset, "predictions", predictions, classes) + fouc.add_coco_labels(coco_dataset, "predictions", predictions, categories) # Verify that predictions were added to two images print(coco_dataset.count("predictions")) # 2 diff --git a/docs/source/release-notes.rst b/docs/source/release-notes.rst index ec04235961..d78f155c1c 100644 --- a/docs/source/release-notes.rst +++ b/docs/source/release-notes.rst @@ -3,6 +3,41 @@ FiftyOne Release Notes .. default-role:: code +FiftyOne Teams 2.1.1 +-------------------- +*Released October 14, 2024* + +Includes all updates from :ref:`FiftyOne 1.0.1 `, plus: + +- Fixed an issue with Auth0 connections for deployments behind proxies +- Bumped dependency requirement `voxel51-eta>=0.13` + +.. _release-notes-v1.0.1: + +FiftyOne 1.0.1 +-------------- +*Released October 14, 2024* + +App + +- Video playback now supports the timeline API + `#4878 `_ +- Added utils to support a `rerun `_ panel + `#4876 `_ +- Fixed a bug that prevented |Classifications| labels from rendering + `#4891 `_ +- Fixed a bug that prevented the `fiftyone quickstart` and + `fiftyone app launch` commands from launching the App + `#4888 `_ + +Core + +- COCO exports now use 1-based categories + `#4884 `_ +- Fixed a bug when passing the `classes` argument to load specific classes in + :ref:`COCO format ` + `#4884 `_ + FiftyOne Teams 2.1.0 -------------------- *Released October 1, 2024* diff --git a/docs/source/user_guide/app.rst b/docs/source/user_guide/app.rst index bef8645d53..24257b872b 100644 --- a/docs/source/user_guide/app.rst +++ b/docs/source/user_guide/app.rst @@ -85,6 +85,9 @@ opened in a new tab of your web browser. See # Blocks execution until the App is closed session.wait() + # Or block execution indefinitely with a negative wait value + # session.wait(-1) + .. note:: When working inside a Docker container, FiftyOne should automatically diff --git a/docs/source/user_guide/dataset_creation/datasets.rst b/docs/source/user_guide/dataset_creation/datasets.rst index c25550a191..ac4085bb3b 100644 --- a/docs/source/user_guide/dataset_creation/datasets.rst +++ b/docs/source/user_guide/dataset_creation/datasets.rst @@ -1499,9 +1499,8 @@ where `labels.json` is a JSON file in the following format: ... ], "categories": [ - ... { - "id": 2, + "id": 1, "name": "cat", "supercategory": "animal", "keypoints": ["nose", "head", ...], @@ -1524,7 +1523,7 @@ where `labels.json` is a JSON file in the following format: { "id": 1, "image_id": 1, - "category_id": 2, + "category_id": 1, "bbox": [260, 177, 231, 199], "segmentation": [...], "keypoints": [224, 226, 2, ...], diff --git a/docs/source/user_guide/export_datasets.rst b/docs/source/user_guide/export_datasets.rst index 293672544a..810601036b 100644 --- a/docs/source/user_guide/export_datasets.rst +++ b/docs/source/user_guide/export_datasets.rst @@ -1646,9 +1646,8 @@ where `labels.json` is a JSON file in the following format: }, "licenses": [], "categories": [ - ... { - "id": 2, + "id": 1, "name": "cat", "supercategory": "animal" }, @@ -1669,7 +1668,7 @@ where `labels.json` is a JSON file in the following format: { "id": 1, "image_id": 1, - "category_id": 2, + "category_id": 1, "bbox": [260, 177, 231, 199], "segmentation": [...], "score": 0.95, diff --git a/e2e-pw/src/oss/poms/modal/imavid-controls.ts b/e2e-pw/src/oss/poms/modal/imavid-controls.ts index acc7af3f25..81228ef68b 100644 --- a/e2e-pw/src/oss/poms/modal/imavid-controls.ts +++ b/e2e-pw/src/oss/poms/modal/imavid-controls.ts @@ -32,10 +32,22 @@ export class ModalImaAsVideoControlsPom { return timelineId; } + private async waitUntilBufferingIsFinished() { + await this.page.waitForFunction( + () => + document + .querySelector("[data-cy=imavid-playhead]") + ?.getAttribute("data-playhead-state") !== "buffering" + ); + } + private async togglePlay() { + await this.waitUntilBufferingIsFinished(); + let currentPlayHeadStatus = await this.playPauseButton.getAttribute( "data-playhead-state" ); + const original = currentPlayHeadStatus; // keep pressing space until play head status changes diff --git a/fiftyone/constants.py b/fiftyone/constants.py index aed1b250ca..4783ee722c 100644 --- a/fiftyone/constants.py +++ b/fiftyone/constants.py @@ -42,7 +42,7 @@ # This setting may be ``None`` if this client has no compatibility with other # versions # -COMPATIBLE_VERSIONS = ">=0.19,<0.25" +COMPATIBLE_VERSIONS = ">=0.19,<1.1" # Package metadata _META = metadata("fiftyone") @@ -106,11 +106,3 @@ # Server setup SERVER_DIR = os.path.join(FIFTYONE_DIR, "server") - -# App setup -try: - from fiftyone.desktop import FIFTYONE_DESKTOP_APP_DIR -except ImportError: - FIFTYONE_DESKTOP_APP_DIR = os.path.normpath( - os.path.join(FIFTYONE_DIR, "../app") - ) diff --git a/fiftyone/core/session/session.py b/fiftyone/core/session/session.py index 1f05184e41..4b01c389b5 100644 --- a/fiftyone/core/session/session.py +++ b/fiftyone/core/session/session.py @@ -1125,7 +1125,7 @@ def wait(self, wait: float = 3) -> None: if wait < 0: while True: time.sleep(10) - elif self.remote: + else: self._wait_closed = False while not self._wait_closed: time.sleep(wait) diff --git a/fiftyone/server/metadata.py b/fiftyone/server/metadata.py index 294d787782..6992447e4a 100644 --- a/fiftyone/server/metadata.py +++ b/fiftyone/server/metadata.py @@ -24,6 +24,7 @@ import fiftyone.core.labels as fol from fiftyone.core.collections import SampleCollection from fiftyone.utils.utils3d import OrthographicProjectionMetadata +from fiftyone.utils.rerun import RrdFile import fiftyone.core.media as fom @@ -33,6 +34,7 @@ fol.Heatmap: "map_path", fol.Segmentation: "mask_path", OrthographicProjectionMetadata: "filepath", + RrdFile: "filepath", } _FFPROBE_BINARY_PATH = shutil.which("ffprobe") diff --git a/fiftyone/utils/coco.py b/fiftyone/utils/coco.py index 76a4fd494b..b2c5a730d9 100644 --- a/fiftyone/utils/coco.py +++ b/fiftyone/utils/coco.py @@ -45,7 +45,7 @@ def add_coco_labels( sample_collection, label_field, labels_or_path, - classes, + categories, label_type="detections", coco_id_field=None, include_annotation_id=False, @@ -68,7 +68,7 @@ def add_coco_labels( { "id": 1, "image_id": 1, - "category_id": 2, + "category_id": 1, "bbox": [260, 177, 231, 199], # optional @@ -88,7 +88,7 @@ def add_coco_labels( { "id": 1, "image_id": 1, - "category_id": 2, + "category_id": 1, "bbox": [260, 177, 231, 199], "segmentation": [...], @@ -109,7 +109,7 @@ def add_coco_labels( { "id": 1, "image_id": 1, - "category_id": 2, + "category_id": 1, "keypoints": [224, 226, 2, ...], "num_keypoints": 10, @@ -129,8 +129,14 @@ def add_coco_labels( will be created if necessary labels_or_path: a list of COCO annotations or the path to a JSON file containing such data on disk - classes: the list of class label strings or a dict mapping class IDs to - class labels + categories: can be any of the following: + + - a list of category dicts in the format of + :meth:`parse_coco_categories` specifying the classes and their + category IDs + - a dict mapping class IDs to class labels + - a list of class labels whose 1-based ordering is assumed to + correspond to the category IDs in the provided COCO labels label_type ("detections"): the type of labels to load. Supported values are ``("detections", "segmentations", "keypoints")`` coco_id_field (None): this parameter determines how to map the @@ -195,10 +201,14 @@ class labels view.compute_metadata() widths, heights = view.values(["metadata.width", "metadata.height"]) - if isinstance(classes, dict): - classes_map = classes + if isinstance(categories, dict): + classes_map = categories + elif not categories: + classes_map = {} + elif isinstance(categories[0], dict): + classes_map = {c["id"]: c["name"] for c in categories} else: - classes_map = {i: label for i, label in enumerate(classes)} + classes_map = {i: label for i, label in enumerate(categories, 1)} labels = [] for _coco_objects, width, height in zip(coco_objects, widths, heights): @@ -563,15 +573,11 @@ def setup(self): self.labels_path, extra_attrs=self.extra_attrs ) - classes = None if classes_map is not None: - classes = _to_classes(classes_map) - - if classes is not None: - info["classes"] = classes + info["classes"] = _to_classes(classes_map) image_ids = _get_matching_image_ids( - classes, + classes_map, images, annotations, image_ids=self.image_ids, @@ -907,12 +913,11 @@ def export_sample(self, image_or_path, label, metadata=None): def close(self, *args): if self._dynamic_classes: - classes = sorted(self._classes) - labels_map_rev = _to_labels_map_rev(classes) + labels_map_rev = _to_labels_map_rev(sorted(self._classes)) for anno in self._annotations: anno["category_id"] = labels_map_rev[anno["category_id"]] - else: - classes = self.classes + elif self.categories is None: + labels_map_rev = _to_labels_map_rev(self.classes) _info = self.info or {} _date_created = datetime.now().replace(microsecond=0).isoformat() @@ -933,10 +938,10 @@ def close(self, *args): categories = [ { "id": i, - "name": l, + "name": c, "supercategory": None, } - for i, l in enumerate(classes) + for c, i in sorted(labels_map_rev.items(), key=lambda t: t[1]) ] labels = { @@ -1681,7 +1686,7 @@ def download_coco_dataset_split( if classes is not None: # Filter by specified classes all_ids, any_ids = _get_images_with_classes( - image_ids, annotations, classes, all_classes + image_ids, annotations, classes, all_classes_map ) else: all_ids = image_ids @@ -1846,7 +1851,7 @@ def _parse_include_license(include_license): def _get_matching_image_ids( - all_classes, + classes_map, images, annotations, image_ids=None, @@ -1862,7 +1867,7 @@ def _get_matching_image_ids( if classes is not None: all_ids, any_ids = _get_images_with_classes( - image_ids, annotations, classes, all_classes + image_ids, annotations, classes, classes_map ) else: all_ids = image_ids @@ -1930,7 +1935,7 @@ def _do_download(args): def _get_images_with_classes( - image_ids, annotations, target_classes, all_classes + image_ids, annotations, target_classes, classes_map ): if annotations is None: logger.warning("Dataset is unlabeled; ignoring classes requirement") @@ -1939,11 +1944,12 @@ def _get_images_with_classes( if etau.is_str(target_classes): target_classes = [target_classes] - bad_classes = [c for c in target_classes if c not in all_classes] + labels_map_rev = {c: i for i, c in classes_map.items()} + + bad_classes = [c for c in target_classes if c not in labels_map_rev] if bad_classes: raise ValueError("Unsupported classes: %s" % bad_classes) - labels_map_rev = _to_labels_map_rev(all_classes) class_ids = {labels_map_rev[c] for c in target_classes} all_ids = [] @@ -2029,7 +2035,7 @@ def _load_image_ids_json(json_path): def _to_labels_map_rev(classes): - return {c: i for i, c in enumerate(classes)} + return {c: i for i, c in enumerate(classes, 1)} def _to_classes(classes_map): diff --git a/fiftyone/utils/quickstart.py b/fiftyone/utils/quickstart.py index 574ab9e2ae..9085f2a31b 100644 --- a/fiftyone/utils/quickstart.py +++ b/fiftyone/utils/quickstart.py @@ -11,9 +11,7 @@ import fiftyone.zoo.datasets as fozd -def quickstart( - video=False, port=None, address=None, remote=False, desktop=None -): +def quickstart(video=False, port=None, address=None, remote=False): """Runs the FiftyOne quickstart. This method loads an interesting dataset from the Dataset Zoo, launches the @@ -27,9 +25,6 @@ def quickstart( ``fiftyone.config.default_app_address`` is used remote (False): whether this is a remote session, and opening the App should not be attempted - desktop (None): whether to launch the App in the browser (False) or as - a desktop App (True). If None, ``fiftyone.config.desktop_app`` is - used. Not applicable to notebook contexts Returns: a tuple containing @@ -39,30 +34,29 @@ def quickstart( the App that was launched """ if video: - return _video_quickstart(port, address, remote, desktop) + return _video_quickstart(port, address, remote) - return _quickstart(port, address, remote, desktop) + return _quickstart(port, address, remote) -def _quickstart(port, address, remote, desktop): +def _quickstart(port, address, remote): print(_QUICKSTART_GUIDE) dataset = fozd.load_zoo_dataset("quickstart") - return _launch_app(dataset, port, address, remote, desktop) + return _launch_app(dataset, port, address, remote) -def _video_quickstart(port, address, remote, desktop): +def _video_quickstart(port, address, remote): print(_VIDEO_QUICKSTART_GUIDE) dataset = fozd.load_zoo_dataset("quickstart-video") - return _launch_app(dataset, port, address, remote, desktop) + return _launch_app(dataset, port, address, remote) -def _launch_app(dataset, port, address, remote, desktop): +def _launch_app(dataset, port, address, remote): session = fos.launch_app( dataset=dataset, port=port, address=address, remote=remote, - desktop=desktop, ) return dataset, session diff --git a/fiftyone/utils/rerun.py b/fiftyone/utils/rerun.py new file mode 100644 index 0000000000..7b72e5601d --- /dev/null +++ b/fiftyone/utils/rerun.py @@ -0,0 +1,25 @@ +""" +Utilities for working with `Rerun `_. + +| Copyright 2017-2024, Voxel51, Inc. +| `voxel51.com `_ +| +""" + +import fiftyone.core.fields as fof +import fiftyone.core.labels as fol +from fiftyone.core.odm import DynamicEmbeddedDocument + + +class RrdFile(DynamicEmbeddedDocument, fol._HasMedia): + """Class for storing a rerun data (rrd) file and its associated metadata. + + Args: + filepath (None): the path to the rrd file + version (None): the version of the rrd file + """ + + _MEDIA_FIELD = "filepath" + + filepath = fof.StringField() + version = fof.StringField() diff --git a/setup.py b/setup.py index f5bc265a82..3652dd44c5 100644 --- a/setup.py +++ b/setup.py @@ -13,7 +13,7 @@ from setuptools import setup, find_packages -VERSION = "1.0.0" +VERSION = "1.0.1" def get_version(): diff --git a/tests/unittests/import_export_tests.py b/tests/unittests/import_export_tests.py index 54798733f5..896429d8a7 100644 --- a/tests/unittests/import_export_tests.py +++ b/tests/unittests/import_export_tests.py @@ -1317,6 +1317,65 @@ def test_coco_detection_dataset(self): {c["id"] for c in categories2}, ) + # Alphabetized 1-based categories by default + + export_dir = self._new_dir() + + dataset.export( + export_dir=export_dir, + dataset_type=fo.types.COCODetectionDataset, + ) + + dataset2 = fo.Dataset.from_dir( + dataset_dir=export_dir, + dataset_type=fo.types.COCODetectionDataset, + label_types="detections", + label_field="predictions", + ) + categories2 = dataset2.info["categories"] + + self.assertListEqual([c["id"] for c in categories2], [1, 2]) + self.assertListEqual([c["name"] for c in categories2], ["cat", "dog"]) + + # Only load matching classes + + export_dir = self._new_dir() + + dataset.export( + export_dir=export_dir, + dataset_type=fo.types.COCODetectionDataset, + ) + + dataset2 = fo.Dataset.from_dir( + dataset_dir=export_dir, + dataset_type=fo.types.COCODetectionDataset, + label_types="detections", + label_field="predictions", + classes="cat", + only_matching=False, + ) + + self.assertEqual(len(dataset2), 2) + self.assertListEqual( + dataset2.distinct("predictions.detections.label"), + ["cat", "dog"], + ) + + dataset3 = fo.Dataset.from_dir( + dataset_dir=export_dir, + dataset_type=fo.types.COCODetectionDataset, + label_types="detections", + label_field="predictions", + classes="cat", + only_matching=True, + ) + + self.assertEqual(len(dataset3), 2) + self.assertListEqual( + dataset3.distinct("predictions.detections.label"), + ["cat"], + ) + @drop_datasets def test_voc_detection_dataset(self): dataset = self._make_dataset() @@ -1758,16 +1817,19 @@ def test_add_yolo_labels(self): @drop_datasets def test_add_coco_labels(self): dataset = self._make_dataset() + classes = dataset.distinct("predictions.detections.label") + categories = [{"id": i, "name": l} for i, l in enumerate(classes, 1)] export_dir = self._new_dir() dataset.export( export_dir=export_dir, dataset_type=fo.types.COCODetectionDataset, + categories=categories, ) coco_labels_path = os.path.join(export_dir, "labels.json") - fouc.add_coco_labels(dataset, "coco", coco_labels_path, classes) + fouc.add_coco_labels(dataset, "coco", coco_labels_path, categories) self.assertEqual( dataset.count_values("predictions.detections.label"), dataset.count_values("coco.detections.label"),