截流自动化的商城平台
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

CouponLogic.php 18KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454
  1. <?php
  2. // +----------------------------------------------------------------------
  3. // | likeshop开源商城系统
  4. // +----------------------------------------------------------------------
  5. // | 欢迎阅读学习系统程序代码,建议反馈是我们前进的动力
  6. // | gitee下载:https://gitee.com/likeshop_gitee
  7. // | github下载:https://github.com/likeshop-github
  8. // | 访问官网:https://www.likeshop.cn
  9. // | 访问社区:https://home.likeshop.cn
  10. // | 访问手册:http://doc.likeshop.cn
  11. // | 微信公众号:likeshop技术社区
  12. // | likeshop系列产品在gitee、github等公开渠道开源版本可免费商用,未经许可不能去除前后端官方版权标识
  13. // | likeshop系列产品收费版本务必购买商业授权,购买去版权授权后,方可去除前后端官方版权标识
  14. // | 禁止对系统程序代码以任何目的,任何形式的再发布
  15. // | likeshop团队版权所有并拥有最终解释权
  16. // +----------------------------------------------------------------------
  17. // | author: likeshop.cn.team
  18. // +----------------------------------------------------------------------
  19. namespace app\shop\logic\coupon;
  20. use app\common\model\user\User;
  21. use app\common\server\UrlServer;
  22. use app\common\basics\Logic;
  23. use think\facade\Db;
  24. use app\common\model\coupon\Coupon;
  25. use app\common\model\coupon\CouponGoods;
  26. use app\common\model\coupon\CouponList;
  27. use app\common\model\user\UserLevel;
  28. class CouponLogic extends Logic{
  29. public static function lists($get){
  30. // 下架本店过了发放时间的优惠券
  31. $now = time();
  32. $updateWhere = [
  33. ['send_time_start','<',$now],
  34. ['send_time_end','<',$now],
  35. ['status','=',1],
  36. ['shop_id', '=', $get['shop_id']]
  37. ];
  38. Coupon::where($updateWhere)
  39. ->update(['status'=>0,'update_time'=>$now]);
  40. $where = [
  41. ['del','=',0],
  42. ['shop_id','=', $get['shop_id']]
  43. ];
  44. if(empty($get['type'])) {
  45. // 已下架
  46. $where[] = ['status', '=', '0'];
  47. }else{
  48. $where[] = ['status', '=', '1'];
  49. }
  50. // 名称
  51. if(isset($get['name']) && !empty($get['name'])) {
  52. $where[] = ['name', 'like', '%'.trim($get['name']).'%' ];
  53. }
  54. // 领取方式
  55. if(isset($get['get_type']) && !empty($get['get_type'])) {
  56. $where[] = ['get_type', '=', $get['get_type'] ];
  57. }
  58. // 创建时间
  59. if(isset($get['start_time']) && !empty($get['start_time'])) {
  60. $where[] = ['create_time', '>=', strtotime($get['start_time']) ];
  61. }
  62. if(isset($get['end_time']) && !empty($get['end_time'])) {
  63. $where[] = ['create_time', '<=', strtotime($get['end_time']) ];
  64. }
  65. $coupon_count = Coupon::where($where)->count();
  66. $coupon_list = Coupon::field('id,name,money,use_goods_type,use_goods_type as use_goods_type_desc,condition_type,condition_money,condition_type as condition_type_desc,send_total_type,send_total_type as send_total_type_desc,send_total,get_type,get_type as get_type_desc,status,status as statusDesc,send_time_start,send_time_end,send_time_start as send_time,use_time_type,use_time_start,use_time_end,use_time,use_time_type as use_time_desc,create_time')
  67. ->where($where)
  68. ->page($get['page'], $get['limit'])
  69. ->order('id desc')
  70. ->select()
  71. ->toArray();
  72. return ['count' => $coupon_count, 'list' => $coupon_list];
  73. }
  74. //新增优惠券
  75. public static function add($post){
  76. $time = time();
  77. //拼接数据
  78. $add_data = [
  79. 'shop_id' => $post['shop_id'],
  80. 'name' => trim($post['name']),
  81. 'money' => $post['money'],
  82. 'send_time_start' => strtotime($post['send_time_start']),
  83. 'send_time_end' => strtotime($post['send_time_end']),
  84. 'send_total_type' => $post['send_total_type'],
  85. 'send_total' => $post['send_total_type'] == 2 ? $post['send_total'] : '',
  86. 'condition_type' => $post['condition_type'],
  87. 'condition_money' => $post['condition_type'] == 2 ? $post['condition_money'] : '',
  88. 'use_time_type' => $post['use_time_type'],
  89. 'use_time_start' => $post['use_time_type'] == 1 ? strtotime($post['use_time_start']) : '',
  90. 'use_time_end' => $post['use_time_type'] == 1 ? strtotime($post['use_time_end']) : '',
  91. 'use_time' => $post['use_time_type'] == 2 ? $post['use_time'] : '',
  92. 'get_type' => $post['get_type'],
  93. 'get_num_type' => $post['get_num_type'],
  94. 'get_num' => $post['get_num'],
  95. 'use_goods_type' => $post['use_goods_type'],
  96. 'status' => $post['status'],
  97. 'create_time' => $time,
  98. 'update_time' => $time,
  99. ];
  100. //用券时间
  101. if($post['use_time_type'] == 3){
  102. $update_data['use_time'] = $post['tomorrow_use_time'];
  103. }
  104. //领取次数
  105. if($post['get_num_type'] == 3){
  106. $update_data['get_num'] = $post['day_get_num'];
  107. }
  108. //提交订单
  109. Db::startTrans();
  110. try {
  111. $coupon = Coupon::create($add_data);
  112. if($coupon && $coupon['use_goods_type'] != 1){
  113. $goods_coupon = [];
  114. $now = time();
  115. foreach ($post['goods_ids'] as $item){
  116. $goods_coupon[] = [
  117. 'coupon_id' => $coupon->id,
  118. 'goods_id' => $item,
  119. 'create_time' => $now,
  120. ];
  121. }
  122. $couponGoods = new CouponGoods();
  123. $couponGoods->saveAll($goods_coupon);
  124. }
  125. Db::commit();
  126. return true;
  127. } catch (\Exception $e) {
  128. Db::rollback();
  129. self::$error = $e->getMessage();
  130. return false;
  131. }
  132. }
  133. public static function edit($post){
  134. //拼接数据
  135. $update_data = [
  136. 'shop_id' => $post['shop_id'],
  137. 'name' => $post['name'],
  138. 'money' => $post['money'],
  139. 'send_time_start' => strtotime($post['send_time_start']),
  140. 'send_time_end' => strtotime($post['send_time_end']),
  141. 'send_total_type' => $post['send_total_type'],
  142. 'send_total' => $post['send_total_type'] == 2 ? $post['send_total'] : '',
  143. 'condition_type' => $post['condition_type'],
  144. 'condition_money' => $post['condition_type'] == 2 ? $post['condition_money'] : '',
  145. 'use_time_type' => $post['use_time_type'],
  146. 'use_time_start' => $post['use_time_type'] == 1 ? strtotime($post['use_time_start']) : '',
  147. 'use_time_end' => $post['use_time_type'] == 1 ? strtotime($post['use_time_end']) : '',
  148. 'use_time' => $post['use_time_type'] == 2 ? $post['use_time'] : '',
  149. 'get_type' => $post['get_type'],
  150. 'get_num_type' => $post['get_num_type'],
  151. 'get_num' => $post['get_num_type'] == 2 ? $post['get_num'] : '',
  152. 'use_goods_type' => $post['use_goods_type'],
  153. 'status' => $post['status'],
  154. 'update_time' => time()
  155. ];
  156. //用券时间
  157. if($post['use_time_type'] == 3){
  158. $update_data['use_time'] = $post['tomorrow_use_time'];
  159. }
  160. //领取次数
  161. if($post['get_num_type'] == 3){
  162. $update_data['get_num'] = $post['day_get_num'];
  163. }
  164. Db::startTrans();
  165. try {
  166. $coupon = Coupon::update($update_data,['id'=>$post['id']]);
  167. CouponGoods::where(['coupon_id'=>$post['id']])->delete();
  168. if($coupon && $coupon['use_goods_type'] != 1){
  169. $goods_coupon = [];
  170. $now = time();
  171. foreach ($post['goods_ids'] as $item){
  172. $goods_coupon[] = [
  173. 'coupon_id' => $post['id'],
  174. 'goods_id' => $item,
  175. 'create_time' => $now,
  176. ];
  177. }
  178. $couponGoods = new CouponGoods();
  179. $couponGoods->saveAll($goods_coupon);
  180. }
  181. Db::commit();
  182. return true;
  183. } catch (\Exception $e) {
  184. Db::rollback();
  185. self::$error = $e->getMessage();
  186. return false;
  187. }
  188. }
  189. public static function getCoupon($id,$get_data = false){
  190. $coupon = Coupon::findOrEmpty($id);
  191. $coupon['goods_coupon'] = [];
  192. if($get_data) {
  193. $coupon = $coupon->getData();
  194. $coupon['send_time_start'] = date('Y-m-d H:i:s',$coupon['send_time_start']);
  195. $coupon['send_time_end'] = date('Y-m-d H:i:s',$coupon['send_time_end']);
  196. if($coupon['use_goods_type'] != 1){ // 非全部商品
  197. $goods_coupon= CouponGoods::alias('cg')
  198. ->join('goods g','cg.goods_id = g.id')
  199. ->where(['coupon_id'=>$id])
  200. ->field('g.id,name,max_price,min_price,stock')
  201. ->select();
  202. foreach ($goods_coupon as &$item){
  203. $item['price'] = '¥'.$item['min_price'].'~'.'¥'.$item['max_price'];
  204. if($item['max_price'] == $item['min_price']){
  205. $item['price'] = '¥'.$item['min_price'];
  206. }
  207. }
  208. $coupon['goods_coupon'] = $goods_coupon;
  209. }
  210. if($coupon['use_time_start']){
  211. $coupon['use_time_start'] = date('Y-m-d H:i:s',$coupon['use_time_start']);
  212. $coupon['use_time_end'] = date('Y-m-d H:i:s',$coupon['use_time_end']);
  213. }
  214. }
  215. return $coupon;
  216. }
  217. /*
  218. * 发放记录
  219. */
  220. public static function log($get){
  221. $where[] = ['cl.del','=',0];
  222. $where[] = ['cl.coupon_id','=',$get['id']];
  223. if(isset($get['keyword']) && $get['keyword']){
  224. switch($get['search_type']) {
  225. case 'sn';
  226. $where[] = ['u.sn', '=', $get['keyword']];
  227. break;
  228. case 'nickname';
  229. $where[] = ['u.nickname', '=', $get['keyword']];
  230. break;
  231. case 'mobile';
  232. $where[] = ['u.mobile', '=', $get['keyword']];
  233. break;
  234. }
  235. }
  236. if(isset($get['status']) && $get['status'] != '') {
  237. $where[] = ['cl.status', '=', $get['status']];
  238. }
  239. $log_count = CouponList::alias('cl')
  240. ->join('user u','cl.user_id = u.id')
  241. ->where($where)
  242. ->count();
  243. $log_list = CouponList::alias('cl')
  244. ->join('user u','cl.user_id = u.id')
  245. ->where($where)
  246. ->field('cl.coupon_id,cl.status as cl_status,coupon_code,cl.create_time as cl_create_time,cl.use_time,u.nickname,u.avatar,u.mobile,u.sn,u.level')
  247. ->page($get['page'], $get['limit'])
  248. ->select();
  249. $coupon_list = Coupon::where(['del'=>0])->column('name','id');
  250. $level_name = UserLevel::where(['del'=>0])->column('name','id');
  251. foreach ($log_list as &$item)
  252. {
  253. $item['coupon_name'] = $coupon_list[$item['coupon_id']] ?? '';
  254. $item['avatar'] = UrlServer::getFileUrl($item['avatar']);
  255. $item['level_name'] = $level_name[$item['level']] ?? '';
  256. $item['status_desc'] = $item['cl_status'] ? '已使用' : '未使用';
  257. $item['cl_create_time'] = date('Y-m-d H:i:s',$item['cl_create_time']);
  258. $item['use_time_desc'] = $item['use_time'] ? date('Y-m-d H:i:s',$item['use_time']) : '';
  259. }
  260. return ['count'=>$log_count , 'lists'=>$log_list];
  261. }
  262. public static function changeStatus($id)
  263. {
  264. try{
  265. $coupon = Coupon::findOrEmpty($id);
  266. if ($coupon['send_time_end'] <= time() && $coupon['status'] == 0) {
  267. throw new \Exception('已过发放时间,无法上架');
  268. }
  269. $coupon->status = $coupon->status ? 0 : 1;
  270. $coupon->update_time = time();
  271. $coupon->save();
  272. return true;
  273. }catch(\Exception $e) {
  274. self::$error = $e->getMessage();
  275. return false;
  276. }
  277. }
  278. /*
  279. * 删除优惠券 删除已领取用户的优惠券
  280. */
  281. public static function del($id){
  282. Db::startTrans();
  283. try{
  284. $time = time();
  285. // 优惠券主表
  286. Coupon::update([
  287. 'id' => $id,
  288. 'del' => 1,
  289. 'update_time' => $time
  290. ]);
  291. // 已发放的优惠券
  292. CouponList::where(['coupon_id'=>$id])->update([
  293. 'del' => 1,
  294. 'update_time' => $time
  295. ]);
  296. Db::commit();
  297. return true;
  298. }catch(\Exception $e) {
  299. Db::rollback();
  300. self::$error = $e->getMessage();
  301. return false;
  302. }
  303. }
  304. /**
  305. * 发放优惠券
  306. */
  307. public static function sendCoupon($post){
  308. $user_ids = $post['user_ids'];
  309. $coupon_ids = $post['coupon_ids'];
  310. $coupon_list = Coupon::where('id', 'in', $coupon_ids)->column('*', 'id');
  311. $user_list = User::where('id', 'in', $user_ids)->column('id,nickname', 'id');
  312. // 判断是否会超发
  313. foreach($coupon_ids as $coupon_id) {
  314. $coupon = $coupon_list[$coupon_id];
  315. if($coupon['send_total_type'] == 2) { // 限制数量的券
  316. $already_issue = CouponList::where([
  317. ['coupon_id','=',$coupon_id],
  318. ['del','=',0]
  319. ])->count(); // 已发放数量
  320. $target_issue = count($user_ids) + $already_issue; // 目标发放总数(若能正常发放的话)
  321. if($target_issue > $coupon['send_total']) {
  322. self::$error = $coupon['name'].'的发放数量已达到限制';
  323. return false;
  324. }
  325. }
  326. }
  327. // 判断用户是否超出领取数量
  328. foreach($coupon_ids as $coupon_id) {
  329. $coupon = $coupon_list[$coupon_id];
  330. if($coupon['get_num_type'] == 2) { // 限制次数
  331. foreach($user_ids as $user_id) {
  332. $count = CouponList::where([
  333. 'user_id' => $user_id,
  334. 'coupon_id' => $coupon_id,
  335. 'del' => 0,
  336. ])->count();
  337. if($count >= $coupon['get_num']) {
  338. self::$error = $user_list[$user_id]['nickname'].'已超出领取的限制次数';
  339. return false;
  340. }
  341. }
  342. }
  343. if($coupon['get_num_type'] == 3) { // 每天领取次数限制
  344. $timeStr = date('Y-m-d', time());
  345. $today = strtotime($timeStr.' 00:00:00');
  346. foreach($user_ids as $user_id) {
  347. $count = CouponList::where([
  348. ['user_id', '=', $user_id],
  349. ['coupon_id', '=', $coupon_id],
  350. ['del', '=', 0],
  351. ['create_time', '>', $today],
  352. ])->count();
  353. if($count >= $coupon['get_num']) {
  354. self::$error = $user_list[$user_id]['nickname'].'已超出每日领取的限制次数';
  355. return false;
  356. }
  357. }
  358. }
  359. }
  360. try{
  361. $addData = [];
  362. $time = time();
  363. foreach ($coupon_ids as $coupon_id){
  364. foreach ($user_ids as $user_id){
  365. $addData[] = [
  366. 'user_id' => $user_id,
  367. 'coupon_id' => $coupon_id,
  368. 'status' => 0,
  369. 'coupon_code' => create_coupon_code(),
  370. 'create_time' => $time,
  371. 'update_time' => $time
  372. ];
  373. }
  374. }
  375. // 批量添加
  376. $couponList = new CouponList();
  377. $couponList->saveAll($addData);
  378. return true;
  379. }catch(\Exception $e){
  380. self::$error = $e->getMessage();
  381. return false;
  382. }
  383. }
  384. public static function getShopCouponList($get)
  385. {
  386. $where = [
  387. ['del', '=', 0], // 未删除
  388. ['status', '=', 1], // 上架中
  389. ['get_type', '=', 2], // 商家赠送
  390. ['send_time_start', '<=', time()], // 开始发放时间早于或等于当前时间
  391. ['send_time_end', '>', time()], // 结束发放时间大于当前时间
  392. ['shop_id', '=', $get['shop_id']]
  393. ];
  394. if(isset($get['name']) && !empty($get['name'])) {
  395. $where[] = ['name', 'like', '%'.trim($get['name']).'%'];
  396. }
  397. if(isset($get['use_goods_type']) && !empty($get['use_goods_type'])) {
  398. $where[] = ['use_goods_type', '=', $get['use_goods_type']];
  399. }
  400. $lists = Coupon::field('id,name,send_time_start,send_time_end,send_time_start as send_time,money,get_type as get_type_desc,condition_type as condition_type_desc_two,condition_money,send_total_type as send_info,send_total,use_goods_type as use_goods_type_desc,use_time_type,use_time_start,use_time_end,use_time,use_time as use_time_desc,status as status_desc,create_time')
  401. ->where($where)
  402. ->page($get['page'], $get['limit'])
  403. ->order('id', 'desc')
  404. ->select()
  405. ->toArray();
  406. $count = Coupon::field('id,name,send_time_start,send_time_end,money,get_type,condition_type,send_total_type')
  407. ->where($where)
  408. ->count();
  409. return [
  410. 'count' => $count,
  411. 'lists' => $lists
  412. ];
  413. }
  414. }