import { getCurrentInstance } from 'vue' import { camelize, capitalize, extend, hasOwn, hyphenate, isArray, isObject, isString, isFunction, looseEqual, toRawType, } from '@vue/shared' import isEqualWith from 'lodash/isEqualWith' import { isClient } from '@vueuse/core' import { debugWarn, throwError } from './error' import type { ComponentPublicInstance, CSSProperties, Ref } from 'vue' import type { TimeoutHandle, Nullable, ComponentSize } from './types' export const SCOPE = 'Util' export function toObject<T>(arr: Array<T>): Record<string, T> { const res = {} for (let i = 0; i < arr.length; i++) { if (arr[i]) { extend(res, arr[i]) } } return res } export const getValueByPath = (obj, paths = ''): unknown => { let ret: unknown = obj paths.split('.').map((path) => { ret = ret?.[path] }) return ret } export function getPropByPath( obj: any, path: string, strict: boolean ): { o: unknown k: string v: Nullable<unknown> } { let tempObj = obj let key, value if (obj && hasOwn(obj, path)) { key = path value = tempObj?.[path] } else { path = path.replace(/\[(\w+)\]/g, '.$1') path = path.replace(/^\./, '') const keyArr = path.split('.') let i = 0 for (i; i < keyArr.length - 1; i++) { if (!tempObj && !strict) break const key = keyArr[i] if (key in tempObj) { tempObj = tempObj[key] } else { if (strict) { throwError(SCOPE, 'Please transfer a valid prop path to form item!') } break } } key = keyArr[i] value = tempObj?.[keyArr[i]] } return { o: tempObj, k: key, v: value, } } /** * Generate random number in range [0, 1000] * Maybe replace with [uuid](https://www.npmjs.com/package/uuid) */ export const generateId = (): number => Math.floor(Math.random() * 10000) // use isEqual instead // export const valueEquals export const escapeRegexpString = (value = ''): string => String(value).replace(/[|\\{}()[\]^$+*?.]/g, '\\$&') // Use native Array.find, Array.findIndex instead // coerce truthy value to array export const coerceTruthyValueToArray = (arr) => { if (!arr && arr !== 0) { return [] } return Array.isArray(arr) ? arr : [arr] } // drop IE and (Edge < 79) support // export const isIE // export const isEdge export const isFirefox = function (): boolean { return isClient && !!window.navigator.userAgent.match(/firefox/i) } export const autoprefixer = function (style: CSSProperties): CSSProperties { const rules = ['transform', 'transition', 'animation'] const prefixes = ['ms-', 'webkit-'] rules.forEach((rule) => { const value = style[rule] if (rule && value) { prefixes.forEach((prefix) => { style[prefix + rule] = value }) } }) return style } export const kebabCase = hyphenate // reexport from lodash & vue shared export { isVNode } from 'vue' export { hasOwn, // isEmpty, // isEqual, isObject, isArray, isString, capitalize, camelize, looseEqual, extend, } export const isBool = (val: unknown): val is boolean => typeof val === 'boolean' export const isNumber = (val: unknown): val is number => typeof val === 'number' export const isHTMLElement = (val: unknown) => toRawType(val).startsWith('HTML') export function rafThrottle<T extends (...args: any) => any>(fn: T): T { let locked = false return function (this: ThisParameterType<T>, ...args: any[]) { if (locked) return locked = true window.requestAnimationFrame(() => { Reflect.apply(fn, this, args) locked = false }) } as T } export const clearTimer = (timer: Ref<TimeoutHandle>) => { clearTimeout(timer.value) timer.value = null } /** * Generating a random int in range (0, max - 1) * @param max {number} */ export function getRandomInt(max: number) { return Math.floor(Math.random() * Math.floor(max)) } export function isUndefined(val: any): val is undefined { return val === undefined } /** * @deprecated please use `useGlobalConfig` in hooks. */ export function useGlobalConfig(): { size?: ComponentSize; zIndex?: number } { const vm: any = getCurrentInstance() if ('$ELEMENT' in vm.proxy) { return vm.proxy.$ELEMENT } return {} } export function isEmpty(val: unknown) { if ( (!val && val !== 0) || (isArray(val) && !val.length) || (isObject(val) && !Object.keys(val).length) ) return true return false } export function arrayFlat(arr: unknown[]) { return arr.reduce((acm: unknown[], item) => { const val = Array.isArray(item) ? arrayFlat(item) : item return acm.concat(val) }, []) } export function deduplicate<T>(arr: T[]) { return Array.from(new Set(arr)) } export function addUnit(value: string | number) { if (isString(value)) { return value } else if (isNumber(value)) { return `${value}px` } debugWarn(SCOPE, 'binding value must be a string or number') return '' } /** * Enhance `lodash.isEqual` for it always return false even two functions have completely same statements. * @param obj The value to compare * @param other The other value to compare * @returns Returns `true` if the values are equivalent, else `false`. * @example * lodash.isEqual(() => 1, () => 1) // false * isEqualWith(() => 1, () => 1) // true */ export function isEqualWithFunction(obj: any, other: any) { return isEqualWith(obj, other, (objVal, otherVal) => { return isFunction(objVal) && isFunction(otherVal) ? `${objVal}` === `${otherVal}` : undefined }) } /** * Generate function for attach ref for the h renderer * @param ref Ref<HTMLElement | ComponentPublicInstance> * @returns (val: T) => void */ export const refAttacher = <T extends HTMLElement | ComponentPublicInstance>( ref: Ref<T> ) => { return (val: T) => { ref.value = val } }