1 // Copyright (C) The Arvados Authors. All rights reserved.
3 // SPDX-License-Identifier: AGPL-3.0
5 import React from "react";
6 import { Grid, Paper, Toolbar, StyleRulesCallback, withStyles, WithStyles, TablePagination, IconButton, Tooltip, Button } from "@material-ui/core";
7 import { ColumnSelector } from "components/column-selector/column-selector";
8 import { DataTable, DataColumns, DataTableFetchMode } from "components/data-table/data-table";
9 import { DataColumn } from "components/data-table/data-column";
10 import { SearchInput } from "components/search-input/search-input";
11 import { ArvadosTheme } from "common/custom-theme";
12 import { MultiselectToolbar } from "components/multiselect-toolbar/MultiselectToolbar";
13 import { TCheckedList } from "components/data-table/data-table";
14 import { createTree } from "models/tree";
15 import { DataTableFilters } from "components/data-table-filters/data-table-filters-tree";
16 import { CloseIcon, IconType, MaximizeIcon, UnMaximizeIcon, MoreVerticalIcon } from "components/icon/icon";
17 import { PaperProps } from "@material-ui/core/Paper";
18 import { MPVPanelProps } from "components/multi-panel-view/multi-panel-view";
20 type CssRules = "titleWrapper" | "searchBox" | "headerMenu" | "toolbar" | "footer" | "root" | "moreOptionsButton" | "title" | 'subProcessTitle' | "dataTable" | "container";
22 const styles: StyleRulesCallback<CssRules> = (theme: ArvadosTheme) => ({
25 justifyContent: "space-between",
32 paddingRight: theme.spacing.unit,
45 display: "inline-block",
46 paddingLeft: theme.spacing.unit * 2,
47 paddingTop: theme.spacing.unit * 2,
52 display: "inline-block",
53 paddingLeft: theme.spacing.unit * 2,
54 paddingTop: theme.spacing.unit * 2,
73 interface DataExplorerDataProps<T> {
74 fetchMode: DataTableFetchMode;
76 itemsAvailable: number;
77 columns: DataColumns<T, any>;
81 rowsPerPageOptions: number[];
83 contextMenuColumn: boolean;
84 defaultViewIcon?: IconType;
85 defaultViewMessages?: string[];
87 currentRoute?: string;
88 hideColumnSelector?: boolean;
89 paperProps?: PaperProps;
90 actions?: React.ReactNode;
91 hideSearchInput?: boolean;
92 title?: React.ReactNode;
93 progressBar?: React.ReactNode;
95 currentItemUuid: string;
97 isMSToolbarVisible: boolean;
98 checkedList: TCheckedList;
102 interface DataExplorerActionProps<T> {
103 onSetColumns: (columns: DataColumns<T, any>) => void;
104 onSearch: (value: string) => void;
105 onRowClick: (item: T) => void;
106 onRowDoubleClick: (item: T) => void;
107 onColumnToggle: (column: DataColumn<T, any>) => void;
108 onContextMenu: (event: React.MouseEvent<HTMLElement>, item: T) => void;
109 onSortToggle: (column: DataColumn<T, any>) => void;
110 onFiltersChange: (filters: DataTableFilters, column: DataColumn<T, any>) => void;
111 onChangePage: (page: number) => void;
112 onChangeRowsPerPage: (rowsPerPage: number) => void;
113 onLoadMore: (page: number) => void;
114 extractKey?: (item: T) => React.Key;
115 toggleMSToolbar: (isVisible: boolean) => void;
116 setCheckedListOnStore: (checkedList: TCheckedList) => void;
119 type DataExplorerProps<T> = DataExplorerDataProps<T> & DataExplorerActionProps<T> & WithStyles<CssRules> & MPVPanelProps;
121 export const DataExplorer = withStyles(styles)(
122 class DataExplorerGeneric<T> extends React.Component<DataExplorerProps<T>> {
124 multiSelectToolbarInTitle = !this.props.title && !this.props.progressBar;
126 componentDidMount() {
127 if (this.props.onSetColumns) {
128 this.props.onSetColumns(this.props.columns);
169 setCheckedListOnStore,
176 className={classes.root}
179 data-cy={this.props["data-cy"]}
185 className={classes.container}
187 <div className={classes.titleWrapper} style={currentRoute?.includes('search-results') || !!progressBar ? {marginBottom: '-20px'} : {}}>
192 className={!!progressBar ? classes.subProcessTitle : classes.title}
197 {!!progressBar && progressBar}
198 {this.multiSelectToolbarInTitle && <MultiselectToolbar />}
199 {(!hideColumnSelector || !hideSearchInput || !!actions) && (
201 className={classes.headerMenu}
205 <Toolbar className={classes.toolbar}>
206 <Grid container justify="space-between" wrap="nowrap" alignItems="center">
207 {!hideSearchInput && (
208 <div className={classes.searchBox}>
209 {!hideSearchInput && (
220 {!hideColumnSelector && (
223 onColumnToggle={onColumnToggle}
227 {doUnMaximizePanel && panelMaximized && (
229 title={`Unmaximize ${panelName || "panel"}`}
232 <IconButton onClick={doUnMaximizePanel}>
237 {doMaximizePanel && !panelMaximized && (
239 title={`Maximize ${panelName || "panel"}`}
242 <IconButton onClick={doMaximizePanel}>
249 title={`Close ${panelName || "panel"}`}
253 disabled={panelMaximized}
254 onClick={doHidePanel}
264 {!this.multiSelectToolbarInTitle && <MultiselectToolbar />}
268 className={classes.dataTable}
269 style={currentRoute?.includes('search-results') || !!progressBar ? {marginTop: '-10px'} : {}}
272 columns={this.props.contextMenuColumn ? [...columns, this.contextMenuColumn] : columns}
274 onRowClick={(_, item: T) => onRowClick(item)}
275 onContextMenu={onContextMenu}
276 onRowDoubleClick={(_, item: T) => onRowDoubleClick(item)}
277 onFiltersChange={onFiltersChange}
278 onSortToggle={onSortToggle}
279 extractKey={extractKey}
280 defaultViewIcon={defaultViewIcon}
281 defaultViewMessages={defaultViewMessages}
282 currentItemUuid={currentItemUuid}
283 currentRoute={paperKey}
284 toggleMSToolbar={toggleMSToolbar}
285 setCheckedListOnStore={setCheckedListOnStore}
286 checkedList={checkedList}
288 isNotFound={this.props.isNotFound}
295 <Toolbar className={classes.footer}>
298 <span data-cy="element-path">{elementPath.length > 2 ? elementPath : ''}</span>
302 container={!elementPath}
305 {fetchMode === DataTableFetchMode.PAGINATED ? (
307 count={itemsAvailable}
308 rowsPerPage={rowsPerPage}
309 rowsPerPageOptions={rowsPerPageOptions}
310 page={this.props.page}
311 onChangePage={this.changePage}
312 onChangeRowsPerPage={this.changeRowsPerPage}
313 // Disable next button on empty lists since that's not default behavior
314 nextIconButtonProps={itemsAvailable > 0 ? {} : { disabled: true }}
320 onClick={this.loadMore}
323 style={{width: '100%', margin: '10px'}}
324 disabled={working || items.length >= itemsAvailable}
337 changePage = (event: React.MouseEvent<HTMLButtonElement>, page: number) => {
338 this.props.onChangePage(page);
341 changeRowsPerPage: React.ChangeEventHandler<HTMLTextAreaElement | HTMLInputElement> = event => {
342 this.props.onChangeRowsPerPage(parseInt(event.target.value, 10));
346 this.props.onLoadMore(this.props.page + 1);
349 renderContextMenuTrigger = (item: T) => (
359 className={this.props.classes.moreOptionsButton}
361 event.stopPropagation()
362 this.props.onContextMenu(event, item)
371 contextMenuColumn: DataColumn<any, any> = {
375 filters: createTree(),
376 key: "context-actions",
377 render: this.renderContextMenuTrigger,