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/multiselectToolbar/MultiselectToolbar';
19 type CssRules = 'searchBox' | 'headerMenu' | 'toolbar' | 'footer' | 'root' | 'moreOptionsButton' | 'title' | 'dataTable' | 'container';
21 const styles: StyleRulesCallback<CssRules> = (theme: ArvadosTheme) => ({
27 paddingRight: theme.spacing.unit,
39 display: 'inline-block',
40 paddingLeft: theme.spacing.unit * 2,
41 paddingTop: theme.spacing.unit * 2,
55 flexDirection: 'row-reverse',
56 justifyContent: 'space-between',
60 interface DataExplorerDataProps<T> {
61 fetchMode: DataTableFetchMode;
63 itemsAvailable: number;
64 columns: DataColumns<T, any>;
68 rowsPerPageOptions: number[];
70 contextMenuColumn: boolean;
71 defaultViewIcon?: IconType;
72 defaultViewMessages?: string[];
74 currentRefresh?: string;
75 currentRoute?: string;
76 hideColumnSelector?: boolean;
77 paperProps?: PaperProps;
78 actions?: React.ReactNode;
79 hideSearchInput?: boolean;
80 title?: React.ReactNode;
82 currentItemUuid: string;
84 isMSToolbarVisible: boolean;
87 interface DataExplorerActionProps<T> {
88 onSetColumns: (columns: DataColumns<T, any>) => void;
89 onSearch: (value: string) => void;
90 onRowClick: (item: T) => void;
91 onRowDoubleClick: (item: T) => void;
92 onColumnToggle: (column: DataColumn<T, any>) => void;
93 onContextMenu: (event: React.MouseEvent<HTMLElement>, item: T) => void;
94 onSortToggle: (column: DataColumn<T, any>) => void;
95 onFiltersChange: (filters: DataTableFilters, column: DataColumn<T, any>) => void;
96 onChangePage: (page: number) => void;
97 onChangeRowsPerPage: (rowsPerPage: number) => void;
98 onLoadMore: (page: number) => void;
99 extractKey?: (item: T) => React.Key;
100 toggleMSToolbar: (isVisible: boolean) => void;
103 type DataExplorerProps<T> = DataExplorerDataProps<T> & DataExplorerActionProps<T> & WithStyles<CssRules> & MPVPanelProps;
105 export const DataExplorer = withStyles(styles)(
106 class DataExplorerGeneric<T> extends React.Component<DataExplorerProps<T>> {
113 componentDidUpdate(prevProps: DataExplorerProps<T>) {
114 const currentRefresh = this.props.currentRefresh || '';
115 const currentRoute = this.props.currentRoute || '';
117 if (currentRoute !== this.state.prevRoute) {
118 // Component already mounted, but the user comes from a route change,
119 // like browsing through a project hierarchy.
121 showLoading: this.props.working,
122 prevRoute: currentRoute,
126 if (currentRefresh !== this.state.prevRefresh) {
127 // Component already mounted, but the user just clicked the
130 showLoading: this.props.working,
131 prevRefresh: currentRefresh,
134 if (this.state.showLoading && !this.props.working) {
141 componentDidMount() {
142 if (this.props.onSetColumns) {
143 this.props.onSetColumns(this.props.columns);
145 // Component just mounted, so we need to show the loading indicator.
147 showLoading: this.props.working,
148 prevRefresh: this.props.currentRefresh || '',
149 prevRoute: this.props.currentRoute || '',
191 <Paper className={classes.root} {...paperProps} key={paperKey} data-cy={this.props['data-cy']}>
192 <Grid container direction='column' wrap='nowrap' className={classes.container}>
195 <Grid item xs className={classes.title}>
199 {(!hideColumnSelector || !hideSearchInput || !!actions) && (
200 <Grid className={classes.headerMenu} item xs>
201 <Toolbar className={classes.toolbar}>
202 {!hideSearchInput && (
203 <div className={classes.searchBox}>
204 {!hideSearchInput && <SearchInput label={searchLabel} value={searchValue} selfClearProp={''} onSearch={onSearch} />}
208 {!hideColumnSelector && <ColumnSelector columns={columns} onColumnToggle={onColumnToggle} />}
209 {doUnMaximizePanel && panelMaximized && (
210 <Tooltip title={`Unmaximize ${panelName || 'panel'}`} disableFocusListener>
211 <IconButton onClick={doUnMaximizePanel}>
216 {doMaximizePanel && !panelMaximized && (
217 <Tooltip title={`Maximize ${panelName || 'panel'}`} disableFocusListener>
218 <IconButton onClick={doMaximizePanel}>
224 <Tooltip title={`Close ${panelName || 'panel'}`} disableFocusListener>
225 <IconButton disabled={panelMaximized} onClick={doHidePanel}>
231 {isMSToolbarVisible && <MultiselectToolbar />}
235 <Grid item xs='auto' className={classes.dataTable}>
237 columns={this.props.contextMenuColumn ? [...columns, this.contextMenuColumn] : columns}
239 onRowClick={(_, item: T) => onRowClick(item)}
240 onContextMenu={onContextMenu}
241 onRowDoubleClick={(_, item: T) => onRowDoubleClick(item)}
242 onFiltersChange={onFiltersChange}
243 onSortToggle={onSortToggle}
244 extractKey={extractKey}
245 working={this.state.showLoading}
246 defaultViewIcon={defaultViewIcon}
247 defaultViewMessages={defaultViewMessages}
248 currentItemUuid={currentItemUuid}
249 currentRoute={paperKey}
250 toggleMSToolbar={toggleMSToolbar}
254 <Toolbar className={classes.footer}>
257 <span data-cy='element-path'>{elementPath}</span>
260 <Grid container={!elementPath} justify='flex-end'>
261 {fetchMode === DataTableFetchMode.PAGINATED ? (
263 count={itemsAvailable}
264 rowsPerPage={rowsPerPage}
265 rowsPerPageOptions={rowsPerPageOptions}
266 page={this.props.page}
267 onChangePage={this.changePage}
268 onChangeRowsPerPage={this.changeRowsPerPage}
269 // Disable next button on empty lists since that's not default behavior
270 nextIconButtonProps={itemsAvailable > 0 ? {} : { disabled: true }}
274 <Button variant='text' size='medium' onClick={this.loadMore}>
286 changePage = (event: React.MouseEvent<HTMLButtonElement>, page: number) => {
287 this.props.onChangePage(page);
290 changeRowsPerPage: React.ChangeEventHandler<HTMLTextAreaElement | HTMLInputElement> = (event) => {
291 this.props.onChangeRowsPerPage(parseInt(event.target.value, 10));
295 this.props.onLoadMore(this.props.page + 1);
298 renderContextMenuTrigger = (item: T) => (
299 <Grid container justify='center'>
300 <Tooltip title='More options' disableFocusListener>
301 <IconButton className={this.props.classes.moreOptionsButton} onClick={(event) => this.props.onContextMenu(event, item)}>
308 contextMenuColumn: DataColumn<any, any> = {
312 filters: createTree(),
313 key: 'context-actions',
314 render: this.renderContextMenuTrigger,