UNPKG

kepler.gl

Version:

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

273 lines (226 loc) 28.3 kB
"use strict"; var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault"); Object.defineProperty(exports, "__esModule", { value: true }); 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 _react = require("react"); var _d3Array = require("d3-array"); var _window = require("global/window"); var _console = _interopRequireDefault(require("global/console")); var _defaultSettings = require("../../../constants/default-settings"); 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; } } function AnimationControllerFactory() { /** * 4 Animation Window Types * 1. free * |-> |-> * Current time is a fixed range, animate a moving window that calls next animation frames continuously * The increment id based on domain / BASE_SPEED * SPEED * * 2. incremental * | |-> * Same as free, current time is a growing range, only the max value of range increment during animation. * The increment is also based on domain / BASE_SPEED * SPEED * * 3. point * o -> o * Current time is a point, animate a moving point calls next animation frame continuously * The increment is based on domain / BASE_SPEED * SPEED * * 4. interval * o ~> o * Current time is a point. An array of sorted time steps are provided, * animate a moving point jumps to the next step */ var AnimationController = /*#__PURE__*/function (_Component) { (0, _inherits2["default"])(AnimationController, _Component); var _super = _createSuper(AnimationController); function AnimationController() { var _this; (0, _classCallCheck2["default"])(this, AnimationController); 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), "state", { isAnimating: false }); (0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "_timer", null); (0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "_animate", function (delay) { _this._startTime = new Date().getTime(); var loop = function loop() { var current = new Date().getTime(); // @ts-ignore var delta = current - _this._startTime; if (delta >= delay) { _this._nextFrame(); _this._startTime = new Date().getTime(); } else { _this._timer = (0, _window.requestAnimationFrame)(loop); } }; _this._timer = (0, _window.requestAnimationFrame)(loop); }); (0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "_resetAnimationByDomain", function () { var _this$props = _this.props, domain = _this$props.domain, value = _this$props.value, animationWindow = _this$props.animationWindow; if (Array.isArray(value)) { if (animationWindow === _defaultSettings.ANIMATION_WINDOW.incremental) { _this.props.updateAnimation([value[0], value[0] + 1]); } else { _this.props.updateAnimation([domain[0], domain[0] + value[1] - value[0]]); } } else { _this.props.updateAnimation(domain[0]); } }); (0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "_resetAnimtionByTimeStep", function () { // go to the first steps _this.props.updateAnimation([_this.props.steps[0], 0]); }); (0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "_resetAnimation", function () { if (_this.props.animationWindow === _defaultSettings.ANIMATION_WINDOW.interval) { _this._resetAnimtionByTimeStep(); } else { _this._resetAnimationByDomain(); } }); (0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "_startAnimation", function () { var speed = _this.props.speed; _this._clearTimer(); if (speed > 0) { if (_this.props.animationWindow === _defaultSettings.ANIMATION_WINDOW.interval) { // animate by interval // 30*600 var steps = _this.props.steps; if (!Array.isArray(steps) || !steps.length) { _console["default"].warn('animation steps should be an array'); return; } // when speed = 1, animation should loop through 600 frames at 60 FPS // calculate delay based on # steps var delay = _defaultSettings.BASE_SPEED * (1000 / _defaultSettings.FPS) / steps.length / (speed || 1); _this._animate(delay); } else { _this._timer = (0, _window.requestAnimationFrame)(_this._nextFrame); } } _this.setState({ isAnimating: true }); }); (0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "_clearTimer", function () { if (_this._timer) { (0, _window.cancelAnimationFrame)(_this._timer); _this._timer = null; } }); (0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "_pauseAnimation", function () { _this._clearTimer(); _this.setState({ isAnimating: false }); }); (0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "_nextFrame", function () { _this._timer = null; var nextValue = _this.props.animationWindow === _defaultSettings.ANIMATION_WINDOW.interval ? _this._nextFrameByTimeStep() : _this._nextFrameByDomain(); _this.props.updateAnimation(nextValue); }); return _this; } (0, _createClass2["default"])(AnimationController, [{ key: "componentDidMount", value: function componentDidMount() { this._startOrPauseAnimation(); } }, { key: "componentDidUpdate", value: function componentDidUpdate() { this._startOrPauseAnimation(); } }, { key: "componentWillUnmount", value: function componentWillUnmount() { if (this._timer) { (0, _window.cancelAnimationFrame)(this._timer); } } }, { key: "_startOrPauseAnimation", value: function _startOrPauseAnimation() { var _this$props2 = this.props, isAnimating = _this$props2.isAnimating, speed = _this$props2.speed; if (!this._timer && isAnimating && speed > 0) { this._startAnimation(); } else if (this._timer && !isAnimating) { this._pauseAnimation(); } } }, { key: "_nextFrameByDomain", value: function _nextFrameByDomain() { var _this$props3 = this.props, domain = _this$props3.domain, value = _this$props3.value, speed = _this$props3.speed, baseSpeed = _this$props3.baseSpeed, animationWindow = _this$props3.animationWindow; var delta = (domain[1] - domain[0]) / baseSpeed * speed; // loop when reaches the end // current time is a range if (Array.isArray(value)) { var value0; var value1; var readEnd = value[1] + delta > domain[1]; if (animationWindow === _defaultSettings.ANIMATION_WINDOW.incremental) { value0 = value[0]; value1 = readEnd ? value[0] + 1 : value[1] + delta; } else { value0 = readEnd ? domain[0] : value[0] + delta; value1 = value0 + value[1] - value[0]; } return [value0, value1]; } // current time is a point return value + delta > domain[1] ? domain[0] : value + delta; } }, { key: "_nextFrameByTimeStep", value: function _nextFrameByTimeStep() { var _this$props4 = this.props, steps = _this$props4.steps, value = _this$props4.value; var val = Array.isArray(value) ? value[0] : value; var index = (0, _d3Array.bisectLeft)(steps, val); var nextIdx = index >= steps.length - 1 ? 0 : index + 1; return [steps[nextIdx], nextIdx]; } }, { key: "render", value: function render() { var isAnimating = this.state.isAnimating; var children = this.props.children; return typeof children === 'function' ? children(isAnimating, this._startAnimation, this._pauseAnimation, this._resetAnimation) : null; } }]); return AnimationController; }(_react.Component); (0, _defineProperty2["default"])(AnimationController, "defaultProps", { baseSpeed: _defaultSettings.BASE_SPEED, speed: 1, steps: null, animationWindow: _defaultSettings.ANIMATION_WINDOW.free }); return AnimationController; } var _default = AnimationControllerFactory; exports["default"] = _default; //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/common/animation-control/animation-controller.js"],"names":["AnimationControllerFactory","AnimationController","isAnimating","delay","_startTime","Date","getTime","loop","current","delta","_nextFrame","_timer","props","domain","value","animationWindow","Array","isArray","ANIMATION_WINDOW","incremental","updateAnimation","steps","interval","_resetAnimtionByTimeStep","_resetAnimationByDomain","speed","_clearTimer","length","Console","warn","BASE_SPEED","FPS","_animate","setState","nextValue","_nextFrameByTimeStep","_nextFrameByDomain","_startOrPauseAnimation","_startAnimation","_pauseAnimation","baseSpeed","value0","value1","readEnd","val","index","nextIdx","state","children","_resetAnimation","Component","free"],"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 {Component} from 'react';\nimport {bisectLeft} from 'd3-array';\nimport {requestAnimationFrame, cancelAnimationFrame} from 'global/window';\nimport Console from 'global/console';\nimport {BASE_SPEED, FPS, ANIMATION_WINDOW} from 'constants/default-settings';\n\nfunction AnimationControllerFactory() {\n  /**\n   * 4 Animation Window Types\n   * 1. free\n   *  |->  |->\n   * Current time is a fixed range, animate a moving window that calls next animation frames continuously\n   * The increment id based on domain / BASE_SPEED * SPEED\n   *\n   * 2. incremental\n   * |    |->\n   * Same as free, current time is a growing range, only the max value of range increment during animation.\n   * The increment is also based on domain / BASE_SPEED * SPEED\n   *\n   * 3. point\n   * o -> o\n   * Current time is a point, animate a moving point calls next animation frame continuously\n   * The increment is based on domain / BASE_SPEED * SPEED\n   *\n   * 4. interval\n   * o ~> o\n   * Current time is a point. An array of sorted time steps are provided,\n   * animate a moving point jumps to the next step\n   */\n  class AnimationController extends Component {\n    // TODO: convert the entire component to use hooks in the next PR\n    static defaultProps = {\n      baseSpeed: BASE_SPEED,\n      speed: 1,\n      steps: null,\n      animationWindow: ANIMATION_WINDOW.free\n    };\n\n    state = {\n      isAnimating: false\n    };\n\n    componentDidMount() {\n      this._startOrPauseAnimation();\n    }\n\n    componentDidUpdate() {\n      this._startOrPauseAnimation();\n    }\n\n    componentWillUnmount() {\n      if (this._timer) {\n        cancelAnimationFrame(this._timer);\n      }\n    }\n\n    _timer = null;\n\n    _startOrPauseAnimation() {\n      const {isAnimating, speed} = this.props;\n      if (!this._timer && isAnimating && speed > 0) {\n        this._startAnimation();\n      } else if (this._timer && !isAnimating) {\n        this._pauseAnimation();\n      }\n    }\n\n    _animate = delay => {\n      this._startTime = new Date().getTime();\n\n      const loop = () => {\n        const current = new Date().getTime();\n        // @ts-ignore\n        const delta = current - this._startTime;\n\n        if (delta >= delay) {\n          this._nextFrame();\n          this._startTime = new Date().getTime();\n        } else {\n          this._timer = requestAnimationFrame(loop);\n        }\n      };\n\n      this._timer = requestAnimationFrame(loop);\n    };\n\n    _resetAnimationByDomain = () => {\n      const {domain, value, animationWindow} = this.props;\n      if (Array.isArray(value)) {\n        if (animationWindow === ANIMATION_WINDOW.incremental) {\n          this.props.updateAnimation([value[0], value[0] + 1]);\n        } else {\n          this.props.updateAnimation([domain[0], domain[0] + value[1] - value[0]]);\n        }\n      } else {\n        this.props.updateAnimation(domain[0]);\n      }\n    };\n\n    _resetAnimtionByTimeStep = () => {\n      // go to the first steps\n      this.props.updateAnimation([this.props.steps[0], 0]);\n    };\n\n    _resetAnimation = () => {\n      if (this.props.animationWindow === ANIMATION_WINDOW.interval) {\n        this._resetAnimtionByTimeStep();\n      } else {\n        this._resetAnimationByDomain();\n      }\n    };\n\n    _startAnimation = () => {\n      const {speed} = this.props;\n      this._clearTimer();\n      if (speed > 0) {\n        if (this.props.animationWindow === ANIMATION_WINDOW.interval) {\n          // animate by interval\n          // 30*600\n          const {steps} = this.props;\n          if (!Array.isArray(steps) || !steps.length) {\n            Console.warn('animation steps should be an array');\n            return;\n          }\n          // when speed = 1, animation should loop through 600 frames at 60 FPS\n          // calculate delay based on # steps\n          const delay = (BASE_SPEED * (1000 / FPS)) / steps.length / (speed || 1);\n          this._animate(delay);\n        } else {\n          this._timer = requestAnimationFrame(this._nextFrame);\n        }\n      }\n      this.setState({isAnimating: true});\n    };\n\n    _clearTimer = () => {\n      if (this._timer) {\n        cancelAnimationFrame(this._timer);\n        this._timer = null;\n      }\n    };\n\n    _pauseAnimation = () => {\n      this._clearTimer();\n      this.setState({isAnimating: false});\n    };\n\n    _nextFrame = () => {\n      this._timer = null;\n      const nextValue =\n        this.props.animationWindow === ANIMATION_WINDOW.interval\n          ? this._nextFrameByTimeStep()\n          : this._nextFrameByDomain();\n\n      this.props.updateAnimation(nextValue);\n    };\n\n    _nextFrameByDomain() {\n      const {domain, value, speed, baseSpeed, animationWindow} = this.props;\n      const delta = ((domain[1] - domain[0]) / baseSpeed) * speed;\n\n      // loop when reaches the end\n      // current time is a range\n      if (Array.isArray(value)) {\n        let value0;\n        let value1;\n        const readEnd = value[1] + delta > domain[1];\n        if (animationWindow === ANIMATION_WINDOW.incremental) {\n          value0 = value[0];\n          value1 = readEnd ? value[0] + 1 : value[1] + delta;\n        } else {\n          value0 = readEnd ? domain[0] : value[0] + delta;\n          value1 = value0 + value[1] - value[0];\n        }\n        return [value0, value1];\n      }\n\n      // current time is a point\n      return value + delta > domain[1] ? domain[0] : value + delta;\n    }\n\n    _nextFrameByTimeStep() {\n      const {steps, value} = this.props;\n      const val = Array.isArray(value) ? value[0] : value;\n      const index = bisectLeft(steps, val);\n      const nextIdx = index >= steps.length - 1 ? 0 : index + 1;\n\n      return [steps[nextIdx], nextIdx];\n    }\n\n    render() {\n      const {isAnimating} = this.state;\n      const {children} = this.props;\n\n      return typeof children === 'function'\n        ? children(isAnimating, this._startAnimation, this._pauseAnimation, this._resetAnimation)\n        : null;\n    }\n  }\n\n  return AnimationController;\n}\n\nexport default AnimationControllerFactory;\n"]}