261 lines
9.2 KiB
Vue
261 lines
9.2 KiB
Vue
|
|
<template>
|
|||
|
|
<section class="space-y-6">
|
|||
|
|
<ListSection :page="page" :total-pages="totalPages" @prev="page--" @next="page++">
|
|||
|
|
<template #title>奖励发放</template>
|
|||
|
|
<template #subtitle>查看奖励发放状态与明细。</template>
|
|||
|
|
<template #filters>
|
|||
|
|
<input class="mos-input !py-1 !px-2 !text-xs w-48" v-model="query" placeholder="搜索用户" />
|
|||
|
|
<input class="mos-input !py-1 !px-2 !text-xs" type="date" v-model="startDate" />
|
|||
|
|
<input class="mos-input !py-1 !px-2 !text-xs" type="date" v-model="endDate" />
|
|||
|
|
</template>
|
|||
|
|
<template #actions>
|
|||
|
|
<button class="mos-btn mos-btn-secondary !py-1 !px-2 !text-xs" @click="selectAll">
|
|||
|
|
{{ allSelected ? '取消全选' : '全选' }}
|
|||
|
|
</button>
|
|||
|
|
<button class="mos-btn mos-btn-secondary !py-1 !px-2 !text-xs" @click="batchIssue">批量发放</button>
|
|||
|
|
<button class="mos-btn mos-btn-secondary !py-1 !px-2 !text-xs" @click="batchRollback">批量回滚</button>
|
|||
|
|
<input class="mos-input !py-1 !px-2 !text-xs w-44" v-model="batchReason" placeholder="批量回滚原因" />
|
|||
|
|
</template>
|
|||
|
|
<template #default>
|
|||
|
|
<div class="space-y-3">
|
|||
|
|
<div v-for="reward in pagedRewards" :key="reward.id" class="space-y-2 rounded-xl border border-mosquito-line px-4 py-3">
|
|||
|
|
<div class="flex items-center justify-between">
|
|||
|
|
<div class="flex items-center gap-3">
|
|||
|
|
<input
|
|||
|
|
type="checkbox"
|
|||
|
|
class="h-4 w-4"
|
|||
|
|
:checked="selectedIds.includes(reward.id)"
|
|||
|
|
@click.stop
|
|||
|
|
@change.stop="toggleSelect(reward.id)"
|
|||
|
|
/>
|
|||
|
|
<div>
|
|||
|
|
<div class="text-sm font-semibold text-mosquito-ink">{{ reward.userName }}</div>
|
|||
|
|
<div class="mos-muted text-xs">批次:{{ reward.batchId }} · {{ reward.batchStatus }}</div>
|
|||
|
|
<div class="mos-muted text-xs">发放时间:{{ formatDate(reward.issuedAt) }}</div>
|
|||
|
|
<div v-if="reward.note" class="mos-muted text-xs">备注:{{ reward.note }}</div>
|
|||
|
|
</div>
|
|||
|
|
</div>
|
|||
|
|
<div class="flex items-center gap-3 text-xs text-mosquito-ink/70">
|
|||
|
|
<span>{{ reward.points }} 积分</span>
|
|||
|
|
<span class="rounded-full bg-mosquito-accent/10 px-2 py-1 text-[10px] font-semibold text-mosquito-brand">{{ reward.status }}</span>
|
|||
|
|
<button
|
|||
|
|
class="mos-btn mos-btn-secondary !py-1 !px-2 !text-xs"
|
|||
|
|
@click="handleActionClick(reward)"
|
|||
|
|
>
|
|||
|
|
{{ actionLabel(reward) }}
|
|||
|
|
</button>
|
|||
|
|
</div>
|
|||
|
|
</div>
|
|||
|
|
<div
|
|||
|
|
v-if="actioningId === reward.id"
|
|||
|
|
class="flex flex-wrap items-center gap-2 rounded-xl border border-dashed border-mosquito-line px-4 py-3 text-xs"
|
|||
|
|
>
|
|||
|
|
<input class="mos-input !py-1 !px-2 !text-xs flex-1" v-model="actionReason" placeholder="请输入原因" />
|
|||
|
|
<button class="mos-btn mos-btn-secondary !py-1 !px-2 !text-xs" @click="cancelAction">取消</button>
|
|||
|
|
<button class="mos-btn mos-btn-accent !py-1 !px-2 !text-xs" @click="confirmAction(reward)">确认</button>
|
|||
|
|
</div>
|
|||
|
|
</div>
|
|||
|
|
</div>
|
|||
|
|
</template>
|
|||
|
|
<template #footer>
|
|||
|
|
<ExportFieldPanel
|
|||
|
|
title="导出字段"
|
|||
|
|
:fields="exportFields"
|
|||
|
|
:selected="exportSelected"
|
|||
|
|
@update:selected="setExportSelected"
|
|||
|
|
@export="exportRewards"
|
|||
|
|
/>
|
|||
|
|
</template>
|
|||
|
|
</ListSection>
|
|||
|
|
</section>
|
|||
|
|
</template>
|
|||
|
|
|
|||
|
|
<script setup lang="ts">
|
|||
|
|
import { computed, onMounted, ref, watch } from 'vue'
|
|||
|
|
import { useDataService } from '../services'
|
|||
|
|
import { downloadCsv } from '../utils/export'
|
|||
|
|
import { useAuditStore } from '../stores/audit'
|
|||
|
|
import ListSection from '../components/ListSection.vue'
|
|||
|
|
import ExportFieldPanel, { type ExportField } from '../components/ExportFieldPanel.vue'
|
|||
|
|
import { useExportFields } from '../composables/useExportFields'
|
|||
|
|
import { normalizeRewardReason } from '../utils/reward'
|
|||
|
|
|
|||
|
|
type RewardItem = {
|
|||
|
|
id: string
|
|||
|
|
userName: string
|
|||
|
|
points: number
|
|||
|
|
status: string
|
|||
|
|
issuedAt: string
|
|||
|
|
batchId: string
|
|||
|
|
batchStatus: string
|
|||
|
|
note?: string
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
const rewards = ref<RewardItem[]>([])
|
|||
|
|
const service = useDataService()
|
|||
|
|
const auditStore = useAuditStore()
|
|||
|
|
const query = ref('')
|
|||
|
|
const selectedIds = ref<string[]>([])
|
|||
|
|
const startDate = ref('')
|
|||
|
|
const endDate = ref('')
|
|||
|
|
const batchReason = ref('')
|
|||
|
|
const actioningId = ref<string | null>(null)
|
|||
|
|
const actionType = ref<'rollback' | 'retry' | null>(null)
|
|||
|
|
const actionReason = ref('')
|
|||
|
|
const page = ref(0)
|
|||
|
|
const pageSize = 6
|
|||
|
|
|
|||
|
|
const formatDate = (value: string) => new Date(value).toLocaleDateString('zh-CN')
|
|||
|
|
|
|||
|
|
const exportFields: ExportField[] = [
|
|||
|
|
{ key: 'userName', label: '用户', required: true },
|
|||
|
|
{ key: 'points', label: '积分' },
|
|||
|
|
{ key: 'status', label: '状态' },
|
|||
|
|
{ key: 'issuedAt', label: '发放时间' },
|
|||
|
|
{ key: 'batchId', label: '批次编号' },
|
|||
|
|
{ key: 'batchStatus', label: '批次状态' },
|
|||
|
|
{ key: 'note', label: '备注' }
|
|||
|
|
]
|
|||
|
|
const { selected: exportSelected, setSelected: setExportSelected } = useExportFields(
|
|||
|
|
exportFields,
|
|||
|
|
exportFields.map((field) => field.key)
|
|||
|
|
)
|
|||
|
|
|
|||
|
|
const exportRewards = () => {
|
|||
|
|
const headers = exportFields
|
|||
|
|
.filter((field) => exportSelected.value.includes(field.key))
|
|||
|
|
.map((field) => field.label)
|
|||
|
|
const rows = rewards.value.map((item) =>
|
|||
|
|
exportFields
|
|||
|
|
.filter((field) => exportSelected.value.includes(field.key))
|
|||
|
|
.map((field) => {
|
|||
|
|
if (field.key === 'userName') return item.userName
|
|||
|
|
if (field.key === 'points') return String(item.points)
|
|||
|
|
if (field.key === 'status') return item.status
|
|||
|
|
if (field.key === 'issuedAt') return formatDate(item.issuedAt)
|
|||
|
|
if (field.key === 'batchId') return item.batchId
|
|||
|
|
if (field.key === 'batchStatus') return item.batchStatus
|
|||
|
|
return item.note ?? ''
|
|||
|
|
})
|
|||
|
|
)
|
|||
|
|
downloadCsv('rewards-demo.csv', headers, rows)
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
|
|||
|
|
onMounted(async () => {
|
|||
|
|
rewards.value = await service.getRewards()
|
|||
|
|
})
|
|||
|
|
|
|||
|
|
const applyIssue = (reward: RewardItem) => {
|
|||
|
|
reward.status = '已发放'
|
|||
|
|
reward.note = undefined
|
|||
|
|
auditStore.addLog('发放奖励', reward.userName)
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
const rollbackIssue = (reward: RewardItem, reason: string) => {
|
|||
|
|
reward.status = '待发放'
|
|||
|
|
reward.note = `回滚原因:${reason}`
|
|||
|
|
auditStore.addLog('回滚奖励', `${reward.userName}:${reason}`)
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
const retryIssue = (reward: RewardItem, reason: string) => {
|
|||
|
|
reward.status = '已发放'
|
|||
|
|
reward.note = `重试原因:${reason}`
|
|||
|
|
auditStore.addLog('重试发放奖励', `${reward.userName}:${reason}`)
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
const actionLabel = (reward: RewardItem) => {
|
|||
|
|
if (reward.status === '已发放') return '回滚'
|
|||
|
|
if (reward.status === '发放失败') return '重试'
|
|||
|
|
return '发放'
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
const handleActionClick = (reward: RewardItem) => {
|
|||
|
|
if (reward.status === '已发放') {
|
|||
|
|
actioningId.value = reward.id
|
|||
|
|
actionType.value = 'rollback'
|
|||
|
|
actionReason.value = ''
|
|||
|
|
return
|
|||
|
|
}
|
|||
|
|
if (reward.status === '发放失败') {
|
|||
|
|
actioningId.value = reward.id
|
|||
|
|
actionType.value = 'retry'
|
|||
|
|
actionReason.value = ''
|
|||
|
|
return
|
|||
|
|
}
|
|||
|
|
applyIssue(reward)
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
const cancelAction = () => {
|
|||
|
|
actioningId.value = null
|
|||
|
|
actionType.value = null
|
|||
|
|
actionReason.value = ''
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
const confirmAction = (reward: RewardItem) => {
|
|||
|
|
const reason = normalizeRewardReason(actionReason.value)
|
|||
|
|
if (actionType.value === 'rollback') {
|
|||
|
|
rollbackIssue(reward, reason)
|
|||
|
|
} else if (actionType.value === 'retry') {
|
|||
|
|
retryIssue(reward, reason)
|
|||
|
|
}
|
|||
|
|
cancelAction()
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
const filteredRewards = computed(() => {
|
|||
|
|
return rewards.value.filter((item) => {
|
|||
|
|
const matchesQuery = item.userName.includes(query.value.trim())
|
|||
|
|
const startOk = startDate.value ? new Date(item.issuedAt).getTime() >= new Date(startDate.value).getTime() : true
|
|||
|
|
const endOk = endDate.value ? new Date(item.issuedAt).getTime() <= new Date(endDate.value).getTime() : true
|
|||
|
|
return matchesQuery && startOk && endOk
|
|||
|
|
})
|
|||
|
|
})
|
|||
|
|
|
|||
|
|
const allSelected = computed(() => {
|
|||
|
|
return filteredRewards.value.length > 0 && filteredRewards.value.every((item) => selectedIds.value.includes(item.id))
|
|||
|
|
})
|
|||
|
|
|
|||
|
|
const toggleSelect = (id: string) => {
|
|||
|
|
if (selectedIds.value.includes(id)) {
|
|||
|
|
selectedIds.value = selectedIds.value.filter((item) => item !== id)
|
|||
|
|
} else {
|
|||
|
|
selectedIds.value = [...selectedIds.value, id]
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
const selectAll = () => {
|
|||
|
|
if (allSelected.value) {
|
|||
|
|
selectedIds.value = []
|
|||
|
|
} else {
|
|||
|
|
selectedIds.value = filteredRewards.value.map((item) => item.id)
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
const batchIssue = () => {
|
|||
|
|
filteredRewards.value
|
|||
|
|
.filter((item) => selectedIds.value.includes(item.id))
|
|||
|
|
.forEach(applyIssue)
|
|||
|
|
selectedIds.value = []
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
const batchRollback = () => {
|
|||
|
|
const reason = normalizeRewardReason(batchReason.value, '批量回滚')
|
|||
|
|
filteredRewards.value
|
|||
|
|
.filter((item) => selectedIds.value.includes(item.id))
|
|||
|
|
.forEach((item) => rollbackIssue(item, reason))
|
|||
|
|
selectedIds.value = []
|
|||
|
|
batchReason.value = ''
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
const totalPages = computed(() => Math.max(1, Math.ceil(filteredRewards.value.length / pageSize)))
|
|||
|
|
|
|||
|
|
const pagedRewards = computed(() => {
|
|||
|
|
const start = page.value * pageSize
|
|||
|
|
return filteredRewards.value.slice(start, start + pageSize)
|
|||
|
|
})
|
|||
|
|
|
|||
|
|
watch([query, startDate, endDate], () => {
|
|||
|
|
page.value = 0
|
|||
|
|
})
|
|||
|
|
</script>
|