Merge branch 'master'
[arvados-workbench2.git] / src / store / navigation / navigation-action.ts
1 // Copyright (C) The Arvados Authors. All rights reserved.
2 //
3 // SPDX-License-Identifier: AGPL-3.0
4
5 import { Dispatch } from "redux";
6 import { getProjectList, projectActions } from "../project/project-action";
7 import { push } from "react-router-redux";
8 import { TreeItemStatus } from "~/components/tree/tree";
9 import { findTreeItem } from "../project/project-reducer";
10 import { RootState } from "../store";
11 import { ResourceKind } from "~/models/resource";
12 import { projectPanelActions } from "../project-panel/project-panel-action";
13 import { getCollectionUrl } from "~/models/collection";
14 import { getProjectUrl, ProjectResource } from "~/models/project";
15 import { ProjectService } from "~/services/project-service/project-service";
16 import { ServiceRepository } from "~/services/services";
17 import { sidePanelActions } from "../side-panel/side-panel-action";
18 import { SidePanelId } from "../side-panel/side-panel-reducer";
19 import { getUuidObjectType, ObjectTypes } from "~/models/object-types";
20
21 export const getResourceUrl = (resourceKind: ResourceKind, resourceUuid: string): string => {
22     switch (resourceKind) {
23         case ResourceKind.PROJECT: return getProjectUrl(resourceUuid);
24         case ResourceKind.COLLECTION: return getCollectionUrl(resourceUuid);
25         default:
26             return '';
27     }
28 };
29
30 export enum ItemMode {
31     BOTH,
32     OPEN,
33     ACTIVE
34 }
35
36 export const setProjectItem = (itemId: string, itemMode: ItemMode) =>
37     (dispatch: Dispatch, getState: () => RootState, services: ServiceRepository) => {
38         const { projects, router } = getState();
39         const treeItem = findTreeItem(projects.items, itemId);
40
41         if (treeItem) {
42             const resourceUrl = getResourceUrl(treeItem.data.kind, treeItem.data.uuid);
43
44             if (itemMode === ItemMode.ACTIVE || itemMode === ItemMode.BOTH) {
45                 if (router.location && !router.location.pathname.includes(resourceUrl)) {
46                     dispatch(push(resourceUrl));
47                 }
48                 dispatch(projectActions.TOGGLE_PROJECT_TREE_ITEM_ACTIVE(treeItem.data.uuid));
49             }
50
51             const promise = treeItem.status === TreeItemStatus.LOADED
52                 ? Promise.resolve()
53                 : dispatch<any>(getProjectList(itemId));
54
55             promise
56                 .then(() => dispatch<any>(() => {
57                     if (itemMode === ItemMode.OPEN || itemMode === ItemMode.BOTH) {
58                         dispatch(projectActions.TOGGLE_PROJECT_TREE_ITEM_OPEN(treeItem.data.uuid));
59                     }
60                     dispatch(projectPanelActions.RESET_PAGINATION());
61                     dispatch(projectPanelActions.REQUEST_ITEMS());
62                 }));
63         } else {
64             const uuid = services.authService.getUuid();
65             if (itemId === uuid) {
66                 dispatch(projectActions.TOGGLE_PROJECT_TREE_ITEM_ACTIVE(uuid));
67                 dispatch(projectPanelActions.RESET_PAGINATION());
68                 dispatch(projectPanelActions.REQUEST_ITEMS());
69             }
70         }
71     };
72
73 export const restoreBranch = (itemId: string) =>
74     async (dispatch: Dispatch, getState: () => RootState, services: ServiceRepository) => {
75         const ancestors = await loadProjectAncestors(itemId, services.projectService);
76         const uuids = ancestors.map(ancestor => ancestor.uuid);
77         await loadBranch(uuids, dispatch);
78         dispatch(sidePanelActions.TOGGLE_SIDE_PANEL_ITEM_OPEN(SidePanelId.PROJECTS));
79         uuids.forEach(uuid => {
80             dispatch(projectActions.TOGGLE_PROJECT_TREE_ITEM_OPEN(uuid));
81         });
82     };
83
84 export const loadProjectAncestors = async (uuid: string, projectService: ProjectService): Promise<Array<ProjectResource>> => {
85     if (getUuidObjectType(uuid) === ObjectTypes.USER) {
86         return [];
87     } else {
88         const currentProject = await projectService.get(uuid);
89         const ancestors = await loadProjectAncestors(currentProject.ownerUuid, projectService);
90         return [...ancestors, currentProject];
91     }
92 };
93
94 const loadBranch = async (uuids: string[], dispatch: Dispatch): Promise<any> => {
95     const [uuid, ...rest] = uuids;
96     if (uuid) {
97         await dispatch<any>(getProjectList(uuid));
98         return loadBranch(rest, dispatch);
99     }
100 };