1 // Copyright (C) The Arvados Authors. All rights reserved.
3 // SPDX-License-Identifier: AGPL-3.0
5 import React from 'react';
17 } from '@material-ui/core';
18 import { ColumnSelector } from 'components/column-selector/column-selector';
19 import { DataTable, DataColumns, DataTableFetchMode } from 'components/data-table/data-table';
20 import { DataColumn } from 'components/data-table/data-column';
21 import { SearchInput } from 'components/search-input/search-input';
22 import { ArvadosTheme } from 'common/custom-theme';
23 import { createTree } from 'models/tree';
24 import { DataTableFilters } from 'components/data-table-filters/data-table-filters-tree';
25 import { CloseIcon, IconType, MaximizeIcon, UnMaximizeIcon, MoreOptionsIcon } from 'components/icon/icon';
26 import { PaperProps } from '@material-ui/core/Paper';
27 import { MPVPanelProps } from 'components/multi-panel-view/multi-panel-view';
28 import { MultiselectToolbar } from 'components/multiselectToolbar/MultiselectToolbar';
29 import { TCheckedList } from 'components/data-table/data-table';
42 const styles: StyleRulesCallback<CssRules> = (theme: ArvadosTheme) => ({
48 paddingRight: theme.spacing.unit,
60 display: 'inline-block',
61 paddingLeft: theme.spacing.unit * 2,
62 paddingTop: theme.spacing.unit * 2,
76 flexDirection: 'row-reverse',
77 justifyContent: 'space-between',
81 interface DataExplorerDataProps<T> {
82 fetchMode: DataTableFetchMode;
84 itemsAvailable: number;
85 columns: DataColumns<T, any>;
89 rowsPerPageOptions: number[];
91 contextMenuColumn: boolean;
92 defaultViewIcon?: IconType;
93 defaultViewMessages?: string[];
95 currentRefresh?: string;
96 currentRoute?: string;
97 hideColumnSelector?: boolean;
98 paperProps?: PaperProps;
99 actions?: React.ReactNode;
100 hideSearchInput?: boolean;
101 title?: React.ReactNode;
103 currentItemUuid: string;
104 elementPath?: string;
105 isMSToolbarVisible: boolean;
108 interface DataExplorerActionProps<T> {
109 onSetColumns: (columns: DataColumns<T, any>) => void;
110 onSearch: (value: string) => void;
111 onRowClick: (item: T) => void;
112 onRowDoubleClick: (item: T) => void;
113 onColumnToggle: (column: DataColumn<T, any>) => void;
114 onContextMenu: (event: React.MouseEvent<HTMLElement>, item: T) => void;
115 onSortToggle: (column: DataColumn<T, any>) => void;
116 onFiltersChange: (filters: DataTableFilters, column: DataColumn<T, any>) => void;
117 onChangePage: (page: number) => void;
118 onChangeRowsPerPage: (rowsPerPage: number) => void;
119 onLoadMore: (page: number) => void;
120 extractKey?: (item: T) => React.Key;
121 toggleMSToolbar: (isVisible: boolean) => void;
122 setCheckedListOnStore: (checkedList: TCheckedList) => void;
125 type DataExplorerProps<T> = DataExplorerDataProps<T> &
126 DataExplorerActionProps<T> &
127 WithStyles<CssRules> &
130 export const DataExplorer = withStyles(styles)(
131 class DataExplorerGeneric<T> extends React.Component<DataExplorerProps<T>> {
138 componentDidUpdate(prevProps: DataExplorerProps<T>) {
139 const currentRefresh = this.props.currentRefresh || '';
140 const currentRoute = this.props.currentRoute || '';
142 if (currentRoute !== this.state.prevRoute) {
143 // Component already mounted, but the user comes from a route change,
144 // like browsing through a project hierarchy.
146 showLoading: this.props.working,
147 prevRoute: currentRoute,
151 if (currentRefresh !== this.state.prevRefresh) {
152 // Component already mounted, but the user just clicked the
155 showLoading: this.props.working,
156 prevRefresh: currentRefresh,
159 if (this.state.showLoading && !this.props.working) {
166 componentDidMount() {
167 if (this.props.onSetColumns) {
168 this.props.onSetColumns(this.props.columns);
170 // Component just mounted, so we need to show the loading indicator.
172 showLoading: this.props.working,
173 prevRefresh: this.props.currentRefresh || '',
174 prevRoute: this.props.currentRoute || '',
213 setCheckedListOnStore,
216 <Paper className={classes.root} {...paperProps} key={paperKey} data-cy={this.props['data-cy']}>
217 <Grid container direction='column' wrap='nowrap' className={classes.container}>
220 <Grid item xs className={classes.title}>
224 {(!hideColumnSelector || !hideSearchInput || !!actions) && (
225 <Grid className={classes.headerMenu} item xs>
226 <Toolbar className={classes.toolbar}>
227 {!hideSearchInput && (
228 <div className={classes.searchBox}>
229 {!hideSearchInput && (
240 {!hideColumnSelector && (
241 <ColumnSelector columns={columns} onColumnToggle={onColumnToggle} />
243 {doUnMaximizePanel && panelMaximized && (
244 <Tooltip title={`Unmaximize ${panelName || 'panel'}`} disableFocusListener>
245 <IconButton onClick={doUnMaximizePanel}>
250 {doMaximizePanel && !panelMaximized && (
251 <Tooltip title={`Maximize ${panelName || 'panel'}`} disableFocusListener>
252 <IconButton onClick={doMaximizePanel}>
258 <Tooltip title={`Close ${panelName || 'panel'}`} disableFocusListener>
259 <IconButton disabled={panelMaximized} onClick={doHidePanel}>
265 <MultiselectToolbar />
269 <Grid item xs='auto' className={classes.dataTable}>
271 columns={this.props.contextMenuColumn ? [...columns, this.contextMenuColumn] : columns}
273 onRowClick={(_, item: T) => onRowClick(item)}
274 onContextMenu={onContextMenu}
275 onRowDoubleClick={(_, item: T) => onRowDoubleClick(item)}
276 onFiltersChange={onFiltersChange}
277 onSortToggle={onSortToggle}
278 extractKey={extractKey}
279 working={this.state.showLoading}
280 defaultViewIcon={defaultViewIcon}
281 defaultViewMessages={defaultViewMessages}
282 currentItemUuid={currentItemUuid}
283 currentRoute={paperKey}
284 toggleMSToolbar={toggleMSToolbar}
285 setCheckedListOnStore={setCheckedListOnStore}
289 <Toolbar className={classes.footer}>
292 <span data-cy='element-path'>{elementPath}</span>
295 <Grid container={!elementPath} justify='flex-end'>
296 {fetchMode === DataTableFetchMode.PAGINATED ? (
298 count={itemsAvailable}
299 rowsPerPage={rowsPerPage}
300 rowsPerPageOptions={rowsPerPageOptions}
301 page={this.props.page}
302 onChangePage={this.changePage}
303 onChangeRowsPerPage={this.changeRowsPerPage}
304 // Disable next button on empty lists since that's not default behavior
305 nextIconButtonProps={itemsAvailable > 0 ? {} : { disabled: true }}
309 <Button variant='text' size='medium' onClick={this.loadMore}>
321 changePage = (event: React.MouseEvent<HTMLButtonElement>, page: number) => {
322 this.props.onChangePage(page);
325 changeRowsPerPage: React.ChangeEventHandler<HTMLTextAreaElement | HTMLInputElement> = (event) => {
326 this.props.onChangeRowsPerPage(parseInt(event.target.value, 10));
330 this.props.onLoadMore(this.props.page + 1);
333 renderContextMenuTrigger = (item: T) => (
334 <Grid container justify='center'>
335 <Tooltip title='More options' disableFocusListener>
337 className={this.props.classes.moreOptionsButton}
338 onClick={(event) => this.props.onContextMenu(event, item)}
346 contextMenuColumn: DataColumn<any, any> = {
350 filters: createTree(),
351 key: 'context-actions',
352 render: this.renderContextMenuTrigger,