UNPKG

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
"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