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