From 4a7aaf6852c8af9622ee86c799ef6067054fd308 Mon Sep 17 00:00:00 2001
From: yuan <123@>
Date: 星期五, 26 六月 2026 20:08:39 +0800
Subject: [PATCH] fix: 指标绑定的添加绑定选择页面优化增加模糊查询

---
 src/views/qualityManagement/metricBinding/index.vue |  179 ++++++++++++++++++++++++++++++++++++++++++++++++-----------
 1 files changed, 144 insertions(+), 35 deletions(-)

diff --git a/src/views/qualityManagement/metricBinding/index.vue b/src/views/qualityManagement/metricBinding/index.vue
index 1ac268a..ba989ea 100644
--- a/src/views/qualityManagement/metricBinding/index.vue
+++ b/src/views/qualityManagement/metricBinding/index.vue
@@ -120,25 +120,45 @@
     <el-dialog
       v-model="bindingDialogVisible"
       title="娣诲姞缁戝畾"
-      width="520px"
+      width="640px"
       @close="closeBindingDialog"
     >
-      <el-form label-width="100px">
-        <el-form-item label="浜у搧">
-          <el-tree-select
-            v-model="selectedProductIds"
-            multiple
-            collapse-tags
-            collapse-tags-tooltip
-            placeholder="璇烽�夋嫨浜у搧锛堝彲澶氶�夛級"
-            clearable
-            check-strictly
-            :data="productOptions"
-            :render-after-expand="false"
-            style="width: 100%"
-          />
-        </el-form-item>
-      </el-form>
+      <div class="binding-dialog">
+        <el-input
+          v-model="productSearchKeyword"
+          placeholder="鎼滅储浜у搧"
+          clearable
+          prefix-icon="Search"
+          class="binding-search"
+        />
+        <el-tree
+          ref="productTreeRef"
+          :key="productTreeKey"
+          v-loading="productTreeLoading"
+          :data="productTreeData"
+          show-checkbox
+          check-strictly
+          node-key="id"
+          :props="{ label: 'label', children: 'children', disabled: 'disabled' }"
+          :filter-node-method="filterProductNode"
+          class="product-binding-tree"
+          @check="handleProductTreeCheck"
+        />
+        <div v-if="selectedProductLabels.length" class="selected-products">
+          <div class="selected-title">宸查�夋嫨锛坽{ selectedProductLabels.length }}锛�</div>
+          <div class="selected-tags">
+            <el-tag
+              v-for="item in selectedProductLabels"
+              :key="item.id"
+              closable
+              type="info"
+              @close="removeSelectedProduct(item.id)"
+            >
+              {{ item.label }}
+            </el-tag>
+          </div>
+        </div>
+      </div>
       <template #footer>
         <span class="dialog-footer">
           <el-button @click="closeBindingDialog">鍙栨秷</el-button>
@@ -150,8 +170,7 @@
 </template>
 
 <script setup>
-import { Search } from '@element-plus/icons-vue'
-import { ref, reactive, toRefs, onMounted, getCurrentInstance } from 'vue'
+import { ref, reactive, toRefs, computed, watch, nextTick, onMounted, getCurrentInstance } from 'vue'
 import { ElMessageBox } from 'element-plus'
 import PIMTable from '@/components/PIMTable/PIMTable.vue'
 import { productTreeList } from '@/api/basicData/product.js'
@@ -261,28 +280,74 @@
 const bindingDialogVisible = ref(false)
 
 // 浜у搧鏍戯紙鐢ㄤ簬缁戝畾閫夋嫨锛�
-const productOptions = ref([])
+const productTreeData = ref([])
+const productTreeLoading = ref(false)
+const productTreeRef = ref(null)
+const productTreeKey = ref(0)
+const productSearchKeyword = ref('')
 const selectedProductIds = ref([])
 
