UNPKG

kepler.gl.geoiq

Version:

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

277 lines (217 loc) 28.2 kB
"use strict"; var _interopRequireWildcard = require("@babel/runtime/helpers/interopRequireWildcard"); var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault"); Object.defineProperty(exports, "__esModule", { value: true }); exports["default"] = exports.DatasetTabs = exports.DatasetModalTab = exports.DataTableModal = void 0; 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 _getPrototypeOf3 = _interopRequireDefault(require("@babel/runtime/helpers/getPrototypeOf")); var _assertThisInitialized2 = _interopRequireDefault(require("@babel/runtime/helpers/assertThisInitialized")); var _inherits2 = _interopRequireDefault(require("@babel/runtime/helpers/inherits")); var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty")); var _taggedTemplateLiteral2 = _interopRequireDefault(require("@babel/runtime/helpers/taggedTemplateLiteral")); var _react = _interopRequireWildcard(require("react")); var _styledComponents = _interopRequireDefault(require("styled-components")); var _window = _interopRequireDefault(require("global/window")); var _defaultSettings = require("../../constants/default-settings"); var _fieldToken = _interopRequireDefault(require("../common/field-token")); var _datasetLabel = _interopRequireDefault(require("../common/dataset-label")); var _icons = require("../common/icons"); function _templateObject3() { var data = (0, _taggedTemplateLiteral2["default"])(["\n align-items: center;\n border-bottom: 3px solid ", ";\n cursor: pointer;\n display: flex;\n height: 35px;\n margin: 0 3px;\n padding: 0 5px;\n\n :first-child {\n margin-left: 0;\n padding-left: 0;\n }\n"]); _templateObject3 = function _templateObject3() { return data; }; return data; } function _templateObject2() { var data = (0, _taggedTemplateLiteral2["default"])(["\n display: flex;\n padding: 0 ", ";\n"]); _templateObject2 = function _templateObject2() { return data; }; return data; } function ownKeys(object, enumerableOnly) { var keys = Object.keys(object); if (Object.getOwnPropertySymbols) { var symbols = Object.getOwnPropertySymbols(object); if (enumerableOnly) symbols = symbols.filter(function (sym) { return Object.getOwnPropertyDescriptor(object, sym).enumerable; }); keys.push.apply(keys, symbols); } return keys; } function _objectSpread(target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i] != null ? arguments[i] : {}; if (i % 2) { ownKeys(Object(source), true).forEach(function (key) { (0, _defineProperty2["default"])(target, key, source[key]); }); } else if (Object.getOwnPropertyDescriptors) { Object.defineProperties(target, Object.getOwnPropertyDescriptors(source)); } else { ownKeys(Object(source)).forEach(function (key) { Object.defineProperty(target, key, Object.getOwnPropertyDescriptor(source, key)); }); } } return target; } function _templateObject() { var data = (0, _taggedTemplateLiteral2["default"])(["\n .react-grid-Main {\n outline: 0;\n }\n\n .react-grid-Grid {\n border: 0;\n }\n\n .react-grid-Cell {\n border-right: 0;\n border-bottom: ", ";\n padding-left: 16px;\n }\n\n .react-grid-HeaderCell {\n border-right: 0;\n border-bottom: 0;\n background: ", ";\n color: ", ";\n padding: 14px 8px 14px 0;\n }\n .react-grid-Cell:first-child,\n .react-grid-HeaderCell:first-child {\n padding-left: ", ";\n }\n .react-grid-Cell:last-child,\n .react-grid-HeaderCell:last-child {\n padding-right: ", ";\n }\n .react-grid-Cell__value {\n color: ", ";\n }\n .react-grid-Canvas {\n ", ";\n }\n"]); _templateObject = function _templateObject() { return data; }; return data; } var ReactDataGrid = _window["default"].navigator ? require('react-data-grid/dist/react-data-grid.min') : null; var shouldPreventScrollBack = false; if (_window["default"].navigator && _window["default"].navigator.userAgent) { var navigator = _window["default"].navigator; // Detect browsers // http://stackoverflow.com/questions/5899783/detect-safari-using-jquery var isMac = navigator.userAgent.match(/Macintosh/); var is_chrome = navigator.userAgent.indexOf('Chrome') > -1; var is_safari = navigator.userAgent.indexOf('Safari') > -1; var is_firefox = navigator.userAgent.indexOf('Firefox') > -1; // prevent chrome scroll back shouldPreventScrollBack = isMac && (is_chrome || is_safari || is_firefox); } var dgSettings = { sidePadding: '38px' }; var DataGridWrapper = _styledComponents["default"].div(_templateObject(), function (props) { return props.theme.panelBorderLT; }, function (props) { return props.theme.panelBackgroundLT; }, function (props) { return props.theme.titleColorLT; }, dgSettings.sidePadding, dgSettings.sidePadding, function (props) { return props.theme.labelColorLT; }, function (props) { return props.theme.modalScrollBar; }); var BooleanFormatter = function BooleanFormatter(_ref) { var value = _ref.value; return _react["default"].createElement("span", null, String(value)); }; var DataTableModal = /*#__PURE__*/ function (_Component) { (0, _inherits2["default"])(DataTableModal, _Component); function DataTableModal() { var _getPrototypeOf2; var _this; (0, _classCallCheck2["default"])(this, DataTableModal); for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) { args[_key] = arguments[_key]; } _this = (0, _possibleConstructorReturn2["default"])(this, (_getPrototypeOf2 = (0, _getPrototypeOf3["default"])(DataTableModal)).call.apply(_getPrototypeOf2, [this].concat(args))); (0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "_onMouseWheel", function (e) { // Prevent futile scroll, which would trigger the Back/Next page event // https://github.com/micho/jQuery.preventMacBackScroll // This prevents scroll when reaching the topmost or leftmost // positions of a container. // react-data-grid canvas element can be scrolled var canvas = _this._root.querySelector('.react-grid-Canvas'); // If canvas can not be scrolled left anymore when we try to scroll left var prevent_left = e.deltaX < 0 && canvas.scrollLeft <= 0; // If canvas can not be scrolled up when we try to scroll up var prevent_up = e.deltaY < 0 && canvas.scrollTop <= 0; if (prevent_left || prevent_up) { e.preventDefault(); } }); return _this; } (0, _createClass2["default"])(DataTableModal, [{ key: "render", value: function render() { var _this2 = this; var _this$props = this.props, datasets = _this$props.datasets, dataId = _this$props.dataId, showDatasetTable = _this$props.showDatasetTable; if (!datasets || !dataId) { return null; } var activeDataset = datasets[dataId]; // TODO: this should be all data var rows = activeDataset.data; var columns = activeDataset.fields.map(function (field, i) { return _objectSpread({}, field, { key: i, headerRenderer: _react["default"].createElement(FieldHeader, field), resizable: true, formatter: field.type === _defaultSettings.ALL_FIELD_TYPES["boolean"] ? BooleanFormatter : undefined }); }).filter(function (_ref2) { var name = _ref2.name; return name !== '_geojson'; }); return _react["default"].createElement("div", { ref: function ref(_ref3) { _this2._root = _ref3; }, className: "dataset-modal", style: { overflow: 'scroll' } }, _react["default"].createElement(DatasetTabs, { activeDataset: activeDataset, datasets: datasets, showDatasetTable: showDatasetTable }), _react["default"].createElement(DataGridWrapper, { onWheel: shouldPreventScrollBack ? this._onMouseWheel : null }, ReactDataGrid ? _react["default"].createElement(ReactDataGrid, { headerRowHeight: 72, columns: columns, minColumnWidth: 172, minWidth: this.props.width, minHeight: this.props.height - 65, rowGetter: function rowGetter(i) { return rows[i]; }, rowHeight: 48, rowsCount: rows.length }) : null)); } }]); return DataTableModal; }(_react.Component); exports.DataTableModal = DataTableModal; var tagContainerStyle = { display: 'flex', flexDirection: 'column', justifyContent: 'space-between' }; var FieldHeader = function FieldHeader(_ref4) { var name = _ref4.name, type = _ref4.type; return _react["default"].createElement("div", { style: tagContainerStyle }, _react["default"].createElement("div", { style: { display: 'flex', alignItems: 'center' } }, _react["default"].createElement("div", { style: { marginRight: type === 'timestamp' ? '2px' : '18px', height: '16px' } }, type === 'timestamp' ? _react["default"].createElement(_icons.Clock, { height: "16px" }) : null), name), _react["default"].createElement("div", { style: { marginLeft: '18px' } }, _react["default"].createElement(_fieldToken["default"], { type: type }))); }; var DatasetCatalog = _styledComponents["default"].div(_templateObject2(), dgSettings.sidePadding); var DatasetModalTab = _styledComponents["default"].div(_templateObject3(), function (props) { return props.active ? 'black' : 'transparent'; }); exports.DatasetModalTab = DatasetModalTab; var DatasetTabs = function DatasetTabs(_ref5) { var activeDataset = _ref5.activeDataset, datasets = _ref5.datasets, showDatasetTable = _ref5.showDatasetTable; return _react["default"].createElement(DatasetCatalog, { className: "dataset-modal-catalog" }, Object.values(datasets).map(function (dataset) { return dataset.externalData && dataset.externalData === true ? null : _react["default"].createElement(DatasetModalTab, { className: "dataset-modal-tab", active: dataset === activeDataset, key: dataset.id, onClick: function onClick() { return showDatasetTable(dataset.id); } }, _react["default"].createElement(_datasetLabel["default"], { dataset: dataset })); })); }; exports.DatasetTabs = DatasetTabs; var DataTableModalFactory = function DataTableModalFactory() { return DataTableModal; }; var _default = DataTableModalFactory; exports["default"] = _default; //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/components/modals/data-table-modal.js"],"names":["ReactDataGrid","window","navigator","require","shouldPreventScrollBack","userAgent","isMac","match","is_chrome","indexOf","is_safari","is_firefox","dgSettings","sidePadding","DataGridWrapper","styled","div","props","theme","panelBorderLT","panelBackgroundLT","titleColorLT","labelColorLT","modalScrollBar","BooleanFormatter","value","String","DataTableModal","e","canvas","_root","querySelector","prevent_left","deltaX","scrollLeft","prevent_up","deltaY","scrollTop","preventDefault","datasets","dataId","showDatasetTable","activeDataset","rows","data","columns","fields","map","field","i","key","headerRenderer","resizable","formatter","type","ALL_FIELD_TYPES","undefined","filter","name","ref","overflow","_onMouseWheel","width","height","length","Component","tagContainerStyle","display","flexDirection","justifyContent","FieldHeader","alignItems","marginRight","marginLeft","DatasetCatalog","DatasetModalTab","active","DatasetTabs","Object","values","dataset","externalData","id","DataTableModalFactory"],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;AAoBA;;AACA;;AACA;;AAEA;;AACA;;AACA;;AACA;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AACA,IAAMA,aAAa,GAAGC,mBAAOC,SAAP,GAClBC,OAAO,CAAC,0CAAD,CADW,GAElB,IAFJ;AAIA,IAAIC,uBAAuB,GAAG,KAA9B;;AAEA,IAAIH,mBAAOC,SAAP,IAAoBD,mBAAOC,SAAP,CAAiBG,SAAzC,EAAoD;AAAA,MAC3CH,SAD2C,GAC9BD,kBAD8B,CAC3CC,SAD2C,EAElD;AACA;;AACA,MAAMI,KAAK,GAAGJ,SAAS,CAACG,SAAV,CAAoBE,KAApB,CAA0B,WAA1B,CAAd;AACA,MAAMC,SAAS,GAAGN,SAAS,CAACG,SAAV,CAAoBI,OAApB,CAA4B,QAA5B,IAAwC,CAAC,CAA3D;AACA,MAAMC,SAAS,GAAGR,SAAS,CAACG,SAAV,CAAoBI,OAApB,CAA4B,QAA5B,IAAwC,CAAC,CAA3D;AACA,MAAME,UAAU,GAAGT,SAAS,CAACG,SAAV,CAAoBI,OAApB,CAA4B,SAA5B,IAAyC,CAAC,CAA7D,CAPkD,CASlD;;AACAL,EAAAA,uBAAuB,GAAGE,KAAK,KAAKE,SAAS,IAAIE,SAAb,IAA0BC,UAA/B,CAA/B;AACD;;AAED,IAAMC,UAAU,GAAG;AACjBC,EAAAA,WAAW,EAAE;AADI,CAAnB;;AAIA,IAAMC,eAAe,GAAGC,6BAAOC,GAAV,oBAWA,UAAAC,KAAK;AAAA,SAAIA,KAAK,CAACC,KAAN,CAAYC,aAAhB;AAAA,CAXL,EAkBH,UAAAF,KAAK;AAAA,SAAIA,KAAK,CAACC,KAAN,CAAYE,iBAAhB;AAAA,CAlBF,EAmBR,UAAAH,KAAK;AAAA,SAAIA,KAAK,CAACC,KAAN,CAAYG,YAAhB;AAAA,CAnBG,EAwBDT,UAAU,CAACC,WAxBV,EA4BAD,UAAU,CAACC,WA5BX,EA+BR,UAAAI,KAAK;AAAA,SAAIA,KAAK,CAACC,KAAN,CAAYI,YAAhB;AAAA,CA/BG,EAkCf,UAAAL,KAAK;AAAA,SAAIA,KAAK,CAACC,KAAN,CAAYK,cAAhB;AAAA,CAlCU,CAArB;;AAsCA,IAAMC,gBAAgB,GAAG,SAAnBA,gBAAmB;AAAA,MAAEC,KAAF,QAAEA,KAAF;AAAA,SAAa,8CAAOC,MAAM,CAACD,KAAD,CAAb,CAAb;AAAA,CAAzB;;IAEaE,c;;;;;;;;;;;;;;;;;sGACK,UAAAC,CAAC,EAAI;AACnB;AACA;AACA;AACA;AAEA;AACA,UAAMC,MAAM,GAAG,MAAKC,KAAL,CAAWC,aAAX,CAAyB,oBAAzB,CAAf,CAPmB,CASnB;;;AACA,UAAMC,YAAY,GAAGJ,CAAC,CAACK,MAAF,GAAW,CAAX,IAAgBJ,MAAM,CAACK,UAAP,IAAqB,CAA1D,CAVmB,CAWnB;;AACA,UAAMC,UAAU,GAAGP,CAAC,CAACQ,MAAF,GAAW,CAAX,IAAgBP,MAAM,CAACQ,SAAP,IAAoB,CAAvD;;AAEA,UAAIL,YAAY,IAAIG,UAApB,EAAgC;AAC9BP,QAAAA,CAAC,CAACU,cAAF;AACD;AACF,K;;;;;;6BAEQ;AAAA;;AAAA,wBACsC,KAAKrB,KAD3C;AAAA,UACAsB,QADA,eACAA,QADA;AAAA,UACUC,MADV,eACUA,MADV;AAAA,UACkBC,gBADlB,eACkBA,gBADlB;;AAGP,UAAI,CAACF,QAAD,IAAa,CAACC,MAAlB,EAA0B;AACxB,eAAO,IAAP;AACD;;AAED,UAAME,aAAa,GAAGH,QAAQ,CAACC,MAAD,CAA9B,CAPO,CAQP;;AACA,UAAMG,IAAI,GAAGD,aAAa,CAACE,IAA3B;AACA,UAAMC,OAAO,GAAGH,aAAa,CAACI,MAAd,CACbC,GADa,CACT,UAACC,KAAD,EAAQC,CAAR;AAAA,iCACAD,KADA;AAEHE,UAAAA,GAAG,EAAED,CAFF;AAGHE,UAAAA,cAAc,EAAE,gCAAC,WAAD,EAAiBH,KAAjB,CAHb;AAIHI,UAAAA,SAAS,EAAE,IAJR;AAKHC,UAAAA,SAAS,EACPL,KAAK,CAACM,IAAN,KAAeC,2CAAf,GAAyC/B,gBAAzC,GAA4DgC;AAN3D;AAAA,OADS,EASbC,MATa,CASN;AAAA,YAAEC,IAAF,SAAEA,IAAF;AAAA,eAAYA,IAAI,KAAK,UAArB;AAAA,OATM,CAAhB;AAWA,aACE;AACE,QAAA,GAAG,EAAE,aAAAC,KAAG,EAAI;AACV,UAAA,MAAI,CAAC7B,KAAL,GAAa6B,KAAb;AACD,SAHH;AAIE,QAAA,SAAS,EAAC,eAJZ;AAKE,QAAA,KAAK,EAAE;AAACC,UAAAA,QAAQ,EAAE;AAAX;AALT,SAOE,gCAAC,WAAD;AACE,QAAA,aAAa,EAAElB,aADjB;AAEE,QAAA,QAAQ,EAAEH,QAFZ;AAGE,QAAA,gBAAgB,EAAEE;AAHpB,QAPF,EAYE,gCAAC,eAAD;AACE,QAAA,OAAO,EAAErC,uBAAuB,GAAG,KAAKyD,aAAR,GAAwB;AAD1D,SAGG7D,aAAa,GACZ,gCAAC,aAAD;AACE,QAAA,eAAe,EAAE,EADnB;AAEE,QAAA,OAAO,EAAE6C,OAFX;AAGE,QAAA,cAAc,EAAE,GAHlB;AAIE,QAAA,QAAQ,EAAE,KAAK5B,KAAL,CAAW6C,KAJvB;AAKE,QAAA,SAAS,EAAE,KAAK7C,KAAL,CAAW8C,MAAX,GAAoB,EALjC;AAME,QAAA,SAAS,EAAE,mBAAAd,CAAC;AAAA,iBAAIN,IAAI,CAACM,CAAD,CAAR;AAAA,SANd;AAOE,QAAA,SAAS,EAAE,EAPb;AAQE,QAAA,SAAS,EAAEN,IAAI,CAACqB;AARlB,QADY,GAWV,IAdN,CAZF,CADF;AA+BD;;;EAxEiCC,gB;;;AA2EpC,IAAMC,iBAAiB,GAAG;AACxBC,EAAAA,OAAO,EAAE,MADe;AAExBC,EAAAA,aAAa,EAAE,QAFS;AAGxBC,EAAAA,cAAc,EAAE;AAHQ,CAA1B;;AAMA,IAAMC,WAAW,GAAG,SAAdA,WAAc;AAAA,MAAEZ,IAAF,SAAEA,IAAF;AAAA,MAAQJ,IAAR,SAAQA,IAAR;AAAA,SAClB;AAAK,IAAA,KAAK,EAAEY;AAAZ,KACE;AAAK,IAAA,KAAK,EAAE;AAACC,MAAAA,OAAO,EAAE,MAAV;AAAkBI,MAAAA,UAAU,EAAE;AAA9B;AAAZ,KACE;AACE,IAAA,KAAK,EAAE;AACLC,MAAAA,WAAW,EAAElB,IAAI,KAAK,WAAT,GAAuB,KAAvB,GAA+B,MADvC;AAELS,MAAAA,MAAM,EAAE;AAFH;AADT,KAMGT,IAAI,KAAK,WAAT,GAAuB,gCAAC,YAAD;AAAO,IAAA,MAAM,EAAC;AAAd,IAAvB,GAAiD,IANpD,CADF,EASGI,IATH,CADF,EAYE;AAAK,IAAA,KAAK,EAAE;AAACe,MAAAA,UAAU,EAAE;AAAb;AAAZ,KACE,gCAAC,sBAAD;AAAY,IAAA,IAAI,EAAEnB;AAAlB,IADF,CAZF,CADkB;AAAA,CAApB;;AAmBA,IAAMoB,cAAc,GAAG3D,6BAAOC,GAAV,qBAELJ,UAAU,CAACC,WAFN,CAApB;;AAKO,IAAM8D,eAAe,GAAG5D,6BAAOC,GAAV,qBAEC,UAAAC,KAAK;AAAA,SAAKA,KAAK,CAAC2D,MAAN,GAAe,OAAf,GAAyB,aAA9B;AAAA,CAFN,CAArB;;;;AAeA,IAAMC,WAAW,GAAG,SAAdA,WAAc;AAAA,MAAEnC,aAAF,SAAEA,aAAF;AAAA,MAAiBH,QAAjB,SAAiBA,QAAjB;AAAA,MAA2BE,gBAA3B,SAA2BA,gBAA3B;AAAA,SACzB,gCAAC,cAAD;AAAgB,IAAA,SAAS,EAAC;AAA1B,KACGqC,MAAM,CAACC,MAAP,CAAcxC,QAAd,EAAwBQ,GAAxB,CAA4B,UAAAiC,OAAO;AAAA,WAClCA,OAAO,CAACC,YAAR,IAAwBD,OAAO,CAACC,YAAR,KAAyB,IAAjD,GAAwD,IAAxD,GACE,gCAAC,eAAD;AACE,MAAA,SAAS,EAAC,mBADZ;AAEE,MAAA,MAAM,EAAED,OAAO,KAAKtC,aAFtB;AAGE,MAAA,GAAG,EAAEsC,OAAO,CAACE,EAHf;AAIE,MAAA,OAAO,EAAE;AAAA,eAAMzC,gBAAgB,CAACuC,OAAO,CAACE,EAAT,CAAtB;AAAA;AAJX,OAME,gCAAC,wBAAD;AAAc,MAAA,OAAO,EAAEF;AAAvB,MANF,CAFgC;AAAA,GAAnC,CADH,CADyB;AAAA,CAApB;;;;AAiBP,IAAMG,qBAAqB,GAAG,SAAxBA,qBAAwB;AAAA,SAAMxD,cAAN;AAAA,CAA9B;;eACewD,qB","sourcesContent":["// Copyright (c) 2019 Uber Technologies, Inc.\n//\n// Permission is hereby granted, free of charge, to any person obtaining a copy\n// of this software and associated documentation files (the \"Software\"), to deal\n// in the Software without restriction, including without limitation the rights\n// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell\n// copies of the Software, and to permit persons to whom the Software is\n// furnished to do so, subject to the following conditions:\n//\n// The above copyright notice and this permission notice shall be included in\n// all copies or substantial portions of the Software.\n//\n// THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\n// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\n// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN\n// THE SOFTWARE.\n\nimport React, {Component} from 'react';\nimport styled from 'styled-components';\nimport window from 'global/window';\n\nimport {ALL_FIELD_TYPES} from 'constants/default-settings';\nimport FieldToken from 'components/common/field-token';\nimport DatasetLabel from 'components/common/dataset-label';\nimport {Clock} from 'components/common/icons/index';\nconst ReactDataGrid = window.navigator\n  ? require('react-data-grid/dist/react-data-grid.min')\n  : null;\n\nlet shouldPreventScrollBack = false;\n\nif (window.navigator && window.navigator.userAgent) {\n  const {navigator} = window;\n  // Detect browsers\n  // http://stackoverflow.com/questions/5899783/detect-safari-using-jquery\n  const isMac = navigator.userAgent.match(/Macintosh/);\n  const is_chrome = navigator.userAgent.indexOf('Chrome') > -1;\n  const is_safari = navigator.userAgent.indexOf('Safari') > -1;\n  const is_firefox = navigator.userAgent.indexOf('Firefox') > -1;\n\n  // prevent chrome scroll back\n  shouldPreventScrollBack = isMac && (is_chrome || is_safari || is_firefox);\n}\n\nconst dgSettings = {\n  sidePadding: '38px'\n};\n\nconst DataGridWrapper = styled.div`\n  .react-grid-Main {\n    outline: 0;\n  }\n\n  .react-grid-Grid {\n    border: 0;\n  }\n\n  .react-grid-Cell {\n    border-right: 0;\n    border-bottom: ${props => props.theme.panelBorderLT};\n    padding-left: 16px;\n  }\n\n  .react-grid-HeaderCell {\n    border-right: 0;\n    border-bottom: 0;\n    background: ${props => props.theme.panelBackgroundLT};\n    color: ${props => props.theme.titleColorLT};\n    padding: 14px 8px 14px 0;\n  }\n  .react-grid-Cell:first-child,\n  .react-grid-HeaderCell:first-child {\n    padding-left: ${dgSettings.sidePadding};\n  }\n  .react-grid-Cell:last-child,\n  .react-grid-HeaderCell:last-child {\n    padding-right: ${dgSettings.sidePadding};\n  }\n  .react-grid-Cell__value {\n    color: ${props => props.theme.labelColorLT};\n  }\n  .react-grid-Canvas {\n    ${props => props.theme.modalScrollBar};\n  }\n`;\n\nconst BooleanFormatter = ({value}) => <span>{String(value)}</span>;\n\nexport class DataTableModal extends Component {\n  _onMouseWheel = e => {\n    // Prevent futile scroll, which would trigger the Back/Next page event\n    // https://github.com/micho/jQuery.preventMacBackScroll\n    // This prevents scroll when reaching the topmost or leftmost\n    // positions of a container.\n\n    // react-data-grid canvas element can be scrolled\n    const canvas = this._root.querySelector('.react-grid-Canvas');\n\n    // If canvas can not be scrolled left anymore when we try to scroll left\n    const prevent_left = e.deltaX < 0 && canvas.scrollLeft <= 0;\n    // If canvas can not be scrolled up when we try to scroll up\n    const prevent_up = e.deltaY < 0 && canvas.scrollTop <= 0;\n\n    if (prevent_left || prevent_up) {\n      e.preventDefault();\n    }\n  };\n\n  render() {\n    const {datasets, dataId, showDatasetTable} = this.props;\n\n    if (!datasets || !dataId) {\n      return null;\n    }\n\n    const activeDataset = datasets[dataId];\n    // TODO: this should be all data\n    const rows = activeDataset.data;\n    const columns = activeDataset.fields\n      .map((field, i) => ({\n        ...field,\n        key: i,\n        headerRenderer: <FieldHeader {...field} />,\n        resizable: true,\n        formatter:\n          field.type === ALL_FIELD_TYPES.boolean ? BooleanFormatter : undefined\n      }))\n      .filter(({name}) => name !== '_geojson');\n\n    return (\n      <div\n        ref={ref => {\n          this._root = ref;\n        }}\n        className=\"dataset-modal\"\n        style={{overflow: 'scroll'}}\n      >\n        <DatasetTabs\n          activeDataset={activeDataset}\n          datasets={datasets}\n          showDatasetTable={showDatasetTable}\n        />\n        <DataGridWrapper\n          onWheel={shouldPreventScrollBack ? this._onMouseWheel : null}\n        >\n          {ReactDataGrid ? (\n            <ReactDataGrid\n              headerRowHeight={72}\n              columns={columns}\n              minColumnWidth={172}\n              minWidth={this.props.width}\n              minHeight={this.props.height - 65}\n              rowGetter={i => rows[i]}\n              rowHeight={48}\n              rowsCount={rows.length}\n            />\n          ) : null}\n        </DataGridWrapper>\n      </div>\n    );\n  }\n}\n\nconst tagContainerStyle = {\n  display: 'flex',\n  flexDirection: 'column',\n  justifyContent: 'space-between'\n};\n\nconst FieldHeader = ({name, type}) => (\n  <div style={tagContainerStyle}>\n    <div style={{display: 'flex', alignItems: 'center'}}>\n      <div\n        style={{\n          marginRight: type === 'timestamp' ? '2px' : '18px',\n          height: '16px'\n        }}\n      >\n        {type === 'timestamp' ? <Clock height=\"16px\" /> : null}\n      </div>\n      {name}\n    </div>\n    <div style={{marginLeft: '18px'}}>\n      <FieldToken type={type} />\n    </div>\n  </div>\n);\n\nconst DatasetCatalog = styled.div`\n  display: flex;\n  padding: 0 ${dgSettings.sidePadding};\n`;\n\nexport const DatasetModalTab = styled.div`\n  align-items: center;\n  border-bottom: 3px solid ${props => (props.active ? 'black' : 'transparent')};\n  cursor: pointer;\n  display: flex;\n  height: 35px;\n  margin: 0 3px;\n  padding: 0 5px;\n\n  :first-child {\n    margin-left: 0;\n    padding-left: 0;\n  }\n`;\n\nexport const DatasetTabs = ({activeDataset, datasets, showDatasetTable}) => (\n  <DatasetCatalog className=\"dataset-modal-catalog\">\n    {Object.values(datasets).map(dataset =>\n      dataset.externalData && dataset.externalData === true ? null : (\n        <DatasetModalTab\n          className=\"dataset-modal-tab\"\n          active={dataset === activeDataset}\n          key={dataset.id}\n          onClick={() => showDatasetTable(dataset.id)}\n        >\n          <DatasetLabel dataset={dataset} />\n        </DatasetModalTab>\n      )\n    )}\n  </DatasetCatalog>\n);\n\nconst DataTableModalFactory = () => DataTableModal;\nexport default DataTableModalFactory;\n"]}