UNPKG

vuetify

Version:

Vue Material Component Framework

146 lines (143 loc) 4.81 kB
// Utilities import { computed, shallowRef, watchEffect } from 'vue'; import { deepEqual, getPropertyFromItem, isPrimitive, omit, pick, propsFactory } from "../util/index.js"; // Types // Composables export const makeItemsProps = propsFactory({ items: { type: Array, default: () => [] }, itemTitle: { type: [String, Array, Function], default: 'title' }, itemValue: { type: [String, Array, Function], default: 'value' }, itemChildren: { type: [Boolean, String, Array, Function], default: 'children' }, itemProps: { type: [Boolean, String, Array, Function], default: 'props' }, returnObject: Boolean, valueComparator: Function }, 'list-items'); export function transformItem(props, item) { const title = getPropertyFromItem(item, props.itemTitle, item); const value = getPropertyFromItem(item, props.itemValue, title); const children = getPropertyFromItem(item, props.itemChildren); const itemProps = props.itemProps === true ? typeof item === 'object' && item != null && !Array.isArray(item) ? 'children' in item ? omit(item, ['children']) : item : undefined : getPropertyFromItem(item, props.itemProps); const _props = { title, value, ...itemProps }; return { title: String(_props.title ?? ''), value: _props.value, props: _props, children: Array.isArray(children) ? transformItems(props, children) : undefined, raw: item }; } export function transformItems(props, items) { const _props = pick(props, ['itemTitle', 'itemValue', 'itemChildren', 'itemProps', 'returnObject', 'valueComparator']); const array = []; for (const item of items) { array.push(transformItem(_props, item)); } return array; } export function useItems(props) { const items = computed(() => transformItems(props, props.items)); const hasNullItem = computed(() => items.value.some(item => item.value === null)); const itemsMap = shallowRef(new Map()); const keylessItems = shallowRef([]); watchEffect(() => { const _items = items.value; const map = new Map(); const keyless = []; for (let i = 0; i < _items.length; i++) { const item = _items[i]; if (isPrimitive(item.value) || item.value === null) { let values = map.get(item.value); if (!values) { values = []; map.set(item.value, values); } values.push(item); } else { keyless.push(item); } } itemsMap.value = map; keylessItems.value = keyless; }); function transformIn(value) { // Cache unrefed values outside the loop, // proxy getters can be slow when you call them a billion times const _items = itemsMap.value; const _allItems = items.value; const _keylessItems = keylessItems.value; const _hasNullItem = hasNullItem.value; const _returnObject = props.returnObject; const hasValueComparator = !!props.valueComparator; const valueComparator = props.valueComparator || deepEqual; const _props = pick(props, ['itemTitle', 'itemValue', 'itemChildren', 'itemProps', 'returnObject', 'valueComparator']); const returnValue = []; main: for (const v of value) { // When the model value is null, return an InternalItem // based on null only if null is one of the items if (!_hasNullItem && v === null) continue; // String model value means value is a custom input value from combobox // Don't look up existing items if the model value is a string if (_returnObject && typeof v === 'string') { returnValue.push(transformItem(_props, v)); continue; } // Fast path, items with primitive values and no // custom valueComparator can use a constant-time // map lookup instead of searching the items array const fastItems = _items.get(v); // Slow path, always use valueComparator. // This is O(n^2) so we really don't want to // do it for more than a couple hundred items. if (hasValueComparator || !fastItems) { for (const item of hasValueComparator ? _allItems : _keylessItems) { if (valueComparator(v, item.value)) { returnValue.push(item); continue main; } } // Not an existing item, construct it from the model (#4000) returnValue.push(transformItem(_props, v)); continue; } returnValue.push(...fastItems); } return returnValue; } function transformOut(value) { return props.returnObject ? value.map(_ref => { let { raw } = _ref; return raw; }) : value.map(_ref2 => { let { value } = _ref2; return value; }); } return { items, transformIn, transformOut }; } //# sourceMappingURL=list-items.js.map