kepler.gl.geoiq
Version:
kepler.gl is a webgl based application to visualize large scale location data in the browser
226 lines (185 loc) • 20.2 kB
JavaScript
;
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.createNewDataEntry = createNewDataEntry;
exports.removeSuffixAndDelimiters = removeSuffixAndDelimiters;
exports.findPointFieldPairs = findPointFieldPairs;
exports.datasetColorMaker = void 0;
var _typeof2 = _interopRequireDefault(require("@babel/runtime/helpers/typeof"));
var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty"));
var _regenerator = _interopRequireDefault(require("@babel/runtime/regenerator"));
var _colorUtils = require("./color-utils");
var _lodash = _interopRequireDefault(require("lodash.uniq"));
var _defaultSettings = require("../constants/default-settings");
var _utils = require("./utils");
var _dataProcessor = require("../processors/data-processor");
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 _marked =
/*#__PURE__*/
_regenerator["default"].mark(generateColor);
// apply a color for each dataset
// to use as label colors
var datasetColors = ['#8F2FBF', '#005CFF', '#C06C84', '#F8B195', '#547A82', '#3EACA8', '#A2D4AB'].map(_colorUtils.hexToRgb);
/**
* Random color generator
*/
function generateColor() {
var index;
return _regenerator["default"].wrap(function generateColor$(_context) {
while (1) {
switch (_context.prev = _context.next) {
case 0:
index = 0;
case 1:
if (!(index < datasetColors.length + 1)) {
_context.next = 7;
break;
}
if (index === datasetColors.length) {
index = 0;
}
_context.next = 5;
return datasetColors[index++];
case 5:
_context.next = 1;
break;
case 7:
case "end":
return _context.stop();
}
}
}, _marked);
}
var datasetColorMaker = generateColor();
exports.datasetColorMaker = datasetColorMaker;
function getNewDatasetColor(datasets) {
var presetColors = datasetColors.map(String);
var usedColors = (0, _lodash["default"])(Object.values(datasets).map(function (d) {
return String(d.color);
})).filter(function (c) {
return presetColors.includes(c);
});
if (usedColors.length === presetColors.length) {
// if we already depleted the pool of color
return datasetColorMaker.next().value;
}
var color = datasetColorMaker.next().value;
while (usedColors.includes(String(color))) {
color = datasetColorMaker.next().value;
}
return color;
}
function createNewDataEntry(_ref) {
var _ref$info = _ref.info,
info = _ref$info === void 0 ? {} : _ref$info,
data = _ref.data;
var datasets = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {};
var validatedData = (0, _dataProcessor.validateInputData)(data);
if (!validatedData) {
return {};
}
var allData = validatedData.rows;
var datasetInfo = _objectSpread({
id: (0, _utils.generateHashId)(4),
label: 'new dataset'
}, info);
var dataId = datasetInfo.id; // add tableFieldIndex and id to fields
// TODO: don't need id and name and tableFieldIndex anymore
// Add value accessor instead
var fields = validatedData.fields.map(function (f, i) {
return _objectSpread({}, f, {
id: f.name,
tableFieldIndex: i + 1
});
});
return (0, _defineProperty2["default"])({}, dataId, _objectSpread({}, datasetInfo, {
color: datasetInfo.color || getNewDatasetColor(datasets),
id: dataId,
allData: allData,
// TODO: no need to make a copy anymore, only save fieldedIndex
data: allData.slice(),
filteredIndex: allData.map(function (_, i) {
return i;
}),
filteredIndexForDomain: allData.map(function (_, i) {
return i;
}),
fieldPairs: findPointFieldPairs(fields),
fields: fields
}));
}
function removeSuffixAndDelimiters(layerName, suffix) {
return layerName.replace(new RegExp(suffix, 'ig'), '').replace(/[_,.]+/g, ' ').trim();
}
/**
* Find point fields pairs from fields
*
* @param {Array} fields
* @returns {Array} found point fields
*/
function findPointFieldPairs(fields) {
var allNames = fields.map(function (f) {
return f.name.toLowerCase();
}); // get list of all fields with matching suffixes
return allNames.reduce(function (carry, fieldName, idx) {
// This search for pairs will early exit if found.
var _iteratorNormalCompletion = true;
var _didIteratorError = false;
var _iteratorError = undefined;
try {
for (var _iterator = _defaultSettings.TRIP_POINT_FIELDS[Symbol.iterator](), _step; !(_iteratorNormalCompletion = (_step = _iterator.next()).done); _iteratorNormalCompletion = true) {
var suffixPair = _step.value;
// match first suffix```
if (fieldName.endsWith(suffixPair[0])) {
var _ret = function () {
// match second suffix
var otherPattern = new RegExp("".concat(suffixPair[0], "$"));
var partner = fieldName.replace(otherPattern, suffixPair[1]);
var partnerIdx = allNames.findIndex(function (d) {
return d === partner;
});
if (partnerIdx > -1) {
var defaultName = removeSuffixAndDelimiters(fieldName, suffixPair[0]);
carry.push({
defaultName: defaultName,
pair: {
lat: {
fieldIdx: idx,
value: fields[idx].name
},
lng: {
fieldIdx: partnerIdx,
value: fields[partnerIdx].name
}
},
suffix: suffixPair
});
return {
v: carry
};
}
}();
if ((0, _typeof2["default"])(_ret) === "object") return _ret.v;
}
}
} catch (err) {
_didIteratorError = true;
_iteratorError = err;
} finally {
try {
if (!_iteratorNormalCompletion && _iterator["return"] != null) {
_iterator["return"]();
}
} finally {
if (_didIteratorError) {
throw _iteratorError;
}
}
}
return carry;
}, []);
}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/utils/dataset-utils.js"],"names":["generateColor","datasetColors","map","hexToRgb","index","length","datasetColorMaker","getNewDatasetColor","datasets","presetColors","String","usedColors","Object","values","d","color","filter","c","includes","next","value","createNewDataEntry","info","data","validatedData","allData","rows","datasetInfo","id","label","dataId","fields","f","i","name","tableFieldIndex","slice","filteredIndex","_","filteredIndexForDomain","fieldPairs","findPointFieldPairs","removeSuffixAndDelimiters","layerName","suffix","replace","RegExp","trim","allNames","toLowerCase","reduce","carry","fieldName","idx","TRIP_POINT_FIELDS","suffixPair","endsWith","otherPattern","partner","partnerIdx","findIndex","defaultName","push","pair","lat","fieldIdx","lng"],"mappings":"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","sourcesContent":["// Copyright (c) 2019 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 {hexToRgb} from './color-utils';\nimport uniq from 'lodash.uniq';\nimport {TRIP_POINT_FIELDS} from 'constants/default-settings';\nimport {generateHashId} from './utils';\nimport {validateInputData} from 'processors/data-processor';\n// apply a color for each dataset\n// to use as label colors\nconst datasetColors = [\n  '#8F2FBF',\n  '#005CFF',\n  '#C06C84',\n  '#F8B195',\n  '#547A82',\n  '#3EACA8',\n  '#A2D4AB'\n].map(hexToRgb);\n\n/**\n * Random color generator\n */\nfunction* generateColor() {\n  let index = 0;\n  while (index < datasetColors.length + 1) {\n    if (index === datasetColors.length) {\n      index = 0;\n    }\n    yield datasetColors[index++];\n  }\n}\n\nexport const datasetColorMaker = generateColor();\n\nfunction getNewDatasetColor(datasets) {\n  const presetColors = datasetColors.map(String);\n  const usedColors = uniq(\n    Object.values(datasets).map(d => String(d.color))\n  ).filter(c => presetColors.includes(c));\n\n  if (usedColors.length === presetColors.length) {\n    // if we already depleted the pool of color\n    return datasetColorMaker.next().value;\n  }\n\n  let color = datasetColorMaker.next().value;\n  while (usedColors.includes(String(color))) {\n    color = datasetColorMaker.next().value;\n  }\n\n  return color;\n}\n\nexport function createNewDataEntry({info = {}, data}, datasets = {}) {\n  const validatedData = validateInputData(data);\n  if (!validatedData) {\n    return {};\n  }\n\n  const allData = validatedData.rows;\n  const datasetInfo = {\n    id: generateHashId(4),\n    label: 'new dataset',\n    ...info\n  };\n  const dataId = datasetInfo.id;\n\n  // add tableFieldIndex and id to fields\n  // TODO: don't need id and name and tableFieldIndex anymore\n  // Add value accessor instead\n  const fields = validatedData.fields.map((f, i) => ({\n    ...f,\n    id: f.name,\n    tableFieldIndex: i + 1\n  }));\n\n  return {\n    [dataId]: {\n      ...datasetInfo,\n      color: datasetInfo.color || getNewDatasetColor(datasets),\n      id: dataId,\n      allData,\n      // TODO: no need to make a copy anymore, only save fieldedIndex\n      data: allData.slice(),\n      filteredIndex: allData.map((_, i) => i),\n      filteredIndexForDomain: allData.map((_, i) => i),\n      fieldPairs: findPointFieldPairs(fields),\n      fields\n    }\n  };\n}\n\nexport function removeSuffixAndDelimiters(layerName, suffix) {\n  return layerName\n    .replace(new RegExp(suffix, 'ig'), '')\n    .replace(/[_,.]+/g, ' ')\n    .trim();\n}\n\n/**\n * Find point fields pairs from fields\n *\n * @param {Array} fields\n * @returns {Array} found point fields\n */\nexport function findPointFieldPairs(fields) {\n  const allNames = fields.map(f => f.name.toLowerCase());\n\n  // get list of all fields with matching suffixes\n  return allNames.reduce((carry, fieldName, idx) => {\n    // This search for pairs will early exit if found.\n    for (const suffixPair of TRIP_POINT_FIELDS) {\n      // match first suffix```\n      if (fieldName.endsWith(suffixPair[0])) {\n        // match second suffix\n        const otherPattern = new RegExp(`${suffixPair[0]}\\$`);\n        const partner = fieldName.replace(otherPattern, suffixPair[1]);\n\n        const partnerIdx = allNames.findIndex(d => d === partner);\n        if (partnerIdx > -1) {\n          const defaultName = removeSuffixAndDelimiters(\n            fieldName,\n            suffixPair[0]\n          );\n\n          carry.push({\n            defaultName,\n            pair: {\n              lat: {\n                fieldIdx: idx,\n                value: fields[idx].name\n              },\n              lng: {\n                fieldIdx: partnerIdx,\n                value: fields[partnerIdx].name\n              }\n            },\n            suffix: suffixPair\n          });\n          return carry;\n        }\n      }\n    }\n    return carry;\n  }, []);\n}\n"]}