kepler.gl.geoiq
Version:
kepler.gl is a webgl based application to visualize large scale location data in the browser
300 lines (245 loc) • 36.3 kB
JavaScript
"use strict";
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
var _typeof = require("@babel/runtime/helpers/typeof");
Object.defineProperty(exports, "__esModule", {
value: true
});
exports["default"] = TimeRangeSliderFactory;
var _slicedToArray2 = _interopRequireDefault(require("@babel/runtime/helpers/slicedToArray"));
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 _moment = _interopRequireDefault(require("moment"));
var _window = require("global/window");
var _lodash = _interopRequireDefault(require("lodash.throttle"));
var _styledComponents = _interopRequireDefault(require("styled-components"));
var _reselect = require("reselect");
var _icons = require("./icons");
var _styledComponents2 = require("./styled-components");
var _rangeSlider = _interopRequireDefault(require("./range-slider"));
var _timeSliderMarker = _interopRequireDefault(require("./time-slider-marker"));
var _playbackControls = _interopRequireDefault(require("./animation-control/playback-controls"));
var _defaultSettings = require("../../constants/default-settings");
var _templateObject, _templateObject2;
function _getRequireWildcardCache(nodeInterop) { if (typeof WeakMap !== "function") return null; var cacheBabelInterop = new WeakMap(); var cacheNodeInterop = new WeakMap(); return (_getRequireWildcardCache = function _getRequireWildcardCache(nodeInterop) { return nodeInterop ? cacheNodeInterop : cacheBabelInterop; })(nodeInterop); }
function _interopRequireWildcard(obj, nodeInterop) { if (!nodeInterop && obj && obj.__esModule) { return obj; } if (obj === null || _typeof(obj) !== "object" && typeof obj !== "function") { return { "default": obj }; } var cache = _getRequireWildcardCache(nodeInterop); if (cache && cache.has(obj)) { return cache.get(obj); } var newObj = {}; var hasPropertyDescriptor = Object.defineProperty && Object.getOwnPropertyDescriptor; for (var key in obj) { if (key !== "default" && Object.prototype.hasOwnProperty.call(obj, key)) { var desc = hasPropertyDescriptor ? Object.getOwnPropertyDescriptor(obj, key) : null; if (desc && (desc.get || desc.set)) { Object.defineProperty(newObj, key, desc); } else { newObj[key] = obj[key]; } } } newObj["default"] = obj; if (cache) { cache.set(obj, newObj); } return newObj; }
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 animationControlWidth = 140;
var StyledSliderContainer = _styledComponents["default"].div(_templateObject || (_templateObject = (0, _taggedTemplateLiteral2["default"])(["\n align-items: flex-end;\n display: flex;\n flex-direction: row;\n justify-content: space-between;\n\n .time-range-slider__control {\n margin-bottom: 12px;\n margin-right: 30px;\n }\n\n .playback-control-button {\n padding: 9px 12px;\n }\n"])));
TimeRangeSliderFactory.deps = [_playbackControls["default"]];
function TimeRangeSliderFactory(PlaybackControls) {
var TimeRangeSlider = /*#__PURE__*/function (_Component) {
(0, _inherits2["default"])(TimeRangeSlider, _Component);
var _super = _createSuper(TimeRangeSlider);
function TimeRangeSlider(_props) {
var _this;
(0, _classCallCheck2["default"])(this, TimeRangeSlider);
_this = _super.call(this, _props);
(0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "timeSelector", function (props) {
return props.currentTime;
});
(0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "formatSelector", function (props) {
return props.format;
});
(0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "displayTimeSelector", (0, _reselect.createSelector)(_this.timeSelector, _this.formatSelector, function (currentTime, format) {
var groupTime = Array.isArray(currentTime) ? currentTime : [currentTime];
return groupTime.reduce(function (accu, curr) {
var displayDateTime = _moment["default"].utc(curr).format(format);
var _displayDateTime$spli = displayDateTime.split(' '),
_displayDateTime$spli2 = (0, _slicedToArray2["default"])(_displayDateTime$spli, 2),
displayDate = _displayDateTime$spli2[0],
displayTime = _displayDateTime$spli2[1];
if (!accu.displayDate.includes(displayDate)) {
accu.displayDate.push(displayDate);
}
accu.displayTime.push(displayTime);
return accu;
}, {
displayDate: [],
displayTime: []
});
}));
(0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "_sliderUpdate", function (args) {
_this._sliderThrottle.cancel();
_this._sliderThrottle(args);
});
(0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "_resetAnimation", function () {
var _this$props = _this.props,
domain = _this$props.domain,
value = _this$props.value;
var value0 = domain[0];
var value1 = value0 + value[1] - value[0];
_this.props.onChange([value0, value1]);
});
(0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "_startAnimation", function () {
_this._pauseAnimation();
_this.props.toggleAnimation();
_this.setState({
isAnimating: true
});
});
(0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "_pauseAnimation", function () {
if (_this._animation) {
(0, _window.cancelAnimationFrame)(_this._animation);
_this.props.toggleAnimation();
_this._animation = null;
}
_this.setState({
isAnimating: false
});
});
(0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "_nextFrame", function () {
_this._animation = null;
var _this$props2 = _this.props,
domain = _this$props2.domain,
value = _this$props2.value;
var speed = (domain[1] - domain[0]) / _defaultSettings.BASE_SPEED * _this.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];
_this.props.onChange([value0, value1]);
});
_this.state = {
isAnimating: false,
width: 288
};
_this._animation = null;
_this._sliderThrottle = (0, _lodash["default"])(function () {
var _this$props3;
return (_this$props3 = _this.props).onChange.apply(_this$props3, arguments);
}, 20);
return _this;
}
(0, _createClass2["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 _this$props4 = this.props,
domain = _this$props4.domain,
value = _this$props4.value,
isEnlarged = _this$props4.isEnlarged,
isLargeData = _this$props4.isLargeData,
hideTimeTitle = _this$props4.hideTimeTitle;
var isAnimating = this.state.isAnimating;
return (/*#__PURE__*/_react["default"].createElement("div", {
className: "time-range-slider"
}, !hideTimeTitle ? /*#__PURE__*/_react["default"].createElement(TimeTitle, {
timeFormat: this.props.timeFormat,
value: value,
isEnlarged: isEnlarged
}) : null, /*#__PURE__*/_react["default"].createElement(StyledSliderContainer, {
className: "time-range-slider__container",
isEnlarged: isEnlarged
}, !isLargeData && isEnlarged ? /*#__PURE__*/_react["default"].createElement(PlaybackControls, {
isAnimatable: this.props.isAnimatable,
isEnlarged: isEnlarged,
isAnimating: isAnimating,
pauseAnimation: this._pauseAnimation,
resetAnimation: this._resetAnimation,
startAnimation: this._startAnimation,
buttonHeight: "12px",
buttonStyle: "secondary"
}) : null, /*#__PURE__*/_react["default"].createElement("div", {
style: {
width: isEnlarged ? "calc(100% - ".concat(animationControlWidth, "px)") : '100%'
}
}, /*#__PURE__*/_react["default"].createElement(_rangeSlider["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: _timeSliderMarker["default"]
}))))
);
}
}]);
return TimeRangeSlider;
}(_react.Component);
(0, _defineProperty2["default"])(TimeRangeSlider, "propTypes", {
onChange: _propTypes["default"].func.isRequired,
domain: _propTypes["default"].arrayOf(_propTypes["default"].number).isRequired,
value: _propTypes["default"].arrayOf(_propTypes["default"].number).isRequired,
step: _propTypes["default"].number.isRequired,
plotType: _propTypes["default"].string,
histogram: _propTypes["default"].arrayOf(_propTypes["default"].any),
lineChart: _propTypes["default"].object,
toggleAnimation: _propTypes["default"].func.isRequired,
isAnimatable: _propTypes["default"].bool,
isEnlarged: _propTypes["default"].bool,
speed: _propTypes["default"].number,
timeFormat: _propTypes["default"].string,
hideTimeTitle: _propTypes["default"].bool
});
TimeRangeSlider.defaultProps = {
timeFormat: _defaultSettings.DEFAULT_TIME_FORMAT
};
return TimeRangeSlider;
}
var TimeValueWrapper = _styledComponents["default"].div(_templateObject2 || (_templateObject2 = (0, _taggedTemplateLiteral2["default"])(["\n display: flex;\n align-items: center;\n font-size: 11px;\n justify-content: ", ";\n\n .horizontal-bar {\n padding: 0 12px;\n color: ", ";\n }\n\n .time-value {\n display: flex;\n flex-direction: ", ";\n align-items: flex-start;\n\n span {\n color: ", ";\n }\n }\n\n .time-value:last-child {\n align-items: flex-end;\n }\n"])), function (props) {
return props.isEnlarged ? 'center' : 'space-between';
}, function (props) {
return props.theme.titleTextColor;
}, function (props) {
return props.isEnlarged ? 'row' : 'column';
}, function (props) {
return props.theme.titleTextColor;
});
var TimeTitle = function TimeTitle(_ref) {
var value = _ref.value,
isEnlarged = _ref.isEnlarged,
_ref$timeFormat = _ref.timeFormat,
timeFormat = _ref$timeFormat === void 0 ? _defaultSettings.DEFAULT_TIME_FORMAT : _ref$timeFormat;
return (/*#__PURE__*/_react["default"].createElement(TimeValueWrapper, {
isEnlarged: isEnlarged,
className: "time-range-slider__time-title"
}, /*#__PURE__*/_react["default"].createElement(TimeValue, {
key: 0,
value: _moment["default"].utc(value[0]).format(timeFormat),
split: !isEnlarged
}), isEnlarged ? /*#__PURE__*/_react["default"].createElement("div", {
className: "horizontal-bar"
}, /*#__PURE__*/_react["default"].createElement(_icons.Minus, {
height: "12px"
})) : null, /*#__PURE__*/_react["default"].createElement(TimeValue, {
key: 1,
value: _moment["default"].utc(value[1]).format(timeFormat),
split: !isEnlarged
}))
);
};
var TimeValue = function TimeValue(_ref2) {
var value = _ref2.value,
split = _ref2.split;
return (
/*#__PURE__*/
// render two lines if not enlarged
_react["default"].createElement("div", {
className: "time-value"
}, split ? value.split(' ').map(function (v, i) {
return (/*#__PURE__*/_react["default"].createElement("div", {
key: i
}, i === 0 ? /*#__PURE__*/_react["default"].createElement(_styledComponents2.SelectText, null, v) : /*#__PURE__*/_react["default"].createElement(_styledComponents2.SelectTextBold, null, v))
);
}) : /*#__PURE__*/_react["default"].createElement(_styledComponents2.SelectTextBold, null, value))
);
};
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/components/common/time-range-slider.js"],"names":["animationControlWidth","StyledSliderContainer","styled","div","TimeRangeSliderFactory","deps","PlaybackControlsFactory","PlaybackControls","TimeRangeSlider","props","currentTime","format","timeSelector","formatSelector","groupTime","Array","isArray","reduce","accu","curr","displayDateTime","moment","utc","split","displayDate","displayTime","includes","push","args","_sliderThrottle","cancel","domain","value","value0","value1","onChange","_pauseAnimation","toggleAnimation","setState","isAnimating","_animation","speed","BASE_SPEED","state","width","_nextFrame","isEnlarged","isLargeData","hideTimeTitle","timeFormat","isAnimatable","_resetAnimation","_startAnimation","histogram","lineChart","plotType","step","_sliderUpdate","TimeSliderMarker","Component","PropTypes","func","isRequired","arrayOf","number","string","any","object","bool","defaultProps","DEFAULT_TIME_FORMAT","TimeValueWrapper","theme","titleTextColor","TimeTitle","TimeValue","map","v","i"],"mappings":"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","sourcesContent":["// Copyright (c) 2023 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 throttle from 'lodash.throttle';\nimport styled from 'styled-components';\nimport {createSelector} from 'reselect';\n\nimport {Minus} from 'components/common/icons';\nimport {SelectTextBold, SelectText} from 'components/common/styled-components';\nimport RangeSlider from 'components/common/range-slider';\nimport TimeSliderMarker from 'components/common/time-slider-marker';\nimport PlaybackControlsFactory from 'components/common/animation-control/playback-controls';\nimport {BASE_SPEED, DEFAULT_TIME_FORMAT} from 'constants/default-settings';\n\nconst animationControlWidth = 140;\n\nconst StyledSliderContainer = styled.div`\n  align-items: flex-end;\n  display: flex;\n  flex-direction: row;\n  justify-content: space-between;\n\n  .time-range-slider__control {\n    margin-bottom: 12px;\n    margin-right: 30px;\n  }\n\n  .playback-control-button {\n    padding: 9px 12px;\n  }\n`;\n\nTimeRangeSliderFactory.deps = [PlaybackControlsFactory];\n\nexport default function TimeRangeSliderFactory(PlaybackControls) {\n  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      timeFormat: PropTypes.string,\n      hideTimeTitle: PropTypes.bool\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(\n        (...value) => this.props.onChange(...value),\n        20\n      );\n    }\n\n    componentDidUpdate() {\n      if (!this._animation && this.state.isAnimating) {\n        this._animation = requestAnimationFrame(this._nextFrame);\n      }\n    }\n\n    timeSelector = props => props.currentTime;\n    formatSelector = props => props.format;\n    displayTimeSelector = createSelector(\n      this.timeSelector,\n      this.formatSelector,\n      (currentTime, format) => {\n        const groupTime = Array.isArray(currentTime)\n          ? currentTime\n          : [currentTime];\n        return groupTime.reduce(\n          (accu, curr) => {\n            const displayDateTime = moment.utc(curr).format(format);\n            const [displayDate, displayTime] = displayDateTime.split(' ');\n\n            if (!accu.displayDate.includes(displayDate)) {\n              accu.displayDate.push(displayDate);\n            }\n            accu.displayTime.push(displayTime);\n\n            return accu;\n          },\n          {displayDate: [], displayTime: []}\n        );\n      }\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 =\n        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 {\n        domain,\n        value,\n        isEnlarged,\n        isLargeData,\n        hideTimeTitle\n      } = this.props;\n      const {isAnimating} = this.state;\n\n      return (\n        <div className=\"time-range-slider\">\n          {!hideTimeTitle ? (\n            <TimeTitle\n              timeFormat={this.props.timeFormat}\n              value={value}\n              isEnlarged={isEnlarged}\n            />\n          ) : null}\n          <StyledSliderContainer\n            className=\"time-range-slider__container\"\n            isEnlarged={isEnlarged}\n          >\n            {!isLargeData && isEnlarged ? (\n              <PlaybackControls\n                isAnimatable={this.props.isAnimatable}\n                isEnlarged={isEnlarged}\n                isAnimating={isAnimating}\n                pauseAnimation={this._pauseAnimation}\n                resetAnimation={this._resetAnimation}\n                startAnimation={this._startAnimation}\n                buttonHeight=\"12px\"\n                buttonStyle=\"secondary\"\n              />\n            ) : null}\n            <div\n              style={{\n                width: isEnlarged\n                  ? `calc(100% - ${animationControlWidth}px)`\n                  : '100%'\n              }}\n            >\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\n  TimeRangeSlider.defaultProps = {\n    timeFormat: DEFAULT_TIME_FORMAT\n  };\n\n  return TimeRangeSlider;\n}\n\nconst TimeValueWrapper = styled.div`\n  display: flex;\n  align-items: center;\n  font-size: 11px;\n  justify-content: ${props => (props.isEnlarged ? 'center' : 'space-between')};\n\n  .horizontal-bar {\n    padding: 0 12px;\n    color: ${props => props.theme.titleTextColor};\n  }\n\n  .time-value {\n    display: flex;\n    flex-direction: ${props => (props.isEnlarged ? 'row' : 'column')};\n    align-items: flex-start;\n\n    span {\n      color: ${props => props.theme.titleTextColor};\n    }\n  }\n\n  .time-value:last-child {\n    align-items: flex-end;\n  }\n`;\n\nconst TimeTitle = ({value, isEnlarged, timeFormat = DEFAULT_TIME_FORMAT}) => (\n  <TimeValueWrapper\n    isEnlarged={isEnlarged}\n    className=\"time-range-slider__time-title\"\n  >\n    <TimeValue\n      key={0}\n      value={moment.utc(value[0]).format(timeFormat)}\n      split={!isEnlarged}\n    />\n    {isEnlarged ? (\n      <div className=\"horizontal-bar\">\n        <Minus height=\"12px\" />\n      </div>\n    ) : null}\n    <TimeValue\n      key={1}\n      value={moment.utc(value[1]).format(timeFormat)}\n      split={!isEnlarged}\n    />\n  </TimeValueWrapper>\n);\n\nconst TimeValue = ({value, split}) => (\n  // render two lines if not enlarged\n  <div className=\"time-value\">\n    {split ? (\n      value\n        .split(' ')\n        .map((v, i) => (\n          <div key={i}>\n            {i === 0 ? (\n              <SelectText>{v}</SelectText>\n            ) : (\n              <SelectTextBold>{v}</SelectTextBold>\n            )}\n          </div>\n        ))\n    ) : (\n      <SelectTextBold>{value}</SelectTextBold>\n    )}\n  </div>\n);\n"]}