181 lines
		
	
	
		
			5.0 KiB
		
	
	
	
		
			Vue
		
	
	
			
		
		
	
	
			181 lines
		
	
	
		
			5.0 KiB
		
	
	
	
		
			Vue
		
	
	
| import { camelize, filterEmpty } from '../../_util/props-util';
 | |
| import { warning } from '../../vc-util/warning';
 | |
| import type { DataNode, ChangeEventExtra, RawValueType, LegacyCheckedNode } from '../interface';
 | |
| import TreeNode from '../TreeNode';
 | |
| import type { VueNode } from '../../_util/type';
 | |
| import type { DefaultOptionType, FieldNames } from '../TreeSelect';
 | |
| 
 | |
| function isTreeSelectNode(node: any) {
 | |
|   return node && node.type && (node.type as any).isTreeSelectNode;
 | |
| }
 | |
| export function convertChildrenToData(rootNodes: VueNode[]): DataNode[] {
 | |
|   function dig(treeNodes: any[] = []): DataNode[] {
 | |
|     return filterEmpty(treeNodes).map(treeNode => {
 | |
|       // Filter invalidate node
 | |
|       if (!isTreeSelectNode(treeNode)) {
 | |
|         warning(!treeNode, 'TreeSelect/TreeSelectNode can only accept TreeSelectNode as children.');
 | |
|         return null;
 | |
|       }
 | |
|       const slots = (treeNode.children as any) || {};
 | |
|       const key = treeNode.key as string | number;
 | |
|       const props: any = {};
 | |
|       for (const [k, v] of Object.entries(treeNode.props)) {
 | |
|         props[camelize(k)] = v;
 | |
|       }
 | |
|       const { isLeaf, checkable, selectable, disabled, disableCheckbox } = props;
 | |
|       // 默认值为 undefined
 | |
|       const newProps = {
 | |
|         isLeaf: isLeaf || isLeaf === '' || undefined,
 | |
|         checkable: checkable || checkable === '' || undefined,
 | |
|         selectable: selectable || selectable === '' || undefined,
 | |
|         disabled: disabled || disabled === '' || undefined,
 | |
|         disableCheckbox: disableCheckbox || disableCheckbox === '' || undefined,
 | |
|       };
 | |
|       const slotsProps = { ...props, ...newProps };
 | |
|       const {
 | |
|         title = slots.title?.(slotsProps),
 | |
|         switcherIcon = slots.switcherIcon?.(slotsProps),
 | |
|         ...rest
 | |
|       } = props;
 | |
|       const children = slots.default?.();
 | |
|       const dataNode: DataNode = {
 | |
|         ...rest,
 | |
|         title,
 | |
|         switcherIcon,
 | |
|         key,
 | |
|         isLeaf,
 | |
|         ...newProps,
 | |
|       };
 | |
| 
 | |
|       const parsedChildren = dig(children);
 | |
|       if (parsedChildren.length) {
 | |
|         dataNode.children = parsedChildren;
 | |
|       }
 | |
| 
 | |
|       return dataNode;
 | |
|     });
 | |
|   }
 | |
| 
 | |
|   return dig(rootNodes as any[]);
 | |
| }
 | |
| 
 | |
| export function fillLegacyProps(dataNode: DataNode): any {
 | |
|   // Skip if not dataNode exist
 | |
|   if (!dataNode) {
 | |
|     return dataNode;
 | |
|   }
 | |
| 
 | |
|   const cloneNode = { ...dataNode };
 | |
| 
 | |
|   if (!('props' in cloneNode)) {
 | |
|     Object.defineProperty(cloneNode, 'props', {
 | |
|       get() {
 | |
|         warning(
 | |
|           false,
 | |
|           'New `vc-tree-select` not support return node instance as argument anymore. Please consider to remove `props` access.',
 | |
|         );
 | |
|         return cloneNode;
 | |
|       },
 | |
|     });
 | |
|   }
 | |
| 
 | |
|   return cloneNode;
 | |
| }
 | |
| 
 | |
| export function fillAdditionalInfo(
 | |
|   extra: ChangeEventExtra,
 | |
|   triggerValue: RawValueType,
 | |
|   checkedValues: RawValueType[],
 | |
|   treeData: DefaultOptionType[],
 | |
|   showPosition: boolean,
 | |
|   fieldNames: FieldNames,
 | |
| ) {
 | |
|   let triggerNode = null;
 | |
|   let nodeList: LegacyCheckedNode[] = null;
 | |
| 
 | |
|   function generateMap() {
 | |
|     function dig(list: DefaultOptionType[], level = '0', parentIncluded = false) {
 | |
|       return list
 | |
|         .map((option, index) => {
 | |
|           const pos = `${level}-${index}`;
 | |
|           const value = option[fieldNames.value];
 | |
|           const included = checkedValues.includes(value);
 | |
|           const children = dig(option[fieldNames.children] || [], pos, included);
 | |
|           const node = (
 | |
|             <TreeNode {...(option as Required<DefaultOptionType>)}>
 | |
|               {children.map(child => child.node)}
 | |
|             </TreeNode>
 | |
|           );
 | |
| 
 | |
|           // Link with trigger node
 | |
|           if (triggerValue === value) {
 | |
|             triggerNode = node;
 | |
|           }
 | |
| 
 | |
|           if (included) {
 | |
|             const checkedNode: LegacyCheckedNode = {
 | |
|               pos,
 | |
|               node,
 | |
|               children,
 | |
|             };
 | |
| 
 | |
|             if (!parentIncluded) {
 | |
|               nodeList.push(checkedNode);
 | |
|             }
 | |
| 
 | |
|             return checkedNode;
 | |
|           }
 | |
|           return null;
 | |
|         })
 | |
|         .filter(node => node);
 | |
|     }
 | |
| 
 | |
|     if (!nodeList) {
 | |
|       nodeList = [];
 | |
| 
 | |
|       dig(treeData);
 | |
| 
 | |
|       // Sort to keep the checked node length
 | |
|       nodeList.sort(
 | |
|         (
 | |
|           {
 | |
|             node: {
 | |
|               props: { value: val1 },
 | |
|             },
 | |
|           },
 | |
|           {
 | |
|             node: {
 | |
|               props: { value: val2 },
 | |
|             },
 | |
|           },
 | |
|         ) => {
 | |
|           const index1 = checkedValues.indexOf(val1);
 | |
|           const index2 = checkedValues.indexOf(val2);
 | |
|           return index1 - index2;
 | |
|         },
 | |
|       );
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   Object.defineProperty(extra, 'triggerNode', {
 | |
|     get() {
 | |
|       warning(false, '`triggerNode` is deprecated. Please consider decoupling data with node.');
 | |
|       generateMap();
 | |
| 
 | |
|       return triggerNode;
 | |
|     },
 | |
|   });
 | |
|   Object.defineProperty(extra, 'allCheckedNodes', {
 | |
|     get() {
 | |
|       warning(false, '`allCheckedNodes` is deprecated. Please consider decoupling data with node.');
 | |
|       generateMap();
 | |
| 
 | |
|       if (showPosition) {
 | |
|         return nodeList;
 | |
|       }
 | |
| 
 | |
|       return nodeList.map(({ node }) => node);
 | |
|     },
 | |
|   });
 | |
| }
 |