OrderOper.php 40 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072
  1. <?php
  2. namespace app\api\model\order;
  3. use addons\shopro\exception\Exception;
  4. use addons\shopro\model\Cart;
  5. use addons\shopro\model\Config;
  6. use addons\shopro\model\Coupons;
  7. use addons\shopro\model\Share;
  8. use app\admin\model\Pingjia;
  9. use app\api\model\User;
  10. use app\api\model\GoodsComment;
  11. use app\api\model\order\Order;
  12. use addons\shopro\model\OrderAction;
  13. use addons\shopro\model\OrderItem;
  14. use addons\shopro\model\Store;
  15. use app\api\model\UserWalletLog;
  16. use addons\shopro\model\Verify;
  17. use app\admin\model\shopro\order\RefundLog;
  18. use think\Db;
  19. use think\Log;
  20. trait OrderOper
  21. {
  22. use OrderOperSendGet, OrderOperCreate;
  23. // 获取订单号
  24. public static function getSn($user_id)
  25. {
  26. $rand = $user_id < 9999 ? mt_rand(100000, 99999999) : mt_rand(100, 99999);
  27. $order_sn = date('Yhis') . $rand;
  28. $id = str_pad($user_id, (24 - strlen($order_sn)), '0', STR_PAD_BOTH);
  29. return $order_sn . $id;
  30. }
  31. // 计算订单
  32. public static function pre($params, $calc_type = 'pre')
  33. {
  34. self::$calc_type = $calc_type;
  35. $user = User::info();
  36. // 检测必要系统环境
  37. // checkEnv(['bcmath', 'queue']);
  38. // 获取请求参数 "order_type", "groupon_id", "buy_type"
  39. extract(self::preParams($params));
  40. // 检测并重新组装要购买的商品列表
  41. list(
  42. $new_goods_list, // 组合好的新的商品列表
  43. $activity_type, // 订单参与的活动类型,后面还需计算一次
  44. $activity_discounts, // 订单可能参与的所有活动
  45. $need_address, // 是否需要收货地址
  46. $user_address // 用户收货地址
  47. ) = self::preCheck($params);
  48. // 计算订单商品价格,所需积分,运费
  49. list(
  50. $new_goods_list, // 新的商品列表
  51. $goods_original_amount, // 商品原始总价
  52. $goods_amount, // 商品现在总价
  53. $dispatch_amount, // 订单运费(所有商品种最高的,后面还需根据活动再次计算)
  54. $score_amount, // 订单所需支付积分
  55. ) = self::preCalcAmount($params, $new_goods_list, $user_address);
  56. // 记录原始运费
  57. $origin_dispatch_amount = $dispatch_amount;
  58. // 计算订单优惠折扣,优惠券,邮费
  59. list(
  60. $new_goods_list, // 重新赋值活动, 商品上增加了 activity_type
  61. $activity_discount_infos, // 每个活动包含的优惠信息
  62. $activity_discount_money, // 促销活动优惠总金额
  63. $dispatch_discount_money, // 邮费总优惠金额
  64. $free_shipping_goods_ids, // 包邮的商品的ids
  65. $activity_type, // 全部参与的活动类型
  66. $dispatch_amount, // 重新计算的运费
  67. $user_coupons, // 使用的优惠券信息
  68. $coupon_money // 优惠券优惠金额
  69. ) = self::preCalcDiscount(
  70. $params,
  71. $new_goods_list,
  72. $activity_discounts,
  73. $activity_type,
  74. $goods_amount,
  75. $dispatch_amount,
  76. $user_address
  77. );
  78. // 计算订单总金额,需支付金额
  79. list(
  80. $new_goods_list,
  81. $total_amount,
  82. $discount_fee,
  83. $total_fee,
  84. $coupon_fee,
  85. ) = self::preCalcOrder(
  86. $new_goods_list,
  87. $goods_amount,
  88. $origin_dispatch_amount,
  89. $dispatch_amount,
  90. $activity_discount_infos,
  91. $activity_discount_money,
  92. $dispatch_discount_money,
  93. $free_shipping_goods_ids,
  94. $coupon_money
  95. );
  96. // 获取发票金额
  97. $invoice_amount = self::preCalcInvoiceAmount($total_fee, $goods_amount);
  98. // 处理返回结果
  99. return self::preReturnParams(
  100. $goods_original_amount,
  101. $goods_amount,
  102. $origin_dispatch_amount,
  103. $dispatch_amount,
  104. $total_amount,
  105. $total_fee,
  106. $discount_fee,
  107. $coupon_fee,
  108. $activity_discount_money,
  109. $dispatch_discount_money,
  110. $activity_type,
  111. $score_amount,
  112. $new_goods_list,
  113. $need_address,
  114. $activity_discount_infos,
  115. $user_coupons,
  116. $user_address,
  117. $invoice_amount,
  118. $user->score//用户积分
  119. );
  120. }
  121. // 获取可用优惠券列表
  122. public static function coupons($params, $goods_amount = 0)
  123. {
  124. $user = User::info();
  125. extract($params);
  126. $order_type = $order_type ?? 'goods';
  127. $groupon_id = $groupon_id ?? 0; // 拼团的 团 id
  128. $buy_type = $buy_type ?? 'alone'; // 拼团的 购买方式: alone=单独购买,groupon=开团
  129. // 商品总金额
  130. if (!$goods_amount) {
  131. // 下单的时候把计算好的 goods_amount 传进来了,接口直接获取可用列表的时候,需要计算
  132. foreach ($goods_list as $k => $goods) {
  133. $goods_amount += ($goods['goods_price'] * $goods['goods_num']);
  134. }
  135. }
  136. $coupons = Coupons::getCouponsList(Coupons::COUPONS_CAN_USE);
  137. $new_coupons = [];
  138. // 过滤,如果有一个产品不适用,则优惠券不允许使用,不显示
  139. foreach ($coupons as $key => $coupon) {
  140. if ($coupon['goods_ids'] === '0') {
  141. // 所有商品可用
  142. $can_use = true;
  143. } else {
  144. $goods_ids = explode(',', $coupon['goods_ids']);
  145. $can_use = true;
  146. foreach ($goods_list as $k => $goods) {
  147. if (!in_array($goods['goods_id'], $goods_ids)) {
  148. $can_use = false;
  149. break;
  150. }
  151. }
  152. }
  153. // 商品可用 并且 商品金额满足
  154. if ($can_use && $coupon->enough <= $goods_amount) {
  155. $new_coupons[] = $coupon;
  156. }
  157. }
  158. $new_coupons = array_values($new_coupons);
  159. return $new_coupons;
  160. }
  161. public static function createOrder($params)
  162. {
  163. $user = User::info();
  164. // 入参
  165. extract($params);
  166. $remark = $remark ?? null;
  167. $order_type = $order_type ?? 'goods';
  168. $groupon_id = $groupon_id ?? 0; // 拼团的 团 id
  169. $buy_type = $buy_type ?? 'alone'; // 拼团的 购买方式: alone=单独购买,groupon=开团
  170. $invoice = $invoice ?? null; // 发票申请
  171. // $cart_id = $cart_id ?? null; //购物车id
  172. // if($cart_id && is_string($cart_id)){
  173. // $cart_id = [$cart_id];
  174. // }
  175. // 订单计算数据
  176. extract(self::pre($params, "create"));
  177. // 用户使用积分
  178. if ($use_score > 0) {
  179. $score_amount = $use_score;
  180. $total_fee = (($total_fee - $score_amount / config('site.score_cash_offset')) > 0 ?($total_fee - $score_amount / config('site.score_cash_offset')): 0);
  181. }
  182. $order = Db::transaction(function () use (
  183. $user,
  184. $order_type,
  185. $groupon_id,
  186. $buy_type,
  187. $goods_original_amount,
  188. $goods_amount,
  189. $dispatch_amount,
  190. $total_amount,
  191. $score_amount,
  192. $total_fee,
  193. $discount_fee,
  194. $coupon_fee,
  195. $activity_discount_money,
  196. $dispatch_discount_money,
  197. $activity_discount_infos,
  198. $new_goods_list,
  199. $activity_type,
  200. $user_coupons,
  201. $user_address,
  202. $remark,
  203. $from,
  204. $invoice,
  205. $invoice_amount
  206. ) {
  207. // 订单创建前
  208. $data = [
  209. 'user' => $user,
  210. 'order_type' => $order_type,
  211. 'groupon_id' => $groupon_id,
  212. 'buy_type' => $buy_type,
  213. 'goods_original_amount' => $goods_original_amount,
  214. 'goods_amount' => $goods_amount,
  215. 'dispatch_amount' => $dispatch_amount,
  216. 'total_amount' => $total_amount,
  217. 'score_amount' => $score_amount,
  218. 'total_fee' => $total_fee,
  219. 'discount_fee' => $discount_fee,
  220. 'coupon_fee' => $coupon_fee,
  221. 'activity_discount_money' => $activity_discount_money,
  222. 'dispatch_discount_money' => $dispatch_discount_money,
  223. 'activity_discount_infos' => $activity_discount_infos,
  224. 'new_goods_list' => $new_goods_list,
  225. 'activity_type' => $activity_type,
  226. 'user_coupons' => $user_coupons,
  227. 'user_address' => $user_address,
  228. 'remark' => $remark,
  229. 'from' => $from
  230. ];
  231. // 如果是活动,这里面减掉 redis 库存
  232. // todo del 1.4
  233. // \think\Hook::listen('order_create_before', $data);
  234. // 团信息, 如果是参与旧团拼团才会不为 null,(开新团放到支付成功)
  235. $groupon = cache('grouponinfo-' . $user->id);
  236. $groupon = $groupon ? json_decode($groupon, true) : null;
  237. // 立即清除缓存
  238. cache('grouponinfo-' . $user->id, NULL);
  239. $orderData = [];
  240. $orderData['order_sn'] = self::getSn($user->id);
  241. $orderData['user_id'] = $user->id;
  242. $orderData['type'] = $order_type;
  243. $orderData['activity_type'] = $activity_type;
  244. $orderData['goods_amount'] = $goods_amount;
  245. $orderData['dispatch_amount'] = $dispatch_amount;
  246. $orderData['total_amount'] = $total_amount;
  247. $orderData['score_amount'] = $score_amount;
  248. $orderData['total_fee'] = $total_fee;
  249. $orderData['discount_fee'] = $discount_fee;
  250. $orderData['score_fee'] = $score_amount; // 记录score 支付数
  251. $orderData['coupon_fee'] = $coupon_fee;
  252. $orderData['activity_discount_money'] = $activity_discount_money;
  253. $orderData['dispatch_discount_money'] = $dispatch_discount_money;
  254. $orderData['goods_original_amount'] = $goods_original_amount;
  255. if ($user_address) {
  256. $orderData['phone'] = $user_address->phone;
  257. $orderData['consignee'] = $user_address->name;
  258. $orderData['province_name'] = city_name($user_address->province_id);//$user_address->province_name;
  259. $orderData['city_name'] = city_name($user_address->city_id);//$user_address->city_name;
  260. $orderData['area_name'] = city_name($user_address->area_id);//$user_address->area_name;
  261. $orderData['address'] = $user_address->address;
  262. $orderData['province_id'] = $user_address->province_id;
  263. $orderData['city_id'] = $user_address->city_id;
  264. $orderData['area_id'] = $user_address->area_id;
  265. }
  266. // 处理发票申请
  267. if ($invoice_amount > 0) {
  268. if (!empty($invoice) && $invoice['amount'] == $invoice_amount) {
  269. $orderData['invoice_status'] = 1; // 已申请
  270. } else {
  271. $orderData['invoice_status'] = 0; // 未申请
  272. }
  273. } else {
  274. $orderData['invoice_status'] = -1; // 不可开具发票
  275. }
  276. $orderData['status'] = 0;
  277. $orderData['remark'] = $remark;
  278. $orderData['coupons_id'] = $user_coupons ? $user_coupons->id : 0;
  279. $orderData['platform'] = request()->header('platform');
  280. //$ext = $activity_discount_infos ? ['activity_discount_infos' => $activity_discount_infos] : []; // 促销活动信息
  281. $ext = [];
  282. $orderData['ext'] = json_encode($ext);
  283. $order = new Order();
  284. $order->allowField(true)->save($orderData);
  285. // 将优惠券使用掉
  286. if ($user_coupons) {
  287. $user_coupons->use_order_id = $order->id;
  288. $user_coupons->usetime = time();
  289. $user_coupons->save();
  290. }
  291. // 如果需要支付积分,扣除积分
  292. if ($score_amount) {
  293. // $user 为 Common\Auth 对象
  294. User::score(-$score_amount, $user->id, 'score_pay', $order->id, '', [
  295. 'order_id' => $order->id,
  296. 'order_sn' => $order->order_sn,
  297. ]);
  298. }
  299. // 添加发票数据
  300. if ($order->invoice_status == 1) {
  301. \addons\shopro\model\OrderInvoice::create([
  302. 'order_id' => $order->id,
  303. 'user_id' => $order->user_id,
  304. 'amount' => $invoice['amount'],
  305. 'type' => $invoice['type'],
  306. 'header_name' => $invoice['header_name'],
  307. 'tax_no' => $invoice['tax_no'],
  308. 'mobile' => $invoice['mobile'],
  309. ]);
  310. }
  311. // 添加 订单 item
  312. foreach ($new_goods_list as $key => $buyinfo) {
  313. $detail = $buyinfo['detail'];
  314. $current_sku_price = $detail['current_sku_price'];
  315. $orderItem = new OrderItem();
  316. $orderItem->user_id = $user->id;
  317. $orderItem->order_id = $order->id;
  318. $orderItem->goods_id = $buyinfo['goods_id'];
  319. $orderItem->goods_type = $detail['type'];
  320. $orderItem->goods_sku_price_id = $buyinfo['sku_price_id'];
  321. $item_activity_type = (isset($current_sku_price['activity_type']) && $current_sku_price['activity_type']) ? $current_sku_price['activity_type'] : '';
  322. $item_activity_type .= $buyinfo['activity_type'] ? ',' . $buyinfo['activity_type'] : '';
  323. $item_activity_type = trim($item_activity_type, ',');
  324. $orderItem->activity_id = $current_sku_price['activity_id'] ?? 0; // 商品当前的活动类型
  325. $orderItem->activity_type = $item_activity_type ?: null; // 商品当前的活动类型
  326. // 当前商品规格对应的 活动下对应商品规格的 id
  327. $orderItem->item_goods_sku_price_id = isset($current_sku_price['item_goods_sku_price']) ?
  328. $current_sku_price['item_goods_sku_price']['id'] : 0;
  329. $orderItem->goods_sku_text = $current_sku_price['goods_sku_text'];
  330. $orderItem->goods_title = $detail->title;
  331. $orderItem->goods_image = empty($current_sku_price['image']) ? $detail->image : $current_sku_price['image'];
  332. $orderItem->goods_original_price = $detail->original_price;
  333. $orderItem->discount_fee = $buyinfo['discount_fee']; // 平均计算单件商品所享受的折扣
  334. $orderItem->pay_price = $buyinfo['pay_price']; // 平均计算单件商品不算运费,算折扣时候的金额
  335. $orderItem->goods_price = $detail->current_sku_price->price;
  336. $orderItem->goods_num = $buyinfo['goods_num'] ?? 1;
  337. $orderItem->goods_weight = $detail->current_sku_price->weight;
  338. $orderItem->dispatch_status = 0;
  339. $orderItem->dispatch_fee = $buyinfo['dispatch_amount'];
  340. $orderItem->dispatch_type = $buyinfo['dispatch_type'];
  341. $orderItem->dispatch_id = $buyinfo['dispatch_id'] ? $buyinfo['dispatch_id'] : 0;
  342. $orderItem->store_id = $buyinfo['store_id'] ? $buyinfo['store_id'] : 0;
  343. $orderItem->aftersale_status = 0;
  344. $orderItem->comment_status = 0;
  345. $orderItem->refund_status = 0;
  346. $ext = [];
  347. if (isset($buyinfo['dispatch_date'])) {
  348. $ext['dispatch_date'] = $buyinfo['dispatch_date'];
  349. }
  350. if (isset($buyinfo['dispatch_phone'])) {
  351. $ext['dispatch_phone'] = $buyinfo['dispatch_phone'];
  352. }
  353. $orderItem->ext = json_encode($ext);
  354. $orderItem->save();
  355. }
  356. // 订单创建后
  357. $data = [
  358. 'user' => $user,
  359. 'order' => $order,
  360. 'from' => $from,
  361. 'groupon' => $groupon,
  362. 'buy_type' => $buy_type,
  363. 'activity_type' => $activity_type,
  364. 'new_goods_list' => $new_goods_list
  365. ];
  366. \think\Hook::listen('order_create_after', $data);
  367. // self::orderCreateAfter($data);
  368. //删除购物车商品
  369. // if($cart_id) \app\api\model\Cart::where('id', 'IN', $cart_id)->delete();
  370. // 重新获取订单
  371. $order = Order::where('id', $order->id)->find();
  372. return $order;
  373. });
  374. return $order;
  375. }
  376. // 订单创建后
  377. public static function orderCreateAfter(&$params)
  378. {
  379. Log::write("eeeeeeeeeeeeeeee");
  380. $user = $params['user'];
  381. $order = $params['order'];
  382. $from = $params['from'];
  383. $groupon = $params['groupon'];
  384. $buy_type = $params['buy_type'];
  385. $new_goods_list = $params['new_goods_list'];
  386. // 删除购物车
  387. if ($from == 'cart') {
  388. foreach ($new_goods_list as $delCart) {
  389. Cart::where([
  390. 'user_id' => $user->id,
  391. 'goods_id' => $delCart['goods_id'],
  392. 'sku_price_id' => $delCart['sku_price_id'],
  393. ])->delete();
  394. }
  395. }
  396. // 更新订单扩展字段
  397. $order_ext = $order['ext_arr'];
  398. $order_ext['buy_type'] = $buy_type; // 购买方式,alone: 单独购买, groupon: 拼团
  399. $order_ext['groupon_id'] = $groupon['id'] ?? 0; // 如果是拼团,团 id
  400. // 判断需要支付的金额是否大于 0
  401. if ($order['total_fee'] <= 0) {
  402. // 更新订单扩展字段
  403. $order->ext = json_encode($order_ext);
  404. $order->save();
  405. $order = (new \addons\shopro\model\Order)->paymentProcess($order, [
  406. 'order_sn' => $order['order_sn'],
  407. 'transaction_id' => '',
  408. 'notify_time' => date('Y-m-d H:i:s'),
  409. 'buyer_email' => $user->id,
  410. 'payment_json' => json_encode([]),
  411. 'pay_fee' => $order->total_fee,
  412. 'pay_type' => $order['type'] == 'score' ? 'score' : 'wallet' // 支付方式 积分完全支付,或者不需要支付,使用 wallet
  413. ]);
  414. } else {
  415. // 默认取第一个商品
  416. $goods_one = $new_goods_list[0]['detail'];
  417. if (isset($detail['activity']) && $detail['activity']) {
  418. $activity_one = $goods_one['activity'];
  419. // 获取第一个商品的活动规则,活动不存在,自动会使用全局自动关闭
  420. $rules = $activity_one['rules'] ?? [];
  421. } else {
  422. $rules = [];
  423. }
  424. // 优先使用活动的订单关闭时间
  425. if (isset($rules['order_auto_close']) && $rules['order_auto_close'] > 0) {
  426. $close_minue = $rules['order_auto_close'];
  427. } else {
  428. // 添加自动关闭队列
  429. $config = Config::where('name', 'order')->cache(300)->find(); // 读取配置自动缓存 5 分钟
  430. $config = isset($config) ? json_decode($config['value'], true) : [];
  431. $close_minue = (isset($config['order_auto_close']) && $config['order_auto_close'] > 0)
  432. ? $config['order_auto_close'] : 15;
  433. }
  434. // 更新订单,将过期时间存入订单,前台展示支付倒计时
  435. $order_ext['expired_time'] = time() + ($close_minue * 60);
  436. // 队列关闭
  437. \think\Queue::later(($close_minue * 60), '\addons\shopro\job\OrderAutoOper@autoClose', ['order' => $order], 'shopro');
  438. // 更新订单扩展字段
  439. $order->ext = json_encode($order_ext);
  440. $order->save();
  441. }
  442. Log::write($order->ext);
  443. return $order;
  444. }
  445. // 订单列表
  446. public static function getList($params)
  447. {
  448. $user = User::info();
  449. extract($params);
  450. $type =$params['type'];
  451. $order = (new self())->where('user_id', $user->id)->with('item');
  452. switch ($type) {
  453. case 'all':
  454. $order = $order;
  455. break;
  456. case 'nopay':
  457. $order = $order->nopay();
  458. break;
  459. case 'nosend':
  460. $order = $order->payed()->nosend();
  461. break;
  462. case 'noget':
  463. $order = $order->payed()->noget();
  464. break;
  465. case 'nocomment':
  466. $order = $order->payed()->nocomment();
  467. break;
  468. case 'aftersale':
  469. $order = $order->payed()->aftersale(); // 个人中心售后单不在走这里,而是直接走的售后单列表
  470. break;
  471. case 'invalid':
  472. $order = $order->invalid();
  473. break;
  474. case 'cancel':
  475. $order = $order->where('status', -1);
  476. break;
  477. }
  478. $orders = $order->order('id', 'desc')->paginate(10);
  479. // 处理未支付订单 item status_code
  480. $orders = $orders->toArray();
  481. if ($orders['data']) {
  482. $data = $orders['data'];
  483. foreach ($data as $key => $od) {
  484. $data[$key] = self::setOrderItemStatusByOrder($od);
  485. $data[$key]['expired_time'] = $od['createtime'] + (15 * 60);
  486. $time = $od['createtime'] + (15 * 60);
  487. if(time() > $time &&$od['status']==0){
  488. Db::name('shopro_order')->where('id' ,$od['id'])->update(['status'=>-1]);
  489. }
  490. }
  491. $orders['data'] = $data;
  492. }
  493. return $orders;
  494. }
  495. // 订单列表
  496. public static function getItemList($params)
  497. {
  498. $user = User::info();
  499. extract($params);
  500. $order = (new self())->where('user_id', $user->id)->where('status',1);
  501. if($params['type']==1){
  502. $in = '0,1';
  503. }
  504. else{
  505. $in ='-1,2';
  506. }
  507. $orders = $order->order('id','desc')->field('order_sn,id,createtime,status,ext,activity_type')->paginate(10);
  508. // 处理未支付订单 item status_code
  509. $orders = $orders->toArray();
  510. if ($orders['data']) {
  511. $data = $orders['data'];
  512. foreach ($data as $key => $od) {
  513. $data[$key]['item'] = Db::name('shopro_order_item')->where('order_id',$data[$key]['id'])->whereIn('aftersale_status',$in)->select();
  514. $data[$key]['expired_time'] = $od['createtime'] + (15 * 60);
  515. if( empty($data[$key]['item'])){
  516. unset($data[$key]);
  517. }
  518. }
  519. $orders['data'] = array_values($data);
  520. $orders['total'] = count($data);
  521. }
  522. return $orders;
  523. }
  524. // 订单详情
  525. public static function detail($params)
  526. {
  527. $user = User::info();
  528. extract($params);
  529. $order = (new self())->with(['item', 'pingjia'])->where('user_id', $user->id);
  530. if (isset($order_sn)) {
  531. $order = $order->where('order_sn', $order_sn);
  532. }
  533. if (isset($id)) {
  534. $order = $order->where('id', $id);
  535. }
  536. $order = $order->find();
  537. if (!$order) {
  538. new Exception('订单不存在');
  539. }
  540. // 处理未支付订单 item status_code
  541. // $order = self::setOrderItemStatusByOrder($order);
  542. // $order['expired_time'] = $order['createtime'] + (15 * 60);
  543. return $order;
  544. }
  545. public static function showPingjia($order_id)
  546. {
  547. $user = User::info();
  548. return Pingjia::where('order_id', $order_id)->where('user_id', $user->id)->find();
  549. }
  550. // 订单商品详情
  551. public static function itemDetail($params)
  552. {
  553. $user = User::info();
  554. extract($params);
  555. $type = $type ?? 'default';
  556. $order = (new self())->with(['item' => function ($query) use ($order_item_id) {
  557. $query->where('id', $order_item_id);
  558. }])->where('user_id', $user->id);
  559. if (isset($order_sn)) {
  560. $order = $order->where('order_sn', $order_sn);
  561. }
  562. if (isset($id)) {
  563. $order = $order->where('id', $id);
  564. }
  565. $order = $order->find();
  566. if (!$order || !$order->item) {
  567. new Exception('订单不存在');
  568. }
  569. // 处理未支付订单 item status_code
  570. $order = self::setOrderItemStatusByOrder($order);
  571. $item = $order['item'][0];
  572. unset($order['item']); // 移除订单表中的 item
  573. $item['order'] = $order; // 订单信息
  574. if ($type == 'dispatch') {
  575. $store = null;
  576. $verifies = [];
  577. if (in_array($item['dispatch_type'], ['selfetch', 'store']) && $item['store_id']) {
  578. // 自提,商家配送
  579. if (
  580. $item['dispatch_type'] == 'selfetch'
  581. && $item['dispatch_status'] == OrderItem::DISPATCH_STATUS_SENDED
  582. && $item['refund_status'] <= 0
  583. ) {
  584. // 关联核销码
  585. $verifies = Verify::where('order_id', $item['order_id'])
  586. ->where('order_item_id', $item['id'])->select();
  587. }
  588. $store = Store::where('id', $item['store_id'])->find();
  589. }
  590. $item['store'] = $store; // 门店信息
  591. $item['verify'] = $verifies; // 核销券列表
  592. // $item['autosend']
  593. // 订单详情,自动发货内容 待补充
  594. }
  595. return $item;
  596. }
  597. // 取消订单
  598. public static function operCancel($params)
  599. {
  600. $user = User::info();
  601. extract($params);
  602. $order = self::with('item')->where('user_id', $user->id)->where('id', $id)->nopay()->find();
  603. if (!$order) {
  604. new Exception('订单不存在或已取消');
  605. }
  606. // 订单取消
  607. $order = (new self)->doCancel($order, $user);
  608. return $order;
  609. }
  610. public function doCancel($order, $user, $type = 'user')
  611. {
  612. $order = Db::transaction(function () use ($order, $user, $type) {
  613. $data = ['order' => $order];
  614. \think\Hook::listen('order_cancel_before', $data);
  615. $order->status = Order::STATUS_CANCEL; // 取消订单
  616. $order->ext = json_encode($order->setExt($order, ['cancel_time' => time()])); // 取消时间
  617. $order->save();
  618. OrderAction::operAdd($order, null, $user, $type, ($type == 'user' ? '用户' : '管理员') . '取消订单');
  619. // 订单取消,退回库存,退回优惠券积分,等
  620. $data = ['order' => $order];
  621. \think\Hook::listen('order_cancel_after', $data);
  622. return $order;
  623. });
  624. return $order;
  625. }
  626. private static function getItem($order, $order_item_id)
  627. {
  628. if (!$order) {
  629. new Exception('当前订单不存在');
  630. }
  631. $order_item = null;
  632. foreach ($order->item as $item) {
  633. if ($item->id == $order_item_id) {
  634. $order_item = $item;
  635. break;
  636. }
  637. }
  638. if (!$order_item) {
  639. new Exception('订单商品不需要操作');
  640. }
  641. return $order_item;
  642. }
  643. // 确认收货订单
  644. public static function operConfirm($params)
  645. {
  646. $user = User::info();
  647. extract($params);
  648. $order = Db::transaction(function () use ($id, $order_item_id, $user) {
  649. // 加锁查询订单,exist 里面的子查询不会加锁,但是该语句需要等待锁释放才能正常查询,所以下面的获取 item 已经是更改过状态之后的了
  650. $order = self::noget()->where('user_id', $user->id)->where('id', $id)->lock(true)->find();
  651. // 获取要操作的 订单 item
  652. $item = self::getItem($order, $order_item_id);
  653. if ($item->dispatch_status == OrderItem::DISPATCH_STATUS_NOSEND) {
  654. new Exception('当前订单未发货');
  655. }
  656. if ($item->dispatch_status == OrderItem::DISPATCH_STATUS_GETED) {
  657. new Exception('当前订单已收货');
  658. }
  659. $superior = new \app\common\model\User();
  660. $share = new \addons\shopro\model\Share();
  661. $userlog = new \addons\shopro\model\UserWalletLog();
  662. $pid = $share->where('user_id', $user->id)->field('share_id')->find();
  663. if ($pid) {
  664. $before = $superior->where('id', $pid['share_id'])->find();
  665. $money = config('site.rebate') / 100;
  666. $number = $money * $item->goods_price;
  667. $after = $before['money'] + $number;
  668. $superior->where('id', $pid['share_id'])->update(['money' => $after]);
  669. $userlog->write($before, $number, $before['money'], $after, 'commission_income', '', 'money', '一级返利');
  670. $tpid = $share->where('user_id', $pid['share_id'])->field('share_id')->find();
  671. if ($tpid) {
  672. $tbefore = $superior->where('id', $tpid['share_id'])->find();
  673. $tmoney = config('site.trebate') / 100;
  674. $tnumber = $tmoney * $item->goods_price;
  675. $tafter = $tbefore['money'] + $tnumber;
  676. $superior->where('id', $tpid['share_id'])->update(['money' => $tafter]);
  677. $userlog->write($tbefore, $tnumber, $tbefore['money'], $tafter, 'commission_income', '', 'money', '二级返利');
  678. }
  679. }
  680. (new self())->getedItem($order, $item, ['oper_type' => 'user']);
  681. });
  682. return $order;
  683. }
  684. // 删除订单
  685. public static function operDelete($params)
  686. {
  687. $user = User::info();
  688. extract($params);
  689. $order = self::canDelete()->where('user_id', $user->id)->where('id', $id)->find();
  690. if (!$order) {
  691. new Exception('订单不存在或不可删除');
  692. }
  693. $order = Db::transaction(function () use ($order, $user) {
  694. $order->delete(); // 删除订单
  695. OrderAction::operAdd($order, null, $user, 'user', '用户删除订单');
  696. return $order;
  697. });
  698. return $order;
  699. }
  700. // 评价
  701. public static function operComment($params)
  702. {
  703. $user = User::info();
  704. $order = Db::transaction(function () use ($user, $params) {
  705. extract($params);
  706. // 加锁读取订单,直到订单评价状态改变
  707. $order = self::with('item')->payed()->where('user_id', $user->id)->where('id', $id)->lock(true)->find();
  708. // 获取要操作的 订单 item
  709. $item = self::getItem($order, $order_item_id);
  710. if ($item->comment_status == 1) {
  711. new Exception('当前商品已评价');
  712. }
  713. // 订单评价前
  714. $data = ['order' => $order, 'item' => $item];
  715. \think\Hook::listen('order_comment_before', $data); // 重新拿到更新过的订单
  716. $images = (isset($images) && $images) ? $images : null;
  717. // 获取用户评论配置
  718. $config = \addons\shopro\model\Config::where('name', 'order')->value('value');
  719. $config = json_decode($config, true);
  720. GoodsComment::create([
  721. 'goods_id' => $item->goods_id,
  722. 'order_id' => $order->id,
  723. 'order_item_id' => $item->id,
  724. 'user_id' => $user->id,
  725. 'level' => $level,
  726. 'content' => $content,
  727. 'images' => $images ? implode(',', $images) : $images,
  728. // 'status' => !empty($config['user_reply']) ? 'show' : 'hidden'
  729. 'status' => 'show'
  730. ]);
  731. $item->comment_status = OrderItem::COMMENT_STATUS_OK; // 评价成功
  732. $item->save();
  733. OrderAction::operAdd($order, $item, $user, 'user', '用户评价成功');
  734. // 订单评价后
  735. $data = ['order' => $order, 'item' => $item];
  736. \think\Hook::listen('order_comment_after', $data);
  737. });
  738. }
  739. // 个人中心订单数量
  740. public static function statusNum()
  741. {
  742. $user = User::info();
  743. $status_num['nopay'] = self::where('user_id', $user->id)->nopay()->count();
  744. $status_num['nosend'] = self::where('user_id', $user->id)->payed()->nosend()->count();
  745. $status_num['noget'] = self::where('user_id', $user->id)->payed()->noget()->count();
  746. $status_num['nocomment'] = self::where('user_id', $user->id)->payed()->nocomment()->count();
  747. // $status_num['aftersale'] = self::where('user_id', $user->id)->payed()->aftersale()->count();
  748. $status_num['aftersale'] = \addons\shopro\model\OrderAftersale::where('user_id', $user->id)->count();
  749. return $status_num;
  750. }
  751. public function paymentProcess($order, $notify)
  752. {
  753. $order->status = Order::STATUS_PAYED;
  754. $order->paytime = time();
  755. $order->transaction_id = $notify['transaction_id'];
  756. $order->payment_json = $notify['payment_json'];
  757. $order->pay_type = $notify['pay_type'];
  758. $order->pay_fee = $notify['pay_fee'];
  759. $order->save();
  760. $user = User::where('id', $order->user_id)->find();
  761. OrderAction::operAdd($order, null, $user, 'user', '用户支付成功');
  762. // 支付成功后续使用异步队列处理
  763. \think\Queue::push('\addons\shopro\job\OrderPayed@payed', ['order' => $order, 'user' => $user], 'shopro-high');
  764. return $order;
  765. }
  766. // 开始退款
  767. public static function startRefund($order, $item, $refund_money, $user = null, $remark = '')
  768. {
  769. // 订单退款前
  770. $data = ['order' => $order, 'item' => $item];
  771. \think\Hook::listen('order_refund_before', $data);
  772. $item->refund_status = \app\admin\model\shopro\order\OrderItem::REFUND_STATUS_OK; // 同意退款
  773. $item->refund_fee = $refund_money;
  774. $item->save();
  775. \addons\shopro\model\OrderAction::operAdd($order, $item, $user, ($user ? 'admin' : 'system'), $remark . ',退款金额:' . $refund_money);
  776. \app\admin\model\shopro\order\Order::refund($order, $item, $refund_money, $remark);
  777. // 订单退款后
  778. $data = ['order' => $order, 'item' => $item];
  779. \think\Hook::listen('order_refund_after', $data);
  780. }
  781. // 退款
  782. public static function refund($order, $item, $refund_money, $remark = '')
  783. {
  784. // 生成退款单
  785. $refundLog = new RefundLog();
  786. $refundLog->order_sn = $order->order_sn;
  787. $refundLog->refund_sn = RefundLog::getSn($order->user_id);
  788. $refundLog->order_item_id = $item->id;
  789. $refundLog->pay_fee = $order->pay_fee;
  790. $refundLog->refund_fee = $refund_money;
  791. $refundLog->pay_type = $order->pay_type;
  792. $refundLog->save();
  793. if ($order->pay_type == 'wechat' || $order->pay_type == 'alipay') {
  794. // 微信|支付宝退款
  795. // 退款数据
  796. $order_data = [
  797. 'out_trade_no' => $order->order_sn
  798. ];
  799. if ($order->pay_type == 'wechat') {
  800. $total_fee = $order->pay_fee * 100;
  801. $refund_fee = $refund_money * 100;
  802. $order_data = array_merge($order_data, [
  803. 'out_refund_no' => $refundLog->refund_sn,
  804. 'total_fee' => $total_fee,
  805. 'refund_fee' => $refund_fee,
  806. 'refund_desc' => $remark,
  807. ]);
  808. } else {
  809. $order_data = array_merge($order_data, [
  810. 'out_request_no' => $refundLog->refund_sn,
  811. 'refund_amount' => $refund_money,
  812. ]);
  813. }
  814. $notify_url = request()->domain() . '/addons/shopro/pay/notifyr/payment/' . $order->pay_type . '/platform/' . $order->platform;
  815. $pay = new \addons\shopro\library\PayService($order->pay_type, $order->platform, $notify_url);
  816. $result = $pay->refund($order_data);
  817. \think\Log::write('refund-result' . json_encode($result));
  818. if ($order->pay_type == 'wechat') {
  819. // 微信通知回调 pay->notifyr
  820. if ($result['return_code'] == 'SUCCESS' && $result['result_code'] == 'SUCCESS') {
  821. return true;
  822. } else {
  823. throw new \Exception($result['return_msg']);
  824. }
  825. } else {
  826. // 支付宝通知回调 pay->notifyx
  827. if ($result['code'] == "10000") {
  828. return true;
  829. } else {
  830. throw new \Exception($result['msg']);
  831. }
  832. }
  833. // { // 微信返回结果
  834. // "return_code":"SUCCESS",
  835. // "return_msg":"OK",
  836. // "appid":"wx39cd0799d4567dd0",
  837. // "mch_id":"1481069012",
  838. // "nonce_str":"huW9eIAb5BDPn0Ma",
  839. // "sign":"250316740B263FE53F5DFF50AF5A8FA1",
  840. // "result_code":"SUCCESS",
  841. // "transaction_id":"4200000497202004072822298902",
  842. // "out_trade_no":"202010300857029180027000",
  843. // "out_refund_no":"1586241595",
  844. // "refund_id":"50300603862020040700031444448",
  845. // "refund_channel":[],
  846. // "refund_fee":"1",
  847. // "coupon_refund_fee":"0",
  848. // "total_fee":"1",
  849. // "cash_fee":"1",
  850. // "coupon_refund_count":"0",
  851. // "cash_refund_fee":"1
  852. // }
  853. // { // 支付宝返回结果
  854. // "code": "10000",
  855. // "msg": "Success",
  856. // "buyer_logon_id": "157***@163.com",
  857. // "buyer_user_id": "2088902485164146",
  858. // "fund_change": "Y",
  859. // "gmt_refund_pay": "2020-08-15 16:11:45",
  860. // "out_trade_no": "202002460317545607015300",
  861. // "refund_fee": "0.01",
  862. // "send_back_fee": "0.00",
  863. // "trade_no": "2020081522001464141438570535"
  864. // }
  865. } else if ($order->pay_type == 'wallet') {
  866. // 余额退款
  867. if ($refund_money != 0) {
  868. \addons\shopro\model\User::money($refund_money, $order->user_id, 'wallet_refund', $order->id, '', [
  869. 'order_id' => $order->id,
  870. 'order_sn' => $order->order_sn,
  871. 'item_id' => $item->id,
  872. ]);
  873. }
  874. self::refundFinish($order, $item, $refundLog);
  875. return true;
  876. } else if ($order->pay_type == 'score') {
  877. // 积分退款,暂不支持积分退款
  878. }
  879. }
  880. public static function refundFinish($order, $item, $refundLog)
  881. {
  882. // 退款完成
  883. $refundLog->status = 1;
  884. $refundLog->save();
  885. // 退款完成
  886. $item->refund_status = \app\admin\model\shopro\order\OrderItem::REFUND_STATUS_FINISH; // 退款完成
  887. $item->save();
  888. \addons\shopro\model\OrderAction::operAdd($order, $item, null, 'admin', '退款成功');
  889. }
  890. public function setExt($order, $field, $origin = [])
  891. {
  892. $newExt = array_merge($origin, $field);
  893. $orderExt = $order['ext_arr'];
  894. return array_merge($orderExt, $newExt);
  895. }
  896. }