import type { Ref, ShallowRef } from 'vue'; import { shallowRef, watch, toRaw } from 'vue'; import type { DataNode, SimpleModeConfig } from '../interface'; import { convertChildrenToData } from '../utils/legacyUtil'; import type { DefaultOptionType } from '../TreeSelect'; import type { VueNode } from '../../_util/type'; function parseSimpleTreeData( treeData: DataNode[], { id, pId, rootPId }: SimpleModeConfig, ): DataNode[] { const keyNodes = {}; const rootNodeList = []; // Fill in the map const nodeList = treeData.map(node => { const clone = { ...node }; const key = clone[id]; keyNodes[key] = clone; clone.key = clone.key || key; return clone; }); // Connect tree nodeList.forEach(node => { const parentKey = node[pId]; const parent = keyNodes[parentKey]; // Fill parent if (parent) { parent.children = parent.children || []; parent.children.push(node); } // Fill root tree node if (parentKey === rootPId || (!parent && rootPId === null)) { rootNodeList.push(node); } }); return rootNodeList; } /** * Convert `treeData` or `children` into formatted `treeData`. * Will not re-calculate if `treeData` or `children` not change. */ export default function useTreeData( treeData: Ref, children: Ref, simpleMode: Ref, ): ShallowRef { const mergedTreeData = shallowRef(); watch( [simpleMode, treeData, children], () => { const simpleModeValue = simpleMode.value; if (treeData.value) { mergedTreeData.value = simpleMode.value ? parseSimpleTreeData(toRaw(treeData.value), { id: 'id', pId: 'pId', rootPId: null, ...(simpleModeValue !== true ? simpleModeValue : {}), }) : toRaw(treeData.value).slice(); } else { mergedTreeData.value = convertChildrenToData(toRaw(children.value)); } }, { immediate: true, deep: true }, ); return mergedTreeData; }