Merge branch '17426-plug-ins' refs #17426
[arvados-workbench2.git] / src / views-components / side-panel-tree / side-panel-tree.tsx
1 // Copyright (C) The Arvados Authors. All rights reserved.
2 //
3 // SPDX-License-Identifier: AGPL-3.0
4
5 import * as React from "react";
6 import { Dispatch } from "redux";
7 import { connect } from "react-redux";
8 import { TreePicker, TreePickerProps } from "../tree-picker/tree-picker";
9 import { TreeItem } from "~/components/tree/tree";
10 import { ProjectResource } from "~/models/project";
11 import { ListItemTextIcon } from "~/components/list-item-text-icon/list-item-text-icon";
12 import { ProcessIcon, ProjectIcon, FilterGroupIcon, FavoriteIcon, ProjectsIcon, ShareMeIcon, TrashIcon, PublicFavoriteIcon } from '~/components/icon/icon';
13 import { WorkflowIcon } from '~/components/icon/icon';
14 import { activateSidePanelTreeItem, toggleSidePanelTreeItemCollapse, SIDE_PANEL_TREE, SidePanelTreeCategory } from '~/store/side-panel-tree/side-panel-tree-actions';
15 import { openSidePanelContextMenu } from '~/store/context-menu/context-menu-actions';
16 import { noop } from 'lodash';
17 import { ResourceKind } from "~/models/resource";
18 import { IllegalNamingWarning } from "~/components/warning/warning";
19 import { GroupClass } from "~/models/group";
20
21 export interface SidePanelTreeProps {
22     onItemActivation: (id: string) => void;
23     sidePanelProgress?: boolean;
24 }
25
26 type SidePanelTreeActionProps = Pick<TreePickerProps<ProjectResource | string>, 'onContextMenu' | 'toggleItemActive' | 'toggleItemOpen' | 'toggleItemSelection'>;
27
28 const mapDispatchToProps = (dispatch: Dispatch, props: SidePanelTreeProps): SidePanelTreeActionProps => ({
29     onContextMenu: (event, { id }) => {
30         dispatch<any>(openSidePanelContextMenu(event, id));
31     },
32     toggleItemActive: (_, { id }) => {
33         dispatch<any>(activateSidePanelTreeItem(id));
34         props.onItemActivation(id);
35     },
36     toggleItemOpen: (_, { id }) => {
37         dispatch<any>(toggleSidePanelTreeItemCollapse(id));
38     },
39     toggleItemSelection: noop,
40 });
41
42 export const SidePanelTree = connect(undefined, mapDispatchToProps)(
43     (props: SidePanelTreeActionProps) =>
44         <TreePicker {...props} render={renderSidePanelItem} pickerId={SIDE_PANEL_TREE} />);
45
46 const renderSidePanelItem = (item: TreeItem<ProjectResource>) => {
47     const name = typeof item.data === 'string' ? item.data : item.data.name;
48     const warn = typeof item.data !== 'string' && item.data.kind === ResourceKind.PROJECT
49         ? <IllegalNamingWarning name={name} />
50         : undefined;
51     return <ListItemTextIcon
52         icon={getProjectPickerIcon(item)}
53         name={name}
54         nameDecorator={warn}
55         isActive={item.active}
56         hasMargin={true}
57     />;
58 };
59
60 const getProjectPickerIcon = (item: TreeItem<ProjectResource | string>) =>
61     typeof item.data === 'string'
62         ? getSidePanelIcon(item.data)
63         : (item.data && item.data.groupClass === GroupClass.FILTER)
64             ? FilterGroupIcon
65             : ProjectIcon;
66
67 const getSidePanelIcon = (category: string) => {
68     switch (category) {
69         case SidePanelTreeCategory.FAVORITES:
70             return FavoriteIcon;
71         case SidePanelTreeCategory.PROJECTS:
72             return ProjectsIcon;
73         case SidePanelTreeCategory.SHARED_WITH_ME:
74             return ShareMeIcon;
75         case SidePanelTreeCategory.TRASH:
76             return TrashIcon;
77         case SidePanelTreeCategory.WORKFLOWS:
78             return WorkflowIcon;
79         case SidePanelTreeCategory.PUBLIC_FAVORITES:
80             return PublicFavoriteIcon;
81         case SidePanelTreeCategory.ALL_PROCESSES:
82             return ProcessIcon;
83         default:
84             return ProjectIcon;
85     }
86 };