UNPKG

kepler.gl

Version:

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

248 lines (200 loc) 24.2 kB
'use strict'; Object.defineProperty(exports, "__esModule", { value: true }); exports.default = 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 _class, _temp2; var _templateObject = (0, _taggedTemplateLiteral3.default)(['\n .ui-sortable {\n display: block;\n position: relative;\n overflow: visible;\n user-select: none;\n\n :before {\n content: \' \';\n display: table;\n }\n\n :after {\n content: \' \';\n display: table;\n }\n }\n\n .ui-sortable-item.ui-sortable-dragging {\n position: absolute;\n z-index: 1688;\n cursor: move;\n }\n\n .ui-sortable-item.ui-sortable-dragging:hover {\n cursor: move;\n opacity: 0.5;\n }\n\n .ui-sortable-placeholder {\n display: none;\n }\n\n .ui-sortable-placeholder.visible {\n display: block;\n opacity: 0;\n z-index: -1;\n }\n'], ['\n .ui-sortable {\n display: block;\n position: relative;\n overflow: visible;\n user-select: none;\n\n :before {\n content: \' \';\n display: table;\n }\n\n :after {\n content: \' \';\n display: table;\n }\n }\n\n .ui-sortable-item.ui-sortable-dragging {\n position: absolute;\n z-index: 1688;\n cursor: move;\n }\n\n .ui-sortable-item.ui-sortable-dragging:hover {\n cursor: move;\n opacity: 0.5;\n }\n\n .ui-sortable-placeholder {\n display: none;\n }\n\n .ui-sortable-placeholder.visible {\n display: block;\n opacity: 0;\n z-index: -1;\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 _propTypes = require('prop-types'); var _propTypes2 = _interopRequireDefault(_propTypes); var _reactAnythingSortable = require('react-anything-sortable'); var _reactAnythingSortable2 = _interopRequireDefault(_reactAnythingSortable); var _styledComponents = require('styled-components'); var _styledComponents2 = _interopRequireDefault(_styledComponents); var _reselect = require('reselect'); var _layerPanel = require('./layer-panel/layer-panel'); var _layerPanel2 = _interopRequireDefault(_layerPanel); var _sourceDataCatalog = require('./source-data-catalog'); var _sourceDataCatalog2 = _interopRequireDefault(_sourceDataCatalog); var _icons = require('../common/icons'); var _itemSelector = require('../common/item-selector/item-selector'); var _itemSelector2 = _interopRequireDefault(_itemSelector); var _styledComponents3 = require('../common/styled-components'); var _defaultSettings = require('../../constants/default-settings'); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } var StyledSortable = _styledComponents2.default.div(_templateObject); var LayerManager = (_temp2 = _class = function (_Component) { (0, _inherits3.default)(LayerManager, _Component); function LayerManager() { var _ref; var _temp, _this, _ret; (0, _classCallCheck3.default)(this, LayerManager); 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, (_ref = LayerManager.__proto__ || Object.getPrototypeOf(LayerManager)).call.apply(_ref, [this].concat(args))), _this), _this.layerClassSelector = function (props) { return props.layerClasses; }, _this.layerTypeOptionsSelector = (0, _reselect.createSelector)(_this.layerClassSelector, function (layerClasses) { return Object.keys(layerClasses).map(function (key) { var layer = new layerClasses[key](); return { id: key, label: layer.name, icon: layer.layerIcon }; }); }), _this._addEmptyNewLayer = function () { _this.props.addLayer(); }, _this._handleSort = function (order) { _this.props.updateLayerOrder(order); }, _temp), (0, _possibleConstructorReturn3.default)(_this, _ret); } (0, _createClass3.default)(LayerManager, [{ key: 'render', value: function render() { var _props = this.props, layers = _props.layers, datasets = _props.datasets, layerOrder = _props.layerOrder, openModal = _props.openModal; var defaultDataset = Object.keys(datasets)[0]; var layerTypeOptions = this.layerTypeOptionsSelector(this.props); var layerActions = { layerConfigChange: this.props.layerConfigChange, layerVisualChannelConfigChange: this.props.layerVisualChannelConfigChange, layerTypeChange: this.props.layerTypeChange, layerVisConfigChange: this.props.layerVisConfigChange, removeLayer: this.props.removeLayer }; var panelProps = { datasets: datasets, openModal: openModal, layerTypeOptions: layerTypeOptions }; return _react2.default.createElement( StyledSortable, { className: 'layer-manager' }, _react2.default.createElement(_sourceDataCatalog2.default, { datasets: datasets, showDatasetTable: this.props.showDatasetTable, removeDataset: this.props.removeDataset, showDeleteDataset: true }), _react2.default.createElement( _styledComponents3.Button, { onClick: this.props.showAddDataModal, isInactive: !defaultDataset, width: '105px', secondary: true }, _react2.default.createElement(_icons.Add, { height: '12px' }), 'Add Data' ), _react2.default.createElement(_styledComponents3.SidePanelDivider, null), _react2.default.createElement( _styledComponents3.SidePanelSection, null, _react2.default.createElement( _reactAnythingSortable2.default, { onSort: this._handleSort, direction: 'vertical', sortHandle: 'sort--handle', dynamic: true }, layerOrder.map(function (idx) { return _react2.default.createElement(_layerPanel2.default, (0, _extends3.default)({}, panelProps, layerActions, { sortData: idx, key: layers[idx].id, idx: idx, layer: layers[idx] })); }) ) ), _react2.default.createElement( _styledComponents3.SidePanelSection, null, defaultDataset ? _react2.default.createElement( _styledComponents3.Button, { onClick: this._addEmptyNewLayer, width: '105px' }, _react2.default.createElement(_icons.Add, { height: '12px' }), 'Add Layer' ) : null ), _react2.default.createElement(LayerBlendingSelector, { layerBlending: this.props.layerBlending, updateLayerBlending: this.props.updateLayerBlending }) ); } }]); return LayerManager; }(_react.Component), _class.propTypes = { addLayer: _propTypes2.default.func.isRequired, datasets: _propTypes2.default.object.isRequired, layerBlending: _propTypes2.default.string.isRequired, layerClasses: _propTypes2.default.object.isRequired, layers: _propTypes2.default.arrayOf(_propTypes2.default.any).isRequired, layerConfigChange: _propTypes2.default.func.isRequired, layerVisualChannelConfigChange: _propTypes2.default.func.isRequired, layerTypeChange: _propTypes2.default.func.isRequired, layerVisConfigChange: _propTypes2.default.func.isRequired, openModal: _propTypes2.default.func.isRequired, removeLayer: _propTypes2.default.func.isRequired, removeDataset: _propTypes2.default.func.isRequired, showDatasetTable: _propTypes2.default.func.isRequired, updateLayerBlending: _propTypes2.default.func.isRequired, updateLayerOrder: _propTypes2.default.func.isRequired }, _temp2); exports.default = LayerManager; var LayerBlendingSelector = function LayerBlendingSelector(_ref2) { var layerBlending = _ref2.layerBlending, updateLayerBlending = _ref2.updateLayerBlending; return _react2.default.createElement( _styledComponents3.SidePanelSection, null, _react2.default.createElement( _styledComponents3.PanelLabel, null, 'Layer Blending' ), _react2.default.createElement(_itemSelector2.default, { selectedItems: layerBlending, options: Object.keys(_defaultSettings.LAYER_BLENDINGS), multiSelect: false, searchable: false, onChange: updateLayerBlending }) ); }; //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/components/side-panel/layer-manager.js"],"names":["StyledSortable","styled","div","LayerManager","layerClassSelector","props","layerClasses","layerTypeOptionsSelector","Object","keys","map","layer","key","id","label","name","icon","layerIcon","_addEmptyNewLayer","addLayer","_handleSort","updateLayerOrder","order","layers","datasets","layerOrder","openModal","defaultDataset","layerTypeOptions","layerActions","layerConfigChange","layerVisualChannelConfigChange","layerTypeChange","layerVisConfigChange","removeLayer","panelProps","showDatasetTable","removeDataset","showAddDataModal","idx","layerBlending","updateLayerBlending","Component","propTypes","PropTypes","func","isRequired","object","string","arrayOf","any","LayerBlendingSelector","LAYER_BLENDINGS"],"mappings":"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","file":"layer-manager.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 PropTypes from 'prop-types';\nimport Sortable from 'react-anything-sortable';\nimport styled from 'styled-components';\nimport {createSelector} from 'reselect';\n\nimport LayerPanel from './layer-panel/layer-panel';\nimport SourceDataCatalog from './source-data-catalog';\nimport {Add} from 'components/common/icons';\nimport ItemSelector from 'components/common/item-selector/item-selector';\nimport {\n  PanelLabel,\n  SidePanelDivider,\n  SidePanelSection,\n  Button\n} from 'components/common/styled-components';\n\nimport {LAYER_BLENDINGS} from 'constants/default-settings';\n\nconst StyledSortable = styled.div`\n  .ui-sortable {\n    display: block;\n    position: relative;\n    overflow: visible;\n    user-select: none;\n\n    :before {\n      content: ' ';\n      display: table;\n    }\n\n    :after {\n      content: ' ';\n      display: table;\n    }\n  }\n\n  .ui-sortable-item.ui-sortable-dragging {\n    position: absolute;\n    z-index: 1688;\n    cursor: move;\n  }\n\n  .ui-sortable-item.ui-sortable-dragging:hover {\n    cursor: move;\n    opacity: 0.5;\n  }\n\n  .ui-sortable-placeholder {\n    display: none;\n  }\n\n  .ui-sortable-placeholder.visible {\n    display: block;\n    opacity: 0;\n    z-index: -1;\n  }\n`;\n\nexport default class LayerManager extends Component {\n  static propTypes = {\n    addLayer: PropTypes.func.isRequired,\n    datasets: PropTypes.object.isRequired,\n    layerBlending: PropTypes.string.isRequired,\n    layerClasses: PropTypes.object.isRequired,\n    layers: PropTypes.arrayOf(PropTypes.any).isRequired,\n    layerConfigChange: PropTypes.func.isRequired,\n    layerVisualChannelConfigChange: PropTypes.func.isRequired,\n    layerTypeChange: PropTypes.func.isRequired,\n    layerVisConfigChange: PropTypes.func.isRequired,\n    openModal: PropTypes.func.isRequired,\n    removeLayer: PropTypes.func.isRequired,\n    removeDataset: PropTypes.func.isRequired,\n    showDatasetTable: PropTypes.func.isRequired,\n    updateLayerBlending: PropTypes.func.isRequired,\n    updateLayerOrder: PropTypes.func.isRequired\n  };\n\n  layerClassSelector = props => props.layerClasses;\n  layerTypeOptionsSelector = createSelector(\n    this.layerClassSelector,\n    layerClasses => Object.keys(layerClasses).map(key => {\n      const layer = new layerClasses[key]();\n      return {\n        id: key,\n        label: layer.name,\n        icon: layer.layerIcon\n      };\n  }));\n\n  _addEmptyNewLayer = () => {\n    this.props.addLayer();\n  };\n\n  _handleSort = order => {\n    this.props.updateLayerOrder(order);\n  };\n\n  render() {\n    const {layers, datasets, layerOrder, openModal} = this.props;\n    const defaultDataset = Object.keys(datasets)[0];\n    const layerTypeOptions = this.layerTypeOptionsSelector(this.props);\n\n    const layerActions = {\n      layerConfigChange: this.props.layerConfigChange,\n      layerVisualChannelConfigChange: this.props.layerVisualChannelConfigChange,\n      layerTypeChange: this.props.layerTypeChange,\n      layerVisConfigChange: this.props.layerVisConfigChange,\n      removeLayer: this.props.removeLayer\n    };\n\n    const panelProps = {datasets, openModal, layerTypeOptions};\n\n    return (\n      <StyledSortable className=\"layer-manager\">\n        <SourceDataCatalog\n          datasets={datasets}\n          showDatasetTable={this.props.showDatasetTable}\n          removeDataset={this.props.removeDataset}\n          showDeleteDataset\n        />\n        <Button\n          onClick={this.props.showAddDataModal}\n          isInactive={!defaultDataset}\n          width=\"105px\"\n          secondary\n        >\n          <Add height=\"12px\" />Add Data\n        </Button>\n        <SidePanelDivider />\n        <SidePanelSection>\n          <Sortable\n            onSort={this._handleSort}\n            direction=\"vertical\"\n            sortHandle=\"sort--handle\"\n            dynamic\n          >\n            {layerOrder.map(idx => (\n              <LayerPanel\n                {...panelProps}\n                {...layerActions}\n                sortData={idx}\n                key={layers[idx].id}\n                idx={idx}\n                layer={layers[idx]}\n              />\n            ))}\n          </Sortable>\n        </SidePanelSection>\n        <SidePanelSection>\n          {defaultDataset ? (\n            <Button onClick={this._addEmptyNewLayer} width=\"105px\">\n              <Add height=\"12px\" />Add Layer\n            </Button>\n          ) : null}\n        </SidePanelSection>\n        <LayerBlendingSelector\n          layerBlending={this.props.layerBlending}\n          updateLayerBlending={this.props.updateLayerBlending}\n        />\n      </StyledSortable>\n    );\n  }\n}\n\nconst LayerBlendingSelector = ({layerBlending, updateLayerBlending}) => (\n  <SidePanelSection>\n    <PanelLabel>Layer Blending</PanelLabel>\n    <ItemSelector\n      selectedItems={layerBlending}\n      options={Object.keys(LAYER_BLENDINGS)}\n      multiSelect={false}\n      searchable={false}\n      onChange={updateLayerBlending}\n    />\n  </SidePanelSection>\n);\n"]}