index.vue 13 KB

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