Groupon.php 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278
  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->price = $activitySkuPrice['price'];
  99. $skuPrice->groupon_price = $activitySkuPrice['price']; // 不覆盖原来规格价格,用作单独购买,将活动的价格设置为新的拼团价格
  100. $skuPrice->is_leader_discount = 0; // 是否团长优惠
  101. $skuPrice->leader_price = $activitySkuPrice['price']; // 团长优惠价格
  102. $skuPrice->status = $activitySkuPrice['status']; // 采用活动的上下架
  103. $skuPrice->ext = $activitySkuPrice['ext']; // 活动规格 ext, order_item 保存备用
  104. $skuPrice->min_price = $activitySkuPrice['price']; // 当前活动规格最小价格,这里是拼团价
  105. $skuPrice->max_price = $activitySkuPrice['price']; // 用作计算活动中最大价格
  106. // 记录相关活动类型
  107. $skuPrice->activity_type = $activity['type'];
  108. $skuPrice->activity_id = $activity['id'];
  109. $skuPrice->activity_endtime = $activity['end_time'];
  110. $skuPrice->activity_seconds = $activity['end_time'] - time();
  111. // 下单的时候需要存活动 的 sku_price_id)
  112. $skuPrice->item_goods_sku_price = $activitySkuPrice;
  113. break;
  114. }
  115. }
  116. }
  117. return $skuPrices;
  118. }
  119. /**
  120. * 这里要使用 shoproException 抛出异常
  121. *
  122. * @param array $buyInfo
  123. * @param array $activity
  124. * @return array
  125. */
  126. public function buyCheck($buyInfo, $activity)
  127. {
  128. $buy_type = request()->param('buy_type', 'groupon');
  129. $groupon_id = request()->param('groupon_id', 0);
  130. // 拼团
  131. $rules = $activity['rules'];
  132. $is_alone = $rules['is_alone'] ?? 1;
  133. $currentSkuPrice = $buyInfo['current_sku_price'];
  134. $is_leader_discount = $currentSkuPrice['is_leader_discount'];
  135. // 成团人数
  136. $num = $rules['team_num'] ?? 1;
  137. // 额外需要的库存
  138. $need_add_num = 0;
  139. // 要单独购买
  140. if ($buy_type == 'alone') {
  141. // 不允许单独购买
  142. if (!$is_alone) {
  143. throw new ShoproException('该商品不允许单独购买');
  144. }
  145. } else {
  146. // 拼团,临时将拼团价设置为商品价格
  147. if (!$groupon_id && $is_leader_discount) {
  148. // 开新团,并且有团长优惠,使用优惠价格
  149. $buyInfo['current_sku_price']['price'] = $currentSkuPrice['leader_price'];
  150. } else {
  151. // 参与团,或者没有团长优惠
  152. $buyInfo['current_sku_price']['price'] = $currentSkuPrice['groupon_price'];
  153. }
  154. }
  155. // 如果是开新团
  156. if (!$groupon_id && $buy_type == 'groupon') {
  157. // 开团需要的最小库存
  158. $need_add_num = ($num - 1);
  159. }
  160. // 当前库存,小于要购买的数量
  161. $need_num = $buyInfo['goods_num'] + ($need_add_num ?? 0);
  162. if ($currentSkuPrice['stock'] < $need_num) {
  163. if ($need_add_num && $is_alone && !$groupon_id && $buy_type == 'groupon') {
  164. throw new ShoproException('商品库存不足以开团,请选择单独购买');
  165. } else if ($buy_type == 'alone') {
  166. throw new ShoproException('商品库存不足');
  167. } else {
  168. throw new ShoproException('该商品不允商品库存不足以开团许单独购买');
  169. }
  170. }
  171. $buyInfo['is_commission'] = $rules['is_commission'] ?? 0; // 是否参与分销
  172. return $buyInfo;
  173. }
  174. public function buy($buyInfo, $activity)
  175. {
  176. $user = auth_user();
  177. $buy_type = request()->param('buy_type', 'groupon');
  178. $groupon_id = request()->param('groupon_id', 0);
  179. // 参与现有团
  180. if ($buy_type != 'alone' && $groupon_id) {
  181. // 检测并获取要参与的团
  182. $activityGroupon = $this->checkAndGetJoinGroupon($buyInfo, $user, $groupon_id);
  183. }
  184. // 判断 并 增加 redis 销量
  185. $stockSale = new StockSale();
  186. $stockSale->cacheForwardSale($buyInfo);
  187. // (开新团不判断)参与旧团 增加预拼团人数,上面加入团的时候已经判断过一次了,所以这里 99.99% 会加入成功的
  188. if (isset($activityGroupon) && $activityGroupon) {
  189. // 增加拼团预成员人数
  190. $goods = $buyInfo['goods'];
  191. $activity = $goods['activity'];
  192. $this->grouponCacheForwardNum($activityGroupon, $activity, $user);
  193. }
  194. return $buyInfo;
  195. }
  196. public function buyOk($order, $user)
  197. {
  198. $this->joinGroupon($order, $user, function ($activityRules, $itemExt) {
  199. $team_num = $activityRules['team_num'] ?? 1;
  200. return compact('team_num');
  201. });
  202. }
  203. /**
  204. * 拼团购买失败
  205. *
  206. * @param \think\Model $order
  207. * @param string $type
  208. * @return void
  209. */
  210. public function buyFail($order, $type)
  211. {
  212. if ($type == 'invalid') {
  213. if ($order->pay_mode == 'offline') {
  214. // 肯定是已经货到付款的订单取消订单,这时候已经添加了参团记录
  215. $this->refundGrouponLog($order);
  216. } else {
  217. // 订单失效,扣除预拼团人数(只处理正在进行中的团)
  218. $this->grouponCacheBackNum($order, $type);
  219. }
  220. } else {
  221. // type = refund 退款订单将参团标记为已退款
  222. $this->refundGrouponLog($order);
  223. }
  224. // 判断扣除预销量 (活动信息还在 redis)
  225. $stockSale = new StockSale();
  226. $stockSale->cacheBackSale($order);
  227. }
  228. }