kepler.gl
Version:
kepler.gl is a webgl based application to visualize large scale location data in the browser
254 lines (242 loc) • 34.7 kB
JavaScript
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.assignGpuChannel = assignGpuChannel;
exports.assignGpuChannels = assignGpuChannels;
exports.getDatasetFieldIndexForFilter = getDatasetFieldIndexForFilter;
exports.getGpuFilterProps = getGpuFilterProps;
exports.resetFilterGpuMode = resetFilterGpuMode;
exports.setFilterGpuMode = setFilterGpuMode;
var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty"));
var _moment = _interopRequireDefault(require("moment"));
var _constants = require("@kepler.gl/constants");
var _utils = require("@kepler.gl/utils");
var _commonUtils = require("@kepler.gl/common-utils");
function ownKeys(e, r) { var t = Object.keys(e); if (Object.getOwnPropertySymbols) { var o = Object.getOwnPropertySymbols(e); r && (o = o.filter(function (r) { return Object.getOwnPropertyDescriptor(e, r).enumerable; })), t.push.apply(t, o); } return t; }
function _objectSpread(e) { for (var r = 1; r < arguments.length; r++) { var t = null != arguments[r] ? arguments[r] : {}; r % 2 ? ownKeys(Object(t), !0).forEach(function (r) { (0, _defineProperty2["default"])(e, r, t[r]); }) : Object.getOwnPropertyDescriptors ? Object.defineProperties(e, Object.getOwnPropertyDescriptors(t)) : ownKeys(Object(t)).forEach(function (r) { Object.defineProperty(e, r, Object.getOwnPropertyDescriptor(t, r)); }); } return e; } // SPDX-License-Identifier: MIT
// Copyright contributors to the kepler.gl project
/**
* Set gpu mode based on current number of gpu filters exists
*/
function setFilterGpuMode(filter, filters) {
// filter can be applied to multiple datasets, hence gpu filter mode should also be
// an array, however, to keep us sane, for now, we only check if there is available channel for every dataId,
// if all of them has, we set gpu mode to true
// TODO: refactor filter so we don't keep an array of everything
filter.dataId.forEach(function (dataId) {
var gpuFilters = filters.filter(function (f) {
return f.dataId.includes(dataId) && f.gpu;
});
if (filter.gpu && gpuFilters.length === _constants.MAX_GPU_FILTERS) {
(0, _utils.set)(['gpu'], false, filter);
}
});
return filter;
}
/**
* Scan though all filters and assign gpu chanel to gpu filter
*/
function assignGpuChannels(allFilters) {
return allFilters.reduce(function (accu, f, index) {
var filters = accu;
// if gpu is true assign and validate gpu Channel
if (f.gpu) {
f = assignGpuChannel(f, accu);
filters = (0, _utils.set)([index], f, accu);
}
return filters;
}, allFilters);
}
/**
* Assign a new gpu filter a channel based on first availability
*/
function assignGpuChannel(filter, filters) {
// find first available channel
if (!filter.gpu) {
return filter;
}
var gpuChannel = filter.gpuChannel || [];
filter.dataId.forEach(function (dataId, datasetIdx) {
var findGpuChannel = function findGpuChannel(channel) {
return function (f) {
var dataIdx = (0, _commonUtils.toArray)(f.dataId).indexOf(dataId);
return f.id !== filter.id && dataIdx > -1 && f.gpu && (0, _commonUtils.toArray)(f.gpuChannel)[dataIdx] === channel;
};
};
if (Number.isFinite(gpuChannel[datasetIdx]) && !filters.find(findGpuChannel(gpuChannel[datasetIdx]))) {
// if value is already assigned and valid
return;
}
var i = 0;
while (i < _constants.MAX_GPU_FILTERS) {
if (!filters.find(findGpuChannel(i))) {
gpuChannel[datasetIdx] = i;
return;
}
i++;
}
});
// if cannot find channel for all dataid, set gpu back to false
// TODO: refactor filter to handle same filter different gpu mode
if (!gpuChannel.length || !gpuChannel.every(Number.isFinite)) {
return _objectSpread(_objectSpread({}, filter), {}, {
gpu: false
});
}
return _objectSpread(_objectSpread({}, filter), {}, {
gpuChannel: gpuChannel
});
}
/**
* Edit filter.gpu to ensure that only
* X number of gpu filers can coexist.
*/
function resetFilterGpuMode(filters) {
var gpuPerDataset = {};
return filters.map(function (f) {
if (f.gpu) {
var gpu = true;
(0, _commonUtils.toArray)(f.dataId).forEach(function (dataId) {
var count = gpuPerDataset[dataId];
if (count === _constants.MAX_GPU_FILTERS) {
gpu = false;
} else {
gpuPerDataset[dataId] = count ? count + 1 : 1;
}
});
if (!gpu) {
return (0, _utils.set)(['gpu'], false, f);
}
}
return f;
});
}
/**
* Initial filter uniform
*/
function getEmptyFilterRange() {
return new Array(_constants.MAX_GPU_FILTERS).fill(0).map(function () {
return [0, 0];
});
}
/**
* Returns index of the data element.
* @param {any} d Data element with row index info.
* @returns number
*/
var defaultGetIndex = function defaultGetIndex(d) {
return d.index;
};
/**
* Returns value at the specified row from the data container.
* @param dc Data container.
* @param d Data element with row index info.
* @param fieldIndex Column index in the data container.
* @returns
*/
var defaultGetData = function defaultGetData(dc, d, fieldIndex) {
return dc.valueAt(d.index, fieldIndex);
};
var getFilterValueAccessor = function getFilterValueAccessor(channels, dataId, fields) {
return function (dc) {
return function () {
var getIndex = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : defaultGetIndex;
var getData = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : defaultGetData;
return function (d, objectInfo) {
// for empty channel, value is 0 and min max would be [0, 0]
var channelValues = channels.map(function (filter) {
var _filter$domain2;
if (!filter) {
return 0;
}
var fieldIndex = getDatasetFieldIndexForFilter(dataId, filter);
var field = fields[fieldIndex];
var value;
// d can be undefined when called from attribute updater from deck,
// when data is an ArrowTable, so use objectInfo instead.
var data = getData(dc, d || objectInfo, fieldIndex);
if (typeof data === 'function') {
value = data(field);
} else {
value = filter.type === _constants.FILTER_TYPES.timeRange ? field.filterProps && Array.isArray(field.filterProps.mappedValue) ? field.filterProps.mappedValue[getIndex(d)] : _moment["default"].utc(data).valueOf() : data;
}
return (0, _commonUtils.notNullorUndefined)(value) ? Array.isArray(value) ? value.map(function (v) {
var _filter$domain;
return v - ((_filter$domain = filter.domain) === null || _filter$domain === void 0 ? void 0 : _filter$domain[0]);
}) : value - ((_filter$domain2 = filter.domain) === null || _filter$domain2 === void 0 ? void 0 : _filter$domain2[0]) : Number.MIN_SAFE_INTEGER;
});
// TODO: can we refactor the above to avoid the transformation below?
var arrChannel = channelValues.find(function (v) {
return Array.isArray(v);
});
if (Array.isArray(arrChannel)) {
// Convert info form supported by DataFilterExtension (relevant for TripLayer)
var vals = [];
// if there are multiple arrays, they should have the same length
var _loop = function _loop(i) {
vals.push(channelValues.map(function (v) {
return Array.isArray(v) ? v[i] : v;
}));
};
for (var i = 0; i < arrChannel.length; i++) {
_loop(i);
}
return vals;
}
return channelValues;
};
};
};
};
function isFilterTriggerEqual(a, b) {
return a === b || (a === null || a === void 0 ? void 0 : a.name) === (b === null || b === void 0 ? void 0 : b.name) && (a === null || a === void 0 ? void 0 : a.domain0) === (b === null || b === void 0 ? void 0 : b.domain0);
}
/**
* Get filter properties for gpu filtering
*/
function getGpuFilterProps(filters, dataId, fields, oldGpuFilter) {
var filterRange = getEmptyFilterRange();
var triggers = {};
// array of filter for each channel, undefined, if no filter is assigned to that channel
var channels = [];
var _loop2 = function _loop2(i) {
var _filter$domain3, _filter$domain4, _oldGpuFilter$filterV, _filter$domain5;
var filter = filters.find(function (f) {
return f.gpu && f.dataId.includes(dataId) && f.gpuChannel && f.gpuChannel[f.dataId.indexOf(dataId)] === i;
});
filterRange[i][0] = filter ? filter.value[0] - ((_filter$domain3 = filter.domain) === null || _filter$domain3 === void 0 ? void 0 : _filter$domain3[0]) : 0;
filterRange[i][1] = filter ? filter.value[1] - ((_filter$domain4 = filter.domain) === null || _filter$domain4 === void 0 ? void 0 : _filter$domain4[0]) : 0;
var oldFilterTrigger = (oldGpuFilter === null || oldGpuFilter === void 0 || (_oldGpuFilter$filterV = oldGpuFilter.filterValueUpdateTriggers) === null || _oldGpuFilter$filterV === void 0 ? void 0 : _oldGpuFilter$filterV["gpuFilter_".concat(i)]) || null;
var trigger = filter ? {
name: filter.name[filter.dataId.indexOf(dataId)],
domain0: (_filter$domain5 = filter.domain) === null || _filter$domain5 === void 0 ? void 0 : _filter$domain5[0]
} : null;
// don't create a new object, cause deck.gl use shallow compare
triggers["gpuFilter_".concat(i)] = isFilterTriggerEqual(trigger, oldFilterTrigger) ? oldFilterTrigger : trigger;
channels.push(filter);
};
for (var i = 0; i < _constants.MAX_GPU_FILTERS; i++) {
_loop2(i);
}
var filterValueAccessor = getFilterValueAccessor(channels, dataId, fields);
return {
filterRange: filterRange,
filterValueUpdateTriggers: triggers,
filterValueAccessor: filterValueAccessor
};
}
/**
* Return dataset field index from filter.fieldIdx
* The index matches the same dataset index for filter.dataId
*/
function getDatasetFieldIndexForFilter(dataId, filter) {
var datasetIndex = (0, _commonUtils.toArray)(filter.dataId).indexOf(dataId);
if (datasetIndex < 0) {
return -1;
}
var fieldIndex = filter.fieldIdx[datasetIndex];
return (0, _commonUtils.notNullorUndefined)(fieldIndex) ? fieldIndex : -1;
}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_moment","_interopRequireDefault","require","_constants","_utils","_commonUtils","ownKeys","e","r","t","Object","keys","getOwnPropertySymbols","o","filter","getOwnPropertyDescriptor","enumerable","push","apply","_objectSpread","arguments","length","forEach","_defineProperty2","getOwnPropertyDescriptors","defineProperties","defineProperty","setFilterGpuMode","filters","dataId","gpuFilters","f","includes","gpu","MAX_GPU_FILTERS","set","assignGpuChannels","allFilters","reduce","accu","index","assignGpuChannel","gpuChannel","datasetIdx","findGpuChannel","channel","dataIdx","toArray","indexOf","id","Number","isFinite","find","i","every","resetFilterGpuMode","gpuPerDataset","map","count","getEmptyFilterRange","Array","fill","defaultGetIndex","d","defaultGetData","dc","fieldIndex","valueAt","getFilterValueAccessor","channels","fields","getIndex","undefined","getData","objectInfo","channelValues","_filter$domain2","getDatasetFieldIndexForFilter","field","value","data","type","FILTER_TYPES","timeRange","filterProps","isArray","mappedValue","moment","utc","valueOf","notNullorUndefined","v","_filter$domain","domain","MIN_SAFE_INTEGER","arrChannel","vals","_loop","isFilterTriggerEqual","a","b","name","domain0","getGpuFilterProps","oldGpuFilter","filterRange","triggers","_loop2","_filter$domain3","_filter$domain4","_oldGpuFilter$filterV","_filter$domain5","oldFilterTrigger","filterValueUpdateTriggers","concat","trigger","filterValueAccessor","datasetIndex","fieldIdx"],"sources":["../src/gpu-filter-utils.ts"],"sourcesContent":["// SPDX-License-Identifier: MIT\n// Copyright contributors to the kepler.gl project\n\nimport moment from 'moment';\nimport {MAX_GPU_FILTERS, FILTER_TYPES} from '@kepler.gl/constants';\nimport {Field, Filter} from '@kepler.gl/types';\nimport {set, DataContainerInterface} from '@kepler.gl/utils';\nimport {toArray, notNullorUndefined} from '@kepler.gl/common-utils';\n\nimport {GpuFilter} from './kepler-table';\n\n/**\n * Set gpu mode based on current number of gpu filters exists\n */\nexport function setFilterGpuMode(filter: Filter, filters: Filter[]) {\n  // filter can be applied to multiple datasets, hence gpu filter mode should also be\n  // an array, however, to keep us sane, for now, we only check if there is available channel for every dataId,\n  // if all of them has, we set gpu mode to true\n  // TODO: refactor filter so we don't keep an array of everything\n\n  filter.dataId.forEach(dataId => {\n    const gpuFilters = filters.filter(f => f.dataId.includes(dataId) && f.gpu);\n\n    if (filter.gpu && gpuFilters.length === MAX_GPU_FILTERS) {\n      set(['gpu'], false, filter);\n    }\n  });\n\n  return filter;\n}\n\n/**\n * Scan though all filters and assign gpu chanel to gpu filter\n */\nexport function assignGpuChannels(allFilters: Filter[]) {\n  return allFilters.reduce((accu, f, index) => {\n    let filters = accu;\n\n    // if gpu is true assign and validate gpu Channel\n    if (f.gpu) {\n      f = assignGpuChannel(f, accu);\n      filters = set([index], f, accu);\n    }\n\n    return filters;\n  }, allFilters);\n}\n/**\n * Assign a new gpu filter a channel based on first availability\n */\nexport function assignGpuChannel(filter: Filter, filters: Filter[]) {\n  // find first available channel\n  if (!filter.gpu) {\n    return filter;\n  }\n\n  const gpuChannel = filter.gpuChannel || [];\n\n  filter.dataId.forEach((dataId, datasetIdx) => {\n    const findGpuChannel = channel => f => {\n      const dataIdx = toArray(f.dataId).indexOf(dataId);\n      return (\n        f.id !== filter.id && dataIdx > -1 && f.gpu && toArray(f.gpuChannel)[dataIdx] === channel\n      );\n    };\n\n    if (\n      Number.isFinite(gpuChannel[datasetIdx]) &&\n      !filters.find(findGpuChannel(gpuChannel[datasetIdx]))\n    ) {\n      // if value is already assigned and valid\n      return;\n    }\n\n    let i = 0;\n\n    while (i < MAX_GPU_FILTERS) {\n      if (!filters.find(findGpuChannel(i))) {\n        gpuChannel[datasetIdx] = i;\n        return;\n      }\n      i++;\n    }\n  });\n\n  // if cannot find channel for all dataid, set gpu back to false\n  // TODO: refactor filter to handle same filter different gpu mode\n  if (!gpuChannel.length || !gpuChannel.every(Number.isFinite)) {\n    return {\n      ...filter,\n      gpu: false\n    };\n  }\n\n  return {\n    ...filter,\n    gpuChannel\n  };\n}\n/**\n * Edit filter.gpu to ensure that only\n * X number of gpu filers can coexist.\n */\nexport function resetFilterGpuMode(filters: Filter[]): Filter[] {\n  const gpuPerDataset = {};\n\n  return filters.map(f => {\n    if (f.gpu) {\n      let gpu = true;\n      toArray(f.dataId).forEach(dataId => {\n        const count = gpuPerDataset[dataId];\n\n        if (count === MAX_GPU_FILTERS) {\n          gpu = false;\n        } else {\n          gpuPerDataset[dataId] = count ? count + 1 : 1;\n        }\n      });\n\n      if (!gpu) {\n        return set(['gpu'], false, f);\n      }\n    }\n\n    return f;\n  });\n}\n\n/**\n * Initial filter uniform\n */\nfunction getEmptyFilterRange() {\n  return new Array(MAX_GPU_FILTERS).fill(0).map(() => [0, 0]);\n}\n\n/**\n * Returns index of the data element.\n * @param {any} d Data element with row index info.\n * @returns number\n */\nconst defaultGetIndex = d => d.index;\n\n/**\n * Returns value at the specified row from the data container.\n * @param dc Data container.\n * @param d Data element with row index info.\n * @param fieldIndex Column index in the data container.\n * @returns\n */\nconst defaultGetData = (dc: DataContainerInterface, d: any, fieldIndex: number) => {\n  return dc.valueAt(d.index, fieldIndex);\n};\n\nconst getFilterValueAccessor =\n  (channels: (Filter | undefined)[], dataId: string, fields: any[]) =>\n  (dc: DataContainerInterface) =>\n  (getIndex = defaultGetIndex, getData = defaultGetData) =>\n  (d, objectInfo?: {index: number}) => {\n    // for empty channel, value is 0 and min max would be [0, 0]\n    const channelValues = channels.map(filter => {\n      if (!filter) {\n        return 0;\n      }\n      const fieldIndex = getDatasetFieldIndexForFilter(dataId, filter);\n      const field = fields[fieldIndex];\n\n      let value;\n      // d can be undefined when called from attribute updater from deck,\n      // when data is an ArrowTable, so use objectInfo instead.\n      const data = getData(dc, d || objectInfo, fieldIndex);\n      if (typeof data === 'function') {\n        value = data(field);\n      } else {\n        value =\n          filter.type === FILTER_TYPES.timeRange\n            ? field.filterProps && Array.isArray(field.filterProps.mappedValue)\n              ? field.filterProps.mappedValue[getIndex(d)]\n              : moment.utc(data).valueOf()\n            : data;\n      }\n\n      return notNullorUndefined(value)\n        ? Array.isArray(value)\n          ? value.map(v => v - filter.domain?.[0])\n          : value - filter.domain?.[0]\n        : Number.MIN_SAFE_INTEGER;\n    });\n\n    // TODO: can we refactor the above to avoid the transformation below?\n    const arrChannel = channelValues.find(v => Array.isArray(v));\n    if (Array.isArray(arrChannel)) {\n      // Convert info form supported by DataFilterExtension (relevant for TripLayer)\n      const vals: number[][] = [];\n      // if there are multiple arrays, they should have the same length\n      for (let i = 0; i < arrChannel.length; i++) {\n        vals.push(channelValues.map(v => (Array.isArray(v) ? v[i] : v)));\n      }\n      return vals;\n    }\n\n    return channelValues;\n  };\n\nfunction isFilterTriggerEqual(a, b) {\n  return a === b || (a?.name === b?.name && a?.domain0 === b?.domain0);\n}\n\n/**\n * Get filter properties for gpu filtering\n */\nexport function getGpuFilterProps(\n  filters: Filter[],\n  dataId: string,\n  fields: Field[],\n  oldGpuFilter?: GpuFilter\n): GpuFilter {\n  const filterRange = getEmptyFilterRange();\n  const triggers: GpuFilter['filterValueUpdateTriggers'] = {};\n\n  // array of filter for each channel, undefined, if no filter is assigned to that channel\n  const channels: (Filter | undefined)[] = [];\n\n  for (let i = 0; i < MAX_GPU_FILTERS; i++) {\n    const filter = filters.find(\n      f =>\n        f.gpu &&\n        f.dataId.includes(dataId) &&\n        f.gpuChannel &&\n        f.gpuChannel[f.dataId.indexOf(dataId)] === i\n    );\n\n    filterRange[i][0] = filter ? filter.value[0] - filter.domain?.[0] : 0;\n    filterRange[i][1] = filter ? filter.value[1] - filter.domain?.[0] : 0;\n    const oldFilterTrigger = oldGpuFilter?.filterValueUpdateTriggers?.[`gpuFilter_${i}`] || null;\n\n    const trigger = filter\n      ? {\n          name: filter.name[filter.dataId.indexOf(dataId)],\n          domain0: filter.domain?.[0]\n        }\n      : null;\n    // don't create a new object, cause deck.gl use shallow compare\n    triggers[`gpuFilter_${i}`] = isFilterTriggerEqual(trigger, oldFilterTrigger)\n      ? oldFilterTrigger\n      : trigger;\n    channels.push(filter);\n  }\n\n  const filterValueAccessor = getFilterValueAccessor(channels, dataId, fields);\n\n  return {\n    filterRange,\n    filterValueUpdateTriggers: triggers,\n    filterValueAccessor\n  };\n}\n\n/**\n * Return dataset field index from filter.fieldIdx\n * The index matches the same dataset index for filter.dataId\n */\nexport function getDatasetFieldIndexForFilter(dataId: string, filter: Filter): number {\n  const datasetIndex = toArray(filter.dataId).indexOf(dataId);\n  if (datasetIndex < 0) {\n    return -1;\n  }\n\n  const fieldIndex = filter.fieldIdx[datasetIndex];\n\n  return notNullorUndefined(fieldIndex) ? fieldIndex : -1;\n}\n"],"mappings":"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","ignoreList":[]}
;