kepler.gl
Version:
kepler.gl is a webgl based application to visualize large scale location data in the browser
250 lines (210 loc) • 24.6 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 _inherits2 = _interopRequireDefault(require("@babel/runtime/helpers/inherits"));
var _possibleConstructorReturn2 = _interopRequireDefault(require("@babel/runtime/helpers/possibleConstructorReturn"));
var _getPrototypeOf2 = _interopRequireDefault(require("@babel/runtime/helpers/getPrototypeOf"));
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 _createSuper(Derived) { var hasNativeReflectConstruct = _isNativeReflectConstruct(); return function _createSuperInternal() { var Super = (0, _getPrototypeOf2["default"])(Derived), result; if (hasNativeReflectConstruct) { var NewTarget = (0, _getPrototypeOf2["default"])(this).constructor; result = Reflect.construct(Super, arguments, NewTarget); } else { result = Super.apply(this, arguments); } return (0, _possibleConstructorReturn2["default"])(this, result); }; }
function _isNativeReflectConstruct() { if (typeof Reflect === "undefined" || !Reflect.construct) return false; if (Reflect.construct.sham) return false; if (typeof Proxy === "function") return true; try { Boolean.prototype.valueOf.call(Reflect.construct(Boolean, [], function () {})); return true; } catch (e) { return false; } }
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 (dc) {
return function (d) {
return [dc.valueAt(d.index, lng.fieldIdx), dc.valueAt(d.index, lat.fieldIdx), altitude && altitude.fieldIdx > -1 ? dc.valueAt(d.index, altitude.fieldIdx) : 0];
};
};
};
exports.scenegraphPosAccessor = scenegraphPosAccessor;
var scenegraphVisConfigs = {
opacity: 'opacity',
colorRange: 'colorRange',
//
sizeScale: 'sizeScale',
angleX: _objectSpread(_objectSpread({}, _layerFactory.LAYER_VIS_CONFIGS.angle), {}, {
property: 'angleX',
label: 'angle X'
}),
angleY: _objectSpread(_objectSpread({}, _layerFactory.LAYER_VIS_CONFIGS.angle), {}, {
property: 'angleY',
label: 'angle Y'
}),
angleZ: _objectSpread(_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);
var _super = _createSuper(ScenegraphLayer);
function ScenegraphLayer(props) {
var _this;
(0, _classCallCheck2["default"])(this, ScenegraphLayer);
_this = _super.call(this, props);
_this.registerVisConfig(scenegraphVisConfigs);
_this.getPositionAccessor = function (dataContainer) {
return scenegraphPosAccessor(_this.config.columns)(dataContainer);
}; // prepare layer info modal
_this._layerInfoModal = (0, _scenegraphInfoModal["default"])();
return _this;
}
(0, _createClass2["default"])(ScenegraphLayer, [{
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'
}
};
}
}, {
key: "calculateDataAttribute",
value: function calculateDataAttribute(_ref3, getPosition) {
var dataContainer = _ref3.dataContainer,
filteredIndex = _ref3.filteredIndex;
var data = [];
for (var i = 0; i < filteredIndex.length; i++) {
var index = filteredIndex[i];
var pos = getPosition({
index: 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({
position: pos,
index: index
});
}
}
return data;
}
}, {
key: "formatLayerData",
value: function formatLayerData(datasets, oldLayerData) {
var _datasets$this$config = datasets[this.config.dataId],
gpuFilter = _datasets$this$config.gpuFilter,
dataContainer = _datasets$this$config.dataContainer;
var _this$updateData = this.updateData(datasets, oldLayerData),
data = _this$updateData.data;
var getPosition = this.getPositionAccessor(dataContainer);
return {
data: data,
getPosition: getPosition,
getFilterValue: gpuFilter.filterValueAccessor(dataContainer)()
};
}
}, {
key: "updateLayerMeta",
value: function updateLayerMeta(dataContainer, getPosition) {
var bounds = this.getPointsBounds(dataContainer, getPosition);
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(_objectSpread(_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
}
}))];
}
}]);
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","dc","d","valueAt","index","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","dataContainer","config","columns","_layerInfoModal","defaultPointColumnPairs","ScenegraphLayerIcon","id","template","modalProps","title","getPosition","filteredIndex","data","i","length","pos","every","Number","isFinite","push","position","datasets","oldLayerData","dataId","gpuFilter","updateData","getFilterValue","filterValueAccessor","bounds","getPointsBounds","updateMeta","opts","visConfig","DeckScenegraphLayer","getDefaultDeckLayerProps","scenegraph","getTranslation","getScale","getOrientation","getColor","parameters","depthTest","blend","updateTriggers","filterValueUpdateTriggers","Layer"],"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 {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}) => dc => d => [\n  dc.valueAt(d.index, lng.fieldIdx),\n  dc.valueAt(d.index, lat.fieldIdx),\n  altitude && altitude.fieldIdx > -1 ? dc.valueAt(d.index, 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 = dataContainer =>\n      scenegraphPosAccessor(this.config.columns)(dataContainer);\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({dataContainer, filteredIndex}, getPosition) {\n    const data = [];\n\n    for (let i = 0; i < filteredIndex.length; i++) {\n      const index = filteredIndex[i];\n      const pos = getPosition({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          position: pos,\n          index\n        });\n      }\n    }\n    return data;\n  }\n\n  formatLayerData(datasets, oldLayerData) {\n    const {gpuFilter, dataContainer} = datasets[this.config.dataId];\n    const {data} = this.updateData(datasets, oldLayerData);\n    const getPosition = this.getPositionAccessor(dataContainer);\n    return {\n      data,\n      getPosition,\n      getFilterValue: gpuFilter.filterValueAccessor(dataContainer)()\n    };\n  }\n\n  updateLayerMeta(dataContainer, getPosition) {\n    const bounds = this.getPointsBounds(dataContainer, getPosition);\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"]}