| | |
| | | > |
| | | </div> |
| | | <div> |
| | | <el-button type="primary" @click="handleAdd">新增</el-button> |
| | | <el-button type="primary" @click="handleBatchApprove">审批</el-button> |
| | | <el-button @click="handleOut">导出</el-button> |
| | | <el-button type="danger" plain @click="handleDelete">删除</el-button> |
| | |
| | | </el-tag> |
| | | </template> |
| | | </el-table-column> |
| | | <el-table-column label="操作" width="120" align="center" fixed="right"> |
| | | <template #default="scope"> |
| | | <el-button |
| | | v-if="scope.row.approvalStatus !== 1 && scope.row.approvalStatus !== '1' && scope.row.approvalStatus !== 'approved' && scope.row.approvalStatus !== 'APPROVED'" |
| | | link |
| | | type="primary" |
| | | size="small" |
| | | @click="handleEdit(scope.row)">编辑</el-button> |
| | | <span v-else style="color: #999; font-size: 12px;">已通过</span> |
| | | </template> |
| | | </el-table-column> |
| | | </el-table> |
| | | <pagination |
| | | v-show="total > 0" |
| | |
| | | @pagination="paginationChange" |
| | | /> |
| | | </div> |
| | | |
| | | <!-- 新增/编辑对话框 --> |
| | | <el-dialog v-model="dialogVisible" |
| | | :title="dialogTitle" |
| | | width="800" |
| | | @close="closeDialog"> |
| | | <el-form ref="formRef" |
| | | :model="formState" |
| | | label-width="140px" |
| | | label-position="top"> |
| | | <el-form-item label="产品名称" |
| | | prop="productModelId" |
| | | :rules="[ |
| | | { |
| | | required: true, |
| | | message: '请选择产品', |
| | | trigger: 'change', |
| | | } |
| | | ]"> |
| | | <el-button type="primary" |
| | | @click="showProductSelect = true"> |
| | | {{ formState.productName ? formState.productName : '选择产品' }} |
| | | </el-button> |
| | | </el-form-item> |
| | | <el-form-item label="规格" |
| | | prop="productModelName"> |
| | | <el-input v-model="formState.productModelName" disabled /> |
| | | </el-form-item> |
| | | <el-form-item label="单位" |
| | | prop="unit"> |
| | | <el-input v-model="formState.unit" disabled /> |
| | | </el-form-item> |
| | | <el-form-item label="库存类型" |
| | | prop="type" |
| | | :rules="[ |
| | | { |
| | | required: true, |
| | | message: '请选择库存类型', |
| | | trigger: 'change', |
| | | } |
| | | ]"> |
| | | <el-select v-model="formState.type" |
| | | placeholder="请选择库存类型" |
| | | :disabled="isEdit"> |
| | | <el-option label="合格库存" |
| | | value="qualified" /> |
| | | <el-option label="不合格库存" |
| | | value="unqualified" /> |
| | | </el-select> |
| | | </el-form-item> |
| | | <el-form-item label="出库数量" |
| | | prop="qualitity" |
| | | :rules="[ |
| | | { |
| | | required: true, |
| | | message: '请输入出库数量', |
| | | trigger: 'blur', |
| | | }, |
| | | { |
| | | validator: (rule, value, callback) => { |
| | | if (formState.maxStock > 0 && value > formState.maxStock) { |
| | | callback('出库数量不能超过当前批号库存 ' + formState.maxStock); |
| | | } else { |
| | | callback(); |
| | | } |
| | | }, |
| | | trigger: 'blur', |
| | | } |
| | | ]"> |
| | | <el-input-number v-model="formState.qualitity" |
| | | :step="1" |
| | | :min="1" |
| | | :max="formState.maxStock > 0 ? formState.maxStock : undefined" |
| | | style="width: 100%" /> |
| | | </el-form-item> |
| | | <el-form-item label="批号" |
| | | prop="batchNo" |
| | | :rules="[ |
| | | { |
| | | required: true, |
| | | message: '请选择批号', |
| | | trigger: 'change', |
| | | } |
| | | ]"> |
| | | <el-select v-model="formState.batchNo" |
| | | placeholder="请选择批号" |
| | | clearable |
| | | :disabled="isEdit" |
| | | @change="handleBatchNoChange" |
| | | style="width: 100%"> |
| | | <el-option v-for="batch in batchNoList" |
| | | :key="batch" |
| | | :label="batch + ' (库存: ' + (batchNoStockMap[batch] || 0) + ')'" |
| | | :value="batch" /> |
| | | </el-select> |
| | | </el-form-item> |
| | | <el-form-item v-if="formState.batchNo && batchNoStockMap[formState.batchNo]" |
| | | label="当前批号库存" |
| | | prop="currentStock"> |
| | | <el-input v-model="batchNoStockMap[formState.batchNo]" disabled /> |
| | | </el-form-item> |
| | | <el-form-item v-if="isEdit" |
| | | label="来源" |
| | | prop="recordType"> |
| | | <el-select v-model="formState.recordType" |
| | | placeholder="请选择来源" |
| | | disabled> |
| | | <el-option v-for="item in stockRecordTypeOptions" |
| | | :key="item.value" |
| | | :label="item.label" |
| | | :value="item.value" /> |
| | | </el-select> |
| | | </el-form-item> |
| | | <el-form-item label="备注" |
| | | prop="remark"> |
| | | <el-input v-model="formState.remark" |
| | | type="textarea" /> |
| | | </el-form-item> |
| | | </el-form> |
| | | <!-- 产品选择弹窗 --> |
| | | <ProductSelectDialog v-model="showProductSelect" |
| | | @confirm="handleProductSelect" |
| | | :top-product-parent-id="props.topParentProductId" |
| | | request-url="/basic/product/pageModelAndQua" |
| | | single /> |
| | | <template #footer> |
| | | <div class="dialog-footer"> |
| | | <el-button type="primary" @click="handleSubmit">确认</el-button> |
| | | <el-button @click="closeDialog">取消</el-button> |
| | | </div> |
| | | </template> |
| | | </el-dialog> |
| | | </div> |
| | | </template> |
| | | |
| | | <script setup> |
| | | import pagination from "@/components/PIMTable/Pagination.vue"; |
| | | import { ref } from "vue"; |
| | | import { ElMessageBox } from "element-plus"; |
| | | import ProductSelectDialog from "@/views/basicData/product/ProductSelectDialog.vue"; |
| | | import { ref, reactive, toRefs, computed, getCurrentInstance, watch, onMounted } from "vue"; |
| | | import { ElMessageBox, ElMessage } from "element-plus"; |
| | | import useUserStore from "@/store/modules/user"; |
| | | import { getCurrentDate } from "@/utils/index.js"; |
| | | import { |
| | | getStockOutPage, |
| | | delPendingStockOut, |
| | | batchApproveStockOutRecords, |
| | | updateStockOutRecord, |
| | | } from "@/api/inventoryManagement/stockOut.js"; |
| | | import { |
| | | findAllQualifiedStockOutRecordTypeOptions, |
| | | findAllUnQualifiedStockOutRecordTypeOptions, |
| | | } from "@/api/basicData/enum.js"; |
| | | import { addStockOutRecordOnly } from "@/api/inventoryManagement/stockInventory.js"; |
| | | import { addUnqualifiedStockOutRecordOnly } from "@/api/inventoryManagement/stockUninventory.js"; |
| | | |
| | | const userStore = useUserStore(); |
| | | const { proxy } = getCurrentInstance(); |
| | |
| | | const tableLoading = ref(false); |
| | | // 来源类型选项 |
| | | const stockRecordTypeOptions = ref([]); |
| | | // 批号列表(从batchNoMaps获取) |
| | | const batchNoList = ref([]); |
| | | // 批号库存映射 |
| | | const batchNoStockMap = ref({}); |
| | | const page = reactive({ |
| | | current: 1, |
| | | size: 100, |
| | |
| | | }, |
| | | }); |
| | | const { searchForm } = toRefs(data); |
| | | |
| | | // 对话框相关 |
| | | const dialogVisible = ref(false); |
| | | const dialogType = ref('add'); // 'add' 或 'edit' |
| | | const dialogTitle = computed(() => dialogType.value === 'add' ? '新增出库记录' : '编辑出库记录'); |
| | | const isEdit = computed(() => dialogType.value === 'edit'); |
| | | const formRef = ref(); |
| | | const showProductSelect = ref(false); |
| | | |
| | | // 表单数据 |
| | | const formState = ref({ |
| | | id: undefined, |
| | | productId: undefined, |
| | | productModelId: undefined, |
| | | productName: "", |
| | | productModelName: "", |
| | | unit: "", |
| | | type: undefined, |
| | | qualitity: 0, |
| | | batchNo: null, |
| | | recordType: "", |
| | | remark: "", |
| | | maxStock: 0, // 当前选中批号的最大库存 |
| | | }); |
| | | |
| | | // 批号为空时转为 null |
| | | watch( |
| | | () => formState.value.batchNo, |
| | | val => { |
| | | if (val === "") { |
| | | formState.value.batchNo = null; |
| | | } |
| | | } |
| | | ); |
| | | |
| | | // 查询列表 |
| | | /** 搜索按钮操作 */ |
| | |
| | | return "warning"; |
| | | }; |
| | | |
| | | // 新增 |
| | | const handleAdd = () => { |
| | | dialogType.value = 'add'; |
| | | resetForm(); |
| | | // 根据当前tab设置默认库存类型 |
| | | formState.value.type = props.type === '0' ? 'qualified' : 'unqualified'; |
| | | dialogVisible.value = true; |
| | | }; |
| | | |
| | | // 编辑 |
| | | const handleEdit = (row) => { |
| | | dialogType.value = 'edit'; |
| | | resetForm(); |
| | | // 填充表单数据 |
| | | formState.value = { |
| | | id: row.id, |
| | | productId: row.productId, |
| | | productModelId: row.productModelId, |
| | | productName: row.productName, |
| | | productModelName: row.model, |
| | | unit: row.unit, |
| | | type: props.type === '0' ? 'qualified' : 'unqualified', |
| | | qualitity: row.stockOutNum, |
| | | batchNo: row.batchNo, |
| | | recordType: row.recordType, |
| | | remark: row.remark || "", |
| | | maxStock: row.stockOutNum || 0, // 编辑时使用当前出库数量作为最大库存(因为是修改已有记录) |
| | | }; |
| | | // 编辑时从batchNoMaps获取批号列表 |
| | | if (row.batchNoMaps && Object.keys(row.batchNoMaps).length > 0) { |
| | | batchNoList.value = Object.keys(row.batchNoMaps); |
| | | batchNoStockMap.value = row.batchNoMaps; |
| | | } else if (row.batchNo) { |
| | | batchNoList.value = [row.batchNo]; |
| | | batchNoStockMap.value = { [row.batchNo]: row.stockOutNum || 0 }; |
| | | } else { |
| | | batchNoList.value = []; |
| | | batchNoStockMap.value = {}; |
| | | } |
| | | dialogVisible.value = true; |
| | | }; |
| | | |
| | | // 重置表单 |
| | | const resetForm = () => { |
| | | formState.value = { |
| | | id: undefined, |
| | | productId: undefined, |
| | | productModelId: undefined, |
| | | productName: "", |
| | | productModelName: "", |
| | | unit: "", |
| | | type: undefined, |
| | | qualitity: 0, |
| | | batchNo: null, |
| | | recordType: "", |
| | | remark: "", |
| | | maxStock: 0, |
| | | }; |
| | | batchNoList.value = []; |
| | | batchNoStockMap.value = {}; |
| | | }; |
| | | |
| | | // 关闭对话框 |
| | | const closeDialog = () => { |
| | | dialogVisible.value = false; |
| | | resetForm(); |
| | | }; |
| | | |
| | | // 产品选择处理 |
| | | const handleProductSelect = async products => { |
| | | if (products && products.length > 0) { |
| | | const product = products[0]; |
| | | formState.value.productId = product.productId; |
| | | formState.value.productName = product.productName; |
| | | formState.value.productModelName = product.model; |
| | | formState.value.productModelId = product.id; |
| | | formState.value.unit = product.unit; |
| | | // 从batchNoMaps获取批号列表和库存 |
| | | if (product.batchNoMaps && Object.keys(product.batchNoMaps).length > 0) { |
| | | batchNoList.value = Object.keys(product.batchNoMaps); |
| | | batchNoStockMap.value = product.batchNoMaps; |
| | | } else { |
| | | batchNoList.value = []; |
| | | batchNoStockMap.value = {}; |
| | | } |
| | | // 清空已选择的批号和最大库存 |
| | | formState.value.batchNo = null; |
| | | formState.value.maxStock = 0; |
| | | showProductSelect.value = false; |
| | | // 触发表单验证更新 |
| | | proxy.$refs["formRef"]?.validateField("productModelId"); |
| | | } |
| | | }; |
| | | |
| | | // 批号选择变化处理 |
| | | const handleBatchNoChange = (batchNo) => { |
| | | if (batchNo && batchNoStockMap.value[batchNo]) { |
| | | formState.value.maxStock = batchNoStockMap.value[batchNo]; |
| | | // 如果当前出库数量超过最大库存,自动调整为最大库存 |
| | | if (formState.value.qualitity > formState.value.maxStock) { |
| | | formState.value.qualitity = formState.value.maxStock; |
| | | } |
| | | } else { |
| | | formState.value.maxStock = 0; |
| | | } |
| | | }; |
| | | |
| | | // 提交表单 |
| | | const handleSubmit = () => { |
| | | proxy.$refs["formRef"].validate(valid => { |
| | | if (valid) { |
| | | // 验证是否选择了产品 |
| | | if (!formState.value.productModelId) { |
| | | ElMessage.error("请选择产品"); |
| | | return; |
| | | } |
| | | |
| | | if (dialogType.value === 'add') { |
| | | submitAdd(); |
| | | } else { |
| | | submitEdit(); |
| | | } |
| | | } |
| | | }); |
| | | }; |
| | | |
| | | // 提交新增 |
| | | const submitAdd = () => { |
| | | const params = { ...formState.value }; |
| | | |
| | | if (formState.value.type === "qualified") { |
| | | addStockOutRecordOnly(params).then(res => { |
| | | ElMessage.success("新增成功"); |
| | | closeDialog(); |
| | | getList(); |
| | | }).catch(() => { |
| | | ElMessage.error("新增失败"); |
| | | }); |
| | | } else { |
| | | addUnqualifiedStockOutRecordOnly(params).then(res => { |
| | | ElMessage.success("新增成功"); |
| | | closeDialog(); |
| | | getList(); |
| | | }).catch(() => { |
| | | ElMessage.error("新增失败"); |
| | | }); |
| | | } |
| | | }; |
| | | |
| | | // 提交编辑 |
| | | const submitEdit = () => { |
| | | const params = { |
| | | productId: formState.value.productId, |
| | | productModelId: formState.value.productModelId, |
| | | productName: formState.value.productName, |
| | | model: formState.value.productModelName, |
| | | unit: formState.value.unit, |
| | | batchNo: formState.value.batchNo, |
| | | stockOutNum: formState.value.qualitity, |
| | | recordType: formState.value.recordType, |
| | | remark: formState.value.remark, |
| | | }; |
| | | |
| | | updateStockOutRecord(formState.value.id, params).then(() => { |
| | | ElMessage.success("编辑成功"); |
| | | closeDialog(); |
| | | getList(); |
| | | }).catch(() => { |
| | | ElMessage.error("编辑失败"); |
| | | }); |
| | | }; |
| | | |
| | | // 获取来源类型选项 |
| | | const fetchStockRecordTypeOptions = () => { |
| | | if (props.type === "0") { |