| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586 |
- <template>
- <el-row :gutter="20">
- <!-- 左侧部门树 -->
- <el-col :span="4" :xs="24">
- <ContentWrap class="h-1/1" v-if="treeShow">
- <DeptTree @node-click="handleDeptNodeClick" />
- </ContentWrap>
- </el-col>
- <el-col :span="contentSpan" :xs="24">
- <ContentWrap>
- <!-- 搜索工作栏 -->
- <el-form
- class="-mb-15px"
- :model="queryParams"
- ref="queryFormRef"
- :inline="true"
- label-width="68px"
- >
- <!-- <el-form-item label="部门名称" prop="deptName" style="margin-left: 20px">
- <el-input
- v-model="queryParams.deptName"
- placeholder="请输入部门名称"
- clearable
- @keyup.enter="handleQuery"
- class="!w-200px"
- />
- </el-form-item> -->
- <el-form-item label="成套名称" prop="name">
- <el-input
- v-model="queryParams.name"
- placeholder="请输入成套名称"
- clearable
- @keyup.enter="handleQuery"
- class="!w-200px"
- />
- </el-form-item>
- <el-form-item>
- <el-button @click="handleAdd" type="primary"
- ><Icon icon="ep:plus" class="mr-5px" />新增成套</el-button
- >
- <el-button @click="handleQuery"
- ><Icon icon="ep:search" class="mr-5px" /> {{ t('devicePerson.search') }}</el-button
- >
- <el-button @click="resetQuery"
- ><Icon icon="ep:refresh" class="mr-5px" /> {{ t('devicePerson.reset') }}</el-button
- >
- </el-form-item>
- </el-form>
- </ContentWrap>
- <!-- 列表 -->
- <ContentWrap>
- <el-table v-loading="loading" :data="list" :stripe="true" :show-overflow-tooltip="true">
- <el-table-column :label="t('monitor.serial')" width="70" align="center">
- <template #default="scope">
- {{ scope.$index + 1 }}
- </template>
- </el-table-column>
- <el-table-column label="部门名称" align="center" prop="deptName" />
- <el-table-column label="成套名称" align="center" prop="name" />
- <el-table-column label="成套类型" align="center">
- <template #default="scope">
- <dict-tag :type="DICT_TYPE.DEVICE_GROUP_TYPE" :value="scope.row.type" />
- </template>
- </el-table-column>
- <el-table-column label="在线状态" align="center">
- <template #default="scope">
- <el-tag type="success" :underline="false" v-if="scope.row.ifOnline === true"
- >在线</el-tag
- >
- <el-tag type="info" :underline="false" v-else>离线</el-tag>
- </template>
- </el-table-column>
- <el-table-column label="描述" align="center" prop="remark" />
- <el-table-column label="设备数量" align="center" prop="deviceCount">
- <template #default="scope">
- {{ (scope.row.details && scope.row.details.length) || 0 }}
- </template>
- </el-table-column>
- <el-table-column label="主设备" align="center" prop="mainDeviceName">
- <template #default="scope">
- {{
- scope.row.details.filter((item) => item.ifMaster)[0]?.deviceName +
- ' ' +
- scope.row.details.filter((item) => item.ifMaster)[0]?.deviceCode || '无'
- }}
- </template>
- </el-table-column>
- <el-table-column :label="t('devicePerson.operation')" align="center" min-width="120px">
- <template #default="scope">
- <el-button link type="primary" @click="handleEdit(scope.row)"> 编辑 </el-button>
- <el-button link type="danger" @click="handleDelete(scope.row.id)"> 删除 </el-button>
- </template>
- </el-table-column>
- </el-table>
- <!-- 分页 -->
- <Pagination
- :total="total"
- v-model:page="queryParams.pageNo"
- v-model:limit="queryParams.pageSize"
- @pagination="getList"
- />
- </ContentWrap>
- </el-col>
- </el-row>
- <!-- 新增/编辑成套设备对话框 -->
- <el-dialog :title="dialogTitle" v-model="dialogVisible" width="800px" @close="cancel">
- <template #header>
- <div class="my-header" style="padding-bottom: 20px">
- <span>{{ dialogTitle }}</span>
- </div>
- </template>
- <el-form ref="formRef" :model="formData" :rules="formRules" label-width="80px">
- <el-row :gutter="20">
- <el-col :span="8">
- <el-form-item label="成套名称" prop="name">
- <el-input v-model="formData.name" placeholder="请输入成套名称" />
- </el-form-item>
- </el-col>
- <el-col :span="8">
- <el-form-item label="成套类型" prop="type">
- <el-select
- v-model="formData.type"
- placeholder="请选择成套类型"
- clearable
- class="!w-240px"
- >
- <el-option
- v-for="dict in getStrDictOptions(DICT_TYPE.DEVICE_GROUP_TYPE)"
- :key="dict.value"
- :label="dict.label"
- :value="dict.value"
- />
- </el-select>
- </el-form-item>
- </el-col>
- <el-col :span="8">
- <el-form-item :label="t('iotDevice.dept')" prop="deptId">
- <el-tree-select
- v-model="formData.deptId"
- :data="deptList"
- :props="defaultProps"
- check-strictly
- node-key="id"
- filterable
- placeholder="请选择所在部门"
- @change="handleDeptChange"
- />
- </el-form-item>
- </el-col>
- </el-row>
- <el-row :gutter="20">
- <el-col :span="24">
- <el-form-item label="描述" prop="remark">
- <el-input
- v-model="formData.remark"
- type="textarea"
- placeholder="请输入描述"
- :rows="2"
- />
- </el-form-item>
- </el-col>
- </el-row>
- <el-row :gutter="20">
- <el-col :span="24">
- <el-form-item label="选择设备" prop="devices">
- <div class="transfer-container">
- <el-transfer
- v-model="selectedDeviceIds"
- :data="deviceOptions"
- :titles="['设备列表', '已选择设备']"
- :button-texts="['移除', '添加']"
- filterable
- :filter-method="filterDeviceMethod"
- filter-placeholder="请输入设备名称"
- @change="rightDeviceChange"
- >
- <template #left-empty>
- <el-empty :image-size="60" :description="isEdit ? '加载中...' : '请选择设备'" />
- </template>
- <template #right-empty>
- <el-empty :image-size="60" :description="isEdit ? '加载中...' : '请选择设备'" />
- </template>
- </el-transfer>
- </div>
- </el-form-item>
- </el-col>
- </el-row>
- <el-row :gutter="20" v-if="selectedDevices.length > 0">
- <el-col :span="12">
- <el-form-item label="设置主设备" prop="mainDevice" label-width="100">
- <el-select
- v-model="mainDeviceId"
- placeholder="请选择主设备"
- clearable
- filterable
- @change="setMainDevice"
- >
- <el-option
- v-for="device in selectedDevices"
- :key="device.id"
- :label="device.label"
- :value="device.id"
- />
- </el-select>
- </el-form-item>
- </el-col>
- </el-row>
- </el-form>
- <template #footer>
- <el-button @click="cancel">取 消</el-button>
- <el-button type="primary" @click="submit">确 定</el-button>
- </template>
- </el-dialog>
- </template>
- <script setup lang="ts">
- import { IotDeviceApi } from '@/api/pms/device'
- import DeptTree from '@/views/system/user/DeptTree.vue'
- import { defaultProps, handleTree } from '@/utils/tree'
- import * as DeptApi from '@/api/system/dept'
- import { ElMessageBox } from 'element-plus'
- import { DICT_TYPE, getStrDictOptions } from '@/utils/dict'
- const deptList = ref<Tree[]>([]) // 树形结构
- defineOptions({ name: 'IotDeviceComplete' })
- const loading = ref(true) // 列表的加载中
- const { t } = useI18n()
- const list = ref([]) // 列表的数据
- const total = ref(0) // 列表的总页数
- const queryParams = reactive({
- pageNo: 1,
- pageSize: 10,
- deptName: undefined,
- name: undefined,
- deptId: undefined
- })
- const queryFormRef = ref(null) // 搜索的表单
- const contentSpan = ref(20)
- const treeShow = ref(true)
- // 对话框相关
- const dialogVisible = ref(false)
- const dialogTitle = ref('')
- const isEdit = ref(false)
- // 表单相关
- const formRef = ref()
- const formData = ref({
- name: '',
- details: [],
- deptId: '',
- remark: '',
- type: ''
- })
- // 表单验证规则
- const formRules = {
- name: [{ required: true, message: '成套名称不能为空', trigger: 'blur' }],
- code: [{ required: true, message: '成套编码不能为空', trigger: 'blur' }],
- deptId: [{ required: true, message: '请选择部门', trigger: 'change' }],
- devices: [
- {
- required: true,
- validator: (rule: any, value: any, callback: any) => {
- if (selectedDeviceIds.value.length === 0) {
- callback(new Error('请至少选择一个设备'))
- } else {
- callback()
- }
- },
- trigger: 'change'
- }
- ],
- mainDevice: [
- {
- required: true,
- validator: (rule: any, value: any, callback: any) => {
- if (!mainDeviceId.value) {
- callback(new Error('请选择主设备'))
- } else {
- callback()
- }
- },
- trigger: 'change'
- }
- ]
- }
- // 部门树数据
- const selectedDeptId = ref<number | string>('')
- // 新增时部门改变时获取设备列表
- const handleDeptChange = async (deptId) => {
- if (deptId) {
- selectedDeptId.value = deptId
- getDeviceList(deptId)
- }
- // 清空主设备she 设备列表
- selectedDevices.value = []
- mainDeviceId.value = ''
- selectedDeviceIds.value = []
- }
- // 获取设备列表
- const getDeviceList = async (deptId) => {
- try {
- const res = await IotDeviceApi.getIotDeviceSetOptions(deptId)
- deviceOptions.value = res.map((item) => ({
- key: item.id, // 始终使用id作为key
- label: `${item.deviceName} (${item.deviceCode})`,
- ...item
- }))
- } catch (err) {
- console.error(err)
- }
- }
- // 设备选择相关
- const deviceOptions = ref<any[]>([])
- const selectedDeviceIds = ref([])
- const selectedDevices = ref([])
- const mainDeviceId = ref('')
- /** 查询列表 */
- const getList = async () => {
- loading.value = true
- try {
- const data = await IotDeviceApi.getIotDeviceSetList(queryParams)
- list.value = data.list
- total.value = data.total
- } finally {
- loading.value = false
- }
- }
- /** 首页处理部门被点击 */
- const handleDeptNodeClick = async (row) => {
- queryParams.deptId = row.id
- await getList()
- }
- /** 搜索按钮操作 */
- const handleQuery = () => {
- queryParams.pageNo = 1
- getList()
- }
- /** 重置按钮操作 */
- const resetQuery = () => {
- queryFormRef.value.resetFields()
- handleQuery()
- }
- // 设备过滤方法
- const filterDeviceMethod = (query, item) => {
- return item.label.toLowerCase().includes(query.toLowerCase())
- }
- // 更新已选择的设备列表
- const updateSelectedDevices = () => {
- // 根据 selectedDeviceIds 从 deviceOptions 中找到对应的设备
- selectedDevices.value = deviceOptions.value.filter((item) =>
- selectedDeviceIds.value.includes(item.key)
- )
- console.log('selectedDevices>>>>>>>>>>>>>>>>', selectedDevices.value)
- // 构建 details 数据
- formData.value.details = selectedDevices.value.map((item) => ({
- deviceId: item.id,
- deviceName: item.deviceName,
- deviceCode: item.deviceCode,
- deptId: item.deptId,
- ifMaster: item.id === mainDeviceId.value ? true : false
- }))
- // 如果主设备不在当前选择中,则清空主设备
- if (
- mainDeviceId.value &&
- !selectedDevices.value.some((device) => device.id === mainDeviceId.value)
- ) {
- mainDeviceId.value = ''
- }
- }
- const rightDeviceChange = (val) => {
- selectedDeviceIds.value = val
- updateSelectedDevices()
- // 手动触发验证
- if (formRef.value) {
- formRef.value.validateField('devices')
- if (val.length > 0) {
- formRef.value.validateField('mainDevice')
- }
- }
- }
- // 设置主设备
- const setMainDevice = (val) => {
- mainDeviceId.value = val
- // 更新 details 中的 ifMaster 字段
- console.log('selectedDevices.value>>>>>>>>>>>>>>>>', selectedDevices.value)
- formData.value.details = selectedDevices.value.map((item) => ({
- deviceId: item.id,
- deviceName: item.deviceName,
- deviceCode: item.deviceCode,
- deptId: item.deptId,
- ifMaster: item.id === mainDeviceId.value ? true : false
- }))
- console.log('formData.value.details>>>>>>>>>>>>>>>>', formData.value.details)
- // 手动触发验证
- if (formRef.value) {
- formRef.value.validateField('mainDevice')
- }
- }
- // 显示新增对话框
- const handleAdd = () => {
- isEdit.value = false
- dialogTitle.value = '新增成套设备'
- resetForm()
- dialogVisible.value = true
- }
- // 显示编辑对话框
- const handleEdit = (row) => {
- isEdit.value = true
- dialogTitle.value = '编辑成套设备'
- formData.value = {
- ...row
- }
- // 先清空之前的选项
- selectedDeviceIds.value = []
- selectedDevices.value = []
- deviceOptions.value = []
- // 获取设备列表后再设置已选择的设备
- getDeviceList(row.deptId).then(() => {
- // 设置已选择的设备IDs(使用deviceOptions中的key值)
- selectedDeviceIds.value = row.details.map((item) => {
- // 在编辑模式下,使用deviceOptions中对应项的key作为标识
- const option = deviceOptions.value.find((opt) => opt.deviceId === item.deviceId)
- return option ? option.key : item.deviceId
- })
- mainDeviceId.value = row.details.find((item) => item.ifMaster)?.deviceId || ''
- // 更新selectedDevices数组
- nextTick(() => {
- updateSelectedDevices()
- })
- })
- dialogVisible.value = true
- }
- //删除成套
- const handleDelete = async (id: number) => {
- ElMessageBox.confirm('确定要删除该成套设备吗?', '提示', {
- confirmButtonText: '确定',
- cancelButtonText: '取消',
- type: 'warning'
- })
- .then(async () => {
- try {
- await IotDeviceApi.deleteIotDeviceSet(id)
- ElMessage.success('删除成功')
- getList()
- } catch (error) {
- console.error(error)
- }
- })
- .catch(() => {
- // 取消操作
- })
- }
- // 重置表单
- const resetForm = () => {
- formData.value = {
- name: '',
- details: [],
- deptId: '',
- remark: ''
- }
- selectedDeviceIds.value = []
- selectedDevices.value = []
- deviceOptions.value = []
- mainDeviceId.value = ''
- }
- // 取消操作
- const cancel = () => {
- dialogVisible.value = false
- resetForm()
- }
- // 提交表单
- const submit = async () => {
- if (!formRef.value) return
- await formRef.value.validate(async (valid) => {
- if (!valid) return
- try {
- const data = {
- ...formData.value
- }
- console.log('提交数据:', data)
- if (isEdit.value) {
- await IotDeviceApi.updateIotDeviceSet(data)
- ElMessage.success('编辑成功')
- } else {
- await IotDeviceApi.createIotDeviceSet(data)
- ElMessage.success('新增成功')
- }
- dialogVisible.value = false
- getList()
- } catch (error) {
- console.error(error)
- }
- })
- }
- onMounted(async () => {
- getList()
- deptList.value = handleTree(await DeptApi.getSimpleDeptList())
- })
- </script>
- <style scoped>
- .transfer-container {
- display: flex;
- flex-direction: column;
- }
- .transfer-footer {
- margin-top: 8px;
- text-align: right;
- }
- ::deep(.el-transfer-panel) {
- width: 300px;
- }
- ::deep(.el-tree--highlight-current) {
- height: 200px !important;
- }
- /* 调整穿梭框面板主体高度 */
- ::deep(.el-transfer-panel__body) {
- height: 850px !important;
- }
- /* 如果需要进一步调整,可以分别设置左右面板的高度 */
- ::deep(.el-transfer-panel .el-transfer-panel__list) {
- height: 850px !important; /* 调整列表区域高度 */
- }
- .el-transfer {
- --el-transfer-panel-body-height: 500px;
- }
- </style>
|