import type { VNodeProps, ComputedRef, Ref } from 'vue'; import { shallowRef, watch, ref } from 'vue'; import type { GetKey } from '../interface'; type CacheMap = Ref>; export default function useHeights( mergedData: ComputedRef, getKey: GetKey, onItemAdd?: ((item: T) => void) | null, onItemRemove?: ((item: T) => void) | null, ): [(item: T, instance: HTMLElement) => void, () => void, CacheMap, Ref] { const instance = new Map(); const heights = shallowRef({}); const updatedMark = ref(Symbol('update')); watch(mergedData, () => { heights.value = {}; updatedMark.value = Symbol('update'); }); let heightUpdateId = 0; function collectHeight() { heightUpdateId += 1; const currentId = heightUpdateId; Promise.resolve().then(() => { // Only collect when it's latest call if (currentId !== heightUpdateId) return; // let changed = false; instance.forEach((element, key) => { if (element && element.offsetParent) { const { offsetHeight } = element; if (heights.value[key!] !== offsetHeight) { //changed = true; updatedMark.value = Symbol('update'); heights.value[key!] = element.offsetHeight; } } }); }); } function setInstance(item: T, ins: HTMLElement) { const key = getKey(item); const origin = instance.get(key); if (ins) { instance.set(key, (ins as any).$el || ins); collectHeight(); } else { instance.delete(key); } // Instance changed if (!origin !== !ins) { if (ins) { onItemAdd?.(item); } else { onItemRemove?.(item); } } } return [setInstance, collectHeight, heights, updatedMark]; }