OrderRefund.php 7.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184
  1. <?php
  2. // +----------------------------------------------------------------------
  3. // | 萤火商城系统 [ 致力于通过产品和服务,帮助商家高效化开拓市场 ]
  4. // +----------------------------------------------------------------------
  5. // | Copyright (c) 2017~2021 https://www.yiovo.com All rights reserved.
  6. // +----------------------------------------------------------------------
  7. // | Licensed 这不是一个自由软件,不允许对程序代码以任何形式任何目的的再发行
  8. // +----------------------------------------------------------------------
  9. // | Author: 萤火科技 <admin@yiovo.com>
  10. // +----------------------------------------------------------------------
  11. declare (strict_types = 1);
  12. namespace app\store\model;
  13. use app\store\model\User as UserModel;
  14. use app\common\model\OrderRefund as OrderRefundModel;
  15. use app\common\enum\order\refund\RefundType as RefundTypeEnum;
  16. use app\common\enum\order\refund\AuditStatus as AuditStatusEnum;
  17. use app\common\enum\order\refund\RefundStatus as RefundStatusEnum;
  18. use app\common\service\Message as MessageService;
  19. use app\common\service\order\Refund as RefundService;
  20. /**
  21. * 售后单模型
  22. * Class OrderRefund
  23. * @package app\api\model
  24. */
  25. class OrderRefund extends OrderRefundModel
  26. {
  27. /**
  28. * 获取售后单列表
  29. * @param array $param
  30. * @return mixed
  31. */
  32. public function getList(array $param = [])
  33. {
  34. // 检索查询条件
  35. $filter = $this->getFilter($param);
  36. // 获取列表数据
  37. return $this->alias('refund')
  38. ->field('refund.*, order.order_no')
  39. ->with(['orderGoods.image', 'orderData', 'user.avatar'])
  40. ->join('order', 'order.order_id = refund.order_id')
  41. ->join('user', 'user.user_id = order.user_id')
  42. ->where($filter)
  43. ->order(['refund.create_time' => 'desc', 'refund.' . $this->getPk()])
  44. ->paginate(10);
  45. }
  46. /**
  47. * 获取售后单详情
  48. * @param int $orderRefundId
  49. * @return OrderRefund|false|null
  50. */
  51. public function getDetail(int $orderRefundId)
  52. {
  53. return static::detail($orderRefundId, [
  54. 'orderData', 'images.file', 'orderGoods.image', 'express', 'address', 'user'
  55. ]) ?: false;
  56. }
  57. /**
  58. * 检索查询条件
  59. * @param array $param
  60. * @return array
  61. */
  62. private function getFilter(array $param = []): array
  63. {
  64. // 默认查询条件
  65. $params = $this->setQueryDefaultValue($param, [
  66. 'searchType' => '', // 关键词类型 (10订单号 20会员昵称 30会员ID)
  67. 'searchValue' => '', // 关键词内容
  68. 'refundType' => -1, // 售后类型
  69. 'refundStatus' => -1, // 售后单状态
  70. 'betweenTime' => [], // 申请时间
  71. ]);
  72. // 检查查询条件
  73. $filter = [];
  74. // 关键词
  75. if (!empty($params['searchValue'])) {
  76. $searchWhere = [
  77. 10 => ['order.order_no', 'like', "%{$params['searchValue']}%"],
  78. 20 => ['user.nick_name', 'like', "%{$params['searchValue']}%"],
  79. 30 => ['order.user_id', '=', (int)$params['searchValue']]
  80. ];
  81. array_key_exists($params['searchType'], $searchWhere) && $filter[] = $searchWhere[$params['searchType']];
  82. }
  83. // 起止时间
  84. if (!empty($params['betweenTime'])) {
  85. $times = between_time($params['betweenTime']);
  86. $filter[] = ['refund.create_time', '>=', $times['start_time']];
  87. $filter[] = ['refund.create_time', '<', $times['end_time'] + 86400];
  88. }
  89. // 售后类型
  90. $params['refundType'] > -1 && $filter[] = ['refund.type', '=', (int)$params['refundType']];
  91. // 处理状态
  92. $params['refundStatus'] > -1 && $filter[] = ['refund.status', '=', (int)$params['refundStatus']];
  93. return $filter;
  94. }
  95. /**
  96. * 商家审核
  97. * @param array $data
  98. * @return bool
  99. */
  100. public function audit(array $data): bool
  101. {
  102. if ($data['audit_status'] == AuditStatusEnum::REJECTED && empty($data['refuse_desc'])) {
  103. $this->error = '请输入拒绝原因';
  104. return false;
  105. }
  106. if ($data['audit_status'] == AuditStatusEnum::REVIEWED && empty($data['address_id'])) {
  107. $this->error = '请选择退货地址';
  108. return false;
  109. }
  110. $this->transaction(function () use ($data) {
  111. // 拒绝申请, 标记售后单状态为已拒绝
  112. $data['audit_status'] == AuditStatusEnum::REJECTED && $data['status'] = RefundStatusEnum::REJECTED;
  113. // 同意换货申请, 标记售后单状态为已完成
  114. $data['audit_status'] == AuditStatusEnum::REVIEWED && $this['type'] == RefundTypeEnum::EXCHANGE && $data['status'] = RefundStatusEnum::COMPLETED;
  115. // 更新退款单状态
  116. $this->save($data);
  117. // 同意售后申请, 记录退货地址
  118. if ($data['audit_status'] == AuditStatusEnum::REVIEWED) {
  119. (new OrderRefundAddress)->add((int)$this['order_refund_id'], (int)$data['address_id']);
  120. }
  121. // 订单详情
  122. $order = Order::detail($this['order_id']);
  123. // 发送消息通知
  124. MessageService::send('order.refund', [
  125. 'refund' => $this, // 退款单信息
  126. 'order_no' => $order['order_no'] // 订单信息
  127. ], $this['store_id']);
  128. });
  129. return true;
  130. }
  131. /**
  132. * 确认收货并退款
  133. * @param array $data
  134. * @return bool
  135. */
  136. public function receipt(array $data): bool
  137. {
  138. // 订单详情
  139. $order = Order::detail($this['order_id']);
  140. if ($data['refund_money'] > min($order['pay_price'], $this['orderGoods']['total_pay_price'])) {
  141. $this->error = '退款金额不能大于商品实付款金额';
  142. return false;
  143. }
  144. // 事务处理
  145. $this->transaction(function () use ($order, $data) {
  146. // 更新售后单状态
  147. $this->save([
  148. 'refund_money' => $data['refund_money'],
  149. 'is_receipt' => 1,
  150. 'status' => RefundStatusEnum::COMPLETED
  151. ]);
  152. // 消减用户的实际消费金额
  153. // 条件:判断订单是否已结算
  154. if ($order['is_settled'] == true) {
  155. (new UserModel)->setDecUserExpend($order['user_id'], $data['refund_money']);
  156. }
  157. // 执行原路退款
  158. (new RefundService)->execute($order, (string)$data['refund_money']);
  159. // 发送消息通知
  160. MessageService::send('order.refund', [
  161. 'refund' => $this, // 退款单信息
  162. 'order_no' => $order['order_no'], // 订单信息
  163. ], $this['store_id']);
  164. });
  165. return true;
  166. }
  167. /**
  168. * 获取待处理售后单数量
  169. * @return int
  170. */
  171. public function getRefundTotal()
  172. {
  173. return $this->where('status', '=', RefundStatusEnum::NORMAL)->count();
  174. }
  175. }