You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

583 lines
19 KiB

2 months ago
1 month ago
2 months ago
2 months ago
2 months ago
2 months ago
2 months ago
1 month ago
1 month ago
1 month ago
1 month ago
1 month ago
1 month ago
1 month ago
1 month ago
1 month ago
1 month ago
2 months ago
2 months ago
2 months ago
2 months ago
1 month ago
2 months ago
2 months ago
1 month ago
1 month ago
2 months ago
2 months ago
1 month ago
2 months ago
2 months ago
2 months ago
2 months ago
1 month ago
2 months ago
2 months ago
2 months ago
2 months ago
2 months ago
2 months ago
2 months ago
2 months ago
2 months ago
2 months ago
2 months ago
2 months ago
  1. <template>
  2. <el-card style="margin-bottom: 0.5vh;margin-top:0.5vh;width:82.5vw">
  3. <el-col style="margin-bottom: 0.5vh">
  4. <el-text size="large">精网号</el-text>
  5. <el-input v-model="searchForm.jwcode" placeholder="请输入精网号" style="width: 12vw;margin-right:1vw" clearable/>
  6. <el-text size="large">商品名</el-text>
  7. <el-select v-model="searchForm.goodsName" placeholder="请输入商品名" style="width: 12vw;margin-right:1vw"
  8. clearable>
  9. <el-option v-for="item in refundGoodsOptions" :key="item" :label="item" :value="item"></el-option>
  10. </el-select>
  11. <el-text size="large">退款方式</el-text>
  12. <el-select v-model="searchForm.refundModel" placeholder="请选择" style="width: 12vw;margin-right:1vw" clearable>
  13. <el-option label="全部退款" value="0"/>
  14. <el-option label="部分退款" value="1"/>
  15. </el-select>
  16. <el-text size="large">所属地区</el-text>
  17. <el-cascader v-model="selectedMarketPath" :options="market" placeholder="请选择所属地区" clearable
  18. style="width:12vw"
  19. @change="handleMarketChange"/>
  20. </el-col>
  21. <el-col>
  22. <el-text size="large">
  23. {{ activeName === 'wait' ? '提交时间:' : '审核时间:' }}
  24. </el-text>
  25. <el-date-picker v-model="dateRange" type="datetimerange" range-separator="" start-placeholder="开始时间"
  26. class="time-controls"
  27. end-placeholder="结束时间" style="margin-right:1vw;width:25vw" @change="handleDatePickerChange"
  28. :default-time="defaultTime" :disabled-date="disabledDate"/>
  29. <el-button @click="getToday()" :type="activeTimeRange === 'today' ? 'primary' : ''"></el-button>
  30. <el-button @click="getYesterday()" :type="activeTimeRange === 'yesterday' ? 'primary' : ''"></el-button>
  31. <el-button @click="get7Days()" :type="activeTimeRange === '7days' ? 'primary' : ''">近7天</el-button>
  32. <el-button type="success" @click="resetSearch">重置</el-button>
  33. <el-button type="primary" @click="handleSearch">查询</el-button>
  34. </el-col>
  35. </el-card>
  36. <el-card>
  37. <el-tabs v-model="activeName" type="card" @tab-click="handleClick">
  38. <el-tab-pane label="待审核" name="wait"></el-tab-pane>
  39. <el-tab-pane label="已通过" name="pass"></el-tab-pane>
  40. <el-tab-pane label="已驳回" name="reject"></el-tab-pane>
  41. </el-tabs>
  42. <div>
  43. 总条数{{ format3(stats.totalNum) }}&nbsp;&nbsp;&nbsp;&nbsp;
  44. 退款总金币数{{ format3(stats.permanentGolds + stats.freeGolds + stats.taskGolds) }}金币&nbsp;&nbsp;&nbsp;&nbsp;
  45. 永久金币{{ format3(stats.permanentGolds) }}金币&nbsp;&nbsp;&nbsp;&nbsp;
  46. 免费金币{{ format3(stats.freeGolds.toFixed(2)) }}金币&nbsp;&nbsp;&nbsp;&nbsp;
  47. 任务金币{{ format3(stats.taskGolds.toFixed(2)) }}金币
  48. </div>
  49. <el-table :data="tableData" style="height:55vh" @sort-change="handleSortChange">
  50. <el-table-column type="index" label="序号" width="60"/>
  51. <el-table-column prop="name" label="姓名" width="120"/>
  52. <el-table-column prop="jwcode" label="精网号" width="120"/>
  53. <el-table-column prop="market" label="所属地区" width="120"/>
  54. <el-table-column prop="refundType" label="退款类型" width="120"/>
  55. <el-table-column prop="refundModel" label="退款方式" width="120">
  56. <template #default="{ row }">
  57. {{ row.refundModel === 0 ? '全部退款' : '部分退款' }}
  58. </template>
  59. </el-table-column>
  60. <el-table-column prop="goodsName" label="退款商品" width="120" show-overflow-tooltip/>
  61. <el-table-column prop="sumGold" label="退款金额" width="120" sortable="custom">
  62. <template #default="{ row }">
  63. {{ row.sumGold / 100 }}
  64. </template>
  65. </el-table-column>
  66. <el-table-column prop="permanentGold" label="永久金币" width="120" sortable="custom">
  67. <template #default="{ row }">
  68. {{ row.permanentGold / 100 }}
  69. </template>
  70. </el-table-column>
  71. <el-table-column prop="freeGold" label="免费金币" width="120" sortable="custom">
  72. <template #default="{ row }">
  73. {{ (row.freeJune + row.freeDecember) / 100 }}
  74. </template>
  75. </el-table-column>
  76. <el-table-column prop="taskGold" label="任务金币" width="120" sortable="custom">
  77. <template #default="{ row }">
  78. {{ row.taskGold / 100 }}
  79. </template>
  80. </el-table-column>
  81. <el-table-column prop="remark" label="备注" width="150" show-overflow-tooltip/>
  82. <el-table-column prop="adminName" label="提交人" width="120"/>
  83. <el-table-column v-if="checkTab === 'reject'" prop="rejectReason" label="驳回理由" width="150"
  84. show-overflow-tooltip/>
  85. <el-table-column v-if="checkTab !== 'pending'" prop="auditName" label="审核人" width="120"/>
  86. <el-table-column prop="createTime" label="提交时间" width="180" sortable="custom">
  87. <template #default="{ row }">
  88. <!-- {{ moment(row.createTime).format('YYYY-MM-DD HH:mm:ss') }}-->
  89. {{
  90. checkTab === 'pending'
  91. ? moment(row.auditTime).format('YYYY-MM-DD HH:mm:ss')
  92. : moment(row.createTime).format('YYYY-MM-DD HH:mm:ss')
  93. }}
  94. </template>
  95. </el-table-column>
  96. <el-table-column v-if="checkTab !== 'pending'" prop="auditTime" label="审核时间" width="180" sortable="custom">
  97. <template #default="{ row }">
  98. {{ row.auditTime ? moment(row.auditTime).format('YYYY-MM-DD HH:mm:ss') : '--' }}
  99. </template>
  100. </el-table-column>
  101. <el-table-column v-if="checkTab === 'pending'" fixed="right" prop="operation" label="操作" width="150px">
  102. <template #default="scope">
  103. <div class="operation">
  104. <el-popconfirm title="确定要通过此条记录吗?" @confirm="handleApprove(scope.row)">
  105. <template #reference>
  106. <el-button :disabled="scope.row.auditStatus === 1 || scope.row.auditStatus === 2" type="primary" text>
  107. 通过
  108. </el-button>
  109. </template>
  110. </el-popconfirm>
  111. <el-button :disabled="scope.row.auditStatus === 1 || scope.row.auditStatus === 2" type="primary" text
  112. @click="showRejectDialog(scope.row)">
  113. 驳回
  114. </el-button>
  115. </div>
  116. </template>
  117. </el-table-column>
  118. </el-table>
  119. <el-pagination class="pagination" v-model:current-page="pagination.pageNum" v-model:page-size="pagination.pageSize"
  120. layout="total, sizes, prev, pager, next, jumper" :total="pagination.total"
  121. @size-change="handlePageSizeChange"
  122. @current-change="handleCurrentChange"></el-pagination>
  123. </el-card>
  124. <el-dialog v-model="rejectDialogVisible" title="驳回理由" width="500px">
  125. <el-form>
  126. <el-form-item label="驳回理由" required>
  127. <el-input v-model="rejectReason" type="textarea" :rows="4" placeholder="请输入驳回理由" maxlength="200"
  128. show-word-limit/>
  129. </el-form-item>
  130. </el-form>
  131. <template #footer>
  132. <span class="dialog-footer">
  133. <el-button @click="rejectDialogVisible = false">取消</el-button>
  134. <el-button type="primary" @click="handleReject">确定</el-button>
  135. </span>
  136. </template>
  137. </el-dialog>
  138. </template>
  139. <script setup>
  140. import {onMounted, reactive, ref} from 'vue'
  141. import {ElMessage} from 'element-plus'
  142. import API from '@/util/http.js'
  143. import moment from 'moment'
  144. import {useAdminStore} from "@/store/index.js";
  145. import {storeToRefs} from "pinia";
  146. import {findMenuById, permissionMapping} from "@/utils/menuTreePermission.js"
  147. import dayjs from "dayjs";
  148. const adminStore = useAdminStore();
  149. const {adminData, menuTree} = storeToRefs(adminStore);
  150. const defaultTime = [
  151. new Date(2000, 1, 1, 0, 0, 0),
  152. new Date(2000, 2, 1, 23, 59, 59),
  153. ]
  154. // 当前激活的时间按钮
  155. const activeTimeRange = ref('')
  156. const scopeValue = ref(null) // 当前行
  157. const rejectDialogVisible = ref(false)
  158. const rejectReason = ref('')
  159. // 状态常量
  160. const STATUS = {
  161. PENDING: 0, // 待审核
  162. APPROVED: 1, // 通过
  163. REJECTED: 2 // 驳回
  164. }
  165. //无法选择的时间
  166. const disabledDate = (time) => {
  167. const limitDate = new Date(2025, 0, 1);
  168. return time.getTime() < limitDate.getTime();
  169. }
  170. // 搜索表单数据
  171. const searchForm = ref({
  172. jwcode: '',
  173. refundModel: '',
  174. goodsName: '',
  175. market: "",
  176. startTime: '',
  177. endTime: '',
  178. auditStatus: '0'
  179. })
  180. const checkTab = ref('pending') // 能否不用STATUS常量,0是待审批,1是已通过,2是驳回,参数status需要Integer
  181. const dateRange = ref([])
  182. const pagination = ref({
  183. pageNum: 1,
  184. pageSize: 50,
  185. total: 0
  186. })
  187. const tableData = ref([])
  188. const marketOptions = ref([])
  189. const refundGoodsOptions = ref([])
  190. const adminInfo = ref({})
  191. // 统计合计数
  192. const stats = ref({
  193. totalNum: 0,
  194. totalCoins: 0,
  195. permanentGolds: 0,
  196. freeGolds: 0,
  197. taskGolds: 0
  198. })
  199. const rejectVisible = ref(false)
  200. const rejectObj = ref({})
  201. const passObj = ref({})
  202. // 标签页默认是待审批
  203. const activeName = ref('wait')
  204. const sortField = ref('')
  205. const sortOrder = ref('')
  206. const market = ref("")
  207. // 处理排序事件
  208. const handleSortChange = (column) => {
  209. if (column.prop === 'sumGold') {
  210. sortField.value = 'sum_gold'
  211. } else if (column.prop === 'permanentGold') {
  212. sortField.value = 'permanent_gold'
  213. } else if (column.prop === 'freeGold') {
  214. sortField.value = 'freeGold'
  215. } else if (column.prop === 'taskGold') {
  216. sortField.value = 'task_gold'
  217. } else if (column.prop === 'createTime') {
  218. sortField.value = 'create_time'
  219. } else if (column.prop === 'auditTime') {
  220. sortField.value = 'audit_time'
  221. } else {
  222. sortField.value = ''
  223. }
  224. sortOrder.value = column.order === 'ascending' ? 'asc' : 'desc'
  225. console.log('排序字段:', sortField.value)
  226. console.log('排序方式:', sortOrder.value)
  227. get()
  228. }
  229. // 显示驳回对话框
  230. const showRejectDialog = (row) => {
  231. scopeValue.value = row
  232. rejectReason.value = ''
  233. rejectDialogVisible.value = true
  234. }
  235. // 查商品名
  236. const getRefundGoods = async () => {
  237. try {
  238. const res = await API({url: '/general/goods'})
  239. refundGoodsOptions.value = res.data || []
  240. } catch (error) {
  241. console.error('获取商品列表失败', error)
  242. }
  243. }
  244. // 搜索方法
  245. const get = async function (val) {
  246. try {
  247. if (typeof val === 'number') {
  248. pagination.value.pageNum = val
  249. }
  250. if (dateRange.value && dateRange.value.length === 2) {
  251. searchForm.value.startTime = moment(dateRange.value[0]).format('YYYY-MM-DD HH:mm:ss')
  252. searchForm.value.endTime = moment(dateRange.value[1]).format('YYYY-MM-DD HH:mm:ss')
  253. } else {
  254. searchForm.value.startTime = ''
  255. searchForm.value.endTime = ''
  256. }
  257. if (searchForm.value.market === '总部' || searchForm.value.market === '研发部') {
  258. searchForm.value.market = '';
  259. }
  260. const params = {
  261. pageNum: pagination.value.pageNum,
  262. pageSize: pagination.value.pageSize,
  263. refundAudit: {
  264. ...searchForm.value,
  265. sortField: sortField.value,
  266. sortOrder: sortOrder.value
  267. }
  268. }
  269. console.log('看看传给后端的参数:', params)
  270. // 校验精网号(数字格式)
  271. if (searchForm.value.jwcode) {
  272. // 纯数字
  273. const numberRegex = /^\d{1,9}$/;
  274. // 检查是否不是数字
  275. if (!numberRegex.test(searchForm.value.jwcode)) {
  276. ElMessage.error('请检查精网号格式')
  277. // 上面提示过了
  278. return
  279. }
  280. }
  281. const res = await API({url: '/audit/selectRefund', data: params})
  282. tableData.value = res.list || []
  283. pagination.value.total = res.total || 0
  284. console.log('查全部的total', pagination.value.total, res.total)
  285. } catch (error) {
  286. console.error('获取数据失败', error)
  287. }
  288. }
  289. // 通过
  290. const handleApprove = async (row) => {
  291. if (findMenuById(menuTree.value, permissionMapping.Refund_Approval)) {
  292. try {
  293. const params = {
  294. orderCode: row.orderCode,
  295. auditId: adminData.value.id,
  296. action: 1,// action的1是通过,2是驳回
  297. rejectReason: ''
  298. }
  299. await API({url: '/audit/audit', data: params})
  300. ElMessage.success('审核通过成功')
  301. get()
  302. getStats()
  303. } catch (error) {
  304. console.error('审核通过失败', error)
  305. ElMessage.error('操作失败')
  306. }
  307. } else {
  308. ElMessage.warning('没有权限')
  309. }
  310. }
  311. // 处理驳回操作
  312. const handleReject = async () => {
  313. if (findMenuById(menuTree.value, permissionMapping.Refund_Approval)) {
  314. if (!rejectReason.value.trim()) {
  315. ElMessage.warning('请输入驳回理由')
  316. return
  317. }
  318. try {
  319. const params = {
  320. orderCode: scopeValue.value.orderCode,
  321. auditId: adminData.value.id,
  322. action: 2,
  323. rejectReason: rejectReason.value
  324. }
  325. await API({url: '/audit/audit', data: params})
  326. ElMessage.success('驳回成功')
  327. rejectDialogVisible.value = false
  328. get()
  329. getStats()
  330. console.log('看看驳回参数', params)
  331. } catch (error) {
  332. console.error('驳回失败', error)
  333. ElMessage.error('操作失败')
  334. }
  335. } else {
  336. ElMessage.warning('没有权限')
  337. }
  338. }
  339. const getStats = async () => {
  340. try {
  341. const params = {
  342. pageNum: pagination.value.pageNum,
  343. pageSize: pagination.value.pageSize,
  344. refundAudit: {
  345. ...searchForm.value
  346. }
  347. }
  348. if (searchForm.value.jwcode) {
  349. // 纯数字
  350. const numberRegex = /^\d{1,9}$/;
  351. // 检查是否不是数字
  352. if (!numberRegex.test(searchForm.value.jwcode)) {
  353. // ElMessage.error('精网号必须为数字格式')
  354. // 上面提示过了
  355. return
  356. }
  357. }
  358. const res = await API({
  359. url: '/audit/sumRefundGold',
  360. data: params
  361. })
  362. stats.value.totalNum = res.totalNum
  363. stats.value.permanentGolds = res.permanentGolds / 100
  364. stats.value.freeGolds = res.freeGolds / 100
  365. stats.value.taskGolds = res.taskGolds / 100
  366. console.log('see see stats和搜索对象', stats.value, params)
  367. } catch (error) {
  368. console.log('请求失败', error)
  369. }
  370. }
  371. // 搜索
  372. const handleSearch = function () {
  373. trimJwCode()
  374. get()
  375. getStats()
  376. }
  377. // 重置
  378. const resetSearch = function () {
  379. const auditStatus = searchForm.value.auditStatus;
  380. searchForm.value = {
  381. jwcode: '',
  382. refundType: '',
  383. goodsName: '',
  384. market: "",
  385. startTime: '',
  386. endTime: '',
  387. sortField: '',
  388. sortOrder: '',
  389. auditStatus: auditStatus
  390. }
  391. dateRange.value = []
  392. activeTimeRange.value = '' // 清除激活状态
  393. selectedMarketPath.value = []
  394. get()
  395. getStats()
  396. }
  397. // 今天
  398. const getToday = function () {
  399. const today = dayjs()
  400. const startTime = today.startOf('day').format('YYYY-MM-DD HH:mm:ss')
  401. const endTime = today.endOf('day').format('YYYY-MM-DD HH:mm:ss')
  402. dateRange.value = [startTime, endTime]
  403. console.log('dateRange', dateRange.value)
  404. activeTimeRange.value = 'today'
  405. get()
  406. getStats()
  407. }
  408. // 昨天
  409. const getYesterday = function () {
  410. const today = dayjs()
  411. const startTime = today.subtract(1, 'day').startOf('day').format('YYYY-MM-DD HH:mm:ss')
  412. const endTime = today.subtract(1, 'day').endOf('day').format('YYYY-MM-DD HH:mm:ss')
  413. dateRange.value = [startTime, endTime]
  414. console.log('dateRange', dateRange.value)
  415. activeTimeRange.value = 'yesterday'
  416. get()
  417. getStats()
  418. }
  419. // 近7天
  420. const get7Days = function () {
  421. const today = dayjs()
  422. const startTime = today.subtract(6, 'day').startOf('day').format('YYYY-MM-DD HH:mm:ss')
  423. const endTime = today.endOf('day').format('YYYY-MM-DD HH:mm:ss')
  424. dateRange.value = [startTime, endTime]
  425. console.log('dateRange', dateRange.value)
  426. activeTimeRange.value = '7days'
  427. get()
  428. getStats()
  429. }
  430. const handleClick = function (tab, event) {
  431. activeName.value = tab.props.name
  432. if (tab.props.name === 'wait') {
  433. adminWait()
  434. } else if (tab.props.name === 'pass') {
  435. adminPass()
  436. } else if (tab.props.name === 'reject') {
  437. adminReject()
  438. }
  439. }
  440. // 当前状态
  441. const getCurrentStatus = () => {
  442. switch (activeName.value) {
  443. case 'wait':
  444. return STATUS.PENDING
  445. case 'pass':
  446. return STATUS.APPROVED
  447. case 'reject':
  448. return STATUS.REJECTED
  449. default:
  450. return ''
  451. }
  452. }
  453. // 待审核
  454. const adminWait = async function () {
  455. checkTab.value = 'pending'
  456. searchForm.value.auditStatus = STATUS.PENDING
  457. await get()
  458. await getStats()
  459. }
  460. // 已通过
  461. const adminPass = async function () {
  462. checkTab.value = 'pass'
  463. searchForm.value.auditStatus = STATUS.APPROVED
  464. await get()
  465. await getStats()
  466. }
  467. // 已驳回
  468. const adminReject = async function () {
  469. checkTab.value = 'reject'
  470. searchForm.value.auditStatus = STATUS.REJECTED
  471. await get()
  472. await getStats()
  473. }
  474. const selectedMarketPath = ref("")
  475. const handleMarketChange = (value) => {
  476. if (value && value.length > 0) {
  477. searchForm.value.market = value[value.length - 1]
  478. } else {
  479. searchForm.value.market = ''
  480. }
  481. }
  482. // 获取地区,修改为级联下拉框
  483. const getMarket = async function () {
  484. try {
  485. const result = await API({
  486. url: '/market/selectMarket',
  487. })
  488. console.log('请求成功', result)
  489. // 递归转换树形结构为级联选择器需要的格式(跳过第一级节点)
  490. const transformTree = (nodes) => {
  491. // 直接处理第一级节点的子节点
  492. const allChildren = nodes.flatMap(node => node.children || []);
  493. return allChildren.map(child => {
  494. const grandchildren = child.children && child.children.length
  495. ? transformTree([child]) // 递归处理子节点
  496. : null;
  497. return {
  498. value: child.name,
  499. label: child.name,
  500. children: grandchildren
  501. };
  502. });
  503. };
  504. market.value = transformTree(result.data)
  505. console.log('转换后的地区树==============', market.value)
  506. } catch (error) {
  507. console.log('请求失败', error)
  508. }
  509. }
  510. const trimJwCode = () => {
  511. if (searchForm.value.jwcode) {
  512. searchForm.value.jwcode = searchForm.value.jwcode.replace(/\s/g, '');
  513. }
  514. }
  515. // 日期选择器变化时清除按钮激活状态
  516. const handleDatePickerChange = () => {
  517. activeTimeRange.value = ''
  518. }
  519. const format3 = (num) => {
  520. // 每三位添加逗号
  521. return num.toLocaleString('en-US')
  522. }
  523. // 表单验证
  524. const rules = reactive({
  525. reason: [{required: true, message: '请输入驳回理由', trigger: 'blur'}]
  526. })
  527. const handlePageSizeChange = function (val) {
  528. pagination.value.pageSize = val
  529. get()
  530. }
  531. const handleCurrentChange = function (val) {
  532. pagination.value.pageNum = val
  533. get()
  534. }
  535. onMounted(async () => {
  536. getRefundGoods()
  537. await getMarket()
  538. await get()
  539. await getStats()
  540. })
  541. </script>
  542. <style scoped>
  543. .pagination {
  544. display: flex;
  545. margin-top: 0.5vh;
  546. }
  547. .operation {
  548. display: flex;
  549. }
  550. .time-controls {
  551. display: flex;
  552. align-items: center;
  553. display: flex;
  554. align-items: center;
  555. gap: 10px;
  556. }
  557. </style>