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
JavaScript
;
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"]}