截流自动化的商城平台
Du kannst nicht mehr als 25 Themen auswählen Themen müssen mit entweder einem Buchstaben oder einer Ziffer beginnen. Sie können Bindestriche („-“) enthalten und bis zu 35 Zeichen lang sein.

AfterSaleLogic.php 20KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535
  1. <?php
  2. namespace app\api\logic;
  3. use app\api\controller\Shop;
  4. use app\common\enum\NoticeEnum;
  5. use app\common\enum\PayEnum;
  6. use app\common\logic\AfterSaleLogLogic;
  7. use app\common\basics\Logic;
  8. use app\common\model\after_sale\{AfterSale, AfterSaleLog};
  9. use app\common\model\NoticeSetting;
  10. use app\common\model\order\OrderGoods;
  11. use app\common\model\order\Order;
  12. use app\common\enum\OrderGoodsEnum;
  13. use app\common\model\shop\Shop as ShopModel;
  14. use app\common\server\AreaServer;
  15. use app\common\server\ConfigServer;
  16. use app\common\server\UrlServer;
  17. use think\facade\Db;
  18. use think\Exception;
  19. use app\common\enum\ShopEnum;
  20. use think\Model;
  21. /**
  22. * 售后
  23. * Class AfterSaleLogic
  24. * @package app\api\logic
  25. */
  26. class AfterSaleLogic extends Logic
  27. {
  28. /**
  29. * @notes 售后退款列表
  30. * @param $user_id
  31. * @param $type
  32. * @param $page
  33. * @param $size
  34. * @return array
  35. * @throws \think\db\exception\DataNotFoundException
  36. * @throws \think\db\exception\DbException
  37. * @throws \think\db\exception\ModelNotFoundException
  38. * @throws \think\exception\DbException
  39. * @author suny
  40. * @date 2021/7/13 6:06 下午
  41. */
  42. public static function lists($user_id, $type, $page, $size)
  43. {
  44. $where = [];
  45. $where[] = ['o.user_id', '=', $user_id];
  46. $data = $result = [];
  47. switch ($type) {
  48. case 'normal':
  49. $where[] = ['g.refund_status', '=', OrderGoodsEnum::REFUND_STATUS_NO];
  50. $where[] = ['o.order_status', 'in', [Order::STATUS_WAIT_RECEIVE, Order::STATUS_FINISH]];
  51. $order = new Order();
  52. $count = $order->alias('o')
  53. ->field('o.id,o.confirm_take_time,o.order_status,o.create_time,s.id as sid,s.name as shop_name,s.type as shop_type')
  54. ->join('order_goods g', 'g.order_id = o.id')
  55. ->join('shop s', 'o.shop_id = s.id')
  56. ->with(['order_goods' => function ($query) {
  57. $query->where('refund_status', OrderGoodsEnum::REFUND_STATUS_NO);
  58. }])
  59. ->where($where)
  60. ->group('o.id')
  61. ->count();
  62. $lists = $order
  63. ->alias('o')
  64. ->field('o.id,o.confirm_take_time,o.order_status,o.order_sn,o.create_time,s.id as sid,s.name as shop_name,s.logo as shop_logo,s.type as shop_type')
  65. ->join('order_goods g', 'g.order_id = o.id')
  66. ->join('shop s', 'o.shop_id = s.id')
  67. ->with(['order_goods' => function ($query) {
  68. $query->where('refund_status', OrderGoodsEnum::REFUND_STATUS_NO);
  69. }])
  70. ->where($where)
  71. ->group('o.id')
  72. ->order('o.id desc')
  73. ->page($page, $size)
  74. ->select()->toArray();
  75. // 获取所有用户的订单售后的退款编号申请时间
  76. $order_ids = array_column($lists, 'id');
  77. $after_sale_lists = AfterSale::where(['order_id' => $order_ids])->column('sn, create_time','order_id');
  78. foreach ($lists as $item) {
  79. $result = [
  80. 'order_id' => $item['id'],
  81. ];
  82. $order_goods = [];
  83. foreach ($item['order_goods'] as $k1 => $good) {
  84. $goods = [
  85. 'goods_id' => $good['goods_id'],
  86. 'item_id' => $good['item_id'],
  87. 'goods_name' => '',
  88. 'goods_num' => $good['goods_num'],
  89. 'goods_price' => $good['goods_price'],
  90. 'image' => '',
  91. ];
  92. $goods_data = $good;
  93. $goods['spec_value_str'] = $goods_data['spec_value'];
  94. $goods['goods_name'] = $goods_data['goods_name'];
  95. $goods['image'] = empty($goods_data['spec_image']) ? UrlServer::getFileUrl($goods_data['image']) : UrlServer::getFileUrl($goods_data['spec_image']);
  96. $order_goods[] = $goods;
  97. }
  98. $result['sid'] = $item['sid'];
  99. $result['shop_name'] = $item['shop_name'];
  100. $result['shop_type'] = ShopEnum::getShopTypeDesc($item['shop_type']);
  101. $result['shop_logo'] = UrlServer::getFileUrl($item['shop_logo']);
  102. $result['order_goods'] = $order_goods;
  103. $result['order_sn'] = $item['order_sn'];
  104. $result['create_time'] = $item['create_time'];
  105. $result['after_sale']['desc'] = '';
  106. $result['after_sale']['able_apply'] = 1;
  107. $result['after_sale']['sn'] = '';
  108. $result['after_sale']['create_time'] = '';
  109. if(isset($after_sale_lists[$item['id']]) && $after_sale_lists[$item['id']]){
  110. $after_sale_info = $after_sale_lists[$item['id']];
  111. $result['after_sale']['sn'] = $after_sale_info['sn'];
  112. $result['after_sale']['create_time'] = date('Y-m-d H:i:s', $after_sale_info['create_time']);
  113. }
  114. if (self::checkAfterSaleDate($item) === false) {
  115. $result['after_sale']['desc'] = '该商品已超过售后期';
  116. $result['after_sale']['able_apply'] = 0;
  117. }
  118. $time = $item['confirm_take_time'] ?? $item['create_time'];
  119. $result['time'] = $time;
  120. $data[] = $result;
  121. }
  122. $list = ['list' => $data, 'page' => $page, 'size' => $size, 'count' => $count, 'more' => is_more($count, $page, $size)];
  123. return $list;
  124. break;
  125. case 'apply':
  126. $where[] = ['g.refund_status', 'in', [OrderGoodsEnum::REFUND_STATUS_APPLY, OrderGoodsEnum::REFUND_STATUS_WAIT]];
  127. $where[] = ['o.order_status', 'in', [Order::STATUS_WAIT_RECEIVE, Order::STATUS_FINISH]];
  128. $where[] = ['a.del', '=', 0];
  129. break;
  130. case 'finish':
  131. $where[] = ['g.refund_status', '=', OrderGoodsEnum::REFUND_STATUS_SUCCESS];
  132. $where[] = ['a.del', '=', 0];
  133. $where[] = ['o.order_status', 'in', [Order::STATUS_WAIT_RECEIVE, Order::STATUS_FINISH, Order::STATUS_CLOSE]];
  134. break;
  135. }
  136. $field = 'g.order_id,g.goods_id,g.item_id,g.goods_num,g.goods_name,g.image,g.spec_value,g.goods_price,a.sn,a.status,a.refund_type,a.id as after_sale_id,a.create_time,s.id as sid,s.name as shop_name,s.logo as shop_logo,s.type as shop_type';
  137. $count = Db::name('order_goods')
  138. ->alias("g")
  139. ->field($field)
  140. ->join('order o', 'g.order_id = o.id')
  141. ->join('after_sale a', 'a.order_goods_id = g.id', 'left')
  142. ->join('shop s', 'o.shop_id = s.id')
  143. ->where($where)
  144. ->count();
  145. $lists = Db::name('order_goods')
  146. ->alias("g")
  147. ->field($field)
  148. ->join('order o', 'g.order_id = o.id')
  149. ->join('after_sale a', 'a.order_goods_id = g.id', 'left')
  150. ->join('shop s', 'o.shop_id = s.id')
  151. ->where($where)
  152. ->order('a.id desc')
  153. ->page($page, $size)
  154. ->select();
  155. foreach ($lists as $k => $item) {
  156. $goods_data = $item;
  157. $goods_name = $goods_data['goods_name'];
  158. $image = empty($goods_data['spec_image']) ? UrlServer::getFileUrl($goods_data['image']) : UrlServer::getFileUrl($goods_data['spec_image']);
  159. $result = [
  160. 'order_id' => $item['order_id'],
  161. 'order_goods' => [[
  162. 'goods_id' => $item['goods_id'],
  163. 'item_id' => $item['item_id'],
  164. 'goods_name' => $goods_name,
  165. 'goods_num' => $item['goods_num'],
  166. 'goods_price' => $item['goods_price'],
  167. 'image' => $image,
  168. 'spec_value_str' => $goods_data['spec_value'],
  169. 'create_time' => $goods_data['create_time'],
  170. ]],
  171. 'after_sale' => [
  172. 'after_sale_id' => $item['after_sale_id'],
  173. 'status' => $item['status'],
  174. 'refund_type' => $item['refund_type'],
  175. 'status_text' => AfterSale::getStatusDesc($item['status']),
  176. 'type_text' => AfterSale::getRefundTypeDesc($item['refund_type']),
  177. 'desc' => AfterSale::getStatusDesc($item['status']),
  178. 'able_apply' => 1,
  179. 'sn' => $item['sn'],
  180. 'create_time' => date('Y-m-d H:i:s', $item['create_time']),
  181. ],
  182. 'time' => date('Y-m-d H:i', $item['create_time']),
  183. ];
  184. $result['sid'] = $item['sid'];
  185. $result['shop_name'] = $item['shop_name'];
  186. $result['shop_type'] = ShopEnum::getShopTypeDesc($item['shop_type']);
  187. $result['shop_logo'] = UrlServer::getFileUrl($item['shop_logo']);
  188. $data[] = $result;
  189. }
  190. $list = [
  191. 'list' => $data,
  192. 'page' => $page,
  193. 'size' => $size,
  194. 'count' => $count,
  195. 'more' => is_more($count, $page, $size)
  196. ];
  197. return $list;
  198. }
  199. /**
  200. * @notes 申请售后
  201. * @param $post
  202. * @param $user_id
  203. * @return array
  204. * @throws Exception
  205. * @throws \think\exception\PDOException
  206. * @author suny
  207. * @date 2021/7/13 6:07 下午
  208. * 验证(收货后多少天内才可申请售后/或已发货,未收货). 售后日志
  209. */
  210. public static function add($post, $user_id)
  211. {
  212. Db::startTrans();
  213. try {
  214. //1,增加售后记录
  215. $order_goods = Db::name('order_goods')
  216. ->alias('g')
  217. ->field('g.id,g.goods_num,g.total_pay_price,g.order_id,g.refund_status,g.goods_id')
  218. ->join('order o', 'o.id = g.order_id')
  219. ->where(['order_id' => $post['order_id'], 'g.item_id' => $post['item_id']])
  220. ->find();
  221. $data = [
  222. 'sn' => createSn('after_sale', 'sn', '', 4),
  223. 'user_id' => $user_id,
  224. 'order_id' => $order_goods['order_id'],
  225. 'order_goods_id' => $order_goods['id'],
  226. 'item_id' => $post['item_id'],
  227. 'goods_id' => $order_goods['goods_id'],
  228. 'goods_num' => $order_goods['goods_num'],
  229. 'refund_reason' => trim($post['reason']),
  230. 'refund_remark' => isset($post['remark']) ? trim($post['remark']) : '',
  231. 'refund_image' => isset($post['img']) ? $post['img'] : '',
  232. 'refund_type' => $post['refund_type'],
  233. 'refund_price' => $order_goods['total_pay_price'],
  234. 'create_time' => time(),
  235. ];
  236. $after_sale_id = Db::name('after_sale')->insertGetId($data);
  237. //2,更改订单商品,退款状态为申请退款
  238. Db::name('order_goods')
  239. ->where('id', $order_goods['id'])
  240. ->update(['refund_status' => OrderGoodsEnum::REFUND_STATUS_APPLY]);
  241. //记录日志
  242. AfterSaleLogLogic::record(
  243. AfterSaleLog::TYPE_USER,
  244. AfterSaleLog::USER_APPLY_REFUND,
  245. $post['order_id'],
  246. $after_sale_id,
  247. $user_id,
  248. AfterSaleLog::USER_APPLY_REFUND
  249. );
  250. $order = Order::with('shop')->where(['id' => $post['order_id']])->find();
  251. if (!empty($order['shop']['mobile'])) {
  252. //通知商家
  253. event('Notice', [
  254. 'scene' => NoticeEnum::AFTER_SALE_NOTICE_SHOP,
  255. 'mobile' => $order['shop']['mobile'],
  256. 'params' => [
  257. 'user_id' => $user_id,
  258. 'after_sale_sn' => $data['sn']
  259. ]
  260. ]);
  261. }
  262. Db::commit();
  263. return ['after_sale_id' => $after_sale_id];
  264. } catch (Exception $e) {
  265. Db::rollback();
  266. throw new Exception($e->getMessage());
  267. }
  268. }
  269. /**
  270. * @notes 售后商品信息
  271. * @param $item_id
  272. * @param $order_id
  273. * @return array
  274. * @throws \think\db\exception\DataNotFoundException
  275. * @throws \think\db\exception\DbException
  276. * @throws \think\db\exception\ModelNotFoundException
  277. * @author suny
  278. * @date 2021/7/13 6:08 下午
  279. */
  280. public static function info($item_id, $order_id)
  281. {
  282. $goods = new OrderGoods();
  283. $where = ['order_id' => $order_id, 'item_id' => $item_id];
  284. $info = $goods->where($where)->find()->toArray();
  285. $goods['image'] = isset($info['image']) ? UrlServer::getFileUrl($info['image']) : '';
  286. $data = [
  287. 'goods' => $info,
  288. 'reason' => AfterSale::getReasonLists(),
  289. ];
  290. return $data;
  291. }
  292. /**
  293. * @notes 上传退货快递信息
  294. * @param $user_id
  295. * @param $post
  296. * @return bool
  297. * @throws \think\db\exception\DataNotFoundException
  298. * @throws \think\db\exception\DbException
  299. * @throws \think\db\exception\ModelNotFoundException
  300. * @author suny
  301. * @date 2021/7/13 6:08 下午
  302. */
  303. public static function express($user_id, $post)
  304. {
  305. $id = $post['id'];
  306. $after_sale = AfterSale::find($id);
  307. $after_sale->express_name = $post['express_name'];
  308. $after_sale->invoice_no = $post['invoice_no'];
  309. $after_sale->express_remark = isset($post['express_remark']) ? trim($post['express_remark']) : null;
  310. $after_sale->express_image = isset($post['express_image']) ? $post['express_image'] : null;
  311. $after_sale->status = AfterSale::STATUS_WAIT_RECEIVE_GOODS;//售后状态
  312. $result = $after_sale->save();
  313. //记录日志
  314. AfterSaleLogLogic::record(
  315. AfterSaleLog::TYPE_USER,
  316. AfterSaleLog::USER_SEND_EXPRESS,
  317. $after_sale['order_id'],
  318. $id,
  319. $user_id,
  320. AfterSaleLog::USER_SEND_EXPRESS
  321. );
  322. return $result;
  323. }
  324. /**
  325. * @notes 撤销申请
  326. * @param $user_id
  327. * @param $post
  328. * @throws \think\db\exception\DataNotFoundException
  329. * @throws \think\db\exception\DbException
  330. * @throws \think\db\exception\ModelNotFoundException
  331. * @author suny
  332. * @date 2021/7/13 6:08 下午
  333. */
  334. public static function cancel($user_id, $post)
  335. {
  336. $id = $post['id'];
  337. $after_sale = AfterSale::find($id);
  338. $after_sale->del = 1;
  339. $after_sale->update_time = time();
  340. $after_sale->save();
  341. //2,更改订单商品,退款状态为申请退款
  342. $order_goods = OrderGoods::find(['id' => $after_sale['order_goods_id']]);
  343. $order_goods->refund_status = OrderGoodsEnum::REFUND_STATUS_NO;
  344. $order_goods->save();
  345. //记录日志
  346. AfterSaleLogLogic::record(
  347. AfterSaleLog::TYPE_USER,
  348. AfterSaleLog::USER_CANCEL_REFUND,
  349. $after_sale['order_id'],
  350. $id,
  351. $user_id,
  352. AfterSaleLog::USER_CANCEL_REFUND
  353. );
  354. }
  355. /**
  356. * @notes 售后详情
  357. * @param $get
  358. * @return array
  359. * @throws \think\db\exception\DataNotFoundException
  360. * @throws \think\db\exception\DbException
  361. * @throws \think\db\exception\ModelNotFoundException
  362. * @author suny
  363. * @date 2021/7/13 6:08 下午
  364. */
  365. public static function detail($get)
  366. {
  367. $after_sale = new AfterSale();
  368. $field = 'a.id,a.sn,a.order_goods_id,a.refund_reason,a.refund_image,a.refund_type,a.refund_price,a.refund_remark,a.create_time,a.status,o.shop_id,s.id sid,s.name shop_name,s.type shop_type';
  369. $detail = $after_sale
  370. ->alias('a')
  371. ->field($field)
  372. ->join('order o', 'o.id = a.order_id')
  373. ->join('shop s', 's.id = o.shop_id')
  374. ->with(['order_goods'])
  375. ->where(['a.id' => $get['id'], 'a.del' => 0])
  376. ->find()
  377. ->toArray();
  378. $detail['shop_type'] = ShopEnum::getShopTypeDesc($detail['shop_type']);
  379. if (!$detail) {
  380. return [];
  381. }
  382. $detail['refund_image'] = empty($detail['refund_image']) ? '' : UrlServer::getFileUrl($detail['refund_image']);
  383. $detail['status_text'] = AfterSale::getStatusDesc($detail['status']);
  384. $detail['order_goods'] = $detail['order_goods'][0];
  385. $goods = $detail['order_goods'];
  386. $image = $goods['image'];
  387. $detail['order_goods']['image'] = empty($goods['spec_image']) ? $image : $goods['spec_image'];
  388. $detail['order_goods']['goods_name'] = $goods['goods_name'];
  389. $detail['order_goods']['spec_value'] = $goods['spec_value'];
  390. $detail['refund_type_text'] = AfterSale::getRefundTypeDesc($detail['refund_type']);
  391. $shop_id = $detail['shop_id'];
  392. $shop_info = ShopModel::where('id', $shop_id)->find();
  393. $refund_address = $shop_info['refund_address'];
  394. $shop_province = $refund_address['province_id'] ?? ''; //省份
  395. $shop_city = $refund_address['city_id'] ?? ''; //城市
  396. $shop_district = $refund_address['district_id'] ?? ''; //县区
  397. $shop_address = $refund_address['address'] ?? ''; //详细地址
  398. if (empty($shop_province) || empty($shop_city) || empty($shop_district)) {
  399. $arr = [];
  400. } else {
  401. $arr = [$shop_province, $shop_city, $shop_district];
  402. }
  403. $address = AreaServer::getAddress($arr, $shop_address);
  404. $shop = [
  405. 'contact' => $refund_address['nickname'] ?? '',
  406. 'mobile' => $refund_address['mobile'] ?? '',
  407. 'address' => $address
  408. ];
  409. $detail['shop'] = $shop;
  410. return $detail;
  411. }
  412. /**
  413. * @notes 重新申请
  414. * @param $user_id
  415. * @param $post
  416. * @return array
  417. * @throws Exception
  418. * @throws \think\exception\PDOException
  419. * @author suny
  420. * @date 2021/7/13 6:08 下午
  421. */
  422. public static function again($user_id, $post)
  423. {
  424. Db::startTrans();
  425. try {
  426. $id = $post['id'];
  427. $after_sale = AfterSale::find($id);
  428. $after_sale->refund_type = $post['refund_type'];
  429. $after_sale->refund_reason = trim($post['reason']);
  430. $after_sale->refund_remark = isset($post['remark']) ? trim($post['remark']) : '';
  431. $after_sale->refund_image = isset($post['img']) ? $post['img'] : '';
  432. $after_sale->status = AfterSale::STATUS_APPLY_REFUND;
  433. $after_sale_result = $after_sale->save();
  434. //2,更改订单商品,退款状态为申请退款
  435. $order_goods = OrderGoods::find(['id' => $after_sale['order_goods_id']]);
  436. $order_goods->refund_status = OrderGoodsEnum::REFUND_STATUS_APPLY;
  437. $order_goods_result = $order_goods->save();
  438. //记录日志
  439. AfterSaleLogLogic::record(
  440. AfterSaleLog::TYPE_USER,
  441. AfterSaleLog::USER_AGAIN_REFUND,
  442. $after_sale['order_id'],
  443. $id,
  444. $user_id,
  445. AfterSaleLog::USER_AGAIN_REFUND
  446. );
  447. Db::commit();
  448. return ['after_sale_id' => $id];
  449. } catch (Exception $e) {
  450. Db::rollback();
  451. throw new Exception($e->getMessage());
  452. }
  453. }
  454. /**
  455. * @notes 检查是否在售后时间内
  456. * @param $order
  457. * @return bool
  458. * @author suny
  459. * @date 2021/7/13 6:08 下午
  460. */
  461. public static function checkAfterSaleDate($order)
  462. {
  463. $now = time();
  464. $refund_days = ConfigServer::get('transaction', 'order_after_sale_days', 7);
  465. if ($refund_days == 0) {
  466. return true;
  467. }
  468. if ($order['order_status'] == Order::STATUS_FINISH) {
  469. $check_time = strtotime('+' . $refund_days . 'day', $order['confirm_take_time']);
  470. if ($now > $check_time) {
  471. return false;
  472. }
  473. }
  474. return true;
  475. }
  476. }