| | |
| | | placeholder="请选择班组成员" |
| | | > |
| | | <el-option |
| | | v-for="user in userTeamOptions" |
| | | v-for="user in reportForm.userIdsList" |
| | | :key="user.userId" |
| | | :label="user.nickName" |
| | | :value="{ userId: user.userId, userName: user.nickName }" |
| | |
| | | </el-form-item> |
| | | </el-col> |
| | | |
| | | <el-col :span="12"> |
| | | <el-form-item label="机台" prop="deviceId"> |
| | | <el-select |
| | | v-model="reportForm.deviceId" |
| | | placeholder="请选择机台" |
| | | filterable |
| | | clearable |
| | | @change="(val) => handleDeviceChange(val)" |
| | | :disabled="isDetail" |
| | | > |
| | | <el-option |
| | | v-for="item in deviceOptions" |
| | | :key="item.id" |
| | | :label="item.deviceName" |
| | | :value="item.id" |
| | | /> |
| | | </el-select> |
| | | </el-form-item> |
| | | </el-col> |
| | | <!-- <el-col :span="12">--> |
| | | <!-- <el-form-item label="机台" prop="deviceId">--> |
| | | <!-- <el-select--> |
| | | <!-- v-model="reportForm.deviceId"--> |
| | | <!-- placeholder="请选择机台"--> |
| | | <!-- filterable--> |
| | | <!-- clearable--> |
| | | <!-- @change="(val) => handleDeviceChange(val)"--> |
| | | <!-- :disabled="isDetail"--> |
| | | <!-- >--> |
| | | <!-- <el-option--> |
| | | <!-- v-for="item in deviceOptions"--> |
| | | <!-- :key="item.id"--> |
| | | <!-- :label="item.deviceName"--> |
| | | <!-- :value="item.id"--> |
| | | <!-- />--> |
| | | <!-- </el-select>--> |
| | | <!-- </el-form-item>--> |
| | | <!-- </el-col>--> |
| | | |
| | | <el-col :span="12"> |
| | | <el-form-item label="审核人" prop="auditUserId"> |
| | |
| | | </el-button> |
| | | </el-col> |
| | | </el-row> |
| | | |
| | | <el-table :data="scheduleRows" border style="width: 100%" v-loading="scheduleLoading"> |
| | | <el-table-column type="index" label="序号" width="70" align="center" /> |
| | | |
| | | <el-table-column type="index" label="序号" width="70" align="center" :index="indexMethod" /> |
| | | <el-table-column label="本次上机机台" min-width="220"> |
| | | <template #default="{ row }"> |
| | | <el-select |
| | |
| | | import {ElMessageBox, ElMessage} from "element-plus"; |
| | | import Pagination from "@/components/PIMTable/Pagination.vue"; |
| | | import dayjs from "dayjs"; |
| | | import { processList } from '@/api/productionManagement/productionProcess.js' |
| | | import { |
| | | productWorkOrderPage, |
| | | updateProductWorkOrder, |
| | |
| | | return [val]; |
| | | }; |
| | | |
| | | const isCurrentUserReportWorker = (row) => { |
| | | const isCurrentUserInUserIds = (row) => { |
| | | const uid = String(currentUserId.value || ""); |
| | | if (!uid) return false; |
| | | if (!row) return false; |
| | | |
| | | const candidateIds = [ |
| | | row.reportUserIds, |
| | | row.reportWorkerIds, |
| | | row.userIdList, |
| | | row.reportUserId, |
| | | row.userId, |
| | | ] |
| | | .flatMap((v) => normalizeArray(v)) |
| | | .map((v) => String(v)) |
| | | const ids = normalizeArray(row?.userIds) |
| | | .map(id => String(id)) |
| | | .filter(Boolean); |
| | | |
| | | if (candidateIds.includes(uid)) return true; |
| | | |
| | | const candidateNames = [ |
| | | row.userNames, |
| | | row.reportUserNames, |
| | | row.reportWorkerNames, |
| | | row.userName, |
| | | ] |
| | | .flatMap((v) => normalizeArray(v)) |
| | | .map((v) => String(v)) |
| | | .filter(Boolean); |
| | | |
| | | if (currentUserName.value && candidateNames.includes(currentUserName.value)) { |
| | | return true; |
| | | } |
| | | |
| | | if (Array.isArray(row.reportWorkerList)) { |
| | | const list = row.reportWorkerList |
| | | .map((item) => String(item?.userId ?? item?.id ?? "")) |
| | | .filter(Boolean); |
| | | if (list.includes(uid)) return true; |
| | | const nameList = row.reportWorkerList |
| | | .map((item) => String(item?.userName ?? item?.nickName ?? "")) |
| | | .filter(Boolean); |
| | | if (currentUserName.value && nameList.includes(currentUserName.value)) return true; |
| | | } |
| | | |
| | | return false; |
| | | return ids.includes(uid); |
| | | }; |
| | | |
| | | const canOperateByReportWorker = computed(() => { |
| | |
| | | return; |
| | | } |
| | | |
| | | const rows = buildScheduleRowsFromRecords(records); |
| | | |
| | | const rows = records.map(record => mapMachineRecordToScheduleRow(record)); |
| | | scheduleRows.value = rows.length > 0 ? rows : [createScheduleRow({})]; |
| | | } catch (error) { |
| | | console.error("获取排产记录失败", error); |
| | |
| | | } |
| | | |
| | | return payload; |
| | | }; |
| | | |
| | | const indexMethod = (index) => { |
| | | return (schedulePage.current - 1) * schedulePage.size + index + 1; |
| | | }; |
| | | |
| | | const mapMachineRecordToScheduleRow = (record) => { |
| | |
| | | clickFun: row => { |
| | | showReportDialog(row); |
| | | }, |
| | | // 用户当前id |
| | | disabled: row => row.completeQuantity !==0 || |
| | | !isCurrentUserInUserIds(row) |
| | | }, |
| | | { |
| | | name: "生产排产", |
| | | clickFun: row => { |
| | | if (!row.canSchedule) { |
| | | ElMessage.warning("当前用户不在该工序人员中,不能生产排产"); |
| | | return; |
| | | } |
| | | openScheduleDialog(row); |
| | | }, |
| | | }, |
| | | disabled: row => !row.canSchedule |
| | | } |
| | | // { |
| | | // name:"审核", |
| | | // color: "#f56c6c", |
| | |
| | | page.size = obj.limit; |
| | | getList(); |
| | | }; |
| | | const getList = () => { |
| | | const getList = async () => { |
| | | tableLoading.value = true; |
| | | const params = {...searchForm.value, ...page}; |
| | | productWorkOrderPage(params) |
| | | .then(res => { |
| | | tableLoading.value = false; |
| | | tableData.value = res.data.records; |
| | | page.total = res.data.total; |
| | | }) |
| | | .catch(() => { |
| | | tableLoading.value = false; |
| | | }); |
| | | |
| | | try { |
| | | await ensureCurrentUser(); |
| | | await processLists(); |
| | | |
| | | const params = { ...searchForm.value, ...page }; |
| | | const res = await productWorkOrderPage(params); |
| | | |
| | | const records = Array.isArray(res?.data?.records) ? res.data.records : []; |
| | | |
| | | tableData.value = records.map(row => ({ |
| | | ...row, |
| | | canSchedule: canScheduleByWorkOrderNo(row) |
| | | })); |
| | | |
| | | page.total = res?.data?.total || 0; |
| | | } finally { |
| | | tableLoading.value = false; |
| | | } |
| | | }; |
| | | |
| | | // 下载并打印工单流转卡(文件流) |
| | |
| | | }; |
| | | |
| | | const showReportDialog = row => { |
| | | // if (!isCurrentUserReportWorker(row)) { |
| | | // ElMessage.warning("当前用户不是该工单的报工人,无法报工"); |
| | | // return; |
| | | // } |
| | | currentReportRowData.value = row; |
| | | reportForm.planQuantity = row.planQuantity - row.completeQuantity; |
| | | reportForm.quantity = |
| | | row.quantity !== undefined && row.quantity !== null ? row.quantity : null; |
| | | reportForm.quantity = row.quantity !== undefined && row.quantity !== null ? row.quantity : null; |
| | | reportForm.productProcessRouteItemId = row.productProcessRouteItemId; |
| | | reportForm.workOrderId = row.id; |
| | | reportForm.reportWork = row.reportWork; |
| | |
| | | reportForm.replenishQty = 0; |
| | | reportForm.teamList = []; |
| | | reportForm.scrapQty = 0; |
| | | reportForm.userIds = row.userIds || []; |
| | | |
| | | const ids = (row.userIds || "") |
| | | .split(",") |
| | | .map(id => id.trim()) |
| | | .filter(Boolean); |
| | | |
| | | reportForm.userIdsList = userTeamOptions.value |
| | | .filter(item => ids.includes(String(item.userId))) |
| | | .map(item => ({ |
| | | userId: item.userId, |
| | | nickName: item.nickName |
| | | })); |
| | | |
| | | |
| | | nextTick(() => { |
| | | reportFormRef.value?.clearValidate(); |
| | |
| | | return; |
| | | } |
| | | |
| | | if (!reportForm.startTime || !reportForm.endTime) { |
| | | ElMessageBox.alert("开始时间和结束时间不能为空", "提示", { |
| | | confirmButtonText: "确定", |
| | | }); |
| | | return; |
| | | } |
| | | |
| | | if (dayjs(reportForm.startTime).isSame(dayjs(reportForm.endTime)) || dayjs(reportForm.startTime).isAfter(dayjs(reportForm.endTime))) { |
| | | ElMessageBox.alert("开始时间必须小于结束时间", "提示", { |
| | | confirmButtonText: "确定", |
| | | }); |
| | | return; |
| | | } |
| | | |
| | | const submitData = { |
| | | ...reportForm, |
| | | quantity: quantity, |
| | |
| | | } |
| | | } |
| | | |
| | | onMounted(() => { |
| | | ensureCurrentUser(); |
| | | getList(); |
| | | const processData = ref([]); |
| | | |
| | | // 查询所有工序 |
| | | const processLists = async () => { |
| | | console.log(processData.value) |
| | | if (processData.value.length > 0) { |
| | | return processData.value; |
| | | } |
| | | const res = await processList(); |
| | | |
| | | processData.value = Array.isArray(res?.data) ? res.data : []; |
| | | return processData.value; |
| | | }; |
| | | |
| | | // 判断当前用户是否能排产 |
| | | const canScheduleByWorkOrderNo = (row) => { |
| | | if (!row) return false; |
| | | |
| | | const uid = String(currentUserId.value || ""); |
| | | if (!uid) return false; |
| | | |
| | | const currentProcess = processData.value.find(item => |
| | | String(item.id) === String(row.processId) |
| | | ); |
| | | |
| | | if (!currentProcess) return false; |
| | | |
| | | const ids = normalizeArray(currentProcess.userIds) |
| | | .map(id => String(id).trim()) |
| | | .filter(Boolean); |
| | | |
| | | return ids.includes(uid); |
| | | }; |
| | | |
| | | onMounted(async () => { |
| | | await ensureCurrentUser(); |
| | | await processLists(); |
| | | await getList(); |
| | | getUserList(); |
| | | getDeviceList(); |
| | | }); |