import { RX_ARRAY_NOTATION } from '../constants/regex' import { identity } from './identity' import { isArray, isNull, isObject, isUndefinedOrNull } from './inspect' /** * Get property defined by dot/array notation in string, returns undefined if not found * * @link https://gist.github.com/jeneg/9767afdcca45601ea44930ea03e0febf#gistcomment-1935901 * * @param {Object} obj * @param {string|Array} path * @return {*} */ export const getRaw = (obj, path, defaultValue = undefined) => { // Handle array of path values path = isArray(path) ? path.join('.') : path // If no path or no object passed if (!path || !isObject(obj)) { return defaultValue } // Handle edge case where user has dot(s) in top-level item field key // See https://github.com/bootstrap-vue/bootstrap-vue/issues/2762 // Switched to `in` operator vs `hasOwnProperty` to handle obj.prototype getters // https://github.com/bootstrap-vue/bootstrap-vue/issues/3463 if (path in obj) { return obj[path] } // Handle string array notation (numeric indices only) path = String(path).replace(RX_ARRAY_NOTATION, '.$1') const steps = path.split('.').filter(identity) // Handle case where someone passes a string of only dots if (steps.length === 0) { return defaultValue } // Traverse path in object to find result // Switched to `in` operator vs `hasOwnProperty` to handle obj.prototype getters // https://github.com/bootstrap-vue/bootstrap-vue/issues/3463 return steps.every(step => isObject(obj) && step in obj && !isUndefinedOrNull((obj = obj[step]))) ? obj : isNull(obj) ? null : defaultValue } /** * Get property defined by dot/array notation in string. * * @link https://gist.github.com/jeneg/9767afdcca45601ea44930ea03e0febf#gistcomment-1935901 * * @param {Object} obj * @param {string|Array} path * @param {*} defaultValue (optional) * @return {*} */ export const get = (obj, path, defaultValue = null) => { const value = getRaw(obj, path) return isUndefinedOrNull(value) ? defaultValue : value }