import { isVNode, provide, reactive, getCurrentInstance, type ComponentPublicInstance, type ComponentInternalInstance } from 'vue' export function flattenVNodes(children: any) { const result: any[] = [] const traverse = (children: any) => { if (Array.isArray(children)) { children.forEach((child) => { // #ifdef H5 if (isVNode(child)) { result.push(child) if (child.component?.subTree) { result.push(child.component.subTree) traverse(child.component.subTree.children) } if (child.children) { traverse(child.children) } } // #endif // #ifndef H5 if (child) { result.push(child) if (child.component?.subTree) { result.push(child.component.subTree) traverse(child.component.subTree.children) } if (child.children) { traverse(child.children) } } // #endif }) } } traverse(children) return result } const findVNodeIndex = (vnodes: any[], vnode: any) => { const index = vnodes.indexOf(vnode) if (index === -1) { return vnodes.findIndex( (item) => vnode.key !== undefined && vnode.key !== null && item.type === vnode.type && item.key === vnode.key ) } return index } export function sortChildren( parent: any, publicChildren: ComponentPublicInstance[], internalChildren: ComponentInternalInstance[] ) { const vnodes = flattenVNodes(parent.ctx.children) internalChildren.sort( (a, b) => findVNodeIndex(vnodes, a.vnode) - findVNodeIndex(vnodes, b.vnode) ) const orderedPublicChildren = internalChildren.map((item) => item.proxy!) publicChildren.sort((a, b) => { const indexA = orderedPublicChildren.indexOf(a) const indexB = orderedPublicChildren.indexOf(b) return indexA - indexB }) } export function useChildren(key: any) { const publicChildren: any = reactive([]) const internalChildren: ComponentInternalInstance[] = reactive([]) const parent = getCurrentInstance()! const linkChildren = (value?: any) => { const link = (child: ComponentInternalInstance) => { if (child.proxy) { internalChildren.push(child) publicChildren.push(child.proxy as any) sortChildren(parent, publicChildren, internalChildren) } } const unlink = (child: ComponentInternalInstance) => { const index = internalChildren.indexOf(child) publicChildren.splice(index, 1) internalChildren.splice(index, 1) } provide( key, Object.assign( { link, unlink, children: publicChildren, internalChildren }, value ) ) } return { children: publicChildren, linkChildren } }