DeviceCompleteSet.vue 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589
  1. <template>
  2. <el-row :gutter="20">
  3. <!-- 左侧部门树 -->
  4. <DeptTree @node-click="handleDeptNodeClick" v-model:collapsed="isLeftContentCollapsed" />
  5. <el-col :xs="24" :span="isLeftContentCollapsed ? 24 : 20">
  6. <ContentWrap style="border: none; box-shadow: 0 2px 2px 0 rgba(0, 0, 0, 0.1)">
  7. <!-- 搜索工作栏 -->
  8. <el-form
  9. class="-mb-15px"
  10. :model="queryParams"
  11. ref="queryFormRef"
  12. :inline="true"
  13. label-width="68px"
  14. >
  15. <!-- <el-form-item label="部门名称" prop="deptName" style="margin-left: 20px">
  16. <el-input
  17. v-model="queryParams.deptName"
  18. placeholder="请输入部门名称"
  19. clearable
  20. @keyup.enter="handleQuery"
  21. class="!w-200px"
  22. />
  23. </el-form-item> -->
  24. <el-form-item label="成套名称" prop="name">
  25. <el-input
  26. v-model="queryParams.name"
  27. placeholder="请输入成套名称"
  28. clearable
  29. @keyup.enter="handleQuery"
  30. class="!w-200px"
  31. />
  32. </el-form-item>
  33. <el-form-item>
  34. <el-button @click="handleAdd" type="primary"
  35. ><Icon icon="ep:plus" class="mr-5px" />新增成套</el-button
  36. >
  37. <el-button @click="handleQuery"
  38. ><Icon icon="ep:search" class="mr-5px" /> {{ t('devicePerson.search') }}</el-button
  39. >
  40. <el-button @click="resetQuery"
  41. ><Icon icon="ep:refresh" class="mr-5px" /> {{ t('devicePerson.reset') }}</el-button
  42. >
  43. </el-form-item>
  44. </el-form>
  45. </ContentWrap>
  46. <!-- 列表 -->
  47. <ContentWrap style="border: none; box-shadow: 0 2px 2px 0 rgba(0, 0, 0, 0.1)">
  48. <zm-table :loading="loading" height="calc(85vh - 138px)" :data="list">
  49. <zm-table-column :label="t('monitor.serial')" width="70" align="center">
  50. <template #default="scope">
  51. {{ scope.$index + 1 }}
  52. </template>
  53. </zm-table-column>
  54. <zm-table-column label="部门名称" align="center" prop="deptName" />
  55. <zm-table-column label="成套名称" align="center" prop="name" />
  56. <zm-table-column label="成套类型" align="center">
  57. <template #default="scope">
  58. <dict-tag :type="DICT_TYPE.DEVICE_GROUP_TYPE" :value="scope.row.type" />
  59. </template>
  60. </zm-table-column>
  61. <zm-table-column label="在线状态" align="center">
  62. <template #default="scope">
  63. <el-tag type="success" :underline="false" v-if="scope.row.ifOnline === true"
  64. >在线</el-tag
  65. >
  66. <el-tag type="info" :underline="false" v-else>离线</el-tag>
  67. </template>
  68. </zm-table-column>
  69. <zm-table-column label="描述" align="center" prop="remark" />
  70. <zm-table-column label="设备数量" align="center" prop="deviceCount">
  71. <template #default="scope">
  72. {{ (scope.row.details && scope.row.details.length) || 0 }}
  73. </template>
  74. </zm-table-column>
  75. <zm-table-column label="主设备" align="center" prop="mainDeviceName">
  76. <template #default="scope">
  77. {{
  78. scope.row.details.filter((item) => item.ifMaster)[0]?.deviceName +
  79. ' ' +
  80. scope.row.details.filter((item) => item.ifMaster)[0]?.deviceCode || '无'
  81. }}
  82. </template>
  83. </zm-table-column>
  84. <zm-table-column
  85. :label="t('devicePerson.operation')"
  86. align="center"
  87. min-width="120px"
  88. action
  89. >
  90. <template #default="scope">
  91. <el-button link type="primary" @click="handleEdit(scope.row)"> 编辑 </el-button>
  92. <el-button link type="danger" @click="handleDelete(scope.row.id)"> 删除 </el-button>
  93. </template>
  94. </zm-table-column>
  95. </zm-table>
  96. <!-- 分页 -->
  97. <Pagination
  98. :total="total"
  99. v-model:page="queryParams.pageNo"
  100. v-model:limit="queryParams.pageSize"
  101. @pagination="getList"
  102. />
  103. </ContentWrap>
  104. </el-col>
  105. </el-row>
  106. <!-- 新增/编辑成套设备对话框 -->
  107. <el-dialog :title="dialogTitle" v-model="dialogVisible" width="800px" @close="cancel">
  108. <template #header>
  109. <div class="my-header" style="padding-bottom: 20px">
  110. <span>{{ dialogTitle }}</span>
  111. </div>
  112. </template>
  113. <el-form ref="formRef" :model="formData" :rules="formRules" label-width="80px">
  114. <el-row :gutter="20">
  115. <el-col :span="8">
  116. <el-form-item label="成套名称" prop="name">
  117. <el-input v-model="formData.name" placeholder="请输入成套名称" />
  118. </el-form-item>
  119. </el-col>
  120. <el-col :span="8">
  121. <el-form-item label="成套类型" prop="type">
  122. <el-select
  123. v-model="formData.type"
  124. placeholder="请选择成套类型"
  125. clearable
  126. class="!w-240px"
  127. >
  128. <el-option
  129. v-for="dict in getStrDictOptions(DICT_TYPE.DEVICE_GROUP_TYPE)"
  130. :key="dict.value"
  131. :label="dict.label"
  132. :value="dict.value"
  133. />
  134. </el-select>
  135. </el-form-item>
  136. </el-col>
  137. <el-col :span="8">
  138. <el-form-item :label="t('iotDevice.dept')" prop="deptId">
  139. <el-tree-select
  140. v-model="formData.deptId"
  141. :data="deptList"
  142. :props="defaultProps"
  143. check-strictly
  144. node-key="id"
  145. filterable
  146. placeholder="请选择所在部门"
  147. @change="handleDeptChange"
  148. />
  149. </el-form-item>
  150. </el-col>
  151. </el-row>
  152. <el-row :gutter="20">
  153. <el-col :span="24">
  154. <el-form-item label="描述" prop="remark">
  155. <el-input
  156. v-model="formData.remark"
  157. type="textarea"
  158. placeholder="请输入描述"
  159. :rows="2"
  160. />
  161. </el-form-item>
  162. </el-col>
  163. </el-row>
  164. <el-row :gutter="20">
  165. <el-col :span="24">
  166. <el-form-item label="选择设备" prop="devices">
  167. <div class="transfer-container">
  168. <el-transfer
  169. v-model="selectedDeviceIds"
  170. :data="deviceOptions"
  171. :titles="['设备列表', '已选择设备']"
  172. :button-texts="['移除', '添加']"
  173. filterable
  174. :filter-method="filterDeviceMethod"
  175. filter-placeholder="请输入设备名称"
  176. @change="rightDeviceChange"
  177. >
  178. <template #left-empty>
  179. <el-empty :image-size="60" :description="isEdit ? '加载中...' : '请选择设备'" />
  180. </template>
  181. <template #right-empty>
  182. <el-empty :image-size="60" :description="isEdit ? '加载中...' : '请选择设备'" />
  183. </template>
  184. </el-transfer>
  185. </div>
  186. </el-form-item>
  187. </el-col>
  188. </el-row>
  189. <el-row :gutter="20" v-if="selectedDevices.length > 0">
  190. <el-col :span="12">
  191. <el-form-item label="设置主设备" prop="mainDevice" label-width="100">
  192. <el-select
  193. v-model="mainDeviceId"
  194. placeholder="请选择主设备"
  195. clearable
  196. filterable
  197. @change="setMainDevice"
  198. >
  199. <el-option
  200. v-for="device in selectedDevices"
  201. :key="device.id"
  202. :label="device.label"
  203. :value="device.id"
  204. />
  205. </el-select>
  206. </el-form-item>
  207. </el-col>
  208. </el-row>
  209. </el-form>
  210. <template #footer>
  211. <el-button @click="cancel">取 消</el-button>
  212. <el-button type="primary" @click="submit">确 定</el-button>
  213. </template>
  214. </el-dialog>
  215. </template>
  216. <script setup lang="ts">
  217. import { IotDeviceApi } from '@/api/pms/device'
  218. import DeptTree from '@/views/system/user/DeptTree2.vue'
  219. import { defaultProps, handleTree } from '@/utils/tree'
  220. import * as DeptApi from '@/api/system/dept'
  221. import { ElMessageBox } from 'element-plus'
  222. import { DICT_TYPE, getStrDictOptions } from '@/utils/dict'
  223. const deptList = ref<Tree[]>([]) // 树形结构
  224. import { useTableComponents } from '@/components/ZmTable/useTableComponents'
  225. const { ZmTable, ZmTableColumn } = useTableComponents()
  226. defineOptions({ name: 'IotDeviceComplete' })
  227. const loading = ref(true) // 列表的加载中
  228. const { t } = useI18n()
  229. let isLeftContentCollapsed = ref(false)
  230. const list = ref([]) // 列表的数据
  231. const total = ref(0) // 列表的总页数
  232. const queryParams = reactive({
  233. pageNo: 1,
  234. pageSize: 10,
  235. deptName: undefined,
  236. name: undefined,
  237. deptId: undefined
  238. })
  239. const queryFormRef = ref(null) // 搜索的表单
  240. const contentSpan = ref(20)
  241. const treeShow = ref(true)
  242. // 对话框相关
  243. const dialogVisible = ref(false)
  244. const dialogTitle = ref('')
  245. const isEdit = ref(false)
  246. // 表单相关
  247. const formRef = ref()
  248. const formData = ref({
  249. name: '',
  250. details: [],
  251. deptId: '',
  252. remark: '',
  253. type: ''
  254. })
  255. // 表单验证规则
  256. const formRules = {
  257. name: [{ required: true, message: '成套名称不能为空', trigger: 'blur' }],
  258. code: [{ required: true, message: '成套编码不能为空', trigger: 'blur' }],
  259. deptId: [{ required: true, message: '请选择部门', trigger: 'change' }],
  260. devices: [
  261. {
  262. required: true,
  263. validator: (rule: any, value: any, callback: any) => {
  264. if (selectedDeviceIds.value.length === 0) {
  265. callback(new Error('请至少选择一个设备'))
  266. } else {
  267. callback()
  268. }
  269. },
  270. trigger: 'change'
  271. }
  272. ],
  273. mainDevice: [
  274. {
  275. required: true,
  276. validator: (rule: any, value: any, callback: any) => {
  277. if (!mainDeviceId.value) {
  278. callback(new Error('请选择主设备'))
  279. } else {
  280. callback()
  281. }
  282. },
  283. trigger: 'change'
  284. }
  285. ]
  286. }
  287. // 部门树数据
  288. const selectedDeptId = ref<number | string>('')
  289. // 新增时部门改变时获取设备列表
  290. const handleDeptChange = async (deptId) => {
  291. if (deptId) {
  292. selectedDeptId.value = deptId
  293. getDeviceList(deptId)
  294. }
  295. // 清空主设备she 设备列表
  296. selectedDevices.value = []
  297. mainDeviceId.value = ''
  298. selectedDeviceIds.value = []
  299. }
  300. // 获取设备列表
  301. const getDeviceList = async (deptId) => {
  302. try {
  303. const res = await IotDeviceApi.getIotDeviceSetOptions(deptId)
  304. deviceOptions.value = res.map((item) => ({
  305. key: item.id, // 始终使用id作为key
  306. label: `${item.deviceName} (${item.deviceCode})`,
  307. ...item
  308. }))
  309. } catch (err) {
  310. console.error(err)
  311. }
  312. }
  313. // 设备选择相关
  314. const deviceOptions = ref<any[]>([])
  315. const selectedDeviceIds = ref([])
  316. const selectedDevices = ref([])
  317. const mainDeviceId = ref('')
  318. /** 查询列表 */
  319. const getList = async () => {
  320. loading.value = true
  321. try {
  322. const data = await IotDeviceApi.getIotDeviceSetList(queryParams)
  323. list.value = data.list
  324. total.value = data.total
  325. } finally {
  326. loading.value = false
  327. }
  328. }
  329. /** 首页处理部门被点击 */
  330. const handleDeptNodeClick = async (row) => {
  331. queryParams.deptId = row.id
  332. await getList()
  333. }
  334. /** 搜索按钮操作 */
  335. const handleQuery = () => {
  336. queryParams.pageNo = 1
  337. getList()
  338. }
  339. /** 重置按钮操作 */
  340. const resetQuery = () => {
  341. queryFormRef.value.resetFields()
  342. handleQuery()
  343. }
  344. // 设备过滤方法
  345. const filterDeviceMethod = (query, item) => {
  346. return item.label.toLowerCase().includes(query.toLowerCase())
  347. }
  348. // 更新已选择的设备列表
  349. const updateSelectedDevices = () => {
  350. // 根据 selectedDeviceIds 从 deviceOptions 中找到对应的设备
  351. selectedDevices.value = deviceOptions.value.filter((item) =>
  352. selectedDeviceIds.value.includes(item.key)
  353. )
  354. console.log('selectedDevices>>>>>>>>>>>>>>>>', selectedDevices.value)
  355. // 构建 details 数据
  356. formData.value.details = selectedDevices.value.map((item) => ({
  357. deviceId: item.id,
  358. deviceName: item.deviceName,
  359. deviceCode: item.deviceCode,
  360. deptId: item.deptId,
  361. ifMaster: item.id === mainDeviceId.value ? true : false
  362. }))
  363. // 如果主设备不在当前选择中,则清空主设备
  364. if (
  365. mainDeviceId.value &&
  366. !selectedDevices.value.some((device) => device.id === mainDeviceId.value)
  367. ) {
  368. mainDeviceId.value = ''
  369. }
  370. }
  371. const rightDeviceChange = (val) => {
  372. selectedDeviceIds.value = val
  373. updateSelectedDevices()
  374. // 手动触发验证
  375. if (formRef.value) {
  376. formRef.value.validateField('devices')
  377. if (val.length > 0) {
  378. formRef.value.validateField('mainDevice')
  379. }
  380. }
  381. }
  382. // 设置主设备
  383. const setMainDevice = (val) => {
  384. mainDeviceId.value = val
  385. // 更新 details 中的 ifMaster 字段
  386. console.log('selectedDevices.value>>>>>>>>>>>>>>>>', selectedDevices.value)
  387. formData.value.details = selectedDevices.value.map((item) => ({
  388. deviceId: item.id,
  389. deviceName: item.deviceName,
  390. deviceCode: item.deviceCode,
  391. deptId: item.deptId,
  392. ifMaster: item.id === mainDeviceId.value ? true : false
  393. }))
  394. console.log('formData.value.details>>>>>>>>>>>>>>>>', formData.value.details)
  395. // 手动触发验证
  396. if (formRef.value) {
  397. formRef.value.validateField('mainDevice')
  398. }
  399. }
  400. // 显示新增对话框
  401. const handleAdd = () => {
  402. isEdit.value = false
  403. dialogTitle.value = '新增成套设备'
  404. resetForm()
  405. dialogVisible.value = true
  406. }
  407. // 显示编辑对话框
  408. const handleEdit = (row) => {
  409. isEdit.value = true
  410. dialogTitle.value = '编辑成套设备'
  411. formData.value = {
  412. ...row
  413. }
  414. // 先清空之前的选项
  415. selectedDeviceIds.value = []
  416. selectedDevices.value = []
  417. deviceOptions.value = []
  418. // 获取设备列表后再设置已选择的设备
  419. getDeviceList(row.deptId).then(() => {
  420. // 设置已选择的设备IDs(使用deviceOptions中的key值)
  421. selectedDeviceIds.value = row.details.map((item) => {
  422. // 在编辑模式下,使用deviceOptions中对应项的key作为标识
  423. const option = deviceOptions.value.find((opt) => opt.deviceId === item.deviceId)
  424. return option ? option.key : item.deviceId
  425. })
  426. mainDeviceId.value = row.details.find((item) => item.ifMaster)?.deviceId || ''
  427. // 更新selectedDevices数组
  428. nextTick(() => {
  429. updateSelectedDevices()
  430. })
  431. })
  432. dialogVisible.value = true
  433. }
  434. //删除成套
  435. const handleDelete = async (id: number) => {
  436. ElMessageBox.confirm('确定要删除该成套设备吗?', '提示', {
  437. confirmButtonText: '确定',
  438. cancelButtonText: '取消',
  439. type: 'warning'
  440. })
  441. .then(async () => {
  442. try {
  443. await IotDeviceApi.deleteIotDeviceSet(id)
  444. ElMessage.success('删除成功')
  445. getList()
  446. } catch (error) {
  447. console.error(error)
  448. }
  449. })
  450. .catch(() => {
  451. // 取消操作
  452. })
  453. }
  454. // 重置表单
  455. const resetForm = () => {
  456. formData.value = {
  457. name: '',
  458. details: [],
  459. deptId: '',
  460. remark: ''
  461. }
  462. selectedDeviceIds.value = []
  463. selectedDevices.value = []
  464. deviceOptions.value = []
  465. mainDeviceId.value = ''
  466. }
  467. // 取消操作
  468. const cancel = () => {
  469. dialogVisible.value = false
  470. resetForm()
  471. }
  472. // 提交表单
  473. const submit = async () => {
  474. if (!formRef.value) return
  475. await formRef.value.validate(async (valid) => {
  476. if (!valid) return
  477. try {
  478. const data = {
  479. ...formData.value
  480. }
  481. console.log('提交数据:', data)
  482. if (isEdit.value) {
  483. await IotDeviceApi.updateIotDeviceSet(data)
  484. ElMessage.success('编辑成功')
  485. } else {
  486. await IotDeviceApi.createIotDeviceSet(data)
  487. ElMessage.success('新增成功')
  488. }
  489. dialogVisible.value = false
  490. getList()
  491. } catch (error) {
  492. console.error(error)
  493. }
  494. })
  495. }
  496. onMounted(async () => {
  497. getList()
  498. deptList.value = handleTree(await DeptApi.getSimpleDeptList())
  499. })
  500. </script>
  501. <style scoped>
  502. .transfer-container {
  503. display: flex;
  504. flex-direction: column;
  505. }
  506. .transfer-footer {
  507. margin-top: 8px;
  508. text-align: right;
  509. }
  510. ::deep(.el-transfer-panel) {
  511. width: 300px;
  512. }
  513. ::deep(.el-tree--highlight-current) {
  514. height: 200px !important;
  515. }
  516. /* 调整穿梭框面板主体高度 */
  517. ::deep(.el-transfer-panel__body) {
  518. height: 850px !important;
  519. }
  520. /* 如果需要进一步调整,可以分别设置左右面板的高度 */
  521. ::deep(.el-transfer-panel .el-transfer-panel__list) {
  522. height: 850px !important; /* 调整列表区域高度 */
  523. }
  524. .el-transfer {
  525. --el-transfer-panel-body-height: 500px;
  526. }
  527. </style>