kepler.gl
Version:
kepler.gl is a webgl based application to visualize large scale location data in the browser
231 lines (197 loc) • 27.5 kB
JavaScript
"use strict";
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
Object.defineProperty(exports, "__esModule", {
value: true
});
exports["default"] = exports.LayerColorLegend = exports.MultiColorLegend = exports.SingleColorLegend = exports.LayerSizeLegend = exports.VisualChannelMetric = exports.StyledMapControlLegend = void 0;
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 = _interopRequireDefault(require("react"));
var _propTypes = _interopRequireDefault(require("prop-types"));
var _styledComponents = _interopRequireDefault(require("styled-components"));
var _d3Color = require("d3-color");
var _colorLegend = _interopRequireDefault(require("../common/color-legend"));
var _defaultSettings = require("../../constants/default-settings");
var _reactIntl = require("react-intl");
function _templateObject() {
var data = (0, _taggedTemplateLiteral2["default"])(["\n padding: 10px 0 10px ", "px;\n font-size: 11px;\n border-bottom-color: ", ";\n border-bottom-style: solid;\n border-bottom-width: ", ";\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__title {\n padding-right: ", "px;\n }\n\n .legend--layer_by {\n color: ", ";\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"]);
_templateObject = function _templateObject() {
return data;
};
return data;
}
var StyledMapControlLegend = _styledComponents["default"].div(_templateObject(), function (props) {
return props.theme.mapControl.padding;
}, function (props) {
return props.theme.panelBorderColor;
}, function (props) {
return props.last ? 0 : '1px';
}, function (props) {
return props.theme.mapControl.padding;
}, function (props) {
return props.theme.textColorHl;
}, 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.textColor;
});
exports.StyledMapControlLegend = StyledMapControlLegend;
var VisualChannelMetric = function VisualChannelMetric(_ref) {
var name = _ref.name;
return _react["default"].createElement("div", {
className: "legend--layer__title"
}, _react["default"].createElement("span", {
className: "legend--layer_by"
}, "by "), _react["default"].createElement("span", {
className: "legend--layer_color_field"
}, _react["default"].createElement(_reactIntl.FormattedMessage, {
id: name
})));
};
exports.VisualChannelMetric = VisualChannelMetric;
var LayerSizeLegend = function LayerSizeLegend(_ref2) {
var label = _ref2.label,
name = _ref2.name;
return _react["default"].createElement("div", {
className: "legend--layer_size-schema"
}, _react["default"].createElement("p", null, _react["default"].createElement("span", {
className: "legend--layer_by"
}, _react["default"].createElement(_reactIntl.FormattedMessage, {
id: label
}))), _react["default"].createElement(VisualChannelMetric, {
name: name
}));
};
exports.LayerSizeLegend = LayerSizeLegend;
var propTypes = {
layers: _propTypes["default"].arrayOf(_propTypes["default"].object)
};
var SingleColorDomain = [''];
var SingleColorLegend = _react["default"].memo(function (_ref3) {
var width = _ref3.width,
color = _ref3.color;
return _react["default"].createElement(_colorLegend["default"], {
scaleType: "ordinal",
displayLabel: false,
domain: SingleColorDomain,
fieldType: null,
range: [_d3Color.rgb.apply(void 0, (0, _toConsumableArray2["default"])(color)).toString()],
width: width
});
});
exports.SingleColorLegend = SingleColorLegend;
SingleColorLegend.displayName = 'SingleColorLegend';
var MultiColorLegend = _react["default"].memo(function (_ref4) {
var colorRange = _ref4.colorRange,
colorScale = _ref4.colorScale,
colorDomain = _ref4.colorDomain,
colorField = _ref4.colorField,
width = _ref4.width;
return _react["default"].createElement(_colorLegend["default"], {
scaleType: colorScale,
displayLabel: true,
domain: colorDomain,
fieldType: colorField && colorField.type || 'real',
range: colorRange.colors,
width: width
});
});
exports.MultiColorLegend = MultiColorLegend;
MultiColorLegend.displayName = 'MultiColorLegend';
var LayerColorLegend = _react["default"].memo(function (_ref5) {
var description = _ref5.description,
config = _ref5.config,
width = _ref5.width,
colorChannel = _ref5.colorChannel;
var enableColorBy = description.measure;
var scale = colorChannel.scale,
field = colorChannel.field,
domain = colorChannel.domain,
range = colorChannel.range,
property = colorChannel.property,
key = colorChannel.key;
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 colorRange = config.visConfig[range];
return _react["default"].createElement("div", null, _react["default"].createElement("div", {
className: "legend--layer_type"
}, _react["default"].createElement(_reactIntl.FormattedMessage, {
id: "layer.".concat(key)
})), _react["default"].createElement("div", {
className: "legend--layer_color-schema"
}, _react["default"].createElement("div", null, enableColorBy ? _react["default"].createElement(VisualChannelMetric, {
name: enableColorBy
}) : null, _react["default"].createElement("div", {
className: "legend--layer_color-legend"
}, enableColorBy ? _react["default"].createElement(MultiColorLegend, {
colorScale: colorScale,
colorField: colorField,
colorDomain: colorDomain,
colorRange: colorRange,
width: width
}) : _react["default"].createElement(SingleColorLegend, {
color: config.visConfig[property] || config[property] || config.color,
width: width
})))));
});
exports.LayerColorLegend = LayerColorLegend;
LayerColorLegend.displayName = 'LayerColorLegend';
var isColorChannel = function isColorChannel(visualChannel) {
return [_defaultSettings.CHANNEL_SCALES.color, _defaultSettings.CHANNEL_SCALES.colorAggr].includes(visualChannel.channelScaleType);
};
var MAP_LEGEND_WIDTH = _defaultSettings.DIMENSIONS.mapControl.width - 2 * _defaultSettings.DIMENSIONS.mapControl.padding;
var MapLegend = function MapLegend(_ref6) {
var _ref6$layers = _ref6.layers,
layers = _ref6$layers === void 0 ? [] : _ref6$layers;
return _react["default"].createElement("div", {
className: "map-legend"
}, layers.map(function (layer, index) {
if (!layer.isValidToSave()) {
return null;
}
var colorChannels = Object.values(layer.visualChannels).filter(isColorChannel);
var nonColorChannels = Object.values(layer.visualChannels).filter(function (vc) {
return !isColorChannel(vc);
});
return _react["default"].createElement(StyledMapControlLegend, {
className: "legend--layer",
last: index === layers.length - 1,
key: index
}, _react["default"].createElement("div", {
className: "legend--layer_name"
}, layer.config.label), colorChannels.map(function (colorChannel) {
return !colorChannel.condition || colorChannel.condition(layer.config) ? _react["default"].createElement(LayerColorLegend, {
key: colorChannel.key,
description: layer.getVisualChannelDescription(colorChannel.key),
config: layer.config,
width: MAP_LEGEND_WIDTH,
colorChannel: colorChannel
}) : null;
}), nonColorChannels.map(function (visualChannel) {
var matchCondition = !visualChannel.condition || visualChannel.condition(layer.config);
var enabled = layer.config[visualChannel.field] || visualChannel.defaultMeasure;
var description = layer.getVisualChannelDescription(visualChannel.key);
return matchCondition && enabled ? _react["default"].createElement(LayerSizeLegend, {
key: visualChannel.key,
label: description.label,
name: description.measure
}) : null;
}));
}));
};
MapLegend.propTypes = propTypes;
var _default = MapLegend;
exports["default"] = _default;
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/components/map/map-legend.js"],"names":["StyledMapControlLegend","styled","div","props","theme","mapControl","padding","panelBorderColor","last","textColorHl","subtextColor","textColor","VisualChannelMetric","name","LayerSizeLegend","label","propTypes","layers","PropTypes","arrayOf","object","SingleColorDomain","SingleColorLegend","React","memo","width","color","rgb","toString","displayName","MultiColorLegend","colorRange","colorScale","colorDomain","colorField","type","colors","LayerColorLegend","description","config","colorChannel","enableColorBy","measure","scale","field","domain","range","property","key","map","k","visConfig","isColorChannel","visualChannel","CHANNEL_SCALES","colorAggr","includes","channelScaleType","MAP_LEGEND_WIDTH","DIMENSIONS","MapLegend","layer","index","isValidToSave","colorChannels","Object","values","visualChannels","filter","nonColorChannels","vc","length","condition","getVisualChannelDescription","matchCondition","enabled","defaultMeasure"],"mappings":"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","sourcesContent":["// Copyright (c) 2020 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 from 'react';\nimport PropTypes from 'prop-types';\nimport styled from 'styled-components';\nimport {rgb} from 'd3-color';\nimport ColorLegend from 'components/common/color-legend';\nimport {CHANNEL_SCALES, DIMENSIONS} from 'constants/default-settings';\nimport {FormattedMessage} from 'react-intl';\n\nexport const StyledMapControlLegend = styled.div`\n  padding: 10px 0 10px ${props => props.theme.mapControl.padding}px;\n  font-size: 11px;\n  border-bottom-color: ${props => props.theme.panelBorderColor};\n  border-bottom-style: solid;\n  border-bottom-width: ${props => (props.last ? 0 : '1px')};\n\n  .legend--layer_name {\n    font-size: 12px;\n    padding-right: ${props => props.theme.mapControl.padding}px;\n    color: ${props => props.theme.textColorHl};\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__title {\n    padding-right: ${props => props.theme.mapControl.padding}px;\n  }\n\n  .legend--layer_by {\n    color: ${props => props.theme.subtextColor};\n  }\n\n  .legend--layer_color_field {\n    color: ${props => props.theme.textColor};\n    font-weight: 500;\n  }\n\n  .legend--layer_color-legend {\n    margin-top: 6px;\n  }\n`;\n\nexport const VisualChannelMetric = ({name}) => {\n  return (\n    <div className=\"legend--layer__title\">\n      <span className=\"legend--layer_by\">by </span>\n      <span className=\"legend--layer_color_field\">\n        <FormattedMessage id={name} />\n      </span>\n    </div>\n  );\n};\n\nexport const LayerSizeLegend = ({label, name}) => (\n  <div className=\"legend--layer_size-schema\">\n    <p>\n      <span className=\"legend--layer_by\">\n        <FormattedMessage id={label} />\n      </span>\n    </p>\n    <VisualChannelMetric name={name} />\n  </div>\n);\n\nconst propTypes = {\n  layers: PropTypes.arrayOf(PropTypes.object)\n};\n\nconst SingleColorDomain = [''];\nexport const SingleColorLegend = React.memo(({width, color}) => (\n  <ColorLegend\n    scaleType=\"ordinal\"\n    displayLabel={false}\n    domain={SingleColorDomain}\n    fieldType={null}\n    range={[rgb(...color).toString()]}\n    width={width}\n  />\n));\n\nSingleColorLegend.displayName = 'SingleColorLegend';\n\nexport const MultiColorLegend = React.memo(\n  ({colorRange, colorScale, colorDomain, colorField, width}) => (\n    <ColorLegend\n      scaleType={colorScale}\n      displayLabel\n      domain={colorDomain}\n      fieldType={(colorField && colorField.type) || 'real'}\n      range={colorRange.colors}\n      width={width}\n    />\n  )\n);\n\nMultiColorLegend.displayName = 'MultiColorLegend';\n\nexport const LayerColorLegend = React.memo(({description, config, width, colorChannel}) => {\n  const enableColorBy = description.measure;\n  const {scale, field, domain, range, property, key} = colorChannel;\n  const [colorScale, colorField, colorDomain] = [scale, field, domain].map(k => config[k]);\n  const colorRange = config.visConfig[range];\n\n  return (\n    <div>\n      <div className=\"legend--layer_type\">\n        <FormattedMessage id={`layer.${key}`} />\n      </div>\n      <div className=\"legend--layer_color-schema\">\n        <div>\n          {enableColorBy ? <VisualChannelMetric name={enableColorBy} /> : null}\n          <div className=\"legend--layer_color-legend\">\n            {enableColorBy ? (\n              <MultiColorLegend\n                colorScale={colorScale}\n                colorField={colorField}\n                colorDomain={colorDomain}\n                colorRange={colorRange}\n                width={width}\n              />\n            ) : (\n              <SingleColorLegend\n                color={config.visConfig[property] || config[property] || config.color}\n                width={width}\n              />\n            )}\n          </div>\n        </div>\n      </div>\n    </div>\n  );\n});\n\nLayerColorLegend.displayName = 'LayerColorLegend';\n\nconst isColorChannel = visualChannel =>\n  [CHANNEL_SCALES.color, CHANNEL_SCALES.colorAggr].includes(visualChannel.channelScaleType);\n\nconst MAP_LEGEND_WIDTH = DIMENSIONS.mapControl.width - 2 * DIMENSIONS.mapControl.padding;\n\nconst MapLegend = ({layers = []}) => (\n  <div className=\"map-legend\">\n    {layers.map((layer, index) => {\n      if (!layer.isValidToSave()) {\n        return null;\n      }\n\n      const colorChannels = Object.values(layer.visualChannels).filter(isColorChannel);\n      const nonColorChannels = Object.values(layer.visualChannels).filter(\n        vc => !isColorChannel(vc)\n      );\n\n      return (\n        <StyledMapControlLegend\n          className=\"legend--layer\"\n          last={index === layers.length - 1}\n          key={index}\n        >\n          <div className=\"legend--layer_name\">{layer.config.label}</div>\n          {colorChannels.map(colorChannel =>\n            !colorChannel.condition || colorChannel.condition(layer.config) ? (\n              <LayerColorLegend\n                key={colorChannel.key}\n                description={layer.getVisualChannelDescription(colorChannel.key)}\n                config={layer.config}\n                width={MAP_LEGEND_WIDTH}\n                colorChannel={colorChannel}\n              />\n            ) : null\n          )}\n          {nonColorChannels.map(visualChannel => {\n            const matchCondition =\n              !visualChannel.condition || visualChannel.condition(layer.config);\n            const enabled = layer.config[visualChannel.field] || visualChannel.defaultMeasure;\n\n            const description = layer.getVisualChannelDescription(visualChannel.key);\n\n            return matchCondition && enabled ? (\n              <LayerSizeLegend\n                key={visualChannel.key}\n                label={description.label}\n                name={description.measure}\n              />\n            ) : null;\n          })}\n        </StyledMapControlLegend>\n      );\n    })}\n  </div>\n);\n\nMapLegend.propTypes = propTypes;\n\nexport default MapLegend;\n"]}