StoreOrderSplitServices.php 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214
  1. <?php
  2. // +----------------------------------------------------------------------
  3. // | CRMEB [ CRMEB赋能开发者,助力企业发展 ]
  4. // +----------------------------------------------------------------------
  5. // | Copyright (c) 2016~2020 https://www.crmeb.com All rights reserved.
  6. // +----------------------------------------------------------------------
  7. // | Licensed CRMEB并不是自由软件,未经许可不能去掉CRMEB相关版权
  8. // +----------------------------------------------------------------------
  9. // | Author: CRMEB Team <admin@crmeb.com>
  10. // +----------------------------------------------------------------------
  11. namespace app\services\order;
  12. use app\services\BaseServices;
  13. use app\dao\order\StoreOrderDao;
  14. use app\services\user\UserServices;
  15. use think\exception\ValidateException;
  16. /**
  17. * 订单拆分
  18. * Class StoreOrderSplitServices
  19. * @package app\services\order
  20. */
  21. class StoreOrderSplitServices extends BaseServices
  22. {
  23. /**
  24. * 需要清空恢复默认数据字段
  25. * @var string[]
  26. */
  27. protected $order_data = ['id', 'status', 'refund_status', 'refund_type', 'refund_express', 'refund_reason_wap_img', 'refund_reason_wap_explain', 'refund_reason_time', 'refund_reason_wap', 'refund_reason', 'refund_price', 'delivery_name', 'delivery_code', 'delivery_type', 'delivery_id', 'fictitious_content', 'delivery_uid'];
  28. /**
  29. * 构造方法
  30. * StoreOrderRefundServices constructor.
  31. * @param StoreOrderDao $dao
  32. */
  33. public function __construct(StoreOrderDao $dao)
  34. {
  35. $this->dao = $dao;
  36. }
  37. /**
  38. * 订单拆分
  39. * @param int $id
  40. * @param array $cart_ids
  41. * @param array $orderInfo
  42. * @throws \think\db\exception\DataNotFoundException
  43. * @throws \think\db\exception\DbException
  44. * @throws \think\db\exception\ModelNotFoundException
  45. */
  46. public function split(int $id, array $cart_ids, $orderInfo = [])
  47. {
  48. $ids = array_unique(array_column($cart_ids, 'cart_id'));
  49. if (!$cart_ids || !$ids) {
  50. return false;
  51. }
  52. if (!$orderInfo) {
  53. $orderInfo = $this->dao->get($id, ['*']);
  54. }
  55. if (!$orderInfo) {
  56. throw new ValidateException('订单未能查到,不能拆分订单!');
  57. }
  58. /** @var StoreOrderCreateServices $storeOrderCreateServices */
  59. $storeOrderCreateServices = app()->make(StoreOrderCreateServices::class);
  60. $orderInfo = is_object($orderInfo) ? $orderInfo->toArray() : $orderInfo;
  61. foreach ($this->order_data as $field) {
  62. unset($orderInfo[$field]);
  63. }
  64. $order_data = $orderInfo;
  65. $order_data['pid'] = $id;
  66. mt_srand();
  67. $order_data['order_id'] = $orderInfo['order_id'] . '_' . rand(100, 999);
  68. $order_data['cart_id'] = [];
  69. $order_data['unique'] = $storeOrderCreateServices->getNewOrderId('');
  70. $order_data['add_time'] = time();
  71. $order_data['mark'] = '拆分订单';
  72. $new_order = $this->dao->save($order_data);
  73. if (!$new_order) {
  74. throw new ValidateException('生成新订单失败');
  75. }
  76. $new_id = (int)$new_order->id;
  77. /** @var StoreOrderStatusServices $statusService */
  78. $statusService = app()->make(StoreOrderStatusServices::class);
  79. $statusService->save([
  80. 'oid' => $new_id,
  81. 'change_type' => 'split_create_order',
  82. 'change_message' => '发货拆分订单生成',
  83. 'change_time' => time()
  84. ]);
  85. /** @var StoreOrderCartInfoServices $storeOrderCartInfoServices */
  86. $storeOrderCartInfoServices = app()->make(StoreOrderCartInfoServices::class);
  87. //订单下原商品信息
  88. $cartInfo = $storeOrderCartInfoServices->getColumn(['oid' => $id, 'cart_id' => $ids], 'cart_num,surplus_num,cart_info', 'cart_id');
  89. $cart_data = $cart_data_all = $update_data = [];
  90. $cart_data['oid'] = $new_id;
  91. foreach ($cart_ids as $cart) {
  92. $surplus_num = $cartInfo[$cart['cart_id']]['surplus_num'] ?? 0;
  93. if (!isset($cartInfo[$cart['cart_id']]) || !$surplus_num) continue;
  94. $_info = is_string($cartInfo[$cart['cart_id']]['cart_info']) ? json_decode($cartInfo[$cart['cart_id']]['cart_info'], true) : $cartInfo[$cart['cart_id']]['cart_info'];
  95. $cart_data['cart_id'] = $storeOrderCreateServices->getNewOrderId('');
  96. $cart_data['product_id'] = $_info['product_id'];
  97. $cart_data['old_cart_id'] = $cart['cart_id'];
  98. $cart_data['cart_num'] = $cart['cart_num'];
  99. $cart_data['unique'] = md5($cart_data['cart_id'] . '_' . $cart_data['oid']);
  100. if ($cart['cart_num'] >= $surplus_num) {//拆分完成
  101. $cart_data['cart_num'] = $surplus_num;
  102. $update_data['split_status'] = 2;
  103. $update_data['surplus_num'] = 0;
  104. } else {//拆分部分数量
  105. $update_data['surplus_num'] = bcsub((string)$surplus_num, $cart['cart_num'], 0);
  106. $update_data['split_status'] = $update_data['surplus_num'] > 0 ? 1 : 2;
  107. }
  108. $_info = $this->slpitComputeOrderCart($cart_data['cart_num'], $_info);
  109. $_info['id'] = $cart_data['cart_id'];
  110. $cart_data['cart_info'] = json_encode($_info);
  111. //修改原来订单商品信息
  112. if (false === $storeOrderCartInfoServices->update(['oid' => $id, 'cart_id' => $cart['cart_id']], $update_data)) {
  113. throw new ValidateException('修改原来订单商品拆分状态失败,请稍候重试');
  114. }
  115. $cart_data_all[] = $cart_data;
  116. }
  117. if (!$storeOrderCartInfoServices->saveAll($cart_data_all)) {
  118. throw new ValidateException('新增拆分订单商品信息失败');
  119. }
  120. $new_order = $this->dao->get($new_id);
  121. $this->splitComputeOrder($new_id, $cart_data_all, $new_order);
  122. return $new_order;
  123. }
  124. /**
  125. * 重新计算新订单中价格等信息
  126. * @param int $id
  127. * @param $orderInfo
  128. * @param array $cart_info_data
  129. */
  130. public function splitComputeOrder(int $id, array $cart_info_data, $new_order)
  131. {
  132. $order_update['cart_id'] = array_column($cart_info_data, 'cart_id');
  133. $order_update['total_num'] = array_sum(array_column($cart_info_data, 'cart_num'));
  134. $pay_price = $total_price = $coupon_price = $deduction_price = $use_integral = $pay_postage = $gainIntegral = $one_brokerage = $two_brokerage = 0;
  135. foreach ($cart_info_data as $cart) {
  136. $_info = json_decode($cart['cart_info'], true);
  137. $pay_price = bcadd((string)$pay_price, bcmul((string)$_info['truePrice'], (string)$cart['cart_num'], 4), 2);
  138. $total_price = bcadd((string)$total_price, bcmul((string)$_info['truePrice'], (string)$cart['cart_num'], 4), 2);
  139. $deduction_price = bcadd((string)$deduction_price, (string)$_info['integral_price'], 2);
  140. $coupon_price = bcadd((string)$coupon_price, (string)$_info['coupon_price'], 2);
  141. $use_integral = bcadd((string)$use_integral, (string)$_info['use_integral'], 0);
  142. $pay_postage = bcadd((string)$pay_postage, (string)$_info['postage_price'], 2);
  143. $cartInfoGainIntegral = bcmul((string)$cart['cart_num'], (string)($_info['productInfo']['give_integral'] ?? '0'), 0);
  144. $gainIntegral = bcadd((string)$gainIntegral, (string)$cartInfoGainIntegral, 0);
  145. $one_brokerage = bcadd((string)$one_brokerage, (string)$_info['one_brokerage'], 2);
  146. $two_brokerage = bcadd((string)$two_brokerage, (string)$_info['two_brokerage'], 2);
  147. }
  148. $order_update['coupon_id'] = array_unique(array_column($cart_info_data, 'coupon_id'));
  149. $order_update['pay_price'] = bcadd((string)$pay_price, (string)$pay_postage, 2);
  150. $order_update['total_price'] = $total_price;
  151. $order_update['deduction_price'] = $deduction_price;
  152. $order_update['coupon_price'] = $coupon_price;
  153. $order_update['use_integral'] = $use_integral;
  154. $order_update['gain_integral'] = $gainIntegral;
  155. $order_update['pay_postage'] = $pay_postage;
  156. /** @var UserServices $userServices */
  157. $userServices = app()->make(UserServices::class);
  158. if ($userServices->checkUserPromoter($new_order['spread_uid'])) $order_update['one_brokerage'] = $one_brokerage;
  159. if ($userServices->checkUserPromoter($new_order['spread_two_uid'])) $order_update['two_brokerage'] = $two_brokerage;
  160. if (false === $this->dao->update($id, $order_update, 'id')) {
  161. throw new ValidateException('保存新订单商品信息失败');
  162. }
  163. return true;
  164. }
  165. /**
  166. * 部分发货重新计算订单商品:实际金额、优惠、积分等金额
  167. * @param int $cart_num
  168. * @param array $cart_info
  169. * @return array
  170. */
  171. public function slpitComputeOrderCart(int $cart_num, array $cart_info)
  172. {
  173. if (!$cart_num || !$cart_info) return [];
  174. $new_cart_info = $cart_info;
  175. /** @var StoreOrderCartInfoServices $storeOrderCartInfoServices */
  176. $storeOrderCartInfoServices = app()->make(StoreOrderCartInfoServices::class);
  177. $splitdCartInfo = $storeOrderCartInfoServices->getColumn(['old_cart_id' => $cart_info['id']], 'cart_info', '');
  178. $deliver_num = $cart_num;
  179. if ($splitdCartInfo) {
  180. foreach ($splitdCartInfo as $k => &$v) {
  181. $v = is_string($v) ? json_decode($v, true) : $v;
  182. $deliver_num = bcadd((string)$deliver_num, (string)$v['cart_num'], 0);
  183. }
  184. }
  185. $new_cart_info['cart_num'] = $cart_num;
  186. $compute_arr = ['coupon_price', 'integral_price', 'postage_price', 'use_integral', 'one_brokerage', 'two_brokerage', 'sum_true_price'];
  187. $scale = 2;
  188. foreach ($compute_arr as $field) {
  189. if (!isset($cart_info[$field]) || !$cart_info[$field]) {
  190. $new_cart_info[$field] = 0;
  191. continue;
  192. }
  193. if ($field == 'use_integral') $scale = 0;
  194. if ($deliver_num < $cart_info['cart_num']) {//分批发货 还有剩余
  195. $new_cart_info[$field] = bcmul((string)$cart_num, bcdiv((string)$cart_info[$field], (string)$cart_info['cart_num'], 4), $scale);
  196. } else {
  197. if ($splitdCartInfo) {//分批发货完成
  198. $new_cart_info[$field] = bcsub((string)$cart_info[$field], (string)array_sum(array_column($splitdCartInfo, $field)), $scale);
  199. } else {//第一次直接全部数量发货
  200. $new_cart_info[$field] = $cart_info[$field];
  201. }
  202. }
  203. }
  204. return $new_cart_info;
  205. }
  206. }