2019-06-13 01:53:03 +08:00
|
|
|
import React, { ReactNode } from "react";
|
|
|
|
import { GridCellRenderer, Index, MultiGrid } from "react-virtualized";
|
2019-06-12 16:06:06 +08:00
|
|
|
import "./dataTable.less";
|
|
|
|
|
|
|
|
export type Column<T> = {
|
|
|
|
name: string,
|
|
|
|
width: number,
|
2019-06-13 01:53:03 +08:00
|
|
|
cellRenderer: (record: T) => ReactNode,
|
2019-06-12 16:06:06 +08:00
|
|
|
tooltip?: (record: T) => string,
|
|
|
|
footerValue?: string,
|
|
|
|
footerTooltip?: string,
|
2019-06-13 01:53:03 +08:00
|
|
|
/**
|
|
|
|
* "number" and "integrated" have special meaning.
|
|
|
|
*/
|
2019-06-12 16:06:06 +08:00
|
|
|
className?: string,
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* A table with a fixed header. Optionally has fixed columns and a footer.
|
|
|
|
* TODO: no-content message.
|
|
|
|
*/
|
|
|
|
export class DataTable<T> extends React.Component<{
|
|
|
|
width: number,
|
|
|
|
height: number,
|
|
|
|
rowCount: number,
|
2019-06-13 01:53:03 +08:00
|
|
|
overscanRowCount?: number,
|
2019-06-12 16:06:06 +08:00
|
|
|
columns: Array<Column<T>>,
|
|
|
|
fixedColumnCount?: number,
|
2019-06-13 01:53:03 +08:00
|
|
|
overscanColumnCount?: number,
|
2019-06-12 16:06:06 +08:00
|
|
|
record: (index: Index) => T,
|
|
|
|
footer?: boolean,
|
2019-06-13 01:53:03 +08:00
|
|
|
/**
|
|
|
|
* When this changes, the DataTable will re-render.
|
|
|
|
*/
|
|
|
|
updateTrigger?: any
|
2019-06-12 16:06:06 +08:00
|
|
|
}> {
|
|
|
|
render() {
|
|
|
|
return (
|
|
|
|
<div className="DataTable">
|
|
|
|
<MultiGrid
|
|
|
|
width={this.props.width}
|
|
|
|
height={this.props.height}
|
|
|
|
rowHeight={26}
|
|
|
|
rowCount={this.props.rowCount + 1 + (this.props.footer ? 1 : 0)}
|
|
|
|
fixedRowCount={1}
|
2019-06-13 01:53:03 +08:00
|
|
|
overscanRowCount={this.props.overscanRowCount}
|
2019-06-12 16:06:06 +08:00
|
|
|
columnWidth={this.columnWidth}
|
|
|
|
columnCount={this.props.columns.length}
|
|
|
|
fixedColumnCount={this.props.fixedColumnCount}
|
2019-06-13 01:53:03 +08:00
|
|
|
overscanColumnCount={this.props.overscanColumnCount}
|
2019-06-12 16:06:06 +08:00
|
|
|
cellRenderer={this.cellRenderer}
|
|
|
|
classNameTopLeftGrid="DataTable-header"
|
|
|
|
classNameTopRightGrid="DataTable-header"
|
2019-06-13 01:53:03 +08:00
|
|
|
updateTigger={this.props.updateTrigger}
|
2019-06-12 16:06:06 +08:00
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
private columnWidth = ({ index }: Index): number => {
|
|
|
|
return this.props.columns[index].width;
|
|
|
|
}
|
|
|
|
|
|
|
|
private cellRenderer: GridCellRenderer = ({ columnIndex, rowIndex, style }) => {
|
|
|
|
const column = this.props.columns[columnIndex];
|
2019-06-13 01:53:03 +08:00
|
|
|
let cell: ReactNode;
|
2019-06-12 16:06:06 +08:00
|
|
|
let title: string | undefined;
|
|
|
|
const classes = ['DataTable-cell'];
|
|
|
|
|
|
|
|
if (columnIndex === this.props.columns.length - 1) {
|
|
|
|
classes.push('last-in-row');
|
|
|
|
}
|
|
|
|
|
|
|
|
if (rowIndex === 0) {
|
|
|
|
// Header row
|
2019-06-13 01:53:03 +08:00
|
|
|
cell = title = column.name;
|
2019-06-12 16:06:06 +08:00
|
|
|
} else {
|
|
|
|
// Record or footer row
|
|
|
|
if (column.className) {
|
|
|
|
classes.push(column.className);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this.props.footer && rowIndex === 1 + this.props.rowCount) {
|
|
|
|
// Footer row
|
|
|
|
classes.push('footer-cell');
|
2019-06-13 01:53:03 +08:00
|
|
|
cell = column.footerValue == null ? '' : column.footerValue;
|
2019-06-12 16:06:06 +08:00
|
|
|
title = column.footerTooltip == null ? '' : column.footerTooltip;
|
|
|
|
} else {
|
|
|
|
// Record row
|
|
|
|
const result = this.props.record({ index: rowIndex - 1 });
|
|
|
|
|
2019-06-13 01:53:03 +08:00
|
|
|
cell = column.cellRenderer(result);
|
2019-06-12 16:06:06 +08:00
|
|
|
|
|
|
|
if (column.tooltip) {
|
|
|
|
title = column.tooltip(result);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-13 01:53:03 +08:00
|
|
|
if (typeof cell !== 'string') {
|
|
|
|
classes.push('custom');
|
|
|
|
}
|
|
|
|
|
2019-06-12 16:06:06 +08:00
|
|
|
return (
|
|
|
|
<div
|
|
|
|
className={classes.join(' ')}
|
|
|
|
key={`${columnIndex}, ${rowIndex}`}
|
|
|
|
style={style}
|
|
|
|
title={title}
|
|
|
|
>
|
2019-06-13 01:53:03 +08:00
|
|
|
{typeof cell === 'string' ? (
|
|
|
|
<span className="DataTable-cell-text">{cell}</span>
|
|
|
|
) : cell}
|
2019-06-12 16:06:06 +08:00
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|