You cannot select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
	
	
		
			287 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			Vue
		
	
		
		
			
		
	
	
			287 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			Vue
		
	
|   
											1 year ago
										 | <template> | ||
|  |     <div class="left-tree-container"> | ||
|  |         <j-input placeholder="名称" v-model:value="searchValue" @pressEnter="search" @change="searchChange"> | ||
|  |             <template #suffix> | ||
|  |                 <AIcon type="SearchOutlined" @click="search" /> | ||
|  |             </template> | ||
|  |         </j-input> | ||
|  |         <div v-if="admin" class="add-btn"> | ||
|  |             <j-button type="primary" @click="addGroup"> | ||
|  |                 新增站点 | ||
|  |             </j-button> | ||
|  |         </div> | ||
|  |         <div class="tree"> | ||
|  |             <j-spin :spinning='loading'> | ||
|  |                 <j-tree :tree-data="listData" v-if="listData.length" draggable @dragstart="dragstart" | ||
|  |                     :fieldNames="{ title: 'name', key: 'id', children: 'children' }" blockNode | ||
|  |                     :selectedKeys="selectedKeys" :default-expanded-keys="defaultExpandedKeys" | ||
|  |                     :showLine="{ showLeafIcon: false }" :showIcon="true"> | ||
|  |                     <template #title="{ name, data, level, type, positionX }"> | ||
|  |                         <div v-if="selectId === data.id"> | ||
|  |                             <j-input v-model:value="addName" @blur="() => saveGroup(data)" ref="inputRef" | ||
|  |                                 :maxlength="64"></j-input> | ||
|  |                         </div> | ||
|  |                         <span v-else class='department-tree-item-content'> | ||
|  |                             <span class='title' :style="{ | ||
|  |                                 width: type === 'root' ? '60px' : (level === 2 ? '80px' : '60px'), | ||
|  |                             }"> | ||
|  |                                 <j-ellipsis> | ||
|  |                                     <AIcon v-if="level === 2 && positionX" type="HeartFilled" /> | ||
|  |                                      {{ name }} | ||
|  |                                 </j-ellipsis> | ||
|  |                             </span> | ||
|  |                             <span class="func-btns" :style="{ | ||
|  |                                 width: type === 'root' ? '145px' : (level === 2 ? '80px' : '100px'), | ||
|  |                             }" @click="(e) => e.stopPropagation()"> | ||
|  |                                 <PermissionButton hasPermission="system/Role:groupUpdate" type="link" | ||
|  |                                     :tooltip="{ title: '编辑' }" @click="editGroup(data)"> | ||
|  |                                     <AIcon type="EditOutlined" /> | ||
|  |                                 </PermissionButton> | ||
|  |                                 <PermissionButton v-if="level == 1" hasPermission="system/Role:groupUpdate" type="link" | ||
|  |                                     :tooltip="{ title: '新增子站' }"> | ||
|  |                                     <AIcon type="PlusOutlined" /> | ||
|  |                                 </PermissionButton> | ||
|  |                                 <PermissionButton v-if="level == 1" hasPermission="system/Role:groupUpdate" type="link" | ||
|  |                                     :tooltip="{ title: '新增子监测点' }"> | ||
|  |                                     <AIcon type="PlusCircleOutlined" /> | ||
|  |                                 </PermissionButton> | ||
|  |                                 <PermissionButton hasPermission="system/Role:groupUpdate" type="link" | ||
|  |                                     :tooltip="{ title: '删除' }" :popConfirm="{ | ||
|  |                                         title: `确定要删除吗`, | ||
|  |                                         onConfirm: () => { }, | ||
|  |                                     }"> | ||
|  |                                     <AIcon type="DeleteOutlined" /> | ||
|  |                                 </PermissionButton> | ||
|  |                                 <PermissionButton v-if="level == 2" hasPermission="system/Role:groupUpdate" type="link" | ||
|  |                                     :tooltip="{ title: '移除' }"> | ||
|  |                                     <AIcon type="CloseOutlined" /> | ||
|  |                                 </PermissionButton> | ||
|  |                                 <PermissionButton v-if="type === 'root'" hasPermission="system/Role:groupUpdate" | ||
|  |                                     type="link" :tooltip="{ title: '导入excel' }"> | ||
|  |                                     <AIcon type="UploadOutlined" /> | ||
|  |                                 </PermissionButton> | ||
|  |                                 <PermissionButton v-if="type === 'root'" hasPermission="system/Role:groupUpdate" | ||
|  |                                     type="link" :tooltip="{ title: '导出excel' }"> | ||
|  |                                     <AIcon type="DownloadOutlined" /> | ||
|  |                                 </PermissionButton> | ||
|  |                             </span> | ||
|  |                         </span> | ||
|  |                     </template> | ||
|  |                 </j-tree> | ||
|  |                 <j-empty v-else description="暂无数据" /> | ||
|  |             </j-spin> | ||
|  |         </div> | ||
|  |     </div> | ||
|  | </template> | ||
|  | 
 | ||
