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

TeamLogic.php 21KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495
  1. <?php
  2. namespace app\api\logic;
  3. use app\common\basics\Logic;
  4. use app\common\enum\GoodsEnum;
  5. use app\common\enum\OrderEnum;
  6. use app\common\enum\TeamEnum;
  7. use app\common\model\dev\DevRegion;
  8. use app\common\model\goods\Goods;
  9. use app\common\model\goods\GoodsItem;
  10. use app\common\model\order\Order;
  11. use app\common\model\order\OrderGoods;
  12. use app\common\model\order\OrderTrade;
  13. use app\common\model\shop\Shop;
  14. use app\common\model\team\TeamActivity;
  15. use app\common\model\team\TeamFound;
  16. use app\common\model\team\TeamJoin;
  17. use app\common\model\user\User;
  18. use Exception;
  19. use think\facade\Db;
  20. class TeamLogic extends Logic
  21. {
  22. /**
  23. * @Notes: 获取拼团活动
  24. * @Author: 张无忌
  25. * @param array $get
  26. * @return array|bool
  27. */
  28. public static function activity(array $get)
  29. {
  30. try {
  31. $pageNo = $get['page_no'] ?? 1;
  32. $pageSize = $get['page_size'] ?? 20;
  33. $model = new TeamActivity();
  34. $lists = $model->alias('T')->field([
  35. 'T.id,T.goods_id,T.people_num,T.team_max_price,T.team_min_price,sales_volume',
  36. 'G.name,G.image,G.max_price,G.min_price,G.market_price'
  37. ])
  38. ->where([
  39. ['T.audit', '=', 1],
  40. ['T.status', '=', 1],
  41. ['T.del', '=', 0],
  42. ['T.activity_start_time', '<=', time()],
  43. ['T.activity_end_time', '>=', time()],
  44. ['S.is_freeze', '=', 0],
  45. ['S.is_run', '=', 1],
  46. ])
  47. ->join('goods G', 'G.id = T.goods_id')
  48. ->join('shop S', 'S.id = T.shop_id')
  49. ->paginate([
  50. 'page' => $pageNo,
  51. 'list_rows' => $pageSize,
  52. 'var_page' => 'page'
  53. ])->toArray();
  54. return [
  55. 'list' => $lists['data'],
  56. 'count' => $lists['total'],
  57. 'more' => is_more($lists['total'], $pageNo, $pageSize),
  58. 'page_no' => $pageNo,
  59. 'page_size' => $pageSize
  60. ];
  61. } catch (Exception $e) {
  62. static::$error = $e->getMessage();
  63. return false;
  64. }
  65. }
  66. /**
  67. * @Notes: 开团信息
  68. * @Author: 张无忌
  69. * @param $post
  70. * @param $user_id
  71. * @return bool|array
  72. */
  73. public static function kaituanInfo($post, $user_id)
  74. {
  75. try {
  76. $teamActivity = (new TeamActivity())->alias('T')
  77. ->field([
  78. 'T.id as team_activity_id,T.shop_id,T.effective_time,GI.goods_id,GI.id as item_id,T.people_num,TG.team_price',
  79. 'G.name,G.image,GI.spec_value_str,GI.spec_value_ids,GI.market_price',
  80. 'GI.price,GI.stock,G.delivery_type,G.type'
  81. ])->where([
  82. ['T.goods_id', '=', (int)$post['goods_id']],
  83. ['T.audit', '=', 1],
  84. ['T.status', '=', 1],
  85. ['T.del', '=', 0],
  86. ['T.activity_start_time', '<=', time()],
  87. ['T.activity_end_time', '>=', time()],
  88. ['TG.goods_id', '=', (int)$post['goods_id']],
  89. ['TG.item_id', '=', (int)$post['item_id']],
  90. ])->join('team_goods TG', 'TG.team_id = T.id')
  91. ->join('goods G', 'G.id = TG.goods_id')
  92. ->join('goods_item GI', 'GI.id = TG.item_id')
  93. ->findOrEmpty()->toArray();
  94. if (!$teamActivity) throw new \think\Exception('当前商品未参与拼团活动,下次再来吧');
  95. if ($teamActivity['stock'] - intval($post['count']) < 0) throw new \think\Exception('抱歉,库存不足');
  96. if (empty($post['address_id'])) {
  97. $address = UserAddressLogic::getDefaultAddress($user_id);
  98. } else {
  99. $address = UserAddressLogic::getOneAddress($user_id, ['id'=>$post['address_id']]);
  100. }
  101. // 校验发票信息 返回以店铺id为键,原发票参数为值的数组
  102. $invoice = OrderInvoiceLogic::checkOrderInvoice($post);
  103. if (false === $invoice) {
  104. self::$error = OrderInvoiceLogic::getError();
  105. return false;
  106. }
  107. $user = (new User())->findOrEmpty($user_id)->toArray();
  108. $shop = (new Shop())->field(['id as shop_id,name as shop_name,open_invoice,spec_invoice,run_start_time,run_end_time,weekdays,province_id,city_id,district_id,address'])->findOrEmpty($teamActivity['shop_id'])->toArray();
  109. //配送方式
  110. if ($teamActivity['type'] == GoodsEnum::TYPE_VIRTUAL) {
  111. // 虚拟商品, 配送方式为 虚拟发货
  112. $shop['delivery_types_arr'][] = [
  113. 'delivery_type' => GoodsEnum::DELIVERY_VIRTUAL,
  114. 'delivery_type_text' => GoodsEnum::getDeliveryTypeDesc(GoodsEnum::DELIVERY_VIRTUAL),
  115. ];
  116. } else {
  117. // 快递和自提
  118. $delivery_type_arr = explode(',',$teamActivity['delivery_type']);
  119. foreach ($delivery_type_arr as $delivery_type_val) {
  120. $shop['delivery_types_arr'][] = [
  121. 'delivery_type' => $delivery_type_val,
  122. 'delivery_type_text' => GoodsEnum::getDeliveryTypeDesc($delivery_type_val),
  123. ];
  124. }
  125. }
  126. $shop['delivery_type'] = $post['delivery_type'];
  127. // $shop['delivery_type_text'] = Order::getDeliveryType($deliveryType);
  128. $teamActivity['count'] = $post['count'];
  129. // $teamActivity['delivery_type'] = $deliveryType;
  130. $teamActivity['delivery_types_arr'] = $shop['delivery_types_arr'];
  131. $shop['goods'][] = $teamActivity;
  132. // 运费
  133. $shipping_price = OrderLogic::calculateFreight([
  134. [
  135. 'goods_id' => $teamActivity['goods_id'],
  136. 'item_id' => $teamActivity['item_id'],
  137. 'num' => input('count/d'),
  138. 'delivery_type' => $shop['delivery_type'],
  139. ]
  140. ], $address);
  141. $shop['shipping_price'] = $shipping_price;
  142. $shop['total_amount'] = bcadd($teamActivity['team_price'] * $post['count'], $shipping_price, 2);
  143. // 经营信息
  144. $shop['run_start_time'] = $shop['run_start_time'] ? date('H:i:s', $shop['run_start_time']) : '';
  145. $shop['run_end_time'] = $shop['run_end_time'] ? date('H:i:s', $shop['run_end_time']) : '';
  146. // 店铺地址
  147. $shop['province'] = DevRegion::getAreaName($shop['province_id']);
  148. $shop['city'] = DevRegion::getAreaName($shop['city_id']);
  149. $shop['district'] = DevRegion::getAreaName($shop['district_id']);
  150. // 用户留言
  151. $userRemark = '';
  152. if (!empty($post['remark'])) {
  153. $post['remark'] = is_array($post['remark']) ? $post['remark'] : json_decode($post['remark'], JSON_UNESCAPED_UNICODE);
  154. $userRemark = !empty($post['remark'][0]['remark']) ? $post['remark'][0]['remark'] : '';
  155. }
  156. return [
  157. 'team_id' => $post['team_id'] ?? 0,
  158. 'pay_way' => $post['pay_way'] ?? 1,
  159. 'order_type' => OrderEnum::TEAM_ORDER,
  160. 'total_amount' => bcadd($teamActivity['team_price'] * $post['count'], $shipping_price, 2),
  161. 'total_count' => intval($post['count']),
  162. 'shipping_price' => $shipping_price,
  163. 'user_money' => $user['user_money'],
  164. 'remark' => $userRemark,
  165. 'address' => $address,
  166. 'shop' => $shop,
  167. 'invoice' => array_values($invoice)
  168. ];
  169. } catch (Exception $e) {
  170. static::$error = $e->getMessage();
  171. return false;
  172. }
  173. }
  174. /**
  175. * @Notes: 发起开团/参团
  176. * @Author: 张无忌
  177. * @param $info
  178. * @param $user_id
  179. * @return bool|array
  180. */
  181. public static function kaituan($info, $userInfo)
  182. {
  183. Db::startTrans();
  184. try {
  185. $time = time();
  186. $teamGoods = $info['shop']['goods'][0];
  187. // 参团验证
  188. if ($info['team_id']) {
  189. $teamFound = (new TeamFound())->where(['id'=>$info['team_id']])->findOrEmpty()->toArray();
  190. if (!$teamFound) throw new \think\Exception('选择的团不存在');
  191. if ($teamFound['status'] != 0) throw new \think\Exception('当前拼团已结束,请重新选择拼团');
  192. if ($teamFound['invalid_time'] <= time()) throw new \think\Exception('当前拼团已结束,请重新选择拼团');
  193. if ($teamFound['user_id'] == $userInfo['id']) throw new \think\Exception('您已是该团成员了,不能重复参团哦!');
  194. if ($teamFound['people'] == $teamFound['join']) throw new \think\Exception('当前拼团已满员,请重新选择拼团!');
  195. // 获取已参团记录
  196. $people = (new TeamJoin())->where(['team_id'=>$info['team_id'], 'user_id'=>$userInfo['id']])->findOrEmpty()->toArray();
  197. if ($people) throw new \think\Exception('您已是该团成员了,不能重复参团哦!');
  198. }
  199. //配送方式 转换
  200. $info['shop']['delivery_type'] = OrderEnum::getChangeDeliveryTypeItem($info['shop']['delivery_type']);
  201. // 验证收货地址
  202. if ($info['shop']['delivery_type'] == OrderEnum::DELIVERY_TYPE_EXPRESS && empty($info['address'])) {
  203. throw new \think\Exception('请选择收货地址');
  204. }
  205. // 校验发票信息 返回以店铺id为键,原发票参数为值的数组
  206. $invoice = OrderInvoiceLogic::checkOrderInvoice($info, 'team');
  207. if (false === $invoice) {
  208. throw new \think\Exception(OrderInvoiceLogic::getError());
  209. }
  210. //线下自提-提货码
  211. if ($info['shop']['delivery_type'] == OrderEnum::DELIVERY_TYPE_SELF) {
  212. $pickup_code = create_rand_number('order', 'pickup_code', 6);
  213. }
  214. // 创建交易单
  215. $trade = OrderTrade::create([
  216. 't_sn' => createSn('order_trade', 't_sn'),
  217. 'user_id' => $userInfo['id'],
  218. 'goods_price' => $info['total_amount'],
  219. 'order_amount' => $info['total_amount'],
  220. 'total_amount' => $info['total_amount'],
  221. 'shop_id' => $info['shop']['shop_id'],
  222. 'create_time' => $time
  223. ]);
  224. // 创建订单
  225. $order = Order::create([
  226. 'trade_id' => $trade['id'],
  227. 'shop_id' => $info['shop']['shop_id'],
  228. 'order_sn' => createSn('order', 'order_sn'),
  229. 'user_id' => $userInfo['id'],
  230. 'order_type' => OrderEnum::TEAM_ORDER,
  231. 'order_source' => $userInfo['client'],
  232. 'delivery_type' => $info['shop']['delivery_type'],
  233. 'pay_way' => $info['pay_way'],
  234. 'consignee' => $info['address']['contact'] ?? '',
  235. 'province' => $info['address']['province_id'] ?? 0,
  236. 'city' => $info['address']['city_id'] ?? 0,
  237. 'district' => $info['address']['district_id'] ?? 0,
  238. 'address' => $info['address']['address'] ?? '',
  239. 'mobile' => $info['address']['telephone'] ?? '',
  240. 'goods_price' => $info['total_amount'],
  241. 'total_amount' => $info['total_amount'],
  242. 'order_amount' => $info['total_amount'],
  243. 'total_num' => $info['total_count'],
  244. 'shipping_price' => $info['shipping_price'],
  245. 'discount_amount' => 0,
  246. 'user_remark' => $info['remark'] ?? '',
  247. 'create_time' => $time,
  248. 'pickup_code' => $pickup_code ?? null
  249. ]);
  250. // 创建订单商品
  251. OrderGoods::create([
  252. 'order_id' => $order['id'],
  253. 'goods_id' => $teamGoods['goods_id'],
  254. 'item_id' => $teamGoods['item_id'],
  255. 'goods_num' => $teamGoods['count'],
  256. 'goods_name' => $teamGoods['name'],
  257. 'goods_price' => $teamGoods['team_price'],
  258. 'total_price' => bcadd($teamGoods['team_price'] * $teamGoods['count'], 0, 2),
  259. 'total_pay_price' => bcadd($teamGoods['team_price'] * $teamGoods['count'], 0, 2),
  260. 'discount_price' => 0,
  261. 'spec_value' => $teamGoods['spec_value_str'],
  262. 'spec_value_ids' => $teamGoods['spec_value_ids'],
  263. 'image' => $teamGoods['image'],
  264. 'shop_id' => $info['shop']['shop_id']
  265. ]);
  266. // 增加发票
  267. OrderInvoiceLogic::insertOrderInvoice($info['shop']['shop_id'], $userInfo['id'], $order['id'], $invoice);
  268. // 开新团
  269. $team_id = 0;
  270. if (!$info['team_id']) {
  271. $teamFound = TeamFound::create([
  272. 'shop_id' => $info['shop']['shop_id'],
  273. 'team_activity_id' => $teamGoods['team_activity_id'],
  274. 'team_sn' => createSn('team_found', 'team_sn'),
  275. 'user_id' => $userInfo['id'],
  276. 'status' => 0,
  277. 'join' => 0,
  278. 'people' => $teamGoods['people_num'],
  279. 'goods_snap' => json_encode([
  280. 'id' => $teamGoods['goods_id'],
  281. 'shop_id' => $teamGoods['shop_id'],
  282. 'name' => $teamGoods['name'],
  283. 'image' => $teamGoods['image']
  284. ]),
  285. 'kaituan_time' => $time,
  286. 'invalid_time' => ($teamGoods['effective_time'] * 60 * 60) + time()
  287. ]);
  288. $team_id = $teamFound['id'];
  289. }
  290. // 加入团
  291. TeamJoin::create([
  292. 'shop_id' => $info['shop']['shop_id'],
  293. 'team_activity_id' => $teamGoods['team_activity_id'],
  294. 'team_id' => $team_id ?: $info['team_id'],
  295. 'sn' => createSn('team_join', 'sn'),
  296. 'user_id' => $userInfo['id'],
  297. 'order_id' => $order['id'],
  298. 'identity' => $info['team_id'] ? 2 : 1,
  299. 'team_snap' => json_encode($info['shop']['goods'][0], JSON_UNESCAPED_UNICODE),
  300. 'create_time' => $time,
  301. 'update_time' => $time
  302. ]);
  303. // 扣减库存
  304. (new GoodsItem())->where([
  305. 'goods_id'=>$teamGoods['goods_id'],
  306. 'id' =>$teamGoods['item_id']
  307. ])->update(['stock' => ['dec', $teamGoods['count']]]);
  308. (new Goods())->where([
  309. 'id'=>$teamGoods['goods_id']
  310. ])->update(['stock' => ['dec', $teamGoods['count']]]);
  311. // 更新参数人数
  312. TeamFound::update([
  313. 'join' => ['inc', 1]
  314. ], ['id'=>$team_id ?: $info['team_id']]);
  315. // 更新活动拼团数
  316. TeamActivity::update([
  317. 'sales_volume' => ['inc', 1]
  318. ], ['id'=>$teamGoods['team_activity_id']]);
  319. Db::commit();
  320. return [
  321. 'team_id' => $team_id ?: $info['team_id'],
  322. 'type' => 'trade',
  323. 'trade_id' => $trade['id'],
  324. 'order_id' => $order['id']
  325. ];
  326. } catch (Exception $e) {
  327. Db::rollback();
  328. static::$error = $e->getMessage();
  329. return false;
  330. }
  331. }
  332. /**
  333. * @Notes: 获取拼团记录
  334. * @Author: 张无忌
  335. * @param $get
  336. * @param $user_id
  337. * @return array|bool
  338. */
  339. public static function record($get, $user_id)
  340. {
  341. try {
  342. $pageNo = $get['page_no'] ?? 1;
  343. $pageSize = $get['page_size'] ?? 20;
  344. $where = [];
  345. if (isset($get['type']) and $get['type'] >= 0) {
  346. $type = intval($get['type']);
  347. $where[] = ['TJ.status', '=', $type];
  348. }
  349. $model = new TeamJoin();
  350. $lists = $model->alias('TJ')->field(['TJ.*,S.name as shop_name,O.order_amount'])
  351. ->where(['TJ.user_id'=>$user_id])
  352. ->order('id desc')
  353. ->where($where)
  354. ->join('Shop S', 'S.id = TJ.shop_id')
  355. ->join('order O', 'O.id = TJ.order_id')
  356. ->paginate([
  357. 'page' => $pageNo,
  358. 'list_rows' => $pageSize,
  359. 'var_page' => 'page'
  360. ])->toArray();
  361. $data = [];
  362. foreach ($lists['data'] as &$item) {
  363. $item['team_snap'] = json_decode($item['team_snap'], true);
  364. $data[] = [
  365. 'id' => $item['id'],
  366. 'order_id' => $item['order_id'],
  367. 'shop_name' => $item['shop_name'],
  368. 'people_num' => $item['team_snap']['people_num'],
  369. 'name' => $item['team_snap']['name'],
  370. 'image' => $item['team_snap']['image'],
  371. 'price' => $item['team_snap']['price'],
  372. 'count' => $item['team_snap']['count'],
  373. 'spec_value_str' => $item['team_snap']['spec_value_str'],
  374. 'order_amount' => $item['order_amount'],
  375. 'status' => $item['status'],
  376. 'identity' => $item['identity'],
  377. 'identity_text' => $item['identity'] == 1 ? '团长' : '团员',
  378. 'status_text' => TeamEnum::getStatusDesc($item['status'])
  379. ];
  380. }
  381. return [
  382. 'list' => $data,
  383. 'count' => $lists['total'],
  384. 'more' => is_more($lists['total'], $pageNo, $pageSize),
  385. 'page_no' => $pageNo,
  386. 'page_size' => $pageSize
  387. ];
  388. } catch (Exception $e) {
  389. static::$error = $e->getMessage();
  390. return false;
  391. }
  392. }
  393. /**
  394. * @Notes: 验证团
  395. * @Author: 张无忌
  396. * @param $post
  397. * @param $user_id
  398. * @return bool
  399. */
  400. public static function check($post, $user_id)
  401. {
  402. try {
  403. $teamActivity = (new TeamActivity())->alias('T')
  404. ->field([
  405. 'T.id as team_activity_id,T.shop_id,T.effective_time,GI.goods_id,GI.id as item_id,T.people_num,TG.team_price',
  406. 'G.name,G.image,GI.spec_value_str,GI.spec_value_ids,GI.market_price',
  407. 'GI.price,GI.stock'
  408. ])->where([
  409. ['T.goods_id', '=', (int)$post['goods_id']],
  410. ['T.audit', '=', 1],
  411. ['T.status', '=', 1],
  412. ['T.del', '=', 0],
  413. ['T.activity_start_time', '<=', time()],
  414. ['T.activity_end_time', '>=', time()],
  415. ['TG.goods_id', '=', (int)$post['goods_id']],
  416. ['TG.item_id', '=', (int)$post['item_id']],
  417. ])->join('team_goods TG', 'TG.team_id = T.id')
  418. ->join('goods G', 'G.id = TG.goods_id')
  419. ->join('goods_item GI', 'GI.id = TG.item_id')
  420. ->findOrEmpty()->toArray();
  421. if (!$teamActivity) throw new \think\Exception('当前商品未参与拼团活动,下次再来吧');
  422. if ($teamActivity['stock'] - intval($post['count']) < 0) throw new \think\Exception('抱歉,库存不足');
  423. // 参团验证
  424. if (!empty($post['team_id']) and $post['team_id']) {
  425. $teamFound = (new TeamFound())->where(['id'=>$post['team_id']])->findOrEmpty()->toArray();
  426. if (!$teamFound) throw new \think\Exception('选择的团不存在');
  427. if ($teamFound['status'] != 0) throw new \think\Exception('当前拼团已结束,请重新选择拼团');
  428. if ($teamFound['invalid_time'] <= time()) throw new \think\Exception('当前拼团已结束,请重新选择拼团');
  429. if ($teamFound['user_id'] == $user_id) throw new \think\Exception('您已是该团成员了,不能重复参团哦!');
  430. if ($teamFound['people'] == $teamFound['join']) throw new \think\Exception('当前拼团已满员,请重新选择拼团!');
  431. // 获取已参团记录
  432. $people = (new TeamJoin())->where(['team_id'=>$post['team_id'], 'user_id'=>$user_id])->findOrEmpty()->toArray();
  433. if ($people) throw new \think\Exception('您已是该团成员了,不能重复参团哦!');
  434. }
  435. return true;
  436. } catch (Exception $e) {
  437. static::$error = $e->getMessage();
  438. return false;
  439. }
  440. }
  441. }