CRM-合同:完善 TODO

pull/381/head
puhui999 2024-02-03 20:47:50 +08:00
parent 4549f8ed3c
commit 6d0e242387
16 changed files with 194 additions and 80 deletions

View File

@ -73,7 +73,6 @@ export const getBusinessListByIds = async (val: number[]) => {
} }
// 商机转移 // 商机转移
// TODO @puhui999transferBusiness export const transferBusiness = async (data: TransferReqVO) => {
export const transfer = async (data: TransferReqVO) => { return await request.put({ url: '/crm/business/transfer-business', data })
return await request.put({ url: '/crm/business/transfer', data })
} }

View File

@ -1,4 +1,5 @@
import request from '@/config/axios' import request from '@/config/axios'
import { TransferReqVO } from '@/api/crm/customer'
export interface ClueVO { export interface ClueVO {
id: number id: number
@ -44,3 +45,8 @@ export const deleteClue = async (id: number) => {
export const exportClue = async (params) => { export const exportClue = async (params) => {
return await request.download({ url: `/crm/clue/export-excel`, params }) return await request.download({ url: `/crm/clue/export-excel`, params })
} }
// 线索转移
export const transferClue = async (data: TransferReqVO) => {
return await request.put({ url: '/crm/clue/transfer-clue', data })
}

View File

@ -88,6 +88,6 @@ export const deleteContactBusinessList = async (data: ContactBusinessReqVO) => {
} }
// 联系人转移 // 联系人转移
export const transfer = async (data: TransferReqVO) => { export const transferContact = async (data: TransferReqVO) => {
return await request.put({ url: '/crm/contact/transfer', data }) return await request.put({ url: '/crm/contact/transfer-contact', data })
} }

View File

@ -7,6 +7,7 @@ export interface ContractVO {
name: string name: string
customerId: number customerId: number
businessId: number businessId: number
businessName: string
processInstanceId: number processInstanceId: number
orderDate: Date orderDate: Date
ownerUserId: number ownerUserId: number
@ -18,8 +19,9 @@ export interface ContractVO {
productPrice: number productPrice: number
contactId: number contactId: number
signUserId: number signUserId: number
signUserName: string
contactLastTime: Date contactLastTime: Date
status: number auditStatus: number
remark: string remark: string
productItems: ProductExpandVO[] productItems: ProductExpandVO[]
creatorName: string creatorName: string
@ -71,7 +73,6 @@ export const handleApprove = async (id: number) => {
} }
// 合同转移 // 合同转移
// TODO @puhui999transfer 相关方法,这块要补充下; export const transferContract = async (data: TransferReqVO) => {
export const transfer = async (data: TransferReqVO) => { return await request.put({ url: '/crm/contract/transfer-contract', data })
return await request.put({ url: '/crm/contract/transfer', data })
} }

View File

@ -82,8 +82,8 @@ export interface TransferReqVO {
} }
// 客户转移 // 客户转移
export const transfer = async (data: TransferReqVO) => { export const transferCustomer = async (data: TransferReqVO) => {
return await request.put({ url: '/crm/customer/transfer', data }) return await request.put({ url: '/crm/customer/transfer-customer', data })
} }
// 锁定/解锁客户 // 锁定/解锁客户

View File

@ -1,4 +1,4 @@
<!-- TODO @puhui999这个最好加个注释哈 --> <!-- 列表选择通用组件参考 ProductList 组件使用 -->
<template> <template>
<Dialog v-model="dialogVisible" :appendToBody="true" :scroll="true" :title="title" width="60%"> <Dialog v-model="dialogVisible" :appendToBody="true" :scroll="true" :title="title" width="60%">
<el-table <el-table

View File

@ -79,7 +79,7 @@ const openForm = (type: string, id?: number) => {
/** 联系人转移 */ /** 联系人转移 */
const crmTransferFormRef = ref<InstanceType<typeof CrmTransferForm>>() // ref const crmTransferFormRef = ref<InstanceType<typeof CrmTransferForm>>() // ref
const transfer = () => { const transfer = () => {
crmTransferFormRef.value?.open('联系人转移', contact.value.id, ContactApi.transfer) crmTransferFormRef.value?.open('联系人转移', contact.value.id, ContactApi.transferContact)
} }
/** 获取操作日志 */ /** 获取操作日志 */

View File

@ -7,7 +7,7 @@
:rules="formRules" :rules="formRules"
label-width="110px" label-width="110px"
> >
<el-row> <el-row :gutter="20">
<el-col :span="24" class="mb-10px"> <el-col :span="24" class="mb-10px">
<CardTitle title="基本信息" /> <CardTitle title="基本信息" />
</el-col> </el-col>
@ -138,7 +138,7 @@
</el-col> </el-col>
<el-col :span="12"> <el-col :span="12">
<el-form-item label="产品总金额(元)" prop="productPrice"> <el-form-item label="产品总金额(元)" prop="productPrice">
<el-input v-model="formData.productPrice" placeholder="请输入产品总金额" /> {{ floatToFixed2(formData.productPrice) }}
</el-form-item> </el-form-item>
</el-col> </el-col>
<el-col :span="24"> <el-col :span="24">
@ -172,6 +172,7 @@ import * as ContactApi from '@/api/crm/contact'
import * as BusinessApi from '@/api/crm/business' import * as BusinessApi from '@/api/crm/business'
import ProductList from './components/ProductList.vue' import ProductList from './components/ProductList.vue'
import BPMLModel from '@/views/crm/contract/components/BPMLModel.vue' import BPMLModel from '@/views/crm/contract/components/BPMLModel.vue'
import { floatToFixed2 } from '@/utils'
const { t } = useI18n() // const { t } = useI18n() //
const message = useMessage() // const message = useMessage() //
@ -191,7 +192,7 @@ const formRules = reactive({
const formRef = ref() // Ref const formRef = ref() // Ref
const BPMLModelRef = ref<InstanceType<typeof BPMLModel>>() // TODO @puhui999 bpm model const BPMLModelRef = ref<InstanceType<typeof BPMLModel>>() // TODO @puhui999 bpm model
// TODO @puhui999 //
watch( watch(
() => formData.value.productItems, () => formData.value.productItems,
(val) => { (val) => {

View File

@ -66,14 +66,17 @@ import * as ProductApi from '@/api/crm/product'
import { DICT_TYPE } from '@/utils/dict' import { DICT_TYPE } from '@/utils/dict'
import { fenToYuanFormat } from '@/utils/formatter' import { fenToYuanFormat } from '@/utils/formatter'
import { TableSelectForm } from '@/components/Table/index' import { TableSelectForm } from '@/components/Table/index'
import { floatToFixed2, yuanToFen } from '@/utils'
defineOptions({ name: 'ProductList' }) defineOptions({ name: 'ProductList' })
withDefaults(defineProps<{ modelValue: any[] }>(), { modelValue: () => [] }) const props = withDefaults(defineProps<{ modelValue: ProductApi.ProductExpandVO[] }>(), {
modelValue: () => []
})
const emits = defineEmits<{ const emits = defineEmits<{
(e: 'update:modelValue', v: any[]): void (e: 'update:modelValue', v: any[]): void
}>() }>()
const list = ref<ProductApi.ProductExpandVO[]>([]) // TODO @puhui999 const list = ref<ProductApi.ProductExpandVO[]>([]) //
const multipleSelection = ref<ProductApi.ProductExpandVO[]>([]) // const multipleSelection = ref<ProductApi.ProductExpandVO[]>([]) //
/** 处理删除 */ /** 处理删除 */
@ -92,12 +95,30 @@ const openForm = () => {
/** 计算 totalPrice */ /** 计算 totalPrice */
const getTotalPrice = computed(() => (row: ProductApi.ProductExpandVO) => { const getTotalPrice = computed(() => (row: ProductApi.ProductExpandVO) => {
const totalPrice = (row.price * row.count * row.discountPercent) / 100 const totalPrice =
row.totalPrice = isNaN(totalPrice) ? 0 : totalPrice (Number(row.price) / 100) * Number(row.count) * (1 - Number(row.discountPercent) / 100)
return isNaN(totalPrice) ? 0 : totalPrice row.totalPrice = isNaN(totalPrice) ? 0 : yuanToFen(totalPrice)
return isNaN(totalPrice) ? 0 : totalPrice.toFixed(2)
}) })
const isSetListValue = ref(false) // list
// TODO @puhui999 //
watch(
() => props.modelValue,
(val) => {
if (!val || val.length === 0 || isSetListValue.value) {
return
}
list.value = [
...props.modelValue.map((item) => {
item.totalPrice = floatToFixed2(item.totalPrice) as unknown as number
return item
})
]
isSetListValue.value = true
},
{ immediate: true, deep: true }
)
//
watch( watch(
list, list,
(val) => { (val) => {
@ -109,15 +130,20 @@ watch(
{ deep: true } { deep: true }
) )
// TODO @puhui999 // ,
watch( watch(
multipleSelection, multipleSelection,
(val) => { (val) => {
if (!val || val.length === 0) { if (!val || val.length === 0) {
return return
} }
//
const ids = list.value.map((item) => item.id) const ids = list.value.map((item) => item.id)
list.value.push(...multipleSelection.value.filter((item) => ids.indexOf(item.id) === -1)) const productList = multipleSelection.value.filter((item) => ids.indexOf(item.id) === -1)
if (!productList || productList.length === 0) {
return
}
list.value.push(...productList)
}, },
{ deep: true } { deep: true }
) )

View File

@ -1,4 +1,4 @@
<!-- TODO @puhui999这个组件的注释加下方便大家打开就知道哈 --> <!-- 合同详情头部组件-->
<template> <template>
<div> <div>
<div class="flex items-start justify-between"> <div class="flex items-start justify-between">
@ -17,17 +17,20 @@
</div> </div>
<ContentWrap class="mt-10px"> <ContentWrap class="mt-10px">
<el-descriptions :column="5" direction="vertical"> <el-descriptions :column="5" direction="vertical">
<el-descriptions-item label="客户"> <el-descriptions-item label="客户名称">
{{ contract.customerName }} {{ contract.customerName }}
</el-descriptions-item> </el-descriptions-item>
<el-descriptions-item label="客户签约人"> <el-descriptions-item label="合同金额(元)">
{{ contract.contactName }} {{ floatToFixed2(contract.price) }}
</el-descriptions-item> </el-descriptions-item>
<el-descriptions-item label="合同金额"> <el-descriptions-item label="下单时间">
{{ contract.productPrice }} {{ contract.orderDate ? formatDate(contract.orderDate) : '空' }}
</el-descriptions-item> </el-descriptions-item>
<el-descriptions-item label="创建时间"> <el-descriptions-item label="回款金额(元)">
{{ contract.createTime ? formatDate(contract.createTime) : '空' }} {{ floatToFixed2(contract.price) }}
</el-descriptions-item>
<el-descriptions-item label="负责人">
{{ contract.ownerUserName }}
</el-descriptions-item> </el-descriptions-item>
</el-descriptions> </el-descriptions>
</ContentWrap> </ContentWrap>
@ -35,6 +38,7 @@
<script lang="ts" setup> <script lang="ts" setup>
import * as ContractApi from '@/api/crm/contract' import * as ContractApi from '@/api/crm/contract'
import { formatDate } from '@/utils/formatTime' import { formatDate } from '@/utils/formatTime'
import { floatToFixed2 } from '@/utils'
defineOptions({ name: 'ContractDetailsHeader' }) defineOptions({ name: 'ContractDetailsHeader' })
defineProps<{ contract: ContractApi.ContractVO }>() defineProps<{ contract: ContractApi.ContractVO }>()

View File

@ -1,4 +1,4 @@
<!-- TODO @puhui999这个组件的注释加下方便大家打开就知道哈 --> <!-- 合同详情组件 -->
<template> <template>
<ContentWrap> <ContentWrap>
<el-collapse v-model="activeNames"> <el-collapse v-model="activeNames">
@ -6,14 +6,43 @@
<template #title> <template #title>
<span class="text-base font-bold">基本信息</span> <span class="text-base font-bold">基本信息</span>
</template> </template>
<!-- TODO puhui999: 先出详情样式后补全 --> <el-descriptions :column="3">
<el-descriptions :column="4"> <el-descriptions-item label="合同编号">
{{ contract.no }}
</el-descriptions-item>
<el-descriptions-item label="合同名称"> <el-descriptions-item label="合同名称">
{{ contract.name }} {{ contract.name }}
</el-descriptions-item> </el-descriptions-item>
<el-descriptions-item label="客户名称">
{{ contract.customerName }}
</el-descriptions-item>
<el-descriptions-item label="商机名称">
{{ contract.businessName }}
</el-descriptions-item>
<el-descriptions-item label="合同金额(元)">
{{ contract.price }}
</el-descriptions-item>
<el-descriptions-item label="下单时间">
{{ formatDate(contract.orderDate) }}
</el-descriptions-item>
<el-descriptions-item label="开始时间">
{{ formatDate(contract.startTime) }}
</el-descriptions-item>
<el-descriptions-item label="结束时间">
{{ formatDate(contract.endTime) }}
</el-descriptions-item>
<el-descriptions-item label="客户签约人">
{{ contract.contactName }}
</el-descriptions-item>
<el-descriptions-item label="公司签约人">
{{ contract.signUserName }}
</el-descriptions-item>
<el-descriptions-item label="备注"> <el-descriptions-item label="备注">
{{ contract.remark }} {{ contract.remark }}
</el-descriptions-item> </el-descriptions-item>
<el-descriptions-item label="合同状态">
{{ contract.auditStatus }}
</el-descriptions-item>
</el-descriptions> </el-descriptions>
</el-collapse-item> </el-collapse-item>
<el-collapse-item name="systemInfo"> <el-collapse-item name="systemInfo">

View File

@ -0,0 +1,64 @@
<template>
<el-table :data="list" :show-overflow-tooltip="true" :stripe="true">
<el-table-column align="center" label="产品名称" prop="name" width="160" />
<el-table-column align="center" label="产品类型" prop="categoryName" width="160" />
<el-table-column align="center" label="产品单位" prop="unit">
<template #default="scope">
<dict-tag :type="DICT_TYPE.CRM_PRODUCT_UNIT" :value="scope.row.unit" />
</template>
</el-table-column>
<el-table-column align="center" label="产品编码" prop="no" />
<el-table-column
:formatter="fenToYuanFormat"
align="center"
label="价格(元)"
prop="price"
width="100"
/>
<el-table-column align="center" label="数量" prop="count" width="200" />
<el-table-column align="center" label="折扣(%)" prop="discountPercent" width="200" />
<el-table-column align="center" label="合计" prop="totalPrice" width="100">
<template #default="{ row }: { row: ProductApi.ProductExpandVO }">
{{ getTotalPrice(row) }}
</template>
</el-table-column>
</el-table>
</template>
<script lang="ts" setup>
import { DICT_TYPE } from '@/utils/dict'
import { fenToYuanFormat } from '@/utils/formatter'
import * as ProductApi from '@/api/crm/product'
import { floatToFixed2, yuanToFen } from '@/utils'
defineOptions({ name: 'ContractProductList' })
const props = withDefaults(defineProps<{ modelValue: ProductApi.ProductExpandVO[] }>(), {
modelValue: () => []
})
const list = ref<ProductApi.ProductExpandVO[]>([]) //
/** 计算 totalPrice */
const getTotalPrice = computed(() => (row: ProductApi.ProductExpandVO) => {
const totalPrice =
(Number(row.price) / 100) * Number(row.count) * (1 - Number(row.discountPercent) / 100)
row.totalPrice = isNaN(totalPrice) ? 0 : yuanToFen(totalPrice)
return isNaN(totalPrice) ? 0 : totalPrice.toFixed(2)
})
const isSetListValue = ref(false) // list
//
watch(
() => props.modelValue,
(val) => {
if (!val || val.length === 0 || isSetListValue.value) {
return
}
list.value = [
...props.modelValue.map((item) => {
item.totalPrice = floatToFixed2(item.totalPrice) as unknown as number
return item
})
]
isSetListValue.value = true
},
{ immediate: true, deep: true }
)
</script>

View File

@ -1,10 +1,10 @@
<!-- TODO @puhui999这个组件的注释加下方便大家打开就知道哈 --> <!-- 合同详情页面组件-->
<template> <template>
<ContractDetailsHeader v-loading="loading" :contract="contract"> <ContractDetailsHeader v-loading="loading" :contract="contract">
<el-button v-if="permissionListRef?.validateWrite" @click="openForm('update', contract.id)"> <el-button v-if="permissionListRef?.validateWrite" @click="openForm('update', contract.id)">
编辑 编辑
</el-button> </el-button>
<el-button v-if="permissionListRef?.validateOwnerUser" type="primary" @click="transfer"> <el-button v-if="permissionListRef?.validateOwnerUser" type="primary" @click="transferContract">
转移 转移
</el-button> </el-button>
</ContractDetailsHeader> </ContractDetailsHeader>
@ -13,6 +13,9 @@
<el-tab-pane label="详细资料"> <el-tab-pane label="详细资料">
<ContractDetailsInfo :contract="contract" /> <ContractDetailsInfo :contract="contract" />
</el-tab-pane> </el-tab-pane>
<el-tab-pane label="产品">
<ContractProductList v-model="contract.productItems" />
</el-tab-pane>
<el-tab-pane label="操作日志"> <el-tab-pane label="操作日志">
<OperateLogV2 :log-list="logList" /> <OperateLogV2 :log-list="logList" />
</el-tab-pane> </el-tab-pane>
@ -25,18 +28,11 @@
@quit-team="close" @quit-team="close"
/> />
</el-tab-pane> </el-tab-pane>
<el-tab-pane label="商机" lazy>
<BusinessList
:biz-id="contract.id!"
:biz-type="BizTypeEnum.CRM_CONTRACT"
:customer-id="contract.customerId"
/>
</el-tab-pane>
</el-tabs> </el-tabs>
</el-col> </el-col>
<!-- 表单弹窗添加/修改 --> <!-- 表单弹窗添加/修改 -->
<ContractForm ref="formRef" @success="getContractData" /> <ContractForm ref="formRef" @success="getContractData" />
<CrmTransferForm ref="crmTransferFormRef" @success="close" /> <CrmTransferForm ref="transferFormRef" @success="close" />
</template> </template>
<script lang="ts" setup> <script lang="ts" setup>
import { useTagsViewStore } from '@/store/modules/tagsView' import { useTagsViewStore } from '@/store/modules/tagsView'
@ -44,12 +40,12 @@ import { OperateLogV2VO } from '@/api/system/operatelog'
import * as ContractApi from '@/api/crm/contract' import * as ContractApi from '@/api/crm/contract'
import ContractDetailsHeader from './ContractDetailsHeader.vue' import ContractDetailsHeader from './ContractDetailsHeader.vue'
import ContractDetailsInfo from './ContractDetailsInfo.vue' import ContractDetailsInfo from './ContractDetailsInfo.vue'
import ContractProductList from './ContractProductList.vue'
import { BizTypeEnum } from '@/api/crm/permission' import { BizTypeEnum } from '@/api/crm/permission'
import { getOperateLogPage } from '@/api/crm/operateLog' import { getOperateLogPage } from '@/api/crm/operateLog'
import ContractForm from '@/views/crm/contract/ContractForm.vue' import ContractForm from '@/views/crm/contract/ContractForm.vue'
import CrmTransferForm from '@/views/crm/permission/components/TransferForm.vue' import CrmTransferForm from '@/views/crm/permission/components/TransferForm.vue'
import PermissionList from '@/views/crm/permission/components/PermissionList.vue' import PermissionList from '@/views/crm/permission/components/PermissionList.vue'
import BusinessList from '@/views/crm/business/components/BusinessList.vue'
defineOptions({ name: 'CrmContractDetail' }) defineOptions({ name: 'CrmContractDetail' })
@ -91,10 +87,9 @@ const getOperateLog = async (contractId: number) => {
} }
/** 转移 */ /** 转移 */
// TODO @puhui999transferFormRef const transferFormRef = ref<InstanceType<typeof CrmTransferForm>>() // ref
const crmTransferFormRef = ref<InstanceType<typeof CrmTransferForm>>() // ref const transferContract = () => {
const transfer = () => { transferFormRef.value?.open('合同转移', contract.value.id, ContractApi.transferContract)
crmTransferFormRef.value?.open('合同转移', contract.value.id, ContractApi.transfer)
} }
/** 关闭 */ /** 关闭 */

View File

@ -57,54 +57,44 @@
<!-- TODO 芋艿各种字段要调整 --> <!-- TODO 芋艿各种字段要调整 -->
<ContentWrap> <ContentWrap>
<el-table v-loading="loading" :data="list" :show-overflow-tooltip="true" :stripe="true"> <el-table v-loading="loading" :data="list" :show-overflow-tooltip="true" :stripe="true">
<el-table-column align="center" label="合同编号" prop="id" /> <el-table-column align="center" label="合同编号" prop="no" width="130" />
<el-table-column align="center" label="合同名称" prop="name" /> <el-table-column align="center" label="合同名称" prop="name" width="130" />
<el-table-column align="center" label="客户名称" prop="customerId" /> <el-table-column align="center" label="合同状态" prop="auditStatus" width="130" />
<el-table-column align="center" label="商机名称" prop="businessId" /> <el-table-column align="center" label="客户名称" prop="customerName" width="130" />
<el-table-column align="center" label="工作流名称" prop="processInstanceId" /> <el-table-column align="center" label="商机名称" prop="businessName" width="130" />
<el-table-column align="center" label="合同金额(元)" prop="price" width="130" />
<el-table-column <el-table-column
:formatter="dateFormatter" :formatter="dateFormatter"
align="center" align="center"
label="下单时间" label="下单时间"
prop="orderDate" prop="orderDate"
width="180px" width="180"
/> />
<el-table-column align="center" label="负责人" prop="ownerUserId" />
<el-table-column align="center" label="合同编号" prop="no" />
<el-table-column <el-table-column
:formatter="dateFormatter" :formatter="dateFormatter"
align="center" align="center"
label="开始时间" label="开始时间"
prop="startTime" prop="startTime"
width="180px" width="180"
/> />
<el-table-column <el-table-column
:formatter="dateFormatter" :formatter="dateFormatter"
align="center" align="center"
label="结束时间" label="结束时间"
prop="endTime" prop="endTime"
width="180px" width="180"
/> />
<el-table-column align="center" label="合同金额" prop="price" /> <el-table-column align="center" label="客户签约人" prop="contactName" width="130" />
<el-table-column align="center" label="整单折扣" prop="discountPercent" /> <el-table-column align="center" label="公司签约人" prop="signUserName" width="130" />
<el-table-column align="center" label="产品总金额" prop="productPrice" /> <el-table-column align="center" label="备注" prop="remark" width="130" />
<el-table-column align="center" label="联系人" prop="contactId" /> <el-table-column align="center" label="审核状态" prop="auditStatus" width="130" />
<el-table-column align="center" label="公司签约人" prop="signUserId" />
<el-table-column <el-table-column
:formatter="dateFormatter" :formatter="dateFormatter"
align="center" align="center"
label="最后跟进时间" label="最后跟进时间"
prop="contactLastTime" prop="contactLastTime"
width="180px" width="180"
/> />
<el-table-column
:formatter="dateFormatter"
align="center"
label="创建时间"
prop="createTime"
width="180px"
/>
<el-table-column align="center" label="备注" prop="remark" />
<el-table-column fixed="right" label="操作" width="250"> <el-table-column fixed="right" label="操作" width="250">
<template #default="scope"> <template #default="scope">
<el-button <el-button

View File

@ -124,7 +124,7 @@ const openForm = () => {
/** 客户转移 */ /** 客户转移 */
const crmTransferFormRef = ref<InstanceType<typeof CrmTransferForm>>() // ref const crmTransferFormRef = ref<InstanceType<typeof CrmTransferForm>>() // ref
const transfer = () => { const transfer = () => {
crmTransferFormRef.value?.open('客户转移', customerId.value, CustomerApi.transfer) crmTransferFormRef.value?.open('客户转移', customerId.value, CustomerApi.transferCustomer)
} }
/** 锁定客户 */ /** 锁定客户 */

View File

@ -208,7 +208,7 @@
<!-- 表单弹窗添加/修改 --> <!-- 表单弹窗添加/修改 -->
<CustomerForm ref="formRef" @success="getList" /> <CustomerForm ref="formRef" @success="getList" />
<CustomerImportForm ref="customerImportFormRef" @success="getList" /> <CustomerImportForm ref="importFormRef" @success="getList" />
</template> </template>
<script lang="ts" setup> <script lang="ts" setup>
@ -341,10 +341,9 @@ const handleDelete = async (id: number) => {
} }
/** 导入按钮操作 */ /** 导入按钮操作 */
// TODO @puhui999importFormRef const importFormRef = ref<InstanceType<typeof CustomerImportForm>>()
const customerImportFormRef = ref<InstanceType<typeof CustomerImportForm>>()
const handleImport = () => { const handleImport = () => {
customerImportFormRef.value?.open() importFormRef.value?.open()
} }
/** 导出按钮操作 */ /** 导出按钮操作 */