Преглед на файлове

feat(admin): 添加微信支付转账记录管理功能

- 新增 TransferList.vue 页面实现转账记录列表展示
- 实现转账记录的搜索、筛选和分页功能
- 添加转账状态统计卡片展示各类状态数据
- 实现同意和驳回转账申请的操作功能
- 添加刷新转账状态的功能按钮
- 集成资金明细弹窗查看用户余额记录
- 在路由配置中注册微信支付管理模块
- 添加后台菜单项支持转账记录页面访问
runphp преди 3 седмици
родител
ревизия
a289d870e7
променени са 2 файла, в които са добавени 925 реда и са изтрити 0 реда
  1. 39 0
      frontend/admin/index.js
  2. 886 0
      frontend/admin/views/TransferList.vue

+ 39 - 0
frontend/admin/index.js

@@ -0,0 +1,39 @@
+export default {
+    routes: {
+        path: '/wechatpay',
+        name: 'WechatpayModule',
+        component: () => import('@/layout/index.vue'),
+        meta: {
+            title: '微信支付管理',
+            icon: 'ChatDotRound',
+            permission: 'wechatpay'
+        },
+        children: [
+            {
+                path: 'transfer',
+                name: 'WechatpayTransfer',
+                component: () => import('./views/TransferList.vue'),
+                meta: {
+                    title: '转账记录',
+                    icon: 'List'
+                }
+            }
+        ]
+    },
+
+    menus: [
+        {
+            path: '/wechatpay',
+            title: '微信支付管理',
+            icon: 'ChatDotRound',
+            permission: 'wechatpay',
+            children: [
+                {
+                    path: '/wechatpay/transfer',
+                    title: '转账记录',
+                    icon: 'List'
+                }
+            ]
+        }
+    ]
+}

+ 886 - 0
frontend/admin/views/TransferList.vue

