Groupon.php 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275
  1. <?php
  2. namespace addons\shopro\library\activity\provider;
  3. use addons\shopro\library\activity\traits\Groupon as GrouponTrait;
  4. use addons\shopro\service\StockSale;
  5. use addons\shopro\exception\ShoproException;
  6. /**
  7. * 普通拼团
  8. */
  9. class Groupon extends Base
  10. {
  11. use GrouponTrait;
  12. protected $rules = [
  13. "is_commission" => "require|boolean",
  14. "is_free_shipping" => "require|boolean",
  15. "sales_show_type" => "require",
  16. "team_num" => "require|number",
  17. "is_alone" => "require|boolean",
  18. "is_fictitious" => "require|boolean",
  19. "fictitious_time" => "requireIf:is_fictitious,1|float|egt:0",
  20. "is_team_card" => "require|boolean",
  21. "is_leader_discount" => "require|boolean",
  22. "valid_time" => "require|float|egt:0",
  23. "limit_num" => "number|egt:0",
  24. "refund_type" => "require", // 退款方式 back=原路退回|money=退回到余额
  25. "order_auto_close" => "float|egt:0",
  26. ];
  27. protected $message = [
  28. 'team_num.require' => '请填写成团人数',
  29. ];
  30. protected $default = [
  31. "is_commission" => 0, // 是否参与分销
  32. "is_free_shipping" => 0, // 是否包邮
  33. "sales_show_type" => "real", // real=真实活动销量|goods=商品总销量(包含虚拟销量)
  34. "team_num" => 2, // 成团人数,最少两人
  35. "is_alone" => 0, // 是否允许单独购买
  36. "is_fictitious" => 0, // 是否允许虚拟成团
  37. "fictitious_num" => 0, // 最多虚拟人数 0:不允许虚拟 '' 不限制
  38. "fictitious_time" => 0, // 开团多长时间自动虚拟成团
  39. "is_team_card" => 0, // 参团卡显示
  40. "is_leader_discount" => 0, // 团长优惠
  41. "valid_time" => 0, // 组团有效时间, 0:一直有效
  42. "limit_num" => 0, // 每人限购数量 0:不限购
  43. "refund_type" => "back", // 退款方式 back=原路退回|money=退回到余额
  44. "order_auto_close" => 0, // 订单自动关闭时间,如果为 0 将使用系统级订单自动关闭时间
  45. ];
  46. public function check($params, $activity_id = 0)
  47. {
  48. // 数据验证
  49. $params = parent::check($params);
  50. // 验证添加的活动商品是否至少设置了一个活动规格
  51. $this->checkActivitySkuPrice($params['goods_list']);
  52. // 检测活动之间是否存在冲突
  53. $this->checkActivityConflict($params, $params['goods_list'], $activity_id);
  54. return $params;
  55. }
  56. public function save($activity, $params = [])
  57. {
  58. $goodsList = $params['goods_list'];
  59. $this->saveSkuPrice($goodsList, $activity, function ($skuPrice) use ($activity) {
  60. // 处理 团长优惠
  61. $rules = $activity->rules;
  62. $is_leader_discount = $rules['is_leader_discount'] ?? 0;
  63. $leader_price = $skuPrice['price'];
  64. if ($is_leader_discount && isset($skuPrice['leader_price'])) {
  65. $leader_price = $skuPrice['leader_price'];
  66. }
  67. $ext = [
  68. 'is_leader_discount' => $is_leader_discount,
  69. 'leader_price' => number_format(floatval($leader_price), 2, '.', '')
  70. ];
  71. unset($skuPrice['leader_price']);
  72. $skuPrice['ext'] = $ext;
  73. return $skuPrice;
  74. });
  75. }
  76. public function showSkuPrice($skuPrice)
  77. {
  78. $ext = $skuPrice['ext'] ?? [];
  79. $skuPrice['leader_price'] = $ext['leader_price'] ?? $skuPrice['price'];
  80. return $skuPrice;
  81. }
  82. public function recoverSkuPrices($goods, $activity)
  83. {
  84. $activitySkuPrices = $activity['activity_sku_prices'];
  85. $skuPrices = $goods->sku_prices;
  86. foreach ($skuPrices as $key => &$skuPrice) {
  87. $stock = $skuPrice->stock; // 下面要用
  88. $skuPrice->stock = 0;
  89. $skuPrice->sales = 0;
  90. foreach ($activitySkuPrices as $activitySkuPrice) {
  91. if ($skuPrice->id == $activitySkuPrice['goods_sku_price_id']) {
  92. // 采用活动的 规格内容
  93. $is_leader_discount = $activitySkuPrice['ext']['is_leader_discount'];
  94. $leader_price = $activitySkuPrice['ext']['leader_price'];
  95. $skuPrice->old_price = $skuPrice->price; // 保存原始普通商品规格的价格(计算活动的优惠)
  96. $skuPrice->stock = ($activitySkuPrice['stock'] > $stock) ? $stock : $activitySkuPrice['stock']; // 活动库存不能超过商品库存
  97. $skuPrice->sales = $activitySkuPrice['sales'];
  98. $skuPrice->groupon_price = $activitySkuPrice['price']; // 不覆盖原来规格价格,用作单独购买,将活动的价格设置为新的拼团价格
  99. $skuPrice->is_leader_discount = $is_leader_discount; // 是否团长优惠
  100. $skuPrice->leader_price = $leader_price; // 团长优惠价格
  101. $skuPrice->status = $activitySkuPrice['status']; // 采用活动的上下架
  102. $skuPrice->ext = $activitySkuPrice['ext']; // 活动规格 ext, order_item 保存备用
  103. $skuPrice->min_price = $activitySkuPrice['price']; // 当前活动规格最小价格,这里是拼团价
  104. $skuPrice->max_price = $activitySkuPrice['price']; // 用作计算活动中最大价格
  105. // 记录相关活动类型
  106. $skuPrice->activity_type = $activity['type'];
  107. $skuPrice->activity_id = $activity['id'];
  108. // 下单的时候需要存活动 的 sku_price_id)
  109. $skuPrice->item_goods_sku_price = $activitySkuPrice;
  110. break;
  111. }
  112. }
  113. }
  114. return $skuPrices;
  115. }
  116. /**
  117. * 这里要使用 shoproException 抛出异常
  118. *
  119. * @param array $buyInfo
  120. * @param array $activity
  121. * @return array
  122. */
  123. public function buyCheck($buyInfo, $activity)
  124. {
  125. $buy_type = request()->param('buy_type', 'groupon');
  126. $groupon_id = request()->param('groupon_id', 0);
  127. // 拼团
  128. $rules = $activity['rules'];
  129. $is_alone = $rules['is_alone'] ?? 1;
  130. $currentSkuPrice = $buyInfo['current_sku_price'];
  131. $is_leader_discount = $currentSkuPrice['is_leader_discount'];
  132. // 成团人数
  133. $num = $rules['team_num'] ?? 1;
  134. // 额外需要的库存
  135. $need_add_num = 0;
  136. // 要单独购买
  137. if ($buy_type == 'alone') {
  138. // 不允许单独购买
  139. if (!$is_alone) {
  140. throw new ShoproException('该商品不允许单独购买');
  141. }
  142. } else {
  143. // 拼团,临时将拼团价设置为商品价格
  144. if (!$groupon_id && $is_leader_discount) {
  145. // 开新团,并且有团长优惠,使用优惠价格
  146. $buyInfo['current_sku_price']['price'] = $currentSkuPrice['leader_price'];
  147. } else {
  148. // 参与团,或者没有团长优惠
  149. $buyInfo['current_sku_price']['price'] = $currentSkuPrice['groupon_price'];
  150. }
  151. }
  152. // 如果是开新团
  153. if (!$groupon_id && $buy_type == 'groupon') {
  154. // 开团需要的最小库存
  155. $need_add_num = ($num - 1);
  156. }
  157. // 当前库存,小于要购买的数量
  158. $need_num = $buyInfo['goods_num'] + ($need_add_num ?? 0);
  159. if ($currentSkuPrice['stock'] < $need_num) {
  160. if ($need_add_num && $is_alone && !$groupon_id && $buy_type == 'groupon') {
  161. throw new ShoproException('商品库存不足以开团,请选择单独购买');
  162. } else if ($buy_type == 'alone') {
  163. throw new ShoproException('商品库存不足');
  164. } else {
  165. throw new ShoproException('该商品不允商品库存不足以开团许单独购买');
  166. }
  167. }
  168. $buyInfo['is_commission'] = $rules['is_commission'] ?? 0; // 是否参与分销
  169. return $buyInfo;
  170. }
  171. public function buy($buyInfo, $activity)
  172. {
  173. $user = auth_user();
  174. $buy_type = request()->param('buy_type', 'groupon');
  175. $groupon_id = request()->param('groupon_id', 0);
  176. // 参与现有团
  177. if ($buy_type != 'alone' && $groupon_id) {
  178. // 检测并获取要参与的团
  179. $activityGroupon = $this->checkAndGetJoinGroupon($buyInfo, $user, $groupon_id);
  180. }
  181. // 判断 并 增加 redis 销量
  182. $stockSale = new StockSale();
  183. $stockSale->cacheForwardSale($buyInfo);
  184. // (开新团不判断)参与旧团 增加预拼团人数,上面加入团的时候已经判断过一次了,所以这里 99.99% 会加入成功的
  185. if (isset($activityGroupon) && $activityGroupon) {
  186. // 增加拼团预成员人数
  187. $goods = $buyInfo['goods'];
  188. $activity = $goods['activity'];
  189. $this->grouponCacheForwardNum($activityGroupon, $activity, $user);
  190. }
  191. return $buyInfo;
  192. }
  193. public function buyOk($order, $user)
  194. {
  195. $this->joinGroupon($order, $user, function ($activityRules, $itemExt) {
  196. $team_num = $activityRules['team_num'] ?? 1;
  197. return compact('team_num');
  198. });
  199. }
  200. /**
  201. * 拼团购买失败
  202. *
  203. * @param \think\Model $order
  204. * @param string $type
  205. * @return void
  206. */
  207. public function buyFail($order, $type)
  208. {
  209. if ($type == 'invalid') {
  210. if ($order->pay_mode == 'offline') {
  211. // 肯定是已经货到付款的订单取消订单,这时候已经添加了参团记录
  212. $this->refundGrouponLog($order);
  213. } else {
  214. // 订单失效,扣除预拼团人数(只处理正在进行中的团)
  215. $this->grouponCacheBackNum($order, $type);
  216. }
  217. } else {
  218. // type = refund 退款订单将参团标记为已退款
  219. $this->refundGrouponLog($order);
  220. }
  221. // 判断扣除预销量 (活动信息还在 redis)
  222. $stockSale = new StockSale();
  223. $stockSale->cacheBackSale($order);
  224. }
  225. }