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
JavaScript
'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"]}