Files
integral-shop/backend-adminend/src/views/integral-external/user-integral-detail/index.vue
apple 708bf9af48 feat(sxsy80): 外部用户 UID 筛选与积分明细展示
User list API accepts uid; admin external pages tighten filters and
integral log maps self-bonus rows via wa_selfbonus_log for display.

Made-with: Cursor
2026-04-27 13:30:05 +08:00

372 lines
11 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="divBox">
<!-- 返回按钮 -->
<div class="back-bar">
<el-button size="small" icon="el-icon-arrow-left" @click="goBack">返回</el-button>
</div>
<!-- 用户概览卡片从用户列表进入时展示 uid 时为全部明细模式 -->
<el-card v-if="uid" class="box-card overview-card">
<div class="overview-header">
<span class="user-title">
{{ userInfo.nickname || ('UID: ' + uid) }}
<span class="uid-badge">UID: {{ uid }}</span>
</span>
</div>
<el-row :gutter="20" class="stats-row">
<el-col :xs="12" :sm="8" :md="6">
<div class="stat-item">
<div class="stat-label">积分</div>
<div class="stat-value integral-color">{{ userInfo.integral != null ? userInfo.integral : '-' }}</div>
</div>
</el-col>
<el-col :xs="12" :sm="8" :md="6">
<div class="stat-item">
<div class="stat-label">个人奖金</div>
<div class="stat-value bonus-color">
{{ userInfo.selfBonus != null ? ('¥' + userInfo.selfBonus) : '-' }}
</div>
</div>
</el-col>
</el-row>
</el-card>
<el-card v-else class="box-card overview-card overview-card--all">
<div class="overview-header">
<span class="user-title">全部积分明细</span>
<span class="hint-text">未指定用户时将展示全部记录支持下方条件筛选</span>
</div>
</el-card>
<!-- 积分明细列表 -->
<el-card class="box-card mt10">
<div slot="header" class="clearfix">
<span>积分明细</span>
</div>
<div class="container mb10">
<el-form inline size="small" :model="searchForm" label-width="96px">
<el-row>
<el-col :xs="24" :sm="12" :md="8" :lg="5" :xl="5">
<el-form-item label="用户ID">
<el-input
v-model="searchForm.uidStr"
placeholder="可选,留空查全部"
clearable
class="filter-input"
@keyup.enter.native="handleSearch"
/>
</el-form-item>
</el-col>
<el-col :xs="24" :sm="12" :md="8" :lg="5" :xl="5">
<el-form-item label="用户名称:">
<el-input
v-model="searchForm.nickName"
placeholder="昵称模糊匹配"
clearable
class="filter-input"
@keyup.enter.native="handleSearch"
/>
</el-form-item>
</el-col>
<el-col :xs="24" :sm="12" :md="8" :lg="5" :xl="5">
<el-form-item label="手机号:">
<el-input
v-model="searchForm.phone"
placeholder="手机号模糊匹配"
clearable
class="filter-input"
@keyup.enter.native="handleSearch"
/>
</el-form-item>
</el-col>
<el-col :xs="24" :sm="12" :md="10" :lg="8" :xl="8">
<el-form-item label="时间选择:">
<el-date-picker
v-model="timeVal"
type="daterange"
align="right"
unlink-panels
value-format="yyyy-MM-dd"
format="yyyy-MM-dd"
range-separator=""
start-placeholder="开始日期"
end-placeholder="结束日期"
class="date-range-width"
@change="onchangeTime"
/>
</el-form-item>
</el-col>
<el-col :xs="24" :sm="12" :md="6" :lg="4" :xl="4">
<el-form-item>
<el-button type="primary" icon="el-icon-search" @click="handleSearch">搜索</el-button>
<el-button @click="handleReset">重置</el-button>
</el-form-item>
</el-col>
</el-row>
</el-form>
</div>
<el-table
v-loading="listLoading"
:data="tableData.data"
style="width: 100%"
size="mini"
highlight-current-row
:header-cell-style="{ fontWeight: 'bold' }"
>
<el-table-column prop="id" label="ID" min-width="80" />
<el-table-column prop="uid" label="用户ID" min-width="80" />
<el-table-column prop="nickName" label="用户昵称" min-width="120" show-overflow-tooltip>
<template slot-scope="scope">
<span>{{ scope.row.nickName || '-' }}</span>
</template>
</el-table-column>
<el-table-column label="积分变动" min-width="100">
<template slot-scope="scope">
<el-tag :type="scope.row.type === 1 ? 'success' : 'danger'" size="small">
{{ scope.row.type === 1 ? '+' : '-' }}{{ scope.row.integral }}
</el-tag>
</template>
</el-table-column>
<el-table-column prop="balance" label="剩余积分" min-width="100" />
<el-table-column label="类型" min-width="80">
<template slot-scope="scope">
<el-tag :type="scope.row.type === 1 ? 'success' : 'danger'" size="small">
{{ scope.row.type === 1 ? '增加' : '扣减' }}
</el-tag>
</template>
</el-table-column>
<el-table-column label="关联类型" min-width="100">
<template slot-scope="scope">
<el-tag size="small" effect="plain">{{ linkTypeFilter(scope.row.linkType) }}</el-tag>
</template>
</el-table-column>
<el-table-column label="状态" min-width="100">
<template slot-scope="scope">
<el-tag :type="statusTypeFilter(scope.row.status)" size="small">
{{ statusFilter(scope.row.status) }}
</el-tag>
</template>
</el-table-column>
<el-table-column prop="mark" label="备注" min-width="150" show-overflow-tooltip>
<template slot-scope="scope">
<span>{{ scope.row.mark || '-' }}</span>
</template>
</el-table-column>
<el-table-column prop="createTime" label="创建时间" min-width="150" />
</el-table>
<div class="block mt20">
<el-pagination
:page-sizes="[15, 30, 45, 60]"
:page-size="searchForm.limit"
:current-page="searchForm.page"
layout="total, sizes, prev, pager, next, jumper"
:total="tableData.total"
@size-change="handleSizeChange"
@current-change="handleCurrentChange"
/>
</div>
</el-card>
</div>
</template>
<script>
import { getExternalIntegralLog } from '@/api/integralExternal';
export default {
name: 'IntegralExternalUserDetail',
data() {
return {
uid: null,
userInfo: {
nickname: '',
integral: null,
selfBonus: null,
},
listLoading: false,
tableData: {
data: [],
total: 0,
},
searchForm: {
uidStr: '',
nickName: '',
phone: '',
dateLimit: '',
page: 1,
limit: 15,
},
timeVal: [],
};
},
created() {
const { uid, nickname, integral, selfBonus } = this.$route.query;
this.uid = uid ? Number(uid) : null;
this.userInfo.nickname = nickname || '';
this.userInfo.integral = integral !== '' && integral != null ? Number(integral) : null;
this.userInfo.selfBonus = selfBonus !== '' && selfBonus != null ? Number(selfBonus) : null;
if (this.uid) {
this.searchForm.uidStr = String(this.uid);
}
},
mounted() {
this.getList();
},
methods: {
getList() {
this.listLoading = true;
const uidParsed = this.searchForm.uidStr === '' || this.searchForm.uidStr == null
? null
: parseInt(String(this.searchForm.uidStr).trim(), 10);
const uid = Number.isNaN(uidParsed) ? null : uidParsed;
const params = {
page: this.searchForm.page,
limit: this.searchForm.limit,
uid,
nickName: this.searchForm.nickName ? this.searchForm.nickName.trim() : undefined,
phone: this.searchForm.phone ? this.searchForm.phone.trim() : undefined,
dateLimit: this.searchForm.dateLimit || undefined,
};
Object.keys(params).forEach((k) => {
if (params[k] === undefined || params[k] === null || params[k] === '') {
delete params[k];
}
});
getExternalIntegralLog(params)
.then((res) => {
this.tableData.data = res.list || [];
this.tableData.total = res.total || 0;
})
.catch(() => {})
.finally(() => {
this.listLoading = false;
});
},
handleSearch() {
this.searchForm.page = 1;
this.getList();
},
handleReset() {
this.searchForm.uidStr = '';
this.searchForm.nickName = '';
this.searchForm.phone = '';
this.searchForm.dateLimit = '';
this.searchForm.page = 1;
this.timeVal = [];
this.uid = null;
this.userInfo = { nickname: '', integral: null, selfBonus: null };
this.getList();
},
onchangeTime(e) {
this.timeVal = e;
this.searchForm.dateLimit = e ? e.join(',') : '';
this.handleSearch();
},
handleSizeChange(val) {
this.searchForm.limit = val;
this.getList();
},
handleCurrentChange(val) {
this.searchForm.page = val;
this.getList();
},
goBack() {
this.$router.push('/integral-external/user');
},
linkTypeFilter(type) {
if (type == null || type === '') return '-';
const raw = String(type).trim();
if (!raw) return '-';
const key = raw.toLowerCase();
const typeMap = {
order: '订单',
sign: '签到',
system: '系统',
selfbonus: '个人奖金',
};
return typeMap[key] || `其他(${raw}`;
},
statusFilter(status) {
const statusMap = { 1: '订单创建', 2: '冻结期', 3: '完成', 4: '失效' };
return statusMap[status] || '未知';
},
statusTypeFilter(status) {
const typeMap = { 1: 'info', 2: 'warning', 3: 'success', 4: 'danger' };
return typeMap[status] || 'info';
},
},
};
</script>
<style scoped lang="scss">
.back-bar {
margin-bottom: 12px;
}
.mt10 {
margin-top: 10px;
}
.mt20 {
margin-top: 20px;
}
.mb10 {
margin-bottom: 10px;
}
.overview-card {
.overview-header {
margin-bottom: 16px;
.user-title {
font-size: 16px;
font-weight: bold;
color: #303133;
.uid-badge {
margin-left: 10px;
font-size: 12px;
font-weight: normal;
color: #909399;
background: #f4f4f5;
border-radius: 4px;
padding: 2px 8px;
}
}
}
.stats-row {
.stat-item {
text-align: center;
padding: 12px 0;
.stat-label {
font-size: 13px;
color: #909399;
margin-bottom: 6px;
}
.stat-value {
font-size: 22px;
font-weight: bold;
}
}
}
}
.integral-color {
color: #e6a23c;
}
.bonus-color {
color: #67c23a;
}
.block {
text-align: right;
}
.filter-input {
width: 150px;
}
.date-range-width {
width: 350px;
}
.overview-card--all .hint-text {
display: block;
margin-top: 8px;
font-size: 13px;
color: #909399;
font-weight: normal;
}
</style>