kepler.gl
Version:
kepler.gl is a webgl based application to visualize large scale location data in the browser
138 lines (131 loc) • 18.2 kB
JavaScript
;
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.coordHasLength4 = coordHasLength4;
exports.getAnimationDomainFromTimestamps = getAnimationDomainFromTimestamps;
exports.isTripGeoJsonField = isTripGeoJsonField;
exports.parseTripGeoJsonTimestamp = parseTripGeoJsonTimestamp;
var _geojsonUtils = require("../geojson-layer/geojson-utils");
var _utils = require("@kepler.gl/utils");
var _commonUtils = require("@kepler.gl/common-utils");
// SPDX-License-Identifier: MIT
// Copyright contributors to the kepler.gl project
/**
* Parse geojson from string
* @param {array} samples feature object values
* @returns whether the geometry coordinates has length of 4
*/
function coordHasLength4(samples) {
var hasLength4 = true;
for (var i = 0; i < samples.length; i += 1) {
var _samples$i;
hasLength4 = Array.isArray((_samples$i = samples[i]) === null || _samples$i === void 0 || (_samples$i = _samples$i.geometry) === null || _samples$i === void 0 ? void 0 : _samples$i.coordinates) && !samples[i].geometry.coordinates.find(function (c) {
return c.length < 4;
});
if (!hasLength4) {
break;
}
}
return hasLength4;
}
/**
* Check if geojson features are trip layer animatable by meeting 3 conditions
* @param dataContainer geojson feature objects container
* @param {object} field array of geojson feature objects
* @returns whether it is trip layer animatable
*/
function isTripGeoJsonField(dataContainer, field) {
if (dataContainer.numRows() < 1) {
return false;
}
var maxCount = 10000;
var sampleRawFeatures = dataContainer.numRows() > maxCount ? (0, _utils.getSampleContainerData)(dataContainer, maxCount) : dataContainer;
var features = sampleRawFeatures.mapIndex(field.valueAccessor).map(_geojsonUtils.parseGeoJsonRawFeature).filter(_commonUtils.notNullorUndefined);
var featureTypes = (0, _geojsonUtils.getGeojsonFeatureTypes)(features);
// condition 1: contain line string
if (!featureTypes.line) {
return false;
}
// condition 2:sample line strings contain 4 coordinates
if (!coordHasLength4(features)) {
return false;
}
var geom = features[0].geometry;
if (!geom || geom.type === 'GeometryCollection' || !('coordinates' in geom)) {
return false;
}
var tsHolder = geom.coordinates.map(function (coord) {
return String(coord[3]);
});
return Boolean((0, _commonUtils.containValidTime)(tsHolder));
}
/**
* Get unix timestamp from animatable geojson for deck.gl trip layer
* @param dataToFeature array of geojson feature objects, can be null
* @returns
*/
function parseTripGeoJsonTimestamp(dataToFeature) {
// Analyze type based on coordinates of the 1st lineString
// select a sample trip to analyze time format
var empty = {
dataToTimeStamp: [],
animationDomain: null
};
var sampleTrip = dataToFeature.find(function (f) {
var _f$geometry;
return (f === null || f === void 0 || (_f$geometry = f.geometry) === null || _f$geometry === void 0 || (_f$geometry = _f$geometry.coordinates) === null || _f$geometry === void 0 || (_f$geometry = _f$geometry[0]) === null || _f$geometry === void 0 ? void 0 : _f$geometry.length) > 3;
});
// if no valid geometry
if (!sampleTrip) {
return empty;
}
var analyzedType = (0, _commonUtils.containValidTime)(sampleTrip.geometry.coordinates.map(function (coord) {
return coord[3];
}));
if (!analyzedType) {
return empty;
}
var format = analyzedType.format;
var getTimeValue = function getTimeValue(coord) {
return coord && (0, _commonUtils.notNullorUndefined)(coord[3]) ? (0, _utils.timeToUnixMilli)(coord[3], format) : null;
};
var dataToTimeStamp = dataToFeature.map(function (f) {
return f && f.geometry && Array.isArray(f.geometry.coordinates) ? f.geometry.coordinates.map(getTimeValue) : null;
});
var animationDomain = getAnimationDomainFromTimestamps(dataToTimeStamp);
return {
dataToTimeStamp: dataToTimeStamp,
animationDomain: animationDomain
};
}
function findMinFromSorted(list) {
// check if list is null since the default value [] will only be applied when the param is undefined
return (list === null || list === void 0 ? void 0 : list.find(function (d) {
return (0, _commonUtils.notNullorUndefined)(d) && Number.isFinite(d);
})) || null;
}
function findMaxFromSorted() {
var list = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : [];
var i = list.length - 1;
while (i > 0) {
if ((0, _commonUtils.notNullorUndefined)(list[i]) && Number.isFinite(list[i])) {
return list[i];
}
i--;
}
return null;
}
function getAnimationDomainFromTimestamps() {
var dataToTimeStamp = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : [];
return dataToTimeStamp.reduce(function (accu, tss) {
var tsMin = findMinFromSorted(tss);
var tsMax = findMaxFromSorted(tss);
if ((0, _commonUtils.notNullorUndefined)(tsMin) && (0, _commonUtils.notNullorUndefined)(tsMax) && Number.isFinite(tsMin) && Number.isFinite(tsMax)) {
accu[0] = Math.min(accu[0], tsMin);
accu[1] = Math.max(accu[1], tsMax);
}
return accu;
}, [Infinity, -Infinity]);
}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_geojsonUtils","require","_utils","_commonUtils","coordHasLength4","samples","hasLength4","i","length","_samples$i","Array","isArray","geometry","coordinates","find","c","isTripGeoJsonField","dataContainer","field","numRows","maxCount","sampleRawFeatures","getSampleContainerData","features","mapIndex","valueAccessor","map","parseGeoJsonRawFeature","filter","notNullorUndefined","featureTypes","getGeojsonFeatureTypes","line","geom","type","tsHolder","coord","String","Boolean","containValidTime","parseTripGeoJsonTimestamp","dataToFeature","empty","dataToTimeStamp","animationDomain","sampleTrip","f","_f$geometry","analyzedType","format","getTimeValue","timeToUnixMilli","getAnimationDomainFromTimestamps","findMinFromSorted","list","d","Number","isFinite","findMaxFromSorted","arguments","undefined","reduce","accu","tss","tsMin","tsMax","Math","min","max","Infinity"],"sources":["../../src/trip-layer/trip-utils.ts"],"sourcesContent":["// SPDX-License-Identifier: MIT\n// Copyright contributors to the kepler.gl project\n\nimport {parseGeoJsonRawFeature, getGeojsonFeatureTypes} from '../geojson-layer/geojson-utils';\nimport {DataContainerInterface, getSampleContainerData, timeToUnixMilli} from '@kepler.gl/utils';\nimport {containValidTime, notNullorUndefined} from '@kepler.gl/common-utils';\nimport {Feature} from '@turf/helpers';\nimport {GeoJsonProperties, Geometry} from 'geojson';\n\n/**\n * Parse geojson from string\n * @param {array} samples feature object values\n * @returns whether the geometry coordinates has length of 4\n */\nexport function coordHasLength4(samples): boolean {\n  let hasLength4 = true;\n  for (let i = 0; i < samples.length; i += 1) {\n    hasLength4 =\n      Array.isArray(samples[i]?.geometry?.coordinates) &&\n      !samples[i].geometry.coordinates.find(c => c.length < 4);\n\n    if (!hasLength4) {\n      break;\n    }\n  }\n  return hasLength4;\n}\n\n/**\n * Check if geojson features are trip layer animatable by meeting 3 conditions\n * @param dataContainer geojson feature objects container\n * @param {object} field array of geojson feature objects\n * @returns whether it is trip layer animatable\n */\nexport function isTripGeoJsonField(dataContainer: DataContainerInterface, field): boolean {\n  if (dataContainer.numRows() < 1) {\n    return false;\n  }\n\n  const maxCount = 10000;\n  const sampleRawFeatures =\n    dataContainer.numRows() > maxCount\n      ? getSampleContainerData(dataContainer, maxCount)\n      : dataContainer;\n\n  const features: Feature<Geometry, GeoJsonProperties>[] = sampleRawFeatures\n    .mapIndex(field.valueAccessor)\n    .map(parseGeoJsonRawFeature)\n    .filter(notNullorUndefined);\n  const featureTypes = getGeojsonFeatureTypes(features);\n\n  // condition 1: contain line string\n  if (!featureTypes.line) {\n    return false;\n  }\n\n  // condition 2:sample line strings contain 4 coordinates\n  if (!coordHasLength4(features)) {\n    return false;\n  }\n\n  const geom = features[0].geometry;\n  if (!geom || geom.type === 'GeometryCollection' || !('coordinates' in geom)) {\n    return false;\n  }\n  const tsHolder = (geom.coordinates as number[][]).map(coord => String(coord[3]));\n\n  return Boolean(containValidTime(tsHolder));\n}\n\n/**\n * Get unix timestamp from animatable geojson for deck.gl trip layer\n * @param dataToFeature array of geojson feature objects, can be null\n * @returns\n */\nexport function parseTripGeoJsonTimestamp(dataToFeature: any[]) {\n  // Analyze type based on coordinates of the 1st lineString\n  // select a sample trip to analyze time format\n  const empty = {dataToTimeStamp: [], animationDomain: null};\n  const sampleTrip = dataToFeature.find(f => f?.geometry?.coordinates?.[0]?.length > 3);\n\n  // if no valid geometry\n  if (!sampleTrip) {\n    return empty;\n  }\n\n  const analyzedType = containValidTime(sampleTrip.geometry.coordinates.map(coord => coord[3]));\n\n  if (!analyzedType) {\n    return empty;\n  }\n\n  const {format} = analyzedType;\n  const getTimeValue = coord =>\n    coord && notNullorUndefined(coord[3]) ? timeToUnixMilli(coord[3], format) : null;\n\n  const dataToTimeStamp: number[][] = dataToFeature.map(f =>\n    f && f.geometry && Array.isArray(f.geometry.coordinates)\n      ? f.geometry.coordinates.map(getTimeValue)\n      : null\n  );\n\n  const animationDomain = getAnimationDomainFromTimestamps(dataToTimeStamp);\n\n  return {dataToTimeStamp, animationDomain};\n}\n\nfunction findMinFromSorted(list: number[]) {\n  // check if list is null since the default value [] will only be applied when the param is undefined\n  return list?.find(d => notNullorUndefined(d) && Number.isFinite(d)) || null;\n}\n\nfunction findMaxFromSorted(list: number[] = []) {\n  let i = list.length - 1;\n  while (i > 0) {\n    if (notNullorUndefined(list[i]) && Number.isFinite(list[i])) {\n      return list[i];\n    }\n    i--;\n  }\n  return null;\n}\n\nexport function getAnimationDomainFromTimestamps(dataToTimeStamp: number[][] = []) {\n  return dataToTimeStamp.reduce(\n    (accu: [number, number], tss) => {\n      const tsMin = findMinFromSorted(tss);\n      const tsMax = findMaxFromSorted(tss);\n      if (\n        notNullorUndefined(tsMin) &&\n        notNullorUndefined(tsMax) &&\n        Number.isFinite(tsMin) &&\n        Number.isFinite(tsMax)\n      ) {\n        accu[0] = Math.min(accu[0], tsMin);\n        accu[1] = Math.max(accu[1], tsMax);\n      }\n      return accu;\n    },\n    [Infinity, -Infinity]\n  );\n}\n"],"mappings":"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","ignoreList":[]}