feat(mes): 更新维修工单表单逻辑及状态管理

调整维修工单表单的字段展示逻辑,优化完成维修和验收的操作流程。新增维修完成日期和维修结果的必填校验,确保用户在提交时提供必要信息。同时,修正了表单的可编辑状态,提升用户体验。
pull/871/MERGE
YunaiV 2026-04-04 01:17:06 +08:00
parent da1dfa410b
commit 307bac15e1
3 changed files with 104 additions and 86 deletions

View File

@ -50,28 +50,18 @@
</el-form-item> </el-form-item>
</el-col> </el-col>
<el-col :span="8" v-if="showFinishFields"> <el-col :span="8" v-if="showFinishFields">
<!-- TODO @AI默认把 required 加进去放到 rules --> <el-form-item label="维修完成日期" prop="finishDate">
<!-- TODO @AI维修完成时需要允许修改 -->
<el-form-item
label="维修完成日期"
prop="finishDate"
:rules="
formType === 'confirm'
? [{ required: true, message: '维修完成日期不能为空', trigger: 'change' }]
: []
"
>
<!-- TODO @AIisConfirm 这样 -->
<el-date-picker <el-date-picker
v-model="formData.finishDate" v-model="formData.finishDate"
type="datetime" type="datetime"
value-format="x" value-format="x"
placeholder="选择完成日期" placeholder="选择完成日期"
:disabled="formType !== 'confirm'" :disabled="!isConfirm"
/> />
</el-form-item> </el-form-item>
</el-col> </el-col>
<el-col :span="8" v-if="showConfirmFields"> <!-- DONE @AI验收日期由后端自动设置仅详情时展示 -->
<el-col :span="8" v-if="showDetailFields">
<el-form-item label="验收日期" prop="confirmDate"> <el-form-item label="验收日期" prop="confirmDate">
<el-date-picker <el-date-picker
v-model="formData.confirmDate" v-model="formData.confirmDate"
@ -84,9 +74,15 @@
</el-col> </el-col>
</el-row> </el-row>
<el-row> <el-row>
<el-col :span="8" v-if="showFinishFields"> <!-- DONE @AI维修结果在验收(finish)时可选择详情(detail)时只读展示 -->
<el-col :span="8" v-if="isFinish || showDetailFields">
<el-form-item label="维修结果" prop="result"> <el-form-item label="维修结果" prop="result">
<el-select v-model="formData.result" placeholder="请选择维修结果" clearable disabled> <el-select
v-model="formData.result"
placeholder="请选择维修结果"
clearable
:disabled="!isFinish"
>
<el-option <el-option
v-for="dict in getIntDictOptions(DICT_TYPE.MES_DV_REPAIR_RESULT)" v-for="dict in getIntDictOptions(DICT_TYPE.MES_DV_REPAIR_RESULT)"
:key="dict.value" :key="dict.value"
@ -96,12 +92,12 @@
</el-select> </el-select>
</el-form-item> </el-form-item>
</el-col> </el-col>
<el-col :span="8" v-if="showFinishFields"> <el-col :span="8" v-if="showConfirmFields">
<el-form-item label="维修人" prop="acceptedUserId"> <el-form-item label="维修人" prop="acceptedUserId">
<UserSelect v-model="formData.acceptedUserId" placeholder="请选择维修人" disabled /> <UserSelect v-model="formData.acceptedUserId" placeholder="请选择维修人" disabled />
</el-form-item> </el-form-item>
</el-col> </el-col>
<el-col :span="8" v-if="showConfirmFields"> <el-col :span="8" v-if="showDetailFields">
<el-form-item label="验收人" prop="confirmUserId"> <el-form-item label="验收人" prop="confirmUserId">
<UserSelect v-model="formData.confirmUserId" placeholder="请选择验收人" disabled /> <UserSelect v-model="formData.confirmUserId" placeholder="请选择验收人" disabled />
</el-form-item> </el-form-item>
@ -130,23 +126,18 @@
</el-button> </el-button>
<el-button <el-button
v-if="formType === 'update'" v-if="isEditable && formData.status === MesDvRepairStatusEnum.PREPARE"
@click="handleSubmit" @click="handleSubmit"
type="success" type="warning"
:disabled="formLoading" :disabled="formLoading"
> >
</el-button> </el-button>
<el-button <el-button v-if="isConfirm" @click="handleConfirm" type="primary" :disabled="formLoading">
v-if="formType === 'confirm'"
@click="handleConfirm"
type="primary"
:disabled="formLoading"
>
完成维修 完成维修
</el-button> </el-button>
<el-button <el-button
v-if="formType === 'finish'" v-if="isFinish"
@click="handleFinish(MesDvRepairResultEnum.PASS)" @click="handleFinish(MesDvRepairResultEnum.PASS)"
type="success" type="success"
:disabled="formLoading" :disabled="formLoading"
@ -154,7 +145,7 @@
</el-button> </el-button>
<el-button <el-button
v-if="formType === 'finish'" v-if="isFinish"
@click="handleFinish(MesDvRepairResultEnum.FAIL)" @click="handleFinish(MesDvRepairResultEnum.FAIL)"
type="warning" type="warning"
:disabled="formLoading" :disabled="formLoading"
@ -187,8 +178,11 @@ const dialogVisible = ref(false) // 弹窗的是否展示
const formLoading = ref(false) // const formLoading = ref(false) //
const formType = ref<string>('create') // 表单类型create / update / confirm / finish / detail const formType = ref<string>('create') // 表单类型create / update / confirm / finish / detail
const isEditable = computed(() => ['create', 'update'].includes(formType.value)) // const isEditable = computed(() => ['create', 'update'].includes(formType.value)) //
const isDetail = computed(() => !isEditable.value) // confirm/finish/detail const isConfirm = computed(() => formType.value === 'confirm') //
const isHeaderReadonly = computed(() => !isEditable.value) // const isFinish = computed(() => formType.value === 'finish') //
// DONE @AIfinish isDetail
const isDetail = computed(() => formType.value === 'detail') // detail
const isHeaderReadonly = computed(() => ['confirm', 'finish', 'detail'].includes(formType.value)) //
const showFinishFields = computed(() => { const showFinishFields = computed(() => {
const status = formData.value.status const status = formData.value.status
if (status == null) { if (status == null) {
@ -203,6 +197,13 @@ const showConfirmFields = computed(() => {
} }
return status >= MesDvRepairStatusEnum.APPROVING return status >= MesDvRepairStatusEnum.APPROVING
}) // (2) }) // (2)
const showDetailFields = computed(() => {
const status = formData.value.status
if (status == null) {
return false
}
return status >= MesDvRepairStatusEnum.FINISHED
}) // (4)
const dialogTitle = computed(() => { const dialogTitle = computed(() => {
const titles: Record<string, string> = { const titles: Record<string, string> = {
create: '新增维修工单', create: '新增维修工单',
@ -231,9 +232,12 @@ const formRules = reactive({
code: [{ required: true, message: '维修单编码不能为空', trigger: 'blur' }], code: [{ required: true, message: '维修单编码不能为空', trigger: 'blur' }],
name: [{ required: true, message: '维修单名称不能为空', trigger: 'blur' }], name: [{ required: true, message: '维修单名称不能为空', trigger: 'blur' }],
machineryId: [{ required: true, message: '设备不能为空', trigger: 'blur' }], machineryId: [{ required: true, message: '设备不能为空', trigger: 'blur' }],
requireDate: [{ required: true, message: '报修日期不能为空', trigger: 'blur' }] requireDate: [{ required: true, message: '报修日期不能为空', trigger: 'blur' }],
finishDate: [{ required: true, message: '维修完成日期不能为空', trigger: 'change' }],
result: [{ required: true, message: '维修结果不能为空', trigger: 'change' }]
}) })
const formRef = ref() // Ref const formRef = ref() // Ref
const originalFormData = ref<string>('') //
/** 生成维修单编码 */ /** 生成维修单编码 */
const generateCode = async () => { const generateCode = async () => {
@ -254,34 +258,51 @@ const open = async (type: string, id?: number) => {
formLoading.value = false formLoading.value = false
} }
} }
//
originalFormData.value = JSON.stringify(formData.value)
} }
/** 提交表单create/update 模式) */ /** 提交表单create/update 模式) */
const submitForm = async () => { const submitForm = async () => {
//
await formRef.value.validate() await formRef.value.validate()
//
formLoading.value = true formLoading.value = true
try { try {
const data = formData.value as any const data = formData.value as any
if (formType.value === 'create') { if (formType.value === 'create') {
await DvRepairApi.createRepair(data) const res = await DvRepairApi.createRepair(data)
message.success('新增成功') message.success('新增成功')
//
formData.value.id = res
formData.value.status = MesDvRepairStatusEnum.PREPARE
formType.value = 'update'
} else { } else {
await DvRepairApi.updateRepair(data) await DvRepairApi.updateRepair(data)
message.success('修改成功') message.success('修改成功')
} }
dialogVisible.value = false //
originalFormData.value = JSON.stringify(formData.value)
//
emit('success') emit('success')
} finally { } finally {
formLoading.value = false formLoading.value = false
} }
} }
/** 提交维修工单(草稿→维修中) */ /** 提交操作:表单修改过则先保存,再提交(草稿→维修中) */
const handleSubmit = async () => { const handleSubmit = async () => {
//
await formRef.value.validate()
try { try {
// TODO @AI /Users/yunai/Java/yudao-all-in-one/yudao-ui-admin-vue3/src/views/mes/dv/maintenrecord/MaintenRecordForm.vue await message.confirm('确认提交该维修工单?【提交后将不能修改】')
await message.confirm('确认提交该维修工单?提交后将进入维修中状态')
formLoading.value = true formLoading.value = true
// 1.
if (JSON.stringify(formData.value) !== originalFormData.value) {
const data = formData.value as any
await DvRepairApi.updateRepair(data)
}
// 2.
await DvRepairApi.submitRepair(formData.value.id!) await DvRepairApi.submitRepair(formData.value.id!)
message.success('提交成功') message.success('提交成功')
dialogVisible.value = false dialogVisible.value = false

View File

@ -2,11 +2,9 @@
<template> <template>
<div class="overflow-hidden"> <div class="overflow-hidden">
<!-- 操作栏 --> <!-- 操作栏 -->
<el-row class="mb-10px" v-if="!disabled"> <el-button v-if="!disabled" type="primary" plain @click="openForm('create')" class="mb-10px">
<el-button type="primary" plain @click="openForm('create')"> <Icon icon="ep:plus" class="mr-5px" /> 添加明细
<Icon icon="ep:plus" class="mr-5px" /> 添加明细 </el-button>
</el-button>
</el-row>
<!-- 列表 --> <!-- 列表 -->
<el-table v-loading="loading" :data="list" :stripe="true" :show-overflow-tooltip="true"> <el-table v-loading="loading" :data="list" :stripe="true" :show-overflow-tooltip="true">
<el-table-column label="项目名称" align="center" prop="subjectName" /> <el-table-column label="项目名称" align="center" prop="subjectName" />
@ -22,7 +20,6 @@
</template> </template>
</el-table-column> </el-table-column>
</el-table> </el-table>
<Pagination <Pagination
:total="total" :total="total"
v-model:page="queryParams.pageNo" v-model:page="queryParams.pageNo"
@ -31,7 +28,7 @@
/> />
<!-- 表单弹窗添加/修改 --> <!-- 表单弹窗添加/修改 -->
<Dialog :title="formTitle" v-model="formVisible" width="500px"> <Dialog :title="dialogTitle" v-model="dialogVisible" width="500px">
<el-form ref="formRef" :model="formData" :rules="formRules" label-width="80px"> <el-form ref="formRef" :model="formData" :rules="formRules" label-width="80px">
<el-form-item label="项目" prop="subjectId"> <el-form-item label="项目" prop="subjectId">
<el-select <el-select
@ -64,8 +61,8 @@
</el-form-item> </el-form-item>
</el-form> </el-form>
<template #footer> <template #footer>
<el-button @click="formVisible = false"> </el-button> <el-button @click="submitForm" type="primary" :loading="formLoading"> </el-button>
<el-button type="primary" @click="submitForm" :loading="formLoading"> </el-button> <el-button @click="dialogVisible = false"> </el-button>
</template> </template>
</Dialog> </Dialog>
</div> </div>
@ -82,9 +79,10 @@ const props = defineProps<{
disabled?: boolean disabled?: boolean
}>() }>()
const message = useMessage() //
const { t } = useI18n() // const { t } = useI18n() //
const message = useMessage() //
// ==================== ====================
const loading = ref(false) // const loading = ref(false) //
const list = ref([]) // const list = ref([]) //
const total = ref(0) // const total = ref(0) //
@ -94,18 +92,6 @@ const queryParams = reactive({
repairId: props.repairId repairId: props.repairId
}) })
//
const formVisible = ref(false) //
const formTitle = ref('') //
const formLoading = ref(false) //
const formType = ref('') // create - update -
const formRef = ref() // Ref
const formData = ref<any>({}) //
const formRules = reactive({
malfunction: [{ required: true, message: '故障描述不能为空', trigger: 'blur' }]
})
const subjectOptions = ref<any[]>([]) //
/** 查询列表 */ /** 查询列表 */
const getList = async () => { const getList = async () => {
loading.value = true loading.value = true
@ -118,11 +104,41 @@ const getList = async () => {
} }
} }
/** 添加/修改操作 */ /** 删除按钮操作 */
const handleDelete = async (id: number) => {
try {
await message.delConfirm()
await DvRepairLineApi.deleteRepairLine(id)
message.success(t('common.delSuccess'))
await getList()
} catch {}
}
// ==================== / ====================
const dialogVisible = ref(false) //
const dialogTitle = ref('') //
const formLoading = ref(false) //
const lineFormType = ref('') // 行表单的类型create / update
const subjectOptions = ref<any[]>([]) //
const formData = ref<any>({}) //
const formRules = reactive({
malfunction: [{ required: true, message: '故障描述不能为空', trigger: 'blur' }]
})
const formRef = ref() // Ref
/** 获取项目选项 */
const getSubjectOptions = async (query: string) => {
try {
const data = await DvSubjectApi.getSubjectPage({ name: query, pageNo: 1, pageSize: 20 })
subjectOptions.value = data.list
} catch {}
}
/** 打开表单弹窗 */
const openForm = async (type: string, row?: any) => { const openForm = async (type: string, row?: any) => {
formVisible.value = true dialogVisible.value = true
formTitle.value = type === 'create' ? '添加明细' : '编辑明细' dialogTitle.value = type === 'create' ? '添加明细' : '编辑明细'
formType.value = type lineFormType.value = type
if (type === 'create') { if (type === 'create') {
formData.value = { formData.value = {
repairId: props.repairId, repairId: props.repairId,
@ -144,42 +160,23 @@ const openForm = async (type: string, row?: any) => {
/** 提交表单 */ /** 提交表单 */
const submitForm = async () => { const submitForm = async () => {
const valid = await formRef.value.validate() await formRef.value.validate()
if (!valid) return
formLoading.value = true formLoading.value = true
try { try {
if (formType.value === 'create') { if (lineFormType.value === 'create') {
await DvRepairLineApi.createRepairLine(formData.value) await DvRepairLineApi.createRepairLine(formData.value)
message.success(t('common.createSuccess')) message.success(t('common.createSuccess'))
} else { } else {
await DvRepairLineApi.updateRepairLine(formData.value) await DvRepairLineApi.updateRepairLine(formData.value)
message.success(t('common.updateSuccess')) message.success(t('common.updateSuccess'))
} }
formVisible.value = false dialogVisible.value = false
await getList() await getList()
} finally { } finally {
formLoading.value = false formLoading.value = false
} }
} }
/** 删除按钮操作 */
const handleDelete = async (id: number) => {
try {
await message.delConfirm()
await DvRepairLineApi.deleteRepairLine(id)
message.success(t('common.delSuccess'))
await getList()
} catch {}
}
/** 获取项目选项 */
const getSubjectOptions = async (query: string) => {
try {
const data = await DvSubjectApi.getSubjectPage({ name: query, pageNo: 1, pageSize: 20 })
subjectOptions.value = data.list
} catch {}
}
/** 监听工单编号变化 */ /** 监听工单编号变化 */
watch( watch(
() => props.repairId, () => props.repairId,

View File

@ -129,7 +129,7 @@
<dict-tag :type="DICT_TYPE.MES_DV_REPAIR_STATUS" :value="scope.row.status" /> <dict-tag :type="DICT_TYPE.MES_DV_REPAIR_STATUS" :value="scope.row.status" />
</template> </template>
</el-table-column> </el-table-column>
<el-table-column label="操作" align="center" width="220" fixed="right"> <el-table-column label="操作" align="center" width="240" fixed="right">
<template #default="scope"> <template #default="scope">
<!-- 草稿编辑删除 --> <!-- 草稿编辑删除 -->
<el-button <el-button
@ -153,7 +153,7 @@
<!-- 维修中完成维修 --> <!-- 维修中完成维修 -->
<el-button <el-button
link link
type="primary" type="success"
@click="openForm('confirm', scope.row.id)" @click="openForm('confirm', scope.row.id)"
v-hasPermi="['mes:dv-repair:update']" v-hasPermi="['mes:dv-repair:update']"
v-if="scope.row.status === MesDvRepairStatusEnum.CONFIRMED" v-if="scope.row.status === MesDvRepairStatusEnum.CONFIRMED"