UNPKG

kepler.gl

Version:

kepler.gl is a webgl based application to visualize large scale location data in the browser

306 lines (251 loc) 27 kB
'use strict'; Object.defineProperty(exports, "__esModule", { value: true }); exports.default = undefined; var _extends2 = require('babel-runtime/helpers/extends'); var _extends3 = _interopRequireDefault(_extends2); var _taggedTemplateLiteral2 = require('babel-runtime/helpers/taggedTemplateLiteral'); var _taggedTemplateLiteral3 = _interopRequireDefault(_taggedTemplateLiteral2); var _classCallCheck2 = require('babel-runtime/helpers/classCallCheck'); var _classCallCheck3 = _interopRequireDefault(_classCallCheck2); var _createClass2 = require('babel-runtime/helpers/createClass'); var _createClass3 = _interopRequireDefault(_createClass2); var _possibleConstructorReturn2 = require('babel-runtime/helpers/possibleConstructorReturn'); var _possibleConstructorReturn3 = _interopRequireDefault(_possibleConstructorReturn2); var _inherits2 = require('babel-runtime/helpers/inherits'); var _inherits3 = _interopRequireDefault(_inherits2); var _class, _temp2; // Copyright (c) 2018 Uber Technologies, Inc. // // Permission is hereby granted, free of charge, to any person obtaining a copy // of this software and associated documentation files (the "Software"), to deal // in the Software without restriction, including without limitation the rights // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell // copies of the Software, and to permit persons to whom the Software is // furnished to do so, subject to the following conditions: // // The above copyright notice and this permission notice shall be included in // all copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN // THE SOFTWARE. var _templateObject = (0, _taggedTemplateLiteral3.default)(['\n .rv-xy-plot__inner path {\n fill: none;\n stroke-width: 1.5;\n }\n'], ['\n .rv-xy-plot__inner path {\n fill: none;\n stroke-width: 1.5;\n }\n']), _templateObject2 = (0, _taggedTemplateLiteral3.default)(['\n background-color: #d3d8e0;\n border-radius: 2px;\n color: ', ';\n font-size: 9px;\n margin: 4px;\n padding: 3px 6px;\n pointer-events: none;\n user-select: none;\n'], ['\n background-color: #d3d8e0;\n border-radius: 2px;\n color: ', ';\n font-size: 9px;\n margin: 4px;\n padding: 3px 6px;\n pointer-events: none;\n user-select: none;\n']); var _react = require('react'); var _react2 = _interopRequireDefault(_react); var _propTypes = require('prop-types'); var _propTypes2 = _interopRequireDefault(_propTypes); var _d3Scale = require('d3-scale'); var _moment = require('moment'); var _moment2 = _interopRequireDefault(_moment); var _d3Array = require('d3-array'); var _reselect = require('reselect'); var _reactVis = require('react-vis'); var _styledComponents = require('styled-components'); var _styledComponents2 = _interopRequireDefault(_styledComponents); var _rangeBrush = require('./range-brush'); var _rangeBrush2 = _interopRequireDefault(_rangeBrush); var _filterUtils = require('../../utils/filter-utils'); var _base = require('../../styles/base'); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } var chartMargin = { top: 18, bottom: 0, left: 0, right: 0 }; var chartH = 52; var containerH = 78; var histogramStyle = { highlightW: 0.7, unHighlightedW: 0.4, highlightedColor: _base.theme.activeColor, unHighlightedColor: _base.theme.sliderBarColor }; var RangePlot = (_temp2 = _class = function (_Component) { (0, _inherits3.default)(RangePlot, _Component); function RangePlot() { var _ref; var _temp, _this, _ret; (0, _classCallCheck3.default)(this, RangePlot); for (var _len = arguments.length, args = Array(_len), _key = 0; _key < _len; _key++) { args[_key] = arguments[_key]; } return _ret = (_temp = (_this = (0, _possibleConstructorReturn3.default)(this, (_ref = RangePlot.__proto__ || Object.getPrototypeOf(RangePlot)).call.apply(_ref, [this].concat(args))), _this), _this.state = { hoveredDP: null }, _this.domainSelector = function (props) { return props.lineChart && props.lineChart.xDomain; }, _this.hintFormatter = (0, _reselect.createSelector)(_this.domainSelector, function (domain) { return (0, _filterUtils.getTimeWidgetHintFormatter)(domain); }), _this.onMouseMove = function (hoveredDP) { _this.setState({ hoveredDP: hoveredDP }); }, _temp), (0, _possibleConstructorReturn3.default)(_this, _ret); } (0, _createClass3.default)(RangePlot, [{ key: 'render', value: function render() { var _props = this.props, onBrush = _props.onBrush, range = _props.range, value = _props.value, width = _props.width, plotType = _props.plotType, lineChart = _props.lineChart, histogram = _props.histogram; var domain = [histogram[0].x0, histogram[histogram.length - 1].x1]; var brushComponent = _react2.default.createElement(_rangeBrush2.default, { domain: domain, onBrush: onBrush, range: range, value: value, width: width }); return _react2.default.createElement( 'div', { style: { height: containerH + 'px', position: 'relative' } }, plotType === 'lineChart' ? _react2.default.createElement(LineChart, { hoveredDP: this.state.hoveredDP, width: width, height: containerH, margin: chartMargin, children: brushComponent, onMouseMove: this.onMouseMove, yDomain: lineChart.yDomain, hintFormat: this.hintFormatter(this.props), data: lineChart.series }) : _react2.default.createElement(Histogram, { width: width, height: chartH, value: value, margin: chartMargin, histogram: histogram, brushComponent: brushComponent }) ); } }]); return RangePlot; }(_react.Component), _class.propTypes = { value: _propTypes2.default.arrayOf(_propTypes2.default.number).isRequired, histogram: _propTypes2.default.arrayOf(_propTypes2.default.shape({ x0: _propTypes2.default.number, x1: _propTypes2.default.number })), lineChart: _propTypes2.default.object, plotType: _propTypes2.default.string, isEnlarged: _propTypes2.default.bool, onBlur: _propTypes2.default.func, width: _propTypes2.default.number.isRequired }, _temp2); exports.default = RangePlot; var Histogram = function Histogram(_ref2) { var width = _ref2.width, height = _ref2.height, margin = _ref2.margin, histogram = _ref2.histogram, value = _ref2.value, brushComponent = _ref2.brushComponent; var domain = [histogram[0].x0, histogram[histogram.length - 1].x1]; var barWidth = width / histogram.length; var x = (0, _d3Scale.scaleLinear)().domain(domain).range([0, width]); var y = (0, _d3Scale.scaleLinear)().domain([0, (0, _d3Array.max)(histogram, function (d) { return d.count; })]).range([0, height]); return _react2.default.createElement( 'svg', { width: width, height: height, style: { marginTop: margin.top + 'px' } }, _react2.default.createElement( 'g', { className: 'histogram-bars' }, histogram.map(function (bar) { var inRange = bar.x0 >= value[0] && bar.x1 <= value[1]; var fill = inRange ? histogramStyle.highlightedColor : histogramStyle.unHighlightedColor; var wRatio = inRange ? histogramStyle.highlightW : histogramStyle.unHighlightedW; return _react2.default.createElement('rect', { key: bar.x0, fill: fill, height: y(bar.count), width: barWidth * wRatio, x: x(bar.x0) + barWidth * (1 - wRatio) / 2, rx: 1, ry: 1, y: height - y(bar.count) }); }) ), brushComponent ); }; var LineChartWrapper = _styledComponents2.default.div(_templateObject); var LineChart = function LineChart(_ref3) { var width = _ref3.width, height = _ref3.height, yDomain = _ref3.yDomain, hintFormat = _ref3.hintFormat, hoveredDP = _ref3.hoveredDP, margin = _ref3.margin, color = _ref3.color, data = _ref3.data, onMouseMove = _ref3.onMouseMove, children = _ref3.children; var brushData = [{ x: data[0].x, y: yDomain[1], customComponent: function customComponent() { return children; } }]; return _react2.default.createElement( LineChartWrapper, null, _react2.default.createElement( _reactVis.XYPlot, { width: width, height: height, margin: (0, _extends3.default)({}, margin, { bottom: 12 }) }, _react2.default.createElement(_reactVis.LineSeries, { strokeWidth: 2, color: color, data: data, onNearestX: onMouseMove }), _react2.default.createElement(_reactVis.MarkSeries, { data: hoveredDP ? [hoveredDP] : [], color: color, size: 3 }), _react2.default.createElement(_reactVis.CustomSVGSeries, { data: brushData }), hoveredDP ? _react2.default.createElement( _reactVis.Hint, { value: hoveredDP }, _react2.default.createElement(HintContent, (0, _extends3.default)({}, hoveredDP, { format: function format(val) { return _moment2.default.utc(val).format(hintFormat); } })) ) : null ) ); }; var StyledHint = _styledComponents2.default.div(_templateObject2, function (props) { return props.theme.textColorLT; }); var HintContent = function HintContent(_ref4) { var x = _ref4.x, y = _ref4.y, format = _ref4.format; return _react2.default.createElement( StyledHint, null, _react2.default.createElement( 'div', { className: 'hint--x' }, format(x) ), _react2.default.createElement( 'div', { className: 'row' }, y ) ); }; //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/components/common/range-plot.js"],"names":["chartMargin","top","bottom","left","right","chartH","containerH","histogramStyle","highlightW","unHighlightedW","highlightedColor","theme","activeColor","unHighlightedColor","sliderBarColor","RangePlot","state","hoveredDP","domainSelector","props","lineChart","xDomain","hintFormatter","domain","onMouseMove","setState","onBrush","range","value","width","plotType","histogram","x0","length","x1","brushComponent","height","position","yDomain","series","Component","propTypes","PropTypes","arrayOf","number","isRequired","shape","object","string","isEnlarged","bool","onBlur","func","Histogram","margin","barWidth","x","y","d","count","marginTop","map","inRange","bar","fill","wRatio","LineChartWrapper","styled","div","LineChart","hintFormat","color","data","children","brushData","customComponent","moment","utc","val","format","StyledHint","textColorLT","HintContent"],"mappings":"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","file":"range-plot.js","sourcesContent":["// Copyright (c) 2018 Uber Technologies, Inc.\n//\n// Permission is hereby granted, free of charge, to any person obtaining a copy\n// of this software and associated documentation files (the \"Software\"), to deal\n// in the Software without restriction, including without limitation the rights\n// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell\n// copies of the Software, and to permit persons to whom the Software is\n// furnished to do so, subject to the following conditions:\n//\n// The above copyright notice and this permission notice shall be included in\n// all copies or substantial portions of the Software.\n//\n// THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\n// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\n// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN\n// THE SOFTWARE.\n\nimport React, {Component} from 'react';\nimport PropTypes from 'prop-types';\nimport {scaleLinear} from 'd3-scale';\nimport moment from 'moment';\nimport {max} from 'd3-array';\nimport {createSelector} from 'reselect';\nimport {LineSeries, XYPlot, CustomSVGSeries, Hint, MarkSeries} from 'react-vis';\nimport styled from 'styled-components';\nimport RangeBrush from './range-brush';\nimport {getTimeWidgetHintFormatter} from 'utils/filter-utils';\nimport {theme} from 'styles/base';\n\nconst chartMargin = {top: 18, bottom: 0, left: 0, right: 0};\nconst chartH = 52;\nconst containerH = 78;\nconst histogramStyle = {\n  highlightW: 0.7,\n  unHighlightedW: 0.4,\n  highlightedColor: theme.activeColor,\n  unHighlightedColor: theme.sliderBarColor\n};\n\nexport default class RangePlot extends Component {\n  static propTypes = {\n    value: PropTypes.arrayOf(PropTypes.number).isRequired,\n    histogram: PropTypes.arrayOf(\n      PropTypes.shape({\n        x0: PropTypes.number,\n        x1: PropTypes.number\n      })\n    ),\n    lineChart: PropTypes.object,\n    plotType: PropTypes.string,\n    isEnlarged: PropTypes.bool,\n    onBlur: PropTypes.func,\n    width: PropTypes.number.isRequired\n  };\n\n  state = {\n    hoveredDP: null\n  };\n\n  domainSelector = props => props.lineChart && props.lineChart.xDomain;\n  hintFormatter = createSelector(this.domainSelector, domain =>\n    getTimeWidgetHintFormatter(domain)\n  );\n\n  onMouseMove = hoveredDP => {\n    this.setState({hoveredDP});\n  };\n\n  render() {\n    const {\n      onBrush,\n      range,\n      value,\n      width,\n      plotType,\n      lineChart,\n      histogram\n    } = this.props;\n    const domain = [histogram[0].x0, histogram[histogram.length - 1].x1];\n\n    const brushComponent = (\n      <RangeBrush\n        domain={domain}\n        onBrush={onBrush}\n        range={range}\n        value={value}\n        width={width}\n      />\n    );\n\n    return (\n      <div\n        style={{\n          height: `${containerH}px`,\n          position: 'relative'\n        }}\n      >\n        {plotType === 'lineChart' ? (\n          <LineChart\n            hoveredDP={this.state.hoveredDP}\n            width={width}\n            height={containerH}\n            margin={chartMargin}\n            children={brushComponent}\n            onMouseMove={this.onMouseMove}\n            yDomain={lineChart.yDomain}\n            hintFormat={this.hintFormatter(this.props)}\n            data={lineChart.series}\n          />\n        ) : (\n          <Histogram\n            width={width}\n            height={chartH}\n            value={value}\n            margin={chartMargin}\n            histogram={histogram}\n            brushComponent={brushComponent}\n          />\n        )}\n      </div>\n    );\n  }\n}\n\nconst Histogram = ({\n  width,\n  height,\n  margin,\n  histogram,\n  value,\n  brushComponent\n}) => {\n  const domain = [histogram[0].x0, histogram[histogram.length - 1].x1];\n  const barWidth = width / histogram.length;\n\n  const x = scaleLinear()\n    .domain(domain)\n    .range([0, width]);\n\n  const y = scaleLinear()\n    .domain([0, max(histogram, d => d.count)])\n    .range([0, height]);\n\n  return (\n    <svg width={width} height={height} style={{marginTop: `${margin.top}px`}}>\n      <g className=\"histogram-bars\">\n        {histogram.map(bar => {\n          const inRange = bar.x0 >= value[0] && bar.x1 <= value[1];\n          const fill = inRange ? histogramStyle.highlightedColor : histogramStyle.unHighlightedColor;\n          const wRatio = inRange ? histogramStyle.highlightW : histogramStyle.unHighlightedW;\n\n          return (\n            <rect\n              key={bar.x0}\n              fill={fill}\n              height={y(bar.count)}\n              width={barWidth * wRatio}\n              x={x(bar.x0) + barWidth * (1 - wRatio) / 2}\n              rx={1}\n              ry={1}\n              y={height - y(bar.count)}\n            />\n          );\n        })}\n      </g>\n      {brushComponent}\n    </svg>\n  );\n};\n\nconst LineChartWrapper = styled.div`\n  .rv-xy-plot__inner path {\n    fill: none;\n    stroke-width: 1.5;\n  }\n`;\n\nconst LineChart = ({\n  width,\n  height,\n  yDomain,\n  hintFormat,\n  hoveredDP,\n  margin,\n  color,\n  data,\n  onMouseMove,\n  children\n}) => {\n  const brushData = [\n    {x: data[0].x, y: yDomain[1], customComponent: () => children}\n  ];\n\n  return (\n    <LineChartWrapper>\n      <XYPlot width={width} height={height} margin={{...margin, bottom: 12}}>\n        <LineSeries\n          strokeWidth={2}\n          color={color}\n          data={data}\n          onNearestX={onMouseMove}\n        />\n        <MarkSeries\n          data={hoveredDP ? [hoveredDP] : []}\n          color={color}\n          size={3}\n        />\n        <CustomSVGSeries data={brushData} />\n        {hoveredDP ? (\n          <Hint value={hoveredDP}>\n            <HintContent\n              {...hoveredDP}\n              format={val => moment.utc(val).format(hintFormat)}\n            />\n          </Hint>\n        ) : null}\n      </XYPlot>\n    </LineChartWrapper>\n  );\n};\n\nconst StyledHint = styled.div`\n  background-color: #d3d8e0;\n  border-radius: 2px;\n  color: ${props => props.theme.textColorLT};\n  font-size: 9px;\n  margin: 4px;\n  padding: 3px 6px;\n  pointer-events: none;\n  user-select: none;\n`;\nconst HintContent = ({x, y, format}) => (\n  <StyledHint>\n    <div className=\"hint--x\">{format(x)}</div>\n    <div className=\"row\">{y}</div>\n  </StyledHint>\n);\n"]}