kepler.gl
Version:
kepler.gl is a webgl based application to visualize large scale location data in the browser
352 lines (288 loc) • 32.9 kB
JavaScript
'use strict';
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.default = undefined;
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 _taggedTemplateLiteral2 = require('babel-runtime/helpers/taggedTemplateLiteral');
var _taggedTemplateLiteral3 = _interopRequireDefault(_taggedTemplateLiteral2);
var _class, _temp, _initialiseProps;
var _templateObject = (0, _taggedTemplateLiteral3.default)(['\n margin-top: ', ';\n align-items: flex-end;\n display: flex;\n flex-direction: row;\n justify-content: space-between;\n'], ['\n margin-top: ', ';\n align-items: flex-end;\n display: flex;\n flex-direction: row;\n justify-content: space-between;\n']),
_templateObject2 = (0, _taggedTemplateLiteral3.default)(['\n display: flex;\n height: ', ';\n align-items: center;\n font-size: 11px;\n justify-content: ', ';\n color: ', ';\n\n .horizontal-bar {\n padding: 0 12px;\n }\n\n .time-value {\n display: flex;\n flex-direction: ', ';\n align-items: flex-start;\n }\n\n .time-value:last-child {\n align-items: flex-end;\n }\n'], ['\n display: flex;\n height: ', ';\n align-items: center;\n font-size: 11px;\n justify-content: ', ';\n color: ', ';\n\n .horizontal-bar {\n padding: 0 12px;\n }\n\n .time-value {\n display: flex;\n flex-direction: ', ';\n align-items: flex-start;\n }\n\n .time-value:last-child {\n align-items: flex-end;\n }\n']),
_templateObject3 = (0, _taggedTemplateLiteral3.default)(['\n margin-bottom: 12px;\n margin-right: 42px;\n\n &.disabled {\n opacity: 0.4;\n pointer-events: none;\n }\n'], ['\n margin-bottom: 12px;\n margin-right: 42px;\n\n &.disabled {\n opacity: 0.4;\n pointer-events: none;\n }\n']),
_templateObject4 = (0, _taggedTemplateLiteral3.default)(['\n svg {\n margin: 0 6px;\n }\n'], ['\n svg {\n margin: 0 6px;\n }\n']); // 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 _react = require('react');
var _react2 = _interopRequireDefault(_react);
var _propTypes = require('prop-types');
var _propTypes2 = _interopRequireDefault(_propTypes);
var _moment = require('moment');
var _moment2 = _interopRequireDefault(_moment);
var _window = require('global/window');
var _classnames = require('classnames');
var _classnames2 = _interopRequireDefault(_classnames);
var _lodash = require('lodash.throttle');
var _lodash2 = _interopRequireDefault(_lodash);
var _styledComponents = require('styled-components');
var _styledComponents2 = _interopRequireDefault(_styledComponents);
var _reselect = require('reselect');
var _icons = require('./icons');
var _styledComponents3 = require('./styled-components');
var _filterUtils = require('../../utils/filter-utils');
var _rangeSlider = require('./range-slider');
var _rangeSlider2 = _interopRequireDefault(_rangeSlider);
var _timeSliderMarker = require('./time-slider-marker');
var _timeSliderMarker2 = _interopRequireDefault(_timeSliderMarker);
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
var defaultTimeFormat = function defaultTimeFormat(val) {
return _moment2.default.utc(val).format('MM/DD/YY hh:mma');
};
var animationControlWidth = 140;
var StyledSliderContainer = _styledComponents2.default.div(_templateObject, function (props) {
return props.isEnlarged ? '12px' : '0px';
});
var TimeRangeSlider = (_temp = _class = function (_Component) {
(0, _inherits3.default)(TimeRangeSlider, _Component);
function TimeRangeSlider(props) {
(0, _classCallCheck3.default)(this, TimeRangeSlider);
var _this = (0, _possibleConstructorReturn3.default)(this, (TimeRangeSlider.__proto__ || Object.getPrototypeOf(TimeRangeSlider)).call(this, props));
_initialiseProps.call(_this);
_this.state = {
isAnimating: false,
width: 288
};
_this._animation = null;
_this._sliderThrottle = (0, _lodash2.default)(function () {
var _this$props;
return (_this$props = _this.props).onChange.apply(_this$props, arguments);
}, 20);
return _this;
}
(0, _createClass3.default)(TimeRangeSlider, [{
key: 'componentDidUpdate',
value: function componentDidUpdate() {
if (!this._animation && this.state.isAnimating) {
this._animation = (0, _window.requestAnimationFrame)(this._nextFrame);
}
}
}, {
key: 'render',
value: function render() {
var _props = this.props,
domain = _props.domain,
value = _props.value,
isEnlarged = _props.isEnlarged;
var isAnimating = this.state.isAnimating;
return _react2.default.createElement(
'div',
{ className: 'time-range-slider' },
_react2.default.createElement(TimeTitle, {
timeFormat: this.titleFormatter(this.props),
value: value,
isEnlarged: isEnlarged
}),
_react2.default.createElement(
StyledSliderContainer,
{
className: 'time-range-slider__container',
isEnlarged: isEnlarged },
isEnlarged ? _react2.default.createElement(AnimationControls, {
isAnimatable: this.props.isAnimatable,
isEnlarged: isEnlarged,
isAnimating: isAnimating,
pauseAnimation: this._pauseAnimation,
resetAnimation: this._resetAnimation,
startAnimation: this._startAnimation
}) : null,
_react2.default.createElement(
'div',
{ style: { width: isEnlarged ? 'calc(100% - ' + animationControlWidth + 'px)' : '100%' } },
_react2.default.createElement(_rangeSlider2.default, {
range: domain,
value0: value[0],
value1: value[1],
histogram: this.props.histogram,
lineChart: this.props.lineChart,
plotType: this.props.plotType,
isEnlarged: isEnlarged,
showInput: false,
step: this.props.step,
onChange: this._sliderUpdate,
xAxis: _timeSliderMarker2.default
})
)
)
);
}
}]);
return TimeRangeSlider;
}(_react.Component), _class.propTypes = {
onChange: _propTypes2.default.func.isRequired,
domain: _propTypes2.default.arrayOf(_propTypes2.default.number).isRequired,
value: _propTypes2.default.arrayOf(_propTypes2.default.number).isRequired,
step: _propTypes2.default.number.isRequired,
plotType: _propTypes2.default.string,
histogram: _propTypes2.default.arrayOf(_propTypes2.default.any),
lineChart: _propTypes2.default.object,
toggleAnimation: _propTypes2.default.func.isRequired,
isAnimatable: _propTypes2.default.bool,
isEnlarged: _propTypes2.default.bool,
speed: _propTypes2.default.number
}, _initialiseProps = function _initialiseProps() {
var _this2 = this;
this.domainSelector = function (props) {
return props.domain;
};
this.titleFormatter = (0, _reselect.createSelector)(this.domainSelector, function (domain) {
return (0, _filterUtils.getTimeWidgetTitleFormatter)(domain);
});
this._sliderUpdate = function (args) {
_this2._sliderThrottle.cancel();
_this2._sliderThrottle(args);
};
this._resetAnimation = function () {
var _props2 = _this2.props,
domain = _props2.domain,
value = _props2.value;
var value0 = domain[0];
var value1 = value0 + value[1] - value[0];
_this2.props.onChange([value0, value1]);
};
this._startAnimation = function () {
_this2._pauseAnimation();
_this2.props.toggleAnimation();
_this2.setState({ isAnimating: true });
};
this._pauseAnimation = function () {
if (_this2._animation) {
(0, _window.cancelAnimationFrame)(_this2._animation);
_this2.props.toggleAnimation();
_this2._animation = null;
}
_this2.setState({ isAnimating: false });
};
this._nextFrame = function () {
_this2._animation = null;
var _props3 = _this2.props,
domain = _props3.domain,
value = _props3.value;
var speed = (domain[1] - domain[0]) / _filterUtils.BASE_SPEED * _this2.props.speed;
// loop when reaches the end
var value0 = value[1] + speed > domain[1] ? domain[0] : value[0] + speed;
var value1 = value0 + value[1] - value[0];
_this2.props.onChange([value0, value1]);
};
}, _temp);
exports.default = TimeRangeSlider;
var TimeValueWrapper = _styledComponents2.default.div(_templateObject2, function (props) {
return props.theme.secondaryInputHeight;
}, function (props) {
return props.isEnlarged ? 'center' : 'space-between';
}, function (props) {
return props.theme.labelColor;
}, function (props) {
return props.isEnlarged ? 'row' : 'column';
});
var TimeTitle = function TimeTitle(_ref) {
var value = _ref.value,
isEnlarged = _ref.isEnlarged,
_ref$timeFormat = _ref.timeFormat,
timeFormat = _ref$timeFormat === undefined ? defaultTimeFormat : _ref$timeFormat;
return _react2.default.createElement(
TimeValueWrapper,
{ isEnlarged: isEnlarged },
_react2.default.createElement(TimeValue, { key: 0, value: _moment2.default.utc(value[0]).format(timeFormat), split: !isEnlarged }),
isEnlarged ? _react2.default.createElement(
'div',
{ className: 'horizontal-bar' },
_react2.default.createElement(_icons.Minus, { height: '12px' })
) : null,
_react2.default.createElement(TimeValue, { key: 1, value: _moment2.default.utc(value[1]).format(timeFormat), split: !isEnlarged })
);
};
var TimeValue = function TimeValue(_ref2) {
var value = _ref2.value,
split = _ref2.split;
return (
// render two lines if not enlarged
_react2.default.createElement(
'div',
{ className: 'time-value' },
split ? value.split(' ').map(function (v, i) {
return _react2.default.createElement(
'div',
{ key: i },
i === 0 ? _react2.default.createElement(
_styledComponents3.SelectText,
null,
v
) : _react2.default.createElement(
_styledComponents3.SelectTextBold,
null,
v
)
);
}) : _react2.default.createElement(
_styledComponents3.SelectTextBold,
null,
value
)
)
);
};
var StyledAnimationControls = _styledComponents2.default.div(_templateObject3);
var IconButton = _styledComponents3.Button.extend(_templateObject4);
var AnimationControls = function AnimationControls(_ref3) {
var isAnimatable = _ref3.isAnimatable,
isAnimating = _ref3.isAnimating,
pauseAnimation = _ref3.pauseAnimation,
resetAnimation = _ref3.resetAnimation,
startAnimation = _ref3.startAnimation;
return _react2.default.createElement(
StyledAnimationControls,
{
className: (0, _classnames2.default)('time-range-slider__control', { disabled: !isAnimatable })
},
_react2.default.createElement(
_styledComponents3.ButtonGroup,
null,
_react2.default.createElement(
IconButton,
{ className: 'playback-control-button',
onClick: resetAnimation, secondary: true },
_react2.default.createElement(_icons.Reset, { height: '12px' })
),
_react2.default.createElement(
IconButton,
{ className: (0, _classnames2.default)('playback-control-button', { active: isAnimating }),
onClick: isAnimating ? pauseAnimation : startAnimation, secondary: true },
isAnimating ? _react2.default.createElement(_icons.Pause, { height: '12px' }) : _react2.default.createElement(_icons.Play, { height: '12px' })
)
)
);
};
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/components/common/time-range-slider.js"],"names":["defaultTimeFormat","moment","utc","val","format","animationControlWidth","StyledSliderContainer","styled","div","props","isEnlarged","TimeRangeSlider","state","isAnimating","width","_animation","_sliderThrottle","onChange","_nextFrame","domain","value","titleFormatter","isAnimatable","_pauseAnimation","_resetAnimation","_startAnimation","histogram","lineChart","plotType","step","_sliderUpdate","TimeSliderMarker","Component","propTypes","PropTypes","func","isRequired","arrayOf","number","string","any","object","toggleAnimation","bool","speed","domainSelector","cancel","args","value0","value1","setState","BASE_SPEED","TimeValueWrapper","theme","secondaryInputHeight","labelColor","TimeTitle","timeFormat","TimeValue","split","map","v","i","StyledAnimationControls","IconButton","Button","extend","AnimationControls","pauseAnimation","resetAnimation","startAnimation","disabled","active"],"mappings":"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","file":"time-range-slider.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 moment from 'moment';\nimport {requestAnimationFrame, cancelAnimationFrame} from 'global/window';\nimport classnames from 'classnames';\nimport throttle from 'lodash.throttle';\nimport styled from 'styled-components';\nimport {createSelector} from 'reselect';\nimport {Play, Reset, Pause, Minus} from 'components/common/icons';\nimport {SelectTextBold, SelectText, Button, ButtonGroup} from 'components/common/styled-components';\nimport {getTimeWidgetTitleFormatter, BASE_SPEED} from 'utils/filter-utils';\nimport RangeSlider from './range-slider';\nimport TimeSliderMarker from './time-slider-marker';\n\nconst defaultTimeFormat = val => moment.utc(val).format('MM/DD/YY hh:mma');\nconst animationControlWidth = 140;\n\nconst StyledSliderContainer = styled.div`\n  margin-top: ${props => props.isEnlarged ? '12px' : '0px'};\n  align-items: flex-end;\n  display: flex;\n  flex-direction: row;\n  justify-content: space-between;\n`;\n\nexport default class TimeRangeSlider extends Component {\n  static propTypes = {\n    onChange: PropTypes.func.isRequired,\n    domain: PropTypes.arrayOf(PropTypes.number).isRequired,\n    value: PropTypes.arrayOf(PropTypes.number).isRequired,\n    step: PropTypes.number.isRequired,\n    plotType: PropTypes.string,\n    histogram: PropTypes.arrayOf(PropTypes.any),\n    lineChart: PropTypes.object,\n    toggleAnimation: PropTypes.func.isRequired,\n    isAnimatable: PropTypes.bool,\n    isEnlarged: PropTypes.bool,\n    speed: PropTypes.number\n  };\n\n  constructor(props) {\n    super(props);\n    this.state = {\n      isAnimating: false,\n      width: 288\n    };\n    this._animation = null;\n    this._sliderThrottle = throttle((...value) => this.props.onChange(...value), 20);\n  }\n\n  componentDidUpdate() {\n    if (!this._animation && this.state.isAnimating) {\n      this._animation = requestAnimationFrame(this._nextFrame);\n    }\n  }\n\n  domainSelector = props => props.domain;\n  titleFormatter = createSelector(this.domainSelector, domain =>\n    getTimeWidgetTitleFormatter(domain)\n  );\n\n  _sliderUpdate = args => {\n    this._sliderThrottle.cancel();\n    this._sliderThrottle(args);\n  };\n\n  _resetAnimation = () => {\n    const {domain, value} = this.props;\n    const value0 = domain[0];\n    const value1 = value0 + value[1] - value[0];\n    this.props.onChange([value0, value1]);\n  };\n\n  _startAnimation = () => {\n    this._pauseAnimation();\n    this.props.toggleAnimation();\n    this.setState({isAnimating: true});\n  };\n\n  _pauseAnimation = () => {\n    if (this._animation) {\n      cancelAnimationFrame(this._animation);\n      this.props.toggleAnimation();\n      this._animation = null;\n    }\n    this.setState({isAnimating: false});\n  };\n\n  _nextFrame = () => {\n    this._animation = null;\n\n    const {domain, value} = this.props;\n    const speed = ((domain[1] - domain[0]) / BASE_SPEED) * this.props.speed;\n\n    // loop when reaches the end\n    const value0 = value[1] + speed > domain[1] ? domain[0] : value[0] + speed;\n    const value1 = value0 + value[1] - value[0];\n    this.props.onChange([value0, value1]);\n  };\n\n  render() {\n    const {domain, value, isEnlarged} = this.props;\n    const {isAnimating} = this.state;\n\n    return (\n      <div className=\"time-range-slider\">\n        <TimeTitle\n          timeFormat={this.titleFormatter(this.props)}\n          value={value}\n          isEnlarged={isEnlarged}\n        />\n        <StyledSliderContainer\n          className=\"time-range-slider__container\"\n          isEnlarged={isEnlarged}>\n          {isEnlarged ? <AnimationControls\n            isAnimatable={this.props.isAnimatable}\n            isEnlarged={isEnlarged}\n            isAnimating={isAnimating}\n            pauseAnimation={this._pauseAnimation}\n            resetAnimation={this._resetAnimation}\n            startAnimation={this._startAnimation}\n          /> : null}\n          <div style={{width: isEnlarged ? `calc(100% - ${animationControlWidth}px)` : '100%'}}>\n            <RangeSlider\n              range={domain}\n              value0={value[0]}\n              value1={value[1]}\n              histogram={this.props.histogram}\n              lineChart={this.props.lineChart}\n              plotType={this.props.plotType}\n              isEnlarged={isEnlarged}\n              showInput={false}\n              step={this.props.step}\n              onChange={this._sliderUpdate}\n              xAxis={TimeSliderMarker}\n            />\n          </div>\n        </StyledSliderContainer>\n      </div>\n    );\n  }\n}\n\nconst TimeValueWrapper = styled.div`\n  display: flex;\n  height: ${props => props.theme.secondaryInputHeight};\n  align-items: center;\n  font-size: 11px;\n  justify-content: ${props => props.isEnlarged ? 'center' : 'space-between'};\n  color: ${props => props.theme.labelColor};\n\n  .horizontal-bar {\n    padding: 0 12px;\n  }\n\n  .time-value {\n    display: flex;\n    flex-direction: ${props => props.isEnlarged ? 'row' : 'column'};\n    align-items: flex-start;\n  }\n\n  .time-value:last-child {\n    align-items: flex-end;\n  }\n`;\n\nconst TimeTitle = ({value, isEnlarged, timeFormat = defaultTimeFormat}) => (\n  <TimeValueWrapper isEnlarged={isEnlarged}>\n    <TimeValue key={0} value={moment.utc(value[0]).format(timeFormat)} split={!isEnlarged}/>\n    {isEnlarged ? (\n      <div className=\"horizontal-bar\">\n        <Minus height=\"12px\"/>\n      </div>\n    ) : null}\n    <TimeValue key={1} value={moment.utc(value[1]).format(timeFormat)} split={!isEnlarged}/>\n  </TimeValueWrapper>\n);\n\nconst TimeValue = ({value, split}) => (\n  // render two lines if not enlarged\n  <div className=\"time-value\">\n    {split ? value.split(' ').map((v, i) => (\n      <div key={i}>\n        {i === 0 ? <SelectText>{v}</SelectText> :\n        <SelectTextBold>{v}</SelectTextBold>}\n      </div>\n    )) : <SelectTextBold>{value}</SelectTextBold>}\n  </div>\n);\n\nconst StyledAnimationControls = styled.div`\n  margin-bottom: 12px;\n  margin-right: 42px;\n\n  &.disabled {\n    opacity: 0.4;\n    pointer-events: none;\n  }\n`;\n\nconst IconButton = Button.extend`\n  svg {\n    margin: 0 6px;\n  }\n`;\n\nconst AnimationControls = ({\n  isAnimatable,\n  isAnimating,\n  pauseAnimation,\n  resetAnimation,\n  startAnimation\n}) => (\n  <StyledAnimationControls\n    className={classnames('time-range-slider__control', {disabled: !isAnimatable})}\n  >\n    <ButtonGroup>\n      <IconButton className=\"playback-control-button\"\n        onClick={resetAnimation} secondary>\n        <Reset height=\"12px\"/>\n      </IconButton>\n      <IconButton className={classnames('playback-control-button', {active: isAnimating})}\n        onClick={isAnimating ? pauseAnimation : startAnimation} secondary>\n        {isAnimating ? <Pause height=\"12px\"/> : <Play height=\"12px\"/>}\n      </IconButton>\n    </ButtonGroup>\n  </StyledAnimationControls>\n);\n"]}