1 // Copyright (C) The Arvados Authors. All rights reserved.
3 // SPDX-License-Identifier: AGPL-3.0
5 import React from "react";
6 import { connect } from "react-redux";
7 import { StyleRulesCallback, withStyles, WithStyles, Toolbar, Tooltip, IconButton } from "@material-ui/core";
8 import { ArvadosTheme } from "common/custom-theme";
9 import { RootState } from "store/store";
10 import { Dispatch } from "redux";
11 import { TCheckedList } from "components/data-table/data-table";
12 import { ContextMenuResource } from "store/context-menu/context-menu-actions";
13 import { Resource, ResourceKind, extractUuidKind } from "models/resource";
14 import { getResource } from "store/resources/resources";
15 import { ResourcesState } from "store/resources/resources";
16 import { MultiSelectMenuAction, MultiSelectMenuActionSet, MultiSelectMenuActionNames } from "views-components/multiselect-toolbar/ms-menu-actions";
17 import { ContextMenuAction } from "views-components/context-menu/context-menu-action-set";
18 import { multiselectActionsFilters, TMultiselectActionsFilters, msResourceKind } from "./ms-toolbar-action-filters";
19 import { kindToActionSet, findActionByName } from "./ms-kind-action-differentiator";
20 import { msToggleTrashAction } from "views-components/multiselect-toolbar/ms-project-action-set";
21 import { copyToClipboardAction } from "store/open-in-new-tab/open-in-new-tab.actions";
22 import { ContainerRequestResource } from "models/container-request";
23 import { FavoritesState } from "store/favorites/favorites-reducer";
24 import { resourceIsFrozen } from "common/frozen-resources";
25 import { ProjectResource } from "models/project";
27 type CssRules = "root" | "button";
29 const styles: StyleRulesCallback<CssRules> = (theme: ArvadosTheme) => ({
35 margin: "1rem auto auto 0.5rem",
37 transition: "width 150ms",
45 export type MultiselectToolbarProps = {
46 checkedList: TCheckedList;
47 selectedUuid: string | null
49 executeMulti: (action: ContextMenuAction, checkedList: TCheckedList, resources: ResourcesState) => void;
53 resources: ResourcesState;
54 favorites: FavoritesState
57 export const MultiselectToolbar = connect(
61 withStyles(styles)((props: MultiselectToolbarProps & WithStyles<CssRules>) => {
62 const { classes, checkedList, selectedUuid: singleSelectedUuid, iconProps } = props;
63 const singleProjectKind = singleSelectedUuid ? resourceSubKind(singleSelectedUuid, iconProps.resources) : ''
64 const currentResourceKinds = singleProjectKind ? singleProjectKind : Array.from(selectedToKindSet(checkedList));
66 const currentPathIsTrash = window.location.pathname === "/trash";
70 currentPathIsTrash && selectedToKindSet(checkedList).size
71 ? [msToggleTrashAction]
72 : selectActionsByKind(currentResourceKinds as string[], multiselectActionsFilters)
73 .filter((action) => (singleSelectedUuid === null ? action.isForMulti : true));
78 className={classes.root}
79 style={{ width: `${actions.length * 2.5}rem` }}
82 actions.map((action, i) =>
85 className={classes.button}
86 title={currentPathIsTrash || action.useAlts(singleSelectedUuid, iconProps) ? action.altName : action.name}
90 <IconButton onClick={() => props.executeMulti(action, checkedList, iconProps.resources)}>
91 {currentPathIsTrash || action.useAlts(singleSelectedUuid, iconProps) ? action.altIcon && action.altIcon({}) : action.icon({})}
96 className={classes.button}
101 <IconButton onClick={() => props.executeMulti(action, checkedList, iconProps.resources)}>{action.icon({})}</IconButton>
114 export function selectedToArray(checkedList: TCheckedList): Array<string> {
115 const arrayifiedSelectedList: Array<string> = [];
116 for (const [key, value] of Object.entries(checkedList)) {
117 if (value === true) {
118 arrayifiedSelectedList.push(key);
121 return arrayifiedSelectedList;
124 export function selectedToKindSet(checkedList: TCheckedList): Set<string> {
125 const setifiedList = new Set<string>();
126 for (const [key, value] of Object.entries(checkedList)) {
127 if (value === true) {
128 setifiedList.add(extractUuidKind(key) as string);
134 function groupByKind(checkedList: TCheckedList, resources: ResourcesState): Record<string, ContextMenuResource[]> {
136 selectedToArray(checkedList).forEach(uuid => {
137 const resource = getResource(uuid)(resources) as ContainerRequestResource | Resource;
138 if (!result[resource.kind]) result[resource.kind] = [];
139 result[resource.kind].push(resource);
144 function filterActions(actionArray: MultiSelectMenuActionSet, filters: Set<string>): Array<MultiSelectMenuAction> {
145 return actionArray[0].filter(action => filters.has(action.name as string));
148 const resourceSubKind = (uuid: string, resources: ResourcesState) => {
149 const resource = getResource(uuid)(resources) as ContainerRequestResource | Resource;
150 switch (resource.kind) {
151 case ResourceKind.PROJECT:
152 if(resourceIsFrozen(resource, resources)) return [msResourceKind.PROJECT_FROZEN]
153 if((resource as ProjectResource).canWrite === false) return [msResourceKind.PROJECT_READONLY]
154 if((resource as ProjectResource).groupClass === "filter") return [msResourceKind.PROJECT_FILTER]
155 return [msResourceKind.PROJECT]
157 return [resource.kind]
161 function selectActionsByKind(currentResourceKinds: Array<string>, filterSet: TMultiselectActionsFilters) {
162 const rawResult: Set<MultiSelectMenuAction> = new Set();
163 const resultNames = new Set();
164 const allFiltersArray: MultiSelectMenuAction[][] = []
165 currentResourceKinds.forEach(kind => {
166 if (filterSet[kind]) {
167 const actions = filterActions(...filterSet[kind]);
168 allFiltersArray.push(actions);
169 actions.forEach(action => {
170 if (!resultNames.has(action.name)) {
171 rawResult.add(action);
172 resultNames.add(action.name);
178 const filteredNameSet = allFiltersArray.map(filterArray => {
179 const resultSet = new Set<string>();
180 filterArray.forEach(action => resultSet.add(action.name as string || ""));
184 const filteredResult = Array.from(rawResult).filter(action => {
185 for (let i = 0; i < filteredNameSet.length; i++) {
186 if (!filteredNameSet[i].has(action.name as string)) return false;
191 return filteredResult.sort((a, b) => {
192 const nameA = a.name || "";
193 const nameB = b.name || "";
204 export const isExactlyOneSelected = (checkedList: TCheckedList) => {
207 for (const uuid in checkedList) {
208 if (checkedList[uuid] === true) {
213 return tally === 1 ? current : null
216 //--------------------------------------------------//
218 function mapStateToProps({multiselect, resources, favorites}: RootState) {
220 checkedList: multiselect.checkedList as TCheckedList,
221 selectedUuid: isExactlyOneSelected(multiselect.checkedList),
229 function mapDispatchToProps(dispatch: Dispatch) {
231 executeMulti: (selectedAction: ContextMenuAction, checkedList: TCheckedList, resources: ResourcesState): void => {
232 const kindGroups = groupByKind(checkedList, resources);
233 switch (selectedAction.name) {
234 case MultiSelectMenuActionNames.MOVE_TO:
235 case MultiSelectMenuActionNames.REMOVE:
236 const firstResource = getResource(selectedToArray(checkedList)[0])(resources) as ContainerRequestResource | Resource;
237 const action = findActionByName(selectedAction.name as string, kindToActionSet[firstResource.kind]);
238 if (action) action.execute(dispatch, kindGroups[firstResource.kind]);
240 case MultiSelectMenuActionNames.COPY_TO_CLIPBOARD:
241 const selectedResources = selectedToArray(checkedList).map(uuid => getResource(uuid)(resources));
242 dispatch<any>(copyToClipboardAction(selectedResources));
245 for (const kind in kindGroups) {
246 const action = findActionByName(selectedAction.name as string, kindToActionSet[kind]);
247 if (action) action.execute(dispatch, kindGroups[kind]);