From bad4d6a38254eca3915f48bf6f1fa962d7ee28c4 Mon Sep 17 00:00:00 2001 From: Boris Sekachev Date: Tue, 12 Sep 2023 12:22:12 +0300 Subject: [PATCH 1/6] Do not reload the page when changing job state --- cvat-core/src/session-implementation.ts | 6 +++++- cvat-ui/src/actions/annotation-actions.ts | 2 +- cvat-ui/src/actions/jobs-actions.ts | 19 ++++++++++++++++++ cvat-ui/src/actions/tasks-actions.ts | 20 ------------------- .../components/job-item/job-actions-menu.tsx | 3 +-- .../top-bar/annotation-menu.tsx | 3 +-- cvat-ui/src/reducers/annotation-reducer.ts | 10 ++++++++++ cvat-ui/src/reducers/notifications-reducer.ts | 16 +++++++++++++++ 8 files changed, 53 insertions(+), 26 deletions(-) diff --git a/cvat-core/src/session-implementation.ts b/cvat-core/src/session-implementation.ts index 6a064722aa2e..f84d08f41209 100644 --- a/cvat-core/src/session-implementation.ts +++ b/cvat-core/src/session-implementation.ts @@ -71,8 +71,12 @@ export function implementJob(Job) { } const data = await serverProxy.jobs.save(this.id, jobData); + const updatedJob = new Job(data); + this.stage = updatedJob.stage; + this.state = updatedJob.state; + this.assignee = updatedJob.assignee; this._updateTrigger.reset(); - return new Job(data); + return this; } const jobSpec = { diff --git a/cvat-ui/src/actions/annotation-actions.ts b/cvat-ui/src/actions/annotation-actions.ts index 414dd911c5e0..076a35f3f04f 100644 --- a/cvat-ui/src/actions/annotation-actions.ts +++ b/cvat-ui/src/actions/annotation-actions.ts @@ -29,7 +29,7 @@ import { ShapeType, Workspace, } from 'reducers'; -import { updateJobAsync } from './tasks-actions'; +import { updateJobAsync } from './jobs-actions'; import { switchToolsBlockerState } from './settings-actions'; interface AnnotationsParameters { diff --git a/cvat-ui/src/actions/jobs-actions.ts b/cvat-ui/src/actions/jobs-actions.ts index 4386f3d8c0b3..22453aef0fe5 100644 --- a/cvat-ui/src/actions/jobs-actions.ts +++ b/cvat-ui/src/actions/jobs-actions.ts @@ -19,6 +19,8 @@ export enum JobsActionTypes { GET_JOB_PREVIEW_SUCCESS = 'GET_JOB_PREVIEW_SUCCESS', GET_JOB_PREVIEW_FAILED = 'GET_JOB_PREVIEW_FAILED', CREATE_JOB_FAILED = 'CREATE_JOB_FAILED', + UPDATE_JOB_SUCCESS = 'UPDATE_JOB_SUCCESS', + UPDATE_JOB_FAILED = 'UPDATE_JOB_FAILED', DELETE_JOB = 'DELETE_JOB', DELETE_JOB_SUCCESS = 'DELETE_JOB_SUCCESS', DELETE_JOB_FAILED = 'DELETE_JOB_FAILED', @@ -46,6 +48,12 @@ const jobsActions = { createJobFailed: (error: any) => ( createAction(JobsActionTypes.CREATE_JOB_FAILED, { error }) ), + updateJobSuccess: (job: Job) => ( + createAction(JobsActionTypes.UPDATE_JOB_SUCCESS, { job }) + ), + updateJobFailed: (jobID: number, error: any) => ( + createAction(JobsActionTypes.UPDATE_JOB_FAILED, { jobID, error }) + ), deleteJob: (jobID: number) => ( createAction(JobsActionTypes.DELETE_JOB, { jobID }) ), @@ -93,6 +101,17 @@ export const createJobAsync = (data: JobData): ThunkAction => async (dispatch) = } }; +export function updateJobAsync(jobInstance: Job): ThunkAction { + return async (dispatch): Promise => { + try { + const updated = await jobInstance.save(); + dispatch(jobsActions.updateJobSuccess(updated)); + } catch (error) { + dispatch(jobsActions.updateJobFailed(jobInstance.id, error)); + } + }; +} + export const deleteJobAsync = (job: Job): ThunkAction => async (dispatch) => { dispatch(jobsActions.deleteJob(job.id)); try { diff --git a/cvat-ui/src/actions/tasks-actions.ts b/cvat-ui/src/actions/tasks-actions.ts index 3ab967e1a458..e5f9e5edcaf8 100644 --- a/cvat-ui/src/actions/tasks-actions.ts +++ b/cvat-ui/src/actions/tasks-actions.ts @@ -22,7 +22,6 @@ export enum TasksActionTypes { DELETE_TASK_SUCCESS = 'DELETE_TASK_SUCCESS', DELETE_TASK_FAILED = 'DELETE_TASK_FAILED', CREATE_TASK_FAILED = 'CREATE_TASK_FAILED', - UPDATE_JOB_FAILED = 'UPDATE_JOB_FAILED', SWITCH_MOVE_TASK_MODAL_VISIBLE = 'SWITCH_MOVE_TASK_MODAL_VISIBLE', GET_TASK_PREVIEW = 'GET_TASK_PREVIEW', GET_TASK_PREVIEW_SUCCESS = 'GET_TASK_PREVIEW_SUCCESS', @@ -294,25 +293,6 @@ ThunkAction, {}, {}, AnyAction> { }; } -function updateJobFailed(jobID: number, error: any): AnyAction { - const action = { - type: TasksActionTypes.UPDATE_JOB_FAILED, - payload: { jobID, error }, - }; - - return action; -} - -export function updateJobAsync(jobInstance: any): ThunkAction, {}, {}, AnyAction> { - return async (dispatch: ActionCreator): Promise => { - try { - await jobInstance.save(); - } catch (error) { - dispatch(updateJobFailed(jobInstance.id, error)); - } - }; -} - export function switchMoveTaskModalVisible(visible: boolean, taskId: number | null = null): AnyAction { const action = { type: TasksActionTypes.SWITCH_MOVE_TASK_MODAL_VISIBLE, diff --git a/cvat-ui/src/components/job-item/job-actions-menu.tsx b/cvat-ui/src/components/job-item/job-actions-menu.tsx index e7911dd05541..fb966b1c36f9 100644 --- a/cvat-ui/src/components/job-item/job-actions-menu.tsx +++ b/cvat-ui/src/components/job-item/job-actions-menu.tsx @@ -13,9 +13,8 @@ import { exportActions } from 'actions/export-actions'; import { Job, JobStage, JobType, getCore, } from 'cvat-core-wrapper'; -import { deleteJobAsync } from 'actions/jobs-actions'; +import { deleteJobAsync, updateJobAsync } from 'actions/jobs-actions'; import { importActions } from 'actions/import-actions'; -import { updateJobAsync } from 'actions/tasks-actions'; const core = getCore(); diff --git a/cvat-ui/src/containers/annotation-page/top-bar/annotation-menu.tsx b/cvat-ui/src/containers/annotation-page/top-bar/annotation-menu.tsx index 0e9908c71a92..e94108e2f18e 100644 --- a/cvat-ui/src/containers/annotation-page/top-bar/annotation-menu.tsx +++ b/cvat-ui/src/containers/annotation-page/top-bar/annotation-menu.tsx @@ -11,7 +11,7 @@ import { MenuInfo } from 'rc-menu/lib/interface'; import { CombinedState, JobStage } from 'reducers'; import AnnotationMenuComponent, { Actions } from 'components/annotation-page/top-bar/annotation-menu'; -import { updateJobAsync } from 'actions/tasks-actions'; +import { updateJobAsync } from 'actions/jobs-actions'; import { saveAnnotationsAsync, setForceExitAnnotationFlag as setForceExitAnnotationFlagAction, @@ -110,7 +110,6 @@ function AnnotationMenuContainer(props: Props): JSX.Element { } else if (action.startsWith('state:')) { [, jobInstance.state] = action.split(':'); updateJob(jobInstance); - window.location.reload(); } else if (action === Actions.LOAD_JOB_ANNO) { showImportModal(jobInstance); } diff --git a/cvat-ui/src/reducers/annotation-reducer.ts b/cvat-ui/src/reducers/annotation-reducer.ts index d7d395adb6f1..2b9a1b287da7 100644 --- a/cvat-ui/src/reducers/annotation-reducer.ts +++ b/cvat-ui/src/reducers/annotation-reducer.ts @@ -5,6 +5,7 @@ import { AnyAction } from 'redux'; import { AnnotationActionTypes } from 'actions/annotation-actions'; +import { JobsActionTypes } from 'actions/jobs-actions'; import { AuthActionTypes } from 'actions/auth-actions'; import { BoundariesActionTypes } from 'actions/boundaries-actions'; import { Canvas, CanvasMode } from 'cvat-canvas-wrapper'; @@ -237,6 +238,15 @@ export default (state = defaultState, action: AnyAction): AnnotationState => { }, }; } + case JobsActionTypes.UPDATE_JOB_SUCCESS: { + return { + ...state, + job: { + ...state.job, + instance: action.payload.job, + }, + }; + } case AnnotationActionTypes.GET_DATA_FAILED: { return { ...state, diff --git a/cvat-ui/src/reducers/notifications-reducer.ts b/cvat-ui/src/reducers/notifications-reducer.ts index 8abb82d7cc5c..4190eb38fb44 100644 --- a/cvat-ui/src/reducers/notifications-reducer.ts +++ b/cvat-ui/src/reducers/notifications-reducer.ts @@ -1563,6 +1563,22 @@ export default function (state = defaultState, action: AnyAction): Notifications }, }; } + case JobsActionTypes.UPDATE_JOB_FAILED: { + return { + ...state, + errors: { + ...state.errors, + jobs: { + ...state.errors.jobs, + updating: { + message: 'Could not update job', + reason: action.payload.error.toString(), + className: 'cvat-notification-notice-update-job-failed', + }, + }, + }, + }; + } case JobsActionTypes.DELETE_JOB_FAILED: { const { jobID } = action.payload; return { From 102a29e418d1e1cb45f48a8232453dad93fa5d11 Mon Sep 17 00:00:00 2001 From: Boris Sekachev Date: Wed, 13 Sep 2023 11:11:50 +0300 Subject: [PATCH 2/6] Some refactoring applied --- cvat-core/src/session-implementation.ts | 20 +++++--- cvat-core/src/session.ts | 5 +- cvat-ui/src/actions/jobs-actions.ts | 7 ++- .../top-bar/annotation-menu.tsx | 5 +- .../top-bar/annotation-menu.tsx | 51 ++++++++++++------- cvat-ui/src/reducers/annotation-reducer.ts | 5 +- cvat-ui/src/reducers/index.ts | 6 --- 7 files changed, 59 insertions(+), 40 deletions(-) diff --git a/cvat-core/src/session-implementation.ts b/cvat-core/src/session-implementation.ts index f84d08f41209..d21422477744 100644 --- a/cvat-core/src/session-implementation.ts +++ b/cvat-core/src/session-implementation.ts @@ -70,12 +70,20 @@ export function implementJob(Job) { jobData.assignee = jobData.assignee.id; } - const data = await serverProxy.jobs.save(this.id, jobData); - const updatedJob = new Job(data); - this.stage = updatedJob.stage; - this.state = updatedJob.state; - this.assignee = updatedJob.assignee; - this._updateTrigger.reset(); + let updatedJob = null; + try { + const data = await serverProxy.jobs.save(this.id, jobData); + updatedJob = new Job(data); + this._updateTrigger.reset(); + } catch (error) { + updatedJob = new Job(this._initialData); + throw error; + } finally { + this.stage = updatedJob.stage; + this.state = updatedJob.state; + this.assignee = updatedJob.assignee; + } + return this; } diff --git a/cvat-core/src/session.ts b/cvat-core/src/session.ts index 1c9189903868..5fc657fff922 100644 --- a/cvat-core/src/session.ts +++ b/cvat-core/src/session.ts @@ -381,7 +381,7 @@ export class Job extends Session { log: CallableFunction; }; - constructor(initialData: SerializedJob) { + constructor(initialData: Readonly) { super(); const data = { id: undefined, @@ -536,6 +536,9 @@ export class Job extends Session { _updateTrigger: { get: () => updateTrigger, }, + _initialData: { + get: () => initialData, + }, }), ); diff --git a/cvat-ui/src/actions/jobs-actions.ts b/cvat-ui/src/actions/jobs-actions.ts index 22453aef0fe5..9126dfba9e0f 100644 --- a/cvat-ui/src/actions/jobs-actions.ts +++ b/cvat-ui/src/actions/jobs-actions.ts @@ -101,14 +101,17 @@ export const createJobAsync = (data: JobData): ThunkAction => async (dispatch) = } }; -export function updateJobAsync(jobInstance: Job): ThunkAction { - return async (dispatch): Promise => { +export function updateJobAsync(jobInstance: Job): ThunkAction> { + return async (dispatch): Promise => { try { const updated = await jobInstance.save(); dispatch(jobsActions.updateJobSuccess(updated)); } catch (error) { dispatch(jobsActions.updateJobFailed(jobInstance.id, error)); + return false; } + + return true; }; } diff --git a/cvat-ui/src/components/annotation-page/top-bar/annotation-menu.tsx b/cvat-ui/src/components/annotation-page/top-bar/annotation-menu.tsx index 75d608140161..d187f78e7b4b 100644 --- a/cvat-ui/src/components/annotation-page/top-bar/annotation-menu.tsx +++ b/cvat-ui/src/components/annotation-page/top-bar/annotation-menu.tsx @@ -16,8 +16,7 @@ import Collapse from 'antd/lib/collapse'; // eslint-disable-next-line import/no-extraneous-dependencies import { MenuInfo } from 'rc-menu/lib/interface'; import CVATTooltip from 'components/common/cvat-tooltip'; -import { getCore } from 'cvat-core-wrapper'; -import { JobStage } from 'reducers'; +import { getCore, JobStage } from 'cvat-core-wrapper'; const core = getCore(); @@ -213,7 +212,7 @@ function AnnotationMenuComponent(props: Props & RouteComponentProps): JSX.Elemen {JobState.COMPLETED} - {[JobStage.ANNOTATION, JobStage.REVIEW].includes(jobStage) ? + {[JobStage.ANNOTATION, JobStage.VALIDATION].includes(jobStage) ? Finish the job : null} {jobStage === JobStage.ACCEPTANCE ? Renew the job : null} diff --git a/cvat-ui/src/containers/annotation-page/top-bar/annotation-menu.tsx b/cvat-ui/src/containers/annotation-page/top-bar/annotation-menu.tsx index e94108e2f18e..c9580f9bd14c 100644 --- a/cvat-ui/src/containers/annotation-page/top-bar/annotation-menu.tsx +++ b/cvat-ui/src/containers/annotation-page/top-bar/annotation-menu.tsx @@ -1,5 +1,5 @@ // Copyright (C) 2020-2022 Intel Corporation -// Copyright (C) 2022 CVAT.ai Corporation +// Copyright (C) 2022-2023 CVAT.ai Corporation // // SPDX-License-Identifier: MIT @@ -9,7 +9,7 @@ import { connect } from 'react-redux'; // eslint-disable-next-line import/no-extraneous-dependencies import { MenuInfo } from 'rc-menu/lib/interface'; -import { CombinedState, JobStage } from 'reducers'; +import { CombinedState } from 'reducers'; import AnnotationMenuComponent, { Actions } from 'components/annotation-page/top-bar/annotation-menu'; import { updateJobAsync } from 'actions/jobs-actions'; import { @@ -19,22 +19,25 @@ import { } from 'actions/annotation-actions'; import { exportActions } from 'actions/export-actions'; import { importActions } from 'actions/import-actions'; -import { getCore } from 'cvat-core-wrapper'; +import { + getCore, Job, JobStage, JobState, +} from 'cvat-core-wrapper'; +import { message } from 'antd'; const core = getCore(); interface StateToProps { - jobInstance: any; + jobInstance: Job; stopFrame: number; } interface DispatchToProps { - showExportModal: (jobInstance: any) => void; - showImportModal: (jobInstance: any) => void; + showExportModal: (jobInstance: Job) => void; + showImportModal: (jobInstance: Job) => void; removeAnnotations(startnumber: number, endnumber: number, delTrackKeyframesOnly: boolean): void; setForceExitAnnotationFlag(forceExit: boolean): void; - saveAnnotations(jobInstance: any, afterSave?: () => void): void; - updateJob(jobInstance: any): void; + saveAnnotations(jobInstance: Job, afterSave?: () => void): void; + updateJob(jobInstance: Job): Promise; } function mapStateToProps(state: CombinedState): StateToProps { @@ -55,10 +58,10 @@ function mapStateToProps(state: CombinedState): StateToProps { function mapDispatchToProps(dispatch: any): DispatchToProps { return { - showExportModal(jobInstance: any): void { + showExportModal(jobInstance: Job): void { dispatch(exportActions.openExportDatasetModal(jobInstance)); }, - showImportModal(jobInstance: any): void { + showImportModal(jobInstance: Job): void { dispatch(importActions.openImportDatasetModal(jobInstance)); }, removeAnnotations(startnumber: number, endnumber: number, delTrackKeyframesOnly:boolean) { @@ -67,11 +70,11 @@ function mapDispatchToProps(dispatch: any): DispatchToProps { setForceExitAnnotationFlag(forceExit: boolean): void { dispatch(setForceExitAnnotationFlagAction(forceExit)); }, - saveAnnotations(jobInstance: any, afterSave?: () => void): void { + saveAnnotations(jobInstance: Job, afterSave?: () => void): void { dispatch(saveAnnotationsAsync(jobInstance, afterSave)); }, - updateJob(jobInstance: any): void { - dispatch(updateJobAsync(jobInstance)); + updateJob(jobInstance: Job): Promise { + return dispatch(updateJobAsync(jobInstance)); }, }; } @@ -98,18 +101,28 @@ function AnnotationMenuContainer(props: Props): JSX.Element { } else if (action === Actions.RENEW_JOB) { jobInstance.state = core.enums.JobState.NEW; jobInstance.stage = JobStage.ANNOTATION; - updateJob(jobInstance); - window.location.reload(); + updateJob(jobInstance).then((success) => { + if (success) { + message.info('Job renewed', 2); + } + }); } else if (action === Actions.FINISH_JOB) { jobInstance.stage = JobStage.ACCEPTANCE; jobInstance.state = core.enums.JobState.COMPLETED; - updateJob(jobInstance); - history.push(`/tasks/${jobInstance.taskId}`); + updateJob(jobInstance).then((success) => { + if (success) { + history.push(`/tasks/${jobInstance.taskId}`); + } + }); } else if (action === Actions.OPEN_TASK) { history.push(`/tasks/${jobInstance.taskId}`); } else if (action.startsWith('state:')) { - [, jobInstance.state] = action.split(':'); - updateJob(jobInstance); + [, jobInstance.state] = action.split(':') as [string, JobState]; + updateJob(jobInstance).then((success) => { + if (success) { + message.info('Job state updated', 2); + } + }); } else if (action === Actions.LOAD_JOB_ANNO) { showImportModal(jobInstance); } diff --git a/cvat-ui/src/reducers/annotation-reducer.ts b/cvat-ui/src/reducers/annotation-reducer.ts index 2b9a1b287da7..ebe4c833d8da 100644 --- a/cvat-ui/src/reducers/annotation-reducer.ts +++ b/cvat-ui/src/reducers/annotation-reducer.ts @@ -10,7 +10,7 @@ import { AuthActionTypes } from 'actions/auth-actions'; import { BoundariesActionTypes } from 'actions/boundaries-actions'; import { Canvas, CanvasMode } from 'cvat-canvas-wrapper'; import { Canvas3d } from 'cvat-canvas3d-wrapper'; -import { DimensionType } from 'cvat-core-wrapper'; +import { DimensionType, JobStage } from 'cvat-core-wrapper'; import { clamp } from 'utils/math'; import { SettingsActionTypes } from 'actions/settings-actions'; @@ -18,7 +18,6 @@ import { ActiveControl, AnnotationState, ContextMenuType, - JobStage, ObjectType, ShapeType, Workspace, @@ -159,7 +158,7 @@ export default (state = defaultState, action: AnyAction): AnnotationState => { groundTruthJobFramesMeta, } = action.payload; - const isReview = job.stage === JobStage.REVIEW; + const isReview = job.stage === JobStage.VALIDATION; let workspaceSelected = Workspace.STANDARD; const defaultLabel = job.labels.length ? job.labels[0] : null; diff --git a/cvat-ui/src/reducers/index.ts b/cvat-ui/src/reducers/index.ts index 1ecc25f23d84..3522cb4b9ada 100644 --- a/cvat-ui/src/reducers/index.ts +++ b/cvat-ui/src/reducers/index.ts @@ -375,12 +375,6 @@ export enum TaskStatus { COMPLETED = 'completed', } -export enum JobStage { - ANNOTATION = 'annotation', - REVIEW = 'validation', - ACCEPTANCE = 'acceptance', -} - export interface ActiveInference { status: RQStatus; progress: number; From d75268ef22493aed687baf3ae0b1a81551acf916 Mon Sep 17 00:00:00 2001 From: Boris Sekachev Date: Wed, 13 Sep 2023 11:13:33 +0300 Subject: [PATCH 3/6] Run CI From cd166d8a3c1663c5e1f8ff8390b8c5b96667459e Mon Sep 17 00:00:00 2001 From: Boris Sekachev Date: Thu, 14 Sep 2023 09:33:46 +0300 Subject: [PATCH 4/6] Updated version & changelog --- CHANGELOG.md | 2 +- cvat-ui/package.json | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 2246700a217c..c0a4d88d9a19 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -11,7 +11,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 - Admin actions for easy activation/deactivation of users () ### Changed -- TDB +- Do not reload annotation view when renew the job or update job state () ### Deprecated - TDB diff --git a/cvat-ui/package.json b/cvat-ui/package.json index 900b7c1e98ca..5266d6798848 100644 --- a/cvat-ui/package.json +++ b/cvat-ui/package.json @@ -1,6 +1,6 @@ { "name": "cvat-ui", - "version": "1.56.0", + "version": "1.56.1", "description": "CVAT single-page application", "main": "src/index.tsx", "scripts": { From 99fd1108560fb3b79753f0e736f9af673f106f4b Mon Sep 17 00:00:00 2001 From: Boris Sekachev Date: Thu, 14 Sep 2023 15:09:14 +0300 Subject: [PATCH 5/6] Adjusted on jobs page --- cvat-ui/src/actions/jobs-actions.ts | 5 ++++ .../components/job-item/job-actions-menu.tsx | 17 ++++-------- cvat-ui/src/components/job-item/job-item.tsx | 2 +- cvat-ui/src/components/jobs-page/job-card.tsx | 11 +++++--- .../src/components/jobs-page/jobs-content.tsx | 10 +++++-- .../src/components/jobs-page/jobs-page.tsx | 11 +++++--- cvat-ui/src/reducers/jobs-reducer.ts | 26 +++++++++++++++++++ 7 files changed, 60 insertions(+), 22 deletions(-) diff --git a/cvat-ui/src/actions/jobs-actions.ts b/cvat-ui/src/actions/jobs-actions.ts index 9126dfba9e0f..37cba5999da7 100644 --- a/cvat-ui/src/actions/jobs-actions.ts +++ b/cvat-ui/src/actions/jobs-actions.ts @@ -19,6 +19,7 @@ export enum JobsActionTypes { GET_JOB_PREVIEW_SUCCESS = 'GET_JOB_PREVIEW_SUCCESS', GET_JOB_PREVIEW_FAILED = 'GET_JOB_PREVIEW_FAILED', CREATE_JOB_FAILED = 'CREATE_JOB_FAILED', + UPDATE_JOB = 'UPDATE_JOB', UPDATE_JOB_SUCCESS = 'UPDATE_JOB_SUCCESS', UPDATE_JOB_FAILED = 'UPDATE_JOB_FAILED', DELETE_JOB = 'DELETE_JOB', @@ -48,6 +49,9 @@ const jobsActions = { createJobFailed: (error: any) => ( createAction(JobsActionTypes.CREATE_JOB_FAILED, { error }) ), + updateJob: () => ( + createAction(JobsActionTypes.UPDATE_JOB) + ), updateJobSuccess: (job: Job) => ( createAction(JobsActionTypes.UPDATE_JOB_SUCCESS, { job }) ), @@ -104,6 +108,7 @@ export const createJobAsync = (data: JobData): ThunkAction => async (dispatch) = export function updateJobAsync(jobInstance: Job): ThunkAction> { return async (dispatch): Promise => { try { + dispatch(jobsActions.updateJob()); const updated = await jobInstance.save(); dispatch(jobsActions.updateJobSuccess(updated)); } catch (error) { diff --git a/cvat-ui/src/components/job-item/job-actions-menu.tsx b/cvat-ui/src/components/job-item/job-actions-menu.tsx index fb966b1c36f9..ed5646f3038a 100644 --- a/cvat-ui/src/components/job-item/job-actions-menu.tsx +++ b/cvat-ui/src/components/job-item/job-actions-menu.tsx @@ -1,6 +1,7 @@ // Copyright (C) 2023 CVAT.ai Corporation // // SPDX-License-Identifier: MIT + import React, { useCallback } from 'react'; import { useDispatch } from 'react-redux'; import { useHistory } from 'react-router'; @@ -13,14 +14,14 @@ import { exportActions } from 'actions/export-actions'; import { Job, JobStage, JobType, getCore, } from 'cvat-core-wrapper'; -import { deleteJobAsync, updateJobAsync } from 'actions/jobs-actions'; +import { deleteJobAsync } from 'actions/jobs-actions'; import { importActions } from 'actions/import-actions'; const core = getCore(); interface Props { job: Job; - onJobUpdate?: (job: Job) => void; + onJobUpdate: (job: Job) => void; } function JobActionsMenu(props: Props): JSX.Element { @@ -61,19 +62,11 @@ function JobActionsMenu(props: Props): JSX.Element { } else if (action.key === 'renew_job') { job.state = core.enums.JobState.NEW; job.stage = JobStage.ANNOTATION; - if (onJobUpdate) { - onJobUpdate(job); - } else { - dispatch(updateJobAsync(job)); - } + onJobUpdate(job); } else if (action.key === 'finish_job') { job.stage = JobStage.ACCEPTANCE; job.state = core.enums.JobState.COMPLETED; - if (onJobUpdate) { - onJobUpdate(job); - } else { - dispatch(updateJobAsync(job)); - } + onJobUpdate(job); } }} > diff --git a/cvat-ui/src/components/job-item/job-item.tsx b/cvat-ui/src/components/job-item/job-item.tsx index 614d058d3426..abdb6d8fb5f8 100644 --- a/cvat-ui/src/components/job-item/job-item.tsx +++ b/cvat-ui/src/components/job-item/job-item.tsx @@ -30,7 +30,7 @@ import CVATTooltip from 'components/common/cvat-tooltip'; import JobActionsMenu from './job-actions-menu'; interface Props { - job: Job, + job: Job; task: Task, onJobUpdate: (job: Job) => void; } diff --git a/cvat-ui/src/components/jobs-page/job-card.tsx b/cvat-ui/src/components/jobs-page/job-card.tsx index 112662aa6706..7177a33e2858 100644 --- a/cvat-ui/src/components/jobs-page/job-card.tsx +++ b/cvat-ui/src/components/jobs-page/job-card.tsx @@ -1,5 +1,5 @@ // Copyright (C) 2022 Intel Corporation -// Copyright (C) 2022 CVAT.ai Corporation +// Copyright (C) 2022-2023 CVAT.ai Corporation // // SPDX-License-Identifier: MIT @@ -9,6 +9,8 @@ import Card from 'antd/lib/card'; import Descriptions from 'antd/lib/descriptions'; import { MoreOutlined } from '@ant-design/icons'; import Dropdown from 'antd/lib/dropdown'; + +import { Job } from 'cvat-core-wrapper'; import { useCardHeightHOC } from 'utils/hooks'; import Preview from 'components/common/preview'; import JobActionsMenu from 'components/job-item/job-actions-menu'; @@ -22,11 +24,12 @@ const useCardHeight = useCardHeightHOC({ }); interface Props { - job: any; + job: Job; + onJobUpdate: (job: Job) => void; } function JobCardComponent(props: Props): JSX.Element { - const { job } = props; + const { job, onJobUpdate } = props; const [expanded, setExpanded] = useState(false); const history = useHistory(); const height = useCardHeight(); @@ -73,7 +76,7 @@ function JobCardComponent(props: Props): JSX.Element { {job.assignee.username} ) : null} - }> + }> diff --git a/cvat-ui/src/components/jobs-page/jobs-content.tsx b/cvat-ui/src/components/jobs-page/jobs-content.tsx index db34eece8cfd..e9a4e266dab3 100644 --- a/cvat-ui/src/components/jobs-page/jobs-content.tsx +++ b/cvat-ui/src/components/jobs-page/jobs-content.tsx @@ -1,4 +1,5 @@ // Copyright (C) 2022 Intel Corporation +// Copyright (C) 2023 CVAT.ai Corporation // // SPDX-License-Identifier: MIT @@ -9,7 +10,12 @@ import { CombinedState } from 'reducers'; import { Job, JobType } from 'cvat-core-wrapper'; import JobCard from './job-card'; -function JobsContentComponent(): JSX.Element { +interface Props { + onJobUpdate(job: Job): void; +} + +function JobsContentComponent(props: Props): JSX.Element { + const { onJobUpdate } = props; const jobs = useSelector((state: CombinedState) => state.jobs.current); const dimensions = { md: 22, @@ -22,7 +28,7 @@ function JobsContentComponent(): JSX.Element { {jobs.filter((job: Job) => job.type === JobType.ANNOTATION).map((job: Job): JSX.Element => ( - + ))} diff --git a/cvat-ui/src/components/jobs-page/jobs-page.tsx b/cvat-ui/src/components/jobs-page/jobs-page.tsx index 0a6dea86241e..899075163d38 100644 --- a/cvat-ui/src/components/jobs-page/jobs-page.tsx +++ b/cvat-ui/src/components/jobs-page/jobs-page.tsx @@ -1,9 +1,10 @@ // Copyright (C) 2022 Intel Corporation +// Copyright (C) 2023 CVAT.ai Corporation // // SPDX-License-Identifier: MIT import './styles.scss'; -import React, { useEffect, useState } from 'react'; +import React, { useCallback, useEffect, useState } from 'react'; import { useHistory } from 'react-router'; import { useDispatch, useSelector } from 'react-redux'; import Spin from 'antd/lib/spin'; @@ -12,9 +13,10 @@ import Pagination from 'antd/lib/pagination'; import Empty from 'antd/lib/empty'; import Text from 'antd/lib/typography/Text'; +import { Job } from 'cvat-core-wrapper'; import { updateHistoryFromQuery } from 'components/resource-sorting-filtering'; import { CombinedState, Indexable } from 'reducers'; -import { getJobsAsync } from 'actions/jobs-actions'; +import { getJobsAsync, updateJobAsync } from 'actions/jobs-actions'; import TopBarComponent from './top-bar'; import JobsContentComponent from './jobs-content'; @@ -26,6 +28,9 @@ function JobsPageComponent(): JSX.Element { const query = useSelector((state: CombinedState) => state.jobs.query); const fetching = useSelector((state: CombinedState) => state.jobs.fetching); const count = useSelector((state: CombinedState) => state.jobs.count); + const onJobUpdate = useCallback((job: Job) => { + dispatch(updateJobAsync(job)); + }, []); const queryParams = new URLSearchParams(history.location.search); const updatedQuery = { ...query }; @@ -51,7 +56,7 @@ function JobsPageComponent(): JSX.Element { const content = count ? ( <> - + { + if (job === action.payload.job) { + return action.payload.job; + } + return job; + }) + ) : state.current, + fetching: false, + }; + } + case JobsActionTypes.UPDATE_JOB_FAILED: { + return { + ...state, + fetching: false, + }; + } default: { return state; } From 77a694085cd0595c4f6cbf77c1da5388d8c14d4c Mon Sep 17 00:00:00 2001 From: Boris Sekachev Date: Thu, 14 Sep 2023 17:21:17 +0300 Subject: [PATCH 6/6] Update cvat-ui/src/components/job-item/job-item.tsx Co-authored-by: Kirill Lakhov --- cvat-ui/src/components/job-item/job-item.tsx | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/cvat-ui/src/components/job-item/job-item.tsx b/cvat-ui/src/components/job-item/job-item.tsx index abdb6d8fb5f8..312e65aa2b56 100644 --- a/cvat-ui/src/components/job-item/job-item.tsx +++ b/cvat-ui/src/components/job-item/job-item.tsx @@ -31,7 +31,7 @@ import JobActionsMenu from './job-actions-menu'; interface Props { job: Job; - task: Task, + task: Task; onJobUpdate: (job: Job) => void; }