kepler.gl
Version:
kepler.gl is a webgl based application to visualize large scale location data in the browser
251 lines (214 loc) • 23.5 kB
JavaScript
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
Object.defineProperty(exports, "__esModule", {
value: true
});
exports["default"] = exports.scenegraphVisConfigs = exports.scenegraphPosAccessor = exports.scenegraphOptionalColumns = exports.scenegraphRequiredColumns = void 0;
var _classCallCheck2 = _interopRequireDefault(require("@babel/runtime/helpers/classCallCheck"));
var _createClass2 = _interopRequireDefault(require("@babel/runtime/helpers/createClass"));
var _possibleConstructorReturn2 = _interopRequireDefault(require("@babel/runtime/helpers/possibleConstructorReturn"));
var _getPrototypeOf2 = _interopRequireDefault(require("@babel/runtime/helpers/getPrototypeOf"));
var _inherits2 = _interopRequireDefault(require("@babel/runtime/helpers/inherits"));
var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty"));
var _meshLayers = require("@deck.gl/mesh-layers");
var _core = require("@loaders.gl/core");
var _gltf = require("@loaders.gl/gltf");
var _baseLayer = _interopRequireDefault(require("../base-layer"));
var _scenegraphLayerIcon = _interopRequireDefault(require("./scenegraph-layer-icon"));
var _scenegraphInfoModal = _interopRequireDefault(require("./scenegraph-info-modal"));
var _layerFactory = require("../layer-factory");
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; }
var scenegraphRequiredColumns = ['lat', 'lng'];
exports.scenegraphRequiredColumns = scenegraphRequiredColumns;
var scenegraphOptionalColumns = ['altitude'];
exports.scenegraphOptionalColumns = scenegraphOptionalColumns;
function fetch(url, _ref) {
var propName = _ref.propName,
layer = _ref.layer;
if (propName === 'scenegraph') {
return (0, _core.load)(url, _gltf.GLTFLoader, layer.getLoadOptions());
}
return fetch(url).then(function (response) {
return response.json();
});
}
var scenegraphPosAccessor = function scenegraphPosAccessor(_ref2) {
var lat = _ref2.lat,
lng = _ref2.lng,
altitude = _ref2.altitude;
return function (d) {
return [// lng
d.data[lng.fieldIdx], // lat
d.data[lat.fieldIdx], // altitude
altitude && altitude.fieldIdx > -1 ? d.data[altitude.fieldIdx] : 0];
};
};
exports.scenegraphPosAccessor = scenegraphPosAccessor;
var scenegraphVisConfigs = {
opacity: 'opacity',
colorRange: 'colorRange',
//
sizeScale: 'sizeScale',
angleX: _objectSpread({}, _layerFactory.LAYER_VIS_CONFIGS.angle, {
property: 'angleX',
label: 'angle X'
}),
angleY: _objectSpread({}, _layerFactory.LAYER_VIS_CONFIGS.angle, {
property: 'angleY',
label: 'angle Y'
}),
angleZ: _objectSpread({}, _layerFactory.LAYER_VIS_CONFIGS.angle, {
property: 'angleZ',
defaultValue: 90,
label: 'angle Z'
})
};
exports.scenegraphVisConfigs = scenegraphVisConfigs;
var DEFAULT_MODEL = 'https://raw.githubusercontent.com/KhronosGroup/glTF-Sample-Models/master/2.0/Duck/glTF-Binary/Duck.glb';
var DEFAULT_TRANSITION = [0, 0, 0];
var DEFAULT_SCALE = [1, 1, 1];
var DEFAULT_COLOR = [255, 255, 255, 255];
var ScenegraphLayer =
/*#__PURE__*/
function (_Layer) {
(0, _inherits2["default"])(ScenegraphLayer, _Layer);
function ScenegraphLayer(props) {
var _this;
(0, _classCallCheck2["default"])(this, ScenegraphLayer);
_this = (0, _possibleConstructorReturn2["default"])(this, (0, _getPrototypeOf2["default"])(ScenegraphLayer).call(this, props));
_this.registerVisConfig(scenegraphVisConfigs);
_this.getPositionAccessor = function () {
return scenegraphPosAccessor(_this.config.columns);
}; // prepare layer info modal
_this._layerInfoModal = (0, _scenegraphInfoModal["default"])();
return _this;
}
(0, _createClass2["default"])(ScenegraphLayer, [{
key: "calculateDataAttribute",
value: function calculateDataAttribute(_ref3, getPosition) {
var allData = _ref3.allData,
filteredIndex = _ref3.filteredIndex;
var data = [];
for (var i = 0; i < filteredIndex.length; i++) {
var index = filteredIndex[i];
var pos = getPosition({
data: allData[index]
}); // if doesn't have point lat or lng, do not add the point
// deck.gl can't handle position = null
if (pos.every(Number.isFinite)) {
data.push({
data: allData[index],
position: pos,
// index is important for filter
index: index
});
}
}
return data;
}
}, {
key: "formatLayerData",
value: function formatLayerData(datasets, oldLayerData) {
var gpuFilter = datasets[this.config.dataId].gpuFilter;
var _this$updateData = this.updateData(datasets, oldLayerData),
data = _this$updateData.data;
var getPosition = this.getPositionAccessor();
return {
data: data,
getPosition: getPosition,
getFilterValue: gpuFilter.filterValueAccessor()
};
}
}, {
key: "updateLayerMeta",
value: function updateLayerMeta(allData, getPosition) {
var bounds = this.getPointsBounds(allData, function (d) {
return getPosition({
data: d
});
});
this.updateMeta({
bounds: bounds
});
}
}, {
key: "renderLayer",
value: function renderLayer(opts) {
var data = opts.data,
gpuFilter = opts.gpuFilter;
var _this$config$visConfi = this.config.visConfig,
_this$config$visConfi2 = _this$config$visConfi.sizeScale,
sizeScale = _this$config$visConfi2 === void 0 ? 1 : _this$config$visConfi2,
_this$config$visConfi3 = _this$config$visConfi.angleX,
angleX = _this$config$visConfi3 === void 0 ? 0 : _this$config$visConfi3,
_this$config$visConfi4 = _this$config$visConfi.angleY,
angleY = _this$config$visConfi4 === void 0 ? 0 : _this$config$visConfi4,
_this$config$visConfi5 = _this$config$visConfi.angleZ,
angleZ = _this$config$visConfi5 === void 0 ? 90 : _this$config$visConfi5;
return [new _meshLayers.ScenegraphLayer(_objectSpread({}, this.getDefaultDeckLayerProps(opts), {}, data, {
fetch: fetch,
scenegraph: this.config.visConfig.scenegraph || DEFAULT_MODEL,
sizeScale: sizeScale,
getTranslation: DEFAULT_TRANSITION,
getScale: DEFAULT_SCALE,
getOrientation: [angleX, angleY, angleZ],
getColor: DEFAULT_COLOR,
// parameters
parameters: {
depthTest: true,
blend: false
},
// update triggers
updateTriggers: {
getOrientation: {
angleX: angleX,
angleY: angleY,
angleZ: angleZ
},
getPosition: this.config.columns,
getFilterValue: gpuFilter.filterValueUpdateTriggers
}
}))];
}
}, {
key: "type",
get: function get() {
return '3D';
}
}, {
key: "requiredLayerColumns",
get: function get() {
return scenegraphRequiredColumns;
}
}, {
key: "optionalColumns",
get: function get() {
return scenegraphOptionalColumns;
}
}, {
key: "columnPairs",
get: function get() {
return this.defaultPointColumnPairs;
}
}, {
key: "layerIcon",
get: function get() {
return _scenegraphLayerIcon["default"];
}
}, {
key: "layerInfoModal",
get: function get() {
return {
id: 'scenegraphInfo',
template: this._layerInfoModal,
modalProps: {
title: 'How to use Scenegraph'
}
};
}
}]);
return ScenegraphLayer;
}(_baseLayer["default"]);
exports["default"] = ScenegraphLayer;
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/layers/scenegraph-layer/scenegraph-layer.js"],"names":["scenegraphRequiredColumns","scenegraphOptionalColumns","fetch","url","propName","layer","GLTFLoader","getLoadOptions","then","response","json","scenegraphPosAccessor","lat","lng","altitude","d","data","fieldIdx","scenegraphVisConfigs","opacity","colorRange","sizeScale","angleX","LAYER_VIS_CONFIGS","angle","property","label","angleY","angleZ","defaultValue","DEFAULT_MODEL","DEFAULT_TRANSITION","DEFAULT_SCALE","DEFAULT_COLOR","ScenegraphLayer","props","registerVisConfig","getPositionAccessor","config","columns","_layerInfoModal","getPosition","allData","filteredIndex","i","length","index","pos","every","Number","isFinite","push","position","datasets","oldLayerData","gpuFilter","dataId","updateData","getFilterValue","filterValueAccessor","bounds","getPointsBounds","updateMeta","opts","visConfig","DeckScenegraphLayer","getDefaultDeckLayerProps","scenegraph","getTranslation","getScale","getOrientation","getColor","parameters","depthTest","blend","updateTriggers","filterValueUpdateTriggers","defaultPointColumnPairs","ScenegraphLayerIcon","id","template","modalProps","title","Layer"],"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 {ScenegraphLayer as DeckScenegraphLayer} from '@deck.gl/mesh-layers';\nimport {load} from '@loaders.gl/core';\nimport {GLTFLoader} from '@loaders.gl/gltf';\n\nimport Layer from '../base-layer';\nimport ScenegraphLayerIcon from './scenegraph-layer-icon';\nimport ScenegraphInfoModalFactory from './scenegraph-info-modal';\nimport {LAYER_VIS_CONFIGS} from 'layers/layer-factory';\n\nexport const scenegraphRequiredColumns = ['lat', 'lng'];\nexport const scenegraphOptionalColumns = ['altitude'];\n\nfunction fetch(url, {propName, layer}) {\n  if (propName === 'scenegraph') {\n    return load(url, GLTFLoader, layer.getLoadOptions());\n  }\n\n  return fetch(url).then(response => response.json());\n}\n\nexport const scenegraphPosAccessor = ({lat, lng, altitude}) => d => [\n  // lng\n  d.data[lng.fieldIdx],\n  // lat\n  d.data[lat.fieldIdx],\n  // altitude\n  altitude && altitude.fieldIdx > -1 ? d.data[altitude.fieldIdx] : 0\n];\n\nexport const scenegraphVisConfigs = {\n  opacity: 'opacity',\n  colorRange: 'colorRange',\n  //\n  sizeScale: 'sizeScale',\n  angleX: {\n    ...LAYER_VIS_CONFIGS.angle,\n    property: 'angleX',\n    label: 'angle X'\n  },\n  angleY: {\n    ...LAYER_VIS_CONFIGS.angle,\n    property: 'angleY',\n    label: 'angle Y'\n  },\n  angleZ: {\n    ...LAYER_VIS_CONFIGS.angle,\n    property: 'angleZ',\n    defaultValue: 90,\n    label: 'angle Z'\n  }\n};\n\nconst DEFAULT_MODEL =\n  'https://raw.githubusercontent.com/KhronosGroup/glTF-Sample-Models/master/2.0/Duck/glTF-Binary/Duck.glb';\nconst DEFAULT_TRANSITION = [0, 0, 0];\nconst DEFAULT_SCALE = [1, 1, 1];\nconst DEFAULT_COLOR = [255, 255, 255, 255];\n\nexport default class ScenegraphLayer extends Layer {\n  constructor(props) {\n    super(props);\n\n    this.registerVisConfig(scenegraphVisConfigs);\n    this.getPositionAccessor = () => scenegraphPosAccessor(this.config.columns);\n\n    // prepare layer info modal\n    this._layerInfoModal = ScenegraphInfoModalFactory();\n  }\n\n  get type() {\n    return '3D';\n  }\n\n  get requiredLayerColumns() {\n    return scenegraphRequiredColumns;\n  }\n\n  get optionalColumns() {\n    return scenegraphOptionalColumns;\n  }\n\n  get columnPairs() {\n    return this.defaultPointColumnPairs;\n  }\n\n  get layerIcon() {\n    return ScenegraphLayerIcon;\n  }\n\n  get layerInfoModal() {\n    return {\n      id: 'scenegraphInfo',\n      template: this._layerInfoModal,\n      modalProps: {\n        title: 'How to use Scenegraph'\n      }\n    };\n  }\n\n  calculateDataAttribute({allData, filteredIndex}, getPosition) {\n    const data = [];\n\n    for (let i = 0; i < filteredIndex.length; i++) {\n      const index = filteredIndex[i];\n      const pos = getPosition({data: allData[index]});\n\n      // if doesn't have point lat or lng, do not add the point\n      // deck.gl can't handle position = null\n      if (pos.every(Number.isFinite)) {\n        data.push({\n          data: allData[index],\n          position: pos,\n          // index is important for filter\n          index\n        });\n      }\n    }\n    return data;\n  }\n\n  formatLayerData(datasets, oldLayerData) {\n    const {gpuFilter} = datasets[this.config.dataId];\n    const {data} = this.updateData(datasets, oldLayerData);\n    const getPosition = this.getPositionAccessor();\n    return {\n      data,\n      getPosition,\n      getFilterValue: gpuFilter.filterValueAccessor()\n    };\n  }\n\n  updateLayerMeta(allData, getPosition) {\n    const bounds = this.getPointsBounds(allData, d => getPosition({data: d}));\n    this.updateMeta({bounds});\n  }\n\n  renderLayer(opts) {\n    const {data, gpuFilter} = opts;\n\n    const {\n      visConfig: {sizeScale = 1, angleX = 0, angleY = 0, angleZ = 90}\n    } = this.config;\n\n    return [\n      new DeckScenegraphLayer({\n        ...this.getDefaultDeckLayerProps(opts),\n        ...data,\n        fetch,\n        scenegraph: this.config.visConfig.scenegraph || DEFAULT_MODEL,\n        sizeScale,\n        getTranslation: DEFAULT_TRANSITION,\n        getScale: DEFAULT_SCALE,\n        getOrientation: [angleX, angleY, angleZ],\n        getColor: DEFAULT_COLOR,\n        // parameters\n        parameters: {depthTest: true, blend: false},\n        // update triggers\n        updateTriggers: {\n          getOrientation: {angleX, angleY, angleZ},\n          getPosition: this.config.columns,\n          getFilterValue: gpuFilter.filterValueUpdateTriggers\n        }\n      })\n    ];\n  }\n}\n"]}
;