From a3c508233dd94b50c8005ec3a5d40b91341d6434 Mon Sep 17 00:00:00 2001
From: spring <2396852758@qq.com>
Date: 星期四, 29 一月 2026 17:58:30 +0800
Subject: [PATCH] Merge branch 'dev_New' of http://114.132.189.42:9002/r/product-inventory-management into dev_New

---
 src/views/safeProduction/safeWorkApproval/components/approvalDia.vue |  530 ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
 1 files changed, 530 insertions(+), 0 deletions(-)

diff --git a/src/views/safeProduction/safeWorkApproval/components/approvalDia.vue b/src/views/safeProduction/safeWorkApproval/components/approvalDia.vue
new file mode 100644
index 0000000..8d40a2e
--- /dev/null
+++ b/src/views/safeProduction/safeWorkApproval/components/approvalDia.vue
@@ -0,0 +1,530 @@
+<template>
+  <div>
+    <el-dialog v-model="dialogFormVisible"
+               :title="operationType === 'approval' ? '瀹℃壒' : '璇︽儏'"
+               width="700px"
+               @close="closeDia">
+      <el-form :model="form"
+               label-width="140px"
+               label-position="top"
+               ref="formRef">
+        <el-row>
+          <el-col :span="24">
+            <el-form-item label="娴佺▼缂栧彿锛�"
+                          prop="approveId">
+              <el-input v-model="form.approveId"
+                        placeholder="鑷姩缂栧彿"
+                        clearable
+                        disabled />
+            </el-form-item>
+          </el-col>
+        </el-row>
+        <el-row>
+          <el-col :span="24">
+            <el-form-item label="鐢宠閮ㄩ棬锛�">
+              <el-select disabled
+                         v-model="form.approveDeptId"
+                         placeholder="閫夋嫨閮ㄩ棬">
+                <el-option v-for="user in productOptions"
+                           :key="user.deptId"
+                           :label="user.deptName"
+                           :value="user.deptId" />
+              </el-select>
+            </el-form-item>
+          </el-col>
+        </el-row>
+        <el-row v-if="!isQuotationApproval && !isPurchaseApproval">
+          <el-col :span="24">
+            <el-form-item :label="props.approveType == 5 ? '閲囪喘鍚堝悓鍙凤細' : '瀹℃壒浜嬬敱锛�'"
+                          prop="approveReason">
+              <el-input v-model="form.approveReason"
+                        placeholder="璇疯緭鍏�"
+                        clearable
+                        type="textarea"
+                        disabled />
+            </el-form-item>
+          </el-col>
+        </el-row>
+        <!-- 瀹℃壒浜洪�夋嫨锛堝姩鎬佽妭鐐癸級 -->
+        <el-row :gutter="30">
+          <el-col :span="12">
+            <el-form-item label="鐢宠浜猴細"
+                          prop="approveUser">
+              <el-select v-model="form.approveUser"
+                         placeholder="閫夋嫨浜哄憳"
+                         disabled>
+                <el-option v-for="user in userList"
+                           :key="user.userId"
+                           :label="user.nickName"
+                           :value="user.userId" />
+              </el-select>
+            </el-form-item>
+          </el-col>
+          <el-col :span="12">
+            <el-form-item label="鐢宠鏃ユ湡锛�"
+                          prop="approveTime">
+              <el-date-picker v-model="form.approveTime"
+                              type="date"
+                              placeholder="璇烽�夋嫨鏃ユ湡"
+                              value-format="YYYY-MM-DD"
+                              format="YYYY-MM-DD"
+                              clearable
+                              style="width: 100%"
+                              disabled />
+            </el-form-item>
+          </el-col>
+        </el-row>
+      </el-form>
+      <!-- 鎶ヤ环瀹℃壒锛氬睍绀烘姤浠疯鎯咃紙澶嶇敤閿�鍞姤浠�"鏌ョ湅璇︽儏瀵硅瘽妗�"鍐呭缁撴瀯锛� -->
+      <div v-if="isQuotationApproval"
+           style="margin: 10px 0 18px;">
+        <el-divider content-position="left">鎶ヤ环璇︽儏</el-divider>
+        <el-skeleton :loading="quotationLoading"
+                     animated>
+          <template #template>
+            <el-skeleton-item variant="h3"
+                              style="width: 30%" />
+            <el-skeleton-item variant="text"
+                              style="width: 100%" />
+            <el-skeleton-item variant="text"
+                              style="width: 100%" />
+          </template>
+          <template #default>
+            <el-empty v-if="!currentQuotation || !currentQuotation.quotationNo"
+                      description="鏈煡璇㈠埌瀵瑰簲鎶ヤ环璇︽儏" />
+            <template v-else>
+              <el-descriptions :column="2"
+                               border>
+                <el-descriptions-item label="鎶ヤ环鍗曞彿">{{ currentQuotation.quotationNo }}</el-descriptions-item>
+                <el-descriptions-item label="瀹㈡埛鍚嶇О">{{ currentQuotation.customer }}</el-descriptions-item>
+                <el-descriptions-item label="涓氬姟鍛�">{{ currentQuotation.salesperson }}</el-descriptions-item>
+                <el-descriptions-item label="鎶ヤ环鏃ユ湡">{{ currentQuotation.quotationDate }}</el-descriptions-item>
+                <el-descriptions-item label="鏈夋晥鏈熻嚦">{{ currentQuotation.validDate }}</el-descriptions-item>
+                <el-descriptions-item label="浠樻鏂瑰紡">{{ currentQuotation.paymentMethod }}</el-descriptions-item>
+                <el-descriptions-item label="鎶ヤ环鎬婚"
+                                      :span="2">
+                  <span style="font-size: 18px; color: #e6a23c; font-weight: bold;">
+                    楼{{ Number(currentQuotation.totalAmount ?? 0).toFixed(2) }}
+                  </span>
+                </el-descriptions-item>
+              </el-descriptions>
+              <div style="margin-top: 20px;">
+                <h4>浜у搧鏄庣粏</h4>
+                <el-table :data="currentQuotation.products || []"
+                          border
+                          style="width: 100%">
+                  <el-table-column prop="product"
+                                   label="浜у搧鍚嶇О" />
+                  <el-table-column prop="specification"
+                                   label="瑙勬牸鍨嬪彿" />
+                  <el-table-column prop="unit"
+                                   label="鍗曚綅" />
+                  <el-table-column prop="unitPrice"
+                                   label="鍗曚环">
+                    <template #default="scope">楼{{ Number(scope.row.unitPrice ?? 0).toFixed(2) }}</template>
+                  </el-table-column>
+                </el-table>
+              </div>
+              <div v-if="currentQuotation.remark"
+                   style="margin-top: 20px;">
+                <h4>澶囨敞</h4>
+                <p>{{ currentQuotation.remark }}</p>
+              </div>
+            </template>
+          </template>
+        </el-skeleton>
+      </div>
+      <!-- 閲囪喘瀹℃壒锛氬睍绀洪噰璐鎯� -->
+      <div v-if="isPurchaseApproval"
+           style="margin: 10px 0 18px;">
+        <el-divider content-position="left">閲囪喘璇︽儏</el-divider>
+        <el-skeleton :loading="purchaseLoading"
+                     animated>
+          <template #template>
+            <el-skeleton-item variant="h3"
+                              style="width: 30%" />
+            <el-skeleton-item variant="text"
+                              style="width: 100%" />
+            <el-skeleton-item variant="text"
+                              style="width: 100%" />
+          </template>
+          <template #default>
+            <el-empty v-if="!currentPurchase || !currentPurchase.purchaseContractNumber"
+                      description="鏈煡璇㈠埌瀵瑰簲閲囪喘璇︽儏" />
+            <template v-else>
+              <el-descriptions :column="2"
+                               border>
+                <el-descriptions-item label="閲囪喘鍚堝悓鍙�">{{ currentPurchase.purchaseContractNumber }}</el-descriptions-item>
+                <el-descriptions-item label="渚涘簲鍟嗗悕绉�">{{ currentPurchase.supplierName }}</el-descriptions-item>
+                <el-descriptions-item label="椤圭洰鍚嶇О">{{ currentPurchase.projectName }}</el-descriptions-item>
+                <el-descriptions-item label="閿�鍞悎鍚屽彿">{{ currentPurchase.salesContractNo }}</el-descriptions-item>
+                <el-descriptions-item label="绛捐鏃ユ湡">{{ currentPurchase.executionDate }}</el-descriptions-item>
+                <el-descriptions-item label="褰曞叆鏃ユ湡">{{ currentPurchase.entryDate }}</el-descriptions-item>
+                <el-descriptions-item label="浠樻鏂瑰紡">{{ currentPurchase.paymentMethod }}</el-descriptions-item>
+                <el-descriptions-item label="鍚堝悓閲戦"
+                                      :span="2">
+                  <span style="font-size: 18px; color: #e6a23c; font-weight: bold;">
+                    楼{{ Number(currentPurchase.contractAmount ?? 0).toFixed(2) }}
+                  </span>
+                </el-descriptions-item>
+              </el-descriptions>
+              <div style="margin-top: 20px;">
+                <h4>浜у搧鏄庣粏</h4>
+                <el-table :data="currentPurchase.productData || []"
+                          border
+                          style="width: 100%">
+                  <el-table-column prop="productCategory"
+                                   label="浜у搧鍚嶇О" />
+                  <el-table-column prop="specificationModel"
+                                   label="瑙勬牸鍨嬪彿" />
+                  <el-table-column prop="unit"
+                                   label="鍗曚綅" />
+                  <el-table-column prop="quantity"
+                                   label="鏁伴噺" />
+                  <el-table-column prop="taxInclusiveUnitPrice"
+                                   label="鍚◣鍗曚环">
+                    <template #default="scope">楼{{ Number(scope.row.taxInclusiveUnitPrice ?? 0).toFixed(2) }}</template>
+                  </el-table-column>
+                  <el-table-column prop="taxInclusiveTotalPrice"
+                                   label="鍚◣鎬讳环">
+                    <template #default="scope">楼{{ Number(scope.row.taxInclusiveTotalPrice ?? 0).toFixed(2) }}</template>
+                  </el-table-column>
+                </el-table>
+              </div>
+            </template>
+          </template>
+        </el-skeleton>
+      </div>
+      <el-form :model="{ activities }"
+               ref="formRef"
+               label-position="top">
+        <el-steps :active="getActiveStep()"
+                  finish-status="success"
+                  process-status="process"
+                  align-center
+                  direction="vertical">
+          <el-step v-for="(activity, index) in activities"
+                   :key="index"
+                   finish-status="success"
+                   :title="getNodeTitle(index, activities.length)"
+                   :description="activity.approveNodeUser"
+                   :icon="getNodeIcon(activity, index)">
+            <template #icon>
+              <el-icon v-if="activity.approveNodeStatus === 2"
+                       color="red"
+                       :size="22">
+                <WarningFilled />
+              </el-icon>
+              <el-icon v-else-if="activity.isShen"
+                       color="#1890ff"
+                       :size="22">
+                <Edit />
+              </el-icon>
+              <el-icon v-else-if="activity.approveNodeStatus === 1"
+                       color="#67C23A"
+                       :size="26">
+                <Check />
+              </el-icon>
+              <el-icon v-else
+                       color="#C0C4CC"
+                       :size="22">
+                <MoreFilled />
+              </el-icon>
+            </template>
+            <template #title>
+              <span style="color: #000000">{{ getNodeTitle(index, activities.length) }}</span>
+            </template>
+            <template #description>
+              <div class="node-user">
+                <div class="avatar-wrapper">
+                  <img :src="userStore.avatar"
+                       class="user-avatar"
+                       alt="" />
+                </div>
+                <span style="color: #000000">{{ activity.approveNodeUser }}-{{activity.isApproval}}</span>
+              </div>
+              <div v-if="!activity.isShen"
+                   class="node-reason">
+                <span>瀹℃壒鎰忚锛�</span>{{ activity.approveNodeReason }}
+              </div>
+              <div v-if="!activity.isShen"
+                   class="node-reason">
+                <span>绛惧悕锛�</span>
+                <img :src="activity.urlTem"
+                     class="signImg"
+                     alt=""
+                     v-if="activity.urlTem" />
+              </div>
+              <div v-else-if="activity.isShen">
+                <el-form-item :prop="'activities.' + index + '.approveNodeReason'"
+                              :rules="[{ required: true, message: '瀹℃壒鎰忚涓嶈兘涓虹┖', trigger: 'blur' }]">
+                  <el-input v-model="activity.approveNodeReason"
+                            clearable
+                            type="textarea"
+                            :disabled="operationType === 'view'"></el-input>
+                </el-form-item>
+              </div>
+            </template>
+          </el-step>
+        </el-steps>
+      </el-form>
+      <template #footer
+                v-if="operationType === 'approval'">
+        <div class="dialog-footer">
+          <el-button type="primary"
+                     @click="submitForm(2)">涓嶉�氳繃</el-button>
+          <el-button type="primary"
+                     @click="submitForm(1)">閫氳繃</el-button>
+          <el-button @click="closeDia">鍙栨秷</el-button>
+        </div>
+      </template>
+    </el-dialog>
+  </div>
+</template>
+
+<script setup>
+  import {
+    computed,
+    getCurrentInstance,
+    nextTick,
+    reactive,
+    ref,
+    toRefs,
+  } from "vue";
+  import {
+    approveProcessDetails,
+    getDept,
+    updateApproveNode,
+  } from "@/api/collaborativeApproval/approvalProcess.js";
+  import useUserStore from "@/store/modules/user.js";
+  import { userListNoPageByTenantId } from "@/api/system/user.js";
+  import {
+    WarningFilled,
+    Edit,
+    Check,
+    MoreFilled,
+  } from "@element-plus/icons-vue";
+  import { getQuotationList } from "@/api/salesManagement/salesQuotation.js";
+  import { getPurchaseByCode } from "@/api/procurementManagement/procurementLedger.js";
+  const emit = defineEmits(["close"]);
+  const { proxy } = getCurrentInstance();
+
+  const props = defineProps({
+    approveType: {
+      type: [Number, String],
+      default: 0,
+    },
+  });
+
+  const dialogFormVisible = ref(false);
+  const operationType = ref("");
+  const activities = ref([]);
+  const formRef = ref(null);
+  const userStore = useUserStore();
+  const productOptions = ref([]);
+  const userList = ref([]);
+  const quotationLoading = ref(false);
+  const currentQuotation = ref({});
+  const purchaseLoading = ref(false);
+  const currentPurchase = ref({});
+  const isQuotationApproval = computed(() => Number(props.approveType) === 6);
+  const isPurchaseApproval = computed(() => Number(props.approveType) === 5);
+
+  const data = reactive({
+    form: {
+      approveTime: "",
+      approveId: "",
+      approveUser: "",
+      approveDeptId: "",
+      approveReason: "",
+      checkResult: "",
+    },
+  });
+  const { form } = toRefs(data);
+
+  // 鑺傜偣鏍囬
+  const getNodeTitle = (index, len) => {
+    if (index === len - 1) return "缁撴潫";
+    return "瀹℃壒";
+  };
+
+  // 鑾峰彇褰撳墠婵�娲绘楠�
+  const getActiveStep = () => {
+    // 濡傛灉鎵�鏈� isShen 閮戒负 false锛岃繑鍥炴渶鍚庝竴涓楠わ紙鍏ㄩ儴瀹屾垚锛�
+    const hasActive = activities.value.some(a => a.isShen === true);
+    if (!hasActive) return activities.value.length;
+    // 褰撳墠鑺傜偣绱㈠紩
+    return activities.value.findIndex(a => a.isShen == true);
+  };
+  // 姝ラicon
+  const getNodeIcon = (activity, index) => {
+    if (activity.approveNodeStatus === 2) return "el-icon-warning"; // 涓嶉�氳繃
+    if (activity.isShen) return "Edit";
+    return "";
+  };
+
+  // 鎵撳紑寮规
+  const openDialog = (type, row) => {
+    operationType.value = type;
+    dialogFormVisible.value = true;
+    currentQuotation.value = {};
+    currentPurchase.value = {};
+    userListNoPageByTenantId().then(res => {
+      userList.value = res.data;
+    });
+    form.value = { ...row };
+    // 绔嬪嵆娓呴櫎琛ㄥ崟楠岃瘉鐘舵�侊紙鍥犱负瀛楁鏄痙isabled鐨勶紝涓嶉渶瑕侀獙璇侊級
+    nextTick(() => {
+      if (formRef.value) {
+        formRef.value.clearValidate();
+      }
+    });
+    // 纭繚閫夐」鍔犺浇瀹屾垚鍚庡啀鍖归厤鍊肩被鍨�
+    getProductOptions().then(() => {
+      // 纭繚鍊肩被鍨嬪尮閰嶏紙濡傛灉閫夐」宸插姞杞斤級
+      if (productOptions.value.length > 0 && form.value.approveDeptId) {
+        const matchedOption = productOptions.value.find(
+          opt =>
+            opt.deptId == form.value.approveDeptId ||
+            String(opt.deptId) === String(form.value.approveDeptId)
+        );
+        if (matchedOption) {
+          form.value.approveDeptId = matchedOption.deptId;
+        }
+      }
+      // 鍐嶆娓呴櫎楠岃瘉锛岀‘淇濋�夐」鍔犺浇鍚庡�煎尮閰嶆纭�
+      nextTick(() => {
+        if (formRef.value) {
+          formRef.value.clearValidate();
+        }
+      });
+    });
+
+    // 鎶ヤ环瀹℃壒锛氱敤瀹℃壒浜嬬敱瀛楁鎵胯浇鐨�"鎶ヤ环鍗曞彿"鍘绘煡鎶ヤ环鍒楄〃
+    if (isQuotationApproval.value) {
+      const quotationNo = row?.approveReason;
+      if (quotationNo) {
+        quotationLoading.value = true;
+        getQuotationList({ quotationNo })
+          .then(res => {
+            const records = res?.data?.records || [];
+            currentQuotation.value = records[0] || {};
+          })
+          .finally(() => {
+            quotationLoading.value = false;
+          });
+      }
+    }
+
+    // 閲囪喘瀹℃壒锛氱敤瀹℃壒浜嬬敱瀛楁鎵胯浇鐨�"閲囪喘鍚堝悓鍙�"鍘绘煡閲囪喘璇︽儏
+    if (isPurchaseApproval.value) {
+      const purchaseContractNumber = row?.approveReason;
+      if (purchaseContractNumber) {
+        purchaseLoading.value = true;
+        getPurchaseByCode({ purchaseContractNumber })
+          .then(res => {
+            currentPurchase.value = res;
+          })
+          .catch(err => {
+            console.error("鏌ヨ閲囪喘璇︽儏澶辫触:", err);
+            proxy.$modal.msgError("鏌ヨ閲囪喘璇︽儏澶辫触");
+          })
+          .finally(() => {
+            purchaseLoading.value = false;
+          });
+      }
+    }
+
+    approveProcessDetails(row.approveId).then(res => {
+      activities.value = res.data;
+      // 澧炲姞isApproval瀛楁
+      activities.value.forEach(item => {
+        if (item.url && item.url.includes("word")) {
+          item.urlTem = item.url.replaceAll("word", "img");
+        } else {
+          item.urlTem = item.url;
+        }
+        if (item.approveNodeStatus === 2) {
+          item.isApproval = "宸查┏鍥�";
+        } else if (item.approveNodeStatus === 1) {
+          item.isApproval = "宸插悓鎰�";
+        } else {
+          item.isApproval = "鏈鎵�";
+        }
+      });
+    });
+  };
+  const getProductOptions = () => {
+    return getDept().then(res => {
+      productOptions.value = res.data;
+    });
+  };
+  // 鎻愪氦瀹℃壒
+  const submitForm = status => {
+    const filteredActivities = activities.value.filter(
+      activity => activity.isShen
+    );
+    if (!filteredActivities || filteredActivities.length === 0) {
+      proxy.$modal.msgError("鏈壘鍒板緟瀹℃壒鐨勮妭鐐�");
+      return;
+    }
+    const currentActivity = filteredActivities[0];
+    if (!currentActivity) {
+      proxy.$modal.msgError("鏈壘鍒板緟瀹℃壒鐨勮妭鐐�");
+      return;
+    }
+    currentActivity.approveNodeStatus = status;
+    // 鍒ゆ柇鏄惁涓烘渶鍚庝竴姝�
+    const isLast =
+      activities.value.findIndex(a => a.isShen) === activities.value.length - 1;
+    updateApproveNode({ ...currentActivity, isLast }).then(() => {
+      proxy.$modal.msgSuccess("鎻愪氦鎴愬姛");
+      closeDia();
+    });
+  };
+  // 鍏抽棴寮规
+  const closeDia = () => {
+    proxy.resetForm("formRef");
+    dialogFormVisible.value = false;
+    quotationLoading.value = false;
+    currentQuotation.value = {};
+    purchaseLoading.value = false;
+    currentPurchase.value = {};
+    emit("close");
+  };
+  defineExpose({
+    openDialog,
+  });
+</script>
+
+<style scoped>
+  .node-user {
+    margin: 10px 0;
+    font-size: 16px;
+    font-weight: 600;
+    display: flex;
+    align-items: center;
+    gap: 8px;
+  }
+  .node-status {
+    color: #1890ff;
+    margin-left: 8px;
+    font-size: 14px;
+  }
+  .node-reason {
+    font-size: 15px;
+    color: #333;
+    margin: 10px 0;
+  }
+  .user-avatar {
+    cursor: pointer;
+    width: 30px;
+    height: 30px;
+    border-radius: 50px;
+  }
+  .signImg {
+    cursor: pointer;
+    width: 200px;
+    height: 60px;
+  }
+</style>
\ No newline at end of file

--
Gitblit v1.9.3