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.

591 lines
19 KiB

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