Skip to content

Commit

Permalink
Fixed frames navigation and GT tracks displaying (#8533)
Browse files Browse the repository at this point in the history
  • Loading branch information
bsekachev authored Oct 15, 2024
1 parent f8fe849 commit ac01fff
Show file tree
Hide file tree
Showing 15 changed files with 103 additions and 97 deletions.
6 changes: 6 additions & 0 deletions changelog.d/20241011_142625_sekachev.bs_fixed_navigation.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
### Fixed

- Incorrect navigation by keyframes when annotation job ends earlier than track in a ground truth job
(<https://github.com/cvat-ai/cvat/pull/8533>)
- Tracks from a ground truth job displayed on wrong frames in review mode when frame step is not equal to 1
(<https://github.com/cvat-ai/cvat/pull/8533>)
78 changes: 32 additions & 46 deletions cvat-core/src/frames.ts
Original file line number Diff line number Diff line change
Expand Up @@ -19,8 +19,7 @@ const frameDataCache: Record<string, {
metaFetchedTimestamp: number;
chunkSize: number;
mode: 'annotation' | 'interpolation';
startFrame: number;
stopFrame: number;
jobStartFrame: number;
decodeForward: boolean;
forwardStep: number;
latestFrameDecodeRequest: number | null;
Expand All @@ -44,7 +43,7 @@ const frameMetaCache: Record<string, Promise<FramesMetaData>> = {};
export class FramesMetaData {
public chunkSize: number;
public deletedFrames: Record<number, boolean>;
public includedFrames: number[];
public includedFrames: number[] | null;
public frameFilter: string;
public frames: {
width: number;
Expand Down Expand Up @@ -170,6 +169,14 @@ export class FramesMetaData {
);
}

getDataFrameNumber(jobRelativeFrame: number): number {
return this.frameStep * jobRelativeFrame + this.startFrame;
}

getJobRelativeFrameNumber(dataFrameNumber: number): number {
return (dataFrameNumber - this.startFrame) / this.frameStep;
}

getUpdated(): Record<string, unknown> {
return this.#updateTrigger.getUpdated(this);
}
Expand Down Expand Up @@ -206,7 +213,7 @@ export class FramesMetaData {

getDataFrameNumbers(): number[] {
if (this.includedFrames) {
return this.includedFrames;
return [...this.includedFrames];
}

return range(this.startFrame, this.stopFrame + 1, this.frameStep);
Expand Down Expand Up @@ -326,18 +333,6 @@ class PrefetchAnalyzer {
}
}

function getDataStartFrame(meta: FramesMetaData, localStartFrame: number): number {
return meta.startFrame - localStartFrame * meta.frameStep;
}

function getDataFrameNumber(frameNumber: number, dataStartFrame: number, step: number): number {
return frameNumber * step + dataStartFrame;
}

function getFrameNumber(dataFrameNumber: number, dataStartFrame: number, step: number): number {
return (dataFrameNumber - dataStartFrame) / step;
}

Object.defineProperty(FrameData.prototype.data, 'implementation', {
value(this: FrameData, onServerRequest) {
return new Promise<{
Expand All @@ -346,21 +341,16 @@ Object.defineProperty(FrameData.prototype.data, 'implementation', {
imageData: ImageBitmap | Blob;
} | Blob>((resolve, reject) => {
const {
meta, provider, prefetchAnalyzer, chunkSize, startFrame,
meta, provider, prefetchAnalyzer, chunkSize, jobStartFrame,
decodeForward, forwardStep, decodedBlocksCacheSize,
} = frameDataCache[this.jobID];

const requestId = +_.uniqueId();
const dataStartFrame = getDataStartFrame(meta, startFrame);
const requestedDataFrameNumber = getDataFrameNumber(
this.number, dataStartFrame, meta.frameStep,
);
const requestedDataFrameNumber = meta.getDataFrameNumber(this.number - jobStartFrame);
const chunkIndex = meta.getFrameChunkIndex(requestedDataFrameNumber);
const segmentFrameNumbers = meta.getDataFrameNumbers().map(
(dataFrameNumber: number) => getFrameNumber(
dataFrameNumber, dataStartFrame, meta.frameStep,
),
);
const segmentFrameNumbers = meta.getDataFrameNumbers().map((dataFrameNumber: number) => (
meta.getJobRelativeFrameNumber(dataFrameNumber) + jobStartFrame
));
const frame = provider.frame(this.number);

function findTheNextNotDecodedChunk(currentFrameIndex: number): number | null {
Expand Down Expand Up @@ -581,7 +571,7 @@ async function saveJobMeta(meta: FramesMetaData, jobID: number): Promise<FramesM
}

function getFrameMeta(jobID, frame): SerializedFramesMetaData['frames'][0] {
const { meta, mode, startFrame } = frameDataCache[jobID];
const { meta, mode, jobStartFrame } = frameDataCache[jobID];
let frameMeta = null;
if (mode === 'interpolation' && meta.frames.length === 1) {
// video tasks have 1 frame info, but image tasks will have many infos
Expand All @@ -590,7 +580,7 @@ function getFrameMeta(jobID, frame): SerializedFramesMetaData['frames'][0] {
if (frame > meta.stopFrame) {
throw new ArgumentError(`Meta information about frame ${frame} can't be received from the server`);
}
frameMeta = meta.frames[frame - startFrame];
frameMeta = meta.frames[frame - jobStartFrame];
} else {
throw new DataError(`Invalid mode is specified ${mode}`);
}
Expand Down Expand Up @@ -639,8 +629,8 @@ export function getContextImage(jobID: number, frame: number): Promise<Record<st

const frameData = frameDataCache[jobID];
const requestId = frame;
const { startFrame } = frameData;
const { related_files: relatedFiles } = frameData.meta.frames[frame - startFrame];
const { jobStartFrame } = frameData;
const { related_files: relatedFiles } = frameData.meta.frames[frame - jobStartFrame];

if (relatedFiles === 0) {
resolve({});
Expand Down Expand Up @@ -726,8 +716,7 @@ export async function getFrame(
chunkType: 'video' | 'imageset',
mode: 'interpolation' | 'annotation', // todo: obsolete, need to remove
frame: number,
startFrame: number,
stopFrame: number,
jobStartFrame: number,
isPlaying: boolean,
step: number,
dimension: DimensionType,
Expand All @@ -750,19 +739,16 @@ export async function getFrame(
Math.floor((2048 * 1024 * 1024) / ((mean + stdDev) * 4 * chunkSize)) || 1, 10,
);

// TODO: migrate to local frame numbers
const dataStartFrame = getDataStartFrame(meta, startFrame);
const dataFrameNumberGetter = (frameNumber: number): number => (
getDataFrameNumber(frameNumber, dataStartFrame, meta.frameStep)
meta.getDataFrameNumber(frameNumber - jobStartFrame)
);

frameDataCache[jobID] = {
meta,
metaFetchedTimestamp: Date.now(),
chunkSize,
mode,
startFrame,
stopFrame,
jobStartFrame,
decodeForward: isPlaying,
forwardStep: step,
provider: new FrameDecoder(
Expand Down Expand Up @@ -864,12 +850,13 @@ export async function findFrame(
let lastUndeletedFrame = null;
const check = (frame): boolean => {
if (meta.includedFrames) {
// meta.includedFrames contains input frame numbers now
const dataStartFrame = meta.startFrame; // this is only true when includedFrames is set
return (meta.includedFrames.includes(
getDataFrameNumber(frame, dataStartFrame, meta.frameStep))
// meta.includedFrames contains absolute frame numbers
const jobStartFrame = 0; // this is only true when includedFrames is set
return (
meta.includedFrames.includes(meta.getDataFrameNumber(frame - jobStartFrame))
) && (!filters.notDeleted || !(frame in meta.deletedFrames));
}

if (filters.notDeleted) {
return !(frame in meta.deletedFrames);
}
Expand All @@ -888,23 +875,22 @@ export async function findFrame(
return lastUndeletedFrame;
}

export function getCachedChunks(jobID): number[] {
export function getCachedChunks(jobID: number): number[] {
if (!(jobID in frameDataCache)) {
return [];
}

return frameDataCache[jobID].provider.cachedChunks(true);
}

export function getJobFrameNumbers(jobID): number[] {
export function getJobFrameNumbers(jobID: number): number[] {
if (!(jobID in frameDataCache)) {
return [];
}

const { meta, startFrame } = frameDataCache[jobID];
const dataStartFrame = getDataStartFrame(meta, startFrame);
const { meta, jobStartFrame } = frameDataCache[jobID];
return meta.getDataFrameNumbers().map((dataFrameNumber: number): number => (
getFrameNumber(dataFrameNumber, dataStartFrame, meta.frameStep)
meta.getJobRelativeFrameNumber(dataFrameNumber) + jobStartFrame
));
}

Expand Down
2 changes: 0 additions & 2 deletions cvat-core/src/session-implementation.ts
Original file line number Diff line number Diff line change
Expand Up @@ -200,7 +200,6 @@ export function implementJob(Job: typeof JobClass): typeof JobClass {
this.mode,
frame,
this.startFrame,
this.stopFrame,
isPlaying,
step,
this.dimension,
Expand Down Expand Up @@ -860,7 +859,6 @@ export function implementTask(Task: typeof TaskClass): typeof TaskClass {
this.mode,
frame,
job.startFrame,
job.stopFrame,
isPlaying,
step,
this.dimension,
Expand Down
4 changes: 3 additions & 1 deletion cvat-ui/src/actions/annotation-actions.ts
Original file line number Diff line number Diff line change
Expand Up @@ -651,7 +651,7 @@ export function changeFrameAsync(
return;
}

if (!isAbleToChangeFrame() || statisticsVisible || propagateVisible) {
if (!isAbleToChangeFrame(toFrame) || statisticsVisible || propagateVisible) {
return;
}

Expand Down Expand Up @@ -914,6 +914,7 @@ export function getJobAsync({
}
}

const jobMeta = await cvat.frames.getMeta('job', job.id);
// frame query parameter does not work for GT job
const frameNumber = Number.isInteger(initialFrame) && gtJob?.id !== job.id ?
initialFrame as number :
Expand Down Expand Up @@ -959,6 +960,7 @@ export function getJobAsync({
payload: {
openTime,
job,
jobMeta,
queryParameters,
groundTruthInstance: gtJob || null,
groundTruthJobFramesMeta,
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -316,7 +316,7 @@ function AttributeAnnotationSidebar(props: StateToProps & DispatchToProps): JSX.
if (activeObjectState && activeObjectState.objectType === ObjectType.TRACK) {
const frame =
typeof activeObjectState.keyframes.next === 'number' ? activeObjectState.keyframes.next : null;
if (frame !== null && isAbleToChangeFrame()) {
if (frame !== null && isAbleToChangeFrame(frame)) {
changeFrame(frame);
}
}
Expand All @@ -326,7 +326,7 @@ function AttributeAnnotationSidebar(props: StateToProps & DispatchToProps): JSX.
if (activeObjectState && activeObjectState.objectType === ObjectType.TRACK) {
const frame =
typeof activeObjectState.keyframes.prev === 'number' ? activeObjectState.keyframes.prev : null;
if (frame !== null && isAbleToChangeFrame()) {
if (frame !== null && isAbleToChangeFrame(frame)) {
changeFrame(frame);
}
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -22,7 +22,7 @@ import { EventScope } from 'cvat-logger';
import { Canvas, HighlightSeverity, CanvasHint } from 'cvat-canvas-wrapper';
import { Canvas3d } from 'cvat-canvas3d-wrapper';
import {
AnnotationConflict, FramesMetaData, ObjectState, QualityConflict, getCore,
AnnotationConflict, ObjectState, QualityConflict, getCore,
} from 'cvat-core-wrapper';
import config from 'config';
import CVATTooltip from 'components/common/cvat-tooltip';
Expand Down Expand Up @@ -117,7 +117,6 @@ interface StateToProps {
conflicts: QualityConflict[];
showGroundTruth: boolean;
highlightedConflict: QualityConflict | null;
groundTruthJobFramesMeta: FramesMetaData | null;
imageFilters: ImageFilter[];
activeControl: ActiveControl;
}
Expand Down Expand Up @@ -154,7 +153,7 @@ function mapStateToProps(state: CombinedState): StateToProps {
annotation: {
canvas: { activeControl, instance: canvasInstance, ready: canvasIsReady },
drawing: { activeLabelID, activeObjectType },
job: { instance: jobInstance, groundTruthInfo: { groundTruthJobFramesMeta } },
job: { instance: jobInstance },
player: {
frame: { data: frameData, number: frame },
frameAngles,
Expand Down Expand Up @@ -255,7 +254,6 @@ function mapStateToProps(state: CombinedState): StateToProps {
conflicts,
showGroundTruth,
highlightedConflict,
groundTruthJobFramesMeta,
imageFilters,
};
}
Expand Down Expand Up @@ -940,15 +938,13 @@ class CanvasWrapperComponent extends React.PureComponent<Props> {
private updateCanvas(): void {
const {
curZLayer, annotations, frameData,
workspace, groundTruthJobFramesMeta,
frame, imageFilters,
workspace, frame, imageFilters,
} = this.props;

const { canvasInstance } = this.props as { canvasInstance: Canvas };
if (frameData !== null && canvasInstance) {
const filteredAnnotations = filterAnnotations(annotations, {
frame,
groundTruthJobFramesMeta,
workspace,
exclude: [ObjectType.TAG],
});
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -206,7 +206,7 @@ function TagAnnotationSidebar(props: StateToProps & DispatchToProps): JSX.Elemen
frameNumber + 1,
jobInstance.stopFrame,
);
if (frame !== null && isAbleToChangeFrame()) {
if (frame !== null && isAbleToChangeFrame(frame)) {
changeFrame(frame);
}
};
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -44,17 +44,8 @@ export function getAllocationTableContents(gtJobMeta: FramesMetaData, gtJob: Job
// A workaround for meta "includedFrames" using source data numbers
// TODO: remove once meta is migrated to relative frame numbers

function getDataStartFrame(meta: FramesMetaData, localStartFrame: number): number {
return meta.startFrame - localStartFrame * meta.frameStep;
}

function getFrameNumber(dataFrameNumber: number, dataStartFrame: number, step: number): number {
return (dataFrameNumber - dataStartFrame) / step;
}

const dataStartFrame = getDataStartFrame(gtJobMeta, gtJob.startFrame);
const jobFrameNumbers = gtJobMeta.getDataFrameNumbers().map((dataFrameID: number) => (
getFrameNumber(dataFrameID, dataStartFrame, gtJobMeta.frameStep)
const jobFrameNumbers = gtJobMeta.getDataFrameNumbers().map((dataFrameNumber: number) => (
gtJobMeta.getJobRelativeFrameNumber(dataFrameNumber) + gtJob.startFrame
));

const jobDataSegmentFrameNumbers = range(
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -211,7 +211,7 @@ class ItemButtonsWrapper extends React.PureComponent<StateToProps & DispatchToPr

private changeFrame(frame: number): void {
const { changeFrame } = this.props;
if (isAbleToChangeFrame()) {
if (isAbleToChangeFrame(frame)) {
changeFrame(frame);
}
}
Expand Down
Loading

0 comments on commit ac01fff

Please sign in to comment.