kepler.gl
Version:
kepler.gl is a webgl based application to visualize large scale location data in the browser
116 lines (113 loc) • 20 kB
JavaScript
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
var _typeof = require("@babel/runtime/helpers/typeof");
Object.defineProperty(exports, "__esModule", {
value: true
});
exports["default"] = void 0;
exports.getTickFormat = getTickFormat;
exports.getXAxis = getXAxis;
exports.updateAxis = updateAxis;
var _taggedTemplateLiteral2 = _interopRequireDefault(require("@babel/runtime/helpers/taggedTemplateLiteral"));
var _react = _interopRequireWildcard(require("react"));
var _momentTimezone = _interopRequireDefault(require("moment-timezone"));
var _d3Scale = require("d3-scale");
var _d3Selection = require("d3-selection");
var _d3Axis = require("d3-axis");
var _styledComponents = _interopRequireDefault(require("styled-components"));
var _utils = require("@kepler.gl/utils");
var _templateObject; // SPDX-License-Identifier: MIT
// Copyright contributors to the kepler.gl project
function _getRequireWildcardCache(e) { if ("function" != typeof WeakMap) return null; var r = new WeakMap(), t = new WeakMap(); return (_getRequireWildcardCache = function _getRequireWildcardCache(e) { return e ? t : r; })(e); }
function _interopRequireWildcard(e, r) { if (!r && e && e.__esModule) return e; if (null === e || "object" != _typeof(e) && "function" != typeof e) return { "default": e }; var t = _getRequireWildcardCache(r); if (t && t.has(e)) return t.get(e); var n = { __proto__: null }, a = Object.defineProperty && Object.getOwnPropertyDescriptor; for (var u in e) if ("default" !== u && {}.hasOwnProperty.call(e, u)) { var i = a ? Object.getOwnPropertyDescriptor(e, u) : null; i && (i.get || i.set) ? Object.defineProperty(n, u, i) : n[u] = e[u]; } return n["default"] = e, t && t.set(e, n), n; }
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 stroke: ", ";\n shape-rendering: crispEdges;\n stroke-width: 1;\n }\n\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.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:mm',
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, _utils.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(4).tickPadding(4);
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)"
}));
};
return /*#__PURE__*/_react["default"].memo(TimeSliderMarker);
}
var _default = exports["default"] = TimeSliderMarkerFactory;
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_react","_interopRequireWildcard","require","_momentTimezone","_interopRequireDefault","_d3Scale","_d3Selection","_d3Axis","_styledComponents","_utils","_templateObject","_getRequireWildcardCache","e","WeakMap","r","t","__esModule","_typeof","has","get","n","__proto__","a","Object","defineProperty","getOwnPropertyDescriptor","u","hasOwnProperty","call","i","set","MIN_TICK_WIDTH_LARGE","MIN_TICK_WIDTH_SMALL","HEIGHT","TimeSliderContainer","styled","svg","_taggedTemplateLiteral2","props","theme","axisFontSize","axisFontColor","sliderBarBgd","TICK_FORMATS","millisecond","second","minute","hour","day","week","month","year","getTickFormat","timezone","toMoment","date","moment","tz","formatter","datetimeFormatter","startOf","getXAxis","domain","width","isEnlarged","Array","isArray","every","Number","isFinite","scale","scaleUtc","range","ticks","Math","floor","tickFormat","xAxis","axisBottom","tickSize","tickPadding","updateAxis","xAxisRef","select","current","TimeSliderMarkerFactory","TimeSliderMarker","_ref","_ref$isEnlarged","_ref$height","height","useRef","useMemo","useEffect","createElement","className","ref","transform","React","memo","_default","exports"],"sources":["../../src/common/time-slider-marker.tsx"],"sourcesContent":["// SPDX-License-Identifier: MIT\n// Copyright contributors to the kepler.gl project\n\nimport React, {useRef, useEffect, useMemo} from 'react';\nimport moment from 'moment-timezone';\nimport {NumberValue, scaleUtc} from 'd3-scale';\nimport {select} from 'd3-selection';\nimport {axisBottom} from 'd3-axis';\nimport styled from 'styled-components';\nimport {datetimeFormatter} from '@kepler.gl/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    stroke: ${props => props.theme.axisFontColor};\n    shape-rendering: crispEdges;\n    stroke-width: 1;\n  }\n\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:mm',\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: string) {\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(\n  domain: Date[] | NumberValue[],\n  width: number,\n  isEnlarged: boolean,\n  timezone: string\n) {\n  if (!Array.isArray(domain) || !domain.every(Number.isFinite)) {\n    return null;\n  }\n  const scale = scaleUtc().domain(domain).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).ticks(ticks).tickSize(4).tickPadding(4);\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\ninterface TimeSliderMarkerProps {\n  width: number;\n  domain: Date[] | NumberValue[];\n  isEnlarged?: boolean;\n  height?: number;\n  timezone: string;\n}\n\nfunction TimeSliderMarkerFactory() {\n  const TimeSliderMarker = ({\n    width,\n    domain,\n    isEnlarged = true,\n    height = HEIGHT,\n    timezone\n  }: TimeSliderMarkerProps) => {\n    const xAxisRef = useRef(null);\n    const xAxis = useMemo(\n      () => getXAxis(domain, width, isEnlarged, timezone),\n      [domain, width, isEnlarged, 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  return React.memo(TimeSliderMarker);\n}\n\nexport default TimeSliderMarkerFactory;\n"],"mappings":"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","ignoreList":[]}
;