UNPKG

kepler.gl.geoiq

Version:

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

213 lines (166 loc) 19.7 kB
"use strict"; var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault"); var _typeof = require("@babel/runtime/helpers/typeof"); Object.defineProperty(exports, "__esModule", { value: true }); exports["default"] = void 0; 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 _styledComponents = _interopRequireDefault(require("styled-components")); var _d3Selection = require("d3-selection"); var _d3Brush = require("d3-brush"); var _templateObject; 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 StyledG = _styledComponents["default"].g(_templateObject || (_templateObject = (0, _taggedTemplateLiteral2["default"])(["\n .selection {\n stroke: none;\n fill: ", ";\n opacity: 1;\n }\n"])), function (props) { return props.theme.rangeBrushBgd; }); var RangeBrush = exports["default"] = /*#__PURE__*/function (_Component) { (0, _inherits2["default"])(RangeBrush, _Component); var _super = _createSuper(RangeBrush); function RangeBrush() { var _this; (0, _classCallCheck2["default"])(this, RangeBrush); 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), "rootContainer", (0, _react.createRef)()); return _this; } (0, _createClass2["default"])(RangeBrush, [{ key: "componentDidMount", value: function componentDidMount() { var _this2 = this; // We want the React app to respond to brush state and vice-versa // but d3-brush fires the same events for both user-initiated brushing // and programmatic brushing (brush.move). We need these flags to // distinguish between the uses. // // We don't use state because that would trigger another `componentDidUpdate` this.brushing = false; this.moving = false; this.root = (0, _d3Selection.select)(this.rootContainer.current); this.brush = (0, _d3Brush.brushX)().on('start', function () { _this2.brushing = true; }).on('brush', function () { if (_this2.moving) { return; } _d3Selection.event.selection === null ? _this2._reset() : _this2._brush(_d3Selection.event.selection); }).on('end', function () { if (!_this2.moving && _d3Selection.event.selection === null) { _this2._reset(); } _this2.brushing = false; _this2.moving = false; }); this.root.call(this.brush); } }, { key: "componentDidUpdate", value: function componentDidUpdate(prevProps) { var _this$props = this.props, _this$props$value = (0, _slicedToArray2["default"])(_this$props.value, 2), val0 = _this$props$value[0], val1 = _this$props$value[1], width = _this$props.width; var _prevProps$value = (0, _slicedToArray2["default"])(prevProps.value, 2), prevVal0 = _prevProps$value[0], prevVal1 = _prevProps$value[1]; if (prevProps.width !== width) { // width change should not trigger this._brush this.moving = true; this.root.call(this.brush); this._move(val0, val1); } if (!this.brushing && !this.moving) { if (prevVal0 !== val0 || prevVal1 !== val1) { this.moving = true; this._move(val0, val1); } } } }, { key: "_reset", value: function _reset() { var _this$props$range = (0, _slicedToArray2["default"])(this.props.range, 2), minValue = _this$props$range[0], maxValue = _this$props$range[1]; this._onBrush(minValue, maxValue); } }, { key: "_move", value: function _move(val0, val1) { var _this$props2 = this.props, _this$props2$domain = (0, _slicedToArray2["default"])(_this$props2.domain, 2), min = _this$props2$domain[0], max = _this$props2$domain[1], width = _this$props2.width; var scale = function scale(x) { return (x - min) * width / (max - min); }; this.brush.move(this.root, [scale(val0), scale(val1)]); } }, { key: "_brush", value: function _brush(_ref) { var _ref2 = (0, _slicedToArray2["default"])(_ref, 2), sel0 = _ref2[0], sel1 = _ref2[1]; var _this$props3 = this.props, _this$props3$domain = (0, _slicedToArray2["default"])(_this$props3.domain, 2), min = _this$props3$domain[0], max = _this$props3$domain[1], width = _this$props3.width; var invert = function invert(x) { return x * (max - min) / width + min; }; this._onBrush(invert(sel0), invert(sel1)); } }, { key: "_onBrush", value: function _onBrush(val0, val1) { var _this$props$value2 = (0, _slicedToArray2["default"])(this.props.value, 2), currentVal0 = _this$props$value2[0], currentVal1 = _this$props$value2[1]; if (currentVal0 === val0 && currentVal1 === val1) { return; } this.props.onBrush(val0, val1); } }, { key: "render", value: function render() { return (/*#__PURE__*/_react["default"].createElement(StyledG, { className: "kg-range-slider__brush" // innerRef={comp => { // this.rootContainer = comp; // }} , ref: this.rootContainer }) ); } }]); return RangeBrush; }(_react.Component); (0, _defineProperty2["default"])(RangeBrush, "propTypes", { domain: _propTypes["default"].arrayOf(_propTypes["default"].number).isRequired, onBrush: _propTypes["default"].func.isRequired, range: _propTypes["default"].arrayOf(_propTypes["default"].number).isRequired, value: _propTypes["default"].arrayOf(_propTypes["default"].number).isRequired, width: _propTypes["default"].number.isRequired }); //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/components/common/range-brush.js"],"names":["StyledG","styled","g","props","theme","rangeBrushBgd","RangeBrush","brushing","moving","root","rootContainer","current","brush","on","event","selection","_reset","_brush","call","prevProps","value","val0","val1","width","prevVal0","prevVal1","_move","range","minValue","maxValue","_onBrush","domain","min","max","scale","x","move","sel0","sel1","invert","currentVal0","currentVal1","onBrush","Component","PropTypes","arrayOf","number","isRequired","func"],"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, createRef} from 'react';\nimport PropTypes from 'prop-types';\nimport styled from 'styled-components';\nimport {event, select} from 'd3-selection';\nimport {brushX} from 'd3-brush';\n\nconst StyledG = styled.g`\n  .selection {\n    stroke: none;\n    fill: ${props => props.theme.rangeBrushBgd};\n    opacity: 1;\n  }\n`;\n\nexport default class RangeBrush extends Component {\n  static propTypes = {\n    domain: PropTypes.arrayOf(PropTypes.number).isRequired,\n    onBrush: PropTypes.func.isRequired,\n    range: PropTypes.arrayOf(PropTypes.number).isRequired,\n    value: PropTypes.arrayOf(PropTypes.number).isRequired,\n    width: PropTypes.number.isRequired\n  };\n\n  componentDidMount() {\n    // We want the React app to respond to brush state and vice-versa\n    // but d3-brush fires the same events for both user-initiated brushing\n    // and programmatic brushing (brush.move). We need these flags to\n    // distinguish between the uses.\n    //\n    // We don't use state because that would trigger another `componentDidUpdate`\n\n    this.brushing = false;\n    this.moving = false;\n\n    this.root = select(this.rootContainer.current);\n    this.brush = brushX()\n      .on('start', () => {\n        this.brushing = true;\n      })\n      .on('brush', () => {\n        if (this.moving) {\n          return;\n        }\n        event.selection === null ? this._reset() : this._brush(event.selection);\n      })\n      .on('end', () => {\n        if (!this.moving && event.selection === null) {\n          this._reset();\n        }\n\n        this.brushing = false;\n        this.moving = false;\n      });\n\n    this.root.call(this.brush);\n  }\n\n  componentDidUpdate(prevProps) {\n    const {\n      value: [val0, val1],\n      width\n    } = this.props;\n    const [prevVal0, prevVal1] = prevProps.value;\n\n    if (prevProps.width !== width) {\n      // width change should not trigger this._brush\n      this.moving = true;\n      this.root.call(this.brush);\n      this._move(val0, val1);\n    }\n\n    if (!this.brushing && !this.moving) {\n      if (prevVal0 !== val0 || prevVal1 !== val1) {\n        this.moving = true;\n        this._move(val0, val1);\n      }\n    }\n  }\n\n  rootContainer = createRef();\n\n  _reset() {\n    const [minValue, maxValue] = this.props.range;\n    this._onBrush(minValue, maxValue);\n  }\n\n  _move(val0, val1) {\n    const {\n      domain: [min, max],\n      width\n    } = this.props;\n    const scale = x => ((x - min) * width) / (max - min);\n    this.brush.move(this.root, [scale(val0), scale(val1)]);\n  }\n\n  _brush([sel0, sel1]) {\n    const {\n      domain: [min, max],\n      width\n    } = this.props;\n    const invert = x => (x * (max - min)) / width + min;\n    this._onBrush(invert(sel0), invert(sel1));\n  }\n\n  _onBrush(val0, val1) {\n    const {\n      value: [currentVal0, currentVal1]\n    } = this.props;\n\n    if (currentVal0 === val0 && currentVal1 === val1) {\n      return;\n    }\n\n    this.props.onBrush(val0, val1);\n  }\n  render() {\n    return (\n      <StyledG\n        className=\"kg-range-slider__brush\"\n        // innerRef={comp => {\n        //   this.rootContainer = comp;\n        // }}\n        ref={this.rootContainer}\n      />\n    );\n  }\n}\n"]}