feat(mes): 添加安灯呼叫记录创建和更新请求 VO

pull/871/MERGE
YunaiV 2026-03-17 21:15:47 +08:00
parent 8774ea3dad
commit ea35423d9e
5 changed files with 193 additions and 184 deletions

View File

@ -45,9 +45,9 @@ export const ProAndonRecordApi = {
return await request.delete({ url: `/mes/pro/andon-record/delete?id=` + id }) return await request.delete({ url: `/mes/pro/andon-record/delete?id=` + id })
}, },
// 处置安灯记录 // 更新安灯记录(保存/已处置)
handleAndonRecord: async (data: any) => { updateAndonRecord: async (data: any) => {
return await request.put({ url: `/mes/pro/andon-record/handle`, data }) return await request.put({ url: `/mes/pro/andon-record/update`, data })
}, },
// 导出安灯记录 Excel // 导出安灯记录 Excel

View File

@ -1,3 +1,4 @@
<!-- TODO @AI应该放到 md -->
<!-- MES 编码规则列表 --> <!-- MES 编码规则列表 -->
<template> <template>
<ContentWrap> <ContentWrap>

View File

@ -1,117 +0,0 @@
<!-- 安灯呼叫记录 处置弹窗 -->
<template>
<Dialog title="处置安灯呼叫" v-model="dialogVisible" width="600px">
<el-form
ref="formRef"
:model="formData"
:rules="formRules"
label-width="100px"
v-loading="formLoading"
>
<!-- 只读展示呼叫信息 -->
<el-form-item label="工作站">
<el-input :model-value="recordInfo.workstationName" disabled />
</el-form-item>
<el-form-item label="发起人">
<el-input :model-value="recordInfo.userNickname" disabled />
</el-form-item>
<el-form-item label="呼叫原因">
<el-input :model-value="recordInfo.reason" disabled />
</el-form-item>
<el-form-item label="级别">
<dict-tag :type="DICT_TYPE.MES_PRO_ANDON_LEVEL" :value="recordInfo.level" />
</el-form-item>
<!-- 可编辑字段 -->
<el-form-item label="处置时间" prop="handleTime">
<el-date-picker
v-model="formData.handleTime"
type="datetime"
value-format="YYYY-MM-DD HH:mm:ss"
placeholder="请选择处置时间"
class="!w-full"
/>
</el-form-item>
<el-form-item label="处置人" prop="handlerUserId">
<UserSelect v-model="formData.handlerUserId" />
</el-form-item>
<el-form-item label="备注" prop="remark">
<el-input v-model="formData.remark" type="textarea" placeholder="请输入处置备注" />
</el-form-item>
</el-form>
<template #footer>
<el-button @click="submitForm" type="primary" :disabled="formLoading"> </el-button>
<el-button @click="dialogVisible = false"> </el-button>
</template>
</Dialog>
</template>
<script setup lang="ts">
import { ProAndonRecordApi } from '@/api/mes/pro/andon/record'
import { DICT_TYPE } from '@/utils/dict'
import { useUserStoreWithOut } from '@/store/modules/user'
import UserSelect from '@/views/system/user/components/UserSelect.vue'
import { formatDate } from '@/utils/formatTime'
defineOptions({ name: 'AndonHandleForm' })
const message = useMessage()
const dialogVisible = ref(false) //
const formLoading = ref(false) //
const formData = ref<any>({}) //
const recordInfo = ref<any>({}) //
const formRules = reactive({
handleTime: [{ required: true, message: '处置时间不能为空', trigger: 'change' }],
handlerUserId: [{ required: true, message: '处置人不能为空', trigger: 'change' }]
})
const formRef = ref() // Ref
/** 打开弹窗 */
const open = async (id: number) => {
dialogVisible.value = true
resetForm()
formLoading.value = true
try {
//
recordInfo.value = await ProAndonRecordApi.getAndonRecord(id)
//
const userStore = useUserStoreWithOut()
formData.value = {
id: id,
handleTime: formatDate(new Date(), 'YYYY-MM-DD HH:mm:ss'),
handlerUserId: userStore.getUser?.id,
remark: undefined
}
} finally {
formLoading.value = false
}
}
defineExpose({ open })
/** 提交处置 */
const emit = defineEmits(['success'])
const submitForm = async () => {
await formRef.value.validate()
formLoading.value = true
try {
await ProAndonRecordApi.handleAndonRecord(formData.value)
message.success('处置成功')
dialogVisible.value = false
emit('success')
} finally {
formLoading.value = false
}
}
/** 重置表单 */
const resetForm = () => {
formData.value = {
id: undefined,
handleTime: undefined,
handlerUserId: undefined,
remark: undefined
}
recordInfo.value = {}
formRef.value?.resetFields()
}
</script>

