UNPKG

kepler.gl

Version:

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

141 lines (112 loc) 18.2 kB
"use strict"; var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault"); var _interopRequireWildcard = require("@babel/runtime/helpers/interopRequireWildcard"); Object.defineProperty(exports, "__esModule", { value: true }); exports.getTickFormat = getTickFormat; exports.getXAxis = getXAxis; exports.updateAxis = updateAxis; exports["default"] = void 0; var _taggedTemplateLiteral2 = _interopRequireDefault(require("@babel/runtime/helpers/taggedTemplateLiteral")); var _react = _interopRequireWildcard(require("react")); var _momentTimezone = _interopRequireDefault(require("moment-timezone")); var _propTypes = _interopRequireDefault(require("prop-types")); var _d3Scale = require("d3-scale"); var _d3Selection = require("d3-selection"); var _d3Axis = require("d3-axis"); var _styledComponents = _interopRequireDefault(require("styled-components")); var _dataUtils = require("../../utils/data-utils"); var _templateObject; var MIN_TICK_WIDTH_LARGE = 80; var MIN_TICK_WIDTH_SMALL = 50; var HEIGHT = 30; var TimeSliderContainer = _styledComponents["default"].svg(_templateObject || (_templateObject = (0, _taggedTemplateLiteral2["default"])(["\n pointer-events: none;\n position: absolute;\n top: 0;\n overflow: visible;\n margin-top: 6px;\n\n .axis text {\n font-size: ", ";\n fill: ", ";\n }\n\n .axis line,\n .axis path {\n fill: none;\n stroke: ", ";\n shape-rendering: crispEdges;\n stroke-width: 2;\n }\n\n .axis .domain {\n display: none;\n }\n\n .value {\n fill: ", ";\n font-size: ", ";\n\n &.start {\n text-anchor: start;\n }\n\n &.end {\n text-anchor: end;\n }\n }\n"])), function (props) { return props.theme.axisFontSize; }, function (props) { return props.theme.axisFontColor; }, function (props) { return props.theme.sliderBarBgd; }, function (props) { return props.theme.axisFontColor; }, function (props) { return props.theme.axisFontSize; }); var TICK_FORMATS = { millisecond: '.SSS', second: ':ss', minute: 'HH:ss', hour: 'HH A', day: 'ddd DD', week: 'MMM DD', month: 'MMM', year: 'YYYY' }; // timezone sensitive tick formatter based on moment // adapted based on d3 time scale tick format https://github.com/d3/d3-scale/blob/master/src/time.js#L59 function getTickFormat(timezone) { // date is js date object var toMoment = timezone ? function (date) { return (0, _momentTimezone["default"])(date).tz(timezone); } : _momentTimezone["default"]; var formatter = (0, _dataUtils.datetimeFormatter)(timezone); return function (date) { return (toMoment(date).startOf('second') < date ? formatter(TICK_FORMATS.millisecond) : toMoment(date).startOf('minute') < date ? formatter(TICK_FORMATS.second) : toMoment(date).startOf('hour') < date ? formatter(TICK_FORMATS.minute) : toMoment(date).startOf('day') < date ? formatter(TICK_FORMATS.hour) : toMoment(date).startOf('month') < date ? toMoment(date).startOf('isoWeek') < date ? formatter(TICK_FORMATS.day) : formatter(TICK_FORMATS.week) : toMoment(date).startOf('year') < date ? formatter(TICK_FORMATS.month) : formatter(TICK_FORMATS.year))(date); }; } // create a helper function so we can test it function getXAxis(domain, width, isEnlarged, timezone) { if (!Array.isArray(domain) || !domain.every(Number.isFinite)) { return null; } var scale = (0, _d3Scale.scaleUtc)().domain(domain).range([0, width]); if (!scale) { return null; } var ticks = Math.floor(width / (isEnlarged ? MIN_TICK_WIDTH_LARGE : MIN_TICK_WIDTH_SMALL)); var tickFormat = timezone ? getTickFormat(timezone) : null; var xAxis = (0, _d3Axis.axisBottom)(scale).ticks(ticks).tickSize(0).tickPadding(12); if (tickFormat) { xAxis.tickFormat(tickFormat); } return xAxis; } function updateAxis(xAxisRef, xAxis) { if (!xAxis) { return; } (0, _d3Selection.select)(xAxisRef.current).call(xAxis); } function TimeSliderMarkerFactory() { var TimeSliderMarker = function TimeSliderMarker(_ref) { var width = _ref.width, domain = _ref.domain, _ref$isEnlarged = _ref.isEnlarged, isEnlarged = _ref$isEnlarged === void 0 ? true : _ref$isEnlarged, _ref$height = _ref.height, height = _ref$height === void 0 ? HEIGHT : _ref$height, timezone = _ref.timezone; var xAxisRef = (0, _react.useRef)(null); var xAxis = (0, _react.useMemo)(function () { return getXAxis(domain, width, isEnlarged, timezone); }, [domain, width, isEnlarged, timezone]); (0, _react.useEffect)(function () { updateAxis(xAxisRef, xAxis); }, [xAxisRef, xAxis]); return /*#__PURE__*/_react["default"].createElement(TimeSliderContainer, { className: "time-slider-marker", width: width, height: height }, /*#__PURE__*/_react["default"].createElement("g", { className: "x axis", ref: xAxisRef, transform: "translate(0, 0)" })); }; TimeSliderMarker.propTypes = { domain: _propTypes["default"].arrayOf(_propTypes["default"].any).isRequired, width: _propTypes["default"].number.isRequired }; return /*#__PURE__*/_react["default"].memo(TimeSliderMarker); } var _default = TimeSliderMarkerFactory; exports["default"] = _default; //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/components/common/time-slider-marker.js"],"names":["MIN_TICK_WIDTH_LARGE","MIN_TICK_WIDTH_SMALL","HEIGHT","TimeSliderContainer","styled","svg","props","theme","axisFontSize","axisFontColor","sliderBarBgd","TICK_FORMATS","millisecond","second","minute","hour","day","week","month","year","getTickFormat","timezone","toMoment","date","tz","moment","formatter","startOf","getXAxis","domain","width","isEnlarged","Array","isArray","every","Number","isFinite","scale","range","ticks","Math","floor","tickFormat","xAxis","tickSize","tickPadding","updateAxis","xAxisRef","current","call","TimeSliderMarkerFactory","TimeSliderMarker","height","propTypes","PropTypes","arrayOf","any","isRequired","number","React","memo"],"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 React, {useRef, useEffect, useMemo} from 'react';\nimport moment from 'moment-timezone';\nimport PropTypes from 'prop-types';\nimport {scaleUtc} from 'd3-scale';\nimport {select} from 'd3-selection';\nimport {axisBottom} from 'd3-axis';\nimport styled from 'styled-components';\nimport {datetimeFormatter} from 'utils/data-utils';\n\nconst MIN_TICK_WIDTH_LARGE = 80;\nconst MIN_TICK_WIDTH_SMALL = 50;\nconst HEIGHT = 30;\n\nconst TimeSliderContainer = styled.svg`\n  pointer-events: none;\n  position: absolute;\n  top: 0;\n  overflow: visible;\n  margin-top: 6px;\n\n  .axis text {\n    font-size: ${props => props.theme.axisFontSize};\n    fill: ${props => props.theme.axisFontColor};\n  }\n\n  .axis line,\n  .axis path {\n    fill: none;\n    stroke: ${props => props.theme.sliderBarBgd};\n    shape-rendering: crispEdges;\n    stroke-width: 2;\n  }\n\n  .axis .domain {\n    display: none;\n  }\n\n  .value {\n    fill: ${props => props.theme.axisFontColor};\n    font-size: ${props => props.theme.axisFontSize};\n\n    &.start {\n      text-anchor: start;\n    }\n\n    &.end {\n      text-anchor: end;\n    }\n  }\n`;\n\nconst TICK_FORMATS = {\n  millisecond: '.SSS',\n  second: ':ss',\n  minute: 'HH:ss',\n  hour: 'HH A',\n  day: 'ddd DD',\n  week: 'MMM DD',\n  month: 'MMM',\n  year: 'YYYY'\n};\n\n// timezone sensitive tick formatter based on moment\n// adapted based on d3 time scale tick format https://github.com/d3/d3-scale/blob/master/src/time.js#L59\nexport function getTickFormat(timezone) {\n  // date is js date object\n  const toMoment = timezone ? date => moment(date).tz(timezone) : moment;\n  const formatter = datetimeFormatter(timezone);\n\n  return date =>\n    (toMoment(date).startOf('second') < date\n      ? formatter(TICK_FORMATS.millisecond)\n      : toMoment(date).startOf('minute') < date\n      ? formatter(TICK_FORMATS.second)\n      : toMoment(date).startOf('hour') < date\n      ? formatter(TICK_FORMATS.minute)\n      : toMoment(date).startOf('day') < date\n      ? formatter(TICK_FORMATS.hour)\n      : toMoment(date).startOf('month') < date\n      ? toMoment(date).startOf('isoWeek') < date\n        ? formatter(TICK_FORMATS.day)\n        : formatter(TICK_FORMATS.week)\n      : toMoment(date).startOf('year') < date\n      ? formatter(TICK_FORMATS.month)\n      : formatter(TICK_FORMATS.year))(date);\n}\n\n// create a helper function so we can test it\nexport function getXAxis(domain, width, isEnlarged, timezone) {\n  if (!Array.isArray(domain) || !domain.every(Number.isFinite)) {\n    return null;\n  }\n  const scale = scaleUtc()\n    .domain(domain)\n    .range([0, width]);\n  if (!scale) {\n    return null;\n  }\n\n  const ticks = Math.floor(width / (isEnlarged ? MIN_TICK_WIDTH_LARGE : MIN_TICK_WIDTH_SMALL));\n  const tickFormat = timezone ? getTickFormat(timezone) : null;\n  const xAxis = axisBottom(scale)\n    .ticks(ticks)\n    .tickSize(0)\n    .tickPadding(12);\n  if (tickFormat) {\n    xAxis.tickFormat(tickFormat);\n  }\n\n  return xAxis;\n}\n\nexport function updateAxis(xAxisRef, xAxis) {\n  if (!xAxis) {\n    return;\n  }\n\n  select(xAxisRef.current).call(xAxis);\n}\n\nfunction TimeSliderMarkerFactory() {\n  const TimeSliderMarker = ({width, domain, isEnlarged = true, height = HEIGHT, timezone}) => {\n    const xAxisRef = useRef(null);\n    const xAxis = useMemo(() => getXAxis(domain, width, isEnlarged, timezone), [\n      domain,\n      width,\n      isEnlarged,\n      timezone\n    ]);\n    useEffect(() => {\n      updateAxis(xAxisRef, xAxis);\n    }, [xAxisRef, xAxis]);\n    return (\n      <TimeSliderContainer className=\"time-slider-marker\" width={width} height={height}>\n        <g className=\"x axis\" ref={xAxisRef} transform=\"translate(0, 0)\" />\n      </TimeSliderContainer>\n    );\n  };\n\n  TimeSliderMarker.propTypes = {\n    domain: PropTypes.arrayOf(PropTypes.any).isRequired,\n    width: PropTypes.number.isRequired\n  };\n\n  return React.memo(TimeSliderMarker);\n}\n\nexport default TimeSliderMarkerFactory;\n"]}