Files
one-pipe-system/src/views/finance/commission/my-commission/index.vue
sexygoat 192c6f1d26
All checks were successful
构建并部署前端到测试环境 / build-and-deploy (push) Successful in 4m25s
fetch(modify):完善按钮权限
2026-02-03 17:20:50 +08:00

878 lines
26 KiB
Vue
Raw Blame History

This file contains ambiguous Unicode characters
This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.
<template>
<div class="my-commission-page">
<!-- 佣金概览卡片 -->
<ElRow :gutter="20" style="margin-bottom: 20px">
<ElCol :xs="24" :sm="12" :md="8" :lg="4">
<ElCard shadow="hover">
<div class="stat-card">
<div
class="stat-icon"
style="background: linear-gradient(135deg, #667eea 0%, #764ba2 100%)"
>
<i class="iconfont-sys">&#xe71d;</i>
</div>
<div class="stat-content">
<div class="stat-label">总佣金</div>
<div class="stat-value">{{ formatMoney(summary.total_commission) }}</div>
</div>
</div>
</ElCard>
</ElCol>
<ElCol :xs="24" :sm="12" :md="8" :lg="4">
<ElCard shadow="hover">
<div class="stat-card">
<div
class="stat-icon"
style="background: linear-gradient(135deg, #f093fb 0%, #f5576c 100%)"
>
<i class="iconfont-sys">&#xe71e;</i>
</div>
<div class="stat-content">
<div class="stat-label">可提现佣金</div>
<div class="stat-value">{{ formatMoney(summary.available_commission) }}</div>
</div>
</div>
</ElCard>
</ElCol>
<ElCol :xs="24" :sm="12" :md="8" :lg="4">
<ElCard shadow="hover">
<div class="stat-card">
<div
class="stat-icon"
style="background: linear-gradient(135deg, #4facfe 0%, #00f2fe 100%)"
>
<i class="iconfont-sys">&#xe720;</i>
</div>
<div class="stat-content">
<div class="stat-label">冻结佣金</div>
<div class="stat-value">{{ formatMoney(summary.frozen_commission) }}</div>
</div>
</div>
</ElCard>
</ElCol>
<ElCol :xs="24" :sm="12" :md="8" :lg="4">
<ElCard shadow="hover">
<div class="stat-card">
<div
class="stat-icon"
style="background: linear-gradient(135deg, #fa709a 0%, #fee140 100%)"
>
<i class="iconfont-sys">&#xe71f;</i>
</div>
<div class="stat-content">
<div class="stat-label">提现中佣金</div>
<div class="stat-value">{{ formatMoney(summary.withdrawing_commission) }}</div>
</div>
</div>
</ElCard>
</ElCol>
<ElCol :xs="24" :sm="12" :md="8" :lg="4">
<ElCard shadow="hover">
<div class="stat-card">
<div
class="stat-icon"
style="background: linear-gradient(135deg, #43e97b 0%, #38f9d7 100%)"
>
<i class="iconfont-sys">&#xe721;</i>
</div>
<div class="stat-content">
<div class="stat-label">已提现佣金</div>
<div class="stat-value">{{ formatMoney(summary.withdrawn_commission) }}</div>
</div>
</div>
</ElCard>
</ElCol>
</ElRow>
<!-- 标签页 -->
<ElCard shadow="never">
<ElTabs v-model="activeTab">
<!-- 佣金明细 -->
<ElTabPane label="佣金明细" name="commission">
<!-- 搜索栏 -->
<ArtSearchBar
v-model:filter="commissionSearchForm"
:items="commissionSearchItems"
:show-expand="false"
@reset="handleCommissionReset"
@search="handleCommissionSearch"
/>
<!-- 表格头部 -->
<ArtTableHeader
:columnList="commissionColumnOptions"
v-model:columns="commissionColumnChecks"
@refresh="handleCommissionRefresh"
style="margin-top: 20px"
>
<template #left>
<ElButton type="primary" @click="showWithdrawalDialog" v-permission="'my_commission:add'">发起提现</ElButton>
</template>
</ArtTableHeader>
<!-- 表格 -->
<ArtTable
ref="commissionTableRef"
row-key="id"
:loading="commissionLoading"
:data="commissionList"
:currentPage="commissionPagination.page"
:pageSize="commissionPagination.pageSize"
:total="commissionPagination.total"
:marginTop="10"
:height="420"
@size-change="handleCommissionSizeChange"
@current-change="handleCommissionCurrentChange"
>
<template #default>
<ElTableColumn
v-for="col in commissionColumns"
:key="col.prop || col.type"
v-bind="col"
/>
</template>
</ArtTable>
</ElTabPane>
<!-- 提现记录 -->
<ElTabPane label="提现记录" name="withdrawal">
<!-- 搜索栏 -->
<ArtSearchBar
v-model:filter="withdrawalSearchForm"
:items="withdrawalSearchItems"
:show-expand="false"
@reset="handleWithdrawalReset"
@search="handleWithdrawalSearch"
/>
<!-- 表格头部 -->
<ArtTableHeader
:columnList="withdrawalColumnOptions"
v-model:columns="withdrawalColumnChecks"
@refresh="handleWithdrawalRefresh"
style="margin-top: 20px"
/>
<!-- 表格 -->
<ArtTable
ref="withdrawalTableRef"
row-key="id"
:loading="withdrawalLoading"
:data="withdrawalList"
:currentPage="withdrawalPagination.page"
:pageSize="withdrawalPagination.pageSize"
:total="withdrawalPagination.total"
:marginTop="10"
:height="500"
@size-change="handleWithdrawalSizeChange"
@current-change="handleWithdrawalCurrentChange"
>
<template #default>
<ElTableColumn
v-for="col in withdrawalColumns"
:key="col.prop || col.type"
v-bind="col"
/>
</template>
</ArtTable>
</ElTabPane>
</ElTabs>
</ElCard>
<!-- 发起提现对话框 -->
<ElDialog
v-model="withdrawalDialogVisible"
title="发起提现"
width="600px"
@close="handleDialogClose"
>
<ElForm
ref="withdrawalFormRef"
:model="withdrawalForm"
:rules="withdrawalRules"
label-width="120px"
>
<ElFormItem label="可提现金额">
<span style="font-size: 20px; font-weight: 500; color: var(--el-color-success)">
{{ formatMoney(summary.available_commission) }}
</span>
</ElFormItem>
<ElFormItem label="提现金额" prop="amount">
<ElInputNumber
v-model="withdrawalForm.amount"
:min="100"
:max="summary.available_commission"
:precision="0"
:step="100"
style="width: 100%"
placeholder="请输入提现金额(分)"
/>
<div style="margin-top: 4px; font-size: 12px; color: var(--el-text-color-secondary)">
金额单位为分如1元=100
</div>
</ElFormItem>
<ElFormItem label="提现方式" prop="withdrawal_method">
<ElRadioGroup v-model="withdrawalForm.withdrawal_method">
<ElRadio :label="WithdrawalMethod.ALIPAY">支付宝</ElRadio>
<ElRadio :label="WithdrawalMethod.WECHAT">微信</ElRadio>
<ElRadio :label="WithdrawalMethod.BANK">银行卡</ElRadio>
</ElRadioGroup>
</ElFormItem>
<!-- 支付宝/微信 -->
<template v-if="withdrawalForm.withdrawal_method !== WithdrawalMethod.BANK">
<ElFormItem label="账户名" prop="account_name">
<ElInput v-model="withdrawalForm.account_name" placeholder="请输入账户名" />
</ElFormItem>
<ElFormItem label="账号" prop="account_number">
<ElInput v-model="withdrawalForm.account_number" placeholder="请输入账号" />
</ElFormItem>
</template>
<!-- 银行卡 -->
<template v-if="withdrawalForm.withdrawal_method === WithdrawalMethod.BANK">
<ElFormItem label="银行名称" prop="bank_name">
<ElSelect
v-model="withdrawalForm.bank_name"
placeholder="请选择银行"
style="width: 100%"
>
<ElOption label="中国工商银行" value="中国工商银行" />
<ElOption label="中国建设银行" value="中国建设银行" />
<ElOption label="中国农业银行" value="中国农业银行" />
<ElOption label="中国银行" value="中国银行" />
<ElOption label="招商银行" value="招商银行" />
<ElOption label="交通银行" value="交通银行" />
<ElOption label="中国邮政储蓄银行" value="中国邮政储蓄银行" />
<ElOption label="其他银行" value="其他银行" />
</ElSelect>
</ElFormItem>
<ElFormItem label="账户名" prop="account_name">
<ElInput v-model="withdrawalForm.account_name" placeholder="请输入账户名" />
</ElFormItem>
<ElFormItem label="卡号" prop="account_number">
<ElInput v-model="withdrawalForm.account_number" placeholder="请输入银行卡号" />
</ElFormItem>
</template>
<ElFormItem label="备注" prop="remark">
<ElInput v-model="withdrawalForm.remark" type="textarea" :rows="3" placeholder="选填" />
</ElFormItem>
</ElForm>
<template #footer>
<div class="dialog-footer">
<ElButton @click="withdrawalDialogVisible = false">取消</ElButton>
<ElButton type="primary" @click="handleSubmitWithdrawal" :loading="submitLoading">
提交
</ElButton>
</div>
</template>
</ElDialog>
</div>
</template>
<script setup lang="ts">
import { h } from 'vue'
import { CommissionService } from '@/api/modules'
import { ElMessage, ElTag } from 'element-plus'
import type { FormInstance, FormRules } from 'element-plus'
import type {
MyCommissionSummary,
MyCommissionRecordItem,
WithdrawalRequestItem,
CommissionRecordQueryParams,
WithdrawalRequestQueryParams,
SubmitWithdrawalParams,
CommissionType,
CommissionStatus,
WithdrawalStatus
} from '@/types/api/commission'
import { WithdrawalMethod } from '@/types/api/commission'
import {
CommissionStatusMap,
CommissionTypeMap,
WithdrawalStatusMap,
WithdrawalMethodMap
} from '@/config/constants/commission'
import type { SearchFormItem } from '@/types'
import { useCheckedColumns } from '@/composables/useCheckedColumns'
import { useAuth } from '@/composables/useAuth'
import { formatDateTime, formatMoney } from '@/utils/business/format'
defineOptions({ name: 'MyCommission' })
const { hasAuth } = useAuth()
// 标签页
const activeTab = ref('commission')
// 佣金概览
const summary = ref<MyCommissionSummary>({
total_commission: 0,
available_commission: 0,
frozen_commission: 0,
withdrawing_commission: 0,
withdrawn_commission: 0
})
// ==================== 佣金明细 ====================
const commissionLoading = ref(false)
const commissionTableRef = ref()
// 搜索表单初始值
const initialCommissionSearchState = {
commission_type: undefined as CommissionType | undefined,
status: undefined as CommissionStatus | undefined,
start_time: '',
end_time: ''
}
// 搜索表单
const commissionSearchForm = reactive({ ...initialCommissionSearchState })
// 搜索表单配置
const commissionSearchItems = computed<SearchFormItem[]>(() => [
{
label: '佣金类型',
prop: 'commission_type',
type: 'select',
options: [
{ label: '一次性佣金', value: 'one_time' },
{ label: '长期佣金', value: 'long_term' }
],
config: {
clearable: true,
placeholder: '请选择佣金类型'
}
},
{
label: '状态',
prop: 'status',
type: 'select',
options: [
{ label: '已冻结', value: 1 },
{ label: '解冻中', value: 2 },
{ label: '已发放', value: 3 },
{ label: '已失效', value: 4 }
],
config: {
clearable: true,
placeholder: '请选择状态'
}
},
{
label: '时间范围',
prop: 'dateRange',
type: 'date',
config: {
type: 'daterange',
clearable: true,
valueFormat: 'YYYY-MM-DD HH:mm:ss',
startPlaceholder: '开始日期',
endPlaceholder: '结束日期'
},
onChange: ({ val }: { prop: string; val: any }) => {
if (val && val.length === 2) {
commissionSearchForm.start_time = val[0]
commissionSearchForm.end_time = val[1]
} else {
commissionSearchForm.start_time = ''
commissionSearchForm.end_time = ''
}
}
}
])
// 分页
const commissionPagination = reactive({
page: 1,
pageSize: 20,
total: 0
})
// 列配置
const commissionColumnOptions = [
{ label: 'ID', prop: 'id' },
{ label: '佣金金额', prop: 'amount' },
{ label: '佣金类型', prop: 'commission_type' },
{ label: '状态', prop: 'status' },
{ label: '订单ID', prop: 'order_id' },
{ label: '创建时间', prop: 'created_at' }
]
const commissionList = ref<MyCommissionRecordItem[]>([])
// 动态列配置
const { columnChecks: commissionColumnChecks, columns: commissionColumns } = useCheckedColumns(
() => [
{
prop: 'id',
label: 'ID',
width: 80
},
{
prop: 'amount',
label: '佣金金额',
width: 140,
formatter: (row: MyCommissionRecordItem) => formatMoney(row.amount)
},
{
prop: 'commission_type',
label: '佣金类型',
width: 120,
formatter: (row: MyCommissionRecordItem) => {
const config = CommissionTypeMap[row.commission_type as keyof typeof CommissionTypeMap]
return h(ElTag, { type: config.type }, () => config.label)
}
},
{
prop: 'status',
label: '状态',
width: 100,
formatter: (row: MyCommissionRecordItem) => {
const config = CommissionStatusMap[row.status as keyof typeof CommissionStatusMap]
return h(ElTag, { type: config.type }, () => config.label)
}
},
{
prop: 'order_id',
label: '订单ID',
width: 100,
formatter: (row: MyCommissionRecordItem) => row.order_id || '-'
},
{
prop: 'shop_id',
label: '店铺ID',
width: 100
},
{
prop: 'created_at',
label: '创建时间',
width: 180,
formatter: (row: MyCommissionRecordItem) => formatDateTime(row.created_at)
}
]
)
// 获取佣金明细
const getCommissionList = async () => {
commissionLoading.value = true
try {
const params = {
pageSize: commissionPagination.pageSize,
current: commissionPagination.page,
commission_type: commissionSearchForm.commission_type,
status: commissionSearchForm.status,
start_time: commissionSearchForm.start_time || undefined,
end_time: commissionSearchForm.end_time || undefined
} as CommissionRecordQueryParams
const res = await CommissionService.getMyCommissionRecords(params)
if (res.code === 0) {
commissionList.value = res.data.items || []
commissionPagination.total = res.data.total || 0
}
} catch (error) {
console.error(error)
} finally {
commissionLoading.value = false
}
}
// 重置搜索
const handleCommissionReset = () => {
Object.assign(commissionSearchForm, { ...initialCommissionSearchState })
commissionPagination.page = 1
getCommissionList()
}
// 搜索
const handleCommissionSearch = () => {
commissionPagination.page = 1
getCommissionList()
}
// 刷新表格
const handleCommissionRefresh = () => {
getCommissionList()
}
// 处理表格分页变化
const handleCommissionSizeChange = (newPageSize: number) => {
commissionPagination.pageSize = newPageSize
getCommissionList()
}
const handleCommissionCurrentChange = (newCurrentPage: number) => {
commissionPagination.page = newCurrentPage
getCommissionList()
}
// ==================== 提现记录 ====================
const withdrawalLoading = ref(false)
const withdrawalTableRef = ref()
// 搜索表单初始值
const initialWithdrawalSearchState = {
status: undefined as WithdrawalStatus | undefined,
start_time: '',
end_time: ''
}
// 搜索表单
const withdrawalSearchForm = reactive({ ...initialWithdrawalSearchState })
// 搜索表单配置
const withdrawalSearchItems = computed<SearchFormItem[]>(() => [
{
label: '状态',
prop: 'status',
type: 'select',
options: [
{ label: '待审核', value: 1 },
{ label: '已通过', value: 2 },
{ label: '已拒绝', value: 3 },
{ label: '已到账', value: 4 }
],
config: {
clearable: true,
placeholder: '请选择状态'
}
},
{
label: '时间范围',
prop: 'dateRange',
type: 'date',
config: {
type: 'daterange',
clearable: true,
valueFormat: 'YYYY-MM-DD HH:mm:ss',
startPlaceholder: '开始日期',
endPlaceholder: '结束日期'
},
onChange: ({ val }: { prop: string; val: any }) => {
if (val && val.length === 2) {
withdrawalSearchForm.start_time = val[0]
withdrawalSearchForm.end_time = val[1]
} else {
withdrawalSearchForm.start_time = ''
withdrawalSearchForm.end_time = ''
}
}
}
])
// 分页
const withdrawalPagination = reactive({
page: 1,
pageSize: 20,
total: 0
})
// 列配置
const withdrawalColumnOptions = [
{ label: 'ID', prop: 'id' },
{ label: '提现单号', prop: 'withdrawal_no' },
{ label: '金额', prop: 'amount' },
{ label: '手续费', prop: 'fee' },
{ label: '实际到账', prop: 'actual_amount' },
{ label: '提现方式', prop: 'withdrawal_method' },
{ label: '状态', prop: 'status' },
{ label: '申请时间', prop: 'created_at' }
]
const withdrawalList = ref<WithdrawalRequestItem[]>([])
// 动态列配置
const { columnChecks: withdrawalColumnChecks, columns: withdrawalColumns } = useCheckedColumns(
() => [
{
prop: 'id',
label: 'ID',
width: 80
},
{
prop: 'withdrawal_no',
label: '提现单号',
minWidth: 160
},
{
prop: 'amount',
label: '金额',
width: 120,
formatter: (row: WithdrawalRequestItem) => formatMoney(row.amount)
},
{
prop: 'fee',
label: '手续费',
width: 100,
formatter: (row: WithdrawalRequestItem) => formatMoney(row.fee)
},
{
prop: 'actual_amount',
label: '实际到账',
width: 120,
formatter: (row: WithdrawalRequestItem) => formatMoney(row.actual_amount)
},
{
prop: 'withdrawal_method',
label: '提现方式',
width: 100,
formatter: (row: WithdrawalRequestItem) => {
const config =
WithdrawalMethodMap[row.withdrawal_method as keyof typeof WithdrawalMethodMap]
return config ? config.label : row.withdrawal_method
}
},
{
prop: 'status',
label: '状态',
width: 100,
formatter: (row: WithdrawalRequestItem) => {
const config = WithdrawalStatusMap[row.status as keyof typeof WithdrawalStatusMap]
return h(ElTag, { type: config.type }, () => config.label)
}
},
{
prop: 'created_at',
label: '申请时间',
width: 180,
formatter: (row: WithdrawalRequestItem) => formatDateTime(row.created_at)
}
]
)
// 获取提现记录
const getWithdrawalList = async () => {
withdrawalLoading.value = true
try {
const params = {
pageSize: withdrawalPagination.pageSize,
current: withdrawalPagination.page,
status: withdrawalSearchForm.status,
start_time: withdrawalSearchForm.start_time || undefined,
end_time: withdrawalSearchForm.end_time || undefined
} as WithdrawalRequestQueryParams
const res = await CommissionService.getMyWithdrawalRequests(params)
if (res.code === 0) {
withdrawalList.value = res.data.items || []
withdrawalPagination.total = res.data.total || 0
}
} catch (error) {
console.error(error)
} finally {
withdrawalLoading.value = false
}
}
// 重置搜索
const handleWithdrawalReset = () => {
Object.assign(withdrawalSearchForm, { ...initialWithdrawalSearchState })
withdrawalPagination.page = 1
getWithdrawalList()
}
// 搜索
const handleWithdrawalSearch = () => {
withdrawalPagination.page = 1
getWithdrawalList()
}
// 刷新表格
const handleWithdrawalRefresh = () => {
getWithdrawalList()
}
// 处理表格分页变化
const handleWithdrawalSizeChange = (newPageSize: number) => {
withdrawalPagination.pageSize = newPageSize
getWithdrawalList()
}
const handleWithdrawalCurrentChange = (newCurrentPage: number) => {
withdrawalPagination.page = newCurrentPage
getWithdrawalList()
}
// ==================== 发起提现 ====================
const withdrawalDialogVisible = ref(false)
const submitLoading = ref(false)
const withdrawalFormRef = ref<FormInstance>()
// 提现表单
const withdrawalForm = reactive<SubmitWithdrawalParams>({
amount: 0,
withdrawal_method: WithdrawalMethod.ALIPAY,
account_name: '',
account_number: '',
bank_name: '',
remark: ''
})
// 提现表单验证规则
const withdrawalRules = reactive<FormRules>({
amount: [
{ required: true, message: '请输入提现金额', trigger: 'blur' },
{
validator: (rule, value, callback) => {
if (value < 100) {
callback(new Error('提现金额不能小于100分1元'))
} else if (value > summary.value.available_commission) {
callback(new Error('提现金额不能大于可提现佣金'))
} else {
callback()
}
},
trigger: 'blur'
}
],
withdrawal_method: [{ required: true, message: '请选择提现方式', trigger: 'change' }],
account_name: [
{ required: true, message: '请输入账户名', trigger: 'blur' },
{ min: 2, max: 50, message: '长度在 2 到 50 个字符', trigger: 'blur' }
],
account_number: [
{ required: true, message: '请输入账号', trigger: 'blur' },
{ min: 5, max: 50, message: '长度在 5 到 50 个字符', trigger: 'blur' }
],
bank_name: [
{
validator: (rule, value, callback) => {
if (withdrawalForm.withdrawal_method === WithdrawalMethod.BANK && !value) {
callback(new Error('请选择银行名称'))
} else {
callback()
}
},
trigger: 'change'
}
]
})
// 显示提现对话框
const showWithdrawalDialog = () => {
if (summary.value.available_commission < 100) {
ElMessage.warning('可提现佣金不足100分1元无法发起提现')
return
}
withdrawalDialogVisible.value = true
}
// 关闭对话框
const handleDialogClose = () => {
withdrawalFormRef.value?.resetFields()
withdrawalForm.amount = 0
withdrawalForm.withdrawal_method = WithdrawalMethod.ALIPAY
withdrawalForm.account_name = ''
withdrawalForm.account_number = ''
withdrawalForm.bank_name = ''
withdrawalForm.remark = ''
}
// 提交提现申请
const handleSubmitWithdrawal = async () => {
if (!withdrawalFormRef.value) return
await withdrawalFormRef.value.validate(async (valid) => {
if (valid) {
submitLoading.value = true
try {
const params: SubmitWithdrawalParams = {
amount: withdrawalForm.amount,
withdrawal_method: withdrawalForm.withdrawal_method,
account_name: withdrawalForm.account_name,
account_number: withdrawalForm.account_number,
remark: withdrawalForm.remark
}
// 如果是银行卡提现,添加银行名称
if (withdrawalForm.withdrawal_method === WithdrawalMethod.BANK) {
params.bank_name = withdrawalForm.bank_name
}
await CommissionService.submitWithdrawalRequest(params)
ElMessage.success('提现申请提交成功')
withdrawalDialogVisible.value = false
// 刷新数据
loadSummary()
if (activeTab.value === 'withdrawal') {
getWithdrawalList()
}
} catch (error) {
console.error(error)
} finally {
submitLoading.value = false
}
}
})
}
// ==================== 初始化 ====================
// 加载佣金概览
const loadSummary = async () => {
try {
const res = await CommissionService.getMyCommissionSummary()
if (res.code === 0) {
summary.value = res.data
}
} catch (error) {
console.error(error)
}
}
// 监听标签页切换
watch(activeTab, (newTab) => {
if (newTab === 'commission') {
getCommissionList()
} else if (newTab === 'withdrawal') {
getWithdrawalList()
}
})
onMounted(() => {
loadSummary()
getCommissionList()
})
</script>
<style lang="scss" scoped>
.my-commission-page {
.stat-card {
display: flex;
gap: 16px;
align-items: center;
.stat-icon {
display: flex;
align-items: center;
justify-content: center;
width: 60px;
height: 60px;
font-size: 28px;
color: white;
border-radius: 12px;
}
.stat-content {
flex: 1;
.stat-label {
margin-bottom: 8px;
font-size: 14px;
color: var(--el-text-color-secondary);
}
.stat-value {
font-size: 20px;
font-weight: 600;
color: var(--el-text-color-primary);
}
}
}
}
</style>