kepler.gl
Version:
kepler.gl is a webgl based application to visualize large scale location data in the browser
396 lines (394 loc) • 68.5 kB
JavaScript
"use strict";
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
var _typeof = require("@babel/runtime/helpers/typeof");
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.LayerColorLegendFactory = LayerColorLegendFactory;
exports.LayerDefaultLegend = void 0;
exports.LayerLegendContentFactory = LayerLegendContentFactory;
exports.LayerLegendHeaderFactory = LayerLegendHeaderFactory;
exports.LayerRadiusLegend = void 0;
exports.SingleColorLegendFactory = SingleColorLegendFactory;
exports["default"] = exports.VisualChannelMetric = exports.StyledMapControlLegend = void 0;
var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty"));
var _slicedToArray2 = _interopRequireDefault(require("@babel/runtime/helpers/slicedToArray"));
var _toConsumableArray2 = _interopRequireDefault(require("@babel/runtime/helpers/toConsumableArray"));
var _taggedTemplateLiteral2 = _interopRequireDefault(require("@babel/runtime/helpers/taggedTemplateLiteral"));
var _react = _interopRequireWildcard(require("react"));
var _styledComponents = _interopRequireDefault(require("styled-components"));
var _d3Color = require("d3-color");
var _d3Format = require("d3-format");
var _reactIntl = require("react-intl");
var _colorLegend = _interopRequireWildcard(require("../common/color-legend"));
var _radiusLegend = _interopRequireDefault(require("../common/radius-legend"));
var _constants = require("@kepler.gl/constants");
var _localization = require("@kepler.gl/localization");
var _viewportMercatorProject = require("viewport-mercator-project");
var _icons = require("../common/icons");
var _panelHeaderAction = _interopRequireDefault(require("../side-panel/panel-header-action"));
var _templateObject, _templateObject2, _templateObject3; // SPDX-License-Identifier: MIT
// Copyright contributors to the kepler.gl project
function _getRequireWildcardCache(e) { if ("function" != typeof WeakMap) return null; var r = new WeakMap(), t = new WeakMap(); return (_getRequireWildcardCache = function _getRequireWildcardCache(e) { return e ? t : r; })(e); }
function _interopRequireWildcard(e, r) { if (!r && e && e.__esModule) return e; if (null === e || "object" != _typeof(e) && "function" != typeof e) return { "default": e }; var t = _getRequireWildcardCache(r); if (t && t.has(e)) return t.get(e); var n = { __proto__: null }, a = Object.defineProperty && Object.getOwnPropertyDescriptor; for (var u in e) if ("default" !== u && {}.hasOwnProperty.call(e, u)) { var i = a ? Object.getOwnPropertyDescriptor(e, u) : null; i && (i.get || i.set) ? Object.defineProperty(n, u, i) : n[u] = e[u]; } return n["default"] = e, t && t.set(e, n), n; }
function ownKeys(e, r) { var t = Object.keys(e); if (Object.getOwnPropertySymbols) { var o = Object.getOwnPropertySymbols(e); r && (o = o.filter(function (r) { return Object.getOwnPropertyDescriptor(e, r).enumerable; })), t.push.apply(t, o); } return t; }
function _objectSpread(e) { for (var r = 1; r < arguments.length; r++) { var t = null != arguments[r] ? arguments[r] : {}; r % 2 ? ownKeys(Object(t), !0).forEach(function (r) { (0, _defineProperty2["default"])(e, r, t[r]); }) : Object.getOwnPropertyDescriptors ? Object.defineProperties(e, Object.getOwnPropertyDescriptors(t)) : ownKeys(Object(t)).forEach(function (r) { Object.defineProperty(e, r, Object.getOwnPropertyDescriptor(t, r)); }); } return e; }
var StyledMapControlLegend = exports.StyledMapControlLegend = _styledComponents["default"].div(_templateObject || (_templateObject = (0, _taggedTemplateLiteral2["default"])(["\n padding: 10px ", "px 10px\n ", "px;\n font-size: 11px;\n font-family: ", ";\n border-bottom-color: ", ";\n border-bottom-style: solid;\n border-bottom-width: ", ";\n width: ", "px;\n box-sizing: border-box;\n\n .legend--layer_name {\n font-size: 12px;\n padding-right: ", "px;\n color: ", ";\n font-weight: 500;\n }\n .legend--layer_type {\n color: ", ";\n font-weight: 500;\n font-size: 11px;\n padding-right: ", "px;\n }\n\n .legend--layer_size-title-row {\n display: flex;\n margin-top: 4px;\n padding-right: ", "px;\n align-items: center;\n }\n\n .legend--layer__title {\n }\n\n .legend--layer__item {\n padding-bottom: 4px;\n }\n .legend--layer_by {\n color: ", ";\n margin-top: 4px;\n }\n\n .legend--layer_color_field {\n color: ", ";\n font-weight: 500;\n }\n\n .legend--layer_color-legend {\n margin-top: 6px;\n }\n"])), function (props) {
return props.theme.mapControl.padding;
}, function (props) {
return props.theme.mapControl.padding;
}, function (props) {
return props.theme.fontFamily;
}, function (props) {
return props.theme.panelBorderColor;
}, function (props) {
return props.last ? 0 : '1px';
}, function (props) {
return props.width;
}, function (props) {
return props.theme.mapControl.padding;
}, function (props) {
return props.theme.textColor;
}, function (props) {
return props.theme.subtextColor;
}, function (props) {
return props.theme.mapControl.padding;
}, function (props) {
return props.theme.mapControl.padding;
}, function (props) {
return props.theme.subtextColor;
}, function (props) {
return props.theme.textColorHl;
});
var StyledLegendHeaderRow = _styledComponents["default"].div(_templateObject2 || (_templateObject2 = (0, _taggedTemplateLiteral2["default"])(["\n display: flex;\n align-items: center;\n justify-content: space-between;\n"])));
var StyledVisibilityToggle = _styledComponents["default"].div(_templateObject3 || (_templateObject3 = (0, _taggedTemplateLiteral2["default"])(["\n cursor: pointer;\n color: ", ";\n display: flex;\n align-items: center;\n margin-left: 8px;\n opacity: ", ";\n\n &:hover {\n color: ", ";\n opacity: 1;\n }\n"])), function (props) {
return props.isVisible ? props.theme.textColor : props.theme.subtextColor;
}, function (props) {
return props.isVisible ? 1 : 0.5;
}, function (props) {
return props.theme.textColorHl;
});
var VisualChannelMetric = exports.VisualChannelMetric = function VisualChannelMetric(_ref) {
var name = _ref.name;
return /*#__PURE__*/_react["default"].createElement("div", {
className: "legend--layer__title"
}, /*#__PURE__*/_react["default"].createElement("span", {
className: "legend--layer_color_field"
}, /*#__PURE__*/_react["default"].createElement(_localization.FormattedMessage, {
id: name
})));
};
var LayerDefaultLegend = exports.LayerDefaultLegend = function LayerDefaultLegend(_ref2) {
var label = _ref2.label,
name = _ref2.name;
return label ? /*#__PURE__*/_react["default"].createElement("div", {
className: "legend--layer_size-schema"
}, /*#__PURE__*/_react["default"].createElement("p", null, /*#__PURE__*/_react["default"].createElement("span", {
className: "legend--layer_by"
}, label ? /*#__PURE__*/_react["default"].createElement(_localization.FormattedMessage, {
id: label
}) : null), /*#__PURE__*/_react["default"].createElement("span", {
className: "legend--layer_by"
}, " by ")), name && /*#__PURE__*/_react["default"].createElement(VisualChannelMetric, {
name: name
})) : null;
};
SingleColorLegendFactory.deps = [_colorLegend.LegendRowFactory];
function SingleColorLegendFactory(LegendRow) {
var SingleColorLegend = function SingleColorLegend(_ref3) {
var color = _ref3.color,
label = _ref3.label;
return /*#__PURE__*/_react["default"].createElement(LegendRow, {
label: label !== null && label !== void 0 ? label : '',
displayLabel: Boolean(label),
color: Array.isArray(color) ? _d3Color.rgb.apply(void 0, (0, _toConsumableArray2["default"])(color)).toString() : color
});
};
SingleColorLegend.displayName = 'SingleColorLegend';
return /*#__PURE__*/_react["default"].memo(SingleColorLegend);
}
LayerColorLegendFactory.deps = [_colorLegend["default"], SingleColorLegendFactory, _panelHeaderAction["default"]];
function LayerColorLegendFactory(ColorLegend, SingleColorLegend, PanelHeaderAction) {
var LayerColorLegend = function LayerColorLegend(_ref4) {
var description = _ref4.description,
config = _ref4.config,
layer = _ref4.layer,
colorChannel = _ref4.colorChannel,
disableEdit = _ref4.disableEdit,
onLayerVisConfigChange = _ref4.onLayerVisConfigChange,
isExport = _ref4.isExport,
mapState = _ref4.mapState,
actionIcons = _ref4.actionIcons;
var intl = (0, _reactIntl.useIntl)();
var enableColorBy = description.measure;
var scale = colorChannel.scale,
field = colorChannel.field,
domain = colorChannel.domain,
range = colorChannel.range,
property = colorChannel.property,
fixed = colorChannel.fixed;
var _map = [scale, field, domain].map(function (k) {
return config[k];
}),
_map2 = (0, _slicedToArray2["default"])(_map, 3),
colorScale = _map2[0],
colorField = _map2[1],
colorDomain = _map2[2];
var isFixed = fixed && config.visConfig[fixed];
var colorRange = config.visConfig[range];
var onUpdateColorLegend = (0, _react.useCallback)(function (colorLegends) {
if (onLayerVisConfigChange) {
onLayerVisConfigChange(layer, (0, _defineProperty2["default"])({}, range, _objectSpread(_objectSpread({}, colorRange), {}, {
colorLegends: colorLegends
})));
}
}, [layer, onLayerVisConfigChange, colorRange, range]);
var _useState = (0, _react.useState)(isExport),
_useState2 = (0, _slicedToArray2["default"])(_useState, 2),
isExpanded = _useState2[0],
setIsExpanded = _useState2[1];
var handleToggleExpanded = function handleToggleExpanded() {
return setIsExpanded(!isExpanded);
};
return /*#__PURE__*/_react["default"].createElement("div", {
className: "legend--layer__item"
}, /*#__PURE__*/_react["default"].createElement("div", {
className: "legend--layer_color-schema"
}, /*#__PURE__*/_react["default"].createElement("div", null, enableColorBy ? /*#__PURE__*/_react["default"].createElement("div", {
className: "legend--layer_size-title-row"
}, /*#__PURE__*/_react["default"].createElement(VisualChannelMetric, {
name: enableColorBy
}), !isExport ? /*#__PURE__*/_react["default"].createElement(PanelHeaderAction, {
id: "legend-collapse-button",
onClick: handleToggleExpanded,
IconComponent: isExpanded ? actionIcons.expanded : actionIcons.collapsed
}) : null) : null, /*#__PURE__*/_react["default"].createElement("div", {
className: "legend--layer_color-legend"
}, enableColorBy ? /*#__PURE__*/_react["default"].createElement(ColorLegend, {
layer: layer,
isExpanded: isExpanded,
scaleType: colorScale,
displayLabel: true,
domain: colorDomain,
fieldType: colorField && colorField.type || 'real',
range: colorRange,
onUpdateColorLegend: onUpdateColorLegend,
disableEdit: disableEdit || Boolean(isExport),
isFixed: isFixed,
mapState: mapState,
labelFormat: colorField !== null && colorField !== void 0 && colorField.displayFormat ? (0, _d3Format.format)(colorField === null || colorField === void 0 ? void 0 : colorField.displayFormat) : null
}) : /*#__PURE__*/_react["default"].createElement(SingleColorLegend, {
color: config.visConfig[property] || config[property] || config.color,
label: intl.formatMessage({
id: "mapLegend.layers.".concat(layer.type, ".singleColor.").concat(colorChannel.key),
defaultMessage: intl.formatMessage({
id: "mapLegend.layers.default.singleColor.".concat(colorChannel.key),
defaultMessage: ' ' // mustn't be empty string or id will be used
})
})
})))));
};
LayerColorLegend.displayName = 'LayerColorLegend';
return /*#__PURE__*/_react["default"].memo(LayerColorLegend);
}
function getLayerRadiusScaleMetersToPixelsMultiplier(layer, mapState) {
// @ts-ignore this actually exist
var _getDistanceScales = (0, _viewportMercatorProject.getDistanceScales)(mapState),
metersPerPixel = _getDistanceScales.metersPerPixel;
// if no field size is defined we need to pass fixed radius = false
var fixedRadius = layer.config.visConfig.fixedRadius && Boolean(layer.config.sizeField);
return layer.getRadiusScaleByZoom(mapState, fixedRadius) / metersPerPixel[0];
}
var LayerRadiusLegend = exports.LayerRadiusLegend = /*#__PURE__*/_react["default"].memo(function (_ref5) {
var layer = _ref5.layer,
width = _ref5.width,
visualChannel = _ref5.visualChannel,
mapState = _ref5.mapState;
var description = layer.getVisualChannelDescription(visualChannel.key);
var config = layer.config;
var enableSizeBy = description.measure;
var scale = visualChannel.scale,
field = visualChannel.field,
domain = visualChannel.domain,
range = visualChannel.range;
var _map3 = [scale, field, domain].map(function (k) {
return config[k];
}),
_map4 = (0, _slicedToArray2["default"])(_map3, 3),
sizeScale = _map4[0],
sizeField = _map4[1],
sizeDomain = _map4[2];
var sizeRange = config.visConfig[range];
if (mapState) {
var radiusMultiplier = getLayerRadiusScaleMetersToPixelsMultiplier(layer, mapState);
sizeRange = sizeRange.map(function (v) {
return v * radiusMultiplier;
});
}
return /*#__PURE__*/_react["default"].createElement("div", null, /*#__PURE__*/_react["default"].createElement("div", {
className: "legend--layer_size-schema"
}, /*#__PURE__*/_react["default"].createElement("div", null, enableSizeBy ? /*#__PURE__*/_react["default"].createElement(VisualChannelMetric, {
name: enableSizeBy
}) : null, /*#__PURE__*/_react["default"].createElement("div", {
className: "legend--layer_size-legend"
}, enableSizeBy ? /*#__PURE__*/_react["default"].createElement(_radiusLegend["default"], {
scaleType: sizeScale,
domain: sizeDomain,
fieldType: sizeField && sizeField.type || 'real',
range: sizeRange,
width: width
}) : null))));
});
var isColorChannel = function isColorChannel(visualChannel) {
return [_constants.CHANNEL_SCALES.color, _constants.CHANNEL_SCALES.colorAggr].includes(visualChannel.channelScaleType);
};
var isRadiusChannel = function isRadiusChannel(visualChannel) {
return [_constants.CHANNEL_SCALES.radius].includes(visualChannel.channelScaleType);
};
function LayerLegendHeaderFactory() {
var LayerLegendHeader = function LayerLegendHeader(_ref6) {
var options = _ref6.options,
layer = _ref6.layer,
onToggleLayerVisibility = _ref6.onToggleLayerVisibility;
var isVisible = layer.config.isVisible;
var onToggle = (0, _react.useCallback)(function () {
if (onToggleLayerVisibility) {
onToggleLayerVisibility(layer);
}
}, [layer, onToggleLayerVisibility]);
if ((options === null || options === void 0 ? void 0 : options.showLayerName) === false) {
return null;
}
return /*#__PURE__*/_react["default"].createElement(StyledLegendHeaderRow, null, /*#__PURE__*/_react["default"].createElement("div", {
className: "legend--layer_name",
style: {
opacity: isVisible ? 1 : 0.5
}
}, layer.config.label), onToggleLayerVisibility ? /*#__PURE__*/_react["default"].createElement(StyledVisibilityToggle, {
isVisible: isVisible,
onClick: onToggle
}, isVisible ? /*#__PURE__*/_react["default"].createElement(_icons.EyeSeen, {
height: "12px"
}) : /*#__PURE__*/_react["default"].createElement(_icons.EyeUnseen, {
height: "12px"
})) : null);
};
return LayerLegendHeader;
}
var defaultActionIcons = {
expanded: _icons.ArrowDown,
collapsed: _icons.ArrowRight
};
LayerLegendContentFactory.deps = [LayerColorLegendFactory];
function LayerLegendContentFactory(LayerColorLegend) {
var LayerLegendContent = function LayerLegendContent(_ref7) {
var layer = _ref7.layer,
containerW = _ref7.containerW,
mapState = _ref7.mapState,
disableEdit = _ref7.disableEdit,
isExport = _ref7.isExport,
onLayerVisConfigChange = _ref7.onLayerVisConfigChange,
actionIcons = _ref7.actionIcons;
var visualChannels = layer.getLegendVisualChannels();
var channelKeys = Object.values(visualChannels);
var colorChannels = channelKeys.filter(isColorChannel);
var nonColorChannels = channelKeys.filter(function (vc) {
return !isColorChannel(vc);
});
var width = containerW - 2 * _constants.DIMENSIONS.mapControl.padding;
// render color by chanel only
var colorChannelToRender = colorChannels.filter(function (cc) {
var _layer$getVisualChann;
return (!cc.condition || cc.condition(layer.config)) && ((_layer$getVisualChann = layer.getVisualChannelDescription(cc.key)) === null || _layer$getVisualChann === void 0 ? void 0 : _layer$getVisualChann.measure);
});
// if no color by chanel, render rest
if (!colorChannelToRender.length) {
colorChannelToRender = colorChannels.filter(function (cc) {
return !cc.condition || cc.condition(layer.config);
});
}
return /*#__PURE__*/_react["default"].createElement(_react["default"].Fragment, null, colorChannelToRender.map(function (colorChannel) {
return /*#__PURE__*/_react["default"].createElement(LayerColorLegend, {
key: colorChannel.key,
colorChannel: colorChannel,
config: layer.config,
description: layer.getVisualChannelDescription(colorChannel.key),
layer: layer,
isExport: isExport,
disableEdit: disableEdit,
mapState: mapState,
onLayerVisConfigChange: onLayerVisConfigChange,
actionIcons: actionIcons
});
}), nonColorChannels.map(function (visualChannel) {
var matchCondition = !visualChannel.condition || visualChannel.condition(layer.config);
var enabled = layer.config[visualChannel.field] || visualChannel.defaultMeasure;
if (matchCondition && enabled) {
var description = layer.getVisualChannelDescription(visualChannel.key);
if (isRadiusChannel(visualChannel)) {
return /*#__PURE__*/_react["default"].createElement(LayerRadiusLegend, {
key: visualChannel.key,
layer: layer,
mapState: mapState,
width: width,
visualChannel: visualChannel
});
}
return /*#__PURE__*/_react["default"].createElement(LayerDefaultLegend, {
key: visualChannel.key,
label: description.label,
name: description.measure
});
}
return null;
}));
};
return LayerLegendContent;
}
MapLegendFactory.deps = [LayerLegendHeaderFactory, LayerLegendContentFactory];
function MapLegendFactory(LayerLegendHeader, LayerLegendContent) {
var MapLegend = function MapLegend(_ref8) {
var _ref8$layers = _ref8.layers,
layers = _ref8$layers === void 0 ? [] : _ref8$layers,
width = _ref8.width,
mapState = _ref8.mapState,
options = _ref8.options,
disableEdit = _ref8.disableEdit,
isExport = _ref8.isExport,
onLayerVisConfigChange = _ref8.onLayerVisConfigChange,
onToggleLayerVisibility = _ref8.onToggleLayerVisibility,
_ref8$actionIcons = _ref8.actionIcons,
actionIcons = _ref8$actionIcons === void 0 ? defaultActionIcons : _ref8$actionIcons;
return /*#__PURE__*/_react["default"].createElement("div", {
className: "map-legend"
}, layers.map(function (layer, index) {
if (!layer.isValidToSave() || layer.config.hidden) {
return null;
}
var containerW = width || _constants.DIMENSIONS.mapControl.width;
return /*#__PURE__*/_react["default"].createElement(StyledMapControlLegend, {
className: "legend--layer",
last: index === layers.length - 1,
key: index,
width: containerW
}, /*#__PURE__*/_react["default"].createElement(LayerLegendHeader, {
isExport: isExport,
options: options,
layer: layer,
onToggleLayerVisibility: onToggleLayerVisibility
}), /*#__PURE__*/_react["default"].createElement(LayerLegendContent, {
containerW: containerW,
layer: layer,
mapState: mapState,
disableEdit: disableEdit,
isExport: isExport,
onLayerVisConfigChange: onLayerVisConfigChange,
actionIcons: actionIcons
}));
}));
};
MapLegend.displayName = 'MapLegend';
return MapLegend;
}
var _default = exports["default"] = MapLegendFactory;
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_react","_interopRequireWildcard","require","_styledComponents","_interopRequireDefault","_d3Color","_d3Format","_reactIntl","_colorLegend","_radiusLegend","_constants","_localization","_viewportMercatorProject","_icons","_panelHeaderAction","_templateObject","_templateObject2","_templateObject3","_getRequireWildcardCache","e","WeakMap","r","t","__esModule","_typeof","has","get","n","__proto__","a","Object","defineProperty","getOwnPropertyDescriptor","u","hasOwnProperty","call","i","set","ownKeys","keys","getOwnPropertySymbols","o","filter","enumerable","push","apply","_objectSpread","arguments","length","forEach","_defineProperty2","getOwnPropertyDescriptors","defineProperties","StyledMapControlLegend","exports","styled","div","_taggedTemplateLiteral2","props","theme","mapControl","padding","fontFamily","panelBorderColor","last","width","textColor","subtextColor","textColorHl","StyledLegendHeaderRow","StyledVisibilityToggle","isVisible","VisualChannelMetric","_ref","name","createElement","className","FormattedMessage","id","LayerDefaultLegend","_ref2","label","SingleColorLegendFactory","deps","LegendRowFactory","LegendRow","SingleColorLegend","_ref3","color","displayLabel","Boolean","Array","isArray","rgb","_toConsumableArray2","toString","displayName","React","memo","LayerColorLegendFactory","ColorLegendFactory","PanelHeaderActionFactory","ColorLegend","PanelHeaderAction","LayerColorLegend","_ref4","description","config","layer","colorChannel","disableEdit","onLayerVisConfigChange","isExport","mapState","actionIcons","intl","useIntl","enableColorBy","measure","scale","field","domain","range","property","fixed","_map","map","k","_map2","_slicedToArray2","colorScale","colorField","colorDomain","isFixed","visConfig","colorRange","onUpdateColorLegend","useCallback","colorLegends","_useState","useState","_useState2","isExpanded","setIsExpanded","handleToggleExpanded","onClick","IconComponent","expanded","collapsed","scaleType","fieldType","type","labelFormat","displayFormat","d3Format","formatMessage","concat","key","defaultMessage","getLayerRadiusScaleMetersToPixelsMultiplier","_getDistanceScales","getDistanceScales","metersPerPixel","fixedRadius","sizeField","getRadiusScaleByZoom","LayerRadiusLegend","_ref5","visualChannel","getVisualChannelDescription","enableSizeBy","_map3","_map4","sizeScale","sizeDomain","sizeRange","radiusMultiplier","v","isColorChannel","CHANNEL_SCALES","colorAggr","includes","channelScaleType","isRadiusChannel","radius","LayerLegendHeaderFactory","LayerLegendHeader","_ref6","options","onToggleLayerVisibility","onToggle","showLayerName","style","opacity","EyeSeen","height","EyeUnseen","defaultActionIcons","ArrowDown","ArrowRight","LayerLegendContentFactory","LayerLegendContent","_ref7","containerW","visualChannels","getLegendVisualChannels","channelKeys","values","colorChannels","nonColorChannels","vc","DIMENSIONS","colorChannelToRender","cc","_layer$getVisualChann","condition","Fragment","matchCondition","enabled","defaultMeasure","MapLegendFactory","MapLegend","_ref8","_ref8$layers","layers","_ref8$actionIcons","index","isValidToSave","hidden","_default"],"sources":["../../src/map/map-legend.tsx"],"sourcesContent":["// SPDX-License-Identifier: MIT\n// Copyright contributors to the kepler.gl project\n\nimport React, {FC, useCallback, useState, ComponentType} from 'react';\nimport styled from 'styled-components';\nimport {rgb} from 'd3-color';\nimport {format as d3Format} from 'd3-format';\nimport {useIntl} from 'react-intl';\nimport ColorLegendFactory, {LegendRowFactory} from '../common/color-legend';\nimport RadiusLegend from '../common/radius-legend';\nimport {CHANNEL_SCALES, DIMENSIONS} from '@kepler.gl/constants';\nimport {FormattedMessage} from '@kepler.gl/localization';\nimport {Layer, LayerBaseConfig, VisualChannel, VisualChannelDescription} from '@kepler.gl/layers';\nimport {LayerVisConfig, MapState, RGBColor} from '@kepler.gl/types';\nimport {getDistanceScales} from 'viewport-mercator-project';\nimport {ArrowDown, ArrowRight, EyeSeen, EyeUnseen} from '../common/icons';\nimport PanelHeaderActionFactory from '../side-panel/panel-header-action';\n\ninterface StyledMapControlLegendProps {\n  width?: number;\n  last?: boolean;\n}\n\nexport const StyledMapControlLegend = styled.div<StyledMapControlLegendProps>`\n  padding: 10px ${props => props.theme.mapControl.padding}px 10px\n    ${props => props.theme.mapControl.padding}px;\n  font-size: 11px;\n  font-family: ${props => props.theme.fontFamily};\n  border-bottom-color: ${props => props.theme.panelBorderColor};\n  border-bottom-style: solid;\n  border-bottom-width: ${props => (props.last ? 0 : '1px')};\n  width: ${props => props.width}px;\n  box-sizing: border-box;\n\n  .legend--layer_name {\n    font-size: 12px;\n    padding-right: ${props => props.theme.mapControl.padding}px;\n    color: ${props => props.theme.textColor};\n    font-weight: 500;\n  }\n  .legend--layer_type {\n    color: ${props => props.theme.subtextColor};\n    font-weight: 500;\n    font-size: 11px;\n    padding-right: ${props => props.theme.mapControl.padding}px;\n  }\n\n  .legend--layer_size-title-row {\n    display: flex;\n    margin-top: 4px;\n    padding-right: ${props => props.theme.mapControl.padding}px;\n    align-items: center;\n  }\n\n  .legend--layer__title {\n  }\n\n  .legend--layer__item {\n    padding-bottom: 4px;\n  }\n  .legend--layer_by {\n    color: ${props => props.theme.subtextColor};\n    margin-top: 4px;\n  }\n\n  .legend--layer_color_field {\n    color: ${props => props.theme.textColorHl};\n    font-weight: 500;\n  }\n\n  .legend--layer_color-legend {\n    margin-top: 6px;\n  }\n`;\n\nconst StyledLegendHeaderRow = styled.div`\n  display: flex;\n  align-items: center;\n  justify-content: space-between;\n`;\n\nconst StyledVisibilityToggle = styled.div<{isVisible: boolean}>`\n  cursor: pointer;\n  color: ${props => (props.isVisible ? props.theme.textColor : props.theme.subtextColor)};\n  display: flex;\n  align-items: center;\n  margin-left: 8px;\n  opacity: ${props => (props.isVisible ? 1 : 0.5)};\n\n  &:hover {\n    color: ${props => props.theme.textColorHl};\n    opacity: 1;\n  }\n`;\n\nexport const VisualChannelMetric = ({name}) => {\n  return (\n    <div className=\"legend--layer__title\">\n      <span className=\"legend--layer_color_field\">\n        <FormattedMessage id={name} />\n      </span>\n    </div>\n  );\n};\n\nexport type LayerSizeLegendProps = {\n  label: string;\n  name: string | undefined;\n};\n\nexport const LayerDefaultLegend: React.FC<LayerSizeLegendProps> = ({label, name}) =>\n  label ? (\n    <div className=\"legend--layer_size-schema\">\n      <p>\n        <span className=\"legend--layer_by\">{label ? <FormattedMessage id={label} /> : null}</span>\n        <span className=\"legend--layer_by\"> by </span>\n      </p>\n      {name && <VisualChannelMetric name={name} />}\n    </div>\n  ) : null;\n\nexport type SingleColorLegendProps = {\n  color: RGBColor;\n  label?: string;\n};\n\nSingleColorLegendFactory.deps = [LegendRowFactory];\n\nexport function SingleColorLegendFactory(LegendRow: ReturnType<typeof LegendRowFactory>) {\n  const SingleColorLegend: React.FC<SingleColorLegendProps> = ({color, label}) => (\n    <LegendRow\n      label={label ?? ''}\n      displayLabel={Boolean(label)}\n      color={Array.isArray(color) ? rgb(...color).toString() : color}\n    />\n  );\n\n  SingleColorLegend.displayName = 'SingleColorLegend';\n\n  return React.memo(SingleColorLegend);\n}\n\nexport type LayerColorLegendProps = {\n  description: VisualChannelDescription;\n  config: LayerBaseConfig;\n  colorChannel: VisualChannel;\n  onLayerVisConfigChange?: (oldLayer: Layer, newVisConfig: Partial<LayerVisConfig>) => void;\n  layer: Layer;\n  disableEdit?: boolean;\n  isExport?: boolean;\n  mapState?: MapState;\n  actionIcons: MapLegendIcons;\n};\n\nLayerColorLegendFactory.deps = [\n  ColorLegendFactory,\n  SingleColorLegendFactory,\n  PanelHeaderActionFactory\n];\nexport function LayerColorLegendFactory(\n  ColorLegend: ReturnType<typeof ColorLegendFactory>,\n  SingleColorLegend: ReturnType<typeof SingleColorLegendFactory>,\n  PanelHeaderAction: ReturnType<typeof PanelHeaderActionFactory>\n) {\n  const LayerColorLegend: React.FC<LayerColorLegendProps> = ({\n    description,\n    config,\n    layer,\n    colorChannel,\n    disableEdit,\n    onLayerVisConfigChange,\n    isExport,\n    mapState,\n    actionIcons\n  }) => {\n    const intl = useIntl();\n    const enableColorBy = description.measure;\n    const {scale, field, domain, range, property, fixed} = colorChannel;\n    const [colorScale, colorField, colorDomain] = [scale, field, domain].map(k => config[k]);\n    const isFixed = fixed && config.visConfig[fixed];\n\n    const colorRange = config.visConfig[range];\n    const onUpdateColorLegend = useCallback(\n      colorLegends => {\n        if (onLayerVisConfigChange) {\n          onLayerVisConfigChange(layer, {\n            [range]: {\n              ...colorRange,\n              colorLegends\n            }\n          });\n        }\n      },\n      [layer, onLayerVisConfigChange, colorRange, range]\n    );\n    const [isExpanded, setIsExpanded] = useState(isExport);\n    const handleToggleExpanded = () => setIsExpanded(!isExpanded);\n    return (\n      <div className=\"legend--layer__item\">\n        <div className=\"legend--layer_color-schema\">\n          <div>\n            {enableColorBy ? (\n              <div className=\"legend--layer_size-title-row\">\n                <VisualChannelMetric name={enableColorBy} />\n                {!isExport ? (\n                  <PanelHeaderAction\n                    id=\"legend-collapse-button\"\n                    onClick={handleToggleExpanded}\n                    IconComponent={isExpanded ? actionIcons.expanded : actionIcons.collapsed}\n                  />\n                ) : null}\n              </div>\n            ) : null}\n            <div className=\"legend--layer_color-legend\">\n              {enableColorBy ? (\n                <ColorLegend\n                  layer={layer}\n                  isExpanded={isExpanded}\n                  scaleType={colorScale}\n                  displayLabel\n                  domain={colorDomain}\n                  fieldType={(colorField && colorField.type) || 'real'}\n                  range={colorRange}\n                  onUpdateColorLegend={onUpdateColorLegend}\n                  disableEdit={disableEdit || Boolean(isExport)}\n                  isFixed={isFixed}\n                  mapState={mapState}\n                  labelFormat={\n                    colorField?.displayFormat ? d3Format(colorField?.displayFormat) : null\n                  }\n                />\n              ) : (\n                <SingleColorLegend\n                  color={config.visConfig[property] || config[property] || config.color}\n                  label={intl.formatMessage({\n                    id: `mapLegend.layers.${layer.type}.singleColor.${colorChannel.key}`,\n                    defaultMessage: intl.formatMessage({\n                      id: `mapLegend.layers.default.singleColor.${colorChannel.key}`,\n                      defaultMessage: ' ' // mustn't be empty string or id will be used\n                    })\n                  })}\n                />\n              )}\n            </div>\n          </div>\n        </div>\n      </div>\n    );\n  };\n\n  LayerColorLegend.displayName = 'LayerColorLegend';\n  return React.memo(LayerColorLegend);\n}\n\nfunction getLayerRadiusScaleMetersToPixelsMultiplier(layer, mapState) {\n  // @ts-ignore this actually exist\n  const {metersPerPixel} = getDistanceScales(mapState);\n  // if no field size is defined we need to pass fixed radius = false\n  const fixedRadius = layer.config.visConfig.fixedRadius && Boolean(layer.config.sizeField);\n  return layer.getRadiusScaleByZoom(mapState, fixedRadius) / metersPerPixel[0];\n}\n\nexport type MapLegendIcons = {\n  expanded: ComponentType<any>;\n  collapsed: ComponentType<any>;\n};\n\nexport type LayerRadiusLegendProps = {\n  layer: Layer;\n  mapState?: MapState;\n  width: number;\n  isExport?: boolean;\n  visualChannel: VisualChannel;\n};\n\nexport const LayerRadiusLegend: FC<LayerRadiusLegendProps> = React.memo(\n  ({layer, width, visualChannel, mapState}) => {\n    const description = layer.getVisualChannelDescription(visualChannel.key);\n    const config = layer.config;\n\n    const enableSizeBy = description.measure;\n    const {scale, field, domain, range} = visualChannel;\n    const [sizeScale, sizeField, sizeDomain] = [scale, field, domain].map(k => config[k]);\n    let sizeRange = config.visConfig[range];\n\n    if (mapState) {\n      const radiusMultiplier = getLayerRadiusScaleMetersToPixelsMultiplier(layer, mapState);\n      sizeRange = sizeRange.map(v => v * radiusMultiplier);\n    }\n\n    return (\n      <div>\n        <div className=\"legend--layer_size-schema\">\n          <div>\n            {enableSizeBy ? <VisualChannelMetric name={enableSizeBy} /> : null}\n            <div className=\"legend--layer_size-legend\">\n              {enableSizeBy ? (\n                <RadiusLegend\n                  scaleType={sizeScale}\n                  domain={sizeDomain}\n                  fieldType={(sizeField && sizeField.type) || 'real'}\n                  range={sizeRange}\n                  width={width}\n                />\n              ) : null}\n            </div>\n          </div>\n        </div>\n      </div>\n    );\n  }\n);\n\nconst isColorChannel = visualChannel =>\n  [CHANNEL_SCALES.color, CHANNEL_SCALES.colorAggr].includes(visualChannel.channelScaleType);\n\nexport type LayerLegendHeaderProps = {\n  layer: Layer;\n  options?: {\n    showLayerName?: boolean;\n  };\n  isExport?: boolean;\n  onToggleLayerVisibility?: (layer: Layer) => void;\n};\n\nconst isRadiusChannel = visualChannel =>\n  [CHANNEL_SCALES.radius].includes(visualChannel.channelScaleType);\n\nexport function LayerLegendHeaderFactory() {\n  const LayerLegendHeader: React.FC<LayerLegendHeaderProps> = ({\n    options,\n    layer,\n    onToggleLayerVisibility\n  }) => {\n    const isVisible = layer.config.isVisible;\n    const onToggle = useCallback(() => {\n      if (onToggleLayerVisibility) {\n        onToggleLayerVisibility(layer);\n      }\n    }, [layer, onToggleLayerVisibility]);\n\n    if (options?.showLayerName === false) {\n      return null;\n    }\n\n    return (\n      <StyledLegendHeaderRow>\n        <div className=\"legend--layer_name\" style={{opacity: isVisible ? 1 : 0.5}}>\n          {layer.config.label}\n        </div>\n        {onToggleLayerVisibility ? (\n          <StyledVisibilityToggle isVisible={isVisible} onClick={onToggle}>\n            {isVisible ? <EyeSeen height=\"12px\" /> : <EyeUnseen height=\"12px\" />}\n          </StyledVisibilityToggle>\n        ) : null}\n      </StyledLegendHeaderRow>\n    );\n  };\n  return LayerLegendHeader;\n}\n\nconst defaultActionIcons = {\n  expanded: ArrowDown,\n  collapsed: ArrowRight\n};\n\nexport type LayerLegendContentProps = {\n  layer: Layer;\n  containerW: number;\n  mapState?: MapState;\n  disableEdit?: boolean;\n  isExport?: boolean;\n  onLayerVisConfigChange?: (oldLayer: Layer, newVisConfig: Partial<LayerVisConfig>) => void;\n  actionIcons: MapLegendIcons;\n};\n\nLayerLegendContentFactory.deps = [LayerColorLegendFactory];\n\nexport function LayerLegendContentFactory(\n  LayerColorLegend: ReturnType<typeof LayerColorLegendFactory>\n) {\n  const LayerLegendContent: React.FC<LayerLegendContentProps> = ({\n    layer,\n    containerW,\n    mapState,\n    disableEdit,\n    isExport,\n    onLayerVisConfigChange,\n    actionIcons\n  }) => {\n    const visualChannels = layer.getLegendVisualChannels();\n    const channelKeys = Object.values(visualChannels);\n    const colorChannels = channelKeys.filter(isColorChannel) as VisualChannel[];\n    const nonColorChannels = channelKeys.filter(vc => !isColorChannel(vc));\n    const width = containerW - 2 * DIMENSIONS.mapControl.padding;\n\n    // render color by chanel only\n    let colorChannelToRender = colorChannels.filter(\n      cc =>\n        (!cc.condition || cc.condition(layer.config)) &&\n        layer.getVisualChannelDescription(cc.key)?.measure\n    );\n    // if no color by chanel, render rest\n    if (!colorChannelToRender.length) {\n      colorChannelToRender = colorChannels.filter(\n        cc => !cc.condition || cc.condition(layer.config)\n      );\n    }\n    return (\n      <>\n        {colorChannelToRender.map(colorChannel => (\n          <LayerColorLegend\n            key={colorChannel.key}\n            colorChannel={colorChannel}\n            config={layer.config}\n            description={layer.getVisualChannelDescription(colorChannel.key)}\n            layer={layer}\n            isExport={isExport}\n            disableEdit={disableEdit}\n            mapState={mapState}\n            onLayerVisConfigChange={onLayerVisConfigChange}\n            actionIcons={actionIcons}\n          />\n        ))}\n        {nonColorChannels.map((visualChannel: VisualChannel) => {\n          const matchCondition = !visualChannel.condition || visualChannel.condition(layer.config);\n          const enabled = layer.config[visualChannel.field] || visualChannel.defaultMeasure;\n\n          if (matchCondition && enabled) {\n            const description = layer.getVisualChannelDescription(visualChannel.key);\n            if (isRadiusChannel(visualChannel)) {\n              return (\n                <LayerRadiusLegend\n                  key={visualChannel.key}\n                  layer={layer}\n                  mapState={mapState}\n                  width={width}\n                  visualChannel={visualChannel}\n                />\n              );\n            }\n            return (\n              <LayerDefaultLegend\n                key={visualChannel.key}\n                label={description.label}\n                name={description.measure}\n              />\n            );\n          }\n          return null;\n        })}\n      </>\n    );\n  };\n\n  return LayerLegendContent;\n}\n\nexport type MapLegendProps = {\n  layers?: ReadonlyArray<Layer>;\n  width?: number;\n  mapState?: MapState;\n  options?: {\n    showLayerName?: boolean;\n  };\n  disableEdit?: boolean;\n  isExport?: boolean;\n  onLayerVisConfigChange?: (oldLayer: Layer, newVisConfig: Partial<LayerVisConfig>) => void;\n  onToggleLayerVisibility?: (layer: Layer) => void;\n  actionIcons?: MapLegendIcons;\n};\n\nMapLegendFactory.deps = [LayerLegendHeaderFactory, LayerLegendContentFactory];\n\nfunction MapLegendFactory(\n  LayerLegendHeader: ReturnType<typeof LayerLegendHeaderFactory>,\n  LayerLegendContent: ReturnType<typeof LayerLegendContentFactory>\n) {\n  const MapLegend: React.FC<MapLegendProps> = ({\n    layers = [],\n    width,\n    mapState,\n    options,\n    disableEdit,\n    isExport,\n    onLayerVisConfigChange,\n    onToggleLayerVisibility,\n    actionIcons = defaultActionIcons\n  }) => (\n    <div className=\"map-legend\">\n      {layers.map((layer, index) => {\n        if (!layer.isValidToSave() || layer.config.hidden) {\n          return null;\n        }\n        const containerW = width || DIMENSIONS.mapControl.width;\n\n        return (\n          <StyledMapControlLegend\n            className=\"legend--layer\"\n            last={index === layers.length - 1}\n            key={index}\n            width={containerW}\n          >\n            <LayerLegendHeader\n              isExport={isExport}\n              options={options}\n              layer={layer}\n              onToggleLayerVisibility={onToggleLayerVisibility}\n            />\n            <LayerLegendContent\n              containerW={containerW}\n              layer={layer}\n              mapState={mapState}\n              disableEdit={disableEdit}\n              isExport={isExport}\n              onLayerVisConfigChange={onLayerVisConfigChange}\n              actionIcons={actionIcons}\n            />\n          </StyledMapControlLegend>\n        );\n      })}\n    </div>\n  );\n\n  MapLegend.displayName = 'MapLegend';\n\n  return MapLegend;\n}\n\nexport default MapLegendFactory;\n"],"mappings":";;;;;;;;;;;;;;;;;;AAGA,IAAAA,MAAA,GAAAC,uBAAA,CAAAC,OAAA;AACA,IAAAC,iBAAA,GAAAC,sBAAA,CAAAF,OAAA;AACA,IAAAG,QAAA,GAAAH,OAAA;AACA,IAAAI,SAAA,GAAAJ,OAAA;AACA,IAAAK,UAAA,GAAAL,OAAA;AACA,IAAAM,YAAA,GAAAP,uBAAA,CAAAC,OAAA;AACA,IAAAO,aAAA,GAAAL,sBAAA,CAAAF,OAAA;AACA,IAAAQ,UAAA,GAAAR,OAAA;AACA,IAAAS,aAAA,GAAAT,OAAA;AAGA,IAAAU,wBAAA,GAAAV,OAAA;AACA,IAAAW,MAAA,GAAAX,OAAA;AACA,IAAAY,kBAAA,GAAAV,sBAAA,CAAAF,OAAA;AAAyE,IAAAa,eAAA,EAAAC,gBAAA,EAAAC,gBAAA,EAhBzE;AACA;AAAA,SAAAC,yBAAAC,CAAA,6BAAAC,OAAA,mBAAAC,CAAA,OAAAD,OAAA,IAAAE,CAAA,OAAAF,OAAA,YAAAF,wBAAA,YAAAA,yBAAAC,CAAA,WAAAA,CAAA,GAAAG,CAAA,GAAAD,CAAA,KAAAF,CAAA;AAAA,SAAAlB,wBAAAkB,CAAA,EAAAE,CAAA,SAAAA,CAAA,IAAAF,CAAA,IAAAA,CAAA,CAAAI,UAAA,SAAAJ,CAAA,eAAAA,CAAA,gBAAAK,OAAA,CAAAL,CAAA,0BAAAA,CAAA,sBAAAA,CAAA,QAAAG,CAAA,GAAAJ,wBAAA,CAAAG,CAAA,OAAAC,CAAA,IAAAA,CAAA,CAAAG,GAAA,CAAAN,CAAA,UAAAG,CAAA,CAAAI,GAAA,CAAAP,CAAA,OAAAQ,CAAA,KAAAC,SAAA,UAAAC,CAAA,GAAAC,MAAA,CAAAC,cAAA,IAAAD,MAAA,CAAAE,wBAAA,WAAAC,CAAA,IAAAd,CAAA,oBAAAc,CAAA,OAAAC,cAAA,CAAAC,IAAA,CAAAhB,CAAA,EAAAc,CAAA,SAAAG,CAAA,GAAAP,CAAA,GAAAC,MAAA,CAAAE,wBAAA,CAAAb,CAAA,EAAAc,CAAA,UAAAG,CAAA,KAAAA,CAAA,CAAAV,GAAA,IAAAU,CAAA,CAAAC,GAAA,IAAAP,MAAA,CAAAC,cAAA,CAAAJ,CAAA,EAAAM,CAAA,EAAAG,CAAA,IAAAT,CAAA,CAAAM,CAAA,IAAAd,CAAA,CAAAc,CAAA,YAAAN,CAAA,cAAAR,CAAA,EAAAG,CAAA,IAAAA,CAAA,CAAAe,GAAA,CAAAlB,CAAA,EAAAQ,CAAA,GAAAA,CAAA;AAAA,SAAAW,QAAAnB,CAAA,EAAAE,CAAA,QAAAC,CAAA,GAAAQ,MAAA,CAAAS,IAAA,CAAApB,CAAA,OAAAW,MAAA,CAAAU,qBAAA,QAAAC,CAAA,GAAAX,MAAA,CAAAU,qBAAA,CAAArB,CAAA,GAAAE,CAAA,KAAAoB,CAAA,GAAAA,CAAA,CAAAC,MAAA,WAAArB,CAAA,WAAAS,MAAA,CAAAE,wBAAA,CAAAb,CAAA,EAAAE,CAAA,EAAAsB,UAAA,OAAArB,CAAA,CAAAsB,IAAA,CAAAC,KAAA,CAAAvB,CAAA,EAAAmB,CAAA,YAAAnB,CAAA;AAAA,SAAAwB,cAAA3B,CAAA,aAAAE,CAAA,MAAAA,CAAA,GAAA0B,SAAA,CAAAC,MAAA,EAAA3B,CAAA,UAAAC,CAAA,WAAAyB,SAAA,CAAA1B,C