Aftersale.php 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410
  1. <?php
  2. namespace app\admin\controller\shopro\order;
  3. use addons\shopro\model\OrderAftersaleLog;
  4. use app\admin\model\shopro\order\Order;
  5. use app\admin\model\shopro\order\OrderItem;
  6. use app\admin\model\shopro\order\Aftersale as OrderAftersale;
  7. use app\common\controller\Backend;
  8. use think\db;
  9. /**
  10. * 订单商品明细
  11. *
  12. * @icon fa fa-circle-o
  13. */
  14. class Aftersale extends Backend
  15. {
  16. /**
  17. * Aftersale模型对象
  18. * @var \app\admin\model\shopro\order\Aftersale
  19. */
  20. protected $model = null;
  21. protected $orderModel = null;
  22. public function _initialize()
  23. {
  24. parent::_initialize();
  25. // 手动加载语言包
  26. $this->loadlang('shopro/order/order');
  27. $this->model = new \app\admin\model\shopro\order\Aftersale;
  28. $this->orderModel = new \app\admin\model\shopro\order\Order;
  29. $this->view->assign("dispatchStatusList", $this->model->getDispatchStatusList());
  30. $this->view->assign("aftersaleStatusList", $this->model->getAftersaleStatusList());
  31. $this->view->assign("refundStatusList", $this->model->getRefundStatusList());
  32. }
  33. /**
  34. * 默认生成的控制器所继承的父类中有index/add/edit/del/multi五个基础方法、destroy/restore/recyclebin三个回收站方法
  35. * 因此在当前控制器中可不用编写增删改查的代码,除非需要自己控制这部分逻辑
  36. * 需要将application/admin/library/traits/Backend.php中对应的方法复制到当前控制器,然后进行修改
  37. */
  38. /**
  39. * 查看
  40. */
  41. public function index()
  42. {
  43. //当前是否为关联查询
  44. $this->relationSearch = false;
  45. //设置过滤方法
  46. $this->request->filter(['strip_tags', 'trim']);
  47. if ($this->request->isAjax())
  48. {
  49. //如果发送的来源是Selectpage,则转发到Selectpage
  50. if ($this->request->request('keyField'))
  51. {
  52. return $this->selectpage();
  53. }
  54. // list($where, $sort, $order, $offset, $limit) = $this->buildparams();
  55. $sort = $this->request->get("sort", !empty($this->model) && $this->model->getPk() ? $this->model->getPk() : 'id');
  56. $order = $this->request->get("order", "DESC");
  57. $offset = $this->request->get("offset", 0);
  58. $limit = $this->request->get("limit", 0);
  59. $search = $this->request->get("search", ''); // 关键字
  60. $status = $this->request->get("status", 'all');
  61. $total = $this->buildSearchOrder() // 返回的是 orderModel
  62. ->order($sort, $order)
  63. ->count();
  64. $list = $this->buildSearchOrder() // 返回的是 orderModel
  65. ->with(['aftersale' => function ($query) {
  66. $query->removeOption('soft_delete')->alias('a')->field('a.*, u.nickname as user_nickname, u.mobile as user_mobile')->join('user u', 'u.id = a.user_id', 'LEFT');
  67. }])
  68. ->order($sort, $order)
  69. ->limit($offset, $limit)
  70. ->select();
  71. // foreach ($list as $row) {
  72. // $row->visible(['id','aftersale_sn','user_id','type','order_id','order_item_id','goods_id','goods_sku_price_id','goods_sku_text','goods_title','goods_image','goods_original_price','discount_fee','goods_price','goods_num','dispatch_status','dispatch_fee','aftersale_status','refund_status','refund_fee','createtime','updatetime','deletetime', 'user', 'activity_type']);
  73. // }
  74. $list = collection($list)->toArray();
  75. $result = array("total" => $total, "rows" => $list);
  76. return $this->success('获取成功', null, $result);
  77. }
  78. return $this->view->fetch();
  79. }
  80. /**
  81. * 详情
  82. */
  83. public function detail($id)
  84. {
  85. if ($this->request->isAjax()) {
  86. $row = $this->model->withTrashed()->with(['user', 'order' => function($query) {
  87. $query->removeOption('soft_delete');
  88. }, 'logs'])->where('id', $id)->find();
  89. if (!$row) {
  90. $this->error(__('No Results were found'));
  91. }
  92. $result = [
  93. 'detail' => $row,
  94. 'logs' => $row->logs,
  95. 'order' => $row->order,
  96. ];
  97. return $this->success('获取成功', null, $result);
  98. }
  99. $this->assignconfig('id', $id);
  100. return $this->view->fetch();
  101. }
  102. /**
  103. * 完成售后
  104. */
  105. public function finish($id)
  106. {
  107. if ($this->request->isAjax()) {
  108. $aftersale = $this->model->withTrashed()->canOper()->where('id', $id)->find();
  109. if (!$aftersale) {
  110. $this->error('售后单不存在或不可完成');
  111. }
  112. $order = Order::withTrashed()->where('id', $aftersale->order_id)->find();
  113. $orderItem = OrderItem::where('id', $aftersale->order_item_id)->find();
  114. if (!$order || !$orderItem) {
  115. $this->error('订单或订单商品不存在');
  116. }
  117. Db::transaction(function () use ($aftersale, $order, $orderItem) {
  118. // 售后单完成前
  119. $data = ['aftersale' => $aftersale, 'order' => $order, 'item' => $orderItem];
  120. \think\Hook::listen('aftersale_finish_before', $data);
  121. $aftersale->aftersale_status = \app\admin\model\shopro\order\Aftersale::AFTERSALE_STATUS_OK; // 售后完成
  122. $aftersale->save();
  123. // 增加售后单变动记录、
  124. OrderAftersaleLog::operAdd($order, $aftersale, $this->auth->getUserInfo(), 'admin', [
  125. 'reason' => '卖家完成售后',
  126. 'content' => '售后订单已完成',
  127. 'images' => []
  128. ]);
  129. $orderItem->aftersale_status = OrderItem::AFTERSALE_STATUS_OK;
  130. $orderItem->save();
  131. \addons\shopro\model\OrderAction::operAdd($order, $orderItem, $this->auth->getUserInfo(), 'admin', '管理员完成售后');
  132. // 售后单完成之后
  133. $data = ['aftersale' => $aftersale, 'order' => $order, 'item' => $orderItem];
  134. \think\Hook::listen('aftersale_finish_after', $data);
  135. });
  136. return $this->success('操作成功');
  137. }
  138. }
  139. /**
  140. * 拒绝售后
  141. */
  142. public function refuse($id = 0)
  143. {
  144. if ($this->request->isAjax()) {
  145. $refuse_msg = $this->request->post('refuse_msg', '');
  146. if (!$refuse_msg) {
  147. $this->error('请输入拒绝原因');
  148. }
  149. $aftersale = $this->model->withTrashed()->canOper()->where('id', $id)->find();
  150. if (!$aftersale) {
  151. $this->error('售后单不存在或不可拒绝');
  152. }
  153. $order = Order::withTrashed()->where('id', $aftersale->order_id)->find();
  154. $orderItem = OrderItem::where('id', $aftersale->order_item_id)->find();
  155. if (!$order || !$orderItem) {
  156. $this->error('订单或订单商品不存在');
  157. }
  158. Db::transaction(function () use ($aftersale, $order, $orderItem, $refuse_msg) {
  159. // 售后单拒绝前
  160. $data = ['aftersale' => $aftersale, 'order' => $order, 'item' => $orderItem];
  161. \think\Hook::listen('aftersale_refuse_before', $data);
  162. $aftersale->aftersale_status = \app\admin\model\shopro\order\Aftersale::AFTERSALE_STATUS_REFUSE; // 售后拒绝
  163. $aftersale->save();
  164. // 增加售后单变动记录
  165. OrderAftersaleLog::operAdd($order, $aftersale, $this->auth->getUserInfo(), 'admin', [
  166. 'reason' => '卖家拒绝售后',
  167. 'content' => $refuse_msg,
  168. 'images' => []
  169. ]);
  170. $orderItem->aftersale_status = OrderItem::AFTERSALE_STATUS_REFUSE; // 拒绝售后
  171. $orderItem->save();
  172. \addons\shopro\model\OrderAction::operAdd($order, $orderItem, $this->auth->getUserInfo(), 'admin', '管理员拒绝订单售后:' . $refuse_msg);
  173. // 售后单拒绝后
  174. $data = ['aftersale' => $aftersale, 'order' => $order, 'item' => $orderItem];
  175. \think\Hook::listen('aftersale_refuse_after', $data);
  176. });
  177. return $this->success('操作成功');
  178. }
  179. }
  180. /**
  181. * 同意退款
  182. */
  183. public function refund($id = 0)
  184. {
  185. if ($this->request->isAjax()) {
  186. $refund_money = round($this->request->post('refund_money', 0), 2);
  187. if ($refund_money < 0) {
  188. $this->error('退款金额不能小于 0');
  189. }
  190. $aftersale = $this->model->withTrashed()->canOper()->where('id', $id)->find();
  191. if (!$aftersale) {
  192. $this->error('售后单不存在或不可退款');
  193. }
  194. $order = Order::withTrashed()->with('item')->where('id', $aftersale->order_id)->find();
  195. if (!$order) {
  196. $this->error('订单不存在');
  197. }
  198. $items = $order->item;
  199. $items = array_column($items, null, 'id');
  200. // 当前订单已退款总金额
  201. $refunded_money = array_sum(array_column($items, 'refund_fee'));
  202. // 剩余可退款金额
  203. $refund_surplus_money = $order->pay_fee - $refunded_money;
  204. // 如果退款金额大于订单支付总金额
  205. if ($refund_money > $refund_surplus_money) {
  206. $this->error('退款总金额不能大于实际支付金额');
  207. }
  208. $orderItem = $items[$aftersale['order_item_id']];
  209. if (!$orderItem || in_array($orderItem['refund_status'], [
  210. \app\admin\model\shopro\order\OrderItem::REFUND_STATUS_OK,
  211. \app\admin\model\shopro\order\OrderItem::REFUND_STATUS_FINISH,
  212. ])) {
  213. $this->error('订单商品已退款,不能重复退款');
  214. }
  215. Db::transaction(function () use ($aftersale, $order, $orderItem, $refund_money, $refund_surplus_money) {
  216. $data = ['aftersale' => $aftersale, 'order' => $order, 'item' => $orderItem];
  217. \think\Hook::listen('aftersale_finish_before', $data);
  218. $aftersale->aftersale_status = \app\admin\model\shopro\order\Aftersale::AFTERSALE_STATUS_OK; // 售后同意
  219. $aftersale->refund_status = \app\admin\model\shopro\order\Aftersale::REFUND_STATUS_FINISH; // 售后同意退款
  220. $aftersale->refund_fee = $refund_money; // 退款金额
  221. $aftersale->save();
  222. // 增加售后单变动记录
  223. OrderAftersaleLog::operAdd($order, $aftersale, $this->auth->getUserInfo(), 'admin', [
  224. 'reason' => '卖家同意退款',
  225. 'content' => '售后订单已退款',
  226. 'images' => []
  227. ]);
  228. $orderItem->aftersale_status = OrderItem::AFTERSALE_STATUS_OK;
  229. $orderItem->save();
  230. \addons\shopro\model\OrderAction::operAdd($order, $orderItem, $this->auth->getUserInfo(), 'admin', '管理员同意售后退款');
  231. // 退款
  232. \app\admin\model\shopro\order\Order::startRefund($order, $orderItem, $refund_money, $this->auth->getUserInfo(), '管理员同意售后退款');
  233. $data = ['aftersale' => $aftersale, 'order' => $order, 'item' => $orderItem];
  234. \think\Hook::listen('aftersale_finish_after', $data);
  235. });
  236. return $this->success('操作成功');
  237. }
  238. }
  239. /**
  240. * 留言
  241. */
  242. public function addLog($id = 0)
  243. {
  244. if ($this->request->isAjax()) {
  245. $reason = $this->request->post('reason', '卖家留言');
  246. $content = $this->request->post('content', '');
  247. $images = $this->request->post('images', []);
  248. if (!$content) {
  249. $this->error('留言内容不能为空');
  250. }
  251. $aftersale = $this->model->withTrashed()->where('id', $id)->find();
  252. if (!$aftersale) {
  253. $this->error('售后单不存在');
  254. }
  255. $order = Order::withTrashed()->with('item')->where('id', $aftersale->order_id)->find();
  256. if (!$order) {
  257. $this->error('订单不存在');
  258. }
  259. Db::transaction(function () use ($order, $aftersale, $reason, $content, $images) {
  260. if ($aftersale['aftersale_status'] == 0) {
  261. $aftersale->aftersale_status = \app\admin\model\shopro\order\Aftersale::AFTERSALE_STATUS_AFTERING; // 售后处理中
  262. $aftersale->save();
  263. }
  264. // 增加售后单变动记录
  265. OrderAftersaleLog::operAdd($order, $aftersale, $this->auth->getUserInfo(), 'admin', [
  266. 'reason' => $reason,
  267. 'content' => $content,
  268. 'images' => $images
  269. ]);
  270. });
  271. return $this->success('操作成功');
  272. }
  273. }
  274. private function buildSearchOrder() {
  275. $search = $this->request->get("search", ''); // 关键字
  276. $status = $this->request->get("status", 'all');
  277. $orders = $this->orderModel->withTrashed();
  278. $orders = $orders->whereExists(function ($query) use ($search, $status) {
  279. extract($this->getModelTable());
  280. $aftersales = $query->table($aftersaleName)->where($aftersaleName . '.order_id=' . $orderName . '.id');
  281. $aftersales = $this->aftersaleSearch($aftersales, $search, $status);
  282. return $aftersales;
  283. });
  284. return $orders;
  285. }
  286. private function aftersaleSearch($aftersales, $search, $status) {
  287. extract($this->getModelTable());
  288. if ($search) {
  289. // 模糊搜索字段
  290. $searcharr = ['goods_title', 'goods_sku_text', 'aftersale_sn'];
  291. foreach ($searcharr as $k => &$v) {
  292. $v = stripos($v, ".") === false ? $aftersaleName . '.' . $v : $v;
  293. }
  294. unset($v);
  295. $aftersales = $aftersales->where(function ($query) use ($searcharr, $search, $aftersaleName) {
  296. $query->where(implode("|", $searcharr), "LIKE", "%{$search}%")
  297. ->whereOr(function ($query) use ($search, $aftersaleName) { // 用户
  298. $query->whereExists(function ($query) use ($search, $aftersaleName) {
  299. $userTableName = (new \app\admin\model\User())->getQuery()->getTable();
  300. $query->table($userTableName)->where($userTableName . '.id=' . $aftersaleName . '.user_id')
  301. ->where(function ($query) use ($search) {
  302. $query->where('nickname', 'like', "%{$search}%")
  303. ->whereOr('mobile', 'like', "%{$search}%");
  304. });
  305. });
  306. });
  307. });
  308. }
  309. // 售后单状态
  310. if ($status != 'all') {
  311. if (in_array($status, ['cancel', 'refuse', 'nooper', 'ing', 'finish'])) {
  312. $aftersales = $aftersales->where(OrderAftersale::getScopeWhere($status));
  313. }
  314. }
  315. return $aftersales;
  316. }
  317. private function getModelTable () {
  318. $orderName = $this->orderModel->getQuery()->getTable();
  319. $aftersaleName = $this->model->getQuery()->getTable();
  320. return compact("orderName", "aftersaleName");
  321. }
  322. }