UNPKG

kepler.gl

Version:

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

200 lines (162 loc) 20.8 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.coordHasLength4 = coordHasLength4; exports.containValidTime = containValidTime; exports.isTripGeoJsonField = isTripGeoJsonField; exports.parseTripGeoJsonTimestamp = parseTripGeoJsonTimestamp; exports.getAnimationDomainFromTimestamps = getAnimationDomainFromTimestamps; var _typeAnalyzer = require("type-analyzer"); var _dataUtils = require("../../utils/data-utils"); var _geojsonUtils = require("../geojson-layer/geojson-utils"); // Copyright (c) 2020 Uber Technologies, Inc. // // Permission is hereby granted, free of charge, to any person obtaining a copy // of this software and associated documentation files (the "Software"), to deal // in the Software without restriction, including without limitation the rights // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell // copies of the Software, and to permit persons to whom the Software is // furnished to do so, subject to the following conditions: // // The above copyright notice and this permission notice shall be included in // all copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN // THE SOFTWARE. /** * Parse geojson from string * @param {array} geojson feature object values * @returns {boolean} whether the geometry coordinates has length of 4 */ function coordHasLength4(samples) { var hasLength4 = true; for (var i = 0; i < samples.length; i += 1) { hasLength4 = !samples[i].geometry.coordinates.find(function (c) { return c.length < 4; }); if (!hasLength4) { break; } } return hasLength4; } /** * Check whether geojson linestring's 4th coordinate is 1) not timestamp 2) unix time stamp 3) real date time * @param {array} data array to be tested if its elements are timestamp * @returns {string} the type of timestamp: unix/datetime/invalid(not timestamp) */ function containValidTime(timestamps) { var formattedTimeStamps = timestamps.map(function (ts) { return { ts: ts }; }); var ignoredDataTypes = Object.keys(_typeAnalyzer.DATA_TYPES).filter(function (type) { return ![_typeAnalyzer.DATA_TYPES.TIME, _typeAnalyzer.DATA_TYPES.DATETIME].includes(type); }); // ignore all types but TIME to improve performance var analyzedType = _typeAnalyzer.Analyzer.computeColMeta(formattedTimeStamps, [], { ignoredDataTypes: ignoredDataTypes })[0]; if (!analyzedType || analyzedType.category !== 'TIME') { return false; } return analyzedType; } /** * Check if geojson features are trip layer animatable by meeting 3 conditions * @param {array} features array of geojson feature objects * @returns {boolean} whether it is trip layer animatable */ function isTripGeoJsonField() { var allData = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : []; var field = arguments.length > 1 ? arguments[1] : undefined; if (!allData.length) { return false; } var getValue = function getValue(d) { return d[field.tableFieldIndex - 1]; }; var maxCount = 10000; var sampleRawFeatures = allData.length > maxCount ? (0, _dataUtils.getSampleData)(allData, maxCount, getValue) : allData.map(getValue); var features = sampleRawFeatures.map(_geojsonUtils.parseGeoJsonRawFeature).filter(function (f) { return f; }); 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; } // condition 3:the 4th coordinate of the first feature line strings is valid time var tsHolder = features[0].geometry.coordinates.map(function (coord) { return coord[3]; }); return Boolean(containValidTime(tsHolder)); } /** * Get unix timestamp from animatable geojson for deck.gl trip layer * @param {Array<Object>} dataToFeature array of geojson feature objects, can be null * @returns {Array<Number>} unix timestamp in milliseconds */ 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) { return f && f.geometry && f.geometry.coordinates && f.geometry.coordinates.length >= 3; }); // if no valid geometry if (!sampleTrip) { return empty; } var analyzedType = 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, _dataUtils.notNullorUndefined)(coord[3]) ? (0, _dataUtils.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() { var list = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : []; return list.find(_dataUtils.notNullorUndefined) || null; } function findMaxFromSorted() { var list = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : []; var i = list.length - 1; while (i > 0) { if ((0, _dataUtils.notNullorUndefined)(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) { accu[0] = Math.min(accu[0], findMinFromSorted(tss)); accu[1] = Math.max(accu[1], findMaxFromSorted(tss)); return accu; }, [Infinity, -Infinity]); } //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/layers/trip-layer/trip-utils.js"],"names":["coordHasLength4","samples","hasLength4","i","length","geometry","coordinates","find","c","containValidTime","timestamps","formattedTimeStamps","map","ts","ignoredDataTypes","Object","keys","DATA_TYPES","filter","type","TIME","DATETIME","includes","analyzedType","Analyzer","computeColMeta","category","isTripGeoJsonField","allData","field","getValue","d","tableFieldIndex","maxCount","sampleRawFeatures","features","parseGeoJsonRawFeature","f","featureTypes","line","tsHolder","coord","Boolean","parseTripGeoJsonTimestamp","dataToFeature","empty","dataToTimeStamp","animationDomain","sampleTrip","format","getTimeValue","Array","isArray","getAnimationDomainFromTimestamps","findMinFromSorted","list","notNullorUndefined","findMaxFromSorted","reduce","accu","tss","Math","min","max","Infinity"],"mappings":"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","sourcesContent":["// Copyright (c) 2020 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 {Analyzer, DATA_TYPES} from 'type-analyzer';\n\nimport {getSampleData, timeToUnixMilli, notNullorUndefined} from 'utils/data-utils';\n\nimport {parseGeoJsonRawFeature, getGeojsonFeatureTypes} from 'layers/geojson-layer/geojson-utils';\n\n/**\n * Parse geojson from string\n * @param {array} geojson feature object values\n * @returns {boolean} whether the geometry coordinates has length of 4\n */\nexport function coordHasLength4(samples) {\n  let hasLength4 = true;\n  for (let i = 0; i < samples.length; i += 1) {\n    hasLength4 = !samples[i].geometry.coordinates.find(c => c.length < 4);\n    if (!hasLength4) {\n      break;\n    }\n  }\n  return hasLength4;\n}\n\n/**\n * Check whether geojson linestring's 4th coordinate is 1) not timestamp 2) unix time stamp 3) real date time\n * @param {array} data array to be tested if its elements are timestamp\n * @returns {string} the type of timestamp: unix/datetime/invalid(not timestamp)\n */\n\nexport function containValidTime(timestamps) {\n  const formattedTimeStamps = timestamps.map(ts => ({ts}));\n  const ignoredDataTypes = Object.keys(DATA_TYPES).filter(\n    type => ![DATA_TYPES.TIME, DATA_TYPES.DATETIME].includes(type)\n  );\n\n  // ignore all types but TIME to improve performance\n  const analyzedType = Analyzer.computeColMeta(formattedTimeStamps, [], {ignoredDataTypes})[0];\n\n  if (!analyzedType || analyzedType.category !== 'TIME') {\n    return false;\n  }\n  return analyzedType;\n}\n\n/**\n * Check if geojson features are trip layer animatable by meeting 3 conditions\n * @param {array} features array of geojson feature objects\n * @returns {boolean} whether it is trip layer animatable\n */\nexport function isTripGeoJsonField(allData = [], field) {\n  if (!allData.length) {\n    return false;\n  }\n  const getValue = d => d[field.tableFieldIndex - 1];\n  const maxCount = 10000;\n  const sampleRawFeatures =\n    allData.length > maxCount ? getSampleData(allData, maxCount, getValue) : allData.map(getValue);\n\n  const features = sampleRawFeatures.map(parseGeoJsonRawFeature).filter(f => f);\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  // condition 3:the 4th coordinate of the first feature line strings is valid time\n  const tsHolder = features[0].geometry.coordinates.map(coord => 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 {Array<Object>} dataToFeature array of geojson feature objects, can be null\n * @returns {Array<Number>} unix timestamp in milliseconds\n */\nexport function parseTripGeoJsonTimestamp(dataToFeature) {\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(\n    f => f && f.geometry && f.geometry.coordinates && f.geometry.coordinates.length >= 3\n  );\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 = 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 = []) {\n  return list.find(notNullorUndefined) || null;\n}\n\nfunction findMaxFromSorted(list = []) {\n  let i = list.length - 1;\n  while (i > 0) {\n    if (notNullorUndefined(list[i])) {\n      return list[i];\n    }\n    i--;\n  }\n  return null;\n}\n\nexport function getAnimationDomainFromTimestamps(dataToTimeStamp = []) {\n  return dataToTimeStamp.reduce(\n    (accu, tss) => {\n      accu[0] = Math.min(accu[0], findMinFromSorted(tss));\n      accu[1] = Math.max(accu[1], findMaxFromSorted(tss));\n      return accu;\n    },\n    [Infinity, -Infinity]\n  );\n}\n"]}