mirror of
https://gitlab.com/bramw/baserow.git
synced 2025-04-22 12:22:24 +00:00
155 lines
4.7 KiB
JavaScript
155 lines
4.7 KiB
JavaScript
import _ from 'lodash'
|
|
|
|
/**
|
|
* Clones the provided JavaScript object and returns that one.
|
|
*
|
|
* @param o
|
|
* @return {object}
|
|
*/
|
|
export function clone(o) {
|
|
return JSON.parse(JSON.stringify(o))
|
|
}
|
|
|
|
/**
|
|
* Creates an object where the key indicates the line number and the value is
|
|
* the string that must be shown on that line number. The line number matches
|
|
* the line number if the value would be stringified with an indent of 4
|
|
* characters `JSON.stringify(value, null, 4)`. The correct value is matched
|
|
* if a value (recursive) object key matches a key of the mapping. However values will
|
|
* not be matched inside the children of a matching key.
|
|
*
|
|
* Example:
|
|
* mappingToStringifiedJSONLines(
|
|
* { key_2: 'Value' },
|
|
* {
|
|
* key_1: 'A random value',
|
|
* key_2: 'Another value'
|
|
* }
|
|
* ) === {
|
|
* 3: 'Value'
|
|
* }
|
|
*/
|
|
export function mappingToStringifiedJSONLines(
|
|
mapping,
|
|
value,
|
|
index = 1,
|
|
lines = {},
|
|
first = true
|
|
) {
|
|
if (Array.isArray(value)) {
|
|
index += 1
|
|
value.forEach((v, i) => {
|
|
index = mappingToStringifiedJSONLines(mapping, v, index, lines, false)
|
|
})
|
|
index += 1
|
|
return first ? lines : index
|
|
} else if (value instanceof Object) {
|
|
index += 1
|
|
Object.keys(value).forEach((k) => {
|
|
let childMapping = mapping
|
|
if (Object.prototype.hasOwnProperty.call(mapping, k)) {
|
|
lines[index] = mapping[k]
|
|
// Only recursively search for more field to line mappings where the current key
|
|
// is not itself the key for a field.
|
|
// For example if this key is a field, then there cannot be any other fields
|
|
// to map within this fields value.
|
|
childMapping = {}
|
|
}
|
|
index = mappingToStringifiedJSONLines(
|
|
childMapping,
|
|
value[k],
|
|
index,
|
|
lines,
|
|
false
|
|
)
|
|
})
|
|
index += 1
|
|
return first ? lines : index
|
|
} else {
|
|
index += 1
|
|
return first ? lines : index
|
|
}
|
|
}
|
|
|
|
export function isPromise(p) {
|
|
return (
|
|
p !== null &&
|
|
typeof p === 'object' &&
|
|
typeof p.then === 'function' &&
|
|
typeof p.catch === 'function'
|
|
)
|
|
}
|
|
|
|
/**
|
|
* Get the value at `path` of `obj`, similar to Lodash `get` function.
|
|
*
|
|
* @param {Object} obj The object that holds the value
|
|
* @param {string | Array[string]} path The path to the value or a list with the path parts
|
|
* @param {any} defaultValue The value to return if the path is not found
|
|
* @return {Object} The value held by the path
|
|
*/
|
|
export function getValueAtPath(obj, path) {
|
|
function _getValueAtPath(obj, keys) {
|
|
const [first, ...rest] = keys
|
|
if (!first) {
|
|
return obj
|
|
}
|
|
if (first in obj) {
|
|
return _getValueAtPath(obj[first], rest)
|
|
}
|
|
if (Array.isArray(obj) && first === '*') {
|
|
const results = obj
|
|
// Call recursively this function transforming the `*` in the path in a list
|
|
// of indexes present in the object, e.g:
|
|
// get(obj, "a.*.b") <=> [get(obj, "a.0.b"), get(obj, "a.1.b"), ...]
|
|
.map((_, index) => _getValueAtPath(obj, [index.toString(), ...rest]))
|
|
// Remove empty results
|
|
// Note: Don't exclude false values such as booleans, empty strings, etc.
|
|
.filter((result) => result !== null && result !== undefined)
|
|
// Return null in case there are no results
|
|
return results.length ? results : null
|
|
}
|
|
return null
|
|
}
|
|
const keys = typeof path === 'string' ? _.toPath(path) : path
|
|
return _getValueAtPath(obj, keys)
|
|
}
|
|
|
|
/**
|
|
* Responsible for setting a value at a given path in `obj`.
|
|
*
|
|
* @param {Object} obj - The object we want to update.
|
|
* @param {String} path - The path, delimited by periods, to the value.
|
|
* @param {Any} value - The value to set at the path.
|
|
* @returns {Object} The object with the updated value.
|
|
*/
|
|
export function setValueAtPath(obj, path, value) {
|
|
return _.set(obj, path, value)
|
|
}
|
|
|
|
/**
|
|
* Uses Object.defineProperty to make Vue provide/inject reactive.
|
|
*
|
|
* @param staticProperties The original object
|
|
* @param reactiveProperties An object containing the properties and values to
|
|
* become reactive
|
|
* @return {object} The original object with the updated properties
|
|
* @see https://stackoverflow.com/questions/65718651/how-do-i-make-vue-2-provide-inject-api-reactive
|
|
*
|
|
* @example
|
|
* const obj = { a: "A", b: "B" }
|
|
* fixPropertyReactivityForProvide(obj, { c: () => "C" }
|
|
* console.log(obj.c) // "c" property is now reactive and will return "C"
|
|
*/
|
|
export function fixPropertyReactivityForProvide(
|
|
staticProperties,
|
|
reactiveProperties
|
|
) {
|
|
Object.entries(reactiveProperties).forEach(([propertyName, getValue]) => {
|
|
Object.defineProperty(staticProperties, propertyName, {
|
|
enumerable: true,
|
|
get: () => getValue(),
|
|
})
|
|
})
|
|
return staticProperties
|
|
}
|