fetch(add): 账户管理

This commit is contained in:
sexygoat
2026-01-23 17:18:24 +08:00
parent 339abca4c0
commit b53fea43c6
93 changed files with 7094 additions and 3153 deletions

View File

@@ -0,0 +1,874 @@
<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">发起提现</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 { formatDateTime, formatMoney } from '@/utils/business/format'
defineOptions({ name: 'MyCommission' })
// 标签页
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>