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
JavaScript
;
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"]}