<template>
|
<div class="app-container">
|
<div class="search_form mb20">
|
<div class="search-row">
|
<div class="search-item">
|
<span class="search_title">生产订单号:</span>
|
<el-input v-model="searchForm.npsNo"
|
style="width: 200px"
|
placeholder="请输入"
|
@change="handleQuery"
|
clearable
|
prefix-icon="Search" />
|
</div>
|
<div class="search-item">
|
<span class="search_title">产品名称:</span>
|
<el-input v-model="searchForm.productName"
|
style="width: 200px"
|
placeholder="请输入"
|
@change="handleQuery"
|
clearable
|
prefix-icon="Search" />
|
</div>
|
<div class="search-item">
|
<span class="search_title">规格:</span>
|
<el-input v-model="searchForm.model"
|
style="width: 200px"
|
placeholder="请输入"
|
@change="handleQuery"
|
clearable
|
prefix-icon="Search" />
|
</div>
|
<div class="search-item">
|
<el-button type="primary"
|
@click="handleQuery">搜索</el-button>
|
</div>
|
</div>
|
</div>
|
<div class="table_list">
|
<el-table :data="tableData"
|
border
|
v-loading="tableLoading"
|
:expand-row-keys="expandedRowKeys"
|
:row-key="(row) => row.productionOrderId"
|
@expand-change="expandChange"
|
style="width: 100%"
|
height="calc(100vh - 18.5em)">
|
<el-table-column type="expand"
|
width="60"
|
fixed="left">
|
<template #default="props">
|
<el-table :data="props.row.children"
|
border
|
v-loading="props.row.childrenLoading">
|
<el-table-column align="center"
|
label="序号"
|
type="index"
|
width="60" />
|
<el-table-column label="工单类型"
|
prop="workOrderType"
|
width="80" />
|
<el-table-column label="工单编号"
|
prop="workOrderNo"
|
width="140"
|
show-overflow-tooltip />
|
<el-table-column label="工序名称"
|
prop="operationName"
|
width="100"
|
show-overflow-tooltip />
|
<el-table-column label="需求数量"
|
prop="planQuantity"
|
width="100" />
|
<el-table-column label="完成数量"
|
prop="completeQuantity"
|
width="100" />
|
<el-table-column label="完成进度"
|
width="140"
|
align="center">
|
<template #default="{ row }">
|
<el-progress :percentage="toProgressPercentage(row?.completionStatus)"
|
:color="progressColor(toProgressPercentage(row?.completionStatus))"
|
:status="toProgressPercentage(row?.completionStatus) >= 100 ? 'success' : ''" />
|
</template>
|
</el-table-column>
|
<el-table-column label="计划开始时间"
|
prop="planStartTime"
|
width="120" />
|
<el-table-column label="计划结束时间"
|
prop="planEndTime"
|
width="120" />
|
<el-table-column label="实际开始时间"
|
prop="actualStartTime"
|
width="120" />
|
<el-table-column label="实际结束时间"
|
prop="actualEndTime"
|
width="120" />
|
<el-table-column label="指定报工人"
|
prop="userNames"
|
width="120"
|
show-overflow-tooltip />
|
<el-table-column label="操作"
|
width="210"
|
align="center"
|
fixed="right">
|
<template #default="{ row }">
|
<el-button link
|
type="primary"
|
@click="handleEdit(row)">计划时间</el-button>
|
<el-button link
|
type="primary"
|
@click="handleAssignReporter(row)">指定报工人</el-button>
|
</template>
|
</el-table-column>
|
</el-table>
|
</template>
|
</el-table-column>
|
<el-table-column align="center"
|
label="序号"
|
type="index"
|
width="60" />
|
<el-table-column label="生产订单号"
|
prop="npsNo"
|
width="160"
|
show-overflow-tooltip />
|
<el-table-column label="产品名称"
|
prop="productName"
|
width="140"
|
show-overflow-tooltip />
|
<el-table-column label="规格"
|
prop="model"
|
show-overflow-tooltip />
|
<el-table-column label="工序名称"
|
prop="operationName"
|
minWidth="160"
|
show-overflow-tooltip />
|
<el-table-column label="计划数量"
|
prop="planQuantity"
|
width="100" />
|
<el-table-column label="完成数量"
|
prop="completeQuantity"
|
width="100" />
|
<el-table-column label="完成进度"
|
width="140"
|
align="center">
|
<template #default="{ row }">
|
<el-progress :percentage="toProgressPercentage(row?.completionStatus)"
|
:color="progressColor(toProgressPercentage(row?.completionStatus))"
|
:status="toProgressPercentage(row?.completionStatus) >= 100 ? 'success' : ''" />
|
</template>
|
</el-table-column>
|
</el-table>
|
<pagination v-show="page.total > 0"
|
:total="page.total"
|
layout="total, sizes, prev, pager, next, jumper"
|
:page="page.current"
|
:limit="page.size"
|
@pagination="paginationChange" />
|
</div>
|
<el-dialog v-model="editDialogVisible"
|
title="编辑计划时间"
|
width="500px">
|
<el-form :model="editrow"
|
label-width="120px">
|
<el-form-item label="计划开始时间">
|
<el-date-picker v-model="editrow.planStartTime"
|
type="date"
|
placeholder="请选择"
|
value-format="YYYY-MM-DD"
|
style="width: 300px" />
|
</el-form-item>
|
<el-form-item label="计划结束时间">
|
<el-date-picker v-model="editrow.planEndTime"
|
type="date"
|
placeholder="请选择"
|
value-format="YYYY-MM-DD"
|
style="width: 300px" />
|
</el-form-item>
|
</el-form>
|
<template #footer>
|
<span class="dialog-footer">
|
<el-button type="primary"
|
@click="handleUpdate">确定</el-button>
|
<el-button @click="editDialogVisible = false">取消</el-button>
|
</span>
|
</template>
|
</el-dialog>
|
<!-- 指定报工人弹窗 -->
|
<el-dialog v-model="assignReporterDialogVisible"
|
title="指定报工人"
|
width="800px">
|
<div class="assign-reporter-content">
|
<div class="selected-tags-box"
|
v-if="selectedEmployeeIds.length > 0">
|
<div class="tags-label">已选择:</div>
|
<div class="tags-list">
|
<el-tag v-for="id in selectedEmployeeIds"
|
:key="id"
|
closable
|
@close="removeEmployeeTag(id)"
|
class="employee-tag">
|
{{ getEmployeeNameById(id) }}
|
</el-tag>
|
</div>
|
</div>
|
<div class="employee-list-container"
|
v-loading="employeeTableLoading">
|
<el-checkbox-group v-model="selectedEmployeeIds">
|
<div class="employee-grid">
|
<div v-for="item in employeeTableData"
|
:key="item.userId"
|
class="employee-item">
|
<el-checkbox :label="item.userId"
|
border>
|
<div class="employee-info">
|
<span class="name">{{ item.nickName }}</span>
|
<span class="dept">{{ item.dept?.deptName }}</span>
|
</div>
|
</el-checkbox>
|
</div>
|
</div>
|
</el-checkbox-group>
|
<div v-if="employeeTableData.length === 0"
|
class="empty-text">
|
暂无匹配人员
|
</div>
|
</div>
|
</div>
|
<template #footer>
|
<span class="dialog-footer">
|
<el-button type="primary"
|
@click="handleSaveReporters">确定</el-button>
|
<el-button @click="assignReporterDialogVisible = false">取消</el-button>
|
</span>
|
</template>
|
</el-dialog>
|
</div>
|
</template>
|
|
<script setup>
|
import { getCurrentInstance, onMounted, reactive, ref, toRefs } from "vue";
|
import { ElMessageBox } from "element-plus";
|
import {
|
productWorkOrderPage,
|
updateProductWorkOrder,
|
assignProductWorkOrder,
|
getWorkOrdersByOrderId,
|
} from "@/api/productionManagement/workOrder.js";
|
import { listUser } from "@/api/system/user.js";
|
import pagination from "@/components/PIMTable/Pagination.vue";
|
|
const { proxy } = getCurrentInstance();
|
|
const tableData = ref([]);
|
const tableLoading = ref(false);
|
const expandedRowKeys = ref([]);
|
const editDialogVisible = ref(false);
|
const editrow = ref(null);
|
const page = reactive({
|
current: 1,
|
size: 10,
|
total: 0,
|
});
|
|
// 指定报工人相关
|
const assignReporterDialogVisible = ref(false);
|
const employeeTableLoading = ref(false);
|
const employeeTableData = ref([]);
|
const employeePage = reactive({
|
current: 1,
|
size: 100,
|
total: 0,
|
});
|
const employeeSearchForm = reactive({
|
staffName: "",
|
});
|
const selectedEmployeeIds = ref([]);
|
const currentWorkOrder = ref(null);
|
|
const data = reactive({
|
searchForm: {
|
npsNo: "",
|
productName: "",
|
model: "",
|
},
|
});
|
const { searchForm } = toRefs(data);
|
|
const toProgressPercentage = val => {
|
const n = Number(val);
|
if (!Number.isFinite(n)) return 0;
|
if (n <= 0) return 0;
|
if (n >= 100) return 100;
|
return parseFloat(n.toFixed(2));
|
};
|
|
const progressColor = percentage => {
|
const p = toProgressPercentage(percentage);
|
if (p < 30) return "#f56c6c";
|
if (p < 50) return "#e6a23c";
|
if (p < 80) return "#409eff";
|
return "#67c23a";
|
};
|
|
const handleQuery = () => {
|
page.current = 1;
|
expandedRowKeys.value = [];
|
getList();
|
};
|
|
const paginationChange = obj => {
|
page.current = obj.page;
|
page.size = obj.limit;
|
expandedRowKeys.value = [];
|
getList();
|
};
|
|
const getList = () => {
|
tableLoading.value = true;
|
const params = { ...searchForm.value, current: page.current, size: page.size };
|
productWorkOrderPage(params)
|
.then(res => {
|
tableLoading.value = false;
|
const records = res.data.records || [];
|
records.forEach(item => {
|
item.children = [];
|
item.childrenLoading = false;
|
});
|
tableData.value = records;
|
page.total = res.data.total;
|
})
|
.catch(() => {
|
tableLoading.value = false;
|
});
|
};
|
|
const expandChange = (row, expandedRows) => {
|
if (expandedRows.length > 0) {
|
expandedRowKeys.value = [];
|
row.childrenLoading = true;
|
getWorkOrdersByOrderId(row.productionOrderId)
|
.then(res => {
|
row.childrenLoading = false;
|
const idx = tableData.value.findIndex(
|
item => item.productionOrderId === row.productionOrderId
|
);
|
if (idx > -1) {
|
tableData.value[idx].children = res.data || [];
|
}
|
expandedRowKeys.value.push(row.productionOrderId);
|
})
|
.catch(() => {
|
row.childrenLoading = false;
|
});
|
} else {
|
expandedRowKeys.value = [];
|
}
|
};
|
|
const handleEdit = row => {
|
editrow.value = JSON.parse(JSON.stringify(row));
|
editDialogVisible.value = true;
|
};
|
|
const handleUpdate = () => {
|
updateProductWorkOrder(editrow.value)
|
.then(() => {
|
proxy.$modal.msgSuccess("提交成功");
|
editDialogVisible.value = false;
|
getList();
|
})
|
.catch(() => {
|
ElMessageBox.alert("修改失败", "提示", {
|
confirmButtonText: "确定",
|
});
|
});
|
};
|
|
const handleAssignReporter = row => {
|
currentWorkOrder.value = row;
|
assignReporterDialogVisible.value = true;
|
if (row.userIds) {
|
try {
|
selectedEmployeeIds.value = JSON.parse(row.userIds);
|
} catch (e) {
|
selectedEmployeeIds.value = [];
|
}
|
} else {
|
selectedEmployeeIds.value = [];
|
}
|
employeeSearchForm.staffName = "";
|
getEmployeeList();
|
};
|
|
const getEmployeeList = () => {
|
employeeTableLoading.value = true;
|
const params = {
|
pageNum: 1,
|
pageSize: 100,
|
};
|
listUser(params)
|
.then(res => {
|
employeeTableLoading.value = false;
|
employeeTableData.value = res.rows;
|
employeePage.total = res.total;
|
})
|
.catch(() => {
|
employeeTableLoading.value = false;
|
});
|
};
|
|
const getEmployeeNameById = id => {
|
const employee = employeeTableData.value.find(item => item.userId === id);
|
return employee ? employee.nickName : id;
|
};
|
|
const removeEmployeeTag = id => {
|
selectedEmployeeIds.value = selectedEmployeeIds.value.filter(
|
item => item !== id
|
);
|
};
|
|
const handleSaveReporters = () => {
|
if (selectedEmployeeIds.value.length === 0) {
|
proxy.$modal.msgWarning("请选择报工人");
|
return;
|
}
|
|
const updateData = {
|
id: currentWorkOrder.value.id,
|
userIds: JSON.stringify(selectedEmployeeIds.value),
|
};
|
|
assignProductWorkOrder(updateData)
|
.then(() => {
|
proxy.$modal.msgSuccess("指定成功");
|
assignReporterDialogVisible.value = false;
|
getList();
|
})
|
.catch(() => {
|
proxy.$modal.msgError("指定失败");
|
});
|
};
|
|
onMounted(() => {
|
getList();
|
});
|
</script>
|
|
<style scoped lang="scss">
|
.search-row {
|
display: flex;
|
align-items: center;
|
gap: 12px;
|
}
|
.search-item {
|
display: flex;
|
align-items: center;
|
}
|
.search_title {
|
margin-right: 8px;
|
font-size: 14px;
|
color: #606266;
|
}
|
.assign-reporter-content {
|
.selected-tags-box {
|
margin-bottom: 16px;
|
padding: 12px;
|
background-color: #f5f7fa;
|
border-radius: 4px;
|
display: flex;
|
align-items: flex-start;
|
|
.tags-label {
|
font-size: 14px;
|
color: #606266;
|
margin-right: 8px;
|
white-space: nowrap;
|
margin-top: 4px;
|
}
|
|
.tags-list {
|
display: flex;
|
flex-wrap: wrap;
|
gap: 8px;
|
|
.employee-tag {
|
margin-bottom: 4px;
|
}
|
}
|
}
|
|
.employee-list-container {
|
max-height: 400px;
|
overflow-y: auto;
|
padding: 10px;
|
border: 1px solid #f0f0f0;
|
border-radius: 4px;
|
|
.employee-grid {
|
display: grid;
|
grid-template-columns: repeat(auto-fill, minmax(160px, 1fr));
|
gap: 12px;
|
}
|
|
.employee-item {
|
:deep(.el-checkbox) {
|
width: 100%;
|
margin-right: 0;
|
height: auto;
|
padding: 8px;
|
|
.el-checkbox__label {
|
width: 100%;
|
}
|
}
|
|
.employee-info {
|
display: flex;
|
flex-direction: column;
|
gap: 4px;
|
|
.name {
|
font-weight: bold;
|
font-size: 14px;
|
color: #303133;
|
}
|
|
.dept {
|
font-size: 12px;
|
color: #909399;
|
}
|
}
|
}
|
|
.empty-text {
|
text-align: center;
|
color: #909399;
|
padding: 20px;
|
}
|
}
|
}
|
</style>
|