UNPKG

kepler.gl

Version:

kepler.gl is a webgl based application to visualize large scale location data in the browser

330 lines (318 loc) 41.9 kB
"use strict"; var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault"); Object.defineProperty(exports, "__esModule", { value: true }); exports.datasetColorMaker = exports.addTimeLabel = void 0; exports.findDefaultColorField = findDefaultColorField; exports.getFieldFormatLabels = getFieldFormatLabels; exports.getFormatLabels = void 0; exports.validateInputData = validateInputData; var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty")); var _typeof2 = _interopRequireDefault(require("@babel/runtime/helpers/typeof")); var _regenerator = _interopRequireDefault(require("@babel/runtime/regenerator")); var _constants = require("@kepler.gl/constants"); var _commonUtils = require("@kepler.gl/common-utils"); var _typeAnalyzer = require("type-analyzer"); var _assert = _interopRequireDefault(require("assert")); var _utils = require("./utils"); var _dataUtils = require("./data-utils"); var _format = require("./format"); var _colorUtils = require("./color-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; } var _marked = /*#__PURE__*/_regenerator["default"].mark(generateColor); // SPDX-License-Identifier: MIT // Copyright contributors to the kepler.gl project // 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 = exports.datasetColorMaker = generateColor(); /** * Field name prefixes and suffixes which should not be considered * as metrics. Fields will still be included if a 'metric word' * is found on the field name, however. */ var EXCLUDED_DEFAULT_FIELDS = [ // Serial numbers and identification numbers '_id', 'id', 'index', 'uuid', 'guid', 'uid', 'gid', 'serial', // Geographic IDs are unlikely to be interesting to color 'zip', 'code', 'post', 'region', 'fips', 'cbgs', 'h3', 's2', // Geographic coords (but not z/elevation/altitude // since that might be a metric) 'lat', 'lon', 'lng', 'latitude', 'longitude', '_x', '_y']; /** * Prefixes and suffixes that indicate a field is a metric. * * Note that these are in order of preference, first being * most preferred. */ var METRIC_DEFAULT_FIELDS = ['metric', 'value', 'sum', 'count', 'unique', 'mean', 'mode', 'median', 'max', 'min', 'deviation', 'variance', 'p99', 'p95', 'p75', 'p50', 'p25', 'p05', // Abbreviations are less preferred 'cnt', 'val']; /** * Choose a field to use as the default color field of a layer. * * The heuristic is: * * First, exclude fields that are on the exclusion list and don't * have names that suggest they contain metrics. Also exclude * field names that are blank. * * Next, look for a field that is of real type and contains one * of the preferred names (in order of the preferred names). * * Next, look for a field that is of integer type and contains * one of the preferred names (in order of the preferred names). * * Next, look for the first field that is of real type (in order * of field index). * * Next, look for the first field that is of integer type (in * order of field index). * * It's possible no field will be chosen (i.e. because all fields * are strings.) * * @param dataset */ function findDefaultColorField(_ref) { var fields = _ref.fields, _ref$fieldPairs = _ref.fieldPairs, fieldPairs = _ref$fieldPairs === void 0 ? [] : _ref$fieldPairs; var fieldsWithoutExcluded = fields.filter(function (field) { if (field.type !== _constants.ALL_FIELD_TYPES.real) { // Only select numeric fields. return false; } if (fieldPairs.find(function (pair) { return pair.pair.lat.value === field.name || pair.pair.lng.value === field.name; })) { // Do not permit lat, lon fields return false; } var normalizedFieldName = field.name.toLowerCase(); if (normalizedFieldName === '') { // Special case excluded name when the name is blank. return false; } var hasExcluded = EXCLUDED_DEFAULT_FIELDS.find(function (f) { return normalizedFieldName.startsWith(f) || normalizedFieldName.endsWith(f); }); var hasInclusion = METRIC_DEFAULT_FIELDS.find(function (f) { return normalizedFieldName.startsWith(f) || normalizedFieldName.endsWith(f); }); return !hasExcluded || hasInclusion; }); var sortedFields = fieldsWithoutExcluded.sort(function (left, right) { var normalizedLeft = left.name.toLowerCase(); var normalizedRight = right.name.toLowerCase(); var leftHasInclusion = METRIC_DEFAULT_FIELDS.findIndex(function (f) { return normalizedLeft.startsWith(f) || normalizedLeft.endsWith(f); }); var rightHasInclusion = METRIC_DEFAULT_FIELDS.findIndex(function (f) { return normalizedRight.startsWith(f) || normalizedRight.endsWith(f); }); if (leftHasInclusion !== rightHasInclusion) { if (leftHasInclusion === -1) { // Elements that do not have the inclusion list should go after those that do. return 1; } else if (rightHasInclusion === -1) { // Elements that do have the inclusion list should go before those that don't. return -1; } // Compare based on order in the inclusion list return leftHasInclusion - rightHasInclusion; } // Compare based on type if (left.type !== right.type) { if (left.type === _constants.ALL_FIELD_TYPES.real) { return -1; } // left is an integer and right is not // and reals come before integers return 1; } // Finally, order based on the order in the datasets columns // @ts-expect-error return left.index - right.index; }); if (sortedFields.length) { // There was a best match return sortedFields[0]; } // No matches return null; } /** * Validate input data, adding missing field types, rename duplicate columns */ function validateInputData(data) { if (!(0, _utils.isPlainObject)(data)) { (0, _assert["default"])('addDataToMap Error: dataset.data cannot be null'); return null; } else if (!Array.isArray(data.fields)) { (0, _assert["default"])('addDataToMap Error: expect dataset.data.fields to be an array'); return null; } else if (!Array.isArray(data.rows) && !Array.isArray(data.cols)) { (0, _assert["default"])('addDataToMap Error: expect dataset.data.rows or cols to be an array'); return null; } var fields = data.fields, rows = data.rows, cols = data.cols; // check if all fields has name, format and type var allValid = fields.every(function (f, i) { if (!(0, _utils.isPlainObject)(f)) { (0, _assert["default"])("fields needs to be an array of object, but find ".concat((0, _typeof2["default"])(f))); fields[i] = { name: "column_".concat(i), type: _constants.ALL_FIELD_TYPES.string }; } if (!f.name) { (0, _assert["default"])("field.name is required but missing in ".concat(JSON.stringify(f))); // assign a name fields[i].name = "column_".concat(i); } if (!f.type || !_constants.ALL_FIELD_TYPES[f.type]) { (0, _assert["default"])("unknown field type ".concat(f.type)); return false; } if (!f.analyzerType) { (0, _assert["default"])("field ".concat(i, " missing analyzerType")); return false; } // check time format is correct based on first 10 not empty element if (f.type === _constants.ALL_FIELD_TYPES.timestamp) { var sample = (cols ? findNonEmptyRowsAtFieldArrow(cols, i, 10) : findNonEmptyRowsAtField(rows, i, 10)).map(function (r) { return { ts: r[i] }; }); var analyzedType = _typeAnalyzer.Analyzer.computeColMeta(sample)[0]; return analyzedType && analyzedType.category === 'TIME' && analyzedType.format === f.format; } // check existing string field is H3 type if (f.type === _constants.ALL_FIELD_TYPES.string) { var _sample = (cols ? findNonEmptyRowsAtFieldArrow(cols, i, 10) : findNonEmptyRowsAtField(rows, i, 10)).map(function (r) { return r[i]; }); return _sample.every(function (item) { return !(0, _commonUtils.h3IsValid)(item); }); } return true; }); if (allValid) { return { rows: rows, fields: fields, cols: cols }; } // if any field has missing type, recalculate it for everyone // because we simply lost faith in humanity var sampleData = cols ? (0, _commonUtils.getSampleForTypeAnalyzeArrow)(cols, fields.map(function (f) { return f.name; })) : (0, _commonUtils.getSampleForTypeAnalyze)({ fields: fields.map(function (f) { return f.name; }), rows: rows }); var fieldOrder = fields.map(function (f) { return f.name; }); var meta = (0, _commonUtils.getFieldsFromData)(sampleData, fieldOrder); var updatedFields = fields.map(function (f, i) { return _objectSpread(_objectSpread({}, f), {}, { type: meta[i].type, format: meta[i].format, analyzerType: meta[i].analyzerType }); }); return _objectSpread({ fields: updatedFields, rows: rows }, cols ? { cols: cols } : {}); } function findNonEmptyRowsAtField(rows, fieldIdx, total) { var sample = []; var i = 0; while (sample.length < total && i < rows.length) { var _rows$i; if ((0, _commonUtils.notNullorUndefined)((_rows$i = rows[i]) === null || _rows$i === void 0 ? void 0 : _rows$i[fieldIdx])) { sample.push(rows[i]); } i++; } return sample; } function findNonEmptyRowsAtFieldArrow(cols, fieldIdx, total) { var sample = []; var numRows = cols[fieldIdx].length; var i = 0; while (sample.length < total && i < numRows) { if ((0, _commonUtils.notNullorUndefined)(cols[fieldIdx].get(i))) { var row = cols.map(function (col) { return col.get(i); }); sample.push(row); } i++; } return sample; } var TIME_DISPLAY = '2020-05-11 14:00'; var addTimeLabel = exports.addTimeLabel = function addTimeLabel(formats) { return formats.map(function (f) { return _objectSpread(_objectSpread({}, f), {}, { label: f.type === _constants.TOOLTIP_FORMAT_TYPES.DATE_TIME || f.type === _constants.TOOLTIP_FORMAT_TYPES.DATE ? (0, _dataUtils.getFormatter)((0, _format.getFormatValue)(f))(TIME_DISPLAY) : f.label }); }); }; function getFieldFormatLabels(fieldType) { var tooltipTypes = fieldType && _constants.FIELD_OPTS[fieldType].format.tooltip || []; var formatLabels = Object.values(_constants.TOOLTIP_FORMATS).filter(function (t) { return tooltipTypes.includes(t.type); }); return addTimeLabel(formatLabels); } var getFormatLabels = exports.getFormatLabels = function getFormatLabels(fields, fieldName) { var _fields$find; var fieldType = (_fields$find = fields.find(function (f) { return f.name === fieldName; })) === null || _fields$find === void 0 ? void 0 : _fields$find.type; return getFieldFormatLabels(fieldType); }; //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_constants","require","_commonUtils","_typeAnalyzer","_assert","_interopRequireDefault","_utils","_dataUtils","_format","_colorUtils","ownKeys","e","r","t","Object","keys","getOwnPropertySymbols","o","filter","getOwnPropertyDescriptor","enumerable","push","apply","_objectSpread","arguments","length","forEach","_defineProperty2","getOwnPropertyDescriptors","defineProperties","defineProperty","_marked","_regenerator","mark","generateColor","datasetColors","map","hexToRgb","index","wrap","generateColor$","_context","prev","next","stop","datasetColorMaker","exports","EXCLUDED_DEFAULT_FIELDS","METRIC_DEFAULT_FIELDS","findDefaultColorField","_ref","fields","_ref$fieldPairs","fieldPairs","fieldsWithoutExcluded","field","type","ALL_FIELD_TYPES","real","find","pair","lat","value","name","lng","normalizedFieldName","toLowerCase","hasExcluded","f","startsWith","endsWith","hasInclusion","sortedFields","sort","left","right","normalizedLeft","normalizedRight","leftHasInclusion","findIndex","rightHasInclusion","validateInputData","data","isPlainObject","assert","Array","isArray","rows","cols","allValid","every","i","concat","_typeof2","string","JSON","stringify","analyzerType","timestamp","sample","findNonEmptyRowsAtFieldArrow","findNonEmptyRowsAtField","ts","analyzedType","Analyzer","computeColMeta","category","format","item","h3IsValid","sampleData","getSampleForTypeAnalyzeArrow","getSampleForTypeAnalyze","fieldOrder","meta","getFieldsFromData","updatedFields","fieldIdx","total","_rows$i","notNullorUndefined","numRows","get","row","col","TIME_DISPLAY","addTimeLabel","formats","label","TOOLTIP_FORMAT_TYPES","DATE_TIME","DATE","getFormatter","getFormatValue","getFieldFormatLabels","fieldType","tooltipTypes","FIELD_OPTS","tooltip","formatLabels","values","TOOLTIP_FORMATS","includes","getFormatLabels","fieldName","_fields$find"],"sources":["../src/dataset-utils.ts"],"sourcesContent":["// SPDX-License-Identifier: MIT\n// Copyright contributors to the kepler.gl project\n\nimport * as arrow from 'apache-arrow';\nimport {\n  ALL_FIELD_TYPES,\n  FIELD_OPTS,\n  TOOLTIP_FORMATS,\n  TOOLTIP_FORMAT_TYPES\n} from '@kepler.gl/constants';\nimport {\n  getSampleForTypeAnalyze,\n  getSampleForTypeAnalyzeArrow,\n  getFieldsFromData\n} from '@kepler.gl/common-utils';\nimport {Analyzer} from 'type-analyzer';\nimport assert from 'assert';\n\nimport {\n  ProcessorResult,\n  RGBColor,\n  Field,\n  FieldPair,\n  TimeLabelFormat,\n  TooltipFields,\n  ProtoDataset\n} from '@kepler.gl/types';\nimport {TooltipFormat} from '@kepler.gl/constants';\nimport {notNullorUndefined, h3IsValid} from '@kepler.gl/common-utils';\n\nimport {isPlainObject} from './utils';\nimport {getFormatter} from './data-utils';\nimport {getFormatValue} from './format';\nimport {hexToRgb} from './color-utils';\n\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(): Generator<RGBColor> {\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\n/**\n * Field name prefixes and suffixes which should not be considered\n * as metrics. Fields will still be included if a 'metric word'\n * is found on the field name, however.\n */\nconst EXCLUDED_DEFAULT_FIELDS = [\n  // Serial numbers and identification numbers\n  '_id',\n  'id',\n  'index',\n  'uuid',\n  'guid',\n  'uid',\n  'gid',\n  'serial',\n  // Geographic IDs are unlikely to be interesting to color\n  'zip',\n  'code',\n  'post',\n  'region',\n  'fips',\n  'cbgs',\n  'h3',\n  's2',\n  // Geographic coords (but not z/elevation/altitude\n  // since that might be a metric)\n  'lat',\n  'lon',\n  'lng',\n  'latitude',\n  'longitude',\n  '_x',\n  '_y'\n];\n\n/**\n * Prefixes and suffixes that indicate a field is a metric.\n *\n * Note that these are in order of preference, first being\n * most preferred.\n */\nconst METRIC_DEFAULT_FIELDS = [\n  'metric',\n  'value',\n  'sum',\n  'count',\n  'unique',\n  'mean',\n  'mode',\n  'median',\n  'max',\n  'min',\n  'deviation',\n  'variance',\n  'p99',\n  'p95',\n  'p75',\n  'p50',\n  'p25',\n  'p05',\n  // Abbreviations are less preferred\n  'cnt',\n  'val'\n];\n\n/**\n * Choose a field to use as the default color field of a layer.\n *\n * The heuristic is:\n *\n * First, exclude fields that are on the exclusion list and don't\n * have names that suggest they contain metrics. Also exclude\n * field names that are blank.\n *\n * Next, look for a field that is of real type and contains one\n * of the preferred names (in order of the preferred names).\n *\n * Next, look for a field that is of integer type and contains\n * one of the preferred names (in order of the preferred names).\n *\n * Next, look for the first field that is of real type (in order\n * of field index).\n *\n * Next, look for the first field that is of integer type (in\n * order of field index).\n *\n * It's possible no field will be chosen (i.e. because all fields\n * are strings.)\n *\n * @param dataset\n */\nexport function findDefaultColorField({\n  fields,\n  fieldPairs = []\n}: {\n  fields: Field[];\n  fieldPairs: FieldPair[];\n}): null | Field {\n  const fieldsWithoutExcluded = fields.filter(field => {\n    if (field.type !== ALL_FIELD_TYPES.real) {\n      // Only select numeric fields.\n      return false;\n    }\n    if (\n      fieldPairs.find(\n        pair => pair.pair.lat.value === field.name || pair.pair.lng.value === field.name\n      )\n    ) {\n      // Do not permit lat, lon fields\n      return false;\n    }\n\n    const normalizedFieldName = field.name.toLowerCase();\n    if (normalizedFieldName === '') {\n      // Special case excluded name when the name is blank.\n      return false;\n    }\n    const hasExcluded = EXCLUDED_DEFAULT_FIELDS.find(\n      f => normalizedFieldName.startsWith(f) || normalizedFieldName.endsWith(f)\n    );\n    const hasInclusion = METRIC_DEFAULT_FIELDS.find(\n      f => normalizedFieldName.startsWith(f) || normalizedFieldName.endsWith(f)\n    );\n    return !hasExcluded || hasInclusion;\n  });\n\n  const sortedFields = fieldsWithoutExcluded.sort((left, right) => {\n    const normalizedLeft = left.name.toLowerCase();\n    const normalizedRight = right.name.toLowerCase();\n    const leftHasInclusion = METRIC_DEFAULT_FIELDS.findIndex(\n      f => normalizedLeft.startsWith(f) || normalizedLeft.endsWith(f)\n    );\n    const rightHasInclusion = METRIC_DEFAULT_FIELDS.findIndex(\n      f => normalizedRight.startsWith(f) || normalizedRight.endsWith(f)\n    );\n    if (leftHasInclusion !== rightHasInclusion) {\n      if (leftHasInclusion === -1) {\n        // Elements that do not have the inclusion list should go after those that do.\n        return 1;\n      } else if (rightHasInclusion === -1) {\n        // Elements that do have the inclusion list should go before those that don't.\n        return -1;\n      }\n      // Compare based on order in the inclusion list\n      return leftHasInclusion - rightHasInclusion;\n    }\n\n    // Compare based on type\n    if (left.type !== right.type) {\n      if (left.type === ALL_FIELD_TYPES.real) {\n        return -1;\n      }\n      // left is an integer and right is not\n      // and reals come before integers\n      return 1;\n    }\n\n    // Finally, order based on the order in the datasets columns\n    // @ts-expect-error\n    return left.index - right.index;\n  });\n\n  if (sortedFields.length) {\n    // There was a best match\n    return sortedFields[0];\n  }\n  // No matches\n  return null;\n}\n\n/**\n * Validate input data, adding missing field types, rename duplicate columns\n */\nexport function validateInputData(data: ProtoDataset['data']): ProcessorResult {\n  if (!isPlainObject(data)) {\n    assert('addDataToMap Error: dataset.data cannot be null');\n    return null;\n  } else if (!Array.isArray(data.fields)) {\n    assert('addDataToMap Error: expect dataset.data.fields to be an array');\n    return null;\n  } else if (!Array.isArray(data.rows) && !Array.isArray(data.cols)) {\n    assert('addDataToMap Error: expect dataset.data.rows or cols to be an array');\n    return null;\n  }\n\n  const {fields, rows, cols} = data;\n\n  // check if all fields has name, format and type\n  const allValid = fields.every((f, i) => {\n    if (!isPlainObject(f)) {\n      assert(`fields needs to be an array of object, but find ${typeof f}`);\n      fields[i] = {name: `column_${i}`, type: ALL_FIELD_TYPES.string};\n    }\n\n    if (!f.name) {\n      assert(`field.name is required but missing in ${JSON.stringify(f)}`);\n      // assign a name\n      fields[i].name = `column_${i}`;\n    }\n\n    if (!f.type || !ALL_FIELD_TYPES[f.type]) {\n      assert(`unknown field type ${f.type}`);\n      return false;\n    }\n\n    if (!f.analyzerType) {\n      assert(`field ${i} missing analyzerType`);\n      return false;\n    }\n\n    // check time format is correct based on first 10 not empty element\n    if (f.type === ALL_FIELD_TYPES.timestamp) {\n      const sample = (\n        cols ? findNonEmptyRowsAtFieldArrow(cols, i, 10) : findNonEmptyRowsAtField(rows, i, 10)\n      ).map(r => ({ts: r[i]}));\n      const analyzedType = Analyzer.computeColMeta(sample)[0];\n      return analyzedType && analyzedType.category === 'TIME' && analyzedType.format === f.format;\n    }\n\n    // check existing string field is H3 type\n    if (f.type === ALL_FIELD_TYPES.string) {\n      const sample = (\n        cols ? findNonEmptyRowsAtFieldArrow(cols, i, 10) : findNonEmptyRowsAtField(rows, i, 10)\n      ).map(r => r[i]);\n      return sample.every(item => !h3IsValid(item));\n    }\n\n    return true;\n  });\n\n  if (allValid) {\n    return {rows, fields, cols};\n  }\n\n  // if any field has missing type, recalculate it for everyone\n  // because we simply lost faith in humanity\n  const sampleData = cols\n    ? getSampleForTypeAnalyzeArrow(\n        cols,\n        fields.map(f => f.name)\n      )\n    : getSampleForTypeAnalyze({\n        fields: fields.map(f => f.name),\n        rows\n      });\n  const fieldOrder = fields.map(f => f.name);\n  const meta = getFieldsFromData(sampleData, fieldOrder);\n  const updatedFields = fields.map((f, i) => ({\n    ...f,\n    type: meta[i].type,\n    format: meta[i].format,\n    analyzerType: meta[i].analyzerType\n  }));\n\n  return {fields: updatedFields, rows, ...(cols ? {cols} : {})};\n}\n\nfunction findNonEmptyRowsAtField(rows: unknown[][], fieldIdx: number, total: number): any[] {\n  const sample: any[] = [];\n  let i = 0;\n  while (sample.length < total && i < rows.length) {\n    if (notNullorUndefined(rows[i]?.[fieldIdx])) {\n      sample.push(rows[i]);\n    }\n    i++;\n  }\n  return sample;\n}\n\nfunction findNonEmptyRowsAtFieldArrow(\n  cols: arrow.Vector[],\n  fieldIdx: number,\n  total: number\n): any[] {\n  const sample: any[] = [];\n  const numRows = cols[fieldIdx].length;\n  let i = 0;\n  while (sample.length < total && i < numRows) {\n    if (notNullorUndefined(cols[fieldIdx].get(i))) {\n      const row = cols.map(col => col.get(i));\n      sample.push(row);\n    }\n    i++;\n  }\n  return sample;\n}\n\nconst TIME_DISPLAY = '2020-05-11 14:00';\n\nexport const addTimeLabel = (formats: TimeLabelFormat[]) =>\n  formats.map(f => ({\n    ...f,\n    label:\n      f.type === TOOLTIP_FORMAT_TYPES.DATE_TIME || f.type === TOOLTIP_FORMAT_TYPES.DATE\n        ? getFormatter(getFormatValue(f))(TIME_DISPLAY)\n        : f.label\n  }));\n\nexport function getFieldFormatLabels(fieldType?: string): TooltipFormat[] {\n  const tooltipTypes = (fieldType && FIELD_OPTS[fieldType].format.tooltip) || [];\n  const formatLabels: TimeLabelFormat[] = Object.values(TOOLTIP_FORMATS).filter(t =>\n    tooltipTypes.includes(t.type)\n  );\n  return addTimeLabel(formatLabels);\n}\n\nexport const getFormatLabels = (fields: TooltipFields[], fieldName: string): TooltipFormat[] => {\n  const fieldType = fields.find(f => f.name === fieldName)?.type;\n  return getFieldFormatLabels(fieldType);\n};\n"],"mappings":"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","ignoreList":[]}