import React, { ReactNode } from "react"; import { GridCellRenderer, Index, MultiGrid } from "react-virtualized"; import "./BigTable.less"; export type Column = { name: string, width: number, cellRenderer: (record: T) => ReactNode, tooltip?: (record: T) => string, footerValue?: string, footerTooltip?: string, /** * "number" and "integrated" have special meaning. */ className?: string, } /** * A table with a fixed header. Optionally has fixed columns and a footer. * Uses windowing to support large amounts of rows and columns. * TODO: no-content message. */ export class BigTable extends React.Component<{ width: number, height: number, rowCount: number, overscanRowCount?: number, columns: Array>, fixedColumnCount?: number, overscanColumnCount?: number, record: (index: Index) => T, footer?: boolean, /** * When this changes, the DataTable will re-render. */ updateTrigger?: any }> { render() { return (
); } private columnWidth = ({ index }: Index): number => { return this.props.columns[index].width; } private cellRenderer: GridCellRenderer = ({ columnIndex, rowIndex, style }) => { const column = this.props.columns[columnIndex]; let cell: ReactNode; 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 cell = title = column.name; } 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'); cell = column.footerValue == null ? '' : column.footerValue; title = column.footerTooltip == null ? '' : column.footerTooltip; } else { // Record row const result = this.props.record({ index: rowIndex - 1 }); cell = column.cellRenderer(result); if (column.tooltip) { title = column.tooltip(result); } } } if (typeof cell !== 'string') { classes.push('custom'); } return (
{typeof cell === 'string' ? ( {cell} ) : cell}
); } }