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

View File

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

View File

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