@@ -0,0 +1,886 @@
+<script setup lang="ts">
+import { ref, onMounted, nextTick } from 'vue'
+import { ElMessage, ElMessageBox } from 'element-plus'
+import request from '@/utils/request'
+import { userApi } from '@/api/user'
+import { Refresh, ArrowRight } from '@element-plus/icons-vue'
+import { imgUrl } from '@/utils/request'
+
+// 定义微信转账记录类型
+interface TransferRecord {
+  id: number
+  user_id: number
+  mch_id: string
+  out_bill_no: string
+  transfer_bill_no: string
+  appid: string
+  state: 'APPLYING' | 'ACCEPTED' | 'PROCESSING' | 'WAIT_USER_CONFIRM' | 'TRANSFERING' | 'SUCCESS' | 'FAIL' | 'CANCELING' | 'CANCELLED'
+  state_text: string
+  transfer_amount: number
+  transfer_remark: string
+  fail_reason: string | null
+  openid: string | null
+  user_name: string | null
+  create_time: string
+  update_time: string
+  fee: number
+  user?: {
+    nickname?: string
+    username?: string
+    mobile?: string
+    avatar?: string
+  }
+}
+
+// 定义余额记录类型
+interface BalanceRecord {
+  id: number
+  user_id: number
+  type: number
+  type_text: string
+  amount: number
+  balance: number
+  description: string
+  create_time: string
+}
+
+// 定义统计数据类型
+interface StatisticsData {
+  state: string
+  count: number
+  amount: number
+}
+
+// 响应式数据
+const transferList = ref<TransferRecord[]>([])
+const statisticsData = ref<StatisticsData[]>([])
+const loading = ref(false)
+const pagination = ref({
+  page: 1,
+  limit: 10,
+  total: 0
+})
+const tableRef = ref()
+const showScrollHint = ref(false)
+const balanceRecordDialogVisible = ref(false)
+const balanceRecordLoading = ref(false)
+const balanceRecords = ref<BalanceRecord[]>([])
+const balanceRecordPagination = ref({
+  page: 1,
+  limit: 10,
+  total: 0
+})
+const currentUserId = ref<number|null>(null)
+
+// 搜索表单
+const searchForm = ref({
+  state: '',
+  out_bill_no: ''
+})
+
+// 组件挂载时获取数据
+onMounted(() => {
+  fetchTransfers()
+  nextTick(() => {
+    checkTableScroll()
+  })
+})
+
+// 检查表格是否需要水平滚动
+const checkTableScroll = () => {
+  nextTick(() => {
+    const tableElement = tableRef.value?.$el
+    if (tableElement) {
+      const tableBodyWrapper = tableElement.querySelector('.el-table__body-wrapper')
+      if (tableBodyWrapper) {
+        showScrollHint.value = tableBodyWrapper.scrollWidth > tableBodyWrapper.clientWidth
+      }
+    }
+  })
+}
+
+// 处理表格滚动事件
+const handleTableScroll = (event: Event) => {
+  const target = event.target as HTMLElement
+  // 可以在这里添加滚动相关的处理逻辑
+}
+
+// 获取转账记录列表
+const fetchTransfers = async () => {
+  loading.value = true
+  try {
+    const response = await request({
+      url: '/wechatpay/transfer_bill',
+      method: 'get',
+      params: {
+        page: pagination.value.page,
+        limit: pagination.value.limit,
+        state: searchForm.value.state || undefined,
+        out_bill_no: searchForm.value.out_bill_no || undefined
+      }
+    })
+    
+    if (response.code === 200) {
+      transferList.value = response.page.data || []
+      pagination.value.total = response.page.total || 0
+      // 更新统计数据
+      statisticsData.value = response.data || []
+      
+      // 检查是否需要显示滚动提示
+      nextTick(() => {
+        checkTableScroll()
+      })
+    } else {
+      ElMessage.error(response.msg || '获取微信转账记录失败')
+    }
+  } catch (error) {
+    ElMessage.error('获取微信转账记录失败')
+    console.error('获取微信转账记录失败:', error)
+  } finally {
+    loading.value = false
+  }
+}
+
+// 重置搜索
+const resetSearch = () => {
+  searchForm.value.state = ''
+  searchForm.value.out_bill_no = ''
+  pagination.value.page = 1
+  fetchTransfers()
+}
+
+// 搜索
+const handleSearch = () => {
+  pagination.value.page = 1
+  fetchTransfers()
+}
+
+// 格式化金额(从分转换为元)
+const formatAmount = (amount: number) => {
+  return (amount / 100).toFixed(2)
+}
+
+// 格式化余额金额(直接保留两位小数)
+const formatBalanceAmount = (amount: number) => {
+  return parseFloat(amount.toString()).toFixed(2)
+}
+
+// 状态标签类型
+const getStateType = (state: string) => {
+  switch (state) {
+    case 'SUCCESS':
+      return 'success'
+    case 'FAIL':
+      return 'danger'
+    case 'PROCESSING':
+    case 'TRANSFERING':
+      return 'warning'
+    case 'APPLYING':
+    case 'ACCEPTED':
+    case 'WAIT_USER_CONFIRM':
+      return 'info'
+    default:
+      return 'info'
+  }
+}
+
+// 处理分页变化
+const handlePageChange = (page: number) => {
+  pagination.value.page = page
+  fetchTransfers()
+}
+
+// 处理页面大小变化
+const handleSizeChange = (size: number) => {
+  pagination.value.limit = size
+  pagination.value.page = 1
+  fetchTransfers()
+}
+
+// 同意转账申请
+const approveTransfer = async (id: number) => {
+  try {
+    await ElMessageBox.confirm('确定要同意此转账申请吗?', '确认操作', {
+      confirmButtonText: '确定',
+      cancelButtonText: '取消',
+      type: 'warning'
+    })
+    
+    const response = await request({
+      url: `/wechatpay/transfer_bill/${id}/approve`,
+      method: 'post'
+    })
+    
+    if (response.code === 200) {
+      ElMessage.success('操作成功')
+      fetchTransfers()
+    } else {
+      ElMessage.error(response.msg || '操作失败')
+    }
+  } catch (error) {
+    if (error !== 'cancel') {
+      ElMessage.error('操作失败')
+      console.error('同意转账申请失败:', error)
+    }
+  }
+}
+
+// 驳回转账申请
+const rejectTransfer = async (id: number) => {
+  try {
+    await ElMessageBox.prompt('请输入驳回原因', '驳回操作', {
+      confirmButtonText: '确定',
+      cancelButtonText: '取消',
+      inputPlaceholder: '请输入驳回原因',
+      inputPattern: /\S/,
+      inputErrorMessage: '驳回原因不能为空'
+    }).then(async ({ value }) => {
+      const response = await request({
+        url: `/wechatpay/transfer_bill/${id}/reject`,
+        method: 'post',
+        data: {
+          fail_reason: value
+        }
+      })
+      
+      if (response.code === 200) {
+        ElMessage.success('操作成功')
+        fetchTransfers()
+      } else {
+        ElMessage.error(response.msg || '操作失败')
+      }
+    }).catch(() => {
+      // 用户取消操作
+    })
+  } catch (error) {
+    ElMessage.error('操作失败')
+    console.error('驳回转账申请失败:', error)
+  }
+}
+
+// 刷新转账状态
+const refreshTransferStatus = async (id: number) => {
+  try {
+    const response = await request({
+      url: `/wechatpay/transfer_bill/${id}/refresh`,
+      method: 'get'
+    })
+    
+    if (response.code === 200) {
+      ElMessage.success('刷新成功')
+      // 重新获取列表数据以更新状态
+      fetchTransfers()
+    } else {
+      ElMessage.error(response.msg || '刷新失败')
+    }
+  } catch (error) {
+    ElMessage.error('刷新失败')
+    console.error('刷新转账状态失败:', error)
+  }
+}
+
+// 显示余额记录对话框
+const showBalanceRecordDialog = (userId: number) => {
+  currentUserId.value = userId
+  balanceRecordDialogVisible.value = true
+  getBalanceRecords()
+}
+
+// 关闭余额记录对话框
+const closeBalanceRecordDialog = () => {
+  balanceRecordDialogVisible.value = false
+  balanceRecords.value = []
+  balanceRecordPagination.value.page = 1
+  balanceRecordPagination.value.total = 0
+  currentUserId.value = null
+}
+
+// 获取余额记录
+const getBalanceRecords = async () => {
+  try {
+    balanceRecordLoading.value = true
+    
+    const params = {
+      user_id: currentUserId.value,
+      page: balanceRecordPagination.value.page,
+      limit: balanceRecordPagination.value.limit
+    }
+    
+    const response = await userApi.getUserBalanceList(params)
+    if (response.code === 200) {
+      balanceRecords.value = response.page?.data || []
+      balanceRecordPagination.value.total = response.page?.total || 0
+    } else {
+      ElMessage.error(response.msg || '获取余额记录失败')
+    }
+  } catch (error) {
+    console.error('获取余额记录失败:', error)
+    ElMessage.error('获取余额记录失败')
+  } finally {
+    balanceRecordLoading.value = false
+  }
+}
+
+// 余额记录分页处理
+const handleBalanceRecordSizeChange = (size: number) => {
+  balanceRecordPagination.value.limit = size
+  balanceRecordPagination.value.page = 1
+  getBalanceRecords()
+}
+
+const handleBalanceRecordCurrentChange = (page: number) => {
+  balanceRecordPagination.value.page = page
+  getBalanceRecords()
+}
+
+</script>
+
+<template>
+  <div class="wechatpay-transfer">
+    <el-card>
+      <template #header>
+        <div class="card-header">
+          <span>微信支付转账记录</span>
+          <div class="header-actions">
+            <el-button type="primary" :icon="Refresh" @click="fetchTransfers" circle />
+          </div>
+        </div>
+      </template>
+      
+      <!-- 搜索表单 -->
+      <div class="search-bar">
+        <el-form :model="searchForm" inline>
+          <el-form-item label="商户单号">
+            <el-input 
+              v-model="searchForm.out_bill_no" 
+              placeholder="请输入商户单号" 
+              clearable 
+              style="width: 200px;"
+            />
+          </el-form-item>
+          <el-form-item label="状态">
+            <el-select 
+              v-model="searchForm.state" 
+              placeholder="请选择状态" 
+              clearable 
+              style="width: 150px;"
+            >
+              <el-option label="申请中" value="APPLYING" />
+              <el-option label="转账已受理" value="ACCEPTED" />
+              <el-option label="转账锁定资金中" value="PROCESSING" />
+              <el-option label="待收款用户确认" value="WAIT_USER_CONFIRM" />
+              <el-option label="转账中" value="TRANSFERING" />
+              <el-option label="转账成功" value="SUCCESS" />
+              <el-option label="转账失败" value="FAIL" />
+              <el-option label="转账撤销中" value="CANCELING" />
+              <el-option label="转账撤销完成" value="CANCELLED" />
+            </el-select>
+          </el-form-item>
+          <el-form-item>
+            <el-button type="primary" @click="handleSearch" :loading="loading">搜索</el-button>
+            <el-button @click="resetSearch">重置</el-button>
+          </el-form-item>
+        </el-form>
+      </div>
+      
+      <!-- 统计数据展示 -->
+      <div class="stats-cards">
+        <el-row :gutter="20" style="margin-bottom: 20px;">
+          <el-col :span="6">
+            <el-card class="stats-card">
+              <div class="stats-item">
+                <div class="stats-label">转账成功</div>
+                <div class="stats-value success">
+                  <div class="count">{{ statisticsData.find(item => item.state === 'SUCCESS')?.count || 0 }} 笔</div>
+                  <div class="amount">¥{{ formatAmount(statisticsData.find(item => item.state === 'SUCCESS')?.amount || 0) }}</div>
+                </div>
+              </div>
+            </el-card>
+          </el-col>
+          <el-col :span="6">
+            <el-card class="stats-card">
+              <div class="stats-item">
+                <div class="stats-label">转账失败</div>
+                <div class="stats-value failed">
+                  <div class="count">{{ statisticsData.find(item => item.state === 'FAIL')?.count || 0 }} 笔</div>
+                  <div class="amount">¥{{ formatAmount(statisticsData.find(item => item.state === 'FAIL')?.amount || 0) }}</div>
+                </div>
+              </div>
+            </el-card>
+          </el-col>
+          <el-col :span="6">
+            <el-card class="stats-card">
+              <div class="stats-item">
+                <div class="stats-label">处理中</div>
+                <div class="stats-value processing">
+                  <div class="count">{{ (statisticsData.find(item => item.state === 'PROCESSING')?.count || 0) + (statisticsData.find(item => item.state === 'TRANSFERING')?.count || 0) }} 笔</div>
+                  <div class="amount">¥{{ formatAmount((statisticsData.find(item => item.state === 'PROCESSING')?.amount || 0) + (statisticsData.find(item => item.state === 'TRANSFERING')?.amount || 0)) }}</div>
+                </div>
+              </div>
+            </el-card>
+          </el-col>
+          <el-col :span="6">
+            <el-card class="stats-card">
+              <div class="stats-item">
+                <div class="stats-label">待处理</div>
+                <div class="stats-value pending">
+                  <div class="count">{{ (statisticsData.find(item => item.state === 'APPLYING')?.count || 0) + (statisticsData.find(item => item.state === 'ACCEPTED')?.count || 0) + (statisticsData.find(item => item.state === 'WAIT_USER_CONFIRM')?.count || 0) }} 笔</div>
+                  <div class="amount">¥{{ formatAmount((statisticsData.find(item => item.state === 'APPLYING')?.amount || 0) + (statisticsData.find(item => item.state === 'ACCEPTED')?.amount || 0) + (statisticsData.find(item => item.state === 'WAIT_USER_CONFIRM')?.amount || 0)) }}</div>
+                </div>
+              </div>
+            </el-card>
+          </el-col>
+        </el-row>
+      </div>
+      
+      <div class="table-wrapper">
+        <div class="table-container">
+          <div class="scroll-hint" v-if="showScrollHint">
+            <el-icon><ArrowRight /></el-icon>
+            <span>左右滑动查看更多</span>
+          </div>
+          <el-table
+            ref="tableRef"
+            v-loading="loading"
+            :data="transferList"
+            style="width: 100%"
+            border
+            @scroll="handleTableScroll"
+          >
+            <el-table-column prop="id" label="ID" width="80" />
+            <el-table-column label="用户信息" width="240">
+              <template #default="scope">
+                <div class="user-info-cell">
+                  <div class="avatar-container">
+                    <el-avatar 
+                      v-if="scope.row.user && scope.row.user.avatar" 
+                      :src="scope.row.user.avatar.startsWith('http') ? scope.row.user.avatar : `${imgUrl}${scope.row.user.avatar}`" 
+                      :size="30" 
+                    />
+                    <el-avatar 
+                      v-else 
+                      :size="30"
+                    >
+                      {{ scope.row.user?.nickname?.charAt(0) || scope.row.user?.username?.charAt(0) || 'U' }}
+                    </el-avatar>
+                  </div>
+                  <div class="user-details">
+                    <div class="nickname">{{ scope.row.user?.nickname || '-' }}</div>
+                    <div class="username"><span class="label">用户名:</span> {{ scope.row.user?.username || '-' }}</div>
+                    <div class="mobile"><span class="label">手机:</span> {{ scope.row.user?.mobile || '-' }}</div>
+                    <div class="user-name"><span class="label">姓名:</span> {{ scope.row.user_name || '-' }}</div>
+                    <div class="openid"><span class="label">OpenID:</span> {{ scope.row.openid || '-' }}</div>
+                  </div>
+                </div>
+              </template>
+            </el-table-column>
+            <el-table-column label="转账金额" width="240">
+              <template #default="scope">
+                <div class="amount-group">
+                  <div class="total-amount">
+                    <span class="label">总金额:</span>
+                    <span class="amount-text">¥{{ formatAmount(scope.row.transfer_amount + scope.row.fee) }}</span>
+                  </div>
+                  <div class="amount-details">
+                    <div class="detail-item">
+                      <span class="label">实际到账:</span>
+                      <span class="value">¥{{ formatAmount(scope.row.transfer_amount) }}</span>
+                    </div>
+                    <div class="detail-item">
+                      <span class="label">手续费:</span>
+                      <span class="value">¥{{ formatAmount(scope.row.fee) }}</span>
+                    </div>
+                    <div class="detail-item">
+                      <el-button 
+                        type="primary" 
+                        size="small" 
+                        link 
+                        style="padding: 0"
+                        @click="showBalanceRecordDialog(scope.row.user_id)"
+                      >
+                        资金明细
+                      </el-button>
+                    </div>
+                  </div>
+                </div>
+              </template>
+            </el-table-column>
+            <el-table-column prop="out_bill_no" label="商户单号" min-width="180" />
+            <el-table-column prop="transfer_bill_no" label="微信转账订单号" min-width="180" />
+            <el-table-column prop="state_text" label="状态" width="150">
+              <template #default="scope">
+                <div style="display: flex; align-items: center;">
+                  <el-tag :type="getStateType(scope.row.state)">
+                    {{ scope.row.state_text }}
+                  </el-tag>
+                  <!-- 当状态为处理中时显示刷新图标 -->
+                  <el-button
+                    v-if="['APPLYING', 'ACCEPTED', 'PROCESSING', 'TRANSFERING', 'WAIT_USER_CONFIRM'].includes(scope.row.state)"
+                    type="text"
+                    :icon="Refresh"
+                    size="small"
+                    @click.stop="refreshTransferStatus(scope.row.id)"
+                    title="刷新状态"
+                    style="margin-left: 4px; color: #606266;"
+                  />
+                </div>
+              </template>
+            </el-table-column>
+            <el-table-column prop="transfer_remark" label="转账备注" min-width="120" />
+            <el-table-column prop="fail_reason" label="失败原因" min-width="120" />
+            <el-table-column prop="create_time" label="创建时间" width="180" />
+            <el-table-column prop="update_time" label="更新时间" width="180" />
+            <el-table-column label="操作" width="150" fixed="right">
+              <template #default="scope">
+                <div v-if="scope.row.state === 'APPLYING'" style="display: flex; gap: 8px;">
+                  <el-button size="small" type="primary" @click.stop="approveTransfer(scope.row.id)">同意</el-button>
+                  <el-button size="small" type="danger" @click.stop="rejectTransfer(scope.row.id)">驳回</el-button>
+                </div>
+                <div v-else>-</div>
+              </template>
+            </el-table-column>
+          </el-table>
+        </div>
+      </div>
+      
+      <!-- 余额记录对话框 -->
+      <el-dialog
+        v-model="balanceRecordDialogVisible"
+        title="资金明细"
+        width="800px"
+        :before-close="closeBalanceRecordDialog"
+      >
+        <el-table
+          v-loading="balanceRecordLoading"
+          :data="balanceRecords"
+          style="width: 100%"
+          border
+          stripe
+        >
+          <el-table-column prop="id" label="ID" width="80" />
+          <el-table-column prop="type_text" label="类型" width="120">
+            <template #default="scope">
+              <el-tag v-if="scope.row.type === 1" type="success">收入</el-tag>
+              <el-tag v-else type="danger">支出</el-tag>
+            </template>
+          </el-table-column>
+          <el-table-column prop="amount" label="金额" width="120">
+            <template #default="scope">
+              <span :class="scope.row.type === 1 ? 'income-amount' : 'expense-amount'">
+                {{ scope.row.type === 1 ? '+' : '-' }}¥{{ formatBalanceAmount(Math.abs(scope.row.amount)) }}
+              </span>
+            </template>
+          </el-table-column>
+          <el-table-column prop="balance" label="余额" width="120">
+            <template #default="scope">
+              ¥{{ formatBalanceAmount(scope.row.balance) }}
+            </template>
+          </el-table-column>
+          <el-table-column prop="description" label="备注" min-width="150" />
+          <el-table-column prop="create_time" label="创建时间" width="180" />
+        </el-table>
+        
+        <div class="pagination-container" style="margin-top: 20px; display: flex; justify-content: flex-end;">
+          <el-pagination
+            v-model:current-page="balanceRecordPagination.page"
+            v-model:page-size="balanceRecordPagination.limit"
+            :page-sizes="[10, 20, 50, 100]"
+            :total="balanceRecordPagination.total"
+            layout="total, sizes, prev, pager, next, jumper"
+            @current-change="handleBalanceRecordCurrentChange"
+            @size-change="handleBalanceRecordSizeChange"
+          />
+        </div>
+        
+        <template #footer>
+          <div class="dialog-footer">
+            <el-button @click="closeBalanceRecordDialog">关闭</el-button>
+          </div>
+        </template>
+      </el-dialog>
+      
+      <div class="pagination-container">
+        <el-pagination
+          v-model:current-page="pagination.page"
+          v-model:page-size="pagination.limit"
+          :page-sizes="[10, 20, 50, 100]"
+          :total="pagination.total"
+          layout="total, sizes, prev, pager, next, jumper"
+          @current-change="handlePageChange"
+          @size-change="handleSizeChange"
+        />
+      </div>
+    </el-card>
+  </div>
+</template>
+
+<style scoped lang="scss">
+.wechatpay-transfer {
+  padding: 20px;
+  
+  .card-header {
+    display: flex;
+    justify-content: space-between;
+    align-items: center;
+    font-weight: 600;
+    font-size: 16px;
+  }
+  
+  .header-actions {
+    display: flex;
+    gap: 10px;
+  }
+  
+  .search-bar {
+    background: #f5f7fa;
+    padding: 15px;
+    border-radius: 4px;
+    margin-bottom: 20px;
+  }
+  
+  .table-container {
+    position: relative;
+    
+    .scroll-hint {
+      position: absolute;
+      top: 50%;
+      right: 10px;
+      transform: translateY(-50%);
+      background: rgba(64, 158, 255, 0.8);
+      color: white;
+      padding: 5px 10px;
+      border-radius: 4px;
+      z-index: 10;
+      display: flex;
+      align-items: center;
+      gap: 5px;
+      font-size: 12px;
+      pointer-events: none;
+      animation: fadeInOut 2s infinite;
+    }
+    
+    @keyframes fadeInOut {
+      0%, 100% { opacity: 0.7; }
+      50% { opacity: 1; }
+    }
+  }
+  
+  .amount-group {
+    .total-amount {
+      display: flex;
+      justify-content: space-between;
+      font-weight: 600;
+      margin-bottom: 4px;
+      
+      .label {
+        color: #606266;
+      }
+      
+      .amount-text {
+        font-size: 16px;
+        color: #e74c3c;
+      }
+    }
+    
+    .amount-details {
+      background-color: #f5f7fa;
+      border-radius: 4px;
+      padding: 6px 8px;
+      
+      .detail-item {
+        display: flex;
+        justify-content: space-between;
+        font-size: 12px;
+        margin-bottom: 2px;
+        
+        &:last-child {
+          margin-bottom: 0;
+        }
+        
+        .label {
+          color: #909399;
+        }
+        
+        .value {
+          color: #606266;
+        }
+      }
+    }
+  }
+  
+  .stats-cards {
+    .stats-card {
+      text-align: center;
+      
+      .stats-item {
+        .stats-label {
+          color: #909399;
+          font-size: 14px;
+          margin-bottom: 8px;
+        }
+        
+        .stats-value {
+          font-weight: bold;
+          
+          .count {
+            font-size: 20px;
+            margin-bottom: 5px;
+          }
+          
+          .amount {
+            font-size: 16px;
+          }
+          
+          &.success {
+            .count, .amount {
+              color: #67c23a;
+            }
+          }
+          
+          &.failed {
+            .count, .amount {
+              color: #f56c6c;
+            }
+          }
+          
+          &.processing {
+            .count, .amount {
+              color: #e6a23c;
+            }
+          }
+          
+          &.pending {
+            .count, .amount {
+              color: #409eff;
+            }
+          }
+        }
+      }
+    }
+  }
+  
+  .amount-text {
+    font-size: 16px;
+    font-weight: 600;
+    color: #e74c3c;
+  }
+  
+  .income-amount {
+    color: #67c23a;
+    font-weight: 600;
+  }
+  
+  .expense-amount {
+    color: #f56c6c;
+    font-weight: 600;
+  }
+  
+  :deep(.el-form--inline .el-form-item) {
+    margin-right: 15px;
+    margin-bottom: 10px;
+  }
+  
+  :deep(.el-table) {
+    .el-table__cell {
+      padding: 8px 0;
+    }
+    
+    // 添加更好的滚动条样式
+    &::-webkit-scrollbar {
+      height: 8px;
+      width: 8px;
+    }
+    
+    &::-webkit-scrollbar-thumb {
+      background-color: #c1c1c1;
+      border-radius: 4px;
+      
+      &:hover {
+        background-color: #a8a8a8;
+      }
+    }
+    
+    &::-webkit-scrollbar-track {
+      background-color: #f1f1f1;
+      border-radius: 4px;
+    }
+  }
+  
+  // 添加阴影效果提示还有更多内容可滚动
+  .table-wrapper {
+    position: relative;
+    
+    &::before,
+    &::after {
+      content: '';
+      position: absolute;
+      top: 0;
+      bottom: 0;
+      width: 30px;
+      pointer-events: none;
+      transition: box-shadow 0.3s;
+      z-index: 1;
+    }
+    
+    &::before {
+      left: 0;
+      box-shadow: inset 10px 0 8px -8px rgba(0, 0, 0, 0.15);
+    }
+    
+    &::after {
+      right: 0;
+      box-shadow: inset -10px 0 8px -8px rgba(0, 0, 0, 0.15);
+    }
+  }
+  
+  .pagination-container {
+    display: flex;
+    justify-content: flex-end;
+    margin-top: 20px;
+  }
+  
+  .user-info-cell {
+    display: flex;
+    align-items: flex-start;
+    
+    .avatar-container {
+      margin-right: 8px;
+      margin-top: 2px;
+    }
+    
+    .user-details {
+      flex: 1;
+      overflow: hidden;
+      
+      .nickname {
+        font-weight: 500;
+        color: #303133;
+        overflow: hidden;
+        text-overflow: ellipsis;
+        white-space: nowrap;
+        margin-bottom: 4px;
+      }
+      
+      .username, .mobile, .openid, .user-name {
+        font-size: 12px;
+        color: #909399;
+        overflow: hidden;
+        text-overflow: ellipsis;
+        white-space: nowrap;
+        margin-bottom: 2px;
+        
+        .label {
+          color: #606266;
+          font-weight: 500;
+          margin-right: 4px;
+        }
+      }
+      
+      .openid, .user-name {
+        font-size: 11px;
+        color: #c0c4cc;
+      }
+    }
+  }
+}
+</style>