list.vue 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393
  1. <template>
  2. <div class="orders-lists">
  3. <el-card class="!border-none" shadow="never">
  4. <el-form :inline="true" class="mb-[-16px]">
  5. <el-form-item label="订单类型" class="w-[280px]">
  6. <el-select class="w-[280px]" v-model="queryParams.type">
  7. <el-option label="全部" value />
  8. <el-option label="后台点单" value="0" />
  9. <el-option label="扫码点单" value="1" />
  10. </el-select>
  11. </el-form-item>
  12. <el-form-item label="订单状态" class="w-[280px]">
  13. <el-select class="w-[280px]" v-model="queryParams.status">
  14. <el-option label="全部" value />
  15. <el-option label="待下单" value="0" />
  16. <el-option label="待结帐" value="1" />
  17. <el-option label="已完成" value="2" />
  18. <el-option label="已退款" value="4" />
  19. </el-select>
  20. </el-form-item>
  21. <el-form-item label="创建时间" class="w-[280px]">
  22. <el-date-picker v-model="createTime" @change="setCreateTime" type="datetimerange"
  23. range-separator="-" start-placeholder="开始时间" end-placeholder="结束时间" />
  24. </el-form-item>
  25. <el-form-item label="结账时间" class="w-[280px]">
  26. <el-date-picker v-model="checkoutTime" @change="setCheckoutTime" type="datetimerange"
  27. range-separator="-" start-placeholder="开始时间" end-placeholder="结束时间" />
  28. </el-form-item>
  29. <el-form-item class="w-[280px]">
  30. <el-button type="primary" @click="resetPage">查询</el-button>
  31. <el-button @click="resetParams">重置</el-button>
  32. </el-form-item>
  33. </el-form>
  34. </el-card>
  35. <el-card class="!border-none mt-4" shadow="never">
  36. <el-table v-loading="pager.loading" :data="pager.lists" row-key="id" :expand-row-keys="expandRowKeys" @expand-change="handleExpandChange">
  37. <el-table-column type="expand" >
  38. <template #default="{ row }">
  39. <div class="order-detail-wrapper" v-loading="!orderDetails[row.id]">
  40. <template v-if="orderDetails[row.id]">
  41. <div class="detail-header">
  42. <span>订单详情</span>
  43. <span class="time">下单时间:{{ timeFormat(orderDetails[row.id].createTime) }}</span>
  44. </div>
  45. <el-table :data="orderDetails[row.id].dishes" border class="detail-table">
  46. <el-table-column label="菜品图片" width="120" align="center">
  47. <template #default="{ row }">
  48. <el-image :src="row.image" :preview-src-list="[row.image]"
  49. class="dish-image" fit="cover" />
  50. </template>
  51. </el-table-column>
  52. <el-table-column prop="name" label="菜品名称">
  53. <template #default="{ row }">
  54. <div>
  55. {{ row.name }}
  56. <div v-if="row.specsList && row.specsList.length" class="mt-1">
  57. <el-tag
  58. v-for="spec in row.specsList"
  59. :key="spec.id"
  60. size="small"
  61. class="mr-1 mb-1"
  62. >
  63. {{ spec.name }}: {{ spec.value }}
  64. </el-tag>
  65. </div>
  66. </div>
  67. </template>
  68. </el-table-column>
  69. <el-table-column prop="amount" label="单价">
  70. <template #default="{ row }">
  71. ¥{{ row.amount }}
  72. </template>
  73. </el-table-column>
  74. <el-table-column prop="number" label="数量" width="120" align="center" />
  75. <el-table-column label="小计" width="120" align="right">
  76. <template #default="{ row }">
  77. ¥{{ row.amount * row.number}}
  78. </template>
  79. </el-table-column>
  80. </el-table>
  81. <div class="detail-footer">
  82. <div class="total-info">
  83. <span>共 {{ getTotalCount(orderDetails[row.id].dishes) }} 件商品</span>
  84. <span class="total-price">
  85. 订单总价:<em>¥{{ orderDetails[row.id].amount}}</em>
  86. </span>
  87. </div>
  88. </div>
  89. </template>
  90. </div>
  91. </template>
  92. </el-table-column>
  93. <el-table-column type="index" min-width="80" />
  94. <el-table-column label="订单号" prop="number" show-overflow-tooltip></el-table-column>
  95. <el-table-column label="支付金额" min-width="160" prop="amount">
  96. <template #default="{ row }">
  97. <div>
  98. <div :style="row.status > 1 && row.status != 6 ? 'text-decoration: line-through;': ''" v-if="row.status > 1 && row.status != 6">订单金额:¥{{row.amount}}</div>
  99. <div style="color: #f01414;" v-if="row.status >= 2 && row.status != 6">
  100. 实付:¥{{row.payAmount || row.amount}} {{row.status==4 ? '退款:¥' + row.refundAmount : ''}}
  101. </div>
  102. <div v-if="row.ticketNo" class="refund-info">
  103. <div style="font-size: 12px;">抵扣券券号:{{ row.ticketNo }}</div>
  104. <div style="font-size: 12px;">抵扣券金额:¥{{ row.ticketAmount }}</div>
  105. </div>
  106. </div>
  107. </template>
  108. </el-table-column>
  109. <el-table-column label="桌号" prop="deskName"></el-table-column>
  110. <el-table-column label="备注" prop="remark" min-width="150" show-tooltip-when-overflow>
  111. <!-- <template #default="{ row }">
  112. <div class="remark-text">
  113. {{row.remark}}
  114. </div>
  115. </template> -->
  116. </el-table-column>
  117. <el-table-column label="类型" prop="type"
  118. :formatter="(row: any) => (row.type == 0 ? '后台点单' : '扫码点单')"></el-table-column>
  119. <el-table-column label="状态" min-width="160" prop="status" >
  120. <template #default="{ row }">
  121. <div>
  122. {{ row.status == 4 ? '退款成功' : row.status == 2 ? '已完成' : row.status == 0 ? '待下单'
  123. : row.status == 1 ? '待支付' : row.status == 5 ? '退款中' : '已取消' }}
  124. <div v-if="row.refundStatus=='SUCCESS'" class="refund-info">
  125. <div>退款时间:{{ row.refundTime }}</div>
  126. <div>退款金额:¥{{ row.refundAmount }}</div>
  127. </div>
  128. </div>
  129. </template>
  130. </el-table-column>
  131. <el-table-column label="结账时间" prop="checkoutTime"></el-table-column>
  132. <el-table-column label="创建时间" prop="createTime"></el-table-column>
  133. <el-table-column label="操作" width="200" fixed="right">
  134. <template #default="{ row }">
  135. <el-button type="primary" link @click="showDetail(row)">查看已点菜品</el-button>
  136. <el-button v-if="row.status=='2' && !row.refundAmount" type="danger" link @click="refund(row)">退款</el-button>
  137. </template>
  138. </el-table-column>
  139. </el-table>
  140. <div class="flex justify-end mt-4">
  141. <pagination v-model="pager" @change="getLists" />
  142. </div>
  143. </el-card>
  144. <el-dialog
  145. v-model="refundDialogVisible"
  146. title="退款"
  147. width="400px"
  148. @close="refundDialogVisible = false">
  149. <div class="refund-dialog-content">
  150. <el-input
  151. v-model="refundAmount"
  152. placeholder="请输入退款金额"
  153. type="number">
  154. <template #prepend><icon :size="25" name="el-icon-money" /></template>
  155. <template #append>元</template>
  156. </el-input>
  157. </div>
  158. <template #footer>
  159. <span class="dialog-footer">
  160. <el-button @click="refundDialogVisible = false">取 消</el-button>
  161. <el-button type="primary" :loading="refundLoading" @click="confirmRefund">确 定</el-button>
  162. </span>
  163. </template>
  164. </el-dialog>
  165. </div>
  166. </template>
  167. <script setup lang="ts">
  168. import { ordersList, getOrderDetail, refundReq } from '@/api/orders'
  169. import { usePaging } from '@/hooks/usePaging'
  170. import { timeFormat } from '@/utils/util'
  171. import feedback from '@/utils/feedback'
  172. // import Money from '@/utils/money'
  173. const createTime = ref()
  174. const checkoutTime = ref()
  175. const queryParams = reactive({
  176. type: '',
  177. status: '',
  178. checkoutTime: '',
  179. createTime: ''
  180. })
  181. const { pager, getLists, resetPage, resetParams } = usePaging({
  182. fetchFun: ordersList,
  183. params: queryParams
  184. })
  185. const setCreateTime = (v : any) => {
  186. queryParams.createTime =
  187. Math.round(v[0].getTime() / 1000).toString() +
  188. ',' +
  189. Math.round(v[1].getTime() / 1000).toString()
  190. }
  191. const setCheckoutTime = (v : any) => {
  192. queryParams.checkoutTime =
  193. Math.round(v[0].getTime() / 1000).toString() +
  194. ',' +
  195. Math.round(v[1].getTime() / 1000).toString()
  196. }
  197. // watch(queryParams, (newV, oldV) => {
  198. // console.log(JSON.stringify(newV))
  199. // })
  200. const expandRowKeys = ref<string[]>([])
  201. const orderDetails = ref<Record<string, any>>({})
  202. const handleExpandChange = (row: any, expandedRows: any) => {
  203. // console.log('当前行展开状态改变:', row, expandedRows);
  204. if (!orderDetails.value[row.id]) {
  205. showDetail(row);
  206. }
  207. }
  208. // 添加这些变量到组件顶层
  209. const refundDialogVisible = ref(false)
  210. const refundAmount = ref(0)
  211. var refundOrder: {
  212. refundTime: string;
  213. refundAmount: number;
  214. payAmount: number;
  215. amount: number;
  216. status: number; id: any;
  217. };
  218. // 添加loading状态变量
  219. const refundLoading = ref(false)
  220. // 修改refund函数
  221. const refund = async (row: any) => {
  222. refundDialogVisible.value = true
  223. refundAmount.value = 0
  224. refundOrder = row;
  225. }
  226. const confirmRefund = async () => {
  227. try {
  228. if(!refundAmount.value || refundAmount.value<=0){
  229. return feedback.msgError('退款金额不合法')
  230. }
  231. if(refundAmount.value > (refundOrder.payAmount || refundOrder.amount)){
  232. return feedback.msgError('退款金额不能大于订单实付金额')
  233. }
  234. refundLoading.value = true
  235. const res = await refundReq({ orderId: refundOrder.id, payAmount: refundAmount.value })
  236. if (res) {
  237. feedback.msgSuccess('退款成功')
  238. refundOrder.status = 5;
  239. // refundOrder.refundAmount = refundAmount.value;
  240. // refundOrder.refundTime = "待处理";
  241. setTimeout(()=>{
  242. getLists();
  243. },200)
  244. } else {
  245. feedback.msgError('退款失败')
  246. }
  247. } catch (error) {
  248. console.error('退款失败:', error)
  249. } finally {
  250. refundLoading.value = false
  251. refundDialogVisible.value = false
  252. }
  253. }
  254. // 展示订单详情
  255. const showDetail = async (row : any) => {
  256. try {
  257. // 如果已经加载过该订单详情,直接展开
  258. if (orderDetails.value[row.id]) {
  259. expandRowKeys.value = [row.id]
  260. return
  261. }
  262. // 调用获取订单详情接口
  263. const res = await getOrderDetail({id:row.id})
  264. console.warn("***getOrderDetail***",res)
  265. if (res) {
  266. // 定义正确的类型
  267. let orderDetail: Record<string, any> = {};
  268. orderDetail.id = res.orderItem.id;
  269. orderDetail.amount = res.orderItem.amount;//总价
  270. orderDetail.createTime = res.orderItem.createTime;
  271. orderDetail.dishes = res.orderDishList;
  272. // 保存订单详情
  273. orderDetails.value[row.id] = orderDetail
  274. // 展开当前行
  275. expandRowKeys.value = [row.id]
  276. } else {
  277. feedback.msgError('获取订单详情失败111')
  278. }
  279. } catch (error) {
  280. console.error('获取订单详情失败:', error)
  281. feedback.msgError('获取订单详情失败')
  282. }
  283. }
  284. // 计算商品总数
  285. const getTotalCount = (dishes : any[]) => {
  286. return dishes.reduce((total, dish) => total + dish.number, 0)
  287. }
  288. // 计算订单总价
  289. // const getTotalPrice = (dishes: any[]) => {
  290. // return dishes.reduce((total, dish) => {
  291. // return Money.add(total, Money.multiply(dish.price, dish.count))
  292. // }, 0)
  293. // }
  294. getLists()
  295. </script>
  296. <style lang="scss" scoped>
  297. .order-detail-wrapper {
  298. padding: 20px;
  299. background: #f8f8f8;
  300. .detail-header {
  301. margin-bottom: 20px;
  302. display: flex;
  303. justify-content: space-between;
  304. align-items: center;
  305. span {
  306. font-size: 14px;
  307. color: #333;
  308. &.time {
  309. color: #999;
  310. }
  311. }
  312. }
  313. .detail-table {
  314. margin-bottom: 20px;
  315. :deep(.dish-image) {
  316. width: 60px;
  317. height: 60px;
  318. border-radius: 4px;
  319. }
  320. }
  321. .detail-footer {
  322. display: flex;
  323. justify-content: flex-end;
  324. .total-info {
  325. text-align: right;
  326. span {
  327. margin-left: 20px;
  328. font-size: 14px;
  329. color: #666;
  330. &.total-price {
  331. em {
  332. font-style: normal;
  333. font-size: 16px;
  334. color: #f56c6c;
  335. font-weight: bold;
  336. }
  337. }
  338. }
  339. }
  340. }
  341. }
  342. .refund-dialog-content {
  343. padding: 20px 0px;
  344. :deep(.el-input) {
  345. width: 100%;
  346. }
  347. }
  348. .dialog-footer {
  349. text-align: right;
  350. .el-button + .el-button {
  351. margin-left: 12px;
  352. }
  353. }
  354. .refund-info {
  355. font-size: 12px;
  356. color: #999;
  357. margin-top: 4px;
  358. }
  359. .remark-text {
  360. display: -webkit-box;
  361. -webkit-box-orient: vertical;
  362. -webkit-line-clamp: 2;
  363. overflow: hidden;
  364. text-overflow: ellipsis;
  365. word-break: break-all;
  366. line-height: 1.5;
  367. max-height: 3em; // 2行的高度
  368. }
  369. </style>