截流自动化的商城平台
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.

ActivityLogic.php 15KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432
  1. <?php
  2. namespace app\shop\logic\team;
  3. use app\common\basics\Logic;
  4. use app\common\enum\GoodsEnum;
  5. use app\common\enum\OrderEnum;
  6. use app\common\enum\OrderLogEnum;
  7. use app\common\enum\TeamEnum;
  8. use app\common\logic\OrderRefundLogic;
  9. use app\common\model\bargain\Bargain;
  10. use app\common\model\goods\Goods;
  11. use app\common\model\order\Order;
  12. use app\common\model\seckill\SeckillGoods;
  13. use app\common\model\team\TeamActivity;
  14. use app\common\model\team\TeamFound;
  15. use app\common\model\team\TeamGoods;
  16. use app\common\model\team\TeamJoin;
  17. use Exception;
  18. use think\facade\Db;
  19. class ActivityLogic extends Logic
  20. {
  21. /**
  22. * @Notes: 获取拼团活动
  23. * @Author: 张无忌
  24. * @param $get
  25. * @param $shop_id
  26. * @return array|bool
  27. */
  28. public static function lists($get, $shop_id)
  29. {
  30. try {
  31. $where[] = ['T.shop_id', '=', $shop_id];
  32. $where[] = ['T.del', '=', 0];
  33. if (!empty($get['datetime']) and $get['datetime']) {
  34. list($start, $end) = explode(' - ', $get['datetime']);
  35. $where[] = ['T.create_time', '>=', strtotime($start.' 00:00:00')];
  36. $where[] = ['T.create_time', '<=', strtotime($end.' 23:59:59')];
  37. }
  38. if (!empty($get['name']) and $get['name']) {
  39. $where[] = ['G.name', 'like', '%'.$get['name'].'%'];
  40. }
  41. if (isset($get['status']) and is_numeric($get['status'])) {
  42. $where[] = ['T.status', '=', intval($get['status'])];
  43. }
  44. if (isset($get['type']) and $get['type']) {
  45. $where[] = ['T.audit', '=', $get['type']-1];
  46. }
  47. $model = new TeamActivity();
  48. $lists = $model->alias('T')->field(['T.*'])
  49. ->where($where)
  50. ->with(['goods'])
  51. ->join('goods G', 'G.id = T.goods_id')
  52. ->paginate([
  53. 'page' => $get['page'] ?? 1,
  54. 'list_rows' => $get['limit'] ?? 20,
  55. 'var_page' => 'page'
  56. ])->toArray();
  57. $teamFoundModel = new TeamFound();
  58. $teamJoinModel = new TeamJoin();
  59. foreach ($lists['data'] as &$item) {
  60. $item['activity_start_time'] = date('Y-m-d H:i', $item['activity_start_time']);
  61. $item['activity_end_time'] = date('Y-m-d H:i', $item['activity_end_time']);
  62. $item['status_text'] = TeamEnum::getTeamStatusDesc($item['status']);
  63. $item['audit_text'] = TeamEnum::getTeamAuditDesc($item['audit']);
  64. $item['team_count'] = $teamFoundModel->where(['team_activity_id'=>$item['id']])->count();
  65. $item['success_found'] = $teamFoundModel->where(['status'=>1, 'team_activity_id'=>$item['id']])->count();
  66. $item['join_found'] = $teamJoinModel->where(['team_activity_id'=>$item['id']])->count();
  67. }
  68. return ['count'=>$lists['total'], 'lists'=>$lists['data']];
  69. } catch (Exception $e) {
  70. static::$error = $e->getMessage();
  71. return false;
  72. }
  73. }
  74. /**
  75. * @Notes: 选择拼团商品
  76. * @Author: 张无忌
  77. * @param $get
  78. * @param $shop_id
  79. * @return array
  80. */
  81. public static function select($get, $shop_id)
  82. {
  83. try {
  84. $where = [];
  85. if (!empty($get['name']) and $get['name']) {
  86. $where[] = ['name', 'like', '%'.$get['name'].'%'];
  87. }
  88. $model = (new Goods());
  89. $lists = $model->field(['id,name,image,stock,max_price,min_price,market_price'])->where([
  90. ['shop_id', '=', $shop_id],
  91. ['audit_status', '=', 1],
  92. ['del', '=', 0],
  93. ['status', '=', 1],
  94. ['type', '=', GoodsEnum::TYPE_ACTUAL]
  95. ])->withAttr('is_activity', function ($value, $data) use($shop_id) {
  96. unset($value);
  97. // 是否是秒杀
  98. $seckill = (new SeckillGoods())->where(['goods_id'=>$data['id']])
  99. ->where(['del'=>0, 'shop_id'=>$shop_id])
  100. ->findOrEmpty()->toArray();
  101. if ($seckill) return '秒杀中';
  102. // 是否是砍价
  103. $bargain = (new Bargain())->where(['goods_id'=>$data['id']])
  104. ->where('del', '=', 0)
  105. ->where('shop_id', '=', $shop_id)
  106. ->findOrEmpty()->toArray();
  107. if ($bargain) return '砍价中';
  108. return '正常';
  109. })->where($where)->with(['GoodsItem'])
  110. ->append(['is_activity'])
  111. ->paginate([
  112. 'page' => $get['page'] ?? 1,
  113. 'list_rows' => $get['limit'] ?? 20,
  114. 'var_page' => 'page'
  115. ])->toArray();
  116. return ['count'=>$lists['total'], 'lists'=>$lists['data']];
  117. } catch (Exception $e) {
  118. return ['error'=>$e->getMessage()];
  119. }
  120. }
  121. /**
  122. * @Notes: 数据统计
  123. * @Author: 张无忌
  124. * @param $shop_id
  125. * @return mixed
  126. */
  127. public static function statistics($shop_id)
  128. {
  129. $where[] = ['del', '=', 0];
  130. $where[] = ['shop_id', '=', $shop_id];
  131. $model = new TeamActivity();
  132. $detail['total'] = $model->where($where)->count();
  133. $detail['stayAudit'] = $model->where($where)->where(['audit'=>0])->count();
  134. $detail['adoptAudit'] = $model->where($where)->where(['audit'=>1])->count();
  135. $detail['refuseAudit'] = $model->where($where)->where(['audit'=>2])->count();
  136. return $detail;
  137. }
  138. /**
  139. * @Notes: 拼团活动详细
  140. * @Author: 张无忌
  141. * @param $id
  142. * @return array
  143. */
  144. public static function detail($id)
  145. {
  146. $model = new TeamActivity();
  147. $detail = $model->field(true)
  148. ->with(['goods', 'teamGoods'])
  149. ->findOrEmpty($id)
  150. ->toArray();
  151. $detail['activity_start_time'] = date('Y-m-d H:i:s', $detail['activity_start_time']);
  152. $detail['activity_end_time'] = date('Y-m-d H:i:s', $detail['activity_end_time']);
  153. return $detail;
  154. }
  155. /**
  156. * @Notes: 新增拼团活动
  157. * @Author: 张无忌
  158. * @param $post
  159. * @param $shop_id
  160. * @return bool
  161. */
  162. public static function add($post, $shop_id)
  163. {
  164. Db::startTrans();
  165. try {
  166. $goods = (new Goods())->findOrEmpty($post['goods_id'])->toArray();
  167. if (!$goods) throw new \think\Exception('选择的商品不存在');
  168. if (strtotime($post['activity_start_time']) >= strtotime($post['activity_end_time'])) {
  169. throw new \think\Exception('团活动开始时间不能少于等于结束时间');
  170. }
  171. // 新增拼团活动信息
  172. $team = TeamActivity::create([
  173. 'shop_id' => $shop_id,
  174. 'goods_id' => $post['goods_id'],
  175. 'people_num' => $post['people_num'],
  176. 'share_title' => $post['share_title'] ?? '',
  177. 'share_intro' => $post['share_intro'] ?? '',
  178. 'team_max_price' => self::getMaxOrMinPrice($post)['max'],
  179. 'team_min_price' => self::getMaxOrMinPrice($post)['min'],
  180. 'audit' => 0,
  181. 'del' => 0,
  182. 'status' => $post['status'],
  183. 'effective_time' => $post['effective_time'],
  184. 'activity_start_time' => strtotime($post['activity_start_time']),
  185. 'activity_end_time' => strtotime($post['activity_end_time']),
  186. 'create_time' => time()
  187. ]);
  188. // 新增拼团商品规格
  189. $lists = [];
  190. foreach ($post['item'] as $key => $value) {
  191. foreach ($value as $K => $item) {
  192. $lists[] = [
  193. 'team_id' => $team['id'],
  194. 'goods_id' => $key,
  195. 'item_id' => $K,
  196. 'team_price' => $item
  197. ];
  198. }
  199. }
  200. if (!empty($lists)) (new TeamGoods())->insertAll($lists);
  201. Db::commit();
  202. return true;
  203. } catch (Exception $e) {
  204. Db::rollback();
  205. static::$error = $e->getMessage();
  206. return false;
  207. }
  208. }
  209. /**
  210. * @Notes: 编辑拼团活动
  211. * @Author: 张无忌
  212. * @param $post
  213. * @param $shop_id
  214. * @return bool
  215. */
  216. public static function edit($post, $shop_id)
  217. {
  218. Db::startTrans();
  219. try {
  220. $goods = (new Goods())->findOrEmpty($post['goods_id'])->toArray();
  221. if (!$goods) throw new \think\Exception('选择的商品不存在');
  222. if (strtotime($post['activity_start_time']) >= strtotime($post['activity_end_time'])) {
  223. throw new \think\Exception('团活动开始时间不能少于等于结束时间');
  224. }
  225. $activity = (new TeamActivity())->findOrEmpty($post['id'])->toArray();
  226. $audit = $activity['audit'] != 1 ? 0 : 1;
  227. // 编辑拼团活动信息
  228. TeamActivity::update([
  229. 'shop_id' => $shop_id,
  230. 'goods_id' => $post['goods_id'],
  231. 'people_num' => $post['people_num'],
  232. 'share_title' => $post['share_title'] ?? '',
  233. 'share_intro' => $post['share_intro'] ?? '',
  234. 'team_max_price' => self::getMaxOrMinPrice($post)['max'],
  235. 'team_min_price' => self::getMaxOrMinPrice($post)['min'],
  236. 'status' => $post['status'],
  237. 'audit' => $audit,
  238. 'effective_time' => $post['effective_time'],
  239. 'activity_start_time' => strtotime($post['activity_start_time']),
  240. 'activity_end_time' => strtotime($post['activity_end_time']),
  241. ], ['id'=>$post['id']]);
  242. // 删除旧的拼团商品
  243. (new TeamGoods())->where(['team_id'=>$post['id']])->delete();
  244. // 更新拼团商品规格
  245. $lists = [];
  246. foreach ($post['item'] as $key => $value) {
  247. foreach ($value as $K => $item) {
  248. $lists[] = [
  249. 'team_id' => $post['id'],
  250. 'goods_id' => $key,
  251. 'item_id' => $K,
  252. 'team_price' => $item
  253. ];
  254. }
  255. }
  256. if (!empty($lists)) (new TeamGoods())->insertAll($lists);
  257. Db::commit();
  258. return true;
  259. } catch (Exception $e) {
  260. Db::rollback();
  261. static::$error = $e->getMessage();
  262. return false;
  263. }
  264. }
  265. /**
  266. * @Notes: 删除拼团活动
  267. * @Author: 张无忌
  268. * @param $id
  269. * @return bool
  270. */
  271. public static function del($id)
  272. {
  273. try {
  274. TeamActivity::update([
  275. 'del' => 1,
  276. 'update_time' => time()
  277. ], ['id'=>$id]);
  278. return true;
  279. } catch (Exception $e) {
  280. static::$error = $e->getMessage();
  281. return false;
  282. }
  283. }
  284. /**
  285. * @Notes: 停止拼团活动
  286. * @Author: 张无忌
  287. * @param $id
  288. * @return bool
  289. */
  290. public static function stop($id)
  291. {
  292. try {
  293. TeamActivity::update([
  294. 'status' => 0,
  295. 'update_time' => time()
  296. ], ['id'=>$id]);
  297. $team_ids = (new TeamFound())->where(['team_activity_id'=>$id, 'status'=>0])->column('id');
  298. $teamJoin = (new TeamJoin())->alias('TJ')
  299. ->field(['TJ.*,O.order_sn,O.order_status,O.pay_status,O.refund_status,O.order_amount'])
  300. ->where('team_id', 'in', $team_ids)
  301. ->join('order O', 'O.id=TJ.order_id')
  302. ->select()->toArray();
  303. self::teamFail($teamJoin, $team_ids, time());
  304. return true;
  305. } catch (Exception $e) {
  306. static::$error = $e->getMessage();
  307. return false;
  308. }
  309. }
  310. /**
  311. * @Notes: 开启拼团活动
  312. * @Author: 张无忌
  313. * @param $id
  314. * @return bool
  315. */
  316. public static function open($id)
  317. {
  318. try {
  319. TeamActivity::update([
  320. 'status' => 1,
  321. 'update_time' => time()
  322. ], ['id'=>$id]);
  323. return true;
  324. } catch (Exception $e) {
  325. static::$error = $e->getMessage();
  326. return false;
  327. }
  328. }
  329. /**
  330. * @Notes: 查看最低活动价和最高活动价
  331. * @Author: 张无忌
  332. * @param $post
  333. * @return array
  334. */
  335. private static function getMaxOrMinPrice($post)
  336. {
  337. $team_price = [];
  338. foreach ($post['item'] as $key => $value) {
  339. foreach ($value as $K => $item) {
  340. array_push($team_price, $item);
  341. }
  342. }
  343. $team_max_price = !empty($team_price) ? max($team_price) : 0;
  344. $team_min_price = !empty($team_price) ? min($team_price) : 0;
  345. return [
  346. 'max' => $team_max_price,
  347. 'min' => $team_min_price
  348. ];
  349. }
  350. /**
  351. * @Notes: 拼团失败
  352. * @Author: 张无忌
  353. * @param $teamJoin (参团列表数据)
  354. * @param $found_ids
  355. * @param $time (时间)
  356. * @throws \think\Exception
  357. */
  358. private static function teamFail($teamJoin, $found_ids, $time)
  359. {
  360. Db::startTrans();
  361. try {
  362. (new TeamFound())->whereIn('id', $found_ids)
  363. ->update(['status'=>TeamEnum::TEAM_STATUS_FAIL, 'team_end_time'=>$time]);
  364. foreach ($teamJoin as $item) {
  365. TeamJoin::update(['status' => TeamEnum::TEAM_STATUS_FAIL, 'update_time' => $time], ['id' => $item['id']]);
  366. if ($item['order_status'] == OrderEnum::ORDER_STATUS_DOWN) continue;
  367. if ($item['refund_status'] != OrderEnum::REFUND_STATUS_NO_REFUND) continue;
  368. $order = (new Order())->findOrEmpty($item['order_id'])->toArray();
  369. // 取消订单
  370. OrderRefundLogic::cancelOrder($order['id'], OrderLogEnum::TYPE_SYSTEM);
  371. if ($order['pay_status'] == OrderEnum::PAY_STATUS_PAID) {
  372. // 更新订单状态
  373. OrderRefundLogic::cancelOrderRefundUpdate($order);
  374. // 订单退款
  375. OrderRefundLogic::refund($order, $order['order_amount'], $order['order_amount']);
  376. }
  377. }
  378. Db::commit();
  379. } catch (Exception $e) {
  380. Db::rollback();
  381. throw new \think\Exception($e->getMessage());
  382. }
  383. }
  384. }