kepler.gl
Version:
kepler.gl is a webgl based application to visualize large scale location data in the browser
249 lines (198 loc) • 25.2 kB
JavaScript
;
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
var _interopRequireWildcard = require("@babel/runtime/helpers/interopRequireWildcard");
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.LegendRow = exports["default"] = void 0;
var _classCallCheck2 = _interopRequireDefault(require("@babel/runtime/helpers/classCallCheck"));
var _createClass2 = _interopRequireDefault(require("@babel/runtime/helpers/createClass"));
var _assertThisInitialized2 = _interopRequireDefault(require("@babel/runtime/helpers/assertThisInitialized"));
var _inherits2 = _interopRequireDefault(require("@babel/runtime/helpers/inherits"));
var _possibleConstructorReturn2 = _interopRequireDefault(require("@babel/runtime/helpers/possibleConstructorReturn"));
var _getPrototypeOf2 = _interopRequireDefault(require("@babel/runtime/helpers/getPrototypeOf"));
var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty"));
var _taggedTemplateLiteral2 = _interopRequireDefault(require("@babel/runtime/helpers/taggedTemplateLiteral"));
var _react = _interopRequireWildcard(require("react"));
var _propTypes = _interopRequireDefault(require("prop-types"));
var _styledComponents = _interopRequireDefault(require("styled-components"));
var _reselect = require("reselect");
var _d3Format = require("d3-format");
var _moment = _interopRequireDefault(require("moment"));
var _defaultSettings = require("../../constants/default-settings");
var _filterUtils = require("../../utils/filter-utils");
var _utils = require("../../utils/utils");
var _templateObject;
function _createSuper(Derived) { var hasNativeReflectConstruct = _isNativeReflectConstruct(); return function _createSuperInternal() { var Super = (0, _getPrototypeOf2["default"])(Derived), result; if (hasNativeReflectConstruct) { var NewTarget = (0, _getPrototypeOf2["default"])(this).constructor; result = Reflect.construct(Super, arguments, NewTarget); } else { result = Super.apply(this, arguments); } return (0, _possibleConstructorReturn2["default"])(this, result); }; }
function _isNativeReflectConstruct() { if (typeof Reflect === "undefined" || !Reflect.construct) return false; if (Reflect.construct.sham) return false; if (typeof Proxy === "function") return true; try { Boolean.prototype.valueOf.call(Reflect.construct(Boolean, [], function () {})); return true; } catch (e) { return false; } }
var ROW_H = 10;
var GAP = 4;
var RECT_W = 20;
var StyledLegend = _styledComponents["default"].div(_templateObject || (_templateObject = (0, _taggedTemplateLiteral2["default"])(["\n ", ";\n\n max-height: 150px;\n overflow-y: auto;\n\n svg {\n text {\n font-size: 9px;\n fill: ", ";\n }\n }\n"])), function (props) {
return props.theme.sidePanelScrollBar;
}, function (props) {
return props.theme.textColor;
});
var defaultFormat = function defaultFormat(d) {
return d;
};
var getTimeLabelFormat = function getTimeLabelFormat(domain) {
var formatter = (0, _filterUtils.getTimeWidgetHintFormatter)(domain);
return function (val) {
return _moment["default"].utc(val).format(formatter);
};
};
var getNumericLabelFormat = function getNumericLabelFormat(domain) {
var diff = domain[1] - domain[0];
if (diff < 10) {
return (0, _d3Format.format)('.2f');
}
return (0, _d3Format.format)('.1f');
};
var getQuantLabelFormat = function getQuantLabelFormat(domain, fieldType) {
// quant scale can only be assigned to linear Fields: real, timestamp, integer
return fieldType === _defaultSettings.ALL_FIELD_TYPES.timestamp ? getTimeLabelFormat(domain) : !fieldType ? defaultFormat : getNumericLabelFormat(domain);
};
var getOrdinalLegends = function getOrdinalLegends(scale) {
var domain = scale.domain();
return {
data: domain.map(scale),
labels: domain
};
};
var getQuantLegends = function getQuantLegends(scale, labelFormat) {
if (typeof scale.invertExtent !== 'function') {
// only quantile, quantize, threshold scale has invertExtent method
return {
data: [],
labels: []
};
}
var labels = scale.range().map(function (d) {
var invert = scale.invertExtent(d);
return "".concat(labelFormat(invert[0]), " to ").concat(labelFormat(invert[1]));
});
return {
data: scale.range(),
labels: labels
};
};
var ColorLegend = /*#__PURE__*/function (_Component) {
(0, _inherits2["default"])(ColorLegend, _Component);
var _super = _createSuper(ColorLegend);
function ColorLegend() {
var _this;
(0, _classCallCheck2["default"])(this, ColorLegend);
for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) {
args[_key] = arguments[_key];
}
_this = _super.call.apply(_super, [this].concat(args));
(0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "domainSelector", function (props) {
return props.domain;
});
(0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "rangeSelector", function (props) {
return props.range;
});
(0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "labelFormatSelector", function (props) {
return props.labelFormat;
});
(0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "scaleTypeSelector", function (props) {
return props.scaleType;
});
(0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "fieldTypeSelector", function (props) {
return props.fieldType;
});
(0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "legendsSelector", (0, _reselect.createSelector)(_this.domainSelector, _this.rangeSelector, _this.scaleTypeSelector, _this.labelFormatSelector, _this.fieldTypeSelector, function (domain, range, scaleType, labelFormat, fieldType) {
var empty = {
data: [],
labels: []
};
if (!range) {
return empty;
}
if ((0, _utils.isObject)(range.colorLegends)) {
return {
data: Object.keys(range.colorLegends),
labels: Object.values(range.colorLegends)
};
} else if (Array.isArray(range.colorMap)) {
return {
data: range.colorMap.map(function (cm) {
return cm[1];
}),
labels: range.colorMap.map(function (cm) {
return cm[0];
})
};
} else if (Array.isArray(range.colors)) {
if (!domain || !scaleType) {
return empty;
}
var scaleFunction = _defaultSettings.SCALE_FUNC[scaleType]; // color scale can only be quantize, quantile or ordinal
// @ts-ignore fix d3 scale
var scale = scaleFunction().domain(domain).range(range.colors);
if (scaleType === _defaultSettings.SCALE_TYPES.ordinal) {
return getOrdinalLegends(scale);
}
var formatLabel = labelFormat || getQuantLabelFormat(scale.domain(), fieldType);
return getQuantLegends(scale, formatLabel);
}
return empty;
}));
return _this;
}
(0, _createClass2["default"])(ColorLegend, [{
key: "render",
value: function render() {
var _this$props = this.props,
width = _this$props.width,
_this$props$displayLa = _this$props.displayLabel,
displayLabel = _this$props$displayLa === void 0 ? true : _this$props$displayLa;
var legends = this.legendsSelector(this.props);
var height = legends.data.length * (ROW_H + GAP);
return /*#__PURE__*/_react["default"].createElement(StyledLegend, null, /*#__PURE__*/_react["default"].createElement("svg", {
width: width,
height: height
}, legends.data.map(function (color, idx) {
return /*#__PURE__*/_react["default"].createElement(LegendRow, {
key: idx,
label: legends.labels[idx],
displayLabel: displayLabel,
color: color,
idx: idx
});
})));
}
}]);
return ColorLegend;
}(_react.Component);
exports["default"] = ColorLegend;
(0, _defineProperty2["default"])(ColorLegend, "propTypes", {
width: _propTypes["default"].number.isRequired,
scaleType: _propTypes["default"].string,
domain: _propTypes["default"].oneOfType([_propTypes["default"].array, _propTypes["default"].object]),
fieldType: _propTypes["default"].string,
range: _propTypes["default"].object,
labelFormat: _propTypes["default"].func
});
var LegendRow = function LegendRow(_ref) {
var _ref$label = _ref.label,
label = _ref$label === void 0 ? '' : _ref$label,
displayLabel = _ref.displayLabel,
color = _ref.color,
idx = _ref.idx;
return /*#__PURE__*/_react["default"].createElement("g", {
transform: "translate(0, ".concat(idx * (ROW_H + GAP), ")")
}, /*#__PURE__*/_react["default"].createElement("rect", {
width: RECT_W,
height: ROW_H,
style: {
fill: color
}
}), /*#__PURE__*/_react["default"].createElement("text", {
x: RECT_W + 8,
y: ROW_H - 1
}, displayLabel ? label.toString() : ''));
};
exports.LegendRow = LegendRow;
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/components/common/color-legend.js"],"names":["ROW_H","GAP","RECT_W","StyledLegend","styled","div","props","theme","sidePanelScrollBar","textColor","defaultFormat","d","getTimeLabelFormat","domain","formatter","val","moment","utc","format","getNumericLabelFormat","diff","getQuantLabelFormat","fieldType","ALL_FIELD_TYPES","timestamp","getOrdinalLegends","scale","data","map","labels","getQuantLegends","labelFormat","invertExtent","range","invert","ColorLegend","scaleType","domainSelector","rangeSelector","scaleTypeSelector","labelFormatSelector","fieldTypeSelector","empty","colorLegends","Object","keys","values","Array","isArray","colorMap","cm","colors","scaleFunction","SCALE_FUNC","SCALE_TYPES","ordinal","formatLabel","width","displayLabel","legends","legendsSelector","height","length","color","idx","Component","PropTypes","number","isRequired","string","oneOfType","array","object","func","LegendRow","label","fill","toString"],"mappings":"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","sourcesContent":["// Copyright (c) 2021 Uber Technologies, Inc.\n//\n// Permission is hereby granted, free of charge, to any person obtaining a copy\n// of this software and associated documentation files (the \"Software\"), to deal\n// in the Software without restriction, including without limitation the rights\n// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell\n// copies of the Software, and to permit persons to whom the Software is\n// furnished to do so, subject to the following conditions:\n//\n// The above copyright notice and this permission notice shall be included in\n// all copies or substantial portions of the Software.\n//\n// THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\n// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\n// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN\n// THE SOFTWARE.\n\nimport React, {Component} from 'react';\nimport PropTypes from 'prop-types';\nimport styled from 'styled-components';\nimport {createSelector} from 'reselect';\nimport {format} from 'd3-format';\nimport moment from 'moment';\nimport {SCALE_TYPES, SCALE_FUNC, ALL_FIELD_TYPES} from 'constants/default-settings';\nimport {getTimeWidgetHintFormatter} from 'utils/filter-utils';\nimport {isObject} from 'utils/utils';\n\nconst ROW_H = 10;\nconst GAP = 4;\nconst RECT_W = 20;\n\nconst StyledLegend = styled.div`\n  ${props => props.theme.sidePanelScrollBar};\n\n  max-height: 150px;\n  overflow-y: auto;\n\n  svg {\n    text {\n      font-size: 9px;\n      fill: ${props => props.theme.textColor};\n    }\n  }\n`;\n\nconst defaultFormat = d => d;\n\nconst getTimeLabelFormat = domain => {\n  const formatter = getTimeWidgetHintFormatter(domain);\n  return val => moment.utc(val).format(formatter);\n};\n\nconst getNumericLabelFormat = domain => {\n  const diff = domain[1] - domain[0];\n\n  if (diff < 10) {\n    return format('.2f');\n  }\n\n  return format('.1f');\n};\n\nconst getQuantLabelFormat = (domain, fieldType) => {\n  // quant scale can only be assigned to linear Fields: real, timestamp, integer\n  return fieldType === ALL_FIELD_TYPES.timestamp\n    ? getTimeLabelFormat(domain)\n    : !fieldType\n    ? defaultFormat\n    : getNumericLabelFormat(domain);\n};\n\nconst getOrdinalLegends = scale => {\n  const domain = scale.domain();\n  return {\n    data: domain.map(scale),\n    labels: domain\n  };\n};\n\nconst getQuantLegends = (scale, labelFormat) => {\n  if (typeof scale.invertExtent !== 'function') {\n    // only quantile, quantize, threshold scale has invertExtent method\n    return {\n      data: [],\n      labels: []\n    };\n  }\n\n  const labels = scale.range().map(d => {\n    const invert = scale.invertExtent(d);\n    return `${labelFormat(invert[0])} to ${labelFormat(invert[1])}`;\n  });\n\n  return {\n    data: scale.range(),\n    labels\n  };\n};\n\nexport default class ColorLegend extends Component {\n  static propTypes = {\n    width: PropTypes.number.isRequired,\n    scaleType: PropTypes.string,\n    domain: PropTypes.oneOfType([PropTypes.array, PropTypes.object]),\n    fieldType: PropTypes.string,\n    range: PropTypes.object,\n    labelFormat: PropTypes.func\n  };\n\n  domainSelector = props => props.domain;\n  rangeSelector = props => props.range;\n  labelFormatSelector = props => props.labelFormat;\n  scaleTypeSelector = props => props.scaleType;\n  fieldTypeSelector = props => props.fieldType;\n\n  legendsSelector = createSelector(\n    this.domainSelector,\n    this.rangeSelector,\n    this.scaleTypeSelector,\n    this.labelFormatSelector,\n    this.fieldTypeSelector,\n    (domain, range, scaleType, labelFormat, fieldType) => {\n      const empty = {\n        data: [],\n        labels: []\n      };\n      if (!range) {\n        return empty;\n      }\n      if (isObject(range.colorLegends)) {\n        return {\n          data: Object.keys(range.colorLegends),\n          labels: Object.values(range.colorLegends)\n        };\n      } else if (Array.isArray(range.colorMap)) {\n        return {\n          data: range.colorMap.map(cm => cm[1]),\n          labels: range.colorMap.map(cm => cm[0])\n        };\n      } else if (Array.isArray(range.colors)) {\n        if (!domain || !scaleType) {\n          return empty;\n        }\n\n        const scaleFunction = SCALE_FUNC[scaleType];\n        // color scale can only be quantize, quantile or ordinal\n        // @ts-ignore fix d3 scale\n        const scale = scaleFunction()\n          .domain(domain)\n          .range(range.colors);\n\n        if (scaleType === SCALE_TYPES.ordinal) {\n          return getOrdinalLegends(scale);\n        }\n\n        const formatLabel = labelFormat || getQuantLabelFormat(scale.domain(), fieldType);\n\n        return getQuantLegends(scale, formatLabel);\n      }\n      return empty;\n    }\n  );\n\n  render() {\n    const {width, displayLabel = true} = this.props;\n\n    const legends = this.legendsSelector(this.props);\n    const height = legends.data.length * (ROW_H + GAP);\n\n    return (\n      <StyledLegend>\n        <svg width={width} height={height}>\n          {legends.data.map((color, idx) => (\n            <LegendRow\n              key={idx}\n              label={legends.labels[idx]}\n              displayLabel={displayLabel}\n              color={color}\n              idx={idx}\n            />\n          ))}\n        </svg>\n      </StyledLegend>\n    );\n  }\n}\n\nexport const LegendRow = ({label = '', displayLabel, color, idx}) => (\n  <g transform={`translate(0, ${idx * (ROW_H + GAP)})`}>\n    <rect width={RECT_W} height={ROW_H} style={{fill: color}} />\n    <text x={RECT_W + 8} y={ROW_H - 1}>\n      {displayLabel ? label.toString() : ''}\n    </text>\n  </g>\n);\n"]}