kepler.gl
Version:
kepler.gl is a webgl based application to visualize large scale location data in the browser
473 lines (374 loc) • 43.6 kB
JavaScript
;
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.unique = unique;
exports.findMapBounds = findMapBounds;
exports.getLatLngBounds = getLatLngBounds;
exports.clamp = clamp;
exports.getSampleData = getSampleData;
exports.timeToUnixMilli = timeToUnixMilli;
exports.maybeToDate = maybeToDate;
exports.notNullorUndefined = notNullorUndefined;
exports.isNumber = isNumber;
exports.isPlainObject = isPlainObject;
exports.numberSort = numberSort;
exports.getSortingFunction = getSortingFunction;
exports.preciseRound = preciseRound;
exports.getRoundingDecimalFromStep = getRoundingDecimalFromStep;
exports.snapToMarks = snapToMarks;
exports.normalizeSliderValue = normalizeSliderValue;
exports.roundValToStep = roundValToStep;
exports.getFormatter = getFormatter;
exports.applyDefaultFormat = applyDefaultFormat;
exports.getBooleanFormatter = getBooleanFormatter;
exports.applyCustomFormat = applyCustomFormat;
exports.datetimeFormatter = datetimeFormatter;
exports.arrayMove = exports.parseFieldValue = exports.FIELD_DISPLAY_FORMAT = exports.defaultFormatter = void 0;
var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty"));
var _slicedToArray2 = _interopRequireDefault(require("@babel/runtime/helpers/slicedToArray"));
var _assert = _interopRequireDefault(require("assert"));
var _defaultSettings = require("../constants/default-settings");
var _tooltip = require("../constants/tooltip");
var _d3Format = require("d3-format");
var _d3Array = require("d3-array");
var _momentTimezone = _interopRequireDefault(require("moment-timezone"));
var _FIELD_DISPLAY_FORMAT;
var MAX_LATITUDE = 90;
var MIN_LATITUDE = -90;
var MAX_LONGITUDE = 180;
var MIN_LONGITUDE = -180;
/**
* simple getting unique values of an array
*
* @param {array} values
* @returns {array} unique values
*/
function unique(values) {
var results = [];
var uniqueSet = new Set(values);
uniqueSet.forEach(function (v) {
if (notNullorUndefined(v)) {
results.push(v);
}
});
return results;
}
/* eslint-disable max-statements */
/**
* return center of map from given points
* @param {array} layers
* @returns {object} coordinates of map center, empty if not found
*/
function findMapBounds(layers) {
// find bounds in formatted layerData
// take ALL layers into account when finding map bounds
var availableLayerBounds = layers.reduce(function (res, l) {
if (l.meta && l.meta.bounds) {
res.push(l.meta.bounds);
}
return res;
}, []); // return null if no layer is available
if (availableLayerBounds.length === 0) {
return null;
} // merge bounds in each layer
var newBounds = availableLayerBounds.reduce(function (res, b) {
return [Math.min(res[0], b[0]), Math.min(res[1], b[1]), Math.max(res[2], b[2]), Math.max(res[3], b[3])];
}, [MAX_LONGITUDE, MAX_LATITUDE, MIN_LONGITUDE, MIN_LATITUDE]);
return newBounds;
}
/* eslint-enable max-statements */
function getLatLngBounds(points, idx, limit) {
var lats = points.map(function (d) {
return Array.isArray(d) && d[idx];
}).filter(Number.isFinite).sort(numberSort);
if (!lats.length) {
return null;
} // clamp to limit
return [Math.max(lats[0], limit[0]), Math.min(lats[lats.length - 1], limit[1])];
}
function clamp(_ref, val) {
var _ref2 = (0, _slicedToArray2["default"])(_ref, 2),
min = _ref2[0],
max = _ref2[1];
return val <= min ? min : val >= max ? max : val;
}
function getSampleData(data) {
var sampleSize = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : 500;
var getValue = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : function (d) {
return d;
};
var sampleStep = Math.max(Math.floor(data.length / sampleSize), 1);
var output = [];
for (var i = 0; i < data.length; i += sampleStep) {
output.push(getValue(data[i]));
}
return output;
}
/**
* Convert different time format to unix milliseconds
* @type {typeof import('./data-utils').timeToUnixMilli}
*/
function timeToUnixMilli(value, format) {
if (notNullorUndefined(value)) {
return typeof value === 'string' ? _momentTimezone["default"].utc(value, format).valueOf() : format === 'x' ? value * 1000 : value;
}
return null;
}
/**
*
* @type {typeof import('./data-utils').maybeToDate}
*/
function maybeToDate(isTime, fieldIdx, format, dc, d) {
if (isTime) {
return timeToUnixMilli(dc.valueAt(d.index, fieldIdx), format);
}
return dc.valueAt(d.index, fieldIdx);
}
/**
* whether null or undefined
* @type {typeof import('./data-utils').notNullorUndefined}
*/
function notNullorUndefined(d) {
return d !== undefined && d !== null;
}
/**
* Whether d is a number, this filtered out NaN as well
* @type {typeof import('./data-utils').notNullorUndefined}
*/
function isNumber(d) {
return Number.isFinite(d);
}
/**
* whether null or undefined
*/
function isPlainObject(obj) {
return obj === Object(obj) && typeof obj !== 'function' && !Array.isArray(obj);
}
/**
* @type {typeof import('./data-utils').numberSort}
*/
function numberSort(a, b) {
return a - b;
}
/**
* @type {typeof import('./data-utils').getSortingFunction}
*/
function getSortingFunction(fieldType) {
switch (fieldType) {
case _defaultSettings.ALL_FIELD_TYPES.real:
case _defaultSettings.ALL_FIELD_TYPES.integer:
case _defaultSettings.ALL_FIELD_TYPES.timestamp:
return numberSort;
default:
return undefined;
}
}
/**
* round number with exact number of decimals
* return as a string
* @type {typeof import('./data-utils').preciseRound}
*/
function preciseRound(num, decimals) {
var t = Math.pow(10, decimals);
return (Math.round(num * t + (decimals > 0 ? 1 : 0) * (Math.sign(num) * (10 / Math.pow(100, decimals)))) / t).toFixed(decimals);
}
/**
* get number of decimals to round to for slider from step
* @param {number} step
* @returns {number} - number of decimal
*/
function getRoundingDecimalFromStep(step) {
if (isNaN(step)) {
(0, _assert["default"])('step is not a number');
(0, _assert["default"])(step);
}
var splitZero = step.toString().split('.');
if (splitZero.length === 1) {
return 0;
}
return splitZero[1].length;
}
/**
* Use in slider, given a number and an array of numbers, return the nears number from the array
* @type {typeof import('./data-utils').snapToMarks}
* @param value
* @param marks
*/
function snapToMarks(value, marks) {
// always use bin x0
var i = (0, _d3Array.bisectLeft)(marks, value);
if (i === 0) {
return marks[i];
} else if (i === marks.length) {
return marks[i - 1];
}
var idx = marks[i] - value < value - marks[i - 1] ? i : i - 1;
return marks[idx];
}
/**
* If marks is provided, snap to marks, if not normalize to step
* @type {typeof import('./data-utils').normalizeSliderValue}
* @param val
* @param minValue
* @param step
* @param marks
*/
function normalizeSliderValue(val, minValue, step, marks) {
if (marks && marks.length) {
return snapToMarks(val, marks);
}
return roundValToStep(minValue, step, val);
}
/**
* round the value to step for the slider
* @type {typeof import('./data-utils').roundValToStep}
* @param minValue
* @param step
* @param val
* @returns - rounded number
*/
function roundValToStep(minValue, step, val) {
if (!isNumber(step) || !isNumber(minValue)) {
return val;
}
var decimal = getRoundingDecimalFromStep(step);
var steps = Math.floor((val - minValue) / step);
var remain = val - (steps * step + minValue); // has to round because javascript turns 0.1 into 0.9999999999999987
remain = Number(preciseRound(remain, 8));
var closest;
if (remain === 0) {
closest = val;
} else if (remain < step / 2) {
closest = steps * step + minValue;
} else {
closest = (steps + 1) * step + minValue;
} // precise round return a string rounded to the defined decimal
var rounded = preciseRound(closest, decimal);
return Number(rounded);
}
/**
* Get the value format based on field and format options
* Used in render tooltip value
* @type {typeof import('./data-utils').defaultFormatter}
*/
var defaultFormatter = function defaultFormatter(v) {
return notNullorUndefined(v) ? String(v) : '';
};
exports.defaultFormatter = defaultFormatter;
var FIELD_DISPLAY_FORMAT = (_FIELD_DISPLAY_FORMAT = {}, (0, _defineProperty2["default"])(_FIELD_DISPLAY_FORMAT, _defaultSettings.ALL_FIELD_TYPES.string, defaultFormatter), (0, _defineProperty2["default"])(_FIELD_DISPLAY_FORMAT, _defaultSettings.ALL_FIELD_TYPES.timestamp, defaultFormatter), (0, _defineProperty2["default"])(_FIELD_DISPLAY_FORMAT, _defaultSettings.ALL_FIELD_TYPES.integer, defaultFormatter), (0, _defineProperty2["default"])(_FIELD_DISPLAY_FORMAT, _defaultSettings.ALL_FIELD_TYPES.real, defaultFormatter), (0, _defineProperty2["default"])(_FIELD_DISPLAY_FORMAT, _defaultSettings.ALL_FIELD_TYPES["boolean"], defaultFormatter), (0, _defineProperty2["default"])(_FIELD_DISPLAY_FORMAT, _defaultSettings.ALL_FIELD_TYPES.date, defaultFormatter), (0, _defineProperty2["default"])(_FIELD_DISPLAY_FORMAT, _defaultSettings.ALL_FIELD_TYPES.geojson, function (d) {
return typeof d === 'string' ? d : isPlainObject(d) ? JSON.stringify(d) : Array.isArray(d) ? "[".concat(String(d), "]") : '';
}), _FIELD_DISPLAY_FORMAT);
/**
* Parse field value and type and return a string representation
* @type {typeof import('./data-utils').parseFieldValue}
*/
exports.FIELD_DISPLAY_FORMAT = FIELD_DISPLAY_FORMAT;
var parseFieldValue = function parseFieldValue(value, type) {
if (!notNullorUndefined(value)) {
return '';
}
return FIELD_DISPLAY_FORMAT[type] ? FIELD_DISPLAY_FORMAT[type](value) : String(value);
};
exports.parseFieldValue = parseFieldValue;
var arrayMoveMutate = function arrayMoveMutate(array, from, to) {
array.splice(to < 0 ? array.length + to : to, 0, array.splice(from, 1)[0]);
};
/**
*
* @param {*} array
* @param {*} from
* @param {*} to
*/
var arrayMove = function arrayMove(array, from, to) {
array = array.slice();
arrayMoveMutate(array, from, to);
return array;
};
/**
* Get the value format based on field and format options
* Used in render tooltip value
* @type {typeof import('./data-utils').getFormatter}
* @param format
* @param field
*/
exports.arrayMove = arrayMove;
function getFormatter(format, field) {
if (!format) {
return defaultFormatter;
}
var tooltipFormat = Object.values(_tooltip.TOOLTIP_FORMATS).find(function (f) {
return f[_tooltip.TOOLTIP_KEY] === format;
});
if (tooltipFormat) {
return applyDefaultFormat(tooltipFormat);
} else if (typeof format === 'string' && field) {
return applyCustomFormat(format, field);
}
return defaultFormatter;
}
function applyDefaultFormat(tooltipFormat) {
if (!tooltipFormat || !tooltipFormat.format) {
return defaultFormatter;
}
switch (tooltipFormat.type) {
case _tooltip.TOOLTIP_FORMAT_TYPES.DECIMAL:
return (0, _d3Format.format)(tooltipFormat.format);
case _tooltip.TOOLTIP_FORMAT_TYPES.DATE:
case _tooltip.TOOLTIP_FORMAT_TYPES.DATE_TIME:
return datetimeFormatter(null)(tooltipFormat.format);
case _tooltip.TOOLTIP_FORMAT_TYPES.PERCENTAGE:
return function (v) {
return "".concat((0, _d3Format.format)(_tooltip.TOOLTIP_FORMATS.DECIMAL_DECIMAL_FIXED_2.format)(v), "%");
};
case _tooltip.TOOLTIP_FORMAT_TYPES.BOOLEAN:
return getBooleanFormatter(tooltipFormat.format);
default:
return defaultFormatter;
}
}
function getBooleanFormatter(format) {
switch (format) {
case '01':
return function (v) {
return v ? '1' : '0';
};
case 'yn':
return function (v) {
return v ? 'yes' : 'no';
};
default:
return defaultFormatter;
}
} // Allow user to specify custom tooltip format via config
function applyCustomFormat(format, field) {
switch (field.type) {
case _defaultSettings.ALL_FIELD_TYPES.real:
case _defaultSettings.ALL_FIELD_TYPES.integer:
return (0, _d3Format.format)(format);
case _defaultSettings.ALL_FIELD_TYPES.date:
case _defaultSettings.ALL_FIELD_TYPES.timestamp:
return datetimeFormatter(null)(format);
default:
return function (v) {
return v;
};
}
}
/**
* Format epoch milliseconds with a format string
* @type {typeof import('./data-utils').datetimeFormatter} timezone
*/
function datetimeFormatter(timezone) {
return timezone ? function (format) {
return function (ts) {
return _momentTimezone["default"].utc(ts).tz(timezone).format(format);
};
} : function (format) {
return function (ts) {
return _momentTimezone["default"].utc(ts).format(format);
};
};
}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/utils/data-utils.js"],"names":["MAX_LATITUDE","MIN_LATITUDE","MAX_LONGITUDE","MIN_LONGITUDE","unique","values","results","uniqueSet","Set","forEach","v","notNullorUndefined","push","findMapBounds","layers","availableLayerBounds","reduce","res","l","meta","bounds","length","newBounds","b","Math","min","max","getLatLngBounds","points","idx","limit","lats","map","d","Array","isArray","filter","Number","isFinite","sort","numberSort","clamp","val","getSampleData","data","sampleSize","getValue","sampleStep","floor","output","i","timeToUnixMilli","value","format","moment","utc","valueOf","maybeToDate","isTime","fieldIdx","dc","valueAt","index","undefined","isNumber","isPlainObject","obj","Object","a","getSortingFunction","fieldType","ALL_FIELD_TYPES","real","integer","timestamp","preciseRound","num","decimals","t","pow","round","sign","toFixed","getRoundingDecimalFromStep","step","isNaN","splitZero","toString","split","snapToMarks","marks","normalizeSliderValue","minValue","roundValToStep","decimal","steps","remain","closest","rounded","defaultFormatter","String","FIELD_DISPLAY_FORMAT","string","date","geojson","JSON","stringify","parseFieldValue","type","arrayMoveMutate","array","from","to","splice","arrayMove","slice","getFormatter","field","tooltipFormat","TOOLTIP_FORMATS","find","f","TOOLTIP_KEY","applyDefaultFormat","applyCustomFormat","TOOLTIP_FORMAT_TYPES","DECIMAL","DATE","DATE_TIME","datetimeFormatter","PERCENTAGE","DECIMAL_DECIMAL_FIXED_2","BOOLEAN","getBooleanFormatter","timezone","ts","tz"],"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 assert from 'assert';\nimport {ALL_FIELD_TYPES} from 'constants/default-settings';\nimport {TOOLTIP_FORMATS, TOOLTIP_FORMAT_TYPES, TOOLTIP_KEY} from 'constants/tooltip';\nimport {format as d3Format} from 'd3-format';\nimport {bisectLeft} from 'd3-array';\nimport moment from 'moment-timezone';\n\nconst MAX_LATITUDE = 90;\nconst MIN_LATITUDE = -90;\nconst MAX_LONGITUDE = 180;\nconst MIN_LONGITUDE = -180;\n\n/**\n * simple getting unique values of an array\n *\n * @param {array} values\n * @returns {array} unique values\n */\nexport function unique(values) {\n  const results = [];\n  const uniqueSet = new Set(values);\n  uniqueSet.forEach(v => {\n    if (notNullorUndefined(v)) {\n      results.push(v);\n    }\n  });\n  return results;\n}\n\n/* eslint-disable max-statements */\n/**\n * return center of map from given points\n * @param {array} layers\n * @returns {object} coordinates of map center, empty if not found\n */\nexport function findMapBounds(layers) {\n  // find bounds in formatted layerData\n  // take ALL layers into account when finding map bounds\n  const availableLayerBounds = layers.reduce((res, l) => {\n    if (l.meta && l.meta.bounds) {\n      res.push(l.meta.bounds);\n    }\n    return res;\n  }, []);\n  // return null if no layer is available\n  if (availableLayerBounds.length === 0) {\n    return null;\n  }\n  // merge bounds in each layer\n  const newBounds = availableLayerBounds.reduce(\n    (res, b) => {\n      return [\n        Math.min(res[0], b[0]),\n        Math.min(res[1], b[1]),\n        Math.max(res[2], b[2]),\n        Math.max(res[3], b[3])\n      ];\n    },\n    [MAX_LONGITUDE, MAX_LATITUDE, MIN_LONGITUDE, MIN_LATITUDE]\n  );\n  return newBounds;\n}\n/* eslint-enable max-statements */\n\nexport function getLatLngBounds(points, idx, limit) {\n  const lats = points\n    .map(d => Array.isArray(d) && d[idx])\n    .filter(Number.isFinite)\n    .sort(numberSort);\n\n  if (!lats.length) {\n    return null;\n  }\n\n  // clamp to limit\n  return [Math.max(lats[0], limit[0]), Math.min(lats[lats.length - 1], limit[1])];\n}\n\nexport function clamp([min, max], val) {\n  return val <= min ? min : val >= max ? max : val;\n}\n\nexport function getSampleData(data, sampleSize = 500, getValue = d => d) {\n  const sampleStep = Math.max(Math.floor(data.length / sampleSize), 1);\n  const output = [];\n  for (let i = 0; i < data.length; i += sampleStep) {\n    output.push(getValue(data[i]));\n  }\n\n  return output;\n}\n\n/**\n * Convert different time format to unix milliseconds\n * @type {typeof import('./data-utils').timeToUnixMilli}\n */\nexport function timeToUnixMilli(value, format) {\n  if (notNullorUndefined(value)) {\n    return typeof value === 'string'\n      ? moment.utc(value, format).valueOf()\n      : format === 'x'\n      ? value * 1000\n      : value;\n  }\n  return null;\n}\n\n/**\n *\n * @type {typeof import('./data-utils').maybeToDate}\n */\nexport function maybeToDate(isTime, fieldIdx, format, dc, d) {\n  if (isTime) {\n    return timeToUnixMilli(dc.valueAt(d.index, fieldIdx), format);\n  }\n\n  return dc.valueAt(d.index, fieldIdx);\n}\n\n/**\n * whether null or undefined\n * @type {typeof import('./data-utils').notNullorUndefined}\n */\nexport function notNullorUndefined(d) {\n  return d !== undefined && d !== null;\n}\n\n/**\n * Whether d is a number, this filtered out NaN as well\n * @type {typeof import('./data-utils').notNullorUndefined}\n */\nexport function isNumber(d) {\n  return Number.isFinite(d);\n}\n/**\n * whether null or undefined\n */\nexport function isPlainObject(obj) {\n  return obj === Object(obj) && typeof obj !== 'function' && !Array.isArray(obj);\n}\n\n/**\n * @type {typeof import('./data-utils').numberSort}\n */\nexport function numberSort(a, b) {\n  return a - b;\n}\n\n/**\n * @type {typeof import('./data-utils').getSortingFunction}\n */\nexport function getSortingFunction(fieldType) {\n  switch (fieldType) {\n    case ALL_FIELD_TYPES.real:\n    case ALL_FIELD_TYPES.integer:\n    case ALL_FIELD_TYPES.timestamp:\n      return numberSort;\n    default:\n      return undefined;\n  }\n}\n\n/**\n * round number with exact number of decimals\n * return as a string\n * @type {typeof import('./data-utils').preciseRound}\n */\nexport function preciseRound(num, decimals) {\n  const t = Math.pow(10, decimals);\n  return (\n    Math.round(\n      num * t + (decimals > 0 ? 1 : 0) * (Math.sign(num) * (10 / Math.pow(100, decimals)))\n    ) / t\n  ).toFixed(decimals);\n}\n\n/**\n * get number of decimals to round to for slider from step\n * @param {number} step\n * @returns {number} - number of decimal\n */\nexport function getRoundingDecimalFromStep(step) {\n  if (isNaN(step)) {\n    assert('step is not a number');\n    assert(step);\n  }\n\n  const splitZero = step.toString().split('.');\n  if (splitZero.length === 1) {\n    return 0;\n  }\n  return splitZero[1].length;\n}\n\n/**\n * Use in slider, given a number and an array of numbers, return the nears number from the array\n * @type {typeof import('./data-utils').snapToMarks}\n * @param value\n * @param marks\n */\nexport function snapToMarks(value, marks) {\n  // always use bin x0\n  const i = bisectLeft(marks, value);\n  if (i === 0) {\n    return marks[i];\n  } else if (i === marks.length) {\n    return marks[i - 1];\n  }\n  const idx = marks[i] - value < value - marks[i - 1] ? i : i - 1;\n  return marks[idx];\n}\n\n/**\n * If marks is provided, snap to marks, if not normalize to step\n * @type {typeof import('./data-utils').normalizeSliderValue}\n * @param val\n * @param minValue\n * @param step\n * @param marks\n */\nexport function normalizeSliderValue(val, minValue, step, marks) {\n  if (marks && marks.length) {\n    return snapToMarks(val, marks);\n  }\n\n  return roundValToStep(minValue, step, val);\n}\n\n/**\n * round the value to step for the slider\n * @type {typeof import('./data-utils').roundValToStep}\n * @param minValue\n * @param step\n * @param val\n * @returns - rounded number\n */\nexport function roundValToStep(minValue, step, val) {\n  if (!isNumber(step) || !isNumber(minValue)) {\n    return val;\n  }\n\n  const decimal = getRoundingDecimalFromStep(step);\n  const steps = Math.floor((val - minValue) / step);\n  let remain = val - (steps * step + minValue);\n\n  // has to round because javascript turns 0.1 into 0.9999999999999987\n  remain = Number(preciseRound(remain, 8));\n\n  let closest;\n  if (remain === 0) {\n    closest = val;\n  } else if (remain < step / 2) {\n    closest = steps * step + minValue;\n  } else {\n    closest = (steps + 1) * step + minValue;\n  }\n\n  // precise round return a string rounded to the defined decimal\n  const rounded = preciseRound(closest, decimal);\n\n  return Number(rounded);\n}\n\n/**\n * Get the value format based on field and format options\n * Used in render tooltip value\n * @type {typeof import('./data-utils').defaultFormatter}\n */\nexport const defaultFormatter = v => (notNullorUndefined(v) ? String(v) : '');\n\nexport const FIELD_DISPLAY_FORMAT = {\n  [ALL_FIELD_TYPES.string]: defaultFormatter,\n  [ALL_FIELD_TYPES.timestamp]: defaultFormatter,\n  [ALL_FIELD_TYPES.integer]: defaultFormatter,\n  [ALL_FIELD_TYPES.real]: defaultFormatter,\n  [ALL_FIELD_TYPES.boolean]: defaultFormatter,\n  [ALL_FIELD_TYPES.date]: defaultFormatter,\n  [ALL_FIELD_TYPES.geojson]: d =>\n    typeof d === 'string'\n      ? d\n      : isPlainObject(d)\n      ? JSON.stringify(d)\n      : Array.isArray(d)\n      ? `[${String(d)}]`\n      : ''\n};\n\n/**\n * Parse field value and type and return a string representation\n * @type {typeof import('./data-utils').parseFieldValue}\n */\nexport const parseFieldValue = (value, type) => {\n  if (!notNullorUndefined(value)) {\n    return '';\n  }\n\n  return FIELD_DISPLAY_FORMAT[type] ? FIELD_DISPLAY_FORMAT[type](value) : String(value);\n};\n\nconst arrayMoveMutate = (array, from, to) => {\n  array.splice(to < 0 ? array.length + to : to, 0, array.splice(from, 1)[0]);\n};\n\n/**\n *\n * @param {*} array\n * @param {*} from\n * @param {*} to\n */\nexport const arrayMove = (array, from, to) => {\n  array = array.slice();\n  arrayMoveMutate(array, from, to);\n  return array;\n};\n\n/**\n * Get the value format based on field and format options\n * Used in render tooltip value\n * @type {typeof import('./data-utils').getFormatter}\n * @param format\n * @param field\n */\nexport function getFormatter(format, field) {\n  if (!format) {\n    return defaultFormatter;\n  }\n  const tooltipFormat = Object.values(TOOLTIP_FORMATS).find(f => f[TOOLTIP_KEY] === format);\n\n  if (tooltipFormat) {\n    return applyDefaultFormat(tooltipFormat);\n  } else if (typeof format === 'string' && field) {\n    return applyCustomFormat(format, field);\n  }\n\n  return defaultFormatter;\n}\n\nexport function applyDefaultFormat(tooltipFormat) {\n  if (!tooltipFormat || !tooltipFormat.format) {\n    return defaultFormatter;\n  }\n\n  switch (tooltipFormat.type) {\n    case TOOLTIP_FORMAT_TYPES.DECIMAL:\n      return d3Format(tooltipFormat.format);\n    case TOOLTIP_FORMAT_TYPES.DATE:\n    case TOOLTIP_FORMAT_TYPES.DATE_TIME:\n      return datetimeFormatter(null)(tooltipFormat.format);\n    case TOOLTIP_FORMAT_TYPES.PERCENTAGE:\n      return v => `${d3Format(TOOLTIP_FORMATS.DECIMAL_DECIMAL_FIXED_2.format)(v)}%`;\n    case TOOLTIP_FORMAT_TYPES.BOOLEAN:\n      return getBooleanFormatter(tooltipFormat.format);\n    default:\n      return defaultFormatter;\n  }\n}\n\nexport function getBooleanFormatter(format) {\n  switch (format) {\n    case '01':\n      return v => (v ? '1' : '0');\n    case 'yn':\n      return v => (v ? 'yes' : 'no');\n    default:\n      return defaultFormatter;\n  }\n}\n// Allow user to specify custom tooltip format via config\nexport function applyCustomFormat(format, field) {\n  switch (field.type) {\n    case ALL_FIELD_TYPES.real:\n    case ALL_FIELD_TYPES.integer:\n      return d3Format(format);\n    case ALL_FIELD_TYPES.date:\n    case ALL_FIELD_TYPES.timestamp:\n      return datetimeFormatter(null)(format);\n    default:\n      return v => v;\n  }\n}\n\n/**\n * Format epoch milliseconds with a format string\n * @type {typeof import('./data-utils').datetimeFormatter} timezone\n */\nexport function datetimeFormatter(timezone) {\n  return timezone\n    ? format => ts =>\n        moment\n          .utc(ts)\n          .tz(timezone)\n          .format(format)\n    : format => ts => moment.utc(ts).format(format);\n}\n"]}