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
JavaScript
"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