kepler.gl
Version:
kepler.gl is a webgl based application to visualize large scale location data in the browser
439 lines (377 loc) • 44.6 kB
JavaScript
'use strict';
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.MapControl = undefined;
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 _toConsumableArray2 = require('babel-runtime/helpers/toConsumableArray');
var _toConsumableArray3 = _interopRequireDefault(_toConsumableArray2);
var _taggedTemplateLiteral2 = require('babel-runtime/helpers/taggedTemplateLiteral');
var _taggedTemplateLiteral3 = _interopRequireDefault(_taggedTemplateLiteral2);
var _class, _temp2;
var _templateObject = (0, _taggedTemplateLiteral3.default)(['\n right: 0;\n width: ', 'px;\n padding: ', 'px;\n z-index: 1;\n top: ', 'px;\n position: absolute;\n'], ['\n right: 0;\n width: ', 'px;\n padding: ', 'px;\n z-index: 1;\n top: ', 'px;\n position: absolute;\n']),
_templateObject2 = (0, _taggedTemplateLiteral3.default)(['\n padding: 4px 0;\n display: flex;\n justify-content: flex-end;\n'], ['\n padding: 4px 0;\n display: flex;\n justify-content: flex-end;\n']),
_templateObject3 = (0, _taggedTemplateLiteral3.default)(['\n align-items: center;\n background-color: ', ';\n border-radius: 18px;\n border: 0;\n box-shadow: 0 6px 12px 0 rgba(0, 0, 0, 0.16);\n color: ', ';\n cursor: pointer;\n display: flex;\n height: 36px;\n justify-content: center;\n margin: 0;\n outline: none;\n padding: 0;\n transition: ', ';\n width: 36px;\n\n :focus {\n outline: none;\n }\n\n :hover {\n cursor: pointer;\n background-color: ', ';\n color: ', ';\n }\n'], ['\n align-items: center;\n background-color: ', ';\n border-radius: 18px;\n border: 0;\n box-shadow: 0 6px 12px 0 rgba(0, 0, 0, 0.16);\n color: ', ';\n cursor: pointer;\n display: flex;\n height: 36px;\n justify-content: center;\n margin: 0;\n outline: none;\n padding: 0;\n transition: ', ';\n width: 36px;\n\n :focus {\n outline: none;\n }\n\n :hover {\n cursor: pointer;\n background-color: ', ';\n color: ', ';\n }\n']),
_templateObject4 = (0, _taggedTemplateLiteral3.default)(['\n background-color: ', ';\n flex-grow: 1;\n z-index: 1;\n p {\n margin-bottom: 0;\n }\n'], ['\n background-color: ', ';\n flex-grow: 1;\n z-index: 1;\n p {\n margin-bottom: 0;\n }\n']),
_templateObject5 = (0, _taggedTemplateLiteral3.default)(['\n ', ' max-height: 500px;\n min-height: 100px;\n overflow: auto;\n'], ['\n ', ' max-height: 500px;\n min-height: 100px;\n overflow: auto;\n']),
_templateObject6 = (0, _taggedTemplateLiteral3.default)(['\n display: flex;\n justify-content: space-between;\n background-color: ', ';\n height: 32px;\n padding: 6px 12px;\n font-size: 11px;\n color: ', ';\n\n button {\n width: 18px;\n height: 18px;\n }\n'], ['\n display: flex;\n justify-content: space-between;\n background-color: ', ';\n height: 32px;\n padding: 6px 12px;\n font-size: 11px;\n color: ', ';\n\n button {\n width: 18px;\n height: 18px;\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 _reselect = require('reselect');
var _styledComponents = require('styled-components');
var _styledComponents2 = _interopRequireDefault(_styledComponents);
var _styledComponents3 = require('../common/styled-components');
var _mapLayerSelector = require('../common/map-layer-selector');
var _mapLayerSelector2 = _interopRequireDefault(_mapLayerSelector);
var _logo = require('../common/logo');
var _logo2 = _interopRequireDefault(_logo);
var _mapLegend = require('./map-legend');
var _mapLegend2 = _interopRequireDefault(_mapLegend);
var _icons = require('../common/icons');
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
var StyledMapControl = _styledComponents2.default.div(_templateObject, function (props) {
return props.theme.mapControl.width;
}, function (props) {
return props.theme.mapControl.padding;
}, function (props) {
return props.top;
});
var StyledMapControlAction = _styledComponents2.default.div(_templateObject2);
var StyledMapControlButton = _styledComponents2.default.div(_templateObject3, function (props) {
return props.active ? props.theme.secondaryBtnActBgd : props.theme.secondaryBtnBgd;
}, function (props) {
return props.active ? props.theme.secondaryBtnActColor : props.theme.secondaryBtnColor;
}, function (props) {
return props.theme.transition;
}, function (props) {
return props.theme.secondaryBtnActBgd;
}, function (props) {
return props.theme.secondaryBtnActColor;
});
var StyledMapControlPanel = _styledComponents2.default.div(_templateObject4, function (props) {
return props.theme.mapPanelBackgroundColor;
});
var StyledMapControlPanelContent = _styledComponents2.default.div(_templateObject5, function (props) {
return props.theme.dropdownScrollBar;
});
var StyledMapControlPanelHeader = _styledComponents2.default.div(_templateObject6, function (props) {
return props.theme.mapPanelHeaderBackgroundColor;
}, function (props) {
return props.theme.secondaryBtnColor;
});
/**
* Generates all layers available for the current map
* TODO: this may be moved into map-container or map-control or even at the reducer level
* @param layers
* @param mapLayers
* @returns {[id, label, isVisible]}
*/
var layerSelector = function layerSelector(layers, mapLayers) {
var availableItems = Object.keys(layers).reduce(function (availableLayers, currentLayerId) {
// is available ? if yes add to available list
var currentLayer = layers[currentLayerId];
// if maplayers exists we need to make sure currentlayer
// is contained in mapLayers in order to add onto availableLayers
// otherwise we add all layers
var layerConfig = mapLayers ? mapLayers[currentLayer.id] : currentLayer.config;
var mustBeAdded = mapLayers && mapLayers[currentLayer.id] ? mapLayers[currentLayer.id].isAvailable : layerConfig.isVisible;
return mustBeAdded ? [].concat((0, _toConsumableArray3.default)(availableLayers), [{
id: currentLayer.id,
name: currentLayer.config.label,
isVisible: mapLayers && mapLayers[currentLayer.id] ? mapLayers[currentLayer.id].isVisible : layerConfig.isVisible,
layer: currentLayer
}]) : availableLayers;
}, []);
return availableItems;
};
var MapControl = exports.MapControl = (_temp2 = _class = function (_Component) {
(0, _inherits3.default)(MapControl, _Component);
function MapControl() {
var _ref;
var _temp, _this, _ret;
(0, _classCallCheck3.default)(this, MapControl);
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 = MapControl.__proto__ || Object.getPrototypeOf(MapControl)).call.apply(_ref, [this].concat(args))), _this), _this.layerSelector = function (state) {
return state.layers;
}, _this.mapLayersSelector = function (state) {
return state.mapLayers;
}, _this.initialDataSelector = (0, _reselect.createSelector)(_this.layerSelector, _this.mapLayersSelector, layerSelector), _temp), (0, _possibleConstructorReturn3.default)(_this, _ret);
}
(0, _createClass3.default)(MapControl, [{
key: 'render',
value: function render() {
var items = this.initialDataSelector(this.props);
if (!items) {
return null;
}
var _props = this.props,
dragRotate = _props.dragRotate,
isSplit = _props.isSplit,
isExport = _props.isExport,
mapIndex = _props.mapIndex,
mapControls = _props.mapControls,
onTogglePerspective = _props.onTogglePerspective,
onToggleSplitMap = _props.onToggleSplitMap,
onMapToggleLayer = _props.onMapToggleLayer,
onToggleMapControl = _props.onToggleMapControl,
scale = _props.scale;
var _mapControls$visibleL = mapControls.visibleLayers,
visibleLayers = _mapControls$visibleL === undefined ? {} : _mapControls$visibleL,
_mapControls$mapLegen = mapControls.mapLegend,
mapLegend = _mapControls$mapLegen === undefined ? {} : _mapControls$mapLegen,
_mapControls$toggle3d = mapControls.toggle3d,
toggle3d = _mapControls$toggle3d === undefined ? {} : _mapControls$toggle3d,
_mapControls$splitMap = mapControls.splitMap,
splitMap = _mapControls$splitMap === undefined ? {} : _mapControls$splitMap;
return _react2.default.createElement(
StyledMapControl,
{ className: 'map-control' },
splitMap.show ? _react2.default.createElement(
ActionPanel,
{ key: 0 },
_react2.default.createElement(
StyledMapControlButton,
{
active: isSplit,
onClick: function onClick(e) {
e.preventDefault();
onToggleSplitMap(isSplit ? mapIndex : undefined);
},
key: 'split-' + isSplit,
className: 'map-control-button split-map',
'data-tip': true,
'data-for': 'action-toggle'
},
isSplit ? _react2.default.createElement(_icons.Delete, { height: '18px' }) : _react2.default.createElement(_icons.Split, { height: '18px' }),
_react2.default.createElement(MapLegendTooltip, {
id: 'action-toggle',
message: isSplit ? 'Close current panel' : 'Switch to dual map view'
})
)
) : null,
isSplit && visibleLayers.show ? _react2.default.createElement(
ActionPanel,
{ key: 1 },
_react2.default.createElement(LayerSelectorPanel, {
items: items,
onMapToggleLayer: onMapToggleLayer,
isActive: visibleLayers.active,
toggleMenuPanel: function toggleMenuPanel() {
return onToggleMapControl('visibleLayers');
}
})
) : null,
toggle3d.show ? _react2.default.createElement(
ActionPanel,
{ key: 2 },
_react2.default.createElement(
StyledMapControlButton,
{
onClick: function onClick(e) {
e.preventDefault();
onTogglePerspective();
},
active: dragRotate,
'data-tip': true,
'data-for': 'action-3d'
},
_react2.default.createElement(_icons.Cube3d, { height: '22px' }),
_react2.default.createElement(MapLegendTooltip, {
id: 'action-3d',
message: dragRotate ? 'Disable 3D Map' : '3D Map'
})
)
) : null,
mapLegend.show ? _react2.default.createElement(
ActionPanel,
{ key: 3 },
_react2.default.createElement(MapLegendPanel, {
items: items,
scale: scale,
isExport: isExport,
onMapToggleLayer: onMapToggleLayer,
isActive: mapLegend.active,
toggleMenuPanel: function toggleMenuPanel() {
return onToggleMapControl('mapLegend');
}
})
) : null
);
}
}]);
return MapControl;
}(_react.Component), _class.propTypes = {
datasets: _propTypes2.default.object.isRequired,
dragRotate: _propTypes2.default.bool.isRequired,
isSplit: _propTypes2.default.bool.isRequired,
layers: _propTypes2.default.arrayOf(_propTypes2.default.object),
mapIndex: _propTypes2.default.number.isRequired,
mapControls: _propTypes2.default.object.isRequired,
onToggleFullScreen: _propTypes2.default.func.isRequired,
onTogglePerspective: _propTypes2.default.func.isRequired,
onToggleSplitMap: _propTypes2.default.func.isRequired,
onToggleMapControl: _propTypes2.default.func.isRequired,
onMapToggleLayer: _propTypes2.default.func.isRequired,
top: _propTypes2.default.number.isRequired,
// optional
scale: _propTypes2.default.number,
mapLayers: _propTypes2.default.object
}, _class.defaultProps = {
isSplit: false,
top: 0
}, _temp2);
var MapControlPanel = function MapControlPanel(_ref2) {
var children = _ref2.children,
header = _ref2.header,
onClick = _ref2.onClick,
_ref2$scale = _ref2.scale,
scale = _ref2$scale === undefined ? 1 : _ref2$scale,
isExport = _ref2.isExport;
return _react2.default.createElement(
StyledMapControlPanel,
{
style: {
transform: 'scale(' + scale + ') translate(calc(-' + 25 * (scale - 1) + '% - ' + 10 * scale + 'px), calc(' + 25 * (scale - 1) + '% + ' + 10 * scale + 'px))'
}
},
_react2.default.createElement(
StyledMapControlPanelHeader,
{ style: { position: 'relative' } },
isExport ? _react2.default.createElement(_logo2.default, { version: false, appName: 'kepler.gl' }) : _react2.default.createElement(
'span',
{ style: { verticalAlign: 'middle' } },
header
),
isExport ? null : _react2.default.createElement(
_styledComponents3.IconRoundSmall,
null,
_react2.default.createElement(_icons.Close, { height: '16px', onClick: onClick })
)
),
_react2.default.createElement(
StyledMapControlPanelContent,
null,
children
)
);
};
var MapLegendPanel = function MapLegendPanel(_ref3) {
var items = _ref3.items,
isActive = _ref3.isActive,
scale = _ref3.scale,
toggleMenuPanel = _ref3.toggleMenuPanel,
isExport = _ref3.isExport;
return !isActive ? _react2.default.createElement(
StyledMapControlButton,
{
key: 2,
'data-tip': true,
'data-for': 'show-legend',
className: 'map-control-button show-legend',
onClick: function onClick(e) {
e.preventDefault();
toggleMenuPanel();
}
},
_react2.default.createElement(_icons.Legend, { height: '22px' }),
_react2.default.createElement(MapLegendTooltip, { id: 'show-legend', message: 'show legend' })
) : _react2.default.createElement(
MapControlPanel,
{
scale: scale,
header: 'Layer Legend',
onClick: toggleMenuPanel,
isExport: isExport
},
_react2.default.createElement(_mapLegend2.default, {
layers: items.filter(function (item) {
return item.isVisible;
}).map(function (item) {
return item.layer;
})
})
);
};
var LayerSelectorPanel = function LayerSelectorPanel(_ref4) {
var items = _ref4.items,
onMapToggleLayer = _ref4.onMapToggleLayer,
isActive = _ref4.isActive,
toggleMenuPanel = _ref4.toggleMenuPanel;
return !isActive ? _react2.default.createElement(
StyledMapControlButton,
{
key: 1,
onClick: function onClick(e) {
e.preventDefault();
toggleMenuPanel();
},
className: 'map-control-button toggle-layer',
'data-tip': true,
'data-for': 'toggle-layer'
},
_react2.default.createElement(_icons.Layers, { height: '22px' }),
_react2.default.createElement(MapLegendTooltip, {
id: 'toggle-layer',
message: isActive ? 'Hide layer panel' : 'Show layer panel'
})
) : _react2.default.createElement(
MapControlPanel,
{ header: 'Visible layers', onClick: toggleMenuPanel },
_react2.default.createElement(_mapLayerSelector2.default, { layers: items, onMapToggleLayer: onMapToggleLayer })
);
};
var ActionPanel = function ActionPanel(_ref5) {
var children = _ref5.children;
return _react2.default.createElement(
StyledMapControlAction,
null,
children
);
};
var MapLegendTooltip = function MapLegendTooltip(_ref6) {
var id = _ref6.id,
message = _ref6.message;
return _react2.default.createElement(
_styledComponents3.Tooltip,
{ id: id, place: 'left', effect: 'solid' },
_react2.default.createElement(
'span',
null,
message
)
);
};
var MapControlFactory = function MapControlFactory() {
return MapControl;
};
exports.default = MapControlFactory;
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/components/map/map-control.js"],"names":["StyledMapControl","styled","div","props","theme","mapControl","width","padding","top","StyledMapControlAction","StyledMapControlButton","active","secondaryBtnActBgd","secondaryBtnBgd","secondaryBtnActColor","secondaryBtnColor","transition","StyledMapControlPanel","mapPanelBackgroundColor","StyledMapControlPanelContent","dropdownScrollBar","StyledMapControlPanelHeader","mapPanelHeaderBackgroundColor","layerSelector","layers","mapLayers","availableItems","Object","keys","reduce","availableLayers","currentLayerId","currentLayer","layerConfig","id","config","mustBeAdded","isAvailable","isVisible","name","label","layer","MapControl","state","mapLayersSelector","initialDataSelector","items","dragRotate","isSplit","isExport","mapIndex","mapControls","onTogglePerspective","onToggleSplitMap","onMapToggleLayer","onToggleMapControl","scale","visibleLayers","mapLegend","toggle3d","splitMap","show","e","preventDefault","undefined","Component","propTypes","datasets","PropTypes","object","isRequired","bool","arrayOf","number","onToggleFullScreen","func","defaultProps","MapControlPanel","children","header","onClick","transform","position","verticalAlign","MapLegendPanel","isActive","toggleMenuPanel","filter","item","map","LayerSelectorPanel","ActionPanel","MapLegendTooltip","message","MapControlFactory"],"mappings":"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","file":"map-control.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 {createSelector} from 'reselect';\nimport styled from 'styled-components';\n\nimport {Tooltip, IconRoundSmall} from 'components/common/styled-components';\nimport MapLayerSelector from 'components/common/map-layer-selector';\nimport KeplerGlLogo from 'components/common/logo';\nimport MapLegend from './map-legend';\nimport {\n  Close,\n  Split,\n  Legend,\n  Cube3d,\n  Delete,\n  Layers\n} from 'components/common/icons';\n\nconst StyledMapControl = styled.div`\n  right: 0;\n  width: ${props => props.theme.mapControl.width}px;\n  padding: ${props => props.theme.mapControl.padding}px;\n  z-index: 1;\n  top: ${props => props.top}px;\n  position: absolute;\n`;\n\nconst StyledMapControlAction = styled.div`\n  padding: 4px 0;\n  display: flex;\n  justify-content: flex-end;\n`;\n\nconst StyledMapControlButton = styled.div`\n  align-items: center;\n  background-color: ${props =>\n    props.active\n      ? props.theme.secondaryBtnActBgd\n      : props.theme.secondaryBtnBgd};\n  border-radius: 18px;\n  border: 0;\n  box-shadow: 0 6px 12px 0 rgba(0, 0, 0, 0.16);\n  color: ${props =>\n    props.active\n      ? props.theme.secondaryBtnActColor\n      : props.theme.secondaryBtnColor};\n  cursor: pointer;\n  display: flex;\n  height: 36px;\n  justify-content: center;\n  margin: 0;\n  outline: none;\n  padding: 0;\n  transition: ${props => props.theme.transition};\n  width: 36px;\n\n  :focus {\n    outline: none;\n  }\n\n  :hover {\n    cursor: pointer;\n    background-color: ${props => props.theme.secondaryBtnActBgd};\n    color: ${props => props.theme.secondaryBtnActColor};\n  }\n`;\n\nconst StyledMapControlPanel = styled.div`\n  background-color: ${props => props.theme.mapPanelBackgroundColor};\n  flex-grow: 1;\n  z-index: 1;\n  p {\n    margin-bottom: 0;\n  }\n`;\n\nconst StyledMapControlPanelContent = styled.div`\n  ${props => props.theme.dropdownScrollBar} max-height: 500px;\n  min-height: 100px;\n  overflow: auto;\n`;\n\nconst StyledMapControlPanelHeader = styled.div`\n  display: flex;\n  justify-content: space-between;\n  background-color: ${props => props.theme.mapPanelHeaderBackgroundColor};\n  height: 32px;\n  padding: 6px 12px;\n  font-size: 11px;\n  color: ${props => props.theme.secondaryBtnColor};\n\n  button {\n    width: 18px;\n    height: 18px;\n  }\n`;\n\n/**\n * Generates all layers available for the current map\n * TODO: this may be moved into map-container or map-control or even at the reducer level\n * @param layers\n * @param mapLayers\n * @returns {[id, label, isVisible]}\n */\nconst layerSelector = (layers, mapLayers) => {\n  const availableItems = Object.keys(layers).reduce(\n    (availableLayers, currentLayerId) => {\n      // is available ? if yes add to available list\n      const currentLayer = layers[currentLayerId];\n      // if maplayers exists we need to make sure currentlayer\n      // is contained in mapLayers in order to add onto availableLayers\n      // otherwise we add all layers\n\n      const layerConfig = mapLayers\n        ? mapLayers[currentLayer.id]\n        : currentLayer.config;\n\n      const mustBeAdded =\n        mapLayers && mapLayers[currentLayer.id]\n          ? mapLayers[currentLayer.id].isAvailable\n          : layerConfig.isVisible;\n\n      return mustBeAdded\n        ? [\n            ...availableLayers,\n            {\n              id: currentLayer.id,\n              name: currentLayer.config.label,\n              isVisible:\n                mapLayers && mapLayers[currentLayer.id]\n                  ? mapLayers[currentLayer.id].isVisible\n                  : layerConfig.isVisible,\n              layer: currentLayer\n            }\n          ]\n        : availableLayers;\n    },\n    []\n  );\n\n  return availableItems;\n};\n\nexport class MapControl extends Component {\n  static propTypes = {\n    datasets: PropTypes.object.isRequired,\n    dragRotate: PropTypes.bool.isRequired,\n    isSplit: PropTypes.bool.isRequired,\n    layers: PropTypes.arrayOf(PropTypes.object),\n    mapIndex: PropTypes.number.isRequired,\n    mapControls: PropTypes.object.isRequired,\n    onToggleFullScreen: PropTypes.func.isRequired,\n    onTogglePerspective: PropTypes.func.isRequired,\n    onToggleSplitMap: PropTypes.func.isRequired,\n    onToggleMapControl: PropTypes.func.isRequired,\n    onMapToggleLayer: PropTypes.func.isRequired,\n    top: PropTypes.number.isRequired,\n\n    // optional\n    scale: PropTypes.number,\n    mapLayers: PropTypes.object\n  };\n\n  static defaultProps = {\n    isSplit: false,\n    top: 0\n  };\n\n  layerSelector = state => state.layers;\n  mapLayersSelector = state => state.mapLayers;\n\n  initialDataSelector = createSelector(\n    this.layerSelector,\n    this.mapLayersSelector,\n    layerSelector\n  );\n\n  render() {\n    const items = this.initialDataSelector(this.props);\n\n    if (!items) {\n      return null;\n    }\n\n    const {\n      dragRotate,\n      isSplit,\n      isExport,\n      mapIndex,\n      mapControls,\n      onTogglePerspective,\n      onToggleSplitMap,\n      onMapToggleLayer,\n      onToggleMapControl,\n      scale\n    } = this.props;\n\n    const {\n      visibleLayers = {},\n      mapLegend = {},\n      toggle3d = {},\n      splitMap = {}\n    } = mapControls;\n\n    return (\n      <StyledMapControl className=\"map-control\">\n        {/* Split Map */}\n        {splitMap.show ? (\n          <ActionPanel key={0}>\n            <StyledMapControlButton\n              active={isSplit}\n              onClick={e => {\n                e.preventDefault();\n                onToggleSplitMap(isSplit ? mapIndex : undefined);\n              }}\n              key={`split-${isSplit}`}\n              className=\"map-control-button split-map\"\n              data-tip\n              data-for=\"action-toggle\"\n            >\n              {isSplit ? <Delete height=\"18px\" /> : <Split height=\"18px\" />}\n              <MapLegendTooltip\n                id=\"action-toggle\"\n                message={\n                  isSplit ? 'Close current panel' : 'Switch to dual map view'\n                }\n              />\n            </StyledMapControlButton>\n          </ActionPanel>\n        ) : null}\n\n        {/* Map Layers */}\n        {isSplit && visibleLayers.show ? (\n          <ActionPanel key={1}>\n            <LayerSelectorPanel\n              items={items}\n              onMapToggleLayer={onMapToggleLayer}\n              isActive={visibleLayers.active}\n              toggleMenuPanel={() => onToggleMapControl('visibleLayers')}\n            />\n          </ActionPanel>\n        ) : null}\n\n        {/* 3D Map */}\n        {toggle3d.show ? (\n          <ActionPanel key={2}>\n            <StyledMapControlButton\n              onClick={e => {\n                e.preventDefault();\n                onTogglePerspective();\n              }}\n              active={dragRotate}\n              data-tip\n              data-for=\"action-3d\"\n            >\n              <Cube3d height=\"22px\" />\n              {/* No icon since we are injecting through css .threeD-map class*/}\n              <MapLegendTooltip\n                id=\"action-3d\"\n                message={dragRotate ? 'Disable 3D Map' : '3D Map'}\n              />\n            </StyledMapControlButton>\n          </ActionPanel>\n        ) : null}\n\n        {/* Map Legend */}\n        {mapLegend.show ? (\n          <ActionPanel key={3}>\n            <MapLegendPanel\n              items={items}\n              scale={scale}\n              isExport={isExport}\n              onMapToggleLayer={onMapToggleLayer}\n              isActive={mapLegend.active}\n              toggleMenuPanel={() => onToggleMapControl('mapLegend')}\n            />\n          </ActionPanel>\n        ) : null}\n      </StyledMapControl>\n    );\n  }\n}\n\nconst MapControlPanel = ({children, header, onClick, scale = 1, isExport}) => (\n  <StyledMapControlPanel\n    style={{\n      transform: `scale(${scale}) translate(calc(-${25 * (scale - 1)}% - ${10 *\n        scale}px), calc(${25 * (scale - 1)}% + ${10 * scale}px))`\n    }}\n  >\n    <StyledMapControlPanelHeader style={{position: 'relative'}}>\n      {isExport ? (\n        <KeplerGlLogo version={false} appName=\"kepler.gl\"/>\n      ) : (\n        <span style={{verticalAlign: 'middle'}}>{header}</span>\n      )}\n      {isExport ? null : (\n        <IconRoundSmall>\n          <Close height=\"16px\" onClick={onClick} />\n        </IconRoundSmall>\n      )}\n    </StyledMapControlPanelHeader>\n    <StyledMapControlPanelContent>{children}</StyledMapControlPanelContent>\n  </StyledMapControlPanel>\n);\n\nconst MapLegendPanel = ({items, isActive, scale, toggleMenuPanel, isExport}) =>\n  !isActive ? (\n    <StyledMapControlButton\n      key={2}\n      data-tip\n      data-for=\"show-legend\"\n      className=\"map-control-button show-legend\"\n      onClick={e => {\n        e.preventDefault();\n        toggleMenuPanel();\n      }}\n    >\n      <Legend height=\"22px\" />\n      <MapLegendTooltip id=\"show-legend\" message={'show legend'} />\n    </StyledMapControlButton>\n  ) : (\n    <MapControlPanel\n      scale={scale}\n      header={'Layer Legend'}\n      onClick={toggleMenuPanel}\n      isExport={isExport}\n    >\n      <MapLegend\n        layers={items.filter(item => item.isVisible).map(item => item.layer)}\n      />\n    </MapControlPanel>\n  );\n\nconst LayerSelectorPanel = ({\n  items,\n  onMapToggleLayer,\n  isActive,\n  toggleMenuPanel\n}) =>\n  !isActive ? (\n    <StyledMapControlButton\n      key={1}\n      onClick={e => {\n        e.preventDefault();\n        toggleMenuPanel();\n      }}\n      className=\"map-control-button toggle-layer\"\n      data-tip\n      data-for=\"toggle-layer\"\n    >\n      <Layers height=\"22px\" />\n      <MapLegendTooltip\n        id=\"toggle-layer\"\n        message={isActive ? 'Hide layer panel' : 'Show layer panel'}\n      />\n    </StyledMapControlButton>\n  ) : (\n    <MapControlPanel header=\"Visible layers\" onClick={toggleMenuPanel}>\n      <MapLayerSelector layers={items} onMapToggleLayer={onMapToggleLayer} />\n    </MapControlPanel>\n  );\n\nconst ActionPanel = ({children}) => (\n  <StyledMapControlAction>{children}</StyledMapControlAction>\n);\n\nconst MapLegendTooltip = ({id, message}) => (\n  <Tooltip id={id} place=\"left\" effect=\"solid\">\n    <span>{message}</span>\n  </Tooltip>\n);\n\nconst MapControlFactory = () => MapControl;\n\nexport default MapControlFactory;\n"]}