UNPKG

react-virtualized

Version:

React components for efficiently rendering large, scrollable lists and tabular data

606 lines (514 loc) 19.6 kB
import _extends from 'babel-runtime/helpers/extends'; import _Object$getPrototypeOf from 'babel-runtime/core-js/object/get-prototype-of'; import _classCallCheck from 'babel-runtime/helpers/classCallCheck'; import _createClass from 'babel-runtime/helpers/createClass'; import _possibleConstructorReturn from 'babel-runtime/helpers/possibleConstructorReturn'; import _inherits from 'babel-runtime/helpers/inherits'; import cn from 'classnames'; import Column from './Column'; import React, { Component, PropTypes } from 'react'; import { findDOMNode } from 'react-dom'; import shallowCompare from 'react-addons-shallow-compare'; import Grid from '../Grid'; import defaultRowRenderer from './defaultRowRenderer'; import SortDirection from './SortDirection'; /** * Table component with fixed headers and virtualized rows for improved performance with large data sets. * This component expects explicit width, height, and padding parameters. */ var Table = function (_Component) { _inherits(Table, _Component); function Table(props) { _classCallCheck(this, Table); var _this = _possibleConstructorReturn(this, (Table.__proto__ || _Object$getPrototypeOf(Table)).call(this, props)); _this.state = { scrollbarWidth: 0 }; _this._createColumn = _this._createColumn.bind(_this); _this._createRow = _this._createRow.bind(_this); _this._onScroll = _this._onScroll.bind(_this); _this._onSectionRendered = _this._onSectionRendered.bind(_this); return _this; } _createClass(Table, [{ key: 'forceUpdateGrid', value: function forceUpdateGrid() { this.Grid.forceUpdate(); } /** See Grid#measureAllCells */ }, { key: 'measureAllRows', value: function measureAllRows() { this.Grid.measureAllCells(); } /** See Grid#recomputeGridSize */ }, { key: 'recomputeRowHeights', value: function recomputeRowHeights() { var index = arguments.length <= 0 || arguments[0] === undefined ? 0 : arguments[0]; this.Grid.recomputeGridSize({ rowIndex: index }); this.forceUpdateGrid(); } }, { key: 'componentDidMount', value: function componentDidMount() { this._setScrollbarWidth(); } }, { key: 'componentDidUpdate', value: function componentDidUpdate() { this._setScrollbarWidth(); } }, { key: 'render', value: function render() { var _this2 = this; var _props = this.props; var children = _props.children; var className = _props.className; var disableHeader = _props.disableHeader; var gridClassName = _props.gridClassName; var gridStyle = _props.gridStyle; var headerHeight = _props.headerHeight; var height = _props.height; var noRowsRenderer = _props.noRowsRenderer; var rowClassName = _props.rowClassName; var rowStyle = _props.rowStyle; var scrollToIndex = _props.scrollToIndex; var style = _props.style; var width = _props.width; var scrollbarWidth = this.state.scrollbarWidth; var availableRowsHeight = height - headerHeight; var rowClass = rowClassName instanceof Function ? rowClassName({ index: -1 }) : rowClassName; var rowStyleObject = rowStyle instanceof Function ? rowStyle({ index: -1 }) : rowStyle; // Precompute and cache column styles before rendering rows and columns to speed things up this._cachedColumnStyles = []; React.Children.toArray(children).forEach(function (column, index) { _this2._cachedColumnStyles[index] = _this2._getFlexStyleForColumn(column, column.props.style); }); // Note that we specify :numChildren, :scrollbarWidth, :sortBy, and :sortDirection as properties on Grid even though these have nothing to do with Grid. // This is done because Grid is a pure component and won't update unless its properties or state has changed. // Any property that should trigger a re-render of Grid then is specified here to avoid a stale display. return React.createElement( 'div', { className: cn('ReactVirtualized__Table', className), style: style }, !disableHeader && React.createElement( 'div', { className: cn('ReactVirtualized__Table__headerRow', rowClass), style: _extends({}, rowStyleObject, { height: headerHeight, paddingRight: scrollbarWidth, width: width }) }, this._getRenderedHeaderRow() ), React.createElement(Grid, _extends({}, this.props, { autoContainerWidth: true, className: cn('ReactVirtualized__Table__Grid', gridClassName), cellRenderer: this._createRow, columnWidth: width, columnCount: 1, height: availableRowsHeight, noContentRenderer: noRowsRenderer, onScroll: this._onScroll, onSectionRendered: this._onSectionRendered, ref: function ref(_ref) { _this2.Grid = _ref; }, scrollbarWidth: scrollbarWidth, scrollToRow: scrollToIndex, style: gridStyle })) ); } }, { key: 'shouldComponentUpdate', value: function shouldComponentUpdate(nextProps, nextState) { return shallowCompare(this, nextProps, nextState); } }, { key: '_createColumn', value: function _createColumn(_ref2) { var column = _ref2.column; var columnIndex = _ref2.columnIndex; var isScrolling = _ref2.isScrolling; var rowData = _ref2.rowData; var rowIndex = _ref2.rowIndex; var _column$props = column.props; var cellDataGetter = _column$props.cellDataGetter; var cellRenderer = _column$props.cellRenderer; var className = _column$props.className; var columnData = _column$props.columnData; var dataKey = _column$props.dataKey; var cellData = cellDataGetter({ columnData: columnData, dataKey: dataKey, rowData: rowData }); var renderedCell = cellRenderer({ cellData: cellData, columnData: columnData, dataKey: dataKey, isScrolling: isScrolling, rowData: rowData, rowIndex: rowIndex }); var style = this._cachedColumnStyles[columnIndex]; var title = typeof renderedCell === 'string' ? renderedCell : null; return React.createElement( 'div', { key: 'Row' + rowIndex + '-Col' + columnIndex, className: cn('ReactVirtualized__Table__rowColumn', className), style: style, title: title }, renderedCell ); } }, { key: '_createHeader', value: function _createHeader(_ref3) { var column = _ref3.column; var index = _ref3.index; var _props2 = this.props; var headerClassName = _props2.headerClassName; var headerStyle = _props2.headerStyle; var onHeaderClick = _props2.onHeaderClick; var sort = _props2.sort; var sortBy = _props2.sortBy; var sortDirection = _props2.sortDirection; var _column$props2 = column.props; var dataKey = _column$props2.dataKey; var disableSort = _column$props2.disableSort; var headerRenderer = _column$props2.headerRenderer; var label = _column$props2.label; var columnData = _column$props2.columnData; var sortEnabled = !disableSort && sort; var classNames = cn('ReactVirtualized__Table__headerColumn', headerClassName, column.props.headerClassName, { 'ReactVirtualized__Table__sortableHeaderColumn': sortEnabled }); var style = this._getFlexStyleForColumn(column, headerStyle); var renderedHeader = headerRenderer({ columnData: columnData, dataKey: dataKey, disableSort: disableSort, label: label, sortBy: sortBy, sortDirection: sortDirection }); var a11yProps = {}; if (sortEnabled || onHeaderClick) { (function () { // If this is a sortable header, clicking it should update the table data's sorting. var newSortDirection = sortBy !== dataKey || sortDirection === SortDirection.DESC ? SortDirection.ASC : SortDirection.DESC; var onClick = function onClick() { sortEnabled && sort({ sortBy: dataKey, sortDirection: newSortDirection }); onHeaderClick && onHeaderClick({ columnData: columnData, dataKey: dataKey }); }; var onKeyDown = function onKeyDown(event) { if (event.key === 'Enter' || event.key === ' ') { onClick(); } }; a11yProps['aria-label'] = column.props['aria-label'] || label || dataKey; a11yProps.role = 'rowheader'; a11yProps.tabIndex = 0; a11yProps.onClick = onClick; a11yProps.onKeyDown = onKeyDown; })(); } return React.createElement( 'div', _extends({}, a11yProps, { key: 'Header-Col' + index, className: classNames, style: style }), renderedHeader ); } }, { key: '_createRow', value: function _createRow(_ref4) { var _this3 = this; var index = _ref4.rowIndex; var isScrolling = _ref4.isScrolling; var key = _ref4.key; var style = _ref4.style; var _props3 = this.props; var children = _props3.children; var onRowClick = _props3.onRowClick; var onRowDoubleClick = _props3.onRowDoubleClick; var onRowMouseOver = _props3.onRowMouseOver; var onRowMouseOut = _props3.onRowMouseOut; var rowClassName = _props3.rowClassName; var rowGetter = _props3.rowGetter; var rowRenderer = _props3.rowRenderer; var rowStyle = _props3.rowStyle; var scrollbarWidth = this.state.scrollbarWidth; var rowClass = rowClassName instanceof Function ? rowClassName({ index: index }) : rowClassName; var rowStyleObject = rowStyle instanceof Function ? rowStyle({ index: index }) : rowStyle; var rowData = rowGetter({ index: index }); var columns = React.Children.toArray(children).map(function (column, columnIndex) { return _this3._createColumn({ column: column, columnIndex: columnIndex, isScrolling: isScrolling, rowData: rowData, rowIndex: index, scrollbarWidth: scrollbarWidth }); }); var className = cn('ReactVirtualized__Table__row', rowClass); var flattenedStyle = _extends({}, style, rowStyleObject, { height: this._getRowHeight(index), paddingRight: scrollbarWidth }); return rowRenderer({ className: className, columns: columns, index: index, isScrolling: isScrolling, key: key, onRowClick: onRowClick, onRowDoubleClick: onRowDoubleClick, onRowMouseOver: onRowMouseOver, onRowMouseOut: onRowMouseOut, rowData: rowData, style: flattenedStyle }); } /** * Determines the flex-shrink, flex-grow, and width values for a cell (header or column). */ }, { key: '_getFlexStyleForColumn', value: function _getFlexStyleForColumn(column) { var customStyle = arguments.length <= 1 || arguments[1] === undefined ? {} : arguments[1]; var flexValue = column.props.flexGrow + ' ' + column.props.flexShrink + ' ' + column.props.width + 'px'; var style = _extends({}, customStyle, { flex: flexValue, msFlex: flexValue, WebkitFlex: flexValue }); if (column.props.maxWidth) { style.maxWidth = column.props.maxWidth; } if (column.props.minWidth) { style.minWidth = column.props.minWidth; } return style; } }, { key: '_getRenderedHeaderRow', value: function _getRenderedHeaderRow() { var _this4 = this; var _props4 = this.props; var children = _props4.children; var disableHeader = _props4.disableHeader; var items = disableHeader ? [] : React.Children.toArray(children); return items.map(function (column, index) { return _this4._createHeader({ column: column, index: index }); }); } }, { key: '_getRowHeight', value: function _getRowHeight(rowIndex) { var rowHeight = this.props.rowHeight; return rowHeight instanceof Function ? rowHeight({ index: rowIndex }) : rowHeight; } }, { key: '_onScroll', value: function _onScroll(_ref5) { var clientHeight = _ref5.clientHeight; var scrollHeight = _ref5.scrollHeight; var scrollTop = _ref5.scrollTop; var onScroll = this.props.onScroll; onScroll({ clientHeight: clientHeight, scrollHeight: scrollHeight, scrollTop: scrollTop }); } }, { key: '_onSectionRendered', value: function _onSectionRendered(_ref6) { var rowOverscanStartIndex = _ref6.rowOverscanStartIndex; var rowOverscanStopIndex = _ref6.rowOverscanStopIndex; var rowStartIndex = _ref6.rowStartIndex; var rowStopIndex = _ref6.rowStopIndex; var onRowsRendered = this.props.onRowsRendered; onRowsRendered({ overscanStartIndex: rowOverscanStartIndex, overscanStopIndex: rowOverscanStopIndex, startIndex: rowStartIndex, stopIndex: rowStopIndex }); } }, { key: '_setScrollbarWidth', value: function _setScrollbarWidth() { var Grid = findDOMNode(this.Grid); var clientWidth = Grid.clientWidth || 0; var offsetWidth = Grid.offsetWidth || 0; var scrollbarWidth = offsetWidth - clientWidth; this.setState({ scrollbarWidth: scrollbarWidth }); } }]); return Table; }(Component); Table.propTypes = { 'aria-label': PropTypes.string, /** * Removes fixed height from the scrollingContainer so that the total height * of rows can stretch the window. Intended for use with WindowScroller */ autoHeight: PropTypes.bool, /** One or more Columns describing the data displayed in this row */ children: function children(props, propName, componentName) { var children = React.Children.toArray(props.children); for (var i = 0; i < children.length; i++) { if (children[i].type !== Column) { return new Error('Table only accepts children of type Column'); } } }, /** Optional CSS class name */ className: PropTypes.string, /** Disable rendering the header at all */ disableHeader: PropTypes.bool, /** * Used to estimate the total height of a Table before all of its rows have actually been measured. * The estimated total height is adjusted as rows are rendered. */ estimatedRowSize: PropTypes.number.isRequired, /** Optional custom CSS class name to attach to inner Grid element. */ gridClassName: PropTypes.string, /** Optional inline style to attach to inner Grid element. */ gridStyle: PropTypes.object, /** Optional CSS class to apply to all column headers */ headerClassName: PropTypes.string, /** Fixed height of header row */ headerHeight: PropTypes.number.isRequired, /** Fixed/available height for out DOM element */ height: PropTypes.number.isRequired, /** Optional renderer to be used in place of table body rows when rowCount is 0 */ noRowsRenderer: PropTypes.func, /** * Optional callback when a column's header is clicked. * ({ columnData: any, dataKey: string }): void */ onHeaderClick: PropTypes.func, /** Optional custom inline style to attach to table header columns. */ headerStyle: PropTypes.object, /** * Callback invoked when a user clicks on a table row. * ({ index: number }): void */ onRowClick: PropTypes.func, /** * Callback invoked when a user double-clicks on a table row. * ({ index: number }): void */ onRowDoubleClick: PropTypes.func, /** * Callback invoked when the mouse leaves a table row. * ({ index: number }): void */ onRowMouseOut: PropTypes.func, /** * Callback invoked when a user moves the mouse over a table row. * ({ index: number }): void */ onRowMouseOver: PropTypes.func, /** * Callback invoked with information about the slice of rows that were just rendered. * ({ startIndex, stopIndex }): void */ onRowsRendered: PropTypes.func, /** * Callback invoked whenever the scroll offset changes within the inner scrollable region. * This callback can be used to sync scrolling between lists, tables, or grids. * ({ clientHeight, scrollHeight, scrollTop }): void */ onScroll: PropTypes.func.isRequired, /** * Number of rows to render above/below the visible bounds of the list. * These rows can help for smoother scrolling on touch devices. */ overscanRowCount: PropTypes.number.isRequired, /** * Optional CSS class to apply to all table rows (including the header row). * This property can be a CSS class name (string) or a function that returns a class name. * If a function is provided its signature should be: ({ index: number }): string */ rowClassName: PropTypes.oneOfType([PropTypes.string, PropTypes.func]), /** * Callback responsible for returning a data row given an index. * ({ index: number }): any */ rowGetter: PropTypes.func.isRequired, /** * Either a fixed row height (number) or a function that returns the height of a row given its index. * ({ index: number }): number */ rowHeight: PropTypes.oneOfType([PropTypes.number, PropTypes.func]).isRequired, /** Number of rows in table. */ rowCount: PropTypes.number.isRequired, /** * Responsible for rendering a table row given an array of columns: * Should implement the following interface: ({ * className: string, * columns: Array, * index: number, * isScrolling: boolean, * onRowClick: ?Function, * onRowDoubleClick: ?Function, * onRowMouseOver: ?Function, * onRowMouseOut: ?Function, * rowData: any, * style: any * }): PropTypes.node */ rowRenderer: PropTypes.func, /** Optional custom inline style to attach to table rows. */ rowStyle: PropTypes.oneOfType([PropTypes.object, PropTypes.func]).isRequired, /** See Grid#scrollToAlignment */ scrollToAlignment: PropTypes.oneOf(['auto', 'end', 'start', 'center']).isRequired, /** Row index to ensure visible (by forcefully scrolling if necessary) */ scrollToIndex: PropTypes.number, /** Vertical offset. */ scrollTop: PropTypes.number, /** * Sort function to be called if a sortable header is clicked. * ({ sortBy: string, sortDirection: SortDirection }): void */ sort: PropTypes.func, /** Table data is currently sorted by this :dataKey (if it is sorted at all) */ sortBy: PropTypes.string, /** Table data is currently sorted in this direction (if it is sorted at all) */ sortDirection: PropTypes.oneOf([SortDirection.ASC, SortDirection.DESC]), /** Optional inline style */ style: PropTypes.object, /** Tab index for focus */ tabIndex: PropTypes.number, /** Width of list */ width: PropTypes.number.isRequired }; Table.defaultProps = { disableHeader: false, estimatedRowSize: 30, headerHeight: 0, headerStyle: {}, noRowsRenderer: function noRowsRenderer() { return null; }, onRowsRendered: function onRowsRendered() { return null; }, onScroll: function onScroll() { return null; }, overscanRowCount: 10, rowRenderer: defaultRowRenderer, rowStyle: {}, scrollToAlignment: 'auto', style: {} }; export default Table;