| | |
| | | </el-dialog> |
| | | <el-dialog v-model="reportDialogVisible" |
| | | title="报工" |
| | | width="500px"> |
| | | width="1400px"> |
| | | <el-form ref="reportFormRef" |
| | | :model="reportForm" |
| | | :rules="reportFormRules" |
| | | label-width="120px"> |
| | | <el-form-item label="待生产数量"> |
| | | <el-input v-model="reportForm.planQuantity" |
| | | readonly |
| | | style="width: 300px" /> |
| | | </el-form-item> |
| | | <el-form-item label="投入总量(kg)" |
| | | <el-row :gutter="30"> |
| | | <el-col :span="12"> |
| | | <el-form-item label="待生产数量"> |
| | | <el-input v-model="reportForm.planQuantity" |
| | | readonly |
| | | style="width: 300px" /> |
| | | </el-form-item> |
| | | </el-col> |
| | | <el-col :span="12"> |
| | | <el-form-item label="投入总量(kg)" |
| | | prop="totalInvestment"> |
| | | <el-input v-model.number="reportForm.totalInvestment" |
| | | type="number" |
| | |
| | | placeholder="请输入投入总量" |
| | | @input="handleTotalInvestmentInput" /> |
| | | </el-form-item> |
| | | <el-form-item label="本次生产数量" |
| | | </el-col> |
| | | <el-col :span="12"><el-form-item label="本次生产数量" |
| | | prop="quantity"> |
| | | <el-input v-model.number="reportForm.quantity" |
| | | type="number" |
| | |
| | | style="width: 300px" |
| | | placeholder="请输入本次生产数量" |
| | | @input="handleQuantityInput" /> |
| | | </el-form-item> |
| | | <el-form-item label="报废数量" |
| | | </el-form-item></el-col> |
| | | <el-col :span="12"><el-form-item label="报废数量" |
| | | prop="scrapQty"> |
| | | <el-input v-model.number="reportForm.scrapQty" |
| | | type="number" |
| | |
| | | style="width: 300px" |
| | | placeholder="请输入报废数量" |
| | | @input="handleScrapQtyInput" /> |
| | | </el-form-item> |
| | | <el-form-item label="检品数量" |
| | | </el-form-item></el-col> |
| | | <el-col :span="12"> |
| | | <el-form-item label="检品数量" |
| | | prop="inspectedQuantity"> |
| | | <el-input v-model.number="reportForm.inspectedQuantity" |
| | | type="number" |
| | |
| | | placeholder="请输入检品数量" |
| | | @input="handleInspectedQuantity"/> |
| | | </el-form-item> |
| | | <el-form-item label="班组信息"> |
| | | </el-col> |
| | | <el-col :span="12"> |
| | | <el-form-item label="班组信息"> |
| | | <el-select v-model="reportForm.userId" |
| | | style="width: 300px" |
| | | placeholder="请选择班组信息" |
| | |
| | | :value="user.userId" /> |
| | | </el-select> |
| | | </el-form-item> |
| | | <el-form-item label="设备选择"> |
| | | <el-select v-model="reportForm.deviceId" |
| | | style="width: 300px" |
| | | placeholder="请选择设备" |
| | | clearable |
| | | filterable |
| | | @change="handleDeviceChange"> |
| | | <el-option v-for="device in deviceOptions" |
| | | :key="device.id" |
| | | :label="device.deviceName" |
| | | :value="device.id" /> |
| | | </el-select> |
| | | </el-form-item> |
| | | <el-form-item v-if="currentDevice" label="设备编号"> |
| | | <el-input v-model="currentDevice.deviceCode" |
| | | disabled |
| | | style="width: 300px" /> |
| | | </el-form-item> |
| | | <el-form-item v-if="currentDevice" label="转数"> |
| | | <el-input v-model="currentDevice.rpm" |
| | | disabled |
| | | style="width: 300px" /> |
| | | </el-form-item> |
| | | <ProductionRecordForm ref="productionRecordFormRef" :list="processParamList"/> |
| | | </el-col> |
| | | </el-row> |
| | | <ProductionRecordForm |
| | | ref="productionRecordFormRef" |
| | | :list="processParamList" |
| | | :device-options="deviceOptions" |
| | | :selected-device-id="reportForm.deviceId" |
| | | /> |
| | | |
| | | <div style="margin-top: 20px"> |
| | | <div style="display: flex; justify-content: flex-end; margin-bottom: 10px"> |
| | | <el-button type="primary" @click="openAddMaterialDialog">新增原材料</el-button> |
| | | </div> |
| | | |
| | | <el-table |
| | | :data="reportForm.drawMaterialList" |
| | | border |
| | | style="width: 100%" |
| | | height="220px" |
| | | empty-text="暂无原材料领用明细" |
| | | > |
| | | <el-table-column type="index" label="序号" width="60" align="center" /> |
| | | <el-table-column prop="productName" label="产品名称" min-width="160" /> |
| | | <el-table-column prop="model" label="型号" min-width="150" /> |
| | | <el-table-column prop="unit" label="单位" width="90" align="center" /> |
| | | <el-table-column prop="reportQty" label="领用数量" width="160" align="center"> |
| | | <template #default="{ row }"> |
| | | <el-input-number |
| | | v-model.number="row.reportQty" |
| | | :min="0" |
| | | :precision="2" |
| | | :controls="false" |
| | | :max="row.requisitionQty || 0" |
| | | :disabled="!row.requisitionQty" |
| | | style="width: 100%" |
| | | /> |
| | | </template> |
| | | </el-table-column> |
| | | <el-table-column prop="remark" label="备注" min-width="160"> |
| | | <template #default="{ row }"> |
| | | <el-input v-model="row.remark" placeholder="请输入备注" clearable /> |
| | | </template> |
| | | </el-table-column> |
| | | <el-table-column label="操作" width="80" align="center"> |
| | | <template #default="{ $index }"> |
| | | <el-button type="danger" link @click="removeDrawMaterialRow($index)">删除</el-button> |
| | | </template> |
| | | </el-table-column> |
| | | </el-table> |
| | | </div> |
| | | </el-form> |
| | | <template #footer> |
| | | <span class="dialog-footer"> |
| | |
| | | </span> |
| | | </template> |
| | | </el-dialog> |
| | | |
| | | <el-dialog |
| | | v-model="addMaterialDialogVisible" |
| | | title="选择原材料" |
| | | width="1000px" |
| | | top="5vh" |
| | | :close-on-click-modal="false" |
| | | append-to-body |
| | | destroy-on-close |
| | | > |
| | | <div> |
| | | <el-table |
| | | :data="availableMaterials" |
| | | border |
| | | style="width: 100%" |
| | | height="45vh" |
| | | row-key="id" |
| | | @selection-change="handleAddMaterialSelectionChange" |
| | | > |
| | | <el-table-column type="selection" width="55" align="center" /> |
| | | <el-table-column type="index" label="序号" width="60" align="center" /> |
| | | <el-table-column prop="productName" label="产品名称" min-width="160" /> |
| | | <el-table-column prop="model" label="型号" min-width="150" /> |
| | | <el-table-column prop="unit" label="单位" width="90" align="center" /> |
| | | <el-table-column prop="requisitionQty" label="可领用数量" width="140" align="center" /> |
| | | </el-table> |
| | | |
| | | <!-- 已选择明细展示放在报工弹框下方的 reportForm.drawMaterialList 表格里 --> |
| | | </div> |
| | | |
| | | <template #footer> |
| | | <span class="dialog-footer"> |
| | | <el-button @click="addMaterialDialogVisible = false">取消</el-button> |
| | | <el-button type="primary" @click="handleConfirmAddMaterial">确定</el-button> |
| | | </span> |
| | | </template> |
| | | </el-dialog> |
| | | <FilesDia ref="workOrderFilesRef" /> |
| | | </div> |
| | | </template> |
| | | |
| | | <script setup> |
| | | import { onMounted, ref, nextTick, computed } from "vue"; |
| | | import { onMounted, ref, nextTick } from "vue"; |
| | | import { ElMessageBox } from "element-plus"; |
| | | import dayjs from "dayjs"; |
| | | import { |
| | |
| | | { |
| | | label: "规格", |
| | | prop: "model", |
| | | }, |
| | | { |
| | | label: "UID码", |
| | | prop: "uidNo", |
| | | }, |
| | | { |
| | | label: "单位", |
| | |
| | | clickFun: row => { |
| | | showReportDialog(row); |
| | | }, |
| | | disabled: row => row.planQuantity <= 0, |
| | | disabled: row => row.planQuantity <= 0 || row.productOrderIsEnd, |
| | | }, |
| | | ], |
| | | }, |
| | |
| | | userId: "", |
| | | productMainId: null, |
| | | deviceId: null, |
| | | // 报工时选择的原材料领用明细 |
| | | drawMaterialList: [], |
| | | otherData: { |
| | | rows: [] |
| | | }, |
| | | }); |
| | | const productionRecordFormRef = ref(); |
| | | |
| | | const currentDevice = computed(() => { |
| | | if (!reportForm.deviceId) return null; |
| | | return deviceOptions.value.find(device => device.id === reportForm.deviceId) || null; |
| | | }); |
| | | // 原材料领用(新增/选择弹框) |
| | | const availableMaterials = ref([]); // 来自当前行的 drawMaterials(json解析成list) |
| | | const addMaterialDialogVisible = ref(false); |
| | | const selectedAddMaterials = ref([]); // 弹框里用户选中的条目(可编辑 reportQty) |
| | | |
| | | // 投入总量验证规则 |
| | | const validateTotalInvestment = (rule, value, callback) => { |
| | |
| | | const handleQuantityInput = value => { |
| | | if (value === "" || value === null || value === undefined) { |
| | | reportForm.quantity = null; |
| | | reportForm.scrapQty = null; |
| | | return; |
| | | } |
| | | const num = Number(value); |
| | |
| | | } |
| | | if (num < 1) { |
| | | reportForm.quantity = null; |
| | | reportForm.scrapQty = null; |
| | | return; |
| | | } |
| | | if (!Number.isInteger(num)) { |
| | | const intValue = Math.floor(num); |
| | | if (intValue < 1) { |
| | | reportForm.quantity = null; |
| | | reportForm.scrapQty = null; |
| | | return; |
| | | } |
| | | reportForm.quantity = intValue; |
| | | } else { |
| | | reportForm.quantity = num; |
| | | } |
| | | // 如果 totalInvestment 有值,自动计算 scrapQty = totalInvestment - quantity |
| | | if (reportForm.totalInvestment !== null && reportForm.totalInvestment !== undefined && reportForm.totalInvestment !== "") { |
| | | const total = Number(reportForm.totalInvestment); |
| | | const qty = Number(reportForm.quantity); |
| | | if (total > qty) { |
| | | reportForm.scrapQty = total - qty; |
| | | } else { |
| | | reportForm.scrapQty = null; |
| | | } |
| | | } |
| | | }; |
| | | |
| | |
| | | return res.data.records |
| | | }; |
| | | |
| | | // 原材料领用:drawMaterials(json -> list)与 drawMaterialList(最终提交) |
| | | const parseMaybeJsonList = (val) => { |
| | | if (!val) return []; |
| | | if (Array.isArray(val)) return val; |
| | | if (typeof val === 'string') { |
| | | try { |
| | | const parsed = JSON.parse(val); |
| | | return Array.isArray(parsed) ? parsed : []; |
| | | } catch (e) { |
| | | return []; |
| | | } |
| | | } |
| | | return []; |
| | | }; |
| | | |
| | | const normalizeDrawMaterialItem = (item) => { |
| | | if (!item) return null; |
| | | return { |
| | | ...item, |
| | | reportQty: item.reportQty ?? item.requisitionQty ?? 0, |
| | | remark: item.remark ?? '', |
| | | }; |
| | | }; |
| | | |
| | | const materialKey = (item) => String(item?.id ?? item?.productModelId ?? ''); |
| | | |
| | | const openAddMaterialDialog = () => { |
| | | if (!availableMaterials.value || availableMaterials.value.length === 0) { |
| | | proxy.$modal.msgWarning("当前工单没有可领用的原材料"); |
| | | return; |
| | | } |
| | | selectedAddMaterials.value = []; |
| | | addMaterialDialogVisible.value = true; |
| | | }; |
| | | |
| | | const handleAddMaterialSelectionChange = (selection) => { |
| | | selectedAddMaterials.value = (selection || []).map((x) => normalizeDrawMaterialItem({ ...x })); |
| | | }; |
| | | |
| | | const handleConfirmAddMaterial = () => { |
| | | if (!selectedAddMaterials.value || selectedAddMaterials.value.length === 0) { |
| | | proxy.$modal.msgWarning("请先选择原材料"); |
| | | return; |
| | | } |
| | | |
| | | const existingKeys = new Set((reportForm.drawMaterialList || []).map(materialKey)); |
| | | const newItems = selectedAddMaterials.value |
| | | .filter((it) => !existingKeys.has(materialKey(it))) |
| | | .map(normalizeDrawMaterialItem) |
| | | .filter(Boolean); |
| | | |
| | | if (newItems.length === 0) { |
| | | proxy.$modal.msgWarning("所选原材料已存在,无需重复添加"); |
| | | addMaterialDialogVisible.value = false; |
| | | return; |
| | | } |
| | | |
| | | reportForm.drawMaterialList = [...(reportForm.drawMaterialList || []), ...newItems]; |
| | | addMaterialDialogVisible.value = false; |
| | | }; |
| | | |
| | | const removeDrawMaterialRow = (index) => { |
| | | if (!Array.isArray(reportForm.drawMaterialList)) return; |
| | | reportForm.drawMaterialList.splice(index, 1); |
| | | }; |
| | | |
| | | const processParamList = ref([]) |
| | | const showReportDialog = async row => { |
| | | currentReportRowData.value = row; |
| | |
| | | |
| | | // 获取工序绑定设备列表 |
| | | getDeviceList(row.processId); |
| | | |
| | | // 原材料:可选项来自当前行 drawMaterials(json -> list) |
| | | availableMaterials.value = parseMaybeJsonList(row.drawMaterials); |
| | | |
| | | // 最终领用清单:优先回显 drawMaterialList(若后端已返回);否则为空 |
| | | const existingDrawList = parseMaybeJsonList(row.drawMaterialList); |
| | | reportForm.drawMaterialList = (existingDrawList || []).map(normalizeDrawMaterialItem).filter(Boolean); |
| | | selectedAddMaterials.value = []; |
| | | addMaterialDialogVisible.value = false; |
| | | |
| | | nextTick(() => { |
| | | reportFormRef.value?.clearValidate(); |
| | |
| | | const data = await productionRecordFormRef.value.submitData(); |
| | | console.log("生产记录表单数据:", data); |
| | | reportForm.otherData.rows = data || []; |
| | | // 机台选择由 ProductionRecordForm 的“机台选择”参数决定 |
| | | reportForm.deviceId = getDeviceFromRecordParams(data) ?? reportForm.deviceId; |
| | | } catch (error) { |
| | | console.error("获取生产记录表单数据失败", error); |
| | | return; |
| | |
| | | ...reportForm, |
| | | quantity: quantity, |
| | | scrapQty: scrapQty, |
| | | // drawMaterialList 直接传数组(不做 JSON.stringify) |
| | | drawMaterialList: reportForm.drawMaterialList || [], |
| | | otherData: JSON.stringify(reportForm.otherData) |
| | | }; |
| | | |
| | |
| | | } |
| | | }; |
| | | |
| | | // 设备选择变化时更新 deviceId |
| | | const handleDeviceChange = deviceId => { |
| | | reportForm.deviceId = deviceId; |
| | | const getDeviceFromRecordParams = (rows) => { |
| | | if (!Array.isArray(rows)) return null; |
| | | const machineRow = rows.find(r => r?.type === '机台选择'); |
| | | return machineRow?.value ?? null; |
| | | }; |
| | | |
| | | onMounted(() => { |