|  |  |  | 
|---|
|  |  |  | <a-card | 
|---|
|  |  |  | :title="$t('menu.resource.title')" | 
|---|
|  |  |  | :bordered="false" | 
|---|
|  |  |  | :style="{ 'width': '100%', 'height': '700px', 'overflow-y': 'auto' }" | 
|---|
|  |  |  | style=" 'width': '100%';   height: calc(100vh - 250px); 'overflow-y': 'auto' " | 
|---|
|  |  |  | > | 
|---|
|  |  |  | <a-button @click="() => onIconClick(null)">新增父级菜单</a-button> | 
|---|
|  |  |  | <a-tree | 
|---|
|  |  |  | class="tree-demo" | 
|---|
|  |  |  | draggable | 
|---|
|  |  |  | 
|---|
|  |  |  | <a-form-item field="menuName" label="资源名称"> | 
|---|
|  |  |  | <a-input v-model="resourceform.menuName" /> | 
|---|
|  |  |  | </a-form-item> | 
|---|
|  |  |  | <a-form-item field="menuName" label="资源图标"> | 
|---|
|  |  |  | <Upload | 
|---|
|  |  |  | :action="uploadAction" | 
|---|
|  |  |  | :limit="1" | 
|---|
|  |  |  | :url="resourceform.icon" | 
|---|
|  |  |  | @update:fileList="updateFileList" | 
|---|
|  |  |  | @success="handleSuccess" | 
|---|
|  |  |  | ></Upload> | 
|---|
|  |  |  | </a-form-item> | 
|---|
|  |  |  | <a-form-item field="menuType" label="资源类型"> | 
|---|
|  |  |  | <a-select | 
|---|
|  |  |  | v-model="resourceform.menuType" | 
|---|
|  |  |  | 
|---|
|  |  |  | :field-names="fieldNames" | 
|---|
|  |  |  | :style="{ width: '320px' }" | 
|---|
|  |  |  | placeholder="请选择" | 
|---|
|  |  |  | /> | 
|---|
|  |  |  | </a-form-item> | 
|---|
|  |  |  | <a-form-item field="description" label="提示词"> | 
|---|
|  |  |  | <a-input | 
|---|
|  |  |  | v-model="resourceform.description" | 
|---|
|  |  |  | placeholder="请输入提示词" | 
|---|
|  |  |  | /> | 
|---|
|  |  |  | </a-form-item> | 
|---|
|  |  |  | <a-form-item field="perms" label="资源控制权限字符"> | 
|---|
|  |  |  | 
|---|
|  |  |  | @cancel="handleCancel" | 
|---|
|  |  |  | @ok="addresource" | 
|---|
|  |  |  | > | 
|---|
|  |  |  | <a-form :model="resourceform" layout="horizontal"> | 
|---|
|  |  |  | <a-form :rules="rules" :model="resourceform" layout="horizontal"> | 
|---|
|  |  |  | <a-form-item field="parentName" label="上级资源" disabled> | 
|---|
|  |  |  | <a-input v-model="resourceform.parentName" /> | 
|---|
|  |  |  | </a-form-item> | 
|---|
|  |  |  | 
|---|
|  |  |  | <a-form-item field="menuName" label="资源名称"> | 
|---|
|  |  |  | <a-input v-model="resourceform.menuName" /> | 
|---|
|  |  |  | </a-form-item> | 
|---|
|  |  |  | <a-form-item field="icon" label="资源图标"> | 
|---|
|  |  |  | <Upload | 
|---|
|  |  |  | :action="uploadAction" | 
|---|
|  |  |  | :limit="1" | 
|---|
|  |  |  | :url="resourceform.icon" | 
|---|
|  |  |  | @update:fileList="updateFileList" | 
|---|
|  |  |  | @success="handleSuccess" | 
|---|
|  |  |  | ></Upload> | 
|---|
|  |  |  | </a-form-item> | 
|---|
|  |  |  | <a-form-item field="menuType" label="资源类型"> | 
|---|
|  |  |  | <a-select | 
|---|
|  |  |  | v-model="resourceform.menuType" | 
|---|
|  |  |  | 
|---|
|  |  |  | :field-names="fieldNames" | 
|---|
|  |  |  | :style="{ width: '320px' }" | 
|---|
|  |  |  | placeholder="请选择" | 
|---|
|  |  |  | /> | 
|---|
|  |  |  | </a-form-item> | 
|---|
|  |  |  | <a-form-item field="description" label="提示词"> | 
|---|
|  |  |  | <a-input | 
|---|
|  |  |  | v-model="resourceform.description" | 
|---|
|  |  |  | placeholder="请输入提示词" | 
|---|
|  |  |  | /> | 
|---|
|  |  |  | </a-form-item> | 
|---|
|  |  |  | <a-form-item field="perms" label="资源控制权限字符"> | 
|---|
|  |  |  | 
|---|
|  |  |  | </template> | 
|---|
|  |  |  |  | 
|---|
|  |  |  | <script lang="ts" setup> | 
|---|
|  |  |  | import { ref } from "vue"; | 
|---|
|  |  |  | import { IconPlus } from "@arco-design/web-vue/es/icon"; | 
|---|
|  |  |  | import { Resource, ResourceAdd, ResourceById, ResourceDelete, ResourceList, ResourceUpdate } from "@/api/authority"; | 
|---|
|  |  |  | import Authheader from "@/views/authority/components/authheader.vue"; | 
|---|
|  |  |  | import { Modal } from "@arco-design/web-vue"; | 
|---|
|  |  |  | import { ref } from 'vue'; | 
|---|
|  |  |  | import { IconPlus } from '@arco-design/web-vue/es/icon'; | 
|---|
|  |  |  | import { | 
|---|
|  |  |  | Resource, | 
|---|
|  |  |  | ResourceAdd, | 
|---|
|  |  |  | ResourceById, | 
|---|
|  |  |  | ResourceDelete, | 
|---|
|  |  |  | ResourceList, | 
|---|
|  |  |  | ResourceUpdate, | 
|---|
|  |  |  | } from '@/api/authority'; | 
|---|
|  |  |  | import Authheader from '@/views/authority/components/authheader.vue'; | 
|---|
|  |  |  | import { Modal } from '@arco-design/web-vue'; | 
|---|
|  |  |  | import { userModelState } from '@/store'; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | let visible = ref(false); | 
|---|
|  |  |  | let treeData = ref([]); | 
|---|
|  |  |  | let showLine = ref(true); | 
|---|
|  |  |  | let menuTips = ref(["权限管理", "资源"]); | 
|---|
|  |  |  | const fieldNames = { value: "key", label: "value" }; | 
|---|
|  |  |  | // 0目录 1菜单 2按钮 | 
|---|
|  |  |  | let options = ref([ | 
|---|
|  |  |  | { | 
|---|
|  |  |  | key: "3", | 
|---|
|  |  |  | value: "目录" | 
|---|
|  |  |  | }, | 
|---|
|  |  |  | { | 
|---|
|  |  |  | key: "0", | 
|---|
|  |  |  | value: "菜单" | 
|---|
|  |  |  | }, | 
|---|
|  |  |  | { | 
|---|
|  |  |  | key: "1", | 
|---|
|  |  |  | value: "按钮" | 
|---|
|  |  |  | } | 
|---|
|  |  |  | ]); | 
|---|
|  |  |  | let resourceform = ref<Resource>({ | 
|---|
|  |  |  | component: "", | 
|---|
|  |  |  | createTime: "", | 
|---|
|  |  |  | description: "", | 
|---|
|  |  |  | icon: "", | 
|---|
|  |  |  | menuId: "", | 
|---|
|  |  |  | menuName: "", | 
|---|
|  |  |  | menuType: "", | 
|---|
|  |  |  | orderNum: "", | 
|---|
|  |  |  | parentId: "", | 
|---|
|  |  |  | parentName: "", | 
|---|
|  |  |  | children: [], | 
|---|
|  |  |  | path: "", | 
|---|
|  |  |  | perms: "", | 
|---|
|  |  |  | status: "", | 
|---|
|  |  |  | syesourcetype: "", | 
|---|
|  |  |  | target: "", | 
|---|
|  |  |  | updateTime: "" | 
|---|
|  |  |  | }); | 
|---|
|  |  |  | const modelStore = userModelState(); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | const onIconClick = (nodeData) => { | 
|---|
|  |  |  | if (nodeData) { | 
|---|
|  |  |  | resourceform.value.parentId = nodeData.menuId; | 
|---|
|  |  |  | resourceform.value.parentName = nodeData.menuName; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | resourceform.value.component = ""; | 
|---|
|  |  |  | resourceform.value.createTime = ""; | 
|---|
|  |  |  | resourceform.value.description = ""; | 
|---|
|  |  |  | resourceform.value.icon = ""; | 
|---|
|  |  |  | resourceform.value.menuId = ""; | 
|---|
|  |  |  | resourceform.value.menuName = ""; | 
|---|
|  |  |  | resourceform.value.menuType = ""; | 
|---|
|  |  |  | resourceform.value.orderNum = "0"; | 
|---|
|  |  |  | resourceform.value.path = ""; | 
|---|
|  |  |  | resourceform.value.perms = ""; | 
|---|
|  |  |  | resourceform.value.status = ""; | 
|---|
|  |  |  | resourceform.value.syesourcetype = ""; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | visible.value = true; | 
|---|
|  |  |  | }; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | const addresource = async () => { | 
|---|
|  |  |  | await ResourceAdd({ | 
|---|
|  |  |  | ...resourceform.value | 
|---|
|  |  |  | } as unknown as Resource).then((res) => { | 
|---|
|  |  |  | ResourceData(""); | 
|---|
|  |  |  | let visible = ref(false); | 
|---|
|  |  |  | let treeData = ref([]); | 
|---|
|  |  |  | let showLine = ref(true); | 
|---|
|  |  |  | let menuTips = ref(['权限管理', '资源']); | 
|---|
|  |  |  | const fieldNames = { value: 'key', label: 'value' }; | 
|---|
|  |  |  | // 0目录 1菜单 2按钮 | 
|---|
|  |  |  | let options = ref([ | 
|---|
|  |  |  | { | 
|---|
|  |  |  | key: '3', | 
|---|
|  |  |  | value: '目录', | 
|---|
|  |  |  | }, | 
|---|
|  |  |  | { | 
|---|
|  |  |  | key: '0', | 
|---|
|  |  |  | value: '菜单', | 
|---|
|  |  |  | }, | 
|---|
|  |  |  | { | 
|---|
|  |  |  | key: '1', | 
|---|
|  |  |  | value: '按钮', | 
|---|
|  |  |  | }, | 
|---|
|  |  |  | ]); | 
|---|
|  |  |  | let resourceform = ref<Resource>({ | 
|---|
|  |  |  | component: '', | 
|---|
|  |  |  | createTime: '', | 
|---|
|  |  |  | description: '', | 
|---|
|  |  |  | icon: '', | 
|---|
|  |  |  | menuId: '', | 
|---|
|  |  |  | menuName: '', | 
|---|
|  |  |  | menuType: '', | 
|---|
|  |  |  | orderNum: '', | 
|---|
|  |  |  | parentId: '', | 
|---|
|  |  |  | parentName: '', | 
|---|
|  |  |  | children: [], | 
|---|
|  |  |  | path: '', | 
|---|
|  |  |  | perms: '', | 
|---|
|  |  |  | status: '', | 
|---|
|  |  |  | syesourcetype: '', | 
|---|
|  |  |  | target: '', | 
|---|
|  |  |  | updateTime: '', | 
|---|
|  |  |  | }); | 
|---|
|  |  |  | }; | 
|---|
|  |  |  | const rules = { | 
|---|
|  |  |  | // parentName: [ | 
|---|
|  |  |  | //   { | 
|---|
|  |  |  | //     required: true, | 
|---|
|  |  |  | //     message: '请输入上级资源', | 
|---|
|  |  |  | //   }, | 
|---|
|  |  |  | // ], | 
|---|
|  |  |  | // menuName: [ | 
|---|
|  |  |  | //   { | 
|---|
|  |  |  | //     required: true, | 
|---|
|  |  |  | //     message: '请输入资源名称', | 
|---|
|  |  |  | //   }, | 
|---|
|  |  |  | // ], | 
|---|
|  |  |  | // menuType: [ | 
|---|
|  |  |  | //   { | 
|---|
|  |  |  | //     required: true, | 
|---|
|  |  |  | //     message: '请选择资源类型', | 
|---|
|  |  |  | //   }, | 
|---|
|  |  |  | // ], | 
|---|
|  |  |  | // description: [ | 
|---|
|  |  |  | //   { | 
|---|
|  |  |  | //     required: true, | 
|---|
|  |  |  | //     message: '请输入提示词', | 
|---|
|  |  |  | //   }, | 
|---|
|  |  |  | // ], | 
|---|
|  |  |  | // perms: [ | 
|---|
|  |  |  | //   { | 
|---|
|  |  |  | //     required: true, | 
|---|
|  |  |  | //     message: '请输入资源控制权限字符', | 
|---|
|  |  |  | //   }, | 
|---|
|  |  |  | // ], | 
|---|
|  |  |  | }; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | const onIconClickDelete = (nodeData) => { | 
|---|
|  |  |  | ResourceDelete(nodeData.menuId).then(() => { | 
|---|
|  |  |  | ResourceData(""); | 
|---|
|  |  |  | }); | 
|---|
|  |  |  | }; | 
|---|
|  |  |  | const onIconClick = (nodeData) => { | 
|---|
|  |  |  | if (nodeData) { | 
|---|
|  |  |  | resourceform.value.parentId = nodeData.menuId; | 
|---|
|  |  |  | resourceform.value.parentName = nodeData.menuName; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | resourceform.value.component = ''; | 
|---|
|  |  |  | resourceform.value.createTime = ''; | 
|---|
|  |  |  | resourceform.value.description = ''; | 
|---|
|  |  |  | resourceform.value.icon = ''; | 
|---|
|  |  |  | resourceform.value.menuId = ''; | 
|---|
|  |  |  | resourceform.value.menuName = ''; | 
|---|
|  |  |  | resourceform.value.menuType = ''; | 
|---|
|  |  |  | resourceform.value.orderNum = '0'; | 
|---|
|  |  |  | resourceform.value.path = ''; | 
|---|
|  |  |  | resourceform.value.perms = ''; | 
|---|
|  |  |  | resourceform.value.status = ''; | 
|---|
|  |  |  | resourceform.value.syesourcetype = ''; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | const showDetail = (id) => { | 
|---|
|  |  |  | ResourceById(id).then((res) => { | 
|---|
|  |  |  | resourceform.value = { ...res.data }; | 
|---|
|  |  |  | }); | 
|---|
|  |  |  | }; | 
|---|
|  |  |  | const editresource = () => { | 
|---|
|  |  |  | ResourceUpdate({ | 
|---|
|  |  |  | ...resourceform.value | 
|---|
|  |  |  | } as unknown as Resource).then((res) => { | 
|---|
|  |  |  | ResourceData(""); | 
|---|
|  |  |  | Modal.success({ | 
|---|
|  |  |  | title: "保存成功", | 
|---|
|  |  |  | content: "保存成功" | 
|---|
|  |  |  | visible.value = true; | 
|---|
|  |  |  | }; | 
|---|
|  |  |  | const formRef = ref(); | 
|---|
|  |  |  | //图片上传 | 
|---|
|  |  |  | const uploadAction = '/api/v1/llm/upload'; // 替换为你的上传API | 
|---|
|  |  |  | const fileList = ref([]); | 
|---|
|  |  |  | const imageUrls = ref([]); | 
|---|
|  |  |  | const uploadUrl = ref([]); | 
|---|
|  |  |  | const httpUrl = modelStore.hrefUrl; | 
|---|
|  |  |  | const updateFileList = (newFileList) => { | 
|---|
|  |  |  | fileList.value = newFileList; | 
|---|
|  |  |  | }; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | const handleSuccess = (urls) => { | 
|---|
|  |  |  | uploadUrl.value = urls; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | const urlsArr = urls.map((url) => { | 
|---|
|  |  |  | return httpUrl + url; | 
|---|
|  |  |  | }); | 
|---|
|  |  |  | }); | 
|---|
|  |  |  | }; | 
|---|
|  |  |  | const reset = (id) => { | 
|---|
|  |  |  | ResourceById(id).then((res) => { | 
|---|
|  |  |  | resourceform.value = { ...res.data }; | 
|---|
|  |  |  | }); | 
|---|
|  |  |  | }; | 
|---|
|  |  |  | imageUrls.value = urlsArr; // 拿到上传的图片地址 | 
|---|
|  |  |  | }; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | const onDrop = ({ dragNode, dropNode, dropPosition }) => { | 
|---|
|  |  |  | const data = treeData.value; | 
|---|
|  |  |  | ResourceUpdate({ | 
|---|
|  |  |  | orderNum: "0", | 
|---|
|  |  |  | parentId: dropNode.menuId, | 
|---|
|  |  |  | menuId: dragNode.menuId | 
|---|
|  |  |  | }); | 
|---|
|  |  |  | const loop = (data, key, callback) => { | 
|---|
|  |  |  | data.some((item, index, arr) => { | 
|---|
|  |  |  | if (item.menuId === key) { | 
|---|
|  |  |  | callback(item, index, arr); | 
|---|
|  |  |  | return true; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | if (item.children) { | 
|---|
|  |  |  | return loop(item.children, key, callback); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | return false; | 
|---|
|  |  |  | const addresource = async (done) => { | 
|---|
|  |  |  | // formRef.value.validate().then(async (res) => { | 
|---|
|  |  |  | //   if (!res) { | 
|---|
|  |  |  | //   } | 
|---|
|  |  |  | //   done(false); | 
|---|
|  |  |  | // }); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | resourceform.value.icon = uploadUrl.value[0] || ''; | 
|---|
|  |  |  | await ResourceAdd({ | 
|---|
|  |  |  | ...resourceform.value, | 
|---|
|  |  |  | } as unknown as Resource).then((res) => { | 
|---|
|  |  |  | ResourceData(''); | 
|---|
|  |  |  | }); | 
|---|
|  |  |  | }; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | loop(data, dragNode.menuId, (_, index, arr) => { | 
|---|
|  |  |  | arr.splice(index, 1); | 
|---|
|  |  |  | }); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | if (dropPosition === 0) { | 
|---|
|  |  |  | loop(data, dropNode.menuId, (item) => { | 
|---|
|  |  |  | item.children = item.children || []; | 
|---|
|  |  |  | item.children.push(dragNode); | 
|---|
|  |  |  | const onIconClickDelete = (nodeData) => { | 
|---|
|  |  |  | ResourceDelete(nodeData.menuId).then(() => { | 
|---|
|  |  |  | ResourceData(''); | 
|---|
|  |  |  | }); | 
|---|
|  |  |  | } else { | 
|---|
|  |  |  | loop(data, dropNode.menuId, (_, index, arr) => { | 
|---|
|  |  |  | arr.splice(dropPosition < 0 ? index : index + 1, 0, dragNode); | 
|---|
|  |  |  | }; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | const showDetail = (id) => { | 
|---|
|  |  |  | ResourceById(id).then((res) => { | 
|---|
|  |  |  | resourceform.value = { ...res.data }; | 
|---|
|  |  |  | resourceform.value.icon = res.data.icon; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | imageUrls.value.push(httpUrl + res.data.icon); | 
|---|
|  |  |  | }); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | }; | 
|---|
|  |  |  | }; | 
|---|
|  |  |  | const editresource = () => { | 
|---|
|  |  |  | resourceform.value.icon = uploadUrl.value[0] || ''; | 
|---|
|  |  |  | ResourceUpdate({ | 
|---|
|  |  |  | ...resourceform.value, | 
|---|
|  |  |  | } as unknown as Resource).then((res) => { | 
|---|
|  |  |  | ResourceData(''); | 
|---|
|  |  |  | Modal.success({ | 
|---|
|  |  |  | title: '保存成功', | 
|---|
|  |  |  | content: '保存成功', | 
|---|
|  |  |  | }); | 
|---|
|  |  |  | }); | 
|---|
|  |  |  | }; | 
|---|
|  |  |  | const reset = (id) => { | 
|---|
|  |  |  | ResourceById(id).then((res) => { | 
|---|
|  |  |  | resourceform.value = { ...res.data }; | 
|---|
|  |  |  | }); | 
|---|
|  |  |  | }; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | const handleCancel = (type) => { | 
|---|
|  |  |  | if (type == 1) { | 
|---|
|  |  |  | visible.value = false; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | }; | 
|---|
|  |  |  | const onDrop = ({ dragNode, dropNode, dropPosition }) => { | 
|---|
|  |  |  | const data = treeData.value; | 
|---|
|  |  |  | ResourceUpdate({ | 
|---|
|  |  |  | orderNum: '0', | 
|---|
|  |  |  | parentId: dropNode.menuId, | 
|---|
|  |  |  | menuId: dragNode.menuId, | 
|---|
|  |  |  | }); | 
|---|
|  |  |  | const loop = (data, key, callback) => { | 
|---|
|  |  |  | data.some((item, index, arr) => { | 
|---|
|  |  |  | if (item.menuId === key) { | 
|---|
|  |  |  | callback(item, index, arr); | 
|---|
|  |  |  | return true; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | if (item.children) { | 
|---|
|  |  |  | return loop(item.children, key, callback); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | return false; | 
|---|
|  |  |  | }); | 
|---|
|  |  |  | }; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | const ResourceData = async (key) => { | 
|---|
|  |  |  | await ResourceList(key).then((res) => { | 
|---|
|  |  |  | treeData.value = [...res.rows]; | 
|---|
|  |  |  | }); | 
|---|
|  |  |  | }; | 
|---|
|  |  |  | loop(data, dragNode.menuId, (_, index, arr) => { | 
|---|
|  |  |  | arr.splice(index, 1); | 
|---|
|  |  |  | }); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | ResourceData(""); | 
|---|
|  |  |  | if (dropPosition === 0) { | 
|---|
|  |  |  | loop(data, dropNode.menuId, (item) => { | 
|---|
|  |  |  | item.children = item.children || []; | 
|---|
|  |  |  | item.children.push(dragNode); | 
|---|
|  |  |  | }); | 
|---|
|  |  |  | } else { | 
|---|
|  |  |  | loop(data, dropNode.menuId, (_, index, arr) => { | 
|---|
|  |  |  | arr.splice(dropPosition < 0 ? index : index + 1, 0, dragNode); | 
|---|
|  |  |  | }); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | }; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | const handleCancel = (type) => { | 
|---|
|  |  |  | if (type == 1) { | 
|---|
|  |  |  | visible.value = false; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | }; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | const ResourceData = async (key) => { | 
|---|
|  |  |  | await ResourceList(key).then((res) => { | 
|---|
|  |  |  | treeData.value = [...res.rows]; | 
|---|
|  |  |  | }); | 
|---|
|  |  |  | }; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | ResourceData(''); | 
|---|
|  |  |  | </script> | 
|---|
|  |  |  |  | 
|---|
|  |  |  | <style scoped> | 
|---|
|  |  |  | .tree-demo :deep(.tree-node-dropover) > :deep(.arco-tree-node-title), | 
|---|
|  |  |  | .tree-demo :deep(.tree-node-dropover) > :deep(.arco-tree-node-title):hover { | 
|---|
|  |  |  | animation: blinkBg 0.4s 2; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @keyframes blinkBg { | 
|---|
|  |  |  | 0% { | 
|---|
|  |  |  | background-color: transparent; | 
|---|
|  |  |  | .tree-demo :deep(.tree-node-dropover) > :deep(.arco-tree-node-title), | 
|---|
|  |  |  | .tree-demo :deep(.tree-node-dropover) > :deep(.arco-tree-node-title):hover { | 
|---|
|  |  |  | animation: blinkBg 0.4s 2; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | 100% { | 
|---|
|  |  |  | background-color: var(--color-primary-light-1); | 
|---|
|  |  |  | @keyframes blinkBg { | 
|---|
|  |  |  | 0% { | 
|---|
|  |  |  | background-color: transparent; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | 100% { | 
|---|
|  |  |  | background-color: var(--color-primary-light-1); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | </style> | 
|---|