kepler.gl
Version:
kepler.gl is a webgl based application to visualize large scale location data in the browser
263 lines (212 loc) • 21.3 kB
JavaScript
'use strict';
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.default = undefined;
var _defineProperty2 = require('babel-runtime/helpers/defineProperty');
var _defineProperty3 = _interopRequireDefault(_defineProperty2);
var _extends3 = require('babel-runtime/helpers/extends');
var _extends4 = _interopRequireDefault(_extends3);
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 border-top: 1px solid ', ';\n'], ['\n border-top: 1px solid ', ';\n']),
_templateObject2 = (0, _taggedTemplateLiteral3.default)(['\n padding-bottom: 6px;\n'], ['\n padding-bottom: 6px;\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 _styledComponents = require('styled-components');
var _styledComponents2 = _interopRequireDefault(_styledComponents);
var _switch = require('../../common/switch');
var _switch2 = _interopRequireDefault(_switch);
var _rangeSlider = require('../../common/range-slider');
var _rangeSlider2 = _interopRequireDefault(_rangeSlider);
var _fieldSelector = require('../../common/field-selector');
var _fieldSelector2 = _interopRequireDefault(_fieldSelector);
var _styledComponents3 = require('../../common/styled-components');
var _sourceDataCatalog = require('../source-data-catalog');
var _interactionUtils = require('../../../utils/interaction-utils');
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
var StyledPanelContent = _styledComponents3.PanelContent.extend(_templateObject, function (props) {
return props.theme.panelBorderColor;
});
var StyledInteractionPanel = _styledComponents2.default.div(_templateObject2);
var InteractionPanel = (_temp2 = _class = function (_Component) {
(0, _inherits3.default)(InteractionPanel, _Component);
function InteractionPanel() {
var _ref;
var _temp, _this, _ret;
(0, _classCallCheck3.default)(this, InteractionPanel);
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 = InteractionPanel.__proto__ || Object.getPrototypeOf(InteractionPanel)).call.apply(_ref, [this].concat(args))), _this), _this.state = { isConfigActive: false }, _this._updateConfig = function (newProp) {
_this.props.onConfigChange((0, _extends4.default)({}, _this.props.config, newProp));
}, _this._enableConfig = function () {
_this.setState({ isConfigActive: !_this.state.isConfigActive });
}, _temp), (0, _possibleConstructorReturn3.default)(_this, _ret);
}
(0, _createClass3.default)(InteractionPanel, [{
key: 'render',
value: function render() {
var _this2 = this;
var _props = this.props,
config = _props.config,
datasets = _props.datasets;
var onChange = function onChange(newConfig) {
return _this2._updateConfig({ config: newConfig });
};
var template = null;
switch (config.id) {
case 'tooltip':
template = _react2.default.createElement(TooltipConfig, {
datasets: datasets,
config: config.config,
width: this.state.innerPanelWidth,
onChange: onChange
});
break;
case 'brush':
template = _react2.default.createElement(BrushConfig, { config: config.config, onChange: onChange });
break;
default:
break;
}
return _react2.default.createElement(
StyledInteractionPanel,
{ className: 'interaction-panel' },
_react2.default.createElement(
_styledComponents3.StyledPanelHeader,
{
className: 'interaction-panel__header',
onClick: this._enableConfig
},
_react2.default.createElement(
_styledComponents3.PanelHeaderContent,
{ className: 'interaction-panel__header__content' },
_react2.default.createElement(
'div',
{ className: 'interaction-panel__header__icon icon' },
_react2.default.createElement(config.iconComponent, { height: '12px' })
),
_react2.default.createElement(
'div',
{ className: 'interaction-panel__header__title' },
_react2.default.createElement(
_styledComponents3.PanelHeaderTitle,
null,
config.id
)
)
),
_react2.default.createElement(
'div',
{ className: 'interaction-panel__header__actions' },
_react2.default.createElement(_switch2.default, {
checked: config.enabled,
id: config.id + '-toggle',
onChange: function onChange() {
return _this2._updateConfig({ enabled: !config.enabled });
},
secondary: true
})
)
),
config.enabled && _react2.default.createElement(
StyledPanelContent,
{ className: 'interaction-panel__content' },
template
)
);
}
}]);
return InteractionPanel;
}(_react.Component), _class.propTypes = {
datasets: _propTypes2.default.object.isRequired,
config: _propTypes2.default.object.isRequired,
onConfigChange: _propTypes2.default.func.isRequired
}, _temp2);
exports.default = InteractionPanel;
var TooltipConfig = function TooltipConfig(_ref2) {
var config = _ref2.config,
datasets = _ref2.datasets,
width = _ref2.width,
onChange = _ref2.onChange;
return _react2.default.createElement(
'div',
null,
Object.keys(config.fieldsToShow).map(function (dataId) {
return _react2.default.createElement(
_styledComponents3.SidePanelSection,
{ key: dataId },
_react2.default.createElement(_sourceDataCatalog.DatasetTag, { dataset: datasets[dataId] }),
_react2.default.createElement(_fieldSelector2.default, {
fields: datasets[dataId].fields,
value: config.fieldsToShow[dataId],
onSelect: function onSelect(fieldsToShow) {
var newConfig = (0, _extends4.default)({}, config, {
fieldsToShow: (0, _extends4.default)({}, config.fieldsToShow, (0, _defineProperty3.default)({}, dataId, fieldsToShow.map(function (d) {
return d.name;
})))
});
onChange(newConfig);
},
closeOnSelect: false,
multiSelect: true
})
);
})
);
};
var BrushConfig = function BrushConfig(_ref3) {
var config = _ref3.config,
_onChange = _ref3.onChange;
return _react2.default.createElement(
_styledComponents3.SidePanelSection,
null,
_react2.default.createElement(
_styledComponents3.PanelLabel,
null,
'Brush Radius (km)'
),
_react2.default.createElement(_rangeSlider2.default, {
range: _interactionUtils.BRUSH_CONFIG.range,
value0: 0,
value1: config.size || 10 / 2,
step: 0.1,
isRanged: false,
onChange: function onChange(value) {
return _onChange((0, _extends4.default)({}, config, { size: value[1] }));
},
inputTheme: 'secondary'
})
);
};
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/side-panel/interaction-panel/interaction-panel.js"],"names":["StyledPanelContent","PanelContent","extend","props","theme","panelBorderColor","StyledInteractionPanel","styled","div","InteractionPanel","state","isConfigActive","_updateConfig","onConfigChange","config","newProp","_enableConfig","setState","datasets","onChange","newConfig","template","id","innerPanelWidth","enabled","Component","propTypes","PropTypes","object","isRequired","func","TooltipConfig","width","Object","keys","fieldsToShow","map","dataId","fields","d","name","BrushConfig","BRUSH_CONFIG","range","size","value"],"mappings":"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","file":"interaction-panel.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 styled from 'styled-components';\nimport Switch from 'components/common/switch';\nimport RangeSlider from 'components/common/range-slider';\nimport FieldSelector from 'components/common/field-selector';\nimport {\n  PanelLabel,\n  SidePanelSection,\n  StyledPanelHeader,\n  PanelHeaderTitle,\n  PanelHeaderContent,\n  PanelContent\n} from 'components/common/styled-components';\nimport {DatasetTag} from '../source-data-catalog';\nimport {BRUSH_CONFIG} from 'utils/interaction-utils';\n\nconst StyledPanelContent = PanelContent.extend`\n  border-top: 1px solid ${props => props.theme.panelBorderColor};\n`;\n\nconst StyledInteractionPanel = styled.div`\n  padding-bottom: 6px;\n`;\n\nexport default class InteractionPanel extends Component {\n  static propTypes = {\n    datasets: PropTypes.object.isRequired,\n    config: PropTypes.object.isRequired,\n    onConfigChange: PropTypes.func.isRequired\n  };\n\n  state = {isConfigActive: false};\n\n  _updateConfig = newProp => {\n    this.props.onConfigChange({\n      ...this.props.config,\n      ...newProp\n    });\n  };\n\n  _enableConfig = () => {\n    this.setState({isConfigActive: !this.state.isConfigActive});\n  };\n\n  render() {\n    const {config, datasets} = this.props;\n    const onChange = newConfig => this._updateConfig({config: newConfig});\n    let template = null;\n\n    switch (config.id) {\n      case 'tooltip':\n        template = (\n          <TooltipConfig\n            datasets={datasets}\n            config={config.config}\n            width={this.state.innerPanelWidth}\n            onChange={onChange}\n          />\n        );\n        break;\n\n      case 'brush':\n        template = <BrushConfig config={config.config} onChange={onChange} />;\n        break;\n\n      default:\n        break;\n    }\n\n    return (\n      <StyledInteractionPanel className=\"interaction-panel\">\n        <StyledPanelHeader\n          className=\"interaction-panel__header\"\n          onClick={this._enableConfig}\n        >\n          <PanelHeaderContent className=\"interaction-panel__header__content\">\n            <div className=\"interaction-panel__header__icon icon\">\n              <config.iconComponent height=\"12px\"/>\n            </div>\n            <div className=\"interaction-panel__header__title\">\n              <PanelHeaderTitle>{config.id}</PanelHeaderTitle>\n            </div>\n          </PanelHeaderContent>\n          <div className=\"interaction-panel__header__actions\">\n            <Switch\n              checked={config.enabled}\n              id={`${config.id}-toggle`}\n              onChange={() => this._updateConfig({enabled: !config.enabled})}\n              secondary\n            />\n          </div>\n        </StyledPanelHeader>\n        {config.enabled && (\n          <StyledPanelContent className=\"interaction-panel__content\">\n            {template}\n          </StyledPanelContent>\n        )}\n      </StyledInteractionPanel>\n    );\n  }\n}\n\nconst TooltipConfig = ({config, datasets, width, onChange}) => (\n  <div>\n    {Object.keys(config.fieldsToShow).map(dataId => (\n      <SidePanelSection key={dataId}>\n        <DatasetTag dataset={datasets[dataId]} />\n        <FieldSelector\n          fields={datasets[dataId].fields}\n          value={config.fieldsToShow[dataId]}\n          onSelect={fieldsToShow => {\n            const newConfig = {\n              ...config,\n              fieldsToShow: {\n                ...config.fieldsToShow,\n                [dataId]: fieldsToShow.map(d => d.name)\n              }\n            };\n            onChange(newConfig);\n          }}\n          closeOnSelect={false}\n          multiSelect\n        />\n      </SidePanelSection>\n    ))}\n  </div>\n);\n\nconst BrushConfig = ({config, onChange}) => (\n  <SidePanelSection>\n    <PanelLabel>Brush Radius (km)</PanelLabel>\n    <RangeSlider\n      range={BRUSH_CONFIG.range}\n      value0={0}\n      value1={config.size || 10 / 2}\n      step={0.1}\n      isRanged={false}\n      onChange={value => onChange({...config, size: value[1]})}\n      inputTheme=\"secondary\"\n    />\n  </SidePanelSection>\n);\n"]}