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 { createTree } from "models/tree";
13 import { DataTableFilters } from "components/data-table-filters/data-table-filters-tree";
14 import { CloseIcon, IconType, MaximizeIcon, UnMaximizeIcon, MoreOptionsIcon } from "components/icon/icon";
15 import { PaperProps } from "@material-ui/core/Paper";
16 import { MPVPanelProps } from "components/multi-panel-view/multi-panel-view";
17 import { MultiselectToolbar } from "components/multiselect-toolbar/MultiselectToolbar";
18 import { TCheckedList } from "components/data-table/data-table";
20 type CssRules = "searchBox" | "headerMenu" | "toolbar" | "footer" | "root" | "moreOptionsButton" | "title" | "dataTable" | "container";
22 const styles: StyleRulesCallback<CssRules> = (theme: ArvadosTheme) => ({
28 paddingRight: theme.spacing.unit,
40 display: "inline-block",
41 paddingLeft: theme.spacing.unit * 2,
42 paddingTop: theme.spacing.unit * 2,
56 flexDirection: "row-reverse",
57 justifyContent: "space-between",
61 interface DataExplorerDataProps<T> {
62 fetchMode: DataTableFetchMode;
64 itemsAvailable: number;
65 columns: DataColumns<T, any>;
69 rowsPerPageOptions: number[];
71 contextMenuColumn: boolean;
72 defaultViewIcon?: IconType;
73 defaultViewMessages?: string[];
75 currentRefresh?: string;
76 currentRoute?: string;
77 hideColumnSelector?: boolean;
78 paperProps?: PaperProps;
79 actions?: React.ReactNode;
80 hideSearchInput?: boolean;
81 title?: React.ReactNode;
83 currentItemUuid: string;
85 isMSToolbarVisible: boolean;
86 checkedList: TCheckedList;
89 interface DataExplorerActionProps<T> {
90 onSetColumns: (columns: DataColumns<T, any>) => void;
91 onSearch: (value: string) => void;
92 onRowClick: (item: T) => void;
93 onRowDoubleClick: (item: T) => void;
94 onColumnToggle: (column: DataColumn<T, any>) => void;
95 onContextMenu: (event: React.MouseEvent<HTMLElement>, item: T) => void;
96 onSortToggle: (column: DataColumn<T, any>) => void;
97 onFiltersChange: (filters: DataTableFilters, column: DataColumn<T, any>) => void;
98 onChangePage: (page: number) => void;
99 onChangeRowsPerPage: (rowsPerPage: number) => void;
100 onLoadMore: (page: number) => void;
101 extractKey?: (item: T) => React.Key;
102 toggleMSToolbar: (isVisible: boolean) => void;
103 setCheckedListOnStore: (checkedList: TCheckedList) => void;
106 type DataExplorerProps<T> = DataExplorerDataProps<T> & DataExplorerActionProps<T> & WithStyles<CssRules> & MPVPanelProps;
108 export const DataExplorer = withStyles(styles)(
109 class DataExplorerGeneric<T> extends React.Component<DataExplorerProps<T>> {
116 componentDidUpdate(prevProps: DataExplorerProps<T>) {
117 const currentRefresh = this.props.currentRefresh || "";
118 const currentRoute = this.props.currentRoute || "";
120 if (currentRoute !== this.state.prevRoute) {
121 // Component already mounted, but the user comes from a route change,
122 // like browsing through a project hierarchy.
124 showLoading: this.props.working,
125 prevRoute: currentRoute,
129 if (currentRefresh !== this.state.prevRefresh) {
130 // Component already mounted, but the user just clicked the
133 showLoading: this.props.working,
134 prevRefresh: currentRefresh,
137 if (this.state.showLoading && !this.props.working) {
144 componentDidMount() {
145 if (this.props.onSetColumns) {
146 this.props.onSetColumns(this.props.columns);
148 // Component just mounted, so we need to show the loading indicator.
150 showLoading: this.props.working,
151 prevRefresh: this.props.currentRefresh || "",
152 prevRoute: this.props.currentRoute || "",
191 setCheckedListOnStore,
196 className={classes.root}
199 data-cy={this.props["data-cy"]}>
204 className={classes.container}>
210 className={classes.title}>
214 {(!hideColumnSelector || !hideSearchInput || !!actions) && (
216 className={classes.headerMenu}
219 <Toolbar className={classes.toolbar}>
220 {!hideSearchInput && (
221 <div className={classes.searchBox}>
222 {!hideSearchInput && (
233 {!hideColumnSelector && (
236 onColumnToggle={onColumnToggle}
239 {doUnMaximizePanel && panelMaximized && (
241 title={`Unmaximize ${panelName || "panel"}`}
242 disableFocusListener>
243 <IconButton onClick={doUnMaximizePanel}>
248 {doMaximizePanel && !panelMaximized && (
250 title={`Maximize ${panelName || "panel"}`}
251 disableFocusListener>
252 <IconButton onClick={doMaximizePanel}>
259 title={`Close ${panelName || "panel"}`}
260 disableFocusListener>
262 disabled={panelMaximized}
263 onClick={doHidePanel}>
269 <MultiselectToolbar />
276 className={classes.dataTable}>
278 columns={this.props.contextMenuColumn ? [...columns, this.contextMenuColumn] : columns}
280 onRowClick={(_, item: T) => onRowClick(item)}
281 onContextMenu={onContextMenu}
282 onRowDoubleClick={(_, item: T) => onRowDoubleClick(item)}
283 onFiltersChange={onFiltersChange}
284 onSortToggle={onSortToggle}
285 extractKey={extractKey}
286 working={this.state.showLoading}
287 defaultViewIcon={defaultViewIcon}
288 defaultViewMessages={defaultViewMessages}
289 currentItemUuid={currentItemUuid}
290 currentRoute={paperKey}
291 toggleMSToolbar={toggleMSToolbar}
292 setCheckedListOnStore={setCheckedListOnStore}
293 checkedList={checkedList}
299 <Toolbar className={classes.footer}>
302 <span data-cy="element-path">{elementPath}</span>
306 container={!elementPath}
308 {fetchMode === DataTableFetchMode.PAGINATED ? (
310 count={itemsAvailable}
311 rowsPerPage={rowsPerPage}
312 rowsPerPageOptions={rowsPerPageOptions}
313 page={this.props.page}
314 onChangePage={this.changePage}
315 onChangeRowsPerPage={this.changeRowsPerPage}
316 // Disable next button on empty lists since that's not default behavior
317 nextIconButtonProps={itemsAvailable > 0 ? {} : { disabled: true }}
324 onClick={this.loadMore}>
336 changePage = (event: React.MouseEvent<HTMLButtonElement>, page: number) => {
337 this.props.onChangePage(page);
340 changeRowsPerPage: React.ChangeEventHandler<HTMLTextAreaElement | HTMLInputElement> = event => {
341 this.props.onChangeRowsPerPage(parseInt(event.target.value, 10));
345 this.props.onLoadMore(this.props.page + 1);
348 renderContextMenuTrigger = (item: T) => (
354 disableFocusListener>
356 className={this.props.classes.moreOptionsButton}
357 onClick={event => this.props.onContextMenu(event, item)}>
364 contextMenuColumn: DataColumn<any, any> = {
368 filters: createTree(),
369 key: "context-actions",
370 render: this.renderContextMenuTrigger,