UNPKG

kepler.gl

Version:

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

449 lines (366 loc) 42.4 kB
'use strict'; Object.defineProperty(exports, "__esModule", { value: true }); var _objectWithoutProperties2 = require('babel-runtime/helpers/objectWithoutProperties'); var _objectWithoutProperties3 = _interopRequireDefault(_objectWithoutProperties2); var _defineProperty2 = require('babel-runtime/helpers/defineProperty'); var _defineProperty3 = _interopRequireDefault(_defineProperty2); var _extends3 = require('babel-runtime/helpers/extends'); var _extends4 = _interopRequireDefault(_extends3); var _toConsumableArray2 = require('babel-runtime/helpers/toConsumableArray'); var _toConsumableArray3 = _interopRequireDefault(_toConsumableArray2); exports.mergeFilters = mergeFilters; exports.mergeLayers = mergeLayers; exports.mergeInteractions = mergeInteractions; exports.mergeInteractionTooltipConfig = mergeInteractionTooltipConfig; exports.mergeLayerBlending = mergeLayerBlending; exports.validateSavedLayerColumns = validateSavedLayerColumns; exports.validateSavedVisualChannels = validateSavedVisualChannels; exports.validateLayerWithData = validateLayerWithData; exports.validateFilterWithData = validateFilterWithData; var _lodash = require('lodash.uniq'); var _lodash2 = _interopRequireDefault(_lodash); var _lodash3 = require('lodash.pick'); var _lodash4 = _interopRequireDefault(_lodash3); var _filterUtils = require('../utils/filter-utils'); var _defaultSettings = require('../constants/default-settings'); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } /** * Merge loaded filters with current state, if no fields or data are loaded * save it for later * * @param {Object} state * @param {Object[]} filtersToMerge * @return {Object} updatedState */ // Copyright (c) 2018 Uber Technologies, Inc. // // Permission is hereby granted, free of charge, to any person obtaining a copy // of this software and associated documentation files (the "Software"), to deal // in the Software without restriction, including without limitation the rights // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell // copies of the Software, and to permit persons to whom the Software is // furnished to do so, subject to the following conditions: // // The above copyright notice and this permission notice shall be included in // all copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN // THE SOFTWARE. function mergeFilters(state, filtersToMerge) { var merged = []; var unmerged = []; var datasets = state.datasets; if (!Array.isArray(filtersToMerge) || !filtersToMerge.length) { return state; } // merge filters filtersToMerge.forEach(function (filter) { // match filter.dataId with current datesets id // uploaded data need to have the same dataId with the filter if (datasets[filter.dataId]) { // datasets is already loaded var validateFilter = validateFilterWithData(datasets[filter.dataId], filter); if (validateFilter) { merged.push(validateFilter); } } else { // datasets not yet loaded unmerged.push(filter); } }); // filter data var updatedFilters = [].concat((0, _toConsumableArray3.default)(state.filters || []), merged); var datasetToFilter = (0, _lodash2.default)(merged.map(function (d) { return d.dataId; })); var updatedDataset = datasetToFilter.reduce(function (accu, dataId) { return (0, _extends4.default)({}, accu, (0, _defineProperty3.default)({}, dataId, (0, _extends4.default)({}, datasets[dataId], (0, _filterUtils.filterData)(datasets[dataId].allData, dataId, updatedFilters)))); }, datasets); return (0, _extends4.default)({}, state, { filters: updatedFilters, datasets: updatedDataset, filterToBeMerged: unmerged }); } /** * Merge layers from de-serialized state, if no fields or data are loaded * save it for later * * @param {object} state * @param {Object[]} layersToMerge * @return {Object} state */ function mergeLayers(state, layersToMerge) { var mergedLayer = []; var unmerged = []; var datasets = state.datasets; if (!Array.isArray(layersToMerge) || !layersToMerge.length) { return state; } layersToMerge.forEach(function (layer) { if (datasets[layer.config.dataId]) { // datasets are already loaded var validateLayer = validateLayerWithData(datasets[layer.config.dataId], layer, state.layerClasses); if (validateLayer) { mergedLayer.push(validateLayer); } } else { // datasets not yet loaded unmerged.push(layer); } }); var layers = [].concat((0, _toConsumableArray3.default)(state.layers), mergedLayer); var newLayerOrder = mergedLayer.map(function (_, i) { return state.layers.length + i; }); // put new layers in front of current layers var layerOrder = [].concat((0, _toConsumableArray3.default)(newLayerOrder), (0, _toConsumableArray3.default)(state.layerOrder)); return (0, _extends4.default)({}, state, { layers: layers, layerOrder: layerOrder, layerToBeMerged: unmerged }); } /** * Merge interactions with saved config * * @param {object} state * @param {Object} interactionToBeMerged * @return {Object} mergedState */ function mergeInteractions(state, interactionToBeMerged) { var merged = {}; var unmerged = {}; if (interactionToBeMerged) { Object.keys(interactionToBeMerged).forEach(function (key) { if (!state.interactionConfig[key]) { return; } var _ref = interactionToBeMerged[key] || {}, enabled = _ref.enabled, configSaved = (0, _objectWithoutProperties3.default)(_ref, ['enabled']); var configToMerge = configSaved; if (key === 'tooltip') { var _mergeInteractionTool = mergeInteractionTooltipConfig(state, configSaved), mergedTooltip = _mergeInteractionTool.mergedTooltip, unmergedTooltip = _mergeInteractionTool.unmergedTooltip; // merge new dataset tooltips with original dataset tooltips configToMerge = { fieldsToShow: (0, _extends4.default)({}, state.interactionConfig[key].config.fieldsToShow, mergedTooltip) }; if (Object.keys(unmergedTooltip).length) { unmerged.tooltip = { fieldsToShow: unmergedTooltip, enabled: enabled }; } } merged[key] = (0, _extends4.default)({}, state.interactionConfig[key], { enabled: enabled, config: (0, _lodash4.default)((0, _extends4.default)({}, state.interactionConfig[key].config, configToMerge), Object.keys(state.interactionConfig[key].config)) }); }); } return (0, _extends4.default)({}, state, { interactionConfig: (0, _extends4.default)({}, state.interactionConfig, merged), interactionToBeMerged: unmerged }); } /** * Merge interactionConfig.tooltip with saved config, * validate fieldsToShow * * @param {string} state * @param {Object} tooltipConfig * @return {Object} - {mergedTooltip: {}, unmergedTooltip: {}} */ function mergeInteractionTooltipConfig(state) { var tooltipConfig = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {}; var unmergedTooltip = {}; var mergedTooltip = {}; if (!tooltipConfig.fieldsToShow || !Object.keys(tooltipConfig.fieldsToShow).length) { return { mergedTooltip: mergedTooltip, unmergedTooltip: unmergedTooltip }; } for (var dataId in tooltipConfig.fieldsToShow) { if (!state.datasets[dataId]) { // is not yet loaded unmergedTooltip[dataId] = tooltipConfig.fieldsToShow[dataId]; } else { (function () { // if dataset is loaded var allFields = state.datasets[dataId].fields.map(function (d) { return d.name; }); var foundFieldsToShow = tooltipConfig.fieldsToShow[dataId].filter(function (name) { return allFields.includes(name); }); mergedTooltip[dataId] = foundFieldsToShow; })(); } } return { mergedTooltip: mergedTooltip, unmergedTooltip: unmergedTooltip }; } /** * Merge layerBlending with saved * * @param {object} state * @param {string} layerBlending * @return {object} merged state */ function mergeLayerBlending(state, layerBlending) { if (layerBlending && _defaultSettings.LAYER_BLENDINGS[layerBlending]) { return (0, _extends4.default)({}, state, { layerBlending: layerBlending }); } return state; } /** * Validate saved layer columns with new data, * update fieldIdx based on new fields * * @param {Object[]} fields * @param {Object} savedCols * @param {Object} emptyCols * @return {null | Object} - validated columns or null */ function validateSavedLayerColumns(fields, savedCols, emptyCols) { var colFound = {}; // find actual column fieldIdx, in case it has changed var allColFound = Object.keys(emptyCols).every(function (key) { var saved = savedCols[key]; colFound[key] = (0, _extends4.default)({}, emptyCols[key]); var fieldIdx = fields.findIndex(function (_ref2) { var name = _ref2.name; return name === saved; }); if (fieldIdx > -1) { // update found columns colFound[key].fieldIdx = fieldIdx; colFound[key].value = saved; return true; } // if col is optional, allow null value return emptyCols[key].optional || false; }); return allColFound && colFound; } /** * Validate saved visual channels config with new data, * refer to vis-state-schema.js VisualChannelSchemaV1 * * @param {Object[]} fields * @param {Object} visualChannels * @param {Object} savedLayer * @return {Object} - validated visual channel in config or {} */ function validateSavedVisualChannels(fields, visualChannels, savedLayer) { return Object.values(visualChannels).reduce(function (found, _ref3) { var field = _ref3.field, scale = _ref3.scale; var foundField = void 0; if (savedLayer.config[field]) { foundField = fields.find(function (fd) { return Object.keys(savedLayer.config[field]).every(function (key) { return savedLayer.config[field][key] === fd[key]; }); }); } return (0, _extends4.default)({}, found, foundField ? (0, _defineProperty3.default)({}, field, foundField) : {}, savedLayer.config[scale] ? (0, _defineProperty3.default)({}, scale, savedLayer.config[scale]) : {}); }, {}); } /** * Validate saved layer config with new data, * update fieldIdx based on new fields * * @param {Object[]} fields * @param {String} dataId * @param {Object} savedLayer * @param {Object} layerClasses * @return {null | Object} - validated layer or null */ function validateLayerWithData(_ref6, savedLayer, layerClasses) { var fields = _ref6.fields, dataId = _ref6.id; var type = savedLayer.type; // layer doesnt have a valid type if (!layerClasses.hasOwnProperty(type) || !savedLayer.config || !savedLayer.config.columns) { return null; } var newLayer = new layerClasses[type]({ id: savedLayer.id, dataId: dataId, label: savedLayer.config.label, color: savedLayer.config.color, isVisible: savedLayer.config.isVisible }); // find column fieldIdx var columns = validateSavedLayerColumns(fields, savedLayer.config.columns, newLayer.getLayerColumns()); if (!columns) { return null; } // visual channel field is saved to be {name, type} // find visual channel field by matching both name and type // refer to vis-state-schema.js VisualChannelSchemaV1 var foundVisualChannelConfigs = validateSavedVisualChannels(fields, newLayer.visualChannels, savedLayer); // copy visConfig over to emptyLayer to make sure it has all the props var visConfig = newLayer.copyLayerConfig(newLayer.config.visConfig, savedLayer.config.visConfig || {}, { notToDeepMerge: 'colorRange' }); newLayer.updateLayerConfig((0, _extends4.default)({ columns: columns, visConfig: visConfig }, foundVisualChannelConfigs)); return newLayer; } /** * Validate saved filter config with new data, * calculate domain and fieldIdx based new fields and data * * @param {Object[]} dataset.fields * @param {Object[]} dataset.allData * @param {Object} filter - filter to be validate * @return {Object | null} - validated filter */ function validateFilterWithData(_ref7, filter) { var fields = _ref7.fields, allData = _ref7.allData; // match filter.name to field.name var fieldIdx = fields.findIndex(function (_ref8) { var name = _ref8.name; return name === filter.name; }); if (fieldIdx < 0) { // if can't find field with same name, discharge filter return null; } var field = fields[fieldIdx]; var value = filter.value; // return filter type, default value, fieldType and fieldDomain from field var filterPropsFromField = (0, _filterUtils.getFilterProps)(allData, field); var matchedFilter = (0, _extends4.default)({}, (0, _filterUtils.getDefaultFilter)(filter.dataId), filter, filterPropsFromField, { freeze: true, fieldIdx: fieldIdx }); var _matchedFilter = matchedFilter, yAxis = _matchedFilter.yAxis; if (yAxis) { var matcheAxis = fields.find(function (_ref9) { var name = _ref9.name, type = _ref9.type; return name === yAxis.name && type === yAxis.type; }); matchedFilter = matcheAxis ? (0, _extends4.default)({}, matchedFilter, { yAxis: matcheAxis }, (0, _filterUtils.getFilterPlot)((0, _extends4.default)({}, matchedFilter, { yAxis: matcheAxis }), allData)) : matchedFilter; } matchedFilter.value = (0, _filterUtils.adjustValueToFilterDomain)(value, matchedFilter); if (matchedFilter.value === null) { // cannt adjust saved value to filter return null; } return matchedFilter; } //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/reducers/vis-state-merger.js"],"names":["mergeFilters","mergeLayers","mergeInteractions","mergeInteractionTooltipConfig","mergeLayerBlending","validateSavedLayerColumns","validateSavedVisualChannels","validateLayerWithData","validateFilterWithData","state","filtersToMerge","merged","unmerged","datasets","Array","isArray","length","forEach","filter","dataId","validateFilter","push","updatedFilters","filters","datasetToFilter","map","d","updatedDataset","reduce","accu","allData","filterToBeMerged","layersToMerge","mergedLayer","layer","config","validateLayer","layerClasses","layers","newLayerOrder","_","i","layerOrder","layerToBeMerged","interactionToBeMerged","Object","keys","interactionConfig","key","enabled","configSaved","configToMerge","mergedTooltip","unmergedTooltip","fieldsToShow","tooltip","tooltipConfig","allFields","fields","name","foundFieldsToShow","includes","layerBlending","LAYER_BLENDINGS","savedCols","emptyCols","colFound","allColFound","every","saved","fieldIdx","findIndex","value","optional","visualChannels","savedLayer","values","found","field","scale","foundField","find","fd","id","type","hasOwnProperty","columns","newLayer","label","color","isVisible","getLayerColumns","foundVisualChannelConfigs","visConfig","copyLayerConfig","notToDeepMerge","updateLayerConfig","filterPropsFromField","matchedFilter","freeze","yAxis","matcheAxis"],"mappings":"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","file":"vis-state-merger.js","sourcesContent":["// Copyright (c) 2018 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 uniq from 'lodash.uniq';\nimport pick from 'lodash.pick';\n\nimport {\n  getDefaultFilter,\n  getFilterProps,\n  getFilterPlot,\n  filterData,\n  adjustValueToFilterDomain\n} from 'utils/filter-utils';\n\nimport {LAYER_BLENDINGS} from 'constants/default-settings';\n\n/**\n * Merge loaded filters with current state, if no fields or data are loaded\n * save it for later\n *\n * @param {Object} state\n * @param {Object[]} filtersToMerge\n * @return {Object} updatedState\n */\nexport function mergeFilters(state, filtersToMerge) {\n  const merged = [];\n  const unmerged = [];\n  const {datasets} = state;\n\n  if (!Array.isArray(filtersToMerge) || !filtersToMerge.length) {\n    return state;\n  }\n\n  // merge filters\n  filtersToMerge.forEach(filter => {\n    // match filter.dataId with current datesets id\n    // uploaded data need to have the same dataId with the filter\n    if (datasets[filter.dataId]) {\n      // datasets is already loaded\n      const validateFilter = validateFilterWithData(\n        datasets[filter.dataId],\n        filter\n      );\n\n      if (validateFilter) {\n        merged.push(validateFilter);\n      }\n    } else {\n      // datasets not yet loaded\n      unmerged.push(filter);\n    }\n  });\n\n  // filter data\n  const updatedFilters = [...(state.filters || []), ...merged];\n  const datasetToFilter = uniq(merged.map(d => d.dataId));\n\n  const updatedDataset = datasetToFilter.reduce(\n    (accu, dataId) => ({\n      ...accu,\n      [dataId]: {\n        ...datasets[dataId],\n        ...filterData(datasets[dataId].allData, dataId, updatedFilters)\n      }\n    }),\n    datasets\n  );\n\n  return {\n    ...state,\n    filters: updatedFilters,\n    datasets: updatedDataset,\n    filterToBeMerged: unmerged\n  };\n}\n\n/**\n * Merge layers from de-serialized state, if no fields or data are loaded\n * save it for later\n *\n * @param {object} state\n * @param {Object[]} layersToMerge\n * @return {Object} state\n */\nexport function mergeLayers(state, layersToMerge) {\n  const mergedLayer = [];\n  const unmerged = [];\n\n  const {datasets} = state;\n\n  if (!Array.isArray(layersToMerge) || !layersToMerge.length) {\n    return state;\n  }\n\n  layersToMerge.forEach(layer => {\n    if (datasets[layer.config.dataId]) {\n      // datasets are already loaded\n      const validateLayer = validateLayerWithData(\n        datasets[layer.config.dataId],\n        layer,\n        state.layerClasses\n      );\n\n      if (validateLayer) {\n        mergedLayer.push(validateLayer);\n      }\n    } else {\n      // datasets not yet loaded\n      unmerged.push(layer);\n    }\n  });\n\n  const layers = [...state.layers, ...mergedLayer];\n  const newLayerOrder = mergedLayer.map((_, i) => state.layers.length + i);\n\n  // put new layers in front of current layers\n  const layerOrder = [...newLayerOrder, ...state.layerOrder];\n\n  return {\n    ...state,\n    layers,\n    layerOrder,\n    layerToBeMerged: unmerged\n  };\n}\n\n/**\n * Merge interactions with saved config\n *\n * @param {object} state\n * @param {Object} interactionToBeMerged\n * @return {Object} mergedState\n */\nexport function mergeInteractions(state, interactionToBeMerged) {\n  const merged = {};\n  const unmerged = {};\n\n  if (interactionToBeMerged) {\n    Object.keys(interactionToBeMerged).forEach(key => {\n      if (!state.interactionConfig[key]) {\n        return;\n      }\n\n      const {enabled, ...configSaved} = interactionToBeMerged[key] || {};\n      let configToMerge = configSaved;\n\n      if (key === 'tooltip') {\n        const {mergedTooltip, unmergedTooltip} = mergeInteractionTooltipConfig(\n          state,\n          configSaved\n        );\n\n        // merge new dataset tooltips with original dataset tooltips\n        configToMerge = {\n          fieldsToShow: {\n            ...state.interactionConfig[key].config.fieldsToShow,\n            ...mergedTooltip\n          }\n        };\n\n        if (Object.keys(unmergedTooltip).length) {\n          unmerged.tooltip = {fieldsToShow: unmergedTooltip, enabled};\n        }\n      }\n\n      merged[key] = {\n        ...state.interactionConfig[key],\n        enabled,\n        config: pick(\n          {\n            ...state.interactionConfig[key].config,\n            ...configToMerge\n          },\n          Object.keys(state.interactionConfig[key].config)\n        )\n      };\n    });\n  }\n\n  return {\n    ...state,\n    interactionConfig: {\n      ...state.interactionConfig,\n      ...merged\n    },\n    interactionToBeMerged: unmerged\n  };\n}\n\n/**\n * Merge interactionConfig.tooltip with saved config,\n * validate fieldsToShow\n *\n * @param {string} state\n * @param {Object} tooltipConfig\n * @return {Object} - {mergedTooltip: {}, unmergedTooltip: {}}\n */\nexport function mergeInteractionTooltipConfig(state, tooltipConfig = {}) {\n  const unmergedTooltip = {};\n  const mergedTooltip = {};\n\n  if (\n    !tooltipConfig.fieldsToShow ||\n    !Object.keys(tooltipConfig.fieldsToShow).length\n  ) {\n    return {mergedTooltip, unmergedTooltip};\n  }\n\n  for (const dataId in tooltipConfig.fieldsToShow) {\n    if (!state.datasets[dataId]) {\n      // is not yet loaded\n      unmergedTooltip[dataId] = tooltipConfig.fieldsToShow[dataId];\n    } else {\n      // if dataset is loaded\n      const allFields = state.datasets[dataId].fields.map(d => d.name);\n      const foundFieldsToShow = tooltipConfig.fieldsToShow[dataId].filter(\n        name => allFields.includes(name)\n      );\n\n      mergedTooltip[dataId] = foundFieldsToShow;\n    }\n  }\n\n  return {mergedTooltip, unmergedTooltip};\n}\n/**\n * Merge layerBlending with saved\n *\n * @param {object} state\n * @param {string} layerBlending\n * @return {object} merged state\n */\nexport function mergeLayerBlending(state, layerBlending) {\n  if (layerBlending && LAYER_BLENDINGS[layerBlending]) {\n    return {\n      ...state,\n      layerBlending\n    };\n  }\n\n  return state;\n}\n\n/**\n * Validate saved layer columns with new data,\n * update fieldIdx based on new fields\n *\n * @param {Object[]} fields\n * @param {Object} savedCols\n * @param {Object} emptyCols\n * @return {null | Object} - validated columns or null\n */\n\nexport function validateSavedLayerColumns(fields, savedCols, emptyCols) {\n  const colFound = {};\n  // find actual column fieldIdx, in case it has changed\n  const allColFound = Object.keys(emptyCols).every(key => {\n    const saved = savedCols[key];\n    colFound[key] = {...emptyCols[key]};\n\n    const fieldIdx = fields.findIndex(({name}) => name === saved);\n\n    if (fieldIdx > -1) {\n      // update found columns\n      colFound[key].fieldIdx = fieldIdx;\n      colFound[key].value = saved;\n      return true;\n    }\n\n    // if col is optional, allow null value\n    return emptyCols[key].optional || false;\n  });\n\n  return allColFound && colFound;\n}\n\n/**\n * Validate saved visual channels config with new data,\n * refer to vis-state-schema.js VisualChannelSchemaV1\n *\n * @param {Object[]} fields\n * @param {Object} visualChannels\n * @param {Object} savedLayer\n * @return {Object} - validated visual channel in config or {}\n */\nexport function validateSavedVisualChannels(\n  fields,\n  visualChannels,\n  savedLayer\n) {\n  return Object.values(visualChannels).reduce((found, {field, scale}) => {\n    let foundField;\n    if (savedLayer.config[field]) {\n      foundField = fields.find(fd =>\n        Object.keys(savedLayer.config[field]).every(\n          key => savedLayer.config[field][key] === fd[key]\n        )\n      );\n    }\n\n    return {\n      ...found,\n      ...(foundField ? {[field]: foundField} : {}),\n      ...(savedLayer.config[scale] ? {[scale]: savedLayer.config[scale]} : {})\n    };\n  }, {});\n}\n\n/**\n * Validate saved layer config with new data,\n * update fieldIdx based on new fields\n *\n * @param {Object[]} fields\n * @param {String} dataId\n * @param {Object} savedLayer\n * @param {Object} layerClasses\n * @return {null | Object} - validated layer or null\n */\nexport function validateLayerWithData({fields, id: dataId}, savedLayer, layerClasses) {\n  const {type} = savedLayer;\n  // layer doesnt have a valid type\n  if (\n    !layerClasses.hasOwnProperty(type) ||\n    !savedLayer.config ||\n    !savedLayer.config.columns\n  ) {\n    return null;\n  }\n\n  const newLayer = new layerClasses[type]({\n    id: savedLayer.id,\n    dataId,\n    label: savedLayer.config.label,\n    color: savedLayer.config.color,\n    isVisible: savedLayer.config.isVisible\n  });\n\n  // find column fieldIdx\n  const columns = validateSavedLayerColumns(\n    fields,\n    savedLayer.config.columns,\n    newLayer.getLayerColumns()\n  );\n\n  if (!columns) {\n    return null;\n  }\n\n  // visual channel field is saved to be {name, type}\n  // find visual channel field by matching both name and type\n  // refer to vis-state-schema.js VisualChannelSchemaV1\n  const foundVisualChannelConfigs = validateSavedVisualChannels(\n    fields,\n    newLayer.visualChannels,\n    savedLayer\n  );\n\n  // copy visConfig over to emptyLayer to make sure it has all the props\n  const visConfig = newLayer.copyLayerConfig(\n    newLayer.config.visConfig,\n    savedLayer.config.visConfig || {},\n    {notToDeepMerge: 'colorRange'}\n  );\n\n  newLayer.updateLayerConfig({\n    columns,\n    visConfig,\n    ...foundVisualChannelConfigs\n  });\n\n  return newLayer;\n}\n\n/**\n * Validate saved filter config with new data,\n * calculate domain and fieldIdx based new fields and data\n *\n * @param {Object[]} dataset.fields\n * @param {Object[]} dataset.allData\n * @param {Object} filter - filter to be validate\n * @return {Object | null} - validated filter\n */\nexport function validateFilterWithData({fields, allData}, filter) {\n  // match filter.name to field.name\n  const fieldIdx = fields.findIndex(({name}) => name === filter.name);\n\n  if (fieldIdx < 0) {\n    // if can't find field with same name, discharge filter\n    return null;\n  }\n\n  const field = fields[fieldIdx];\n  const value = filter.value;\n\n  // return filter type, default value, fieldType and fieldDomain from field\n  const filterPropsFromField = getFilterProps(allData, field);\n\n  let matchedFilter = {\n    ...getDefaultFilter(filter.dataId),\n    ...filter,\n    ...filterPropsFromField,\n    freeze: true,\n    fieldIdx\n  };\n\n  const {yAxis} = matchedFilter;\n  if (yAxis) {\n    const matcheAxis = fields.find(\n      ({name, type}) => name === yAxis.name && type === yAxis.type\n    );\n\n    matchedFilter = matcheAxis\n      ? {\n          ...matchedFilter,\n          yAxis: matcheAxis,\n          ...getFilterPlot({...matchedFilter, yAxis: matcheAxis}, allData)\n        }\n      : matchedFilter;\n  }\n\n  matchedFilter.value = adjustValueToFilterDomain(value, matchedFilter);\n\n  if (matchedFilter.value === null) {\n    // cannt adjust saved value to filter\n    return null;\n  }\n\n  return matchedFilter;\n}\n"]}