From d566670842fb8324eb04c45ec0435c3c6dfdffba Mon Sep 17 00:00:00 2001 From: Dmitry Kalinin Date: Fri, 21 May 2021 11:48:31 +0300 Subject: [PATCH 1/8] Tasks loading only when needed --- cvat-core/src/api-implementation.js | 1 + cvat-core/src/project.js | 31 ++++++++ cvat-ui/src/actions/projects-actions.ts | 76 +++++++++++-------- .../components/projects-page/project-item.tsx | 30 +++----- .../components/projects-page/project-list.tsx | 20 ++--- cvat-ui/src/reducers/interfaces.ts | 7 +- cvat-ui/src/reducers/projects-reducer.ts | 11 ++- cvat/apps/engine/serializers.py | 8 +- 8 files changed, 114 insertions(+), 70 deletions(-) diff --git a/cvat-core/src/api-implementation.js b/cvat-core/src/api-implementation.js index 5e9ff610ca52..69945a8b5988 100644 --- a/cvat-core/src/api-implementation.js +++ b/cvat-core/src/api-implementation.js @@ -238,6 +238,7 @@ // prettier-ignore const projects = projectsData.map((project) => { if (filter.withoutTasks) { + project.task_ids = project.tasks; project.tasks = []; } return project; diff --git a/cvat-core/src/project.js b/cvat-core/src/project.js index 0e2b54789141..6d068ae8cbc8 100644 --- a/cvat-core/src/project.js +++ b/cvat-core/src/project.js @@ -8,6 +8,7 @@ const { ArgumentError } = require('./exceptions'); const { Task } = require('./session'); const { Label } = require('./labels'); + const { getPreview } = require('./frames'); const User = require('./user'); /** @@ -34,6 +35,7 @@ updated_date: undefined, task_subsets: undefined, training_project: undefined, + task_ids: undefined, }; for (const property in data) { @@ -254,6 +256,22 @@ ); } + /** + * Get the first frame of the first task of a project for preview + * @method preview + * @memberof Project + * @returns {string} - jpeg encoded image + * @instance + * @async + * @throws {module:API.cvat.exceptions.PluginError} + * @throws {module:API.cvat.exceptions.ServerError} + * @throws {module:API.cvat.exceptions.ArgumentError} + */ + async preview() { + const result = await PluginRegistry.apiWrapper.call(this, Project.prototype.preview); + return result; + } + /** * Method updates data of a created project or creates new project from scratch * @method save @@ -331,4 +349,17 @@ const result = await serverProxy.projects.delete(this.id); return result; }; + + Project.prototype.preview.implementation = async function () { + let taskId; + if (this.tasks.length) { + taskId = this.tasks[0].id; + } else if (Array.isArray(this._internalData.task_ids) && this._internalData.task_ids.length) { + [taskId] = this._internalData.task_ids; + } else { + return ''; + } + const frameData = await getPreview(taskId); + return frameData; + }; })(); diff --git a/cvat-ui/src/actions/projects-actions.ts b/cvat-ui/src/actions/projects-actions.ts index 5408a42fbe17..cc1b89e59c3e 100644 --- a/cvat-ui/src/actions/projects-actions.ts +++ b/cvat-ui/src/actions/projects-actions.ts @@ -31,8 +31,8 @@ export enum ProjectsActionTypes { // prettier-ignore const projectActions = { getProjects: () => createAction(ProjectsActionTypes.GET_PROJECTS), - getProjectsSuccess: (array: any[], count: number) => ( - createAction(ProjectsActionTypes.GET_PROJECTS_SUCCESS, { array, count }) + getProjectsSuccess: (array: any[], previews: string[], count: number) => ( + createAction(ProjectsActionTypes.GET_PROJECTS_SUCCESS, { array, previews, count }) ), getProjectsFailed: (error: any) => createAction(ProjectsActionTypes.GET_PROJECTS_FAILED, { error }), updateProjectsGettingQuery: (query: Partial) => ( @@ -69,6 +69,12 @@ export function getProjectsAsync(query: Partial): ThunkAction { page: 1, ...query, }; + + // Check if we try to retrive single project of projects list + if (!Object.keys(filteredQuery).includes('id')) { + filteredQuery.withoutTasks = true; + } + for (const key in filteredQuery) { if (filteredQuery[key] === null || typeof filteredQuery[key] === 'undefined') { delete filteredQuery[key]; @@ -84,40 +90,44 @@ export function getProjectsAsync(query: Partial): ThunkAction { } const array = Array.from(result); + const previewPromises = array.map((project): string => (project as any).preview().catch(() => '')); + + // Appropriate tasks fetching proccess needs with retrieving only a single project + if (Object.keys(filteredQuery).includes('id')) { + const tasks: any[] = []; + const taskPreviewPromises: Promise[] = []; + + for (const project of array) { + taskPreviewPromises.push( + ...(project as any).tasks.map((task: any): string => { + tasks.push(task); + return (task as any).frames.preview().catch(() => ''); + }), + ); + } - const tasks: any[] = []; - const taskPreviewPromises: Promise[] = []; - - for (const project of array) { - taskPreviewPromises.push( - ...(project as any).tasks.map((task: any): string => { - tasks.push(task); - return (task as any).frames.preview().catch(() => ''); - }), - ); + const taskPreviews = await Promise.all(taskPreviewPromises); + + const store = getCVATStore(); + const state: CombinedState = store.getState(); + + if (!state.tasks.fetching) { + dispatch( + getTasksSuccess(tasks, taskPreviews, tasks.length, { + page: 1, + assignee: null, + id: null, + mode: null, + name: null, + owner: null, + search: null, + status: null, + }), + ); + } } - const taskPreviews = await Promise.all(taskPreviewPromises); - - dispatch(projectActions.getProjectsSuccess(array, result.count)); - - const store = getCVATStore(); - const state: CombinedState = store.getState(); - - if (!state.tasks.fetching) { - dispatch( - getTasksSuccess(tasks, taskPreviews, tasks.length, { - page: 1, - assignee: null, - id: null, - mode: null, - name: null, - owner: null, - search: null, - status: null, - }), - ); - } + dispatch(projectActions.getProjectsSuccess(array, await Promise.all(previewPromises), result.count)); }; } diff --git a/cvat-ui/src/components/projects-page/project-item.tsx b/cvat-ui/src/components/projects-page/project-item.tsx index 6d80e04ca554..2ca8523f5ec9 100644 --- a/cvat-ui/src/components/projects-page/project-item.tsx +++ b/cvat-ui/src/components/projects-page/project-item.tsx @@ -1,4 +1,4 @@ -// Copyright (C) 2020 Intel Corporation +// Copyright (C) 2020-2021 Intel Corporation // // SPDX-License-Identifier: MIT @@ -22,24 +22,18 @@ interface Props { } export default function ProjectItemComponent(props: Props): JSX.Element { - const { projectInstance } = props; + const { + projectInstance: { instance, preview }, + } = props; const history = useHistory(); - const ownerName = projectInstance.owner ? projectInstance.owner.username : null; - const updated = moment(projectInstance.updatedDate).fromNow(); + const ownerName = instance.owner ? instance.owner.username : null; + const updated = moment(instance.updatedDate).fromNow(); const deletes = useSelector((state: CombinedState) => state.projects.activities.deletes); - const deleted = projectInstance.id in deletes ? deletes[projectInstance.id] : false; - - let projectPreview = null; - if (projectInstance.tasks.length) { - // prettier-ignore - projectPreview = useSelector((state: CombinedState) => ( - state.tasks.current.find((task) => task.instance.id === projectInstance.tasks[0].id)?.preview - )); - } + const deleted = instance.id in deletes ? deletes[instance.id] : false; const onOpenProject = (): void => { - history.push(`/projects/${projectInstance.id}`); + history.push(`/projects/${instance.id}`); }; const style: React.CSSProperties = {}; @@ -52,10 +46,10 @@ export default function ProjectItemComponent(props: Props): JSX.Element { return ( - {projectInstance.name} + {instance.name} )} description={( @@ -88,7 +82,7 @@ export default function ProjectItemComponent(props: Props): JSX.Element { {`Last updated ${updated}`}
- }> + }>
diff --git a/cvat-ui/src/components/projects-page/project-list.tsx b/cvat-ui/src/components/projects-page/project-list.tsx index 1d7138902dac..111b4a1de5fd 100644 --- a/cvat-ui/src/components/projects-page/project-list.tsx +++ b/cvat-ui/src/components/projects-page/project-list.tsx @@ -1,4 +1,4 @@ -// Copyright (C) 2020 Intel Corporation +// Copyright (C) 2020-2021 Intel Corporation // // SPDX-License-Identifier: MIT @@ -8,14 +8,14 @@ import { Row, Col } from 'antd/lib/grid'; import Pagination from 'antd/lib/pagination'; import { getProjectsAsync } from 'actions/projects-actions'; -import { CombinedState } from 'reducers/interfaces'; +import { CombinedState, Project } from 'reducers/interfaces'; import ProjectItem from './project-item'; export default function ProjectListComponent(): JSX.Element { const dispatch = useDispatch(); const projectsCount = useSelector((state: CombinedState) => state.projects.count); const { page } = useSelector((state: CombinedState) => state.projects.gettingQuery); - let projectInstances = useSelector((state: CombinedState) => state.projects.current); + const projectInstances = useSelector((state: CombinedState) => state.projects.current); const gettingQuery = useSelector((state: CombinedState) => state.projects.gettingQuery); function changePage(p: number): void { @@ -27,7 +27,7 @@ export default function ProjectListComponent(): JSX.Element { ); } - projectInstances = projectInstances.reduce((rows, key, index) => { + const projects = projectInstances.reduce((rows, key, index) => { if (index % 4 === 0) { rows.push([key]); } else { @@ -40,12 +40,12 @@ export default function ProjectListComponent(): JSX.Element { <> - {projectInstances.map( - (row: any[]): JSX.Element => ( - - {row.map((instance: any) => ( - - + {projects.map( + (row: Project[]): JSX.Element => ( + + {row.map((project: Project) => ( + + ))} diff --git a/cvat-ui/src/reducers/interfaces.ts b/cvat-ui/src/reducers/interfaces.ts index 841a0ef612d1..fc8a40340d90 100644 --- a/cvat-ui/src/reducers/interfaces.ts +++ b/cvat-ui/src/reducers/interfaces.ts @@ -30,10 +30,13 @@ export interface ProjectsQuery { owner: string | null; name: string | null; status: string | null; - [key: string]: string | number | null | undefined; + [key: string]: string | boolean | number | null | undefined; } -export type Project = any; +export interface Project { + instance: any; + preview: string; +} export interface ProjectsState { initialized: boolean; diff --git a/cvat-ui/src/reducers/projects-reducer.ts b/cvat-ui/src/reducers/projects-reducer.ts index d6a7c1eee491..c3d9d0d43158 100644 --- a/cvat-ui/src/reducers/projects-reducer.ts +++ b/cvat-ui/src/reducers/projects-reducer.ts @@ -1,4 +1,4 @@ -// Copyright (C) 2020 Intel Corporation +// Copyright (C) 2020-2021 Intel Corporation // // SPDX-License-Identifier: MIT @@ -50,12 +50,19 @@ export default (state: ProjectsState = defaultState, action: AnyAction): Project current: [], }; case ProjectsActionTypes.GET_PROJECTS_SUCCESS: { + const combinedWithPreviews = action.payload.array.map( + (project: any, index: number): Project => ({ + instance: project, + preview: action.payload.previews[index], + }), + ); + return { ...state, initialized: true, fetching: false, count: action.payload.count, - current: action.payload.array, + current: combinedWithPreviews, }; } case ProjectsActionTypes.GET_PROJECTS_FAILED: { diff --git a/cvat/apps/engine/serializers.py b/cvat/apps/engine/serializers.py index 15739bbafee4..4f3495594a71 100644 --- a/cvat/apps/engine/serializers.py +++ b/cvat/apps/engine/serializers.py @@ -510,11 +510,9 @@ class Meta: def to_representation(self, instance): response = super().to_representation(instance) - subsets = set() - for task in instance.tasks.all(): - if task.subset: - subsets.add(task.subset) - response['task_subsets'] = list(subsets) + task_subsets = set(instance.tasks.values_list('subset', flat=True)) + task_subsets.discard('') + response['task_subsets'] = list(task_subsets) return response class ProjectSerializer(ProjectWithoutTaskSerializer): From 8d2bc25df1ea4653f56147734d01eaa9b603b45c Mon Sep 17 00:00:00 2001 From: Dmitry Kalinin Date: Fri, 21 May 2021 20:56:25 +0300 Subject: [PATCH 2/8] Fixed project page --- cvat-core/src/project.js | 4 ++-- cvat-ui/src/actions/projects-actions.ts | 23 ++++++++----------- .../components/project-page/project-page.tsx | 10 ++++---- .../src/components/project-page/styles.scss | 4 ++++ 4 files changed, 20 insertions(+), 21 deletions(-) diff --git a/cvat-core/src/project.js b/cvat-core/src/project.js index 6d068ae8cbc8..1eefded59358 100644 --- a/cvat-core/src/project.js +++ b/cvat-core/src/project.js @@ -60,9 +60,9 @@ data.tasks.push(taskInstance); } } - if (!data.task_subsets && data.tasks.length) { + if (!data.task_subsets) { const subsetsSet = new Set(); - for (const task in data.tasks) { + for (const task of data.tasks) { if (task.subset) subsetsSet.add(task.subset); } data.task_subsets = Array.from(subsetsSet); diff --git a/cvat-ui/src/actions/projects-actions.ts b/cvat-ui/src/actions/projects-actions.ts index cc1b89e59c3e..c2f4a747c3f0 100644 --- a/cvat-ui/src/actions/projects-actions.ts +++ b/cvat-ui/src/actions/projects-actions.ts @@ -90,27 +90,23 @@ export function getProjectsAsync(query: Partial): ThunkAction { } const array = Array.from(result); - const previewPromises = array.map((project): string => (project as any).preview().catch(() => '')); // Appropriate tasks fetching proccess needs with retrieving only a single project if (Object.keys(filteredQuery).includes('id')) { const tasks: any[] = []; - const taskPreviewPromises: Promise[] = []; - - for (const project of array) { - taskPreviewPromises.push( - ...(project as any).tasks.map((task: any): string => { - tasks.push(task); - return (task as any).frames.preview().catch(() => ''); - }), - ); - } + const [project] = array; + const taskPreviewPromises: Promise[] = (project as any).tasks.map((task: any): string => { + tasks.push(task); + return (task as any).frames.preview().catch(() => ''); + }); const taskPreviews = await Promise.all(taskPreviewPromises); const store = getCVATStore(); const state: CombinedState = store.getState(); + dispatch(projectActions.getProjectsSuccess(array, taskPreviews, result.count)); + if (!state.tasks.fetching) { dispatch( getTasksSuccess(tasks, taskPreviews, tasks.length, { @@ -125,9 +121,10 @@ export function getProjectsAsync(query: Partial): ThunkAction { }), ); } + } else { + const previewPromises = array.map((project): string => (project as any).preview().catch(() => '')); + dispatch(projectActions.getProjectsSuccess(array, await Promise.all(previewPromises), result.count)); } - - dispatch(projectActions.getProjectsSuccess(array, await Promise.all(previewPromises), result.count)); }; } diff --git a/cvat-ui/src/components/project-page/project-page.tsx b/cvat-ui/src/components/project-page/project-page.tsx index 348b5d0e8a36..4c0de0c55eda 100644 --- a/cvat-ui/src/components/project-page/project-page.tsx +++ b/cvat-ui/src/components/project-page/project-page.tsx @@ -28,18 +28,16 @@ export default function ProjectPageComponent(): JSX.Element { const id = +useParams().id; const dispatch = useDispatch(); const history = useHistory(); - const projects = useSelector((state: CombinedState) => state.projects.current); + const projects = useSelector((state: CombinedState) => state.projects.current.map((project) => project.instance)); const projectsFetching = useSelector((state: CombinedState) => state.projects.fetching); const deletes = useSelector((state: CombinedState) => state.projects.activities.deletes); const taskDeletes = useSelector((state: CombinedState) => state.tasks.activities.deletes); const tasksActiveInferences = useSelector((state: CombinedState) => state.models.inferences); const tasks = useSelector((state: CombinedState) => state.tasks.current); - const projectSubsets = useSelector((state: CombinedState) => { - const [project] = state.projects.current.filter((_project) => _project.id === id); - return project ? ([...new Set(project.tasks.map((task: any) => task.subset))] as string[]) : []; - }); const [project] = projects.filter((_project) => _project.id === id); + const projectSubsets = ['']; + if (project) projectSubsets.push(...project.subsets); const deleteActivity = project && id in deletes ? deletes[id] : null; useEffect(() => { @@ -90,7 +88,7 @@ export default function ProjectPageComponent(): JSX.Element { - {projectSubsets.map((subset) => ( + {projectSubsets.map((subset: string) => ( {subset && {subset}} {tasks diff --git a/cvat-ui/src/components/project-page/styles.scss b/cvat-ui/src/components/project-page/styles.scss index 4769c556bb84..1befd4348a03 100644 --- a/cvat-ui/src/components/project-page/styles.scss +++ b/cvat-ui/src/components/project-page/styles.scss @@ -54,3 +54,7 @@ display: flex; align-items: center; } + +.ant-layout-content { + overflow-y: auto; +} From 1ad35e590f2380e0652cea2580a94d39ee519eb7 Mon Sep 17 00:00:00 2001 From: Dmitry Kalinin Date: Fri, 21 May 2021 21:03:21 +0300 Subject: [PATCH 3/8] Added CHANGELOG, increased packages versions --- CHANGELOG.md | 2 +- cvat-core/package-lock.json | 2 +- cvat-core/package.json | 2 +- cvat-ui/package-lock.json | 2 +- cvat-ui/package.json | 2 +- 5 files changed, 5 insertions(+), 5 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 328e058bef53..5f608b592cec 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -25,7 +25,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 ### Fixed -- +- Project page requests took a long time and many DB queries () ### Security diff --git a/cvat-core/package-lock.json b/cvat-core/package-lock.json index 776e8b2b84e6..2bdc11180320 100644 --- a/cvat-core/package-lock.json +++ b/cvat-core/package-lock.json @@ -1,6 +1,6 @@ { "name": "cvat-core", - "version": "3.12.3", + "version": "3.13.0", "lockfileVersion": 1, "requires": true, "dependencies": { diff --git a/cvat-core/package.json b/cvat-core/package.json index 9ed6e5e7cf03..e5e482fde5cb 100644 --- a/cvat-core/package.json +++ b/cvat-core/package.json @@ -1,6 +1,6 @@ { "name": "cvat-core", - "version": "3.12.3", + "version": "3.13.0", "description": "Part of Computer Vision Tool which presents an interface for client-side integration", "main": "babel.config.js", "scripts": { diff --git a/cvat-ui/package-lock.json b/cvat-ui/package-lock.json index c2a48c19580c..b3a5bce157d2 100644 --- a/cvat-ui/package-lock.json +++ b/cvat-ui/package-lock.json @@ -1,6 +1,6 @@ { "name": "cvat-ui", - "version": "1.20.0", + "version": "1.20.1", "lockfileVersion": 1, "requires": true, "dependencies": { diff --git a/cvat-ui/package.json b/cvat-ui/package.json index f15cd39389f5..a3df3ed409b3 100644 --- a/cvat-ui/package.json +++ b/cvat-ui/package.json @@ -1,6 +1,6 @@ { "name": "cvat-ui", - "version": "1.20.0", + "version": "1.20.1", "description": "CVAT single-page application", "main": "src/index.tsx", "scripts": { From 30cd147240bf89895aa4409e550eaa4b847b4701 Mon Sep 17 00:00:00 2001 From: Dmitry Kalinin Date: Mon, 24 May 2021 11:56:40 +0300 Subject: [PATCH 4/8] Update CHANGELOG.md Co-authored-by: Boris Sekachev --- CHANGELOG.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 5f608b592cec..cd0049d15bab 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -25,7 +25,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 ### Fixed -- Project page requests took a long time and many DB queries () +- Project page requests took a long time and did many DB queries () ### Security From 025cd34642fd488496d1a1438b001d57d210b70f Mon Sep 17 00:00:00 2001 From: Dmitry Kalinin Date: Mon, 24 May 2021 12:43:57 +0300 Subject: [PATCH 5/8] Fixed comments --- cvat-core/src/api-implementation.js | 10 ++++++++++ cvat-core/src/project.js | 9 ++------- cvat-ui/src/actions/projects-actions.ts | 13 +++---------- .../src/components/project-page/project-page.tsx | 2 +- .../src/components/projects-page/project-list.tsx | 2 +- cvat-ui/src/components/projects-page/styles.scss | 4 ++++ 6 files changed, 21 insertions(+), 19 deletions(-) diff --git a/cvat-core/src/api-implementation.js b/cvat-core/src/api-implementation.js index 69945a8b5988..04c982631c93 100644 --- a/cvat-core/src/api-implementation.js +++ b/cvat-core/src/api-implementation.js @@ -227,6 +227,14 @@ checkExclusiveFields(filter, ['id', 'search'], ['page', 'withoutTasks']); + if (typeof filter.withoutTasks === 'undefined') { + if (typeof filter.id === 'undefined') { + filter.withoutTasks = true; + } else { + filter.withoutTasks = false; + } + } + const searchParams = new URLSearchParams(); for (const field of ['name', 'assignee', 'owner', 'search', 'status', 'id', 'page', 'withoutTasks']) { if (Object.prototype.hasOwnProperty.call(filter, field)) { @@ -240,6 +248,8 @@ if (filter.withoutTasks) { project.task_ids = project.tasks; project.tasks = []; + } else { + project.task_ids = project.tasks.map((task) => task.id); } return project; }).map((project) => new Project(project)); diff --git a/cvat-core/src/project.js b/cvat-core/src/project.js index 1eefded59358..e2a137f13eae 100644 --- a/cvat-core/src/project.js +++ b/cvat-core/src/project.js @@ -351,15 +351,10 @@ }; Project.prototype.preview.implementation = async function () { - let taskId; - if (this.tasks.length) { - taskId = this.tasks[0].id; - } else if (Array.isArray(this._internalData.task_ids) && this._internalData.task_ids.length) { - [taskId] = this._internalData.task_ids; - } else { + if (!this._internalData.task_ids.length) { return ''; } - const frameData = await getPreview(taskId); + const frameData = await getPreview(this._internalData.task_ids[0]); return frameData; }; })(); diff --git a/cvat-ui/src/actions/projects-actions.ts b/cvat-ui/src/actions/projects-actions.ts index c2f4a747c3f0..7a8c8b0cd744 100644 --- a/cvat-ui/src/actions/projects-actions.ts +++ b/cvat-ui/src/actions/projects-actions.ts @@ -5,9 +5,8 @@ import { Dispatch, ActionCreator } from 'redux'; import { ActionUnion, createAction, ThunkAction } from 'utils/redux'; -import { ProjectsQuery, CombinedState } from 'reducers/interfaces'; +import { ProjectsQuery } from 'reducers/interfaces'; import { getTasksSuccess, updateTaskSuccess } from 'actions/tasks-actions'; -import { getCVATStore } from 'cvat-store'; import getCore from 'cvat-core-wrapper'; const cvat = getCore(); @@ -60,7 +59,7 @@ const projectActions = { export type ProjectActions = ActionUnion; export function getProjectsAsync(query: Partial): ThunkAction { - return async (dispatch: ActionCreator): Promise => { + return async (dispatch: ActionCreator, getState): Promise => { dispatch(projectActions.getProjects()); dispatch(projectActions.updateProjectsGettingQuery(query)); @@ -70,11 +69,6 @@ export function getProjectsAsync(query: Partial): ThunkAction { ...query, }; - // Check if we try to retrive single project of projects list - if (!Object.keys(filteredQuery).includes('id')) { - filteredQuery.withoutTasks = true; - } - for (const key in filteredQuery) { if (filteredQuery[key] === null || typeof filteredQuery[key] === 'undefined') { delete filteredQuery[key]; @@ -102,8 +96,7 @@ export function getProjectsAsync(query: Partial): ThunkAction { const taskPreviews = await Promise.all(taskPreviewPromises); - const store = getCVATStore(); - const state: CombinedState = store.getState(); + const state = getState(); dispatch(projectActions.getProjectsSuccess(array, taskPreviews, result.count)); diff --git a/cvat-ui/src/components/project-page/project-page.tsx b/cvat-ui/src/components/project-page/project-page.tsx index 4c0de0c55eda..385d58c519ac 100644 --- a/cvat-ui/src/components/project-page/project-page.tsx +++ b/cvat-ui/src/components/project-page/project-page.tsx @@ -28,7 +28,7 @@ export default function ProjectPageComponent(): JSX.Element { const id = +useParams().id; const dispatch = useDispatch(); const history = useHistory(); - const projects = useSelector((state: CombinedState) => state.projects.current.map((project) => project.instance)); + const projects = useSelector((state: CombinedState) => state.projects.current).map((project) => project.instance); const projectsFetching = useSelector((state: CombinedState) => state.projects.fetching); const deletes = useSelector((state: CombinedState) => state.projects.activities.deletes); const taskDeletes = useSelector((state: CombinedState) => state.tasks.activities.deletes); diff --git a/cvat-ui/src/components/projects-page/project-list.tsx b/cvat-ui/src/components/projects-page/project-list.tsx index 111b4a1de5fd..494b501f04dd 100644 --- a/cvat-ui/src/components/projects-page/project-list.tsx +++ b/cvat-ui/src/components/projects-page/project-list.tsx @@ -38,7 +38,7 @@ export default function ProjectListComponent(): JSX.Element { return ( <> - + {projects.map( (row: Project[]): JSX.Element => ( diff --git a/cvat-ui/src/components/projects-page/styles.scss b/cvat-ui/src/components/projects-page/styles.scss index 44fb66158704..dd60013d2279 100644 --- a/cvat-ui/src/components/projects-page/styles.scss +++ b/cvat-ui/src/components/projects-page/styles.scss @@ -117,3 +117,7 @@ object-fit: cover; } } + +.cvat-project-list-content { + padding-bottom: $grid-unit-size; +} From 6e4464ea017019ed9530d81a99ec141fabf33722 Mon Sep 17 00:00:00 2001 From: Dmitry Kalinin Date: Mon, 24 May 2021 13:29:02 +0300 Subject: [PATCH 6/8] Fixed overflow issue --- cvat-ui/src/components/project-page/styles.scss | 9 +++++---- 1 file changed, 5 insertions(+), 4 deletions(-) diff --git a/cvat-ui/src/components/project-page/styles.scss b/cvat-ui/src/components/project-page/styles.scss index 1befd4348a03..71767d2d78b5 100644 --- a/cvat-ui/src/components/project-page/styles.scss +++ b/cvat-ui/src/components/project-page/styles.scss @@ -4,6 +4,11 @@ @import '../../base.scss'; +.cvat-project-page { + overflow-y: auto; + height: 100%; +} + .cvat-project-details { width: 100%; height: auto; @@ -54,7 +59,3 @@ display: flex; align-items: center; } - -.ant-layout-content { - overflow-y: auto; -} From e3ee9a512a89d253f7bed380058a4e5ffc0b4752 Mon Sep 17 00:00:00 2001 From: Dmitry Kalinin Date: Mon, 24 May 2021 15:13:35 +0300 Subject: [PATCH 7/8] Fixed reducer issue --- cvat-ui/src/reducers/projects-reducer.ts | 24 ++++++++++-------------- 1 file changed, 10 insertions(+), 14 deletions(-) diff --git a/cvat-ui/src/reducers/projects-reducer.ts b/cvat-ui/src/reducers/projects-reducer.ts index c3d9d0d43158..d8fba25d88cb 100644 --- a/cvat-ui/src/reducers/projects-reducer.ts +++ b/cvat-ui/src/reducers/projects-reducer.ts @@ -117,13 +117,11 @@ export default (state: ProjectsState = defaultState, action: AnyAction): Project return { ...state, current: state.current.map( - (project): Project => { - if (project.id === action.payload.project.id) { - return action.payload.project; - } - - return project; - }, + (project): Project => ({ + ...project, + instance: project.instance.id === action.payload.project.id ? + action.payload.project : project.instance, + }), ), }; } @@ -131,13 +129,11 @@ export default (state: ProjectsState = defaultState, action: AnyAction): Project return { ...state, current: state.current.map( - (project): Project => { - if (project.id === action.payload.project.id) { - return action.payload.project; - } - - return project; - }, + (project): Project => ({ + ...project, + instance: project.instance.id === action.payload.project.id ? + action.payload.project : project.instance, + }), ), }; } From 089d38b987fe233a26abc6ffec68ceb4fc6a73a8 Mon Sep 17 00:00:00 2001 From: Dmitry Kalinin Date: Mon, 24 May 2021 15:39:14 +0300 Subject: [PATCH 8/8] Fixed cvat-core tests --- cvat-core/tests/api/projects.js | 2 +- cvat-core/tests/mocks/server-proxy.mock.js | 14 ++++++++------ 2 files changed, 9 insertions(+), 7 deletions(-) diff --git a/cvat-core/tests/api/projects.js b/cvat-core/tests/api/projects.js index 5a01b7d141d4..8a0fe3b09485 100644 --- a/cvat-core/tests/api/projects.js +++ b/cvat-core/tests/api/projects.js @@ -16,7 +16,7 @@ const { Project } = require('../../src/project'); describe('Feature: get projects', () => { test('get all projects', async () => { - const result = await window.cvat.projects.get(); + const result = await window.cvat.projects.get({ withoutTasks: false }); expect(Array.isArray(result)).toBeTruthy(); expect(result).toHaveLength(2); for (const el of result) { diff --git a/cvat-core/tests/mocks/server-proxy.mock.js b/cvat-core/tests/mocks/server-proxy.mock.js index 4357e1769efa..a5111756cd97 100644 --- a/cvat-core/tests/mocks/server-proxy.mock.js +++ b/cvat-core/tests/mocks/server-proxy.mock.js @@ -14,16 +14,18 @@ const { frameMetaDummyData, } = require('./dummy-data.mock'); -function QueryStringToJSON(query) { +function QueryStringToJSON(query, ignoreList = []) { const pairs = [...new URLSearchParams(query).entries()]; const result = {}; for (const pair of pairs) { const [key, value] = pair; - if (['id'].includes(key)) { - result[key] = +value; - } else { - result[key] = value; + if (!ignoreList.includes(key)) { + if (['id'].includes(key)) { + result[key] = +value; + } else { + result[key] = value; + } } } @@ -73,7 +75,7 @@ class ServerProxy { } async function getProjects(filter = '') { - const queries = QueryStringToJSON(filter); + const queries = QueryStringToJSON(filter, ['without_tasks']); const result = projectsDummyData.results.filter((x) => { for (const key in queries) { if (Object.prototype.hasOwnProperty.call(queries, key)) {