-const getProductOptions = async () => {
-  // 閬垮厤閲嶅璇锋眰
-  if (productOptions.value?.length) return
-  const res = await productTreeList()
-  productOptions.value = convertIdToValue(Array.isArray(res) ? res : [])
+const markParentNodesDisabled = (nodes) => {
+  return (nodes || []).map((node) => {
+    const children = node.children?.length ? markParentNodesDisabled(node.children) : []
+    return {
+      ...node,
+      children,
+      disabled: children.length > 0
+    }
+  })
 }
 
-function convertIdToValue(data) {
-  return (data || []).map((item) => {
-    const { id, children, ...rest } = item
-    const newItem = {
-      ...rest,
-      value: id
+const buildProductLabelMap = (nodes, map = {}) => {
+  ;(nodes || []).forEach((node) => {
+    if (node.id != null) {
+      map[node.id] = node.label
     }
-    if (children && children.length > 0) {
-      newItem.children = convertIdToValue(children)
+    if (node.children?.length) {
+      buildProductLabelMap(node.children, map)
     }
-    return newItem
   })
+  return map
+}
+
+const productLabelMap = computed(() => buildProductLabelMap(productTreeData.value))
+
+const selectedProductLabels = computed(() =>
+  selectedProductIds.value.map((id) => ({
+    id,
+    label: productLabelMap.value[id] || id
+  }))
+)
+
+const filterProductNode = (value, data) => {
+  if (!value) return true
+  return String(data.label || '').includes(value)
+}
+
+watch(productSearchKeyword, (val) => {
+  productTreeRef.value?.filter(val)
+})
+
+const getProductTreeData = async () => {
+  if (productTreeData.value?.length) return
+  productTreeLoading.value = true
+  try {
+    const res = await productTreeList()
+    productTreeData.value = markParentNodesDisabled(Array.isArray(res) ? res : [])
+  } catch (error) {
+    console.error('鑾峰彇浜у搧鏍戝け璐�:', error)
+  } finally {
+    productTreeLoading.value = false
+  }
+}
+
+const handleProductTreeCheck = () => {
+  selectedProductIds.value = productTreeRef.value?.getCheckedKeys(true) || []
+}
+
+const removeSelectedProduct = (id) => {
+  selectedProductIds.value = selectedProductIds.value.filter((item) => item !== id)
+  productTreeRef.value?.setChecked(id, false, false)
 }
 
 const handleQuery = () => {
@@ -354,12 +419,20 @@
 const openBindingDialog = () => {
   if (!currentStandard.value?.id) return
   selectedProductIds.value = []
-  getProductOptions()
+  productSearchKeyword.value = ''
+  productTreeKey.value += 1
   bindingDialogVisible.value = true
+  nextTick(() => {
+    productTreeRef.value?.setCheckedKeys([])
+    productTreeRef.value?.filter('')
+  })
 }
 
 const closeBindingDialog = () => {
   bindingDialogVisible.value = false
+  selectedProductIds.value = []
+  productSearchKeyword.value = ''
+  productTreeRef.value?.setCheckedKeys([])
 }
 
 const submitBinding = async () => {
@@ -428,6 +501,7 @@
 onMounted(() => {
   getStandardList()
   getProcessList()
+  getProductTreeData()
 })
 </script>
 
@@ -533,4 +607,39 @@
   width: 100%;
   margin-top: 4px;
 }
+
+.binding-dialog {
+  display: flex;
+  flex-direction: column;
+  gap: 12px;
+}
+
+.binding-search {
+  width: 100%;
+}
+
+.product-binding-tree {
+  max-height: 360px;
+  overflow-y: auto;
+  border: 1px solid #ebeef5;
+  border-radius: 4px;
+  padding: 8px;
+}
+
+.selected-products {
+  border-top: 1px solid #ebeef5;
+  padding-top: 12px;
+}
+
+.selected-title {
+  font-size: 13px;
+  color: #606266;
+  margin-bottom: 8px;
+}
+
+.selected-tags {
+  display: flex;
+  flex-wrap: wrap;
+  gap: 8px;
+}
 </style>

--
Gitblit v1.9.3