Active.php 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288
  1. <?php namespace App\Http\Controllers\Api\Redpacket;
  2. use App\Http\Controllers\Api\Api;
  3. use App\Models\Custom;
  4. use App\Models\CustomAmount;
  5. use Illuminate\Support\Facades\DB;
  6. use App\Models\Redpacket\ActiveRecord;
  7. use App\Models\Redpacket\ActiveReward;
  8. use App\Models\Redpacket\Active as Model;
  9. use App\Models\WeiBan\Tags as WeiBanTags;
  10. use App\Models\OrdersProduct;
  11. /**
  12. * 红包活动
  13. *
  14. * @author 刘相欣
  15. *
  16. * */
  17. class Active extends Api{
  18. /**
  19. * 获取抽奖配置 /api/redpacket_active/get_detail
  20. *
  21. * */
  22. public function get_detail(Model $Model,Custom $Custom,ActiveReward $ActiveReward,ActiveRecord $ActiveRecord,WeiBanTags $WeiBanTags,OrdersProduct $OrdersProduct){
  23. // 接口验签
  24. // $this->verify_sign();
  25. // 检查登录
  26. $uid = $this->checkLogin();
  27. // 接收参数
  28. $id = request('id',0);
  29. // 获取活动
  30. $data = $Model->getOne($id);
  31. // 如果存在的话
  32. if( !$data ) return json_send(['code'=>'error','msg'=>'暂无活动','data'=>$data]);
  33. // 获取客户信息
  34. $custom = $Custom->getOne($uid);
  35. // 如果存在的话
  36. if( !$custom ) return json_send(['code'=>'no_login','msg'=>'请登录','data'=>['error'=>'无对应客户']]);
  37. // 如果城市限制并且不在在城市范围内,不允许参加
  38. if ( $data['city_ids'] && !in_array($custom['city_id'],explode(',',$data['city_ids'])) ) return json_send(['code'=>'error','msg'=>'账号不在城市范围,请核对您的城市','data'=>['error'=>'不在标签范围内']]);
  39. // 判断用户ID是否在活动参与范围
  40. if( $data['custom_scope'] && !in_array($uid,explode(',',$data['custom_scope']) ) ) return json_send(['code'=>'error','msg'=>'账号不在指定客户范围','data'=>['error'=>'不在客户范围内']]);
  41. // 判断用户手机是否在活动参与范围
  42. if( $data['phone_scope'] && !in_array($custom['phone'],explode(',',$data['phone_scope']) ) ) return json_send(['code'=>'error','msg'=>'账号不在指定手机号范围','data'=>['error'=>'不在手机号范围内']]);
  43. // 如果限制新老用户参与
  44. if( $data['custom_type'] ) {
  45. // 如果客户注册时间不在活动期间
  46. if( $data['custom_start_time'] > $custom['insert_time'] || $data['custom_end_time'] < $custom['insert_time'] ) return json_send(['code'=>'error','msg'=>'账号不在活动范围','data'=>['error'=>'不在活动时间内注册']]);
  47. }
  48. // 如果限制下单
  49. if( $data['is_order'] ) {
  50. // 查询结果
  51. $query = $OrdersProduct->query()->where([['custom_uid','=',$uid],['status','=',1],['insert_time','>=',$data['order_start_time']],['insert_time','<=',$data['order_end_time']]]);
  52. // 如果商品存在
  53. if( $data['product_scope'] ) $query = $query->whereIn('product_id',explode(',',$data['product_scope']));
  54. // 获取时间段内下单数量
  55. $orderTotal = $query->groupBy('order_id')->count();
  56. // 如果没有下单数据
  57. if( !$orderTotal ) return json_send(['code'=>'error','msg'=>'不符合下单条件','data'=>['error'=>'不符合订单条件']]);
  58. }
  59. // 是否需要获取用户标签
  60. $tags = $data['tag_scope'] || $data['tag_except'] ? $WeiBanTags->getListByWeibanExtid($custom['weiban_extid']) : [];
  61. // 获取用户的标签
  62. $tags = $tags ? array_column($tags,'name') : [];
  63. // 判断是不是可以参与
  64. if( $data['tag_scope'] ) {
  65. // 解析数组
  66. $data['tag_scope'] = explode(',',$data['tag_scope']);
  67. // 计算交集
  68. $intersect = count(array_intersect($data['tag_scope'],$tags));
  69. // 判断标签是否必须同时满足
  70. if( $data['tag_scope_type'] == 1 ) {
  71. // 判断用户的标签是否全部满足于标签限制范围内
  72. if( $intersect != count($data['tag_scope']) ) return json_send(['code'=>'error','msg'=>'账号不在标签范围','data'=>['error'=>'不在标签范围内']]);
  73. }else{
  74. // 判断标签限制并且不在标签限制范围内,不允许参加
  75. if( !$intersect ) return json_send(['code'=>'error','msg'=>'账号不在标签范围','data'=>['error'=>'不在标签范围内']]);
  76. }
  77. }
  78. // 判断是不是可以参与
  79. if( $data['tag_except'] ) {
  80. // 解析数组
  81. $data['tag_except'] = explode(',',$data['tag_except']);
  82. // 计算交集
  83. $intersect = count(array_intersect($data['tag_except'],$tags));
  84. // 如果存在交集,在排除范围,即不可参与
  85. if( $intersect ) return json_send(['code'=>'error','msg'=>'账号不在范围','data'=>['error'=>'在标签排除范围内']]);
  86. }
  87. // 奖品
  88. $reward = $ActiveReward->getListByLottery($data['id']);
  89. // 活动暂无奖品
  90. if( !$reward ) return json_send(['code'=>'error','msg'=>'活动暂未配置奖项','data'=>$data]);
  91. // 查询条件
  92. $map = [];
  93. // 判断周期
  94. if( !empty($data['freq']) ) {
  95. if( $data['freq'] == 1 ) $map = [['insert_time','>=',now()->startOfDay()->getTimestamp()],['insert_time','<=',now()->endOfDay()->getTimestamp()]];
  96. if( $data['freq'] == 2 ) $map = [['insert_time','>=',now()->startOfWeek()->getTimestamp()],['insert_time','<=',now()->endOfWeek()->getTimestamp()]];
  97. if( $data['freq'] == 3 ) $map = [['insert_time','>=',now()->startOfMonth()->getTimestamp()],['insert_time','<=',now()->endOfMonth()->getTimestamp()]];
  98. }
  99. // logo
  100. $data['logo'] = $data['logo'] ? path_compat($data['logo']) : '';
  101. // 查询用户已参与次数
  102. $data['join_num'] = (int)$ActiveRecord->query()->where([['custom_uid','=',$uid],['active_id','=',$id]])->where($map)->count();
  103. // 计算剩余次数
  104. $data['number'] = $data['lucky_num'] - $data['join_num'];
  105. // 最少为0,避免显示异常
  106. $data['number'] = $data['number'] < 0 ? 0 : $data['number'];
  107. // 时间处理
  108. $data['start_date'] = date('Y/m/d H:i',$data['start_time']);
  109. // 时间处理
  110. $data['end_date'] = date('Y/m/d H:i',$data['end_time']);
  111. // 通过活动ID,查询奖品
  112. $data['reward_list'] = [];
  113. // 奖品数据
  114. foreach ($reward as $value) {
  115. // 奖项
  116. $data['reward_list'][] = ['id'=>$value['id'],'money'=>$value['money']];
  117. }
  118. // 默认可以参加活动
  119. $data['allow_join'] = 1;
  120. // 返回结果
  121. return json_send(['code'=>'success','msg'=>'获取成功','data'=>$data]);
  122. }
  123. /**
  124. * 抽奖 /api/redpacket_active/get_reward
  125. *
  126. * */
  127. public function get_reward(Model $Model,Custom $Custom,ActiveRecord $ActiveRecord,ActiveReward $ActiveReward,CustomAmount $CustomAmount,WeiBanTags $WeiBanTags,OrdersProduct $OrdersProduct){
  128. // 接口验签
  129. // $this->verify_sign();
  130. // 检查登录
  131. $uid = $this->checkLogin();
  132. // 获取活动
  133. $activeId = request('active_id',0);
  134. // 如果存在的话
  135. if( !$activeId ) return json_send(['code'=>'error','msg'=>'请选择参与的活动','data'=>['error'=>"抽奖活动ID有误"]]);
  136. // 获取客户信息
  137. $custom = $Custom->getOne($uid);
  138. // 如果存在的话
  139. if( !$custom ) return json_send(['code'=>'no_login','msg'=>'请登录','data'=>['error'=>'无对应客户']]);
  140. // 通过活动ID,查询奖品
  141. $data = $Model->getOne($activeId);
  142. // 如果存在的话
  143. if( !$data ) return json_send(['code'=>'error','msg'=>'活动不存在或未开始','data'=>$data]);
  144. // 活动时间判断
  145. if( $data['start_time'] > time() ) return json_send(['code'=>'error','msg'=>'活动暂未开始','data'=>$data]);
  146. // 活动时间判断
  147. if( $data['end_time'] < time() ) return json_send(['code'=>'error','msg'=>'活动已结束','data'=>$data]);
  148. // 如果城市限制并且不在在城市范围内,不允许参加
  149. if ( $data['city_ids'] && !in_array($custom['city_id'],explode(',',$data['city_ids'])) ) return json_send(['code'=>'error','msg'=>'账号不在城市范围,请核对您的城市','data'=>['error'=>'不在标签范围内']]);
  150. // 判断用户ID是否在活动参与范围
  151. if( $data['custom_scope'] && !in_array($uid,explode(',',$data['custom_scope']) ) ) return json_send(['code'=>'error','msg'=>'账号不在指定客户范围','data'=>['error'=>'不在客户范围内']]);
  152. // 判断用户手机是否在活动参与范围
  153. if( $data['phone_scope'] && !in_array($custom['phone'],explode(',',$data['phone_scope']) ) ) return json_send(['code'=>'error','msg'=>'账号不在指定手机号范围','data'=>['error'=>'不在手机号范围内']]);
  154. // 如果限制新老用户参与
  155. if( $data['custom_type'] ) {
  156. // 如果客户注册时间不在活动期间
  157. if( $data['custom_start_time'] > $custom['insert_time'] || $data['custom_end_time'] < $custom['insert_time'] ) return json_send(['code'=>'error','msg'=>'账号不在活动范围','data'=>['error'=>'不在活动时间内注册']]);
  158. }
  159. // 如果限制下单
  160. if( $data['is_order'] ) {
  161. // 查询结果
  162. $query = $OrdersProduct->query()->where([['custom_uid','=',$uid],['status','=',1],['insert_time','>=',$data['order_start_time']],['insert_time','<=',$data['order_end_time']]]);
  163. // 如果商品存在
  164. if( $data['product_scope'] ) $query = $query->whereIn('product_id',explode(',',$data['product_scope']));
  165. // 获取时间段内下单数量
  166. $orderTotal = $query->groupBy('order_id')->count();
  167. // 如果没有下单数据
  168. if( !$orderTotal ) return json_send(['code'=>'error','msg'=>'不符合下单条件','data'=>['error'=>'不符合订单条件']]);
  169. }
  170. // 是否需要获取用户标签
  171. $tags = $data['tag_scope'] || $data['tag_except'] ? $WeiBanTags->getListByWeibanExtid($custom['weiban_extid']) : [];
  172. // 获取用户的标签
  173. $tags = $tags ? array_column($tags,'name') : [];
  174. // 判断是不是可以参与
  175. if( $data['tag_scope'] ) {
  176. // 解析数组
  177. $data['tag_scope'] = explode(',',$data['tag_scope']);
  178. // 计算交集
  179. $intersect = count(array_intersect($data['tag_scope'],$tags));
  180. // 判断标签是否必须同时满足
  181. if( $data['tag_scope_type'] == 1 ) {
  182. // 判断用户的标签是否全部满足于标签限制范围内
  183. if( $intersect != count($data['tag_scope']) ) return json_send(['code'=>'error','msg'=>'账号不在标签范围','data'=>['error'=>'不在标签范围内']]);
  184. }else{
  185. // 判断标签限制并且不在标签限制范围内,不允许参加
  186. if( !$intersect ) return json_send(['code'=>'error','msg'=>'账号不在标签范围','data'=>['error'=>'不在标签范围内']]);
  187. }
  188. }
  189. // 判断是不是可以参与
  190. if( $data['tag_except'] ) {
  191. // 解析数组
  192. $data['tag_except'] = explode(',',$data['tag_except']);
  193. // 计算交集
  194. $intersect = count(array_intersect($data['tag_except'],$tags));
  195. // 如果存在交集,在排除范围,即不可参与
  196. if( $intersect ) return json_send(['code'=>'error','msg'=>'账号在不可参与范围','data'=>['error'=>'账号在标签排除范围内']]);
  197. }
  198. // 奖品
  199. $reward = $ActiveReward->getListByLottery($data['id']);
  200. // 活动暂无奖品
  201. if( !$reward ) return json_send(['code'=>'error','msg'=>'活动暂未配置奖品','data'=>$data]);
  202. // 查询条件
  203. $map = [['insert_time','>=',$data['start_time']],['insert_time','<=',$data['end_time']]];
  204. // 判断周期
  205. if( !empty($data['freq']) ) {
  206. if( $data['freq'] == 1 ) $map = [['insert_time','>=',now()->startOfDay()->getTimestamp()],['insert_time','<=',now()->endOfDay()->getTimestamp()]];
  207. if( $data['freq'] == 2 ) $map = [['insert_time','>=',now()->startOfWeek()->getTimestamp()],['insert_time','<=',now()->endOfWeek()->getTimestamp()]];
  208. if( $data['freq'] == 3 ) $map = [['insert_time','>=',now()->startOfMonth()->getTimestamp()],['insert_time','<=',now()->endOfMonth()->getTimestamp()]];
  209. }
  210. // 查询用户已参与次数
  211. $data['join_num'] = (int)$ActiveRecord->query()->where([['custom_uid','=',$uid],['active_id','=',$activeId]])->where($map)->count();
  212. // 计算剩余次数
  213. $data['join_num'] = $data['lucky_num'] - $data['join_num'];
  214. // 如果次数不够
  215. if( $data['join_num'] <= 0 ) return json_send(['code'=>'error','msg'=>'抽奖次数已用完','data'=>['error'=>'抽奖次数已用完']]);
  216. // 抽奖记录
  217. $record = ['custom_uid'=>$uid,'active_id'=>$activeId,'reward_id'=>0,'money'=>0,'status'=>8];
  218. // 开启事务
  219. DB::beginTransaction();
  220. try{
  221. // 限制中奖则获取中奖次数
  222. $rewarTotal = $data['max_reward'] ? $ActiveRecord->query()->where([['active_id','=',$data['id']],['custom_uid','=',$uid],['reward_id','>',0]])->where($map)->count() : 0;
  223. // 中奖上限以后不再中奖
  224. $rewardIndex = ($data['max_reward'] && $rewarTotal >= $data['max_reward']) ? 0 : $ActiveReward->getRewardResult($reward);
  225. // 如果中奖,下标不是0
  226. if( $rewardIndex ) {
  227. // 获取奖品
  228. $rewardResult = $reward[$rewardIndex];
  229. // 奖品记录ID
  230. if( !empty($rewardResult['id']) ){
  231. // 是否直接发放
  232. if( $data['is_arrive'] ) {
  233. // 发放红包
  234. $result = $CustomAmount->trade($uid,$activeId,$rewardResult['money'],8,1,'红包活动');
  235. // 发放失败,改为未中奖
  236. if( isset($result['error']) ) $rewardIndex = 0;
  237. // 发放成功,状态为已完成
  238. $record['status'] = 8;
  239. }else{
  240. // 发放状态为进行中
  241. $record['status'] = 1;
  242. }
  243. }
  244. }
  245. // 中奖记录奖品
  246. if( $rewardIndex ) {
  247. // 记录,默认状态为1,进行中
  248. $record['reward_id'] = $reward[$rewardIndex]['id'];
  249. $record['money'] = $reward[$rewardIndex]['money'];
  250. // 扣减次数
  251. $ActiveReward->edit($rewardResult['id'],['reward_total'=>DB::raw('reward_total+-1')]);
  252. }
  253. // 写入记录
  254. $result = $ActiveRecord->add($record);
  255. // 写入失败
  256. if( !$result ) {
  257. // 回退数据
  258. DB::rollBack();
  259. // 提示
  260. return json_send(['code'=>'error','msg'=>'抽奖记录失败,请重试']);
  261. }
  262. // 提交事务
  263. DB::commit();
  264. // 通过活动ID,查询奖品
  265. $rewardList = [];
  266. // 奖品数据
  267. foreach ($reward as $value) {
  268. $rewardList[] = ['id'=>$value['id'],'money'=>$value['money']];
  269. }
  270. // 默认可以参加活动
  271. $data['allow_join'] = 1;
  272. // 返回结果
  273. return json_send(['code'=>'success','msg'=>'抽奖成功','data'=>['reward_list'=>$rewardList,'reward_index'=>$rewardIndex,'join_num'=>$data['join_num']-1]]);
  274. // 异常处理
  275. } catch (\Throwable $th) {
  276. // 回退数据
  277. DB::rollBack();
  278. // 下单失败提示
  279. return json_send(['code'=>'error','msg'=>'抽奖失败,请重试','data'=>['error'=>$th->getMessage().$th->getLine()]]);
  280. }
  281. }
  282. }