View File

@ -1,56 +1,114 @@
<!-- 安灯呼叫记录 新增/详情弹窗 --> <!-- 安灯呼叫记录 新增/处置/详情弹窗 -->
<template> <template>
<Dialog :title="dialogTitle" v-model="dialogVisible" width="600px"> <Dialog :title="dialogTitle" v-model="dialogVisible" width="600px">
<el-form <el-form
ref="formRef" ref="formRef"
:model="formData" :model="formData"
:rules="formRules" :rules="formType === 'create' ? createRules : undefined"
label-width="100px" label-width="100px"
:disabled="formType === 'detail'"
v-loading="formLoading" v-loading="formLoading"
> >
<!-- ====== 呼叫信息create 可编辑update/detail 只读 ====== -->
<el-form-item label="工作站" prop="workstationId"> <el-form-item label="工作站" prop="workstationId">
<MdWorkstationSelect v-model="formData.workstationId" placeholder="请选择工作站" /> <MdWorkstationSelect
v-if="formType === 'create'"
v-model="formData.workstationId"
placeholder="请选择工作站"
/>
<el-input v-else :model-value="formData.workstationName" disabled />
</el-form-item> </el-form-item>
<el-form-item label="发起人" prop="userId"> <el-form-item label="发起人" prop="userId">
<UserSelect v-model="formData.userId" /> <UserSelect v-if="formType === 'create'" v-model="formData.userId" />
<el-input v-else :model-value="formData.userNickname" disabled />
</el-form-item> </el-form-item>
<el-form-item label="生产工单" prop="workOrderId"> <el-form-item label="生产工单" prop="workOrderId">
<ProWorkOrderSelect v-model="formData.workOrderId" placeholder="请选择工单(可选)" /> <ProWorkOrderSelect
v-if="formType === 'create'"
v-model="formData.workOrderId"
placeholder="请选择工单(可选)"
/>
<el-input v-else :model-value="formData.workOrderCode || '-'" disabled />
</el-form-item> </el-form-item>
<el-form-item label="工序" prop="processId"> <el-form-item label="工序" prop="processId">
<ProProcessSelect v-model="formData.processId" placeholder="请选择工序(可选)" /> <ProProcessSelect
v-if="formType === 'create'"
v-model="formData.processId"
placeholder="请选择工序(可选)"
/>
<el-input v-else :model-value="formData.processName || '-'" disabled />
</el-form-item> </el-form-item>
<el-form-item label="呼叫原因" prop="configId"> <el-form-item label="呼叫原因" prop="configId">
<AndonConfigSelect v-model="formData.configId" @change="handleConfigChange" /> <AndonConfigSelect
</el-form-item> v-if="formType === 'create'"
<el-form-item label="级别" prop="level"> v-model="formData.configId"
<dict-tag @change="handleConfigChange"
v-if="formData.level"
:type="DICT_TYPE.MES_PRO_ANDON_LEVEL"
:value="formData.level"
/> />
<span v-else class="text-gray-400">由呼叫原因自动带出</span> <el-input v-else :model-value="formData.reason" disabled />
</el-form-item> </el-form-item>
<el-form-item label="备注" prop="remark"> <el-form-item label="级别">
<el-input v-model="formData.remark" type="textarea" placeholder="请输入备注" /> <template v-if="formType === 'create'">
<dict-tag
v-if="formData.level"
:type="DICT_TYPE.MES_PRO_ANDON_LEVEL"
:value="formData.level"
/>
<span v-else class="text-gray-400">由呼叫原因自动带出</span>
</template>
<dict-tag v-else :type="DICT_TYPE.MES_PRO_ANDON_LEVEL" :value="formData.level" />
</el-form-item> </el-form-item>
<!-- 详情模式额外展示 -->
<template v-if="formType === 'detail'"> <!-- ====== 处置信息update 可编辑detail 只读create 不展示 ====== -->
<template v-if="formType !== 'create'">
<el-divider content-position="left">处置信息</el-divider>
<el-form-item label="状态"> <el-form-item label="状态">
<dict-tag :type="DICT_TYPE.MES_PRO_ANDON_STATUS" :value="formData.status" /> <dict-tag :type="DICT_TYPE.MES_PRO_ANDON_STATUS" :value="formData.status" />
</el-form-item> </el-form-item>
<el-form-item label="处置时间" v-if="formData.handleTime"> <el-form-item label="处置时间">
<el-input :model-value="formData.handleTime" disabled /> <el-date-picker
v-if="formType === 'update'"
v-model="formData.handleTime"
type="datetime"
value-format="YYYY-MM-DD HH:mm:ss"
placeholder="请选择处置时间"
class="!w-full"
/>
<el-input v-else :model-value="formData.handleTime || '-'" disabled />
</el-form-item> </el-form-item>
<el-form-item label="处置人" v-if="formData.handlerUserNickname"> <el-form-item label="处置人">
<el-input :model-value="formData.handlerUserNickname" disabled /> <UserSelect v-if="formType === 'update'" v-model="formData.handlerUserId" />
<el-input v-else :model-value="formData.handlerUserNickname || '-'" disabled />
</el-form-item> </el-form-item>
</template> </template>
<!-- ====== 备注create/update 可编辑detail 只读 ====== -->
<el-form-item label="备注">
<el-input
v-if="formType !== 'detail'"
v-model="formData.remark"
type="textarea"
placeholder="请输入备注"
/>
<span v-else>{{ formData.remark || '-' }}</span>
</el-form-item>
</el-form> </el-form>
<template #footer v-if="formType !== 'detail'">
<el-button @click="submitForm" type="primary" :disabled="formLoading"> </el-button> <!-- ====== 底部按钮 ====== -->
<el-button @click="dialogVisible = false"> </el-button> <template #footer>
<!-- create 模式 -->
<template v-if="formType === 'create'">
<el-button @click="dialogVisible = false"> </el-button>
<el-button type="primary" @click="handleCreate" :disabled="formLoading"> </el-button>
</template>
<!-- update 模式 -->
<template v-else-if="formType === 'update'">
<el-button @click="dialogVisible = false"> </el-button>
<el-button type="primary" @click="handleSave" :disabled="formLoading"> </el-button>
<el-button type="success" @click="handleFinish" :disabled="formLoading">已处置</el-button>
</template>
<!-- detail 模式 -->
<template v-else>
<el-button @click="dialogVisible = false"> </el-button>
</template>
</template> </template>
</Dialog> </Dialog>
</template> </template>
@ -59,51 +117,67 @@
import { ProAndonRecordApi } from '@/api/mes/pro/andon/record' import { ProAndonRecordApi } from '@/api/mes/pro/andon/record'
import { ProAndonConfigVO } from '@/api/mes/pro/andon/config' import { ProAndonConfigVO } from '@/api/mes/pro/andon/config'
import { DICT_TYPE } from '@/utils/dict' import { DICT_TYPE } from '@/utils/dict'
import { useUserStoreWithOut } from '@/store/modules/user'
import { formatDate } from '@/utils/formatTime'
import { MesProAndonStatusEnum } from '@/views/mes/utils/constants'
import MdWorkstationSelect from '@/views/mes/md/workstation/components/MdWorkstationSelect.vue' import MdWorkstationSelect from '@/views/mes/md/workstation/components/MdWorkstationSelect.vue'
import ProWorkOrderSelect from '@/views/mes/pro/workorder/components/ProWorkOrderSelect.vue' import ProWorkOrderSelect from '@/views/mes/pro/workorder/components/ProWorkOrderSelect.vue'
import ProProcessSelect from '@/views/mes/pro/process/components/ProProcessSelect.vue' import ProProcessSelect from '@/views/mes/pro/process/components/ProProcessSelect.vue'
import UserSelect from '@/views/system/user/components/UserSelect.vue' import UserSelect from '@/views/system/user/components/UserSelect.vue'
import AndonConfigSelect from '../config/components/AndonConfigSelect.vue' import AndonConfigSelect from '../config/components/AndonConfigSelect.vue'
import { useUserStoreWithOut } from '@/store/modules/user'
/** 安灯呼叫记录 新增/详情弹窗 */
defineOptions({ name: 'AndonRecordForm' }) defineOptions({ name: 'AndonRecordForm' })
const { t } = useI18n() const { t } = useI18n() //
const message = useMessage() const message = useMessage() //
const dialogVisible = ref(false) // const dialogVisible = ref(false) //
const dialogTitle = ref('') //
const formLoading = ref(false) // 12 const formLoading = ref(false) // 12
const formType = ref('') // create - detail - const formType = ref('') // create - update - detail -
const dialogTitle = computed(() =>
//
formType.value === 'create'
? '新增安灯呼叫'
: formType.value === 'update'
? '处置安灯呼叫'
: '安灯呼叫详情'
)
const formData = ref<any>({}) // const formData = ref<any>({}) //
const formRules = reactive({
workstationId: [{ required: true, message: '工作站不能为空', trigger: 'change' }],
configId: [{ required: true, message: '呼叫原因不能为空', trigger: 'change' }],
level: [{ required: true, message: '级别不能为空', trigger: 'change' }]
})
const formRef = ref() // Ref const formRef = ref() // Ref
const createRules = reactive({
workstationId: [{ required: true, message: '工作站不能为空', trigger: 'change' }],
configId: [{ required: true, message: '呼叫原因不能为空', trigger: 'change' }]
})
const userStore = useUserStoreWithOut()
/** 打开弹窗 */ /** 打开弹窗 */
const open = async (type: string, id?: number) => { const open = async (type: string, id?: number) => {
dialogVisible.value = true dialogVisible.value = true
dialogTitle.value = type === 'create' ? '新增安灯呼叫' : '安灯呼叫详情'
formType.value = type formType.value = type
resetForm() resetForm()
if (type === 'create') { if (type === 'create') {
// //
const userStore = useUserStoreWithOut()
formData.value.userId = userStore.getUser?.id formData.value.userId = userStore.getUser?.id
} else if (type === 'detail') { } else {
// /
formLoading.value = true formLoading.value = true
try { try {
formData.value = await ProAndonRecordApi.getAndonRecord(id!) formData.value = await ProAndonRecordApi.getAndonRecord(id!)
//
if (type === 'update') {
if (!formData.value.handleTime) {
formData.value.handleTime = formatDate(new Date())
}
if (!formData.value.handlerUserId) {
formData.value.handlerUserId = userStore.getUser?.id
}
}
} finally { } finally {
formLoading.value = false formLoading.value = false
} }
} }
} }
defineExpose({ open }) defineExpose({ open }) // open
/** 选择配置后自动填充原因和级别 */ /** 选择配置后自动填充原因和级别 */
const handleConfigChange = (config: ProAndonConfigVO | undefined) => { const handleConfigChange = (config: ProAndonConfigVO | undefined) => {
@ -116,15 +190,69 @@ const handleConfigChange = (config: ProAndonConfigVO | undefined) => {
} }
} }
/** 提交表单 */ /** 提交新增 */
const emit = defineEmits(['success']) const emit = defineEmits(['success']) // success
const submitForm = async () => { const handleCreate = async () => {
//
await formRef.value.validate() await formRef.value.validate()
//
formLoading.value = true formLoading.value = true
try { try {
await ProAndonRecordApi.createAndonRecord(formData.value) await ProAndonRecordApi.createAndonRecord(formData.value)
message.success(t('common.createSuccess')) message.success(t('common.createSuccess'))
dialogVisible.value = false dialogVisible.value = false
//
emit('success')
} finally {
formLoading.value = false
}
}
/** 保存处置信息(保持 ACTIVE 状态) */
const handleSave = async () => {
//
formLoading.value = true
try {
await ProAndonRecordApi.updateAndonRecord({
id: formData.value.id,
handleTime: formData.value.handleTime,
handlerUserId: formData.value.handlerUserId,
remark: formData.value.remark,
status: MesProAndonStatusEnum.ACTIVE
})
message.success('保存成功')
dialogVisible.value = false
//
emit('success')
} finally {
formLoading.value = false
}
}
/** 标记已处置(状态变更为 HANDLED */
const handleFinish = async () => {
//
if (!formData.value.handleTime) {
message.warning('标记已处置时,处置时间不能为空')
return
}
if (!formData.value.handlerUserId) {
message.warning('标记已处置时,处置人不能为空')
return
}
//
formLoading.value = true
try {
await ProAndonRecordApi.updateAndonRecord({
id: formData.value.id,
handleTime: formData.value.handleTime,
handlerUserId: formData.value.handlerUserId,
remark: formData.value.remark,
status: MesProAndonStatusEnum.HANDLED
})
message.success('处置成功')
dialogVisible.value = false
//
emit('success') emit('success')
} finally { } finally {
formLoading.value = false formLoading.value = false

View File

@ -115,7 +115,7 @@
v-if="scope.row.status === MesProAndonStatusEnum.ACTIVE" v-if="scope.row.status === MesProAndonStatusEnum.ACTIVE"
link link
type="success" type="success"
@click="openHandleForm(scope.row.id)" @click="openRecordForm('update', scope.row.id)"
v-hasPermi="['mes:pro-andon-record:update']" v-hasPermi="['mes:pro-andon-record:update']"
> >
处置 处置
@ -148,10 +148,8 @@
/> />
</ContentWrap> </ContentWrap>
<!-- 弹窗新增/详情 --> <!-- 弹窗新增/处置/详情 -->
<AndonRecordForm ref="recordFormRef" @success="getList" /> <AndonRecordForm ref="recordFormRef" @success="getList" />
<!-- 弹窗处置 -->
<AndonHandleForm ref="handleFormRef" @success="getList" />
<!-- 弹窗安灯设置 --> <!-- 弹窗安灯设置 -->
<AndonConfigDialog ref="configDialogRef" /> <AndonConfigDialog ref="configDialogRef" />
</template> </template>
@ -161,7 +159,6 @@ import { dateFormatter } from '@/utils/formatTime'
import download from '@/utils/download' import download from '@/utils/download'
import { ProAndonRecordApi, ProAndonRecordVO } from '@/api/mes/pro/andon/record' import { ProAndonRecordApi, ProAndonRecordVO } from '@/api/mes/pro/andon/record'
import AndonRecordForm from './AndonRecordForm.vue' import AndonRecordForm from './AndonRecordForm.vue'
import AndonHandleForm from './AndonHandleForm.vue'
import AndonConfigDialog from '../config/AndonConfigForm.vue' import AndonConfigDialog from '../config/AndonConfigForm.vue'
import { DICT_TYPE, getIntDictOptions } from '@/utils/dict' import { DICT_TYPE, getIntDictOptions } from '@/utils/dict'
import { MesProAndonStatusEnum } from '@/views/mes/utils/constants' import { MesProAndonStatusEnum } from '@/views/mes/utils/constants'
@ -170,12 +167,12 @@ import UserSelect from '@/views/system/user/components/UserSelect.vue'
defineOptions({ name: 'MesProAndon' }) defineOptions({ name: 'MesProAndon' })
const message = useMessage() const message = useMessage() //
const { t } = useI18n() const { t } = useI18n() //
const loading = ref(true) // const loading = ref(true) //
const list = ref<ProAndonRecordVO[]>([]) //
const total = ref(0) // const total = ref(0) //
const list = ref<ProAndonRecordVO[]>([]) //
const queryParams = reactive({ const queryParams = reactive({
pageNo: 1, pageNo: 1,
pageSize: 10, pageSize: 10,
@ -185,7 +182,7 @@ const queryParams = reactive({
status: undefined, status: undefined,
createTime: undefined createTime: undefined
}) })
const queryFormRef = ref() // Ref const queryFormRef = ref() //
const exportLoading = ref(false) // const exportLoading = ref(false) //
/** 查询列表 */ /** 查询列表 */
@ -212,18 +209,12 @@ const resetQuery = () => {
handleQuery() handleQuery()
} }
/** 新增/详情弹窗 */ /** 新增/处置/详情操作 */
const recordFormRef = ref() const recordFormRef = ref()
const openRecordForm = (type: string, id?: number) => { const openRecordForm = (type: string, id?: number) => {
recordFormRef.value.open(type, id) recordFormRef.value.open(type, id)
} }
/** 处置弹窗 */
const handleFormRef = ref()
const openHandleForm = (id: number) => {
handleFormRef.value.open(id)
}
/** 安灯设置弹窗 */ /** 安灯设置弹窗 */
const configDialogRef = ref() const configDialogRef = ref()
const openConfigDialog = () => { const openConfigDialog = () => {
@ -233,9 +224,12 @@ const openConfigDialog = () => {
/** 删除按钮操作 */ /** 删除按钮操作 */
const handleDelete = async (id: number) => { const handleDelete = async (id: number) => {
try { try {
//
await message.delConfirm() await message.delConfirm()
//
await ProAndonRecordApi.deleteAndonRecord(id) await ProAndonRecordApi.deleteAndonRecord(id)
message.success(t('common.delSuccess')) message.success(t('common.delSuccess'))
//
await getList() await getList()
} catch {} } catch {}
} }
@ -243,17 +237,20 @@ const handleDelete = async (id: number) => {
/** 导出按钮操作 */ /** 导出按钮操作 */
const handleExport = async () => { const handleExport = async () => {
try { try {
//
await message.exportConfirm() await message.exportConfirm()
//
exportLoading.value = true exportLoading.value = true
const data = await ProAndonRecordApi.exportAndonRecord(queryParams) const data = await ProAndonRecordApi.exportAndonRecord(queryParams)
download.excel(data, '安灯呼叫记录.xls') download.excel(data, '安灯呼叫记录.xls')
} catch {
} finally { } finally {
exportLoading.value = false exportLoading.value = false
} }
} }
/** 初始化 **/ /** 初始化 */
onMounted(async () => { onMounted(() => {
await getList() getList()
}) })
</script> </script>