feat(mes): 添加工作站编码字段,移除单位编号字段并优化相关逻辑

pull/871/MERGE
YunaiV 2026-03-15 22:04:27 +08:00
parent d49ba31f78
commit 3860525dfd
3 changed files with 71 additions and 200 deletions

View File

@ -17,8 +17,6 @@ export interface ProTaskVO {
itemName: string // 产品名称 itemName: string // 产品名称
itemCode: string // 产品编码 itemCode: string // 产品编码
itemSpec: string // 规格型号 itemSpec: string // 规格型号
unitMeasureId: number // 单位编号
unitMeasureName: string // 单位名称
quantity: number // 排产数量 quantity: number // 排产数量
producedQuantity: number // 已生产数量 producedQuantity: number // 已生产数量
qualifyQuantity: number // 合格品数量 qualifyQuantity: number // 合格品数量
@ -28,7 +26,7 @@ export interface ProTaskVO {
clientName: string // 客户名称 clientName: string // 客户名称
startTime: Date // 开始生产时间 startTime: Date // 开始生产时间
duration: number // 生产时长工作日1=8小时 duration: number // 生产时长工作日1=8小时
endTime: Date // 结束生产时间 endTime: Date | number // 结束生产时间
colorCode: string // 甘特图显示颜色 colorCode: string // 甘特图显示颜色
requestDate: Date // 需求日期(从工单查) requestDate: Date // 需求日期(从工单查)
finishDate: Date // 完成日期 finishDate: Date // 完成日期

View File

@ -1,15 +1,14 @@
<!-- MES 生产任务列表工单维度排产对话框内使用 --> <!-- MES 生产任务列表工单维度排产对话框内使用 -->
<!-- TODO @芋艿 review -->
<template> <template>
<div> <div>
<!-- DONE @AI详情时整个都不展示 -->
<!-- 操作栏 --> <!-- 操作栏 -->
<div class="mb-10px"> <div v-if="!disabled" class="mb-10px">
<el-button <el-button
type="primary" type="primary"
plain plain
@click="openForm('create')" @click="openForm('create')"
v-hasPermi="['mes:pro-task:create']" v-hasPermi="['mes:pro-task:create']"
:disabled="disabled"
> >
<Icon icon="ep:plus" class="mr-5px" /> 新增任务 <Icon icon="ep:plus" class="mr-5px" /> 新增任务
</el-button> </el-button>
@ -19,26 +18,26 @@
<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="code" width="140" /> <el-table-column label="任务编码" align="center" prop="code" width="140" />
<el-table-column label="任务名称" align="center" prop="name" min-width="150" /> <el-table-column label="任务名称" align="center" prop="name" min-width="150" />
<el-table-column label="工作站" align="center" prop="workstationName" width="120" /> <el-table-column label="工作站编号" align="center" prop="workstationCode" width="120" />
<el-table-column label="工作站名称" align="center" prop="workstationName" width="120" />
<el-table-column label="排产数量" align="center" prop="quantity" width="100" /> <el-table-column label="排产数量" align="center" prop="quantity" width="100" />
<el-table-column label="已生产" align="center" prop="producedQuantity" width="80" /> <el-table-column label="已生产数量" align="center" prop="producedQuantity" width="100" />
<el-table-column label="合格" align="center" prop="qualifyQuantity" width="80" />
<el-table-column <el-table-column
label="开始时间" label="开始生产时间"
align="center" align="center"
prop="startTime" prop="startTime"
:formatter="dateFormatter" :formatter="dateFormatter"
width="170" width="170"
/> />
<el-table-column label="时长(天)" align="center" prop="duration" width="80" /> <el-table-column label="生产时长" align="center" prop="duration" width="80" />
<el-table-column <el-table-column
label="结束时间" label="预计完成时间"
align="center" align="center"
prop="endTime" prop="endTime"
:formatter="dateFormatter" :formatter="dateFormatter"
width="170" width="170"
/> />
<el-table-column label="颜色" align="center" prop="colorCode" width="60"> <el-table-column label="显示颜色" align="center" prop="colorCode" width="100">
<template #default="scope"> <template #default="scope">
<div <div
:style="{ :style="{
@ -51,102 +50,31 @@
></div> ></div>
</template> </template>
</el-table-column> </el-table-column>
<el-table-column label="状态" align="center" prop="status" width="100"> <el-table-column v-if="!disabled" label="操作" align="center" width="160" fixed="right">
<template #default="scope"> <template #default="scope">
<dict-tag :type="DICT_TYPE.MES_PRO_TASK_STATUS" :value="scope.row.status" />
</template>
</el-table-column>
<el-table-column label="操作" align="center" width="200" fixed="right">
<template #default="scope">
<!-- 草稿(0)编辑/删除/开始 -->
<template v-if="scope.row.status === MesProTaskStatusEnum.NORMAL">
<el-button
link
type="primary"
@click="openForm('update', scope.row.id)"
v-hasPermi="['mes:pro-task:update']"
>
编辑
</el-button>
<el-button
link
type="danger"
@click="handleDelete(scope.row.id)"
v-hasPermi="['mes:pro-task:delete']"
>
删除
</el-button>
</template>
<!-- 进行中(1)暂停/完成/取消 -->
<template v-if="scope.row.status === MesProTaskStatusEnum.START">
<el-button
link
type="warning"
@click="handleChangeStatus(scope.row.id, MesProTaskStatusEnum.PAUSE, '暂停')"
v-hasPermi="['mes:pro-task:update']"
>
暂停
</el-button>
<el-button
link
type="success"
@click="handleChangeStatus(scope.row.id, MesProTaskStatusEnum.FINISHED, '完成')"
v-hasPermi="['mes:pro-task:update']"
>
完成
</el-button>
<el-button
link
type="danger"
@click="handleChangeStatus(scope.row.id, MesProTaskStatusEnum.CANCELED, '取消')"
v-hasPermi="['mes:pro-task:update']"
>
取消
</el-button>
</template>
<!-- 暂停(2)继续/完成/取消 -->
<template v-if="scope.row.status === MesProTaskStatusEnum.PAUSE">
<el-button
link
type="primary"
@click="handleChangeStatus(scope.row.id, MesProTaskStatusEnum.START, '继续')"
v-hasPermi="['mes:pro-task:update']"
>
继续
</el-button>
<el-button
link
type="success"
@click="handleChangeStatus(scope.row.id, MesProTaskStatusEnum.FINISHED, '完成')"
v-hasPermi="['mes:pro-task:update']"
>
完成
</el-button>
<el-button
link
type="danger"
@click="handleChangeStatus(scope.row.id, MesProTaskStatusEnum.CANCELED, '取消')"
v-hasPermi="['mes:pro-task:update']"
>
取消
</el-button>
</template>
<!-- 所有状态详情 -->
<el-button <el-button
link link
type="primary" type="primary"
@click="openForm('detail', scope.row.id)" @click="openForm('update', scope.row.id)"
v-hasPermi="['mes:pro-task:query']" v-hasPermi="['mes:pro-task:update']"
> >
详情 编辑
</el-button>
<el-button
link
type="danger"
@click="handleDelete(scope.row.id)"
v-hasPermi="['mes:pro-task:delete']"
>
删除
</el-button> </el-button>
</template> </template>
</el-table-column> </el-table-column>
</el-table> </el-table>
</div> </div>
<!-- 添加/编辑/详情任务弹窗 --> <!-- 添加/编辑任务弹窗 -->
<Dialog :title="dialogTitle" v-model="dialogVisible" width="800px"> <Dialog :title="dialogTitle" v-model="dialogVisible" width="960px">
<el-form <el-form
ref="formRef" ref="formRef"
:model="formData" :model="formData"
@ -155,36 +83,12 @@
v-loading="formLoading" v-loading="formLoading"
> >
<el-row> <el-row>
<el-col :span="12"> <el-col :span="8">
<el-form-item label="所属工单" prop="workOrderId">
<el-input :model-value="workOrderInfo" disabled />
</el-form-item>
</el-col>
<el-col :span="12">
<el-form-item label="工序" prop="processId">
<el-select
v-model="formData.processId"
placeholder="请选择工序"
class="!w-1/1"
:disabled="isDetail"
>
<el-option
v-for="item in props.processList"
:key="item.processId"
:label="item.processName"
:value="item.processId"
/>
</el-select>
</el-form-item>
</el-col>
</el-row>
<el-row>
<el-col :span="12">
<el-form-item label="工作站" prop="workstationId"> <el-form-item label="工作站" prop="workstationId">
<MdWorkstationSelect v-model="formData.workstationId" :disabled="isDetail" /> <MdWorkstationSelect v-model="formData.workstationId" :disabled="isDetail" />
</el-form-item> </el-form-item>
</el-col> </el-col>
<el-col :span="12"> <el-col :span="8">
<el-form-item label="排产数量" prop="quantity"> <el-form-item label="排产数量" prop="quantity">
<el-input-number <el-input-number
v-model="formData.quantity" v-model="formData.quantity"
@ -195,9 +99,14 @@
/> />
</el-form-item> </el-form-item>
</el-col> </el-col>
<el-col :span="8">
<el-form-item label="甘特颜色" prop="colorCode">
<el-color-picker v-model="formData.colorCode" :disabled="isDetail" />
</el-form-item>
</el-col>
</el-row> </el-row>
<el-row> <el-row>
<el-col :span="12"> <el-col :span="8">
<el-form-item label="开始时间" prop="startTime"> <el-form-item label="开始时间" prop="startTime">
<el-date-picker <el-date-picker
v-model="formData.startTime" v-model="formData.startTime"
@ -209,21 +118,19 @@
/> />
</el-form-item> </el-form-item>
</el-col> </el-col>
<el-col :span="12"> <el-col :span="8">
<el-form-item label="生产时长(工作日)" prop="duration"> <el-form-item label="生产时长" prop="duration">
<el-input-number <el-input-number
v-model="formData.duration" v-model="formData.duration"
:min="1" :min="1"
:precision="0" :precision="0"
class="!w-1/1" class="!w-1/1"
:disabled="isDetail" :disabled="isDetail"
@change="computeEndTime" @change="handleDurationChange"
/> />
</el-form-item> </el-form-item>
</el-col> </el-col>
</el-row> <el-col :span="8">
<el-row>
<el-col :span="12">
<el-form-item label="结束时间"> <el-form-item label="结束时间">
<el-date-picker <el-date-picker
v-model="formData.endTime" v-model="formData.endTime"
@ -234,18 +141,6 @@
/> />
</el-form-item> </el-form-item>
</el-col> </el-col>
<el-col :span="12">
<el-form-item label="甘特颜色" prop="colorCode">
<el-color-picker v-model="formData.colorCode" :disabled="isDetail" />
</el-form-item>
</el-col>
</el-row>
<el-row v-if="taskFormType !== 'create'">
<el-col :span="12">
<el-form-item label="任务状态">
<dict-tag :type="DICT_TYPE.MES_PRO_TASK_STATUS" :value="formData.status" />
</el-form-item>
</el-col>
</el-row> </el-row>
<el-row> <el-row>
<el-col :span="24"> <el-col :span="24">
@ -269,23 +164,17 @@
<script setup lang="ts"> <script setup lang="ts">
import { dateFormatter } from '@/utils/formatTime' import { dateFormatter } from '@/utils/formatTime'
import { DICT_TYPE } from '@/utils/dict'
import { ProTaskApi, ProTaskVO } from '@/api/mes/pro/task' import { ProTaskApi, ProTaskVO } from '@/api/mes/pro/task'
import { ProRouteProcessVO } from '@/api/mes/pro/route/process'
import { MesProTaskStatusEnum } 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'
defineOptions({ name: 'ProTaskList' }) defineOptions({ name: 'ProTaskList' })
// DONE @AI workOrderCodeworkOrderNameprocessList
const props = defineProps<{ const props = defineProps<{
workOrderId: number workOrderId: number
workOrderCode?: string
workOrderName?: string
routeId: number routeId: number
processId: number processId: number
itemId?: number itemId?: number
unitMeasureId?: number
processList?: ProRouteProcessVO[]
disabled?: boolean disabled?: boolean
}>() }>()
@ -322,29 +211,25 @@ const handleDelete = async (id: number) => {
} catch {} } catch {}
} }
/** 变更任务状态 */ // ==================== / ====================
const handleChangeStatus = async (id: number, status: number, actionName: string) => { const dialogVisible = ref(false) //
try { const formLoading = ref(false) //
await message.confirm(`确认要${actionName}该任务吗?`) const formType = ref('') // create - update - detail -
await ProTaskApi.updateTask({ id, status } as any) const dialogTitle = computed(() => {
message.success(`任务已${actionName}`) const typeMap: Record<string, string> = {
await getList() detail: '生产任务详情',
} catch {} create: '新增生产任务',
} update: '编辑生产任务'
}
// ==================== // ==================== return typeMap[formType.value] || ''
const dialogVisible = ref(false) })
const dialogTitle = ref('') const formData = ref<ProTaskVO>({
const formLoading = ref(false)
const taskFormType = ref('')
const formData = ref({
id: undefined, id: undefined,
workOrderId: undefined, workOrderId: undefined,
workstationId: undefined, workstationId: undefined,
routeId: undefined, routeId: undefined,
processId: undefined, processId: undefined,
itemId: undefined, itemId: undefined,
unitMeasureId: undefined,
quantity: undefined, quantity: undefined,
startTime: undefined, startTime: undefined,
duration: 1, duration: 1,
@ -352,23 +237,18 @@ const formData = ref({
colorCode: '#00AEF3', colorCode: '#00AEF3',
status: undefined, status: undefined,
remark: undefined remark: undefined
}) } as unknown as ProTaskVO)
const formRules = reactive({ const formRules = reactive({
workstationId: [{ required: true, message: '工作站不能为空', trigger: 'change' }], workstationId: [{ required: true, message: '工作站不能为空', trigger: 'change' }],
processId: [{ required: true, message: '工序不能为空', trigger: 'change' }],
quantity: [{ required: true, message: '排产数量不能为空', trigger: 'blur' }], quantity: [{ required: true, message: '排产数量不能为空', trigger: 'blur' }],
startTime: [{ required: true, message: '开始时间不能为空', trigger: 'change' }], startTime: [{ required: true, message: '开始时间不能为空', trigger: 'change' }],
duration: [{ required: true, message: '生产时长不能为空', trigger: 'blur' }] duration: [{ required: true, message: '生产时长不能为空', trigger: 'blur' }]
}) })
const formRef = ref() const formRef = ref() // Ref
const isDetail = computed(() => formType.value === 'detail')
/** 工单信息(只读展示) */ /** 计算结束时间:开始时间 + 生产时长 * 8小时 */
const workOrderInfo = ref('') const handleDurationChange = () => {
const isDetail = computed(() => taskFormType.value === 'detail')
/** 自动计算结束时间startTime + duration * 8h */
const computeEndTime = () => {
if (formData.value.startTime && formData.value.duration) { if (formData.value.startTime && formData.value.duration) {
const start = const start =
typeof formData.value.startTime === 'number' typeof formData.value.startTime === 'number'
@ -378,34 +258,29 @@ const computeEndTime = () => {
} }
} }
// /** 监听开始时间变化也重新计算 */
watch( watch(
() => formData.value.startTime, () => formData.value.startTime,
() => computeEndTime() () => handleDurationChange()
) )
/** 打开表单弹窗 */ /** 打开表单弹窗 */
const openForm = async (type: string, id?: number) => { const openForm = async (type: string, id?: number) => {
dialogVisible.value = true dialogVisible.value = true
dialogTitle.value = type === 'detail' ? '任务详情' : type === 'create' ? '新增任务' : '编辑任务' formType.value = type
taskFormType.value = type
resetForm() resetForm()
if (type === 'create') { if (type === 'create') {
// //
formData.value.workOrderId = props.workOrderId formData.value.workOrderId = props.workOrderId!
formData.value.routeId = props.routeId formData.value.routeId = props.routeId!
formData.value.processId = props.processId formData.value.processId = props.processId!
formData.value.itemId = props.itemId formData.value.itemId = props.itemId!
formData.value.unitMeasureId = props.unitMeasureId
workOrderInfo.value = `${props.workOrderCode || ''} ${props.workOrderName || ''}`
} else if (id) { } else if (id) {
// / //
formLoading.value = true formLoading.value = true
try { try {
const data = await ProTaskApi.getTask(id) formData.value = await ProTaskApi.getTask(id)
formData.value = data
workOrderInfo.value = `${data.workOrderCode || ''} ${data.workOrderName || ''}`
} finally { } finally {
formLoading.value = false formLoading.value = false
} }
@ -418,7 +293,7 @@ const submitForm = async () => {
formLoading.value = true formLoading.value = true
try { try {
const data = formData.value as unknown as ProTaskVO const data = formData.value as unknown as ProTaskVO
if (taskFormType.value === 'create') { if (formType.value === 'create') {
await ProTaskApi.createTask(data) await ProTaskApi.createTask(data)
message.success(t('common.createSuccess')) message.success(t('common.createSuccess'))
} else { } else {
@ -441,7 +316,6 @@ const resetForm = () => {
routeId: undefined, routeId: undefined,
processId: undefined, processId: undefined,
itemId: undefined, itemId: undefined,
unitMeasureId: undefined,
quantity: undefined, quantity: undefined,
startTime: undefined, startTime: undefined,
duration: 1, duration: 1,
@ -449,17 +323,20 @@ const resetForm = () => {
colorCode: '#00AEF3', colorCode: '#00AEF3',
status: undefined, status: undefined,
remark: undefined remark: undefined
} } as unknown as ProTaskVO
formRef.value?.resetFields() formRef.value?.resetFields()
} }
// processId /** 监听 processId 切换重新加载 */
watch( watch(
() => props.processId, () => props.processId,
() => getList() () => getList()
) )
onMounted(() => getList()) /** 初始化 */
onMounted(() => {
getList()
})
defineExpose({ getList }) defineExpose({ getList })
</script> </script>

View File

@ -137,13 +137,9 @@
> >
<ProTaskList <ProTaskList
:work-order-id="formData.id!" :work-order-id="formData.id!"
:work-order-code="formData.code!"
:work-order-name="formData.name!"
:route-id="currentRouteId" :route-id="currentRouteId"
:process-id="rp.processId" :process-id="rp.processId"
:item-id="formData.productId!" :item-id="formData.productId!"
:unit-measure-id="(formData as any).unitMeasureId"
:process-list="routeProcessList"
/> />
</el-card> </el-card>
<template #footer> <template #footer>