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" | "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,
66 interface DataExplorerDataProps<T> {
67 fetchMode: DataTableFetchMode;
69 itemsAvailable: number;
70 columns: DataColumns<T, any>;
74 rowsPerPageOptions: number[];
76 contextMenuColumn: boolean;
77 defaultViewIcon?: IconType;
78 defaultViewMessages?: string[];
80 currentRefresh?: string;
81 currentRoute?: string;
82 hideColumnSelector?: boolean;
83 paperProps?: PaperProps;
84 actions?: React.ReactNode;
85 hideSearchInput?: boolean;
86 title?: React.ReactNode;
87 progressBar?: React.ReactNode;
89 currentItemUuid: string;
91 isMSToolbarVisible: boolean;
92 checkedList: TCheckedList;
95 interface DataExplorerActionProps<T> {
96 onSetColumns: (columns: DataColumns<T, any>) => void;
97 onSearch: (value: string) => void;
98 onRowClick: (item: T) => void;
99 onRowDoubleClick: (item: T) => void;
100 onColumnToggle: (column: DataColumn<T, any>) => void;
101 onContextMenu: (event: React.MouseEvent<HTMLElement>, item: T) => void;
102 onSortToggle: (column: DataColumn<T, any>) => void;
103 onFiltersChange: (filters: DataTableFilters, column: DataColumn<T, any>) => void;
104 onChangePage: (page: number) => void;
105 onChangeRowsPerPage: (rowsPerPage: number) => void;
106 onLoadMore: (page: number) => void;
107 extractKey?: (item: T) => React.Key;
108 toggleMSToolbar: (isVisible: boolean) => void;
109 setCheckedListOnStore: (checkedList: TCheckedList) => void;
112 type DataExplorerProps<T> = DataExplorerDataProps<T> & DataExplorerActionProps<T> & WithStyles<CssRules> & MPVPanelProps;
114 export const DataExplorer = withStyles(styles)(
115 class DataExplorerGeneric<T> extends React.Component<DataExplorerProps<T>> {
122 multiSelectToolbarInTitle = !this.props.title && !this.props.progressBar;
124 componentDidUpdate(prevProps: DataExplorerProps<T>) {
125 const currentRefresh = this.props.currentRefresh || "";
126 const currentRoute = this.props.currentRoute || "";
128 if (currentRoute !== this.state.prevRoute) {
129 // Component already mounted, but the user comes from a route change,
130 // like browsing through a project hierarchy.
132 showLoading: this.props.working,
133 prevRoute: currentRoute,
137 if (currentRefresh !== this.state.prevRefresh) {
138 // Component already mounted, but the user just clicked the
141 showLoading: this.props.working,
142 prevRefresh: currentRefresh,
145 if (this.state.showLoading && !this.props.working) {
152 componentDidMount() {
153 if (this.props.onSetColumns) {
154 this.props.onSetColumns(this.props.columns);
156 // Component just mounted, so we need to show the loading indicator.
158 showLoading: this.props.working,
159 prevRefresh: this.props.currentRefresh || "",
160 prevRoute: this.props.currentRoute || "",
200 setCheckedListOnStore,
205 className={classes.root}
208 data-cy={this.props["data-cy"]}
214 className={classes.container}
216 <div className={classes.titleWrapper}>
221 className={classes.title}
226 {!!progressBar && progressBar}
227 {this.multiSelectToolbarInTitle && <MultiselectToolbar />}
228 {(!hideColumnSelector || !hideSearchInput || !!actions) && (
230 className={classes.headerMenu}
234 <Toolbar className={classes.toolbar}>
235 <Grid container justify="space-between" wrap="nowrap" alignItems="center">
236 {!hideSearchInput && (
237 <div className={classes.searchBox}>
238 {!hideSearchInput && (
249 {!hideColumnSelector && (
252 onColumnToggle={onColumnToggle}
256 {doUnMaximizePanel && panelMaximized && (
258 title={`Unmaximize ${panelName || "panel"}`}
261 <IconButton onClick={doUnMaximizePanel}>
266 {doMaximizePanel && !panelMaximized && (
268 title={`Maximize ${panelName || "panel"}`}
271 <IconButton onClick={doMaximizePanel}>
278 title={`Close ${panelName || "panel"}`}
282 disabled={panelMaximized}
283 onClick={doHidePanel}
293 {!this.multiSelectToolbarInTitle && <MultiselectToolbar />}
297 className={classes.dataTable}
300 columns={this.props.contextMenuColumn ? [...columns, this.contextMenuColumn] : columns}
302 onRowClick={(_, item: T) => onRowClick(item)}
303 onContextMenu={onContextMenu}
304 onRowDoubleClick={(_, item: T) => onRowDoubleClick(item)}
305 onFiltersChange={onFiltersChange}
306 onSortToggle={onSortToggle}
307 extractKey={extractKey}
308 working={this.state.showLoading}
309 defaultViewIcon={defaultViewIcon}
310 defaultViewMessages={defaultViewMessages}
311 currentItemUuid={currentItemUuid}
312 currentRoute={paperKey}
313 toggleMSToolbar={toggleMSToolbar}
314 setCheckedListOnStore={setCheckedListOnStore}
315 checkedList={checkedList}
322 <Toolbar className={classes.footer}>
325 <span data-cy="element-path">{elementPath}</span>
329 container={!elementPath}
332 {fetchMode === DataTableFetchMode.PAGINATED ? (
334 count={itemsAvailable}
335 rowsPerPage={rowsPerPage}
336 rowsPerPageOptions={rowsPerPageOptions}
337 page={this.props.page}
338 onChangePage={this.changePage}
339 onChangeRowsPerPage={this.changeRowsPerPage}
340 // Disable next button on empty lists since that's not default behavior
341 nextIconButtonProps={itemsAvailable > 0 ? {} : { disabled: true }}
348 onClick={this.loadMore}
361 changePage = (event: React.MouseEvent<HTMLButtonElement>, page: number) => {
362 this.props.onChangePage(page);
365 changeRowsPerPage: React.ChangeEventHandler<HTMLTextAreaElement | HTMLInputElement> = event => {
366 this.props.onChangeRowsPerPage(parseInt(event.target.value, 10));
370 this.props.onLoadMore(this.props.page + 1);
373 renderContextMenuTrigger = (item: T) => (
383 className={this.props.classes.moreOptionsButton}
384 onClick={event => this.props.onContextMenu(event, item)}
392 contextMenuColumn: DataColumn<any, any> = {
396 filters: createTree(),
397 key: "context-actions",
398 render: this.renderContextMenuTrigger,