| | |
| | | <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="请选择产品(可多选)" |
| | | <div class="binding-dialog"> |
| | | <el-input |
| | | v-model="productSearchKeyword" |
| | | placeholder="搜索产品" |
| | | clearable |
| | | check-strictly |
| | | :data="productOptions" |
| | | :render-after-expand="false" |
| | | style="width: 100%" |
| | | prefix-icon="Search" |
| | | class="binding-search" |
| | | /> |
| | | </el-form-item> |
| | | </el-form> |
| | | <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> |
| | |
| | | </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' |
| | |
| | | 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 = () => { |
| | |
| | | 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 () => { |
| | |
| | | onMounted(() => { |
| | | getStandardList() |
| | | getProcessList() |
| | | getProductTreeData() |
| | | }) |
| | | </script> |
| | | |
| | |
| | | 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> |