|  | <script lang="ts" setup> | ||
|  | import { onlyMessage } from '@/utils/comm'; | ||
|  | import { useUserInfo } from '@/store/userInfo'; | ||
|  | import { storeToRefs } from 'pinia'; | ||
|  | import { filterTreeItem } from '@/utils/comm' | ||
|  | const loading = ref<boolean>(false); // 数据加载状态
 | ||
|  | const emit = defineEmits(['selectData']); | ||
|  | const props = defineProps({ | ||
|  |     listData: { | ||
|  |         type: Array, | ||
|  |         default: [] | ||
|  |     } | ||
|  | }) | ||
|  | const userInfoStore = useUserInfo(); | ||
|  | const { userInfos } = storeToRefs(userInfoStore); | ||
|  | const admin = computed(() => { | ||
|  |     return userInfos.value?.username === 'admin'; | ||
|  | }); | ||
|  | 
 | ||
|  | const selectedKeys = ref<string[]>(['global_role']); | ||
|  | const searchValue = ref(); | ||
|  | const inputRef = ref(); | ||
|  | const addName = ref(); | ||
|  | const selectId = ref(); | ||
|  | const mapMonitorList = ref(); | ||
|  | const defaultExpandedKeys = ref<(number | string)[]>([]);  | ||
|  | 
 | ||
|  | const dragstart = ({ event, node }: { event: DragEvent, node: any }) => { | ||
|  |    // console.log("🚀  node:", node)
 | ||
|  |     //event.preventDefault()
 | ||
|  |     event.dataTransfer?.setData('monitorData', JSON.stringify(node.dataRef)); // 使用 text/plain 类型存储节点信息
 | ||
|  | } | ||
|  | 
 | ||
|  | //获取全部id
 | ||
|  | const getTreeId = <T>(treeNode: T[]) => { | ||
|  |     let result: any[] = []; | ||
|  |     const traverse = (node: any) => { | ||
|  |         result?.push(node.id) | ||
|  |         if (node.children && Array.isArray(node.children)) { | ||
|  |             node.children.forEach((child: any) => traverse(child)) | ||
|  |         } | ||
|  |     } | ||
|  |     treeNode?.forEach((child: any) => traverse(child)) | ||
|  |     return result | ||
|  | } | ||
|  | defaultExpandedKeys.value = getTreeId(props.listData) | ||
|  | const queryGroup = async (select?: Boolean, searchName?: string) => { | ||
|  |     // const params = searchName
 | ||
|  |     //     ? {
 | ||
|  |     //         sorts: [{ name: 'createTime', order: 'desc' }],
 | ||
|  |     //         terms: [
 | ||
|  |     //             {
 | ||
|  |     //                 terms: [
 | ||
|  |     //                     {
 | ||
|  |     //                         value: '%' + searchName + '%',
 | ||
|  |     //                         termType: 'like',
 | ||
|  |     //                         column: 'name',
 | ||
|  |     //                     },
 | ||
|  |     //                 ],
 | ||
|  |     //             },
 | ||
|  |     //         ],
 | ||
|  |     //     }
 | ||
|  |     //     : { sorts: [{ name: 'createTime', order: 'desc' }] };
 | ||
|  |     // const req: any = await queryRoleGroup(params);
 | ||
|  |     // if (req.status === 200) {
 | ||
|  |     //     listData.value[0].children = req.result;
 | ||
|  |     //     if (req.result[req.result.length - 1].id === 'default_group') {
 | ||
|  |     //         req.result.unshift(req.result[req.result.length - 1]);
 | ||
|  |     //         req.result.pop();
 | ||
|  |     //     }
 | ||
|  |     //     // if(req.result.length && select){
 | ||
|  |     //     //     selectGroup(req.result[0].id)
 | ||
|  |     //     // }
 | ||
|  |     // }
 | ||
|  | }; | ||
|  | const addGroup = () => { | ||
|  |     // addName.value = '';
 | ||
|  |     // const newId = randomString();
 | ||
|  |     // listData.value[0].children?.splice(1, 0, {
 | ||
|  |     //     name: '',
 | ||
|  |     //     id: newId,
 | ||
|  |     // });
 | ||
|  |     // selectId.value = newId;
 | ||
|  |     // nextTick(() => {
 | ||
|  |     //     inputRef.value.focus();
 | ||
|  |     // });
 | ||
|  | }; | ||
|  | const saveGroup = async (data: any) => { | ||
|  |     if (addName.value === '') { | ||
|  |         if (data.name === '') { | ||
|  |            // listData.value[0].children.splice(1, 1);
 | ||
|  |         } | ||
|  |     } | ||
|  |     else { | ||
|  |         const saveData = { | ||
|  |             name: addName.value, | ||
|  |             id: data.id, | ||
|  |         }; | ||
|  |         const res = { status: 200 }//await saveRoleGroup(saveData);
 | ||
|  |         if (res.status === 200) { | ||
|  |             onlyMessage('操作成功!'); | ||
|  |             //  queryGroup();
 | ||
|  |         } else { | ||
|  |             onlyMessage('操作失败!'); | ||
|  |         } | ||
|  |     } | ||
|  |     setTimeout(() => { | ||
|  |         selectId.value = ''; | ||
|  |     }, 300); | ||
|  | }; | ||
|  | 
 | ||
|  | const search = () => { | ||
|  |     queryGroup(true, searchValue.value); | ||
|  | }; | ||
|  | const searchChange = () => { | ||
|  |     if (searchValue.value === '') { | ||
|  |         queryGroup(); | ||
|  |     } | ||
|  | }; | ||
|  | const editGroup = (data: any) => { | ||
|  |     if (!selectId.value) { | ||
|  |         selectId.value = data.id; | ||
|  |         addName.value = data.name; | ||
|  |         // listData.value[0].children?.forEach((item: any) => {
 | ||
|  |         //     if (item.id === data.id) {
 | ||
|  |         //         item.edit = true;
 | ||
|  |         //     }
 | ||
|  |         // });
 | ||
|  |         nextTick(() => { | ||
|  |             inputRef.value.focus(); | ||
|  |         }); | ||
|  |     } | ||
|  | }; | ||
|  | 
 | ||
|  | onMounted(() => { | ||
|  |     mapMonitorList.value = filterTreeItem(props.listData, 'children', (i) => i.level == 2 && i.positionX) | ||
|  | }); | ||
|  | </script> | ||
|  | <style lang="less" scoped> | ||
|  | .left-tree-container { | ||
|  |     display: flex; | ||
|  |     height: 100%; | ||
|  |     flex-direction: column; | ||
|  | 
 | ||
|  |     .add-btn { | ||
|  |         margin: 24px 0; | ||
|  | 
 | ||
|  |         :deep(.ant-btn-primary) { | ||
|  |             width: 100%; | ||
|  |         } | ||
|  |     } | ||
|  |     :deep(.ant-tree-treenode-disabled){ | ||
|  |         .ant-tree-node-content-wrapper{ | ||
|  |             color: rgba(0, 0, 0, 0.55) | ||
|  |         } | ||
|  |     } | ||
|  | 
 | ||
|  |     :deep(.ant-tree-treenode) { | ||
|  |         width: 100%; | ||
|  | 
 | ||
|  |         .ant-tree-node-content-wrapper { | ||
|  |             flex: 1 1 auto; | ||
|  | 
 | ||
|  |             .ant-tree-title { | ||
|  |                 &:hover { | ||
|  |                     .func-btns { | ||
|  |                         display: block; | ||
|  |                     } | ||
|  |                 } | ||
|  |             } | ||
|  |         } | ||
|  |     } | ||
|  | 
 | ||
|  | 
 | ||
|  |     .tree { | ||
|  |         overflow-y: auto; | ||
|  |         overflow-x: auto; | ||
|  |         flex: 1 1 auto; | ||
|  | 
 | ||
|  |         .department-tree-item-content { | ||
|  |             display: inline-flex; | ||
|  |             align-items: stretch; | ||
|  | 
 | ||
|  |             .title { | ||
|  |                 flex: 1; | ||
|  |                 margin-right: 80px; | ||
|  |             } | ||
|  | 
 | ||
|  |             .func-btns { | ||
|  |                 display: none; | ||
|  |                 font-size: 14px; | ||
|  |                 width: 100px; | ||
|  |                 margin-left: -80px; | ||
|  | 
 | ||
|  |                 :deep(.ant-btn-link) { | ||
|  |                     padding: 0 4px; | ||
|  |                     height: 24px; | ||
|  |                 } | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         .loading { | ||
|  |             display: flex; | ||
|  |             width: 100%; | ||
|  |             justify-content: center; | ||
|  |             margin-top: 20px; | ||
|  |         } | ||
|  |     } | ||
|  | } | ||
|  | </style> |