UNPKG

kepler.gl

Version:

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

289 lines (233 loc) 27.9 kB
'use strict'; Object.defineProperty(exports, "__esModule", { value: true }); exports.DatasetTabs = exports.DatasetModalTab = exports.DataTableModal = undefined; var _extends2 = require('babel-runtime/helpers/extends'); var _extends3 = _interopRequireDefault(_extends2); var _classCallCheck2 = require('babel-runtime/helpers/classCallCheck'); var _classCallCheck3 = _interopRequireDefault(_classCallCheck2); var _createClass2 = require('babel-runtime/helpers/createClass'); var _createClass3 = _interopRequireDefault(_createClass2); var _possibleConstructorReturn2 = require('babel-runtime/helpers/possibleConstructorReturn'); var _possibleConstructorReturn3 = _interopRequireDefault(_possibleConstructorReturn2); var _inherits2 = require('babel-runtime/helpers/inherits'); var _inherits3 = _interopRequireDefault(_inherits2); var _taggedTemplateLiteral2 = require('babel-runtime/helpers/taggedTemplateLiteral'); var _taggedTemplateLiteral3 = _interopRequireDefault(_taggedTemplateLiteral2); var _templateObject = (0, _taggedTemplateLiteral3.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'], ['\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']), _templateObject2 = (0, _taggedTemplateLiteral3.default)(['\n display: flex;\n padding: 0 ', ';\n'], ['\n display: flex;\n padding: 0 ', ';\n']), _templateObject3 = (0, _taggedTemplateLiteral3.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'], ['\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']); // Copyright (c) 2018 Uber Technologies, Inc. // // Permission is hereby granted, free of charge, to any person obtaining a copy // of this software and associated documentation files (the "Software"), to deal // in the Software without restriction, including without limitation the rights // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell // copies of the Software, and to permit persons to whom the Software is // furnished to do so, subject to the following conditions: // // The above copyright notice and this permission notice shall be included in // all copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN // THE SOFTWARE. var _react = require('react'); var _react2 = _interopRequireDefault(_react); var _styledComponents = require('styled-components'); var _styledComponents2 = _interopRequireDefault(_styledComponents); var _window = require('global/window'); var _window2 = _interopRequireDefault(_window); var _defaultSettings = require('../../constants/default-settings'); var _fieldToken = require('../common/field-token'); var _fieldToken2 = _interopRequireDefault(_fieldToken); var _datasetLabel = require('../common/dataset-label'); var _datasetLabel2 = _interopRequireDefault(_datasetLabel); var _icons = require('../common/icons'); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } var ReactDataGrid = _window2.default.navigator ? require('react-data-grid/dist/react-data-grid.min') : null; var shouldPreventScrollBack = false; if (_window2.default.navigator && _window2.default.navigator.userAgent) { var navigator = _window2.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 = _styledComponents2.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 _react2.default.createElement( 'span', null, String(value) ); }; var DataTableModal = exports.DataTableModal = function (_Component) { (0, _inherits3.default)(DataTableModal, _Component); function DataTableModal() { var _ref2; var _temp, _this, _ret; (0, _classCallCheck3.default)(this, DataTableModal); for (var _len = arguments.length, args = Array(_len), _key = 0; _key < _len; _key++) { args[_key] = arguments[_key]; } return _ret = (_temp = (_this = (0, _possibleConstructorReturn3.default)(this, (_ref2 = DataTableModal.__proto__ || Object.getPrototypeOf(DataTableModal)).call.apply(_ref2, [this].concat(args))), _this), _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(); } }, _temp), (0, _possibleConstructorReturn3.default)(_this, _ret); } (0, _createClass3.default)(DataTableModal, [{ key: 'render', value: function render() { var _this2 = this; var _props = this.props, datasets = _props.datasets, dataId = _props.dataId, showDatasetTable = _props.showDatasetTable; if (!datasets || !dataId) { return null; } var activeDataset = datasets[dataId]; var rows = activeDataset.data; var columns = activeDataset.fields.map(function (field, i) { return (0, _extends3.default)({}, field, { key: i, headerRenderer: _react2.default.createElement(FieldHeader, field), resizable: true, formatter: field.type === _defaultSettings.ALL_FIELD_TYPES.boolean ? BooleanFormatter : undefined }); }).filter(function (_ref3) { var name = _ref3.name; return name !== '_geojson'; }); return _react2.default.createElement( 'div', { ref: function ref(_ref4) { _this2._root = _ref4; }, className: 'dataset-modal', style: { overflow: 'scroll' } }, _react2.default.createElement(DatasetTabs, { activeDataset: activeDataset, datasets: datasets, showDatasetTable: showDatasetTable }), _react2.default.createElement( DataGridWrapper, { onWheel: shouldPreventScrollBack ? this._onMouseWheel : null }, ReactDataGrid ? _react2.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); var tagContainerStyle = { display: 'flex', flexDirection: 'column', justifyContent: 'space-between' }; var FieldHeader = function FieldHeader(_ref5) { var name = _ref5.name, type = _ref5.type; return _react2.default.createElement( 'div', { style: tagContainerStyle }, _react2.default.createElement( 'div', { style: { display: 'flex', alignItems: 'center' } }, _react2.default.createElement( 'div', { style: { marginRight: type === 'timestamp' ? '2px' : '18px', height: '16px' } }, type === 'timestamp' ? _react2.default.createElement(_icons.Clock, { height: '16px' }) : null ), name ), _react2.default.createElement( 'div', { style: { marginLeft: '18px' } }, _react2.default.createElement(_fieldToken2.default, { type: type }) ) ); }; var DatasetCatalog = _styledComponents2.default.div(_templateObject2, dgSettings.sidePadding); var DatasetModalTab = exports.DatasetModalTab = _styledComponents2.default.div(_templateObject3, function (props) { return props.active ? 'black' : 'transparent'; }); var DatasetTabs = exports.DatasetTabs = function DatasetTabs(_ref6) { var activeDataset = _ref6.activeDataset, datasets = _ref6.datasets, showDatasetTable = _ref6.showDatasetTable; return _react2.default.createElement( DatasetCatalog, { className: 'dataset-modal-catalog' }, Object.values(datasets).map(function (dataset) { return _react2.default.createElement( DatasetModalTab, { className: 'dataset-modal-tab', active: dataset === activeDataset, key: dataset.id, onClick: function onClick() { return showDatasetTable(dataset.id); } }, _react2.default.createElement(_datasetLabel2.default, { dataset: dataset }) ); }) ); }; var DataTableModalFactory = function DataTableModalFactory() { return DataTableModal; }; exports.default = DataTableModalFactory; //# 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","_onMouseWheel","canvas","_root","querySelector","prevent_left","e","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","boolean","undefined","filter","name","ref","overflow","width","height","length","Component","tagContainerStyle","display","flexDirection","justifyContent","FieldHeader","alignItems","marginRight","marginLeft","DatasetCatalog","DatasetModalTab","active","DatasetTabs","Object","values","dataset","id","DataTableModalFactory"],"mappings":"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","file":"data-table-modal.js","sourcesContent":["// Copyright (c) 2018 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 ? require('react-data-grid/dist/react-data-grid.min') : 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    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 ref={ref => {this._root = ref}} className=\"dataset-modal\" style={{overflow: 'scroll'}}>\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      <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  </DatasetCatalog>\n);\n\nconst DataTableModalFactory = () => DataTableModal;\nexport default DataTableModalFactory;\n"]}