UNPKG

kepler.gl

Version:

kepler.gl is a webgl based application to visualize large scale location data in the browser

467 lines (465 loc) 87 kB
"use strict"; var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault"); var _typeof = require("@babel/runtime/helpers/typeof"); Object.defineProperty(exports, "__esModule", { value: true }); exports["default"] = exports.TableSection = exports.Container = void 0; var _toConsumableArray2 = _interopRequireDefault(require("@babel/runtime/helpers/toConsumableArray")); var _classCallCheck2 = _interopRequireDefault(require("@babel/runtime/helpers/classCallCheck")); var _createClass2 = _interopRequireDefault(require("@babel/runtime/helpers/createClass")); var _possibleConstructorReturn2 = _interopRequireDefault(require("@babel/runtime/helpers/possibleConstructorReturn")); var _getPrototypeOf2 = _interopRequireDefault(require("@babel/runtime/helpers/getPrototypeOf")); var _inherits2 = _interopRequireDefault(require("@babel/runtime/helpers/inherits")); var _extends2 = _interopRequireDefault(require("@babel/runtime/helpers/extends")); var _taggedTemplateLiteral2 = _interopRequireDefault(require("@babel/runtime/helpers/taggedTemplateLiteral")); var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty")); var _react = _interopRequireWildcard(require("react")); var _reactVirtualized = require("react-virtualized"); var _styledComponents = _interopRequireWildcard(require("styled-components")); var _classnames2 = _interopRequireDefault(require("classnames")); var _reselect = require("reselect"); var _lodash = _interopRequireDefault(require("lodash.get")); var _lodash2 = _interopRequireDefault(require("lodash.debounce")); var _icons = require("../icons"); var _grid = _interopRequireDefault(require("./grid")); var _headerCell = _interopRequireDefault(require("./header-cell")); var _utils = require("@kepler.gl/utils"); var _cellSize = require("./cell-size"); var _constants = require("@kepler.gl/constants"); var _templateObject, _templateObject2; // SPDX-License-Identifier: MIT // Copyright contributors to the kepler.gl project function _getRequireWildcardCache(e) { if ("function" != typeof WeakMap) return null; var r = new WeakMap(), t = new WeakMap(); return (_getRequireWildcardCache = function _getRequireWildcardCache(e) { return e ? t : r; })(e); } function _interopRequireWildcard(e, r) { if (!r && e && e.__esModule) return e; if (null === e || "object" != _typeof(e) && "function" != typeof e) return { "default": e }; var t = _getRequireWildcardCache(r); if (t && t.has(e)) return t.get(e); var n = { __proto__: null }, a = Object.defineProperty && Object.getOwnPropertyDescriptor; for (var u in e) if ("default" !== u && {}.hasOwnProperty.call(e, u)) { var i = a ? Object.getOwnPropertyDescriptor(e, u) : null; i && (i.get || i.set) ? Object.defineProperty(n, u, i) : n[u] = e[u]; } return n["default"] = e, t && t.set(e, n), n; } function ownKeys(e, r) { var t = Object.keys(e); if (Object.getOwnPropertySymbols) { var o = Object.getOwnPropertySymbols(e); r && (o = o.filter(function (r) { return Object.getOwnPropertyDescriptor(e, r).enumerable; })), t.push.apply(t, o); } return t; } function _objectSpread(e) { for (var r = 1; r < arguments.length; r++) { var t = null != arguments[r] ? arguments[r] : {}; r % 2 ? ownKeys(Object(t), !0).forEach(function (r) { (0, _defineProperty2["default"])(e, r, t[r]); }) : Object.getOwnPropertyDescriptors ? Object.defineProperties(e, Object.getOwnPropertyDescriptors(t)) : ownKeys(Object(t)).forEach(function (r) { Object.defineProperty(e, r, Object.getOwnPropertyDescriptor(t, r)); }); } return e; } function _callSuper(t, o, e) { return o = (0, _getPrototypeOf2["default"])(o), (0, _possibleConstructorReturn2["default"])(t, _isNativeReflectConstruct() ? Reflect.construct(o, e || [], (0, _getPrototypeOf2["default"])(t).constructor) : o.apply(t, e)); } function _isNativeReflectConstruct() { try { var t = !Boolean.prototype.valueOf.call(Reflect.construct(Boolean, [], function () {})); } catch (t) {} return (_isNativeReflectConstruct = function _isNativeReflectConstruct() { return !!t; })(); } var defaultHeaderRowHeight = 55; var defaultHeaderStatsControlHeight = 40; var defaultRowHeight = 32; var overscanColumnCount = 10; var overscanRowCount = 10; // The default scrollbar width can range anywhere from 12px to 17px var browserScrollBarWidth = 17; var fieldToAlignRight = (0, _defineProperty2["default"])((0, _defineProperty2["default"])({}, _constants.ALL_FIELD_TYPES.integer, true), _constants.ALL_FIELD_TYPES.real, true); var pinnedClassList = { header: 'pinned-columns--header pinned-grid-container', rows: 'pinned-columns--rows pinned-grid-container' }; var unpinnedClassList = { header: 'unpinned-columns--header unpinned-grid-container', rows: 'unpinned-columns--rows unpinned-grid-container' }; var Container = exports.Container = _styledComponents["default"].div(_templateObject || (_templateObject = (0, _taggedTemplateLiteral2["default"])(["\n display: flex;\n font-size: 11px;\n flex-grow: 1;\n color: ", ";\n width: 100%;\n position: relative;\n .ReactVirtualized__Grid:focus,\n .ReactVirtualized__Grid:active {\n outline: 0;\n }\n .body-grid {\n ", "\n }\n\n .cell {\n &::-webkit-scrollbar {\n display: none;\n }\n }\n\n *:focus {\n outline: 0;\n }\n\n .results-table-wrapper {\n position: relative;\n min-height: 100%;\n max-height: 100%;\n display: flex;\n flex-direction: row;\n flex-grow: 1;\n overflow: hidden;\n\n .scroll-in-ui-thread.pinned-columns--header {\n overflow: hidden;\n border-bottom: 1px solid ", ";\n padding-bottom: ", "px;\n }\n .scroll-in-ui-thread.unpinned-columns--header {\n width: 100vw;\n overflow: hidden;\n border-bottom: 1px solid ", ";\n // leave room for scrollbar\n padding-bottom: ", "px;\n }\n\n .scroll-in-ui-thread::after {\n content: '';\n height: 100%;\n left: 0;\n position: absolute;\n pointer-events: none;\n top: 0;\n width: 100%;\n }\n\n .grid-row {\n position: relative;\n display: flex;\n flex-direction: row;\n }\n .grid-column {\n display: flex;\n flex-direction: column;\n flex: 1 1 auto;\n }\n .pinned-grid-container {\n flex: 0 0 75px;\n z-index: 10;\n position: absolute;\n left: 0;\n top: 0;\n border-right: 2px solid ", ";\n }\n .even-row {\n background-color: ", ";\n }\n .odd-row {\n background-color: ", ";\n }\n .cell,\n .header-cell {\n width: 100%;\n height: 100%;\n display: flex;\n flex-direction: column;\n justify-content: center;\n align-items: flex-start;\n text-align: center;\n overflow: hidden;\n // header border is rendered by header container\n border-bottom: 0;\n .n-sort-idx {\n font-size: 9px;\n }\n }\n .cell {\n border-bottom: 1px solid ", ";\n border-right: 1px solid ", ";\n white-space: nowrap;\n overflow: auto;\n padding: 0 ", "px;\n font-size: ", "px;\n\n .result-link {\n text-decoration: none;\n }\n }\n .cell.end-cell,\n .header-cell.end-cell {\n border-right: none;\n padding-right: ", "px;\n }\n .cell.first-cell,\n .header-cell.first-cell {\n padding-left: ", "px;\n }\n .cell.bottom-cell {\n border-bottom: none;\n }\n .cell.align-right {\n align-items: flex-end;\n }\n }\n\n &:focus {\n outline: none;\n }\n"])), function (props) { return props.theme.dataTableTextColor; }, function (props) { return props.hasCustomScrollBarStyle && props.theme.modalScrollBar; }, function (props) { return props.theme.cellBorderColor; }, browserScrollBarWidth, function (props) { return props.theme.cellBorderColor; }, browserScrollBarWidth, function (props) { return props.theme.pinnedGridBorderColor; }, function (props) { return props.theme.evenRowBackground; }, function (props) { return props.theme.oddRowBackground; }, function (props) { return props.theme.cellBorderColor; }, function (props) { return props.theme.cellBorderColor; }, function (props) { return props.theme.cellPaddingSide; }, function (props) { return props.theme.cellFontSize; }, function (props) { return props.theme.cellPaddingSide + props.theme.edgeCellPaddingSide; }, function (props) { return props.theme.cellPaddingSide + props.theme.edgeCellPaddingSide; }); var defaultColumnWidth = 200; var columnWidthFunction = function columnWidthFunction(columns, cellSizeCache, ghost) { return function (_ref) { var index = _ref.index; return (columns[index] || {}).ghost ? ghost : cellSizeCache[columns[index]] || defaultColumnWidth; }; }; /* * This is an accessor method used to generalize getting a cell from a data row */ var defaultGetRowCell = function defaultGetRowCell(_ref2, formatter) { var dataContainer = _ref2.dataContainer, columns = _ref2.columns, column = _ref2.column, colMeta = _ref2.colMeta, rowIndex = _ref2.rowIndex, sortOrder = _ref2.sortOrder; var rowIdx = sortOrder && sortOrder.length ? (0, _lodash["default"])(sortOrder, rowIndex) : rowIndex; var type = colMeta[column].type; var value = dataContainer === null || dataContainer === void 0 ? void 0 : dataContainer.valueAt(rowIdx, columns.indexOf(column)); return value === null || value === undefined || value === '' ? '' : formatter ? formatter(value) : (0, _utils.parseFieldValue)(value, type); }; var StyledStatsControl = _styledComponents["default"].div(_templateObject2 || (_templateObject2 = (0, _taggedTemplateLiteral2["default"])(["\n height: ", "px;\n width: 100%;\n display: flex;\n justify-content: center;\n align-items: stretch;\n position: absolute;\n top: ", "px;\n font-family: ", "px;\n font-size: 12px;\n color: ", ";\n background-color: ", ";\n &:hover {\n cursor: pointer;\n }\n\n > div {\n padding: 0px 24px;\n display: flex;\n align-items: center;\n\n svg {\n margin-left: 12px;\n transition: transform 0.5s ease;\n transform: rotate(", "deg);\n }\n }\n"])), function (props) { return props.theme.headerStatsControlHeight; }, function (props) { return props.top; }, function (props) { return props.theme.fontFamilyMedium; }, function (props) { return props.theme.activeColor; }, function (props) { return props.theme.headerCellStatsControlBackground; }, function (props) { return props.showStats ? 180 : 0; }); var StatsControl = function StatsControl(_ref3) { var top = _ref3.top, showStats = _ref3.showStats, toggleShowStats = _ref3.toggleShowStats; return /*#__PURE__*/_react["default"].createElement(StyledStatsControl, { top: top, showStats: showStats }, /*#__PURE__*/_react["default"].createElement("div", { onClick: toggleShowStats }, showStats ? 'Hide Column Stats' : 'Show Column Stats', /*#__PURE__*/_react["default"].createElement(_icons.ArrowDown, { height: "18px" }))); }; var TableSection = exports.TableSection = function TableSection(_ref4) { var classList = _ref4.classList, isPinned = _ref4.isPinned, columns = _ref4.columns, headerGridProps = _ref4.headerGridProps, fixedWidth = _ref4.fixedWidth, _ref4$fixedHeight = _ref4.fixedHeight, fixedHeight = _ref4$fixedHeight === void 0 ? undefined : _ref4$fixedHeight, onScroll = _ref4.onScroll, scrollTop = _ref4.scrollTop, dataGridProps = _ref4.dataGridProps, columnWidth = _ref4.columnWidth, _ref4$setGridRef = _ref4.setGridRef, setGridRef = _ref4$setGridRef === void 0 ? undefined : _ref4$setGridRef, headerCellRender = _ref4.headerCellRender, dataCellRender = _ref4.dataCellRender, _ref4$scrollLeft = _ref4.scrollLeft, scrollLeft = _ref4$scrollLeft === void 0 ? 0 : _ref4$scrollLeft; var headerHeight = headerGridProps.height; var headerStyle = (0, _react.useMemo)(function () { return { height: "".concat(headerHeight, "px") }; }, [headerHeight]); var contentStyle = (0, _react.useMemo)(function () { return { top: "".concat(headerHeight, "px") }; }, [headerHeight]); return /*#__PURE__*/_react["default"].createElement(_reactVirtualized.AutoSizer, null, function (_ref5) { var width = _ref5.width, height = _ref5.height; var gridDimension = { columnCount: columns.length, columnWidth: columnWidth, width: fixedWidth || width }; var headerGridWidth = fixedWidth || width; var dataGridHeight = fixedHeight || height; return /*#__PURE__*/_react["default"].createElement(_react["default"].Fragment, null, /*#__PURE__*/_react["default"].createElement("div", { className: (0, _classnames2["default"])('scroll-in-ui-thread', classList === null || classList === void 0 ? void 0 : classList.header), style: headerStyle }, /*#__PURE__*/_react["default"].createElement(_grid["default"], (0, _extends2["default"])({ cellRenderer: headerCellRender }, headerGridProps, gridDimension, { height: headerGridProps.height + browserScrollBarWidth, width: headerGridWidth, scrollLeft: scrollLeft, onScroll: onScroll }))), /*#__PURE__*/_react["default"].createElement("div", { className: (0, _classnames2["default"])('scroll-in-ui-thread', classList === null || classList === void 0 ? void 0 : classList.rows), style: contentStyle }, /*#__PURE__*/_react["default"].createElement(_grid["default"], (0, _extends2["default"])({ cellRenderer: dataCellRender }, dataGridProps, gridDimension, { className: isPinned ? 'pinned-grid' : 'body-grid', height: dataGridHeight - headerGridProps.height, onScroll: onScroll, scrollLeft: scrollLeft, scrollTop: scrollTop, setGridRef: setGridRef })))); }); }; var DUMMY_STYLE = {}; DataTableFactory.deps = [_headerCell["default"]]; function DataTableFactory(HeaderCell) { var DataTable = /*#__PURE__*/function (_Component) { function DataTable() { var _this; (0, _classCallCheck2["default"])(this, DataTable); for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) { args[_key] = arguments[_key]; } _this = _callSuper(this, DataTable, [].concat(args)); (0, _defineProperty2["default"])(_this, "pinnedGrid", null); (0, _defineProperty2["default"])(_this, "unpinnedGrid", null); (0, _defineProperty2["default"])(_this, "hasMounted", false); (0, _defineProperty2["default"])(_this, "state", { cellSizeCache: {}, moreOptionsColumn: null, showStats: true }); (0, _defineProperty2["default"])(_this, "root", /*#__PURE__*/(0, _react.createRef)()); (0, _defineProperty2["default"])(_this, "columns", function (props) { return props.columns; }); (0, _defineProperty2["default"])(_this, "pinnedColumns", function (props) { return props.pinnedColumns; }); (0, _defineProperty2["default"])(_this, "unpinnedColumns", (0, _reselect.createSelector)(_this.columns, _this.pinnedColumns, function (columns, pinnedColumns) { return !Array.isArray(pinnedColumns) ? columns : columns.filter(function (c) { return !pinnedColumns.includes(c); }); })); (0, _defineProperty2["default"])(_this, "toggleMoreOptions", function (moreOptionsColumn) { if (_this.hasMounted) _this.setState({ moreOptionsColumn: _this.state.moreOptionsColumn === moreOptionsColumn ? null : moreOptionsColumn }); }); (0, _defineProperty2["default"])(_this, "toggleShowStats", function () { if (_this.hasMounted) _this.setState({ showStats: !_this.state.showStats }); }); (0, _defineProperty2["default"])(_this, "getCellSizeCache", function () { var _this$props = _this.props, _this$props$cellSizeC = _this$props.cellSizeCache, propsCache = _this$props$cellSizeC === void 0 ? {} : _this$props$cellSizeC, fixedWidth = _this$props.fixedWidth, _this$props$pinnedCol = _this$props.pinnedColumns, pinnedColumns = _this$props$pinnedCol === void 0 ? [] : _this$props$pinnedCol; var unpinnedColumns = _this.unpinnedColumns(_this.props); var width = fixedWidth ? fixedWidth : _this.root.current ? _this.root.current.clientWidth : 0; // pin column border is 2 pixel vs 1 pixel var adjustWidth = pinnedColumns.length ? width - 1 : width; var _ref6 = (0, _cellSize.adjustCellsToContainer)(adjustWidth, propsCache, pinnedColumns, unpinnedColumns), cellSizeCache = _ref6.cellSizeCache, ghost = _ref6.ghost; return { cellSizeCache: cellSizeCache, ghost: ghost }; }); (0, _defineProperty2["default"])(_this, "doScaleCellsToWidth", function () { if (_this.hasMounted) _this.setState(_this.getCellSizeCache()); }); (0, _defineProperty2["default"])(_this, "scaleCellsToWidth", (0, _lodash2["default"])(_this.doScaleCellsToWidth, 300)); (0, _defineProperty2["default"])(_this, "renderDataCell", function (columns, isPinned, props) { var _this$props$getRowCel; var getRowCell = (_this$props$getRowCel = _this.props.getRowCell) !== null && _this$props$getRowCel !== void 0 ? _this$props$getRowCel : defaultGetRowCell; var DataCellRenderer = function DataCellRenderer(cellInfo) { var columnIndex = cellInfo.columnIndex, key = cellInfo.key, style = cellInfo.style, rowIndex = cellInfo.rowIndex; var dataContainer = props.dataContainer, colMeta = props.colMeta; var column = columns[columnIndex]; var isGhost = column.ghost; var formatter = isGhost ? null : (0, _utils.getColumnFormatter)(colMeta[column]); var rowCell = isGhost ? '' : getRowCell(_objectSpread(_objectSpread({}, props), {}, { column: column, rowIndex: rowIndex }), formatter); var type = isGhost ? null : colMeta[column].type; var lastRowIndex = dataContainer ? dataContainer.numRows() - 1 : 0; var endCell = columnIndex === columns.length - 1; var firstCell = columnIndex === 0; var bottomCell = rowIndex === lastRowIndex; var alignRight = fieldToAlignRight[Number(type)]; var cell = /*#__PURE__*/_react["default"].createElement("div", { className: (0, _classnames2["default"])('cell', (0, _defineProperty2["default"])((0, _defineProperty2["default"])((0, _defineProperty2["default"])((0, _defineProperty2["default"])((0, _defineProperty2["default"])((0, _defineProperty2["default"])((0, _defineProperty2["default"])({}, rowIndex % 2 === 0 ? 'even-row' : 'odd-row', true), "row-".concat(rowIndex), true), 'pinned-cell', isPinned), 'first-cell', firstCell), 'end-cell', endCell), 'bottom-cell', bottomCell), 'align-right', alignRight)), key: key, style: style, title: isGhost ? undefined : rowCell }, "".concat(rowCell).concat(endCell ? '\n' : '\t')); return cell; }; return DataCellRenderer; }); return _this; } (0, _inherits2["default"])(DataTable, _Component); return (0, _createClass2["default"])(DataTable, [{ key: "componentDidMount", value: function componentDidMount() { this.hasMounted = true; window.addEventListener('resize', this.scaleCellsToWidth); this.scaleCellsToWidth(); } }, { key: "componentDidUpdate", value: function componentDidUpdate(prevProps) { if (this.props.cellSizeCache !== prevProps.cellSizeCache || this.props.pinnedColumns !== prevProps.pinnedColumns) { this.scaleCellsToWidth(); } } }, { key: "componentWillUnmount", value: function componentWillUnmount() { this.hasMounted = false; window.removeEventListener('resize', this.scaleCellsToWidth); } }, { key: "renderHeaderCell", value: function renderHeaderCell(columns, isPinned, props, toggleMoreOptions, moreOptionsColumn) { var _this2 = this; var HeaderCellRenderer = function HeaderCellRenderer(cellInfo) { return /*#__PURE__*/_react["default"].createElement(HeaderCell, { cellInfo: cellInfo, key: cellInfo.columnIndex, columns: columns, isPinned: isPinned, showStats: _this2.state.showStats, props: props, toggleMoreOptions: toggleMoreOptions, moreOptionsColumn: moreOptionsColumn // pass dummy style to prevent warnings from react-virtualized Grid , style: DUMMY_STYLE }); }; return HeaderCellRenderer; } }, { key: "render", value: function render() { var _this3 = this; var _this$props2 = this.props, dataContainer = _this$props2.dataContainer, _this$props2$pinnedCo = _this$props2.pinnedColumns, pinnedColumns = _this$props2$pinnedCo === void 0 ? [] : _this$props2$pinnedCo, _this$props2$theme = _this$props2.theme, theme = _this$props2$theme === void 0 ? {} : _this$props2$theme, fixedWidth = _this$props2.fixedWidth, _this$props2$fixedHei = _this$props2.fixedHeight, fixedHeight = _this$props2$fixedHei === void 0 ? 0 : _this$props2$fixedHei, hasStats = _this$props2.hasStats, hasCustomScrollBarStyle = _this$props2.hasCustomScrollBarStyle; var unpinnedColumns = this.unpinnedColumns(this.props); var _this$state = this.state, _this$state$cellSizeC = _this$state.cellSizeCache, cellSizeCache = _this$state$cellSizeC === void 0 ? {} : _this$state$cellSizeC, moreOptionsColumn = _this$state.moreOptionsColumn, ghost = _this$state.ghost, showStats = _this$state.showStats; var unpinnedColumnsGhost = ghost ? [].concat((0, _toConsumableArray2["default"])(unpinnedColumns), [{ ghost: true }]) : unpinnedColumns; var pinnedColumnsWidth = pinnedColumns.reduce(function (acc, val) { return acc + (0, _lodash["default"])(cellSizeCache, val, 0); }, 0); var hasPinnedColumns = Boolean(pinnedColumns.length); var _theme$headerRowHeigh = theme.headerRowHeight, headerRowHeight = _theme$headerRowHeigh === void 0 ? defaultHeaderRowHeight : _theme$headerRowHeigh, _theme$headerStatsCon = theme.headerStatsControlHeight, headerStatsControlHeight = _theme$headerStatsCon === void 0 ? defaultHeaderStatsControlHeight : _theme$headerStatsCon, _theme$headerRowWStat = theme.headerRowWStatsHeight, headerRowWStatsHeight = _theme$headerRowWStat === void 0 ? defaultHeaderRowHeight : _theme$headerRowWStat, _theme$rowHeight = theme.rowHeight, rowHeight = _theme$rowHeight === void 0 ? defaultRowHeight : _theme$rowHeight; var headerGridProps = { cellSizeCache: cellSizeCache, className: 'header-grid', height: !hasStats ? headerRowHeight : showStats ? headerRowWStatsHeight : headerRowHeight + headerStatsControlHeight, rowCount: 1, rowHeight: !hasStats ? headerRowHeight : showStats ? headerRowWStatsHeight : headerRowHeight + headerStatsControlHeight }; var dataGridProps = { cellSizeCache: cellSizeCache, overscanColumnCount: overscanColumnCount, overscanRowCount: overscanRowCount, rowCount: dataContainer ? dataContainer.numRows() : 0, rowHeight: rowHeight }; return /*#__PURE__*/_react["default"].createElement(Container, { className: "data-table-container", ref: this.root, hasCustomScrollBarStyle: hasCustomScrollBarStyle }, Object.keys(cellSizeCache).length ? /*#__PURE__*/_react["default"].createElement(_react["default"].Fragment, null, /*#__PURE__*/_react["default"].createElement(_reactVirtualized.ScrollSync, null, function (_ref7) { var _onScroll = _ref7.onScroll, scrollLeft = _ref7.scrollLeft, scrollTop = _ref7.scrollTop; return /*#__PURE__*/_react["default"].createElement("div", { className: "results-table-wrapper" }, hasPinnedColumns && /*#__PURE__*/_react["default"].createElement("div", { key: "pinned-columns", className: "pinned-columns grid-row" }, /*#__PURE__*/_react["default"].createElement(TableSection, { classList: pinnedClassList, isPinned: true, columns: pinnedColumns, headerGridProps: headerGridProps, fixedWidth: pinnedColumnsWidth, onScroll: function onScroll(args) { return _onScroll(_objectSpread(_objectSpread({}, args), {}, { scrollLeft: scrollLeft })); }, scrollTop: scrollTop, scrollLeft: scrollLeft, dataGridProps: dataGridProps, setGridRef: function setGridRef(pinnedGrid) { return _this3.pinnedGrid = pinnedGrid; }, columnWidth: columnWidthFunction(pinnedColumns, cellSizeCache), headerCellRender: _this3.renderHeaderCell(pinnedColumns, true, _this3.props, _this3.toggleMoreOptions, moreOptionsColumn), dataCellRender: _this3.renderDataCell(pinnedColumns, true, _this3.props) })), /*#__PURE__*/_react["default"].createElement("div", { key: "unpinned-columns", style: { marginLeft: "".concat(hasPinnedColumns ? "".concat(pinnedColumnsWidth, "px") : '0') }, className: "unpinned-columns grid-column" }, /*#__PURE__*/_react["default"].createElement(TableSection, { classList: unpinnedClassList, isPinned: false, columns: unpinnedColumnsGhost, headerGridProps: headerGridProps, fixedWidth: fixedWidth, fixedHeight: fixedHeight, onScroll: _onScroll, scrollTop: scrollTop, scrollLeft: scrollLeft, dataGridProps: dataGridProps, setGridRef: function setGridRef(unpinnedGrid) { return _this3.unpinnedGrid = unpinnedGrid; }, columnWidth: columnWidthFunction(unpinnedColumnsGhost, cellSizeCache, ghost), headerCellRender: _this3.renderHeaderCell(unpinnedColumnsGhost, false, _this3.props, _this3.toggleMoreOptions, moreOptionsColumn), dataCellRender: _this3.renderDataCell(unpinnedColumnsGhost, false, _this3.props) }))); }), hasStats ? /*#__PURE__*/_react["default"].createElement(StatsControl, { top: headerRowHeight, showStats: showStats, toggleShowStats: this.toggleShowStats }) : null) : null); } }]); }(_react.Component); (0, _defineProperty2["default"])(DataTable, "defaultProps", { dataContainer: null, pinnedColumns: [], colMeta: {}, cellSizeCache: {}, sortColumn: {}, fixedWidth: null, fixedHeight: null, theme: {}, hasStats: false, hasCustomScrollBarStyle: true }); return (0, _styledComponents.withTheme)(DataTable); } var _default = exports["default"] = DataTableFactory; //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_react","_interopRequireWildcard","require","_reactVirtualized","_styledComponents","_classnames2","_interopRequireDefault","_reselect","_lodash","_lodash2","_icons","_grid","_headerCell","_utils","_cellSize","_constants","_templateObject","_templateObject2","_getRequireWildcardCache","e","WeakMap","r","t","__esModule","_typeof","has","get","n","__proto__","a","Object","defineProperty","getOwnPropertyDescriptor","u","hasOwnProperty","call","i","set","ownKeys","keys","getOwnPropertySymbols","o","filter","enumerable","push","apply","_objectSpread","arguments","length","forEach","_defineProperty2","getOwnPropertyDescriptors","defineProperties","_callSuper","_getPrototypeOf2","_possibleConstructorReturn2","_isNativeReflectConstruct","Reflect","construct","constructor","Boolean","prototype","valueOf","defaultHeaderRowHeight","defaultHeaderStatsControlHeight","defaultRowHeight","overscanColumnCount","overscanRowCount","browserScrollBarWidth","fieldToAlignRight","ALL_FIELD_TYPES","integer","real","pinnedClassList","header","rows","unpinnedClassList","Container","exports","styled","div","_taggedTemplateLiteral2","props","theme","dataTableTextColor","hasCustomScrollBarStyle","modalScrollBar","cellBorderColor","pinnedGridBorderColor","evenRowBackground","oddRowBackground","cellPaddingSide","cellFontSize","edgeCellPaddingSide","defaultColumnWidth","columnWidthFunction","columns","cellSizeCache","ghost","_ref","index","defaultGetRowCell","_ref2","formatter","dataContainer","column","colMeta","rowIndex","sortOrder","rowIdx","type","value","valueAt","indexOf","undefined","parseFieldValue","StyledStatsControl","headerStatsControlHeight","top","fontFamilyMedium","activeColor","headerCellStatsControlBackground","showStats","StatsControl","_ref3","toggleShowStats","createElement","onClick","ArrowDown","height","TableSection","_ref4","classList","isPinned","headerGridProps","fixedWidth","_ref4$fixedHeight","fixedHeight","onScroll","scrollTop","dataGridProps","columnWidth","_ref4$setGridRef","setGridRef","headerCellRender","dataCellRender","_ref4$scrollLeft","scrollLeft","headerHeight","headerStyle","useMemo","concat","contentStyle","AutoSizer","_ref5","width","gridDimension","columnCount","headerGridWidth","dataGridHeight","Fragment","className","classnames","style","_extends2","cellRenderer","DUMMY_STYLE","DataTableFactory","deps","HeaderCellFactory","HeaderCell","DataTable","_Component","_this","_classCallCheck2","_len","args","Array","_key","moreOptionsColumn","createRef","pinnedColumns","createSelector","isArray","c","includes","hasMounted","setState","state","_this$props","_this$props$cellSizeC","propsCache","_this$props$pinnedCol","unpinnedColumns","root","current","clientWidth","adjustWidth","_ref6","adjustCellsToContainer","getCellSizeCache","debounce","doScaleCellsToWidth","_this$props$getRowCel","getRowCell","DataCellRenderer","cellInfo","columnIndex","key","isGhost","getColumnFormatter","rowCell","lastRowIndex","numRows","endCell","firstCell","bottomCell","alignRight","Number","cell","title","_inherits2","_createClass2","componentDidMount","window","addEventListener","scaleCellsToWidth","componentDidUpdate","prevProps","componentWillUnmount","removeEventListener","renderHeaderCell","toggleMoreOptions","_this2","HeaderCellRenderer","render","_this3","_this$props2","_this$props2$pinnedCo","_this$props2$theme","_this$props2$fixedHei","hasStats","_this$state","_this$state$cellSizeC","unpinnedColumnsGhost","_toConsumableArray2","pinnedColumnsWidth","reduce","acc","val","hasPinnedColumns","_theme$headerRowHeigh","headerRowHeight","_theme$headerStatsCon","_theme$headerRowWStat","headerRowWStatsHeight","_theme$rowHeight","rowHeight","rowCount","ref","ScrollSync","_ref7","pinnedGrid","renderDataCell","marginLeft","unpinnedGrid","Component","sortColumn","withTheme","_default"],"sources":["../../../src/common/data-table/index.tsx"],"sourcesContent":["// SPDX-License-Identifier: MIT\n// Copyright contributors to the kepler.gl project\n\nimport React, {Component, createRef, useMemo} from 'react';\nimport {ScrollSync, AutoSizer, OnScrollParams, GridProps, Index} from 'react-virtualized';\nimport styled, {withTheme} from 'styled-components';\nimport classnames from 'classnames';\nimport {createSelector} from 'reselect';\nimport get from 'lodash.get';\nimport debounce from 'lodash.debounce';\nimport {ArrowDown} from '../icons';\n\nimport {CellSizeCache} from './cell-size';\n\nimport Grid from './grid';\nimport HeaderCellFactory from './header-cell';\n\nimport {ColMeta} from '@kepler.gl/types';\nimport {parseFieldValue, getColumnFormatter, DataContainerInterface} from '@kepler.gl/utils';\nimport {adjustCellsToContainer} from './cell-size';\n\nimport {ALL_FIELD_TYPES} from '@kepler.gl/constants';\n\nconst defaultHeaderRowHeight = 55;\nconst defaultHeaderStatsControlHeight = 40;\nconst defaultRowHeight = 32;\nconst overscanColumnCount = 10;\nconst overscanRowCount = 10;\n// The default scrollbar width can range anywhere from 12px to 17px\nconst browserScrollBarWidth = 17;\nconst fieldToAlignRight = {\n  [ALL_FIELD_TYPES.integer]: true,\n  [ALL_FIELD_TYPES.real]: true\n};\n\nconst pinnedClassList = {\n  header: 'pinned-columns--header pinned-grid-container',\n  rows: 'pinned-columns--rows pinned-grid-container'\n};\n\nconst unpinnedClassList = {\n  header: 'unpinned-columns--header unpinned-grid-container',\n  rows: 'unpinned-columns--rows unpinned-grid-container'\n};\n\ntype ContainerProps = {\n  hasCustomScrollBarStyle?: boolean;\n};\n\nexport const Container = styled.div<ContainerProps>`\n  display: flex;\n  font-size: 11px;\n  flex-grow: 1;\n  color: ${props => props.theme.dataTableTextColor};\n  width: 100%;\n  position: relative;\n  .ReactVirtualized__Grid:focus,\n  .ReactVirtualized__Grid:active {\n    outline: 0;\n  }\n  .body-grid {\n    ${props => props.hasCustomScrollBarStyle && props.theme.modalScrollBar}\n  }\n\n  .cell {\n    &::-webkit-scrollbar {\n      display: none;\n    }\n  }\n\n  *:focus {\n    outline: 0;\n  }\n\n  .results-table-wrapper {\n    position: relative;\n    min-height: 100%;\n    max-height: 100%;\n    display: flex;\n    flex-direction: row;\n    flex-grow: 1;\n    overflow: hidden;\n\n    .scroll-in-ui-thread.pinned-columns--header {\n      overflow: hidden;\n      border-bottom: 1px solid ${props => props.theme.cellBorderColor};\n      padding-bottom: ${browserScrollBarWidth}px;\n    }\n    .scroll-in-ui-thread.unpinned-columns--header {\n      width: 100vw;\n      overflow: hidden;\n      border-bottom: 1px solid ${props => props.theme.cellBorderColor};\n      // leave room for scrollbar\n      padding-bottom: ${browserScrollBarWidth}px;\n    }\n\n    .scroll-in-ui-thread::after {\n      content: '';\n      height: 100%;\n      left: 0;\n      position: absolute;\n      pointer-events: none;\n      top: 0;\n      width: 100%;\n    }\n\n    .grid-row {\n      position: relative;\n      display: flex;\n      flex-direction: row;\n    }\n    .grid-column {\n      display: flex;\n      flex-direction: column;\n      flex: 1 1 auto;\n    }\n    .pinned-grid-container {\n      flex: 0 0 75px;\n      z-index: 10;\n      position: absolute;\n      left: 0;\n      top: 0;\n      border-right: 2px solid ${props => props.theme.pinnedGridBorderColor};\n    }\n    .even-row {\n      background-color: ${props => props.theme.evenRowBackground};\n    }\n    .odd-row {\n      background-color: ${props => props.theme.oddRowBackground};\n    }\n    .cell,\n    .header-cell {\n      width: 100%;\n      height: 100%;\n      display: flex;\n      flex-direction: column;\n      justify-content: center;\n      align-items: flex-start;\n      text-align: center;\n      overflow: hidden;\n      // header border is rendered by header container\n      border-bottom: 0;\n      .n-sort-idx {\n        font-size: 9px;\n      }\n    }\n    .cell {\n      border-bottom: 1px solid ${props => props.theme.cellBorderColor};\n      border-right: 1px solid ${props => props.theme.cellBorderColor};\n      white-space: nowrap;\n      overflow: auto;\n      padding: 0 ${props => props.theme.cellPaddingSide}px;\n      font-size: ${props => props.theme.cellFontSize}px;\n\n      .result-link {\n        text-decoration: none;\n      }\n    }\n    .cell.end-cell,\n    .header-cell.end-cell {\n      border-right: none;\n      padding-right: ${props => props.theme.cellPaddingSide + props.theme.edgeCellPaddingSide}px;\n    }\n    .cell.first-cell,\n    .header-cell.first-cell {\n      padding-left: ${props => props.theme.cellPaddingSide + props.theme.edgeCellPaddingSide}px;\n    }\n    .cell.bottom-cell {\n      border-bottom: none;\n    }\n    .cell.align-right {\n      align-items: flex-end;\n    }\n  }\n\n  &:focus {\n    outline: none;\n  }\n`;\n\nconst defaultColumnWidth = 200;\n\nexport type SortColumn = {\n  column?: string;\n  mode?: string;\n};\n\nconst columnWidthFunction =\n  (columns, cellSizeCache, ghost?) =>\n  ({index}) => {\n    return (columns[index] || {}).ghost\n      ? ghost\n      : cellSizeCache[columns[index]] || defaultColumnWidth;\n  };\n\ninterface GetRowCellProps {\n  dataContainer: DataContainerInterface | null;\n  columns: (string & {ghost?: boolean})[];\n  column: string;\n  colMeta;\n  rowIndex: number;\n  sortOrder?: number[] | null;\n}\n\n/*\n * This is an accessor method used to generalize getting a cell from a data row\n */\nconst defaultGetRowCell = (\n  {dataContainer, columns, column, colMeta, rowIndex, sortOrder}: GetRowCellProps,\n  formatter\n) => {\n  const rowIdx = sortOrder && sortOrder.length ? get(sortOrder, rowIndex) : rowIndex;\n  const {type} = colMeta[column];\n\n  const value = dataContainer?.valueAt(rowIdx, columns.indexOf(column));\n  return value === null || value === undefined || value === ''\n    ? ''\n    : formatter\n    ? formatter(value)\n    : parseFieldValue(value, type);\n};\n\ntype StatsControlProps = {\n  top: number;\n  showStats?: boolean;\n};\n\nconst StyledStatsControl = styled.div<StatsControlProps>`\n  height: ${props => props.theme.headerStatsControlHeight}px;\n  width: 100%;\n  display: flex;\n  justify-content: center;\n  align-items: stretch;\n  position: absolute;\n  top: ${props => props.top}px;\n  font-family: ${props => props.theme.fontFamilyMedium}px;\n  font-size: 12px;\n  color: ${props => props.theme.activeColor};\n  background-color: ${props => props.theme.headerCellStatsControlBackground};\n  &:hover {\n    cursor: pointer;\n  }\n\n  > div {\n    padding: 0px 24px;\n    display: flex;\n    align-items: center;\n\n    svg {\n      margin-left: 12px;\n      transition: transform 0.5s ease;\n      transform: rotate(${props => (props.showStats ? 180 : 0)}deg);\n    }\n  }\n`;\n\nconst StatsControl = ({\n  top,\n  showStats,\n  toggleShowStats\n}: {\n  top: number;\n  showStats?: boolean;\n  toggleShowStats: () => void;\n}) => (\n  <StyledStatsControl top={top} showStats={showStats}>\n    <div onClick={toggleShowStats}>\n      {showStats ? 'Hide Column Stats' : 'Show Column Stats'}\n      <ArrowDown height=\"18px\" />\n    </div>\n  </StyledStatsControl>\n);\n\ninterface TableSectionProps {\n  classList?: {\n    header: string;\n    rows: string;\n  };\n  isPinned?: boolean;\n  columns: (string & {ghost?: boolean})[];\n  headerGridProps?;\n  fixedWidth?: number | null;\n  fixedHeight?: number | null;\n  onScroll?: (params: OnScrollParams) => void;\n  scrollTop?: number;\n  dataGridProps: {\n    rowHeight: number | ((params: Index) => number);\n    rowCount: number;\n  } & Partial<GridProps>;\n  columnWidth?;\n  setGridRef?: (ref: HTMLDivElement | null) => void;\n  headerCellRender?;\n  dataCellRender?;\n  scrollLeft?: number;\n}\n\nexport const TableSection = ({\n  classList,\n  isPinned,\n  columns,\n  headerGridProps,\n  fixedWidth,\n  fixedHeight = undefined,\n  onScroll,\n  scrollTop,\n  dataGridProps,\n  columnWidth,\n  setGridRef = undefined,\n  headerCellRender,\n  dataCellRender,\n  scrollLeft = 0\n}: TableSectionProps) => {\n  const headerHeight = headerGridProps.height;\n\n  const headerStyle = useMemo(\n    () => ({\n      height: `${headerHeight}px`\n    }),\n    [headerHeight]\n  );\n  const contentStyle = useMemo(\n    () => ({\n      top: `${headerHeight}px`\n    }),\n    [headerHeight]\n  );\n\n  return (\n    <AutoSizer>\n      {({width, height}) => {\n        const gridDimension = {\n          columnCount: columns.length,\n          columnWidth,\n          width: fixedWidth || width\n        };\n        const headerGridWidth = fixedWidth || width;\n        const dataGridHeight = fixedHeight || height;\n\n        return (\n          <>\n            <div\n              className={classnames('scroll-in-ui-thread', classList?.header)}\n              style={headerStyle}\n            >\n              <Grid\n                cellRenderer={headerCellRender}\n                {...headerGridProps}\n                {...gridDimension}\n                height={headerGridProps.height + browserScrollBarWidth}\n                width={headerGridWidth}\n                scrollLeft={scrollLeft}\n                onScroll={onScroll}\n              />\n            </div>\n            <div\n              className={classnames('scroll-in-ui-thread', classList?.rows)}\n              style={contentStyle}\n            >\n              <Grid\n                cellRenderer={dataCellRender}\n                {...dataGridProps}\n                {...gridDimension}\n                className={isPinned ? 'pinned-grid' : 'body-grid'}\n                height={dataGridHeight - headerGridProps.height}\n                onScroll={onScroll}\n                scrollLeft={scrollLeft}\n                scrollTop={scrollTop}\n                setGridRef={setGridRef}\n              />\n            </div>\n          </>\n        );\n      }}\n    </AutoSizer>\n  );\n};\n\nexport interface DataTableProps {\n  dataId?: string;\n  hasStats?: boolean;\n  cellSizeCache?: CellSizeCache;\n  pinnedColumns?: string[];\n  columns: (string & {ghost?: boolean})[];\n  fixedWidth?: number | null;\n  theme?: any;\n  dataContainer: DataContainerInterface | null;\n  fixedHeight?: number | null;\n  colMeta: ColMeta;\n  sortColumn: SortColumn;\n  sortTableColumn: (column: string, mode?: string) => void;\n  pinTableColumn: (column: string) => void;\n  setColumnDisplayFormat?: (formats: {[key: string]: string}) => void;\n  copyTableColumn: (column: string) => void;\n  sortOrder?: number[] | null;\n  showStats?: boolean;\n  hasCustomScrollBarStyle?: boolean;\n  getRowCell?: (renderDataCellProps: GetRowCellProps, formatter: any) => string | number;\n}\n\ninterface DataTableState {\n  cellSizeCache?: CellSizeCache;\n  moreOptionsColumn?;\n  ghost?;\n  showStats?: boolean;\n}\n\nconst DUMMY_STYLE = {};\n\nDataTableFactory.deps = [HeaderCellFactory];\nfunction DataTableFactory(\n  HeaderCell: ReturnType<typeof HeaderCellFactory>\n): React.ComponentType<DataTableProps> {\n  class DataTable extends Component<DataTableProps, DataTableState> {\n    static defaultProps = {\n      dataContainer: null,\n      pinnedColumns: [],\n      colMeta: {},\n      cellSizeCache: {},\n      sortColumn: {},\n      fixedWidth: null,\n      fixedHeight: null,\n      theme: {},\n      hasStats: false,\n      hasCustomScrollBarStyle: true\n    };\n\n    pinnedGrid: HTMLDivElement | null = null;\n    unpinnedGrid: HTMLDivElement | null = null;\n    hasMounted = false;\n\n    state: DataTableState = {\n      cellSizeCache: {},\n      moreOptionsColumn: null,\n      showStats: true\n    };\n\n    componentDidMount() {\n      this.hasMounted = true;\n      window.addEventListener('resize', this.scaleCellsToWidth);\n      this.scaleCellsToWidth();\n    }\n\n    componentDidUpdate(prevProps) {\n      if (\n        this.props.cellSizeCache !== prevProps.cellSizeCache ||\n        this.props.pinnedColumns !== prevProps.pinnedColumns\n      ) {\n        this.scaleCellsToWidth();\n      }\n    }\n\n    componentWillUnmount() {\n      this.hasMounted = false;\n      window.removeEventListener('resize', this.scaleCellsToWidth);\n    }\n\n    root = createRef<HTMLDivElement>();\n    columns = (props: DataTableProps) => props.columns;\n    pinnedColumns = (props: DataTableProps) => props.pinnedColumns;\n    unpinnedColumns = createSelector(this.columns, this.pinnedColumns, (columns, pinnedColumns) =>\n      !Array.isArray(pinnedColumns) ? columns : columns.filter(c => !pinnedColumns.includes(c))\n    );\n\n    toggleMoreOptions = moreOptionsColumn => {\n      if (this.hasMounted)\n        this.setState({\n          moreOptionsColumn:\n            this.state.moreOptionsColumn === moreOptionsColumn ? null : moreOptionsColumn\n        });\n    };\n    toggleShowStats = () => {\n      if (this.hasM