heli-mes/mes-ui/mes-ui-admin-vue3/src/views/heli/materialplan/standardbuy.vue

645 lines
21 KiB
Vue
Raw Normal View History

2025-06-24 13:43:23 +08:00
<template>
<el-card class="hl-card">
<template #header>
2025-06-25 16:03:35 +08:00
<span>标准件采购</span>
2025-06-24 13:43:23 +08:00
</template>
<ContentWrap class="borderxx">
<!-- 搜索工作栏 -->
<el-form class="-mb-15px" :model="queryParams" ref="queryFormRef" :inline="true" label-width="120px">
2025-10-15 19:14:42 +08:00
<el-form-item label="项目编码" prop="projectCode">
<el-input v-model="queryParams.projectCode" placeholder="请输入项目编号" clearable @keyup.enter="handleQuery" class="!w-240px" />
</el-form-item>
2025-06-24 13:43:23 +08:00
<el-form-item label="项目名称" prop="projectName">
<el-input v-model="queryParams.projectName" placeholder="请输入项目名称" clearable @keyup.enter="handleQuery" class="!w-240px" />
</el-form-item>
2025-10-15 19:14:42 +08:00
<el-form-item label="客户简称" prop="customerName">
<el-input v-model="queryParams.customerName" placeholder="请输入客户简称" clearable @keyup.enter="handleQuery" class="!w-240px" />
</el-form-item>
2025-06-24 13:43:23 +08:00
<el-form-item label="子项目名称" prop="projectSubName">
<el-input v-model="queryParams.projectSubName" placeholder="请输入子项目名称" clearable @keyup.enter="handleQuery" class="!w-240px" />
</el-form-item>
<el-form-item label="零件名称" prop="materialName">
<el-input v-model="queryParams.materialName" placeholder="请输入零件名称" clearable @keyup.enter="handleQuery" class="!w-240px" />
</el-form-item>
2025-10-15 19:14:42 +08:00
<el-form-item label="采购状态" prop="mplanStatus">
<el-select @change="e => clearItem(e)" v-model="queryParams.mplanStatus" placeholder="请选择标准件状态" class="!w-240px">
<el-option v-for="dict in getIntDictOptions(DICT_TYPE.HELI_STANDARD_BUY_TYPE)" :key="dict.value" :label="dict.label" :value="dict.value" />
</el-select>
2025-06-24 13:43:23 +08:00
</el-form-item>
<el-form-item label="责任人" prop="ownerName">
<el-input
v-model="queryParams.ownerName"
placeholder="请输入责任人"
clearable
@keyup.enter="handleQuery"
class="!w-240px"
/>
</el-form-item>
2025-10-15 19:14:42 +08:00
2025-06-24 13:43:23 +08:00
<el-form-item label="物料需求单号" prop="projectMaterialPlanNo">
<el-input v-model="queryParams.projectMaterialPlanNo" placeholder="请输入物料需求单号" clearable @keyup.enter="handleQuery" class="!w-240px" />
</el-form-item>
2025-08-27 18:07:43 +08:00
<el-form-item label="供应商" prop="supplierName">
<el-input v-model="queryParams.supplierName" placeholder="请输入供应商" clearable @keyup.enter="handleQuery" class="!w-240px" />
</el-form-item>
2025-06-24 13:43:23 +08:00
<el-form-item style="margin-left:30px">
<el-button @click="handleQuery" type="primary">
<Icon icon="ep:search" class="mr-5px" /> 搜索
</el-button>
<el-button @click="resetQuery">
<Icon icon="ep:refresh" class="mr-5px" /> 重置
</el-button>
<el-button
type="success"
plain
@click="handleExportDetail"
:loading="exportLoading"
>
<Icon icon="ep:download" class="mr-5px" /> 导出
</el-button>
</el-form-item>
</el-form>
</ContentWrap>
<!-- 列表 -->
<el-card class="hl-card-info">
<template #header>
<div class="hl-card-info-icona"></div><span class="hl-card-info-text">标准件明细</span>
<el-button style="margin-left: 20px" @click="opearteForm(0)" type="success" size="large"> </el-button>
<el-button style="margin-left: 20px" @click="opearteForm(1)" type="warning" size="large">去库存</el-button>
2025-08-04 15:43:45 +08:00
<label style="display: inline-flex; align-items: center; margin-left: 20px;">
<span style="order: 1;">同时操作</span>
<el-checkbox class="large-checkbox" v-model="flag" style="order: 2; margin-left: 10px;"/>
</label>
2025-06-24 13:43:23 +08:00
</template>
<el-row>
<el-col>
<el-card class="hl-incard">
<el-form ref="subFormRef" :model="list" v-loading="formLoading" label-width="0" >
<el-table
:row-key="getRowKeys" ref="multipleTable"
2025-08-27 18:07:43 +08:00
@selection-change="handleSelectionChange" v-loading="loading" :data="list" :show-overflow-tooltip="true" class="hl-table" show-summary :summary-method="getSummaries">
2025-06-24 13:43:23 +08:00
<el-table-column
type="selection"
width="55"
:reserve-selection="true"
/>
<el-table-column fixed label="序号" align="center" type="index" width="60" />
<el-table-column label="客户简称" align="center" prop="customerBriefName" min-width="120" fixed/>
<el-table-column label="项目名称" align="center" prop="projectName" min-width="150" fixed/>
<el-table-column label="子项目名称" align="center" prop="projectSubName" min-width="150" fixed/>
<el-table-column label="零件名称" align="center" prop="matName" min-width="150" />
<el-table-column label="材质" align="center" prop="compositionName" min-width="150" />
<el-table-column label="规格型号" align="center" prop="matSpec" min-width="150" />
<el-table-column label="图号" align="center" prop="blueprintNo" min-width="80" />
<el-table-column label="需求数量" align="center" prop="boomAmount" min-width="100" />
<el-table-column label="单位" align="center" prop="matUnit" min-width="100">
<template #default="scope">
<dict-tag :type="DICT_TYPE.HELI_MATERIAL_UNIT" :value="scope.row.matUnit" />
</template>
</el-table-column>
2025-07-02 14:45:30 +08:00
<el-table-column label="物料编码" align="center" prop="materialName" min-width="180" >
2025-06-25 16:03:35 +08:00
2025-06-24 13:43:23 +08:00
<template #default="scope">
2025-07-02 14:45:30 +08:00
<el-input v-model="scope.row.materialName" disabled>
2025-06-24 13:43:23 +08:00
<template #suffix>
<Icon @click="serachLog(scope.row)" icon="ep:search" color="primary"/>
</template>
</el-input>
</template>
</el-table-column>
<el-table-column label="库存量" align="center" prop="matRest" min-width="120" />
2025-06-25 16:03:35 +08:00
<el-table-column align="center" prop="supplierId" min-width="180" >
<template #header><span class="hl-table_header">*</span>供应商</template>
2025-06-24 13:43:23 +08:00
<template #default="scope">
<el-select
v-model="scope.row.supplierId"
size="large"
clearable
filterable
2025-08-04 15:43:45 +08:00
@change="change(scope.row)"
2025-06-24 13:43:23 +08:00
style="width: 165 px"
>
<el-option
v-for="item in supplierList"
:key="item.id"
2025-08-04 15:43:45 +08:00
:label="item.brief"
2025-06-24 13:43:23 +08:00
:value="item.id"
/>
</el-select>
</template>
</el-table-column>
<el-table-column align="center" prop="purchaseAmount" min-width="180" >
<template #header><span class="hl-table_header">*</span>采购数量</template>
<template #default="scope">
2025-07-02 09:47:57 +08:00
<el-input-number :disabled="scope.row.stat" v-model="scope.row.purchaseAmount" :precision="2" :step="0.01" />
</template>
2025-06-24 13:43:23 +08:00
</el-table-column>
2025-06-25 16:03:35 +08:00
<el-table-column align="center" prop="estimatedPrice" min-width="180" >
<template #header><span class="hl-table_header">*</span>预估总价</template>
2025-06-24 13:43:23 +08:00
<template #default="scope">
2025-07-02 09:47:57 +08:00
<el-input-number v-model="scope.row.estimatedPrice" :precision="2" :step="0.01" />
2025-06-24 13:43:23 +08:00
</template>
</el-table-column>
<el-table-column min-width="200px" align="center" >
2025-06-24 13:43:23 +08:00
<template #header><span class="hl-table_header">*</span>预计到货日期</template>
<template #default="{ row, $index }">
<el-form-item :prop="`${$index}.arriveTime`" class="mb-0px!" >
2025-08-04 15:43:45 +08:00
<el-date-picker class="!w-265px" v-model="row.arriveTime" type="date" value-format="x" placeholder="预计到货日期" @change="change1(row)"/>
2025-06-24 13:43:23 +08:00
</el-form-item>
</template>
</el-table-column>
<el-table-column label="采购单号" align="center" prop="purchaseNo" min-width="180" />
<el-table-column label="状态" align="center" prop="mplanStatus" min-width="90" >
<template #default="scope">
<dict-tag :type="DICT_TYPE.HELI_PROJECT_MATERIAL_PLAN_BOOM_STATUS" :value="scope.row.mplanStatus" />
</template>
</el-table-column>
2025-06-24 13:43:23 +08:00
</el-table>
<!-- 分页 -->
<Pagination :total="total" v-model:page="queryParams.pageNo" v-model:limit="queryParams.pageSize" @pagination="getList" />
</el-form>
</el-card>
</el-col>
</el-row>
</el-card>
<div class="hl-footer text-center">
</div>
</el-card>
<matLog ref="matLogRef" @success="getMat" />
</template>
<script setup lang="ts">
import { getIntDictOptions, DICT_TYPE } from '@/utils/dict'
import download from '@/utils/download'
import * as materialPlanBoomApi from '@/api/heli/materialPlanBoom'
import { useCommonStateWithOut } from '@/store/modules/common'
import { useUserStore } from '@/store/modules/user'
import matLog from './storageLog.vue'
import * as supplierApi from '@/api/heli/supplier'
import {inject, ref} from "vue";
const formLoading = ref(false) // 表单的加载中1修改时的数据加载2提交的按钮禁用
const userStore = useUserStore()
const username = userStore.getUser.nickname
2025-09-05 08:19:16 +08:00
defineOptions({ name: 'standardbuy' })
2025-06-24 13:43:23 +08:00
const reload: any = inject('reload')
const commonStore = useCommonStateWithOut()
const message = useMessage() // 消息弹窗
const { t } = useI18n() // 国际化
const router = useRouter()
const matLogRef = ref()
const loading = ref(true) // 列表的加载中
const list = ref([]) // 列表的数据
const total = ref(0) // 列表的总页数
2025-08-04 15:43:45 +08:00
const flag = ref(true) // 列表的加载中
2025-06-24 13:43:23 +08:00
const queryParams = reactive({
pageNo: 1,
pageSize: 10,
code: undefined,
planCode: undefined,
projectCode: undefined,
customerName: undefined,
projectName: undefined,
projectSubName: undefined,
version: undefined,
bomStatus: undefined,
remark: undefined,
mplanStatus: 0,
createTime: [],
ownerName:username,
materialName:undefined,
2025-08-27 18:07:43 +08:00
projectMaterialPlanNo:undefined,
supplierName:undefined,
2025-06-24 13:43:23 +08:00
})
const queryParams1 = reactive({
pageNo: 1,
pageSize: 999,
})
const queryFormRef = ref() // 搜索的表单
const exportLoading = ref(false) // 导出的加载中
/** 查询列表 */
const getList = async () => {
loading.value = true
try {
multipleTable.value.clearSelection();
2025-07-10 21:12:33 +08:00
const idid = history.state.idid;
if(idid){
queryParams.projectMaterialPlanNo=idid
}
2025-06-24 13:43:23 +08:00
const data = await materialPlanBoomApi.getStandardBuyPage(queryParams)
2025-07-10 21:12:33 +08:00
history.state.idid=undefined
2025-06-24 13:43:23 +08:00
list.value = data.list
total.value = data.total
} finally {
loading.value = false
}
}
const ids = ref([])
const multipleTable=ref()
const getRowKeys=(row)=>{
return row.id;
}
const handleSelectionChange = (selection) => {
ids.value = selection;
2025-08-04 15:43:45 +08:00
var bomDetails = list.value;
if (selection.length==0) {
for (let i = 0; i < bomDetails.length; i++) {
bomDetails[i].chkboxEnable=false
}
}else if (selection.length== bomDetails.length) {
for (let i = 0; i < bomDetails.length; i++) {
bomDetails[i].chkboxEnable=true
}
}else {
for (let i = 0; i < bomDetails.length; i++) {
for (let j = 0; j < row.length; j++) {
if (val[j].id==bomDetails[i].id){
bomDetails[i].chkboxEnable=true
break
}else {
bomDetails[i].chkboxEnable=false
}
}
}
}
}
2025-08-27 18:07:43 +08:00
const getSummaries = (param: SummaryMethodProps) => {
const summaryField = ["boomAmount","purchaseAmount","estimatedPrice"];
const { columns, data } = param;
const sums = [];
columns.forEach((column, index) => {
if (index === 0) {
sums[index] = '合计';
return;
}
if (!summaryField.includes(column.property)) {
sums[index] = '';
return;
}
const values = data.map(item => Number(item[column.property]));
if (!values.every(value => isNaN(value))) {
sums[index] = values.reduce((prev, curr) => {
const value = Number(curr);
if (!isNaN(value)) {
return floatAdd(prev, curr);
} else {
return prev;
}
}, 0);
const thousandsField = [];
if (thousandsField.includes(column.property)) {
sums[index] = thousandsFormat(sums[index]);
}
} else {
sums[index] = '';
}
});
// const sumValue = computed(() => {
// return list.value.reduce((sum, item) => sum + item.quantityLent, 0);
// });
// const sums = ['合计','','','','','','','',sumValue];
return sums;
}
const floatAdd = (a, b) => {
var c, d, e;
if (undefined === a || null === a || "" === a || isNaN(a)) { a = 0; }
if (undefined === b || null === b || "" === b || isNaN(b)) { b = 0; }
try {
c = a.toString().split(".")[1].length;
} catch (f) {
c = 0;
}
try {
d = b.toString().split(".")[1].length;
} catch (f) {
d = 0;
}
e = Math.pow(10, Math.max(c, d));
return (floatMul(a, e) + floatMul(b, e)) / e;
};
const floatMul = (a, b) => {
var c = 0,
d = a.toString(),
e = b.toString();
try {
c += d.split(".")[1].length;
} catch (f) {}
try {
c += e.split(".")[1].length;
} catch (f) {}
return Number(d.replace(".", "")) * Number(e.replace(".", "")) / Math.pow(10, c);
}
2025-08-04 15:43:45 +08:00
const change =(row:any)=>{
if (flag.value) {
var bomDetails = list.value;
if (row.chkboxEnable) {
for (let i = 0; i < bomDetails.length; i++) {
if (bomDetails[i].chkboxEnable) {
bomDetails[i].supplierId = row.supplierId
}
}
}
}
}
const change1 =(row:any)=>{
if (flag.value) {
var bomDetails = list.value;
if (row.chkboxEnable) {
for (let i = 0; i < bomDetails.length; i++) {
if (bomDetails[i].chkboxEnable) {
bomDetails[i].arriveTime = row.arriveTime
}
}
}
}
2025-06-24 13:43:23 +08:00
}
2025-06-26 14:37:58 +08:00
const clearItem = (e) =>{
multipleTable.value.clearSelection();
}
2025-06-24 13:43:23 +08:00
/** 导出按钮操作 */
const handleExportDetail = async () => {
try {
// 导出的二次确认
await message.exportConfirm()
// 发起导出
exportLoading.value = true
const data = await materialPlanBoomApi.exportMaterialPlanBoom(queryParams)
download.excel(data, '标准件物料需求计划.xlsx')
} catch {
} finally {
exportLoading.value = false
}
}
2025-07-05 02:52:44 +08:00
const getMat = async (data) => {
2025-06-24 13:43:23 +08:00
//formData.value.boomItemDOList = arrBoom
for(var i = 0 ; i < list.value.length ; i ++){
2025-07-05 02:52:44 +08:00
if(list.value[i].id == data.rowid){
list.value[i].matRest = data.matRest
list.value[i].matCode = data.matCode
list.value[i].materialId = data.materialId
list.value[i].materialName =data.matName
// await updateRow(2,list.value[i]);
2025-06-24 13:43:23 +08:00
break;
}
}
}
const changeSupp = async(e,row) => {
if(e){
await updateRow(4,row)
}
}
const changeDate = async(e,row) => {
if(e){
await updateRow(5,row)
}
}
const serachLog = (row) => {
matLogRef.value.open(row.id,row.matName)
2025-06-24 13:43:23 +08:00
}
/** 搜索按钮操作 */
const handleQuery = () => {
queryParams.pageNo = 1
getList()
}
const opearteForm = async(type) =>{
if(ids.value != null && ids.value.length > 0){
loading.value = true
2025-06-24 13:43:23 +08:00
if(type == 0){
//送审
var isError = false;
var sets = new Set();
for(var i = 0 ; i<ids.value.length ;i++){
var row = ids.value[i];
if(!row.purchaseAmount && row.purchaseAmount != 0){
2025-06-24 13:43:23 +08:00
message.error('零件:'+row.matName+ '采购数量不能为空!')
loading.value = false
2025-06-24 13:43:23 +08:00
return
} else if( row.purchaseAmount == 0){
message.error('零件:'+row.matName+ '采购数量不能为0')
loading.value = false
2025-06-24 13:43:23 +08:00
return
} else if(!row.supplierId){
message.error('零件:'+row.matName+ '供应商不能为空!')
loading.value = false
2025-06-24 13:43:23 +08:00
return
} else if(!row.estimatedPrice && row.estimatedPrice != 0){
2025-06-25 16:03:35 +08:00
message.error('零件:'+row.matName+ '预估总价不能为空!')
loading.value = false
2025-06-24 13:43:23 +08:00
return
} else if( row.estimatedPrice == 0){
message.error('零件:'+row.matName+ '预估总价不能为0')
loading.value = false
2025-06-24 13:43:23 +08:00
return
} else if(!row.arriveTime ){
2025-06-25 16:03:35 +08:00
message.error('零件:'+row.matName+ '预计到货日期不能为空!')
loading.value = false
2025-06-24 13:43:23 +08:00
return
}
if( !sets.has(row.projectId) || sets == null ){
sets.add(row.projectId)
}
}
if(sets != null && sets.size > 1){
2025-06-25 16:03:35 +08:00
message.error('不同项目不能同时生成采购单');
loading.value = false
2025-06-24 13:43:23 +08:00
return
}
var form = {
id:1,
projectMaterialPlanId:1,
boomDetailId:1,
projectPlanSubId:1,
materialPlanBoomDOList:ids.value,
buyType:0
}
try{
2025-07-02 09:47:57 +08:00
console.log(111006)
await materialPlanBoomApi.updateMaterialPlanBoom(form);
message.success("送审成功")
await getList()
}catch{
loading.value = false
}finally{
loading.value = false
}
2025-06-24 13:43:23 +08:00
}else{
//去库存
// for(var i = 0 ; i<ids.value.length ;i++){
// var row = ids.value[i];
// if(!row.materialId){
// message.error('请先查询库存量后进行操作')
// loading.value = false
// return
// }
// }
console.log(ids.value)
2025-06-24 13:43:23 +08:00
var form = {
id:1,
projectMaterialPlanId:1,
boomDetailId:1,
projectPlanSubId:1,
materialPlanBoomDOList:ids.value,
buyType:2
}
2025-06-25 16:03:35 +08:00
try{
await materialPlanBoomApi.updateMaterialPlanBoom(form);
2025-06-24 13:43:23 +08:00
message.success("去库存成功")
await getList()
}catch{
loading.value = false
2025-06-25 16:03:35 +08:00
}finally{
loading.value = false
}
2025-06-24 13:43:23 +08:00
}
}else{
message.error("至少选择一项后进行操作!")
loading.value = false
2025-06-24 13:43:23 +08:00
return ;
}
}
const updateRow = async(type,row) =>{
if(type == 1){
if(row.purchaseAmount != null){
await materialPlanBoomApi.updateMaterialPlanBoom(row)
}
}else if(type == 2){
if(row.matRest != null){
await materialPlanBoomApi.updateMaterialPlanBoom(row)
}
}else if(type == 3){
if(row.estimatedPrice != null){
await materialPlanBoomApi.updateMaterialPlanBoom(row)
}
}else if(type == 4){
if(row.supplierId){
await materialPlanBoomApi.updateMaterialPlanBoom(row)
}
}else if(type == 5){
if(row.arriveTime){
await materialPlanBoomApi.updateMaterialPlanBoom(row)
}
}
}
const supplierList = ref([])
const supplier = async() =>{
var data = await supplierApi.getSupplierPage(queryParams1);
supplierList.value =data.list
//console.log(supplierList/)
}
const handleSelectedUser = (currentIndex, newValue: any) => {
list.value[currentIndex].duEmpId = newValue?.id
}
const submitForm = async () => {
const filteredList = list.value.filter(item => item.duEmpId !== null && item.duEmpId !== undefined);
if (filteredList.length <= 0) {
message.error("提交明细不能为空,请确认");
return;
}
const firstProjectSubId = filteredList[0].projectSubId;
for (let i = 1; i < filteredList.length; i++) {
if (filteredList[i].projectSubId !== firstProjectSubId) {
message.error("标准件明细不属于同一个子项目,请确认");
return;
}
}
for (let i = 0; i < filteredList.length; i++) {
2025-07-02 09:47:57 +08:00
if (filteredList[i].projectMaterialPlanNo!=null&&filteredList[i].projectMaterialPlanNo!=''){
2025-06-24 13:43:23 +08:00
message.error("零件"+filteredList[i].materialName+"已生成物料需求计划,请确认")
return
}
2025-07-02 09:47:57 +08:00
if (filteredList[i].boomArriveDate==null&&filteredList[i].boomArriveDate!=''){
2025-06-24 13:43:23 +08:00
message.error("零件"+filteredList[i].materialName+"要求完成日期为空,请确认")
return
}
2025-07-02 09:47:57 +08:00
if (filteredList[i].duEmpId==null&&filteredList[i].duEmpId!=''){
2025-06-24 13:43:23 +08:00
message.error("零件"+filteredList[i].materialName+"责任人为空,请确认")
return
}
}
formLoading.value = true
try {
await ProcessBomApi.submitForm(filteredList)
message.success("提交成功")
// 发送操作成功的事件
getList()
emit('success')
} finally {
formLoading.value = false
}
}
const withdraw = async () => {
if (queryParams.projectMaterialPlanNo==null){
message.error("物料需求单号为空,不允许撤回")
return
}
await ProcessBomApi.withdraw(queryParams.projectMaterialPlanNo)
message.success("撤回成功")
// 发送操作成功的事件
getList()
emit('success')
}
/** 重置按钮操作 */
const resetQuery = () => {
queryFormRef.value.resetFields()
handleQuery()
}
/** 导出按钮操作 */
const handleExport = async () => {
try {
// 导出的二次确认
await message.exportConfirm()
// 发起导出
exportLoading.value = true
// const data = await ProcessBomApi.exportProcessBom(queryParams)
// download.excel(data, '工艺bom.xls')
2025-06-24 13:43:23 +08:00
} catch {
} finally {
exportLoading.value = false
}
}
/** 初始化 **/
onMounted( async() => {
await supplier()
await getList()
})
</script>
<style>
2025-08-04 15:43:45 +08:00
.large-checkbox {
transform: scale(1.5); /* 方法1使用scale放大 */
/* 或者 */
/* 方法2通过调整字体大小 */
/* font-size: 18px; */
}
2025-06-24 13:43:23 +08:00
</style>