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

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580
  1. <?php
  2. namespace app\api\logic;
  3. use app\api\controller\Shop;
  4. use app\common\enum\BargainEnum;
  5. use app\common\model\goods\GoodsItem;
  6. use app\common\basics\Logic;
  7. use app\common\server\ConfigServer;
  8. use app\common\server\UrlServer;
  9. use think\facade\Db;
  10. use think\Exception;
  11. use app\common\model\bargain\Bargain;
  12. use app\common\model\bargain\BargainItem;
  13. use app\common\model\bargain\BargainKnife;
  14. use app\common\model\bargain\BargainLaunch;
  15. use app\common\model\goods\GoodsSpec;
  16. use app\common\model\goods\GoodsSpecValue;
  17. /**
  18. * Class BargainLogic
  19. * @package app\api\logic
  20. */
  21. class BargainLogic extends Logic
  22. {
  23. /**
  24. * @notes 获取砍价成功人数
  25. * @return int
  26. * @author suny
  27. * @date 2021/7/13 6:17 下午
  28. */
  29. public static function bargainNumber()
  30. {
  31. return BargainLaunch::where(['status' => 1])
  32. ->count();
  33. }
  34. /**
  35. * @notes 砍价列表
  36. * @param $page
  37. * @param $size
  38. * @return array
  39. * @throws \think\db\exception\DataNotFoundException
  40. * @throws \think\db\exception\DbException
  41. * @throws \think\db\exception\ModelNotFoundException
  42. * @author suny
  43. * @date 2021/7/13 6:17 下午
  44. */
  45. public static function lists($page, $size)
  46. {
  47. $now = time();
  48. $where[] = ['B.del', '=', 0];
  49. $where[] = ['B.status', '=', 1];
  50. $where[] = ['B.audit_status', '=', 1];
  51. $where[] = ['activity_start_time', '<', $now];
  52. $where[] = ['activity_end_time', '>', $now];
  53. $bargain_count = Bargain::alias('B')
  54. ->join('Goods G', 'B.goods_id = G.id')
  55. ->where($where)
  56. ->count();
  57. $bargain_list = Bargain::alias('B')
  58. ->join('Goods G', 'B.goods_id = G.id')
  59. ->where($where)
  60. ->page($page, $size)
  61. ->order('id desc')
  62. ->field('B.id,B.bargain_min_price as activity_price,G.id as goods_id,G.name,G.image,G.min_price as price')
  63. ->withAttr('image', function ($value, $data) {
  64. return UrlServer::getFileUrl($value);
  65. })
  66. ->select();
  67. $more = is_more($bargain_count, $page, $size); //是否有下一页
  68. $data = [
  69. 'list' => $bargain_list,
  70. 'page_no' => $page,
  71. 'page_size' => $size,
  72. 'count' => $bargain_count,
  73. 'more' => $more
  74. ];
  75. return $data;
  76. }
  77. /**
  78. * @notes 砍价活动详情
  79. * @param $bargain_id
  80. * @return array|\think\Model|null
  81. * @throws \think\db\exception\DataNotFoundException
  82. * @throws \think\db\exception\DbException
  83. * @throws \think\db\exception\ModelNotFoundException
  84. * @author suny
  85. * @date 2021/7/13 6:17 下午
  86. */
  87. public static function detail($bargain_id)
  88. {
  89. $detail = Bargain::alias('B')
  90. ->join('goods G', 'B.goods_id = G.id')
  91. ->where(['B.id' => $bargain_id])
  92. ->field('B.id,B.shop_id,B.bargain_min_price as activity_price,G.id as goods_id,G.name,G.image,G.min_price as price')
  93. ->withAttr('image', function ($value, $data) {
  94. return UrlServer::getFileUrl($value);
  95. })
  96. ->find();
  97. $detail['goods_item'] = BargainItem::alias('BI')
  98. ->join('goods_item GI', 'BI.item_id = GI.id')
  99. ->where(['BI.goods_id' => $detail['goods_id']])
  100. ->field('BI.floor_price as activity_price,GI.id,GI.image,GI.price,GI.spec_value_ids,GI.spec_value_str')
  101. ->select();
  102. $spec_list = GoodsSpec::where(['goods_id' => $detail['goods_id']])
  103. ->column('*', 'id');
  104. $spec_value_list = GoodsSpecValue::where(['goods_id' => $detail['goods_id']])
  105. ->select();
  106. foreach ($spec_value_list as $spec_value) {
  107. if (isset($spec_list[$spec_value['spec_id']])) {
  108. $spec_list[$spec_value['spec_id']]['spec_value'][] = $spec_value;
  109. }
  110. }
  111. foreach ($detail['goods_item'] as $key => $goods_item) {
  112. if (empty($goods_item['image'])) {
  113. $goods_item['image'] = $detail['image'];
  114. }
  115. $detail['goods_item'][$key]['image'] = UrlServer::getFileUrl($goods_item['image']);
  116. }
  117. $detail['goods_spec'] = array_values($spec_list);
  118. //前端渲染状态
  119. $detail['status'] = -1;
  120. //提示字段
  121. $detail['bargain_tips'] = '您正在发起砍价';
  122. $detail['simple_tips'] = '邀请好友帮忙砍价,砍至' . $detail['activity_price'] . '即可发货';
  123. return $detail;
  124. }
  125. /**
  126. * @notes 发起砍价
  127. * @param $post_data
  128. * @param $user_id
  129. * @return array|false
  130. * @throws \think\exception\PDOException
  131. * @author suny
  132. * @date 2021/7/13 6:17 下午
  133. */
  134. public static function sponsor($post_data, $user_id)
  135. {
  136. Db::startTrans();
  137. try {
  138. $shop_id = Bargain::where('id', $post_data['bargain_id'])->value('shop_id');
  139. //商品规格信息
  140. $goods_item = GoodsItem::alias('GI')
  141. ->join('goods G', 'G.id = GI.goods_id')
  142. ->where(['GI.id' => $post_data['item_id']])
  143. ->field('G.name,G.image as goods_iamge,GI.id as item_id,GI.image,GI.spec_value_str,GI.price')
  144. ->find();
  145. $bargain = Bargain::alias('B')
  146. ->join('bargain_item BI', 'B.id = BI.bargain_id')
  147. ->where(['B.id' => $post_data['bargain_id'], 'BI.item_id' => $post_data['item_id'], 'B.del' => 0])
  148. ->field('B.*,BI.floor_price,BI.item_id,BI.first_knife_price')
  149. ->find();
  150. $now = time();
  151. //砍到价格
  152. $current_price = $goods_item['price'] - $bargain['first_knife_price'];
  153. $status = BargainLaunch::conductStatus;
  154. $payment_limit_time = 0;
  155. $bargain_end_time = 0;
  156. //首刀砍掉后小于零,或低于底价
  157. if ($current_price <= $bargain['floor_price'] || $current_price <= 0) {
  158. //标记砍价成功
  159. $status = BargainLaunch::successStatus;
  160. //砍价的价格低于低价、重新计算每刀价格
  161. if ($current_price < $bargain['floor_price']) {
  162. $current_price = $bargain['floor_price'];
  163. $bargain['first_knife_price'] = $goods_item['price'] - $current_price;
  164. }
  165. //砍价成功后的付款时间(秒)
  166. $payment_limit_time = ConfigServer::get('bargain', 'payment_limit_time', 0) * 60;
  167. if ($payment_limit_time > 0) {
  168. $payment_limit_time = $now + $payment_limit_time;
  169. }
  170. $bargain_end_time = $now;
  171. }
  172. $diff_price = round($current_price - $bargain['floor_price'], 2);
  173. $bargain_sn = createSn('bargain_launch', 'bargain_sn');
  174. $bargain_launch = [
  175. 'bargain_id' => $bargain['id'],
  176. 'goods_id' => $bargain['goods_id'],
  177. 'user_id' => $user_id,
  178. 'order_id' => '',
  179. 'goods_snap' => json_encode($goods_item), //规格信息
  180. 'bargain_snap' => json_encode($bargain), //砍价信息
  181. 'help_number' => 1, //助力次数
  182. 'bargain_price' => $bargain['floor_price'], //商品活动低价
  183. 'current_price' => $current_price, //当前砍到的价格
  184. 'launch_start_time' => $now, //砍价发起时间
  185. 'launch_end_time' => $now + $bargain['time_limit'] * 3600,//砍价结束时间
  186. 'bargain_end_time' => $bargain_end_time, //最后的砍价时间
  187. 'payment_limit_time' => $payment_limit_time, //最后的付款时间
  188. 'status' => $status, //当前砍价状态
  189. 'shop_id' => $shop_id, //店铺ID
  190. 'bargain_sn' => $bargain_sn, //砍价订单编号
  191. ];
  192. //写入发起砍价表
  193. $bargain_launch_id = BargainLaunch::insertGetId($bargain_launch);
  194. $bargain_knife = [
  195. 'bargain_id' => $bargain['id'],
  196. 'launch_id' => $bargain_launch_id,
  197. 'user_id' => $user_id,
  198. 'surplus_price' => $current_price,
  199. 'help_price' => $bargain['first_knife_price'],
  200. 'help_time' => $now
  201. ];
  202. //写入砍价记录表
  203. BargainKnife::insert($bargain_knife);
  204. //砍价进度条
  205. if ($current_price == 0) {
  206. $progress = round(1, 2);
  207. } else {
  208. $progress = round($bargain['floor_price'] / $current_price, 2);
  209. }
  210. Db::commit();
  211. $data = [
  212. 'id' => $bargain_launch_id,
  213. 'knife_price' => $bargain['first_knife_price'],
  214. 'diff_price' => $diff_price,
  215. 'progress' => $progress
  216. ];
  217. return $data;
  218. } catch (Exception $e) {
  219. Db::rollback();
  220. self::$error = $e->getMessage();
  221. return false;
  222. }
  223. }
  224. /**
  225. * @notes 砍价助力
  226. * @param $id
  227. * @param $user_id
  228. * @return array|false
  229. * @throws \think\exception\PDOException
  230. * @author suny
  231. * @date 2021/7/13 6:17 下午
  232. */
  233. public static function knife($id, $user_id)
  234. {
  235. Db::startTrans();
  236. try {
  237. $now = time();
  238. $bargain_launch = new BargainLaunch();
  239. $bargain_launch = $bargain_launch
  240. ->where(['id' => $id])
  241. ->find();
  242. $bargain_snap = $bargain_launch['bargain_snap'];
  243. //每刀随机金额
  244. if (1 == $bargain_snap['knife_type']) {
  245. $knife__price_array = explode(',', $bargain_launch['bargain_snap']['knife_price']);
  246. $knife_price = round(random_float(array_pop($knife__price_array), array_pop($knife__price_array)), 2);
  247. } else { //每刀固定金额
  248. $knife_price = $bargain_launch['bargain_snap']['knife_price'];
  249. }
  250. //商品的低价
  251. $low_price = $bargain_launch['bargain_price'];
  252. //砍价后的金额
  253. $knife_after_price = $bargain_launch['current_price'] - $knife_price;
  254. $status = BargainLaunch::conductStatus; //砍价状态
  255. $payment_limit_time = 0; //砍价成功后的付款时间
  256. $bargain_end_time = 0; //最后的砍刀时间
  257. //如果砍价后低于低价,按当前的价格-低价来得到每刀金额,并标记砍价成功
  258. if ($knife_after_price <= $low_price) {
  259. //砍价成功后的付款时间(秒)
  260. $payment_limit_time = ConfigServer::get('bargain', 'payment_limit_time', 0) * 60;
  261. $status = BargainLaunch::successStatus;
  262. if ($payment_limit_time > 0) {
  263. $payment_limit_time = $now + $payment_limit_time;
  264. }
  265. $bargain_end_time = $now;
  266. $knife_price = round($bargain_launch['current_price'] - $low_price, 2); //砍价的金额 = 当前砍到的价格 - 低价
  267. $knife_after_price = $low_price; //砍价后更新为低价
  268. }
  269. //写入砍价助力表
  270. $knife_data = [
  271. 'bargain_id' => $bargain_snap['id'], //活动id
  272. 'launch_id' => $bargain_launch['id'], //订单id
  273. 'user_id' => $user_id, //用户id
  274. 'surplus_price' => $knife_after_price, //助力后的价格
  275. 'help_price' => $knife_price, //助力的金额
  276. 'help_time' => $now,
  277. ];
  278. BargainKnife::insert($knife_data);
  279. //更新砍价进度
  280. $bargain_launch->help_number = ['inc', 1]; //助力次数+1
  281. $bargain_launch->current_price = $knife_after_price; //助力后的价格
  282. $bargain_launch->status = $status; //砍价状态
  283. //砍价成功 限制多少时间内付款
  284. if ($payment_limit_time > 0) {
  285. $bargain_launch->payment_limit_time = $payment_limit_time;
  286. }
  287. if ($bargain_end_time > 0) {
  288. $bargain_launch->bargain_end_time = $bargain_end_time;
  289. }
  290. $bargain_launch->save();
  291. //进度条、剩余差价
  292. if ($knife_after_price == 0) {
  293. $progress = round(1, 2);
  294. } else {
  295. $progress = round($bargain_snap['floor_price'] / $knife_after_price, 2);
  296. }
  297. $diff_price = round($knife_after_price - $bargain_snap['floor_price'], 2);
  298. Db::commit();
  299. $data = [
  300. 'status' => $status,
  301. 'knife_price' => $knife_price,
  302. 'diff_price' => $diff_price,
  303. 'progress' => $progress
  304. ];
  305. return $data;
  306. } catch (Exception $e) {
  307. Db::rollback();
  308. self::$error = $e->getMessage();
  309. return false;
  310. }
  311. }
  312. /**
  313. * @notes 砍价订单
  314. * @param $type
  315. * @param $user_id
  316. * @param $page
  317. * @param $size
  318. * @return array
  319. * @throws \think\db\exception\DataNotFoundException
  320. * @throws \think\db\exception\DbException
  321. * @throws \think\db\exception\ModelNotFoundException
  322. * @throws \think\exception\DbException
  323. * @author suny
  324. * @date 2021/7/13 6:18 下午
  325. */
  326. public static function orderList($type, $user_id, $page, $size)
  327. {
  328. $where[] = ['user_id', '=', $user_id];
  329. if ('-1' != $type && $type != '') {
  330. $where[] = ['status', '=', $type];
  331. }
  332. $bargain_launch = new BargainLaunch();
  333. $order_count = $bargain_launch
  334. ->where($where)
  335. ->count();
  336. $order_list = $bargain_launch
  337. ->alias('al')
  338. ->join('shop s', 's.id = al.shop_id')
  339. ->where($where)
  340. ->field('al.* ,s.name as shop_name')
  341. ->page($page, $size)
  342. ->order('al.id desc')
  343. ->append(['item_id', 'btn_tips', 'bargain_tips', 'image', 'goods_image', 'name', 'price', 'spec_value_str', 'status_text', 'create_time'])
  344. ->visible(['id', 'bargain_id', 'goods_id', 'item_id', 'image', 'goods_image', 'name', 'price', 'spec_value_str', 'order_id', 'current_price', 'buy_btn', 'bargain_btn', 'pay_btn', 'status_text', 'create_time', 'shop_name'])
  345. ->select();
  346. $more = is_more($order_count, $page, $size); //是否有下一页
  347. $data = [
  348. 'list' => $order_list,
  349. 'page_no' => $page,
  350. 'page_size' => $size,
  351. 'count' => $order_count,
  352. 'more' => $more
  353. ];
  354. return $data;
  355. }
  356. /**
  357. * @notes 砍价详情页
  358. * @param $id
  359. * @param $user_id
  360. * @return array
  361. * @throws \think\db\exception\DataNotFoundException
  362. * @throws \think\db\exception\DbException
  363. * @throws \think\db\exception\ModelNotFoundException
  364. * @throws \think\exception\DbException
  365. * @author suny
  366. * @date 2021/7/13 6:18 下午
  367. */
  368. public static function bargainDetail($id, $user_id)
  369. {
  370. $bargain_launch = new BargainLaunch();
  371. $bargain_launch = $bargain_launch
  372. ->where(['id' => $id])
  373. ->with('bargain_knife.user')
  374. ->append(['item_id', 'image', 'name', 'price', 'spec_value_str', 'over_time', 'diff_price', 'activity_price', 'progress', 'knife_list', 'knife_price', 'share_titles', 'share_intros'])
  375. ->find();
  376. //砍价快照
  377. $bargain_snap = $bargain_launch['bargain_snap'];
  378. $bargain_launch = $bargain_launch->visible(['id', 'bargain_id', 'shop_id', 'user_id', 'goods_id', 'item_id', 'order_id', 'name', 'image', 'price', 'spec_value_str', 'current_price', 'floor_price', 'diff_price', 'progress', 'knife_list', 'status', 'knife_price', 'share_titles', 'share_intros', 'payment_limit_time'])
  379. ->hidden(['bargain_knife'])
  380. ->toArray();
  381. $user_info = current($bargain_launch['knife_list']);
  382. $now = time();
  383. //好友点击分享行为按钮
  384. $sponsor_btn = 0; //我也要砍价按钮
  385. $knife_btn = 0; //帮忙砍一刀按钮
  386. //查看自己砍价行为按钮
  387. $direct_buy_btn = 0; //直接购买按钮
  388. $buy_btn = 0; //立即购买按钮
  389. $invite_btn = 0; //邀请好友按钮
  390. $order_btn = 0; //查看订单按钮
  391. //提示语,通用
  392. $bargain_tips = ''; //砍价提示,最上方提示
  393. $simple_tips = ''; //提示,砍价提示下方提示
  394. $status_tips = ''; //状态提示
  395. //分享用户的头像
  396. $bargain_launch['share_avatar'] = '';
  397. //显示按钮:查看自己的砍价、好友点击分享进来
  398. if ($bargain_launch['user_id'] != $user_id) { //好友点击分享进来
  399. $bargain_launch['status'] = 5;//标记为好友邀请进来的
  400. //我也要砍价按钮
  401. $sponsor_btn = 1;
  402. $bargain_tips = '来自' . $user_info['nickname'] . '的分享';
  403. $bargain_launch['share_avatar'] = $user_info['avatar'];
  404. $simple_tips = '谢谢您的助力,动动手指帮我砍一刀';
  405. if ($bargain_launch['over_time'] <= $now || 1 == $bargain_launch['status'] || 2 == $bargain_launch['status']) {
  406. $status_tips = '砍价已结束,去看看其他商品吧~';
  407. } else {
  408. $bargain_knife = Db::name('bargain_knife')->where(['launch_id' => $bargain_launch['id'], 'user_id' => $user_id])->find();
  409. if (empty($bargain_knife)) {
  410. //帮忙砍一刀按钮
  411. $knife_btn = 1;
  412. }
  413. }
  414. } else {//自己的砍价
  415. $simple_tips = '邀请好友帮砍价,砍至' . $bargain_snap['floor_price'] . '即可发货';
  416. //砍价进行中
  417. if (0 == $bargain_launch['status'] && $bargain_launch['over_time'] > $now) {
  418. //邀请好友按钮
  419. $invite_btn = 1;
  420. $bargain_tips = '砍价中';
  421. if (2 == $bargain_snap['payment_where']) {
  422. //显示直接购买按钮
  423. $direct_buy_btn = 1;
  424. }
  425. $status_tips = '砍价中';
  426. }
  427. //砍价成功
  428. if (1 == $bargain_launch['status'] || (2 == $bargain_snap['payment_where'] && $bargain_launch['over_time'] <= $now)) {
  429. $bargain_tips = '砍价成功';
  430. $status_tips = '恭喜您,砍价成功';
  431. //砍价成功,没有超过付款时间,且没有下单的,显示购买按钮
  432. $buy_btn = 1;
  433. if ($bargain_launch['order_id'] || ($bargain_launch['payment_limit_time'] && $bargain_launch['payment_limit_time'] < $now)) {
  434. $buy_btn = 0;
  435. if ($bargain_launch['payment_limit_time'] && $bargain_launch['payment_limit_time'] < $now) {
  436. $status_tips = '已超过下单时间,无法下单';
  437. }
  438. }
  439. //已下单
  440. if ($bargain_launch['order_id']) {
  441. $order_btn = 1;
  442. }
  443. }
  444. //砍价失败
  445. if (($bargain_launch['over_time'] <= $now && 1 != $bargain_launch['status'] && 2 != $bargain_snap['payment_where']) || 2 == $bargain_launch['status']) {
  446. $bargain_tips = '砍价失败';
  447. $status_tips = '非常遗憾,砍价失败';
  448. }
  449. }
  450. $bargain_launch['sponsor_btn'] = $sponsor_btn;
  451. $bargain_launch['knife_btn'] = $knife_btn;
  452. $bargain_launch['direct_buy_btn'] = $direct_buy_btn;
  453. $bargain_launch['buy_btn'] = $buy_btn;
  454. $bargain_launch['invite_btn'] = $invite_btn;
  455. $bargain_launch['order_btn'] = $order_btn;
  456. $bargain_launch['bargain_tips'] = $bargain_tips;
  457. $bargain_launch['simple_tips'] = $simple_tips;
  458. $bargain_launch['status_tips'] = $status_tips;
  459. return $bargain_launch;
  460. }
  461. /**
  462. * @notes 关闭砍价订单
  463. * @param $id
  464. * @return bool
  465. * @throws \think\db\exception\DataNotFoundException
  466. * @throws \think\db\exception\DbException
  467. * @throws \think\db\exception\ModelNotFoundException
  468. * @author suny
  469. * @date 2021/7/13 6:18 下午
  470. */
  471. public static function closeBargain($id)
  472. {
  473. $bargain_launch = new BargainLaunch();
  474. $bargain_launch = $bargain_launch
  475. ->where(['id' => $id])
  476. ->find();
  477. //标记砍价结束
  478. if ($bargain_launch['launch_end_time'] < time() && 0 == $bargain_launch['status']) {
  479. //任意金额购买标记成功
  480. if (2 == $bargain_launch['bargain_snap']['payment_where']) {
  481. $status = 1;
  482. } else {//砍到低价
  483. $status = 2;
  484. if ($bargain_launch['bargain_price'] > $bargain_launch['current_price']) {
  485. $status = 1;
  486. }
  487. }
  488. $bargain_launch->status = $status;
  489. $bargain_launch->save();
  490. }
  491. return true;
  492. }
  493. }