kepler.gl
Version:
kepler.gl is a webgl based application to visualize large scale location data in the browser
183 lines (146 loc) • 17.9 kB
JavaScript
;
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.parseGeoJsonRawFeature = parseGeoJsonRawFeature;
exports.getGeojsonDataMaps = getGeojsonDataMaps;
exports.parseGeometryFromString = parseGeometryFromString;
exports.getGeojsonBounds = getGeojsonBounds;
exports.getGeojsonFeatureTypes = getGeojsonFeatureTypes;
exports.featureToDeckGlGeoType = void 0;
var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty"));
var _typeof2 = _interopRequireDefault(require("@babel/runtime/helpers/typeof"));
var _wellknown = _interopRequireDefault(require("wellknown"));
var _geojsonNormalize = _interopRequireDefault(require("@mapbox/geojson-normalize"));
var _bbox = _interopRequireDefault(require("@turf/bbox"));
var _dataUtils = require("../../utils/data-utils");
function ownKeys(object, enumerableOnly) { var keys = Object.keys(object); if (Object.getOwnPropertySymbols) { var symbols = Object.getOwnPropertySymbols(object); if (enumerableOnly) symbols = symbols.filter(function (sym) { return Object.getOwnPropertyDescriptor(object, sym).enumerable; }); keys.push.apply(keys, symbols); } return keys; }
function _objectSpread(target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i] != null ? arguments[i] : {}; if (i % 2) { ownKeys(Object(source), true).forEach(function (key) { (0, _defineProperty2["default"])(target, key, source[key]); }); } else if (Object.getOwnPropertyDescriptors) { Object.defineProperties(target, Object.getOwnPropertyDescriptors(source)); } else { ownKeys(Object(source)).forEach(function (key) { Object.defineProperty(target, key, Object.getOwnPropertyDescriptor(source, key)); }); } } return target; }
function parseGeoJsonRawFeature(rawFeature) {
if ((0, _typeof2["default"])(rawFeature) === 'object') {
// Support GeoJson feature as object
// probably need to normalize it as well
var normalized = (0, _geojsonNormalize["default"])(rawFeature);
if (!normalized || !Array.isArray(normalized.features)) {
// fail to normalize GeoJson
return null;
}
return normalized.features[0];
} else if (typeof rawFeature === 'string') {
return parseGeometryFromString(rawFeature);
} else if (Array.isArray(rawFeature)) {
// Support GeoJson LineString as an array of points
return {
type: 'Feature',
geometry: {
// why do we need to flip it...
coordinates: rawFeature.map(function (pts) {
return [pts[1], pts[0]];
}),
type: 'LineString'
}
};
}
return null;
}
/**
* Parse raw data to GeoJson feature
* @param dataContainer
* @param getFeature
* @returns {{}}
*/
function getGeojsonDataMaps(dataContainer, getFeature) {
var acceptableTypes = ['Point', 'MultiPoint', 'LineString', 'MultiLineString', 'Polygon', 'MultiPolygon', 'GeometryCollection'];
var dataToFeature = [];
for (var index = 0; index < dataContainer.numRows(); index++) {
var feature = parseGeoJsonRawFeature(getFeature({
index: index
}));
if (feature && feature.geometry && acceptableTypes.includes(feature.geometry.type)) {
var cleaned = _objectSpread(_objectSpread({}, feature), {}, {
// store index of the data in feature properties
properties: _objectSpread(_objectSpread({}, feature.properties), {}, {
index: index
})
});
dataToFeature[index] = cleaned;
} else {
dataToFeature[index] = null;
}
}
return dataToFeature;
}
/**
* Parse geojson from string
* @param {String} geoString
* @returns {null | Object} geojson object or null if failed
*/
function parseGeometryFromString(geoString) {
var parsedGeo; // try parse as geojson string
// {"type":"Polygon","coordinates":[[[-74.158491,40.83594]]]}
try {
parsedGeo = JSON.parse(geoString);
} catch (e) {// keep trying to parse
} // try parse as wkt
if (!parsedGeo) {
try {
parsedGeo = (0, _wellknown["default"])(geoString);
} catch (e) {
return null;
}
}
if (!parsedGeo) {
return null;
}
var normalized = (0, _geojsonNormalize["default"])(parsedGeo);
if (!normalized || !Array.isArray(normalized.features)) {
// fail to normalize geojson
return null;
}
return normalized.features[0];
}
function getGeojsonBounds() {
var features = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : [];
// 70 ms for 10,000 polygons
// here we only pick couple
var maxCount = 10000;
var samples = features.length > maxCount ? (0, _dataUtils.getSampleData)(features, maxCount) : features;
var nonEmpty = samples.filter(function (d) {
return d && d.geometry && d.geometry.coordinates && d.geometry.coordinates.length;
});
try {
return (0, _bbox["default"])({
type: 'FeatureCollection',
features: nonEmpty
});
} catch (e) {
return null;
}
}
var featureToDeckGlGeoType = {
Point: 'point',
MultiPoint: 'point',
LineString: 'line',
MultiLineString: 'line',
Polygon: 'polygon',
MultiPolygon: 'polygon'
};
/**
* Parse geojson from string
* @param {Array<Object>} allFeatures
* @returns {Object} mapping of feature type existence
*/
exports.featureToDeckGlGeoType = featureToDeckGlGeoType;
function getGeojsonFeatureTypes(allFeatures) {
var featureTypes = {};
for (var f = 0; f < allFeatures.length; f++) {
var feature = allFeatures[f];
var geoType = featureToDeckGlGeoType[feature && feature.geometry && feature.geometry.type];
if (geoType) {
featureTypes[geoType] = true;
}
}
return featureTypes;
}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/layers/geojson-layer/geojson-utils.js"],"names":["parseGeoJsonRawFeature","rawFeature","normalized","Array","isArray","features","parseGeometryFromString","type","geometry","coordinates","map","pts","getGeojsonDataMaps","dataContainer","getFeature","acceptableTypes","dataToFeature","index","numRows","feature","includes","cleaned","properties","geoString","parsedGeo","JSON","parse","e","getGeojsonBounds","maxCount","samples","length","nonEmpty","filter","d","featureToDeckGlGeoType","Point","MultiPoint","LineString","MultiLineString","Polygon","MultiPolygon","getGeojsonFeatureTypes","allFeatures","featureTypes","f","geoType"],"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 wktParser from 'wellknown';\nimport normalize from '@mapbox/geojson-normalize';\nimport bbox from '@turf/bbox';\n\nimport {getSampleData} from 'utils/data-utils';\n\nexport function parseGeoJsonRawFeature(rawFeature) {\n  if (typeof rawFeature === 'object') {\n    // Support GeoJson feature as object\n    // probably need to normalize it as well\n    const normalized = normalize(rawFeature);\n    if (!normalized || !Array.isArray(normalized.features)) {\n      // fail to normalize GeoJson\n      return null;\n    }\n\n    return normalized.features[0];\n  } else if (typeof rawFeature === 'string') {\n    return parseGeometryFromString(rawFeature);\n  } else if (Array.isArray(rawFeature)) {\n    // Support GeoJson  LineString as an array of points\n    return {\n      type: 'Feature',\n      geometry: {\n        // why do we need to flip it...\n        coordinates: rawFeature.map(pts => [pts[1], pts[0]]),\n        type: 'LineString'\n      }\n    };\n  }\n\n  return null;\n}\n/**\n * Parse raw data to GeoJson feature\n * @param dataContainer\n * @param getFeature\n * @returns {{}}\n */\nexport function getGeojsonDataMaps(dataContainer, getFeature) {\n  const acceptableTypes = [\n    'Point',\n    'MultiPoint',\n    'LineString',\n    'MultiLineString',\n    'Polygon',\n    'MultiPolygon',\n    'GeometryCollection'\n  ];\n\n  const dataToFeature = [];\n\n  for (let index = 0; index < dataContainer.numRows(); index++) {\n    const feature = parseGeoJsonRawFeature(getFeature({index}));\n\n    if (feature && feature.geometry && acceptableTypes.includes(feature.geometry.type)) {\n      const cleaned = {\n        ...feature,\n        // store index of the data in feature properties\n        properties: {\n          ...feature.properties,\n          index\n        }\n      };\n\n      dataToFeature[index] = cleaned;\n    } else {\n      dataToFeature[index] = null;\n    }\n  }\n\n  return dataToFeature;\n}\n\n/**\n * Parse geojson from string\n * @param {String} geoString\n * @returns {null | Object} geojson object or null if failed\n */\nexport function parseGeometryFromString(geoString) {\n  let parsedGeo;\n\n  // try parse as geojson string\n  // {\"type\":\"Polygon\",\"coordinates\":[[[-74.158491,40.83594]]]}\n  try {\n    parsedGeo = JSON.parse(geoString);\n  } catch (e) {\n    // keep trying to parse\n  }\n\n  // try parse as wkt\n  if (!parsedGeo) {\n    try {\n      parsedGeo = wktParser(geoString);\n    } catch (e) {\n      return null;\n    }\n  }\n\n  if (!parsedGeo) {\n    return null;\n  }\n\n  const normalized = normalize(parsedGeo);\n\n  if (!normalized || !Array.isArray(normalized.features)) {\n    // fail to normalize geojson\n    return null;\n  }\n\n  return normalized.features[0];\n}\n\nexport function getGeojsonBounds(features = []) {\n  // 70 ms for 10,000 polygons\n  // here we only pick couple\n  const maxCount = 10000;\n  const samples = features.length > maxCount ? getSampleData(features, maxCount) : features;\n\n  const nonEmpty = samples.filter(\n    d => d && d.geometry && d.geometry.coordinates && d.geometry.coordinates.length\n  );\n\n  try {\n    return bbox({\n      type: 'FeatureCollection',\n      features: nonEmpty\n    });\n  } catch (e) {\n    return null;\n  }\n}\n\nexport const featureToDeckGlGeoType = {\n  Point: 'point',\n  MultiPoint: 'point',\n  LineString: 'line',\n  MultiLineString: 'line',\n  Polygon: 'polygon',\n  MultiPolygon: 'polygon'\n};\n\n/**\n * Parse geojson from string\n * @param {Array<Object>} allFeatures\n * @returns {Object} mapping of feature type existence\n */\nexport function getGeojsonFeatureTypes(allFeatures) {\n  const featureTypes = {};\n  for (let f = 0; f < allFeatures.length; f++) {\n    const feature = allFeatures[f];\n    const geoType = featureToDeckGlGeoType[feature && feature.geometry && feature.geometry.type];\n    if (geoType) {\n      featureTypes[geoType] = true;\n    }\n  }\n\n  return featureTypes;\n}\n"]}