Browse Source

✨ ERP:初始化其它入库的表单 40%

YunaiV 1 year ago
parent
commit
53c94af027

+ 5 - 0
src/api/erp/stock/stock/index.ts

@@ -24,6 +24,11 @@ export const StockApi = {
     return await request.get({ url: `/erp/stock/get?id=` + id })
   },
 
+  // 查询产品库存详情
+  getStock2: async (productId: number, warehouseId: number) => {
+    return await request.get({ url: `/erp/stock/get`, params: { productId, warehouseId } })
+  },
+
   // 导出产品库存 Excel
   exportStock: async (params) => {
     return await request.download({ url: `/erp/stock/export-excel`, params })

+ 1 - 1
src/components/UploadFile/src/UploadFile.vue

@@ -160,7 +160,7 @@ const emitUpdateModelValue = () => {
   // 情况1:数组结果
   let result: string | string[] = fileList.value.map((file) => file.url!)
   // 情况2:逗号分隔的字符串
-  if (isString(props.modelValue)) {
+  if (props.limit === 1 || isString(props.modelValue)) {
     result = result.join(',')
   }
   emit('update:modelValue', result)

+ 34 - 45
src/views/erp/stock/in/StockInForm.vue

@@ -8,54 +8,54 @@
       v-loading="formLoading"
     >
       <el-row :gutter="20">
-        <el-col :span="12">
+        <!-- TODO 芋艿:待接入 -->
+        <el-col :span="8">
           <el-form-item label="入库单号" prop="no">
             <el-input v-model="formData.no" placeholder="请输入入库单号" />
           </el-form-item>
         </el-col>
-        <el-col :span="12">
+        <el-col :span="8">
           <el-form-item label="入库时间" prop="inTime">
             <el-date-picker
               v-model="formData.inTime"
               type="date"
               value-format="x"
               placeholder="选择入库时间"
+              class="!w-1/1"
             />
           </el-form-item>
         </el-col>
-        <el-col :span="12">
+        <!-- TODO 芋艿:待接入 -->
+        <el-col :span="8">
           <el-form-item label="供应商" prop="supplierId">
-            <el-input v-model="formData.supplierId" placeholder="请输入供应商编号" />
+            <el-input v-model="formData.supplierId" placeholder="请输入供应商" />
           </el-form-item>
         </el-col>
-        <!-- TODO 芋艿:附件 -->
-        <el-col :span="24">
+        <el-col :span="16">
           <el-form-item label="备注" prop="remark">
-            <el-input type="textarea" v-model="formData.remark" placeholder="请输入备注" />
+            <el-input
+              type="textarea"
+              v-model="formData.remark"
+              :rows="1"
+              placeholder="请输入备注"
+            />
+          </el-form-item>
+        </el-col>
+        <el-col :span="8">
+          <el-form-item label="附件" prop="fileUrl">
+            <UploadFile :is-show-tip="false" v-model="formData.fileUrl" :limit="1" />
           </el-form-item>
         </el-col>
       </el-row>
     </el-form>
     <!-- 子表的表单 -->
-    <el-tabs v-model="subTabsName">
-      <el-tab-pane label="入库产品清单" name="stockInItem">
-        <StockInItemForm ref="stockInItemFormRef" :in-id="formData.id" />
-      </el-tab-pane>
-    </el-tabs>
-    <el-form
-      ref="formRef2"
-      :model="formData"
-      :rules="formRules"
-      label-width="100px"
-      v-loading="formLoading"
-    >
-      <el-form-item label="合计数量" prop="totalCount">
-        <el-input v-model="formData.totalCount" placeholder="请输入合计数量" />
-      </el-form-item>
-      <el-form-item label="合计金额" prop="totalPrice">
-        <el-input v-model="formData.totalPrice" placeholder="请输入合计金额,单位:元" />
-      </el-form-item>
-    </el-form>
+    <ContentWrap>
+      <el-tabs v-model="subTabsName" class="-mt-15px -mb-10px">
+        <el-tab-pane label="入库产品清单" name="stockInItem">
+          <StockInItemForm ref="stockInItemFormRef" :items="formData.items" />
+        </el-tab-pane>
+      </el-tabs>
+    </ContentWrap>
     <template #footer>
       <el-button @click="submitForm" type="primary" :disabled="formLoading">确 定</el-button>
       <el-button @click="dialogVisible = false">取 消</el-button>
@@ -82,18 +82,15 @@ const formData = ref({
   no: undefined,
   supplierId: undefined,
   inTime: undefined,
-  totalCount: undefined,
-  totalPrice: undefined,
-  remark: undefined
+  remark: undefined,
+  fileUrl: '',
+  items: []
 })
 const formRules = reactive({
   no: [{ required: true, message: '入库单号不能为空', trigger: 'blur' }],
-  inTime: [{ required: true, message: '入库时间不能为空', trigger: 'blur' }],
-  totalCount: [{ required: true, message: '合计数量不能为空', trigger: 'blur' }],
-  totalPrice: [{ required: true, message: '合计金额,单位:元不能为空', trigger: 'blur' }]
+  inTime: [{ required: true, message: '入库时间不能为空', trigger: 'blur' }]
 })
 const formRef = ref() // 表单 Ref
-const formRef2 = ref() // 表单 Ref TODO 芋艿:需要优化
 
 /** 子表的表单 */
 const subTabsName = ref('stockInItem')
@@ -122,20 +119,11 @@ const emit = defineEmits(['success']) // 定义 success 事件,用于操作成
 const submitForm = async () => {
   // 校验表单
   await formRef.value.validate()
-  await formRef2.value.validate() // TODO 芋艿:需要在看看
-  // 校验子表单
-  try {
-    await stockInItemFormRef.value.validate()
-  } catch (e) {
-    subTabsName.value = 'stockInItem'
-    return
-  }
+  await stockInItemFormRef.value.validate()
   // 提交请求
   formLoading.value = true
   try {
     const data = formData.value as unknown as StockInVO
-    // 拼接子表的数据
-    data.stockInItems = stockInItemFormRef.value.getData()
     if (formType.value === 'create') {
       await StockInApi.createStockIn(data)
       message.success(t('common.createSuccess'))
@@ -160,9 +148,10 @@ const resetForm = () => {
     inTime: undefined,
     totalCount: undefined,
     totalPrice: undefined,
-    remark: undefined
+    remark: undefined,
+    fileUrl: undefined,
+    items: []
   }
   formRef.value?.resetFields()
-  formRef2.value?.resetFields() // TODO 芋艿:需要在看看
 }
 </script>

+ 127 - 45
src/views/erp/stock/in/components/StockInItemForm.vue

@@ -8,65 +8,101 @@
     :inline-message="true"
   >
     <el-table :data="formData" show-summary class="-mt-10px">
-      <el-table-column label="序号" type="index" width="60" />
-      <el-table-column label="仓库名称" min-width="150">
+      <el-table-column label="序号" type="index" align="center" width="60" />
+      <el-table-column label="仓库名称" min-width="125">
         <template #default="{ row, $index }">
           <el-form-item
             :prop="`${$index}.warehouseId`"
             :rules="formRules.warehouseId"
             class="mb-0px!"
           >
-            <el-input v-model="row.warehouseId" placeholder="请输入仓库编号" />
+            <el-select
+              v-model="row.warehouseId"
+              filterable
+              placeholder="请选择仓库"
+              @change="onChangeWarehouse($event, row)"
+            >
+              <el-option
+                v-for="item in warehouseList"
+                :key="item.id"
+                :label="item.name"
+                :value="item.id"
+              />
+            </el-select>
           </el-form-item>
         </template>
       </el-table-column>
-      <el-table-column label="产品名称" min-width="150">
+      <el-table-column label="产品名称" min-width="180">
         <template #default="{ row, $index }">
           <el-form-item :prop="`${$index}.productId`" :rules="formRules.productId" class="mb-0px!">
-            <el-input v-model="row.productId" placeholder="请输入产品编号" />
+            <el-select
+              v-model="row.productId"
+              filterable
+              @change="onChangeProduct($event, row)"
+              placeholder="请选择产品"
+            >
+              <el-option
+                v-for="item in productList"
+                :key="item.id"
+                :label="item.name"
+                :value="item.id"
+              />
+            </el-select>
+          </el-form-item>
+        </template>
+      </el-table-column>
+      <el-table-column label="库存" min-width="100">
+        <template #default="{ row }">
+          <el-form-item class="mb-0px!">
+            <el-input disabled v-model="row.stockCount" />
           </el-form-item>
         </template>
       </el-table-column>
       <el-table-column label="条码" min-width="150">
         <template #default="{ row }">
           <el-form-item class="mb-0px!">
-            <el-input disabled v-model="row.productId" placeholder="请输入条码" />
+            <el-input disabled v-model="row.productBarCode" />
           </el-form-item>
         </template>
       </el-table-column>
-      <el-table-column label="单位" min-width="150">
+      <el-table-column label="单位" min-width="80">
         <template #default="{ row }">
           <el-form-item class="mb-0px!">
-            <el-input disabled v-model="row.productUnitId" placeholder="请输入单位编号" />
+            <el-input disabled v-model="row.productUnitName" />
           </el-form-item>
         </template>
       </el-table-column>
-      <el-table-column label="数量" prop="count" min-width="150">
+      <el-table-column label="数量" prop="count" fixed="right" min-width="120">
         <template #default="{ row, $index }">
           <el-form-item :prop="`${$index}.count`" :rules="formRules.count" class="mb-0px!">
-            <el-input v-model="row.count" placeholder="请输入商品数量" />
+            <el-input-number
+              v-model="row.count"
+              controls-position="right"
+              :min="1"
+              class="!w-100%"
+            />
           </el-form-item>
         </template>
       </el-table-column>
-      <el-table-column label="商品单价" min-width="150">
+      <el-table-column label="商品单价" fixed="right" min-width="120">
         <template #default="{ row, $index }">
           <el-form-item
             :prop="`${$index}.productPrice`"
             :rules="formRules.productPrice"
             class="mb-0px!"
           >
-            <el-input v-model="row.productPrice" placeholder="请输入商品单价" />
+            <el-input-number v-model="row.productPrice" controls-position="right" :min="1" />
           </el-form-item>
         </template>
       </el-table-column>
-      <el-table-column label="合计金额" min-width="150">
+      <el-table-column label="合计金额" prop="totalPrice" fixed="right" min-width="100">
         <template #default="{ row, $index }">
           <el-form-item
             :prop="`${$index}.totalPrice`"
             :rules="formRules.totalPrice"
             class="mb-0px!"
           >
-            <el-input v-model="row.totalPrice" placeholder="请输入合计金额,单位:元" />
+            <el-input disabled v-model="row.totalPrice" />
           </el-form-item>
         </template>
       </el-table-column>
@@ -90,57 +126,74 @@
 </template>
 <script setup lang="ts">
 import { StockInApi } from '@/api/erp/stock/in'
+import { ProductApi, ProductVO } from '@/api/erp/product/product'
+import { WarehouseApi, WarehouseVO } from '@/api/erp/stock/warehouse'
+import { StockApi } from '@/api/erp/stock/stock'
 
 const props = defineProps<{
-  inId: undefined // 入库编号(主表的关联字段)
+  items: undefined
 }>()
 const formLoading = ref(false) // 表单的加载中
 const formData = ref([])
 const formRules = reactive({
   inId: [{ required: true, message: '入库编号不能为空', trigger: 'blur' }],
-  warehouseId: [{ required: true, message: '仓库编号不能为空', trigger: 'blur' }],
-  productId: [{ required: true, message: '产品编号不能为空', trigger: 'blur' }],
-  productUnitId: [{ required: true, message: '商品单位编号不能为空', trigger: 'blur' }],
-  productPrice: [{ required: true, message: '商品单价不能为空', trigger: 'blur' }],
-  count: [{ required: true, message: '商品数量不能为空', trigger: 'blur' }],
-  totalPrice: [{ required: true, message: '合计金额,单位:元不能为空', trigger: 'blur' }]
+  warehouseId: [{ required: true, message: '仓库不能为空', trigger: 'blur' }],
+  productId: [{ required: true, message: '产品不能为空', trigger: 'blur' }],
+  count: [{ required: true, message: '商品数量不能为空', trigger: 'blur' }]
 })
-const formRef = ref() // 表单 Ref
+const formRef = ref([]) // 表单 Ref
+const productList = ref<ProductVO[]>([]) // 产品列表
+const warehouseList = ref<WarehouseVO[]>([]) // 仓库列表
+const defaultWarehouse = ref<WarehouseVO>(undefined) // 默认仓库
 
 /** 监听主表的关联字段的变化,加载对应的子表数据 */
 watch(
-  () => props.inId,
+  () => props.items,
   async (val) => {
-    // 1. 重置表单
-    formData.value = []
-    // 2. val 非空,则加载数据
-    if (!val) {
+    formData.value = val
+  },
+  { immediate: true }
+)
+
+/** 监听合同产品变化,计算合同产品总价 */
+watch(
+  () => formData.value,
+  (val) => {
+    if (!val || val.length === 0) {
       return
     }
-    try {
-      formLoading.value = true
-      formData.value = await StockInApi.getStockInItemListByInId(val)
-    } finally {
-      formLoading.value = false
-    }
+    // 循环处理
+    val.forEach((item) => {
+      // const product = productList.value.find((product) => product.id === item.productId)
+      // if (product) {
+      //   item.productUnitName = product.unitName
+      //   item.productBarCode = product.barCode
+      //   item.productPrice = product.minPrice
+      //   // TODO 芋艿:加载库存
+      //   item.stockCount = 10
+      // }
+      if (item.productPrice && item.count) {
+        item.totalPrice = item.productPrice * item.count
+      }
+    })
   },
-  { immediate: true }
+  { deep: true }
 )
 
 /** 新增按钮操作 */
 const handleAdd = () => {
   const row = {
     id: undefined,
-    inId: undefined,
-    warehouseId: undefined,
+    warehouseId: defaultWarehouse.value?.id,
     productId: undefined,
-    productUnitId: undefined,
+    productUnitName: undefined, // 产品单位
+    productBarCode: undefined, // 产品条码
     productPrice: undefined,
-    count: undefined,
+    stockCount: undefined,
+    count: 1,
     totalPrice: undefined,
     remark: undefined
   }
-  row.inId = props.inId
   formData.value.push(row)
 }
 
@@ -149,15 +202,44 @@ const handleDelete = (index) => {
   formData.value.splice(index, 1)
 }
 
+/** 处理仓库变更 */
+const onChangeWarehouse = (warehouseId, row) => {
+  // 加载库存
+  setStockCount(row)
+}
+
+/** 处理产品变更 */
+const onChangeProduct = (productId, row) => {
+  const product = productList.value.find((item) => item.id === productId)
+  if (product) {
+    row.productUnitName = product.unitName
+    row.productBarCode = product.barCode
+    row.productPrice = product.minPrice
+  }
+  // 加载库存
+  setStockCount(row)
+}
+
+/** 加载库存 */
+const setStockCount = async (row) => {
+  if (!row.productId || !row.warehouseId) {
+    return
+  }
+  const stock = await StockApi.getStock2(row.productId, row.warehouseId)
+  row.stockCount = stock ? stock.count : 0
+}
+
 /** 表单校验 */
 const validate = () => {
   return formRef.value.validate()
 }
+defineExpose({ validate })
 
-/** 表单值 */
-const getData = () => {
-  return formData.value
-}
-
-defineExpose({ validate, getData })
+/** 初始化 */
+onMounted(async () => {
+  // 加载产品、仓库列表
+  productList.value = await ProductApi.getProductSimpleList()
+  warehouseList.value = await WarehouseApi.getWarehouseSimpleList()
+  defaultWarehouse.value = warehouseList.value.find((item) => item.defaultStatus)
+})
 </script>