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

GoodsLogic.php 26KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655
  1. <?php
  2. namespace app\api\logic;
  3. use app\common\model\distribution\Distribution;
  4. use app\common\model\distribution\DistributionLevel;
  5. use app\common\model\shop\ShopFollow;
  6. use app\common\model\user\User;
  7. use app\common\basics\Logic;
  8. use app\common\enum\FootprintEnum;
  9. use app\common\model\distribution\DistributionGoods;
  10. use app\common\model\goods\Goods;
  11. use app\common\model\goods\GoodsCollect;
  12. use app\common\model\goods\GoodsClick;
  13. use app\common\model\goods\GoodsSpec;
  14. use app\common\model\goods\GoodsComment;
  15. use app\common\model\goods\GoodsCommentImage;
  16. use app\common\model\SearchRecord;
  17. use app\common\enum\GoodsEnum;
  18. use app\common\model\seckill\SeckillGoods;
  19. use app\common\model\shop\Shop;
  20. use app\common\model\team\TeamActivity;
  21. use app\common\model\team\TeamFound;
  22. use app\common\model\team\TeamGoods;
  23. use app\common\model\user\UserLevel;
  24. use app\common\server\ConfigServer;
  25. use app\common\server\UrlServer;
  26. use think\facade\Db;
  27. use think\facade\Validate;
  28. class GoodsLogic extends Logic
  29. {
  30. /**
  31. * 商品详情
  32. */
  33. public static function getGoodsDetail($goodsId, $userId)
  34. {
  35. //获取用户折扣
  36. $discount = 10;
  37. if($userId){
  38. $user = User::where('id', $userId)->find();
  39. if($user && isset($user['level'])){
  40. $user_discount = UserLevel::where('id', $user['level'])->value('discount');
  41. if($user_discount && $user_discount > 0 && $user_discount <= 10){
  42. $discount = $user_discount;
  43. }
  44. }
  45. }
  46. // 销售中商品:未删除/审核通过/已上架
  47. $onSaleWhere = [
  48. 'del' => GoodsEnum::DEL_NORMAL, // 未删除
  49. 'status' => GoodsEnum::STATUS_SHELVES, // 上架中
  50. 'audit_status' => GoodsEnum::AUDIT_STATUS_OK, // 审核通过
  51. ];
  52. $goodsDetail = Goods::with(['goods_image', 'goods_item', 'shop'])
  53. ->field('id,type,name,image,video,remark,content,market_price,min_price,max_price,is_show_stock,stock,sales_actual,sales_virtual,clicks,clicks_virtual,shop_id,poster,delivery_type')
  54. ->where($onSaleWhere)
  55. ->where('id', $goodsId)
  56. ->findOrEmpty();
  57. if ($goodsDetail->isEmpty()) {
  58. self::$error = '商品已下架';
  59. return false;
  60. }
  61. //处理默认配送方式
  62. if ($goodsDetail['type'] == GoodsEnum::TYPE_VIRTUAL) {
  63. $goodsDetail['default_delivery_type'] = GoodsEnum::DELIVERY_VIRTUAL;
  64. } else {
  65. // 快递和自提
  66. $goodsDetail['default_delivery_type'] = (int)explode(',',$goodsDetail['delivery_type'])[0];
  67. }
  68. Db::startTrans();
  69. try{
  70. // 轮播图加域名
  71. foreach($goodsDetail['goods_image'] as &$item) {
  72. $item['uri'] = empty($item['uri']) ? '' : UrlServer::getFileUrl($item['uri']);
  73. }
  74. // 会员价
  75. $goodsDetail['member_price'] = 0;
  76. // 会员价数组
  77. $member_price = [];
  78. foreach ($goodsDetail['goods_item'] as &$goods_item) {
  79. $is_member = Goods::where('id',$goods_item['goods_id'])->value('is_member');
  80. $goods_item['is_member'] = $is_member;
  81. if($is_member == 1 && $discount && $userId){
  82. $goods_item['member_price'] = round($goods_item['price']* $discount/10,2);
  83. $goodsDetail['member_price'] = round($goods_item['price']* $discount/10,2);
  84. $member_price[] = $goodsDetail['member_price'];
  85. }
  86. // 规格图片处理
  87. $goods_item['image'] = empty($goods_item['image']) ? $goodsDetail['image'] : $goods_item['image'];
  88. }
  89. !empty($member_price) && $goodsDetail['member_price'] = min($member_price);
  90. // 增加点击量
  91. $goodsDetail->clicks += 1;
  92. $goodsDetail->save();
  93. // 转数组
  94. $goodsDetailArr = $goodsDetail->toArray();
  95. $goodsDetailArr['poster'] = !empty($goodsDetailArr['poster']) ? UrlServer::getFileUrl($goodsDetailArr['poster']) : '';
  96. // 新增点击记录
  97. GoodsClick::create([
  98. 'shop_id' => $goodsDetailArr['shop_id'],
  99. 'user_id' => $userId,
  100. 'goods_id' => $goodsId,
  101. 'create_time' => time()
  102. ]);
  103. //店铺信息
  104. switch ($goodsDetailArr['shop']['type']){
  105. case 1 :
  106. $type_desc = '官方自营';
  107. break;
  108. case 2 :
  109. $type_desc = '入驻商家';
  110. break;
  111. default :
  112. $type_desc = '入驻商家';
  113. break;
  114. }
  115. $follow = Db::name('shop_follow')->where(['shop_id' => $goodsDetailArr['shop_id'],'status' => 1])->count('id');
  116. $goodsDetailArr['shop']['type_desc'] = $type_desc; //商家类型
  117. $goodsDetailArr['shop']['follow_num'] = $follow; //收藏人数
  118. //客服二维码
  119. $customer_image = ConfigServer::get('shop_customer_service','image','',$goodsDetailArr['shop_id']);
  120. if($customer_image){
  121. $customer_image = UrlServer::getFileUrl($customer_image);
  122. }
  123. $goodsDetailArr['shop']['customer_image'] = $customer_image;
  124. // 用户是否关注店铺
  125. $goodsDetailArr['shop']['shop_follow_status'] = 0;
  126. if($userId) { // 用户已登录
  127. $shopFollow = ShopFollow::where(['user_id'=>$userId, 'shop_id'=>$goodsDetailArr['shop_id']])->findOrEmpty();
  128. if(!$shopFollow->isEmpty()) {
  129. $goodsDetailArr['shop']['shop_follow_status'] = $shopFollow['status'];
  130. }
  131. }
  132. // 店铺在售商品数量
  133. $goodsDetailArr['shop']['goods_on_sale'] = Goods::where($onSaleWhere)
  134. ->where('shop_id', $goodsDetailArr['shop_id'])
  135. ->count();
  136. // 店铺推荐商品列表(9个)
  137. $goodsDetailArr['shop']['goods_list'] = Goods::field('id,name,image,market_price,min_price')
  138. ->where($onSaleWhere)
  139. ->where([
  140. 'shop_id' => $goodsDetailArr['shop_id'],
  141. 'is_recommend' => 1, // 推荐
  142. ])
  143. ->order([
  144. 'sales_actual' => 'desc',
  145. 'id' => 'desc'
  146. ])
  147. ->limit(9)
  148. ->select()
  149. ->toArray();
  150. // 总销量 = 实际销量 + 虚拟销量
  151. $goodsDetailArr['sales_sum'] = $goodsDetailArr['sales_actual'] + $goodsDetailArr['sales_virtual'];
  152. // 标识活动信息
  153. $goodsDetailArr['activity'] = [
  154. 'type' => 0,
  155. 'type_desc' => '普通商品'
  156. ];
  157. // 检查商品是否在参与活动,替换商品价格
  158. $goodsDetailArr = self::checkActivity($goodsDetailArr);
  159. // 是否收藏
  160. $goodsDetailArr['is_collect'] = 0;
  161. if($userId) { // 非游客
  162. $goodsCollect = GoodsCollect::where([
  163. 'user_id' => $userId,
  164. 'goods_id' => $goodsId
  165. ])->findOrEmpty();
  166. if(!$goodsCollect->isEmpty()) {
  167. $goodsDetailArr['is_collect'] = $goodsCollect->status ? 1 : 0;
  168. }
  169. }
  170. // 规格项及规格值信息
  171. $goodsDetailArr['goods_spec'] = GoodsSpec::with('spec_value')
  172. ->where('goods_id', $goodsId)->select();
  173. // 商品评价
  174. $commentCategory = GoodsCommentLogic::category(['goods_id'=>$goodsId]);
  175. $goodsDetailArr['comment']['percent'] = $commentCategory['percent'];
  176. $all_comment = Db::name('goods_comment')->where(['goods_id' => $goodsId])->sum('goods_comment');
  177. $goods_comment_count = Db::name('goods_comment')->where(['goods_id' => $goodsId])->count('id');
  178. if($goods_comment_count){
  179. $goods_comment = round($all_comment / $goods_comment_count,2);
  180. $goodsDetailArr['comment']['goods_comment'] = $goods_comment;
  181. }else{
  182. $goodsDetailArr['comment']['goods_comment'] = 0;
  183. }
  184. // 最新一条评论
  185. $one = GoodsComment::alias('gc')
  186. ->field('gc.id,gc.goods_comment,gc.create_time,gc.comment,u.avatar,u.nickname,g.name as goods_name')
  187. ->leftJoin('user u', 'u.id=gc.user_id')
  188. ->leftJoin('goods g', 'g.id=gc.goods_id')
  189. ->where([
  190. ['gc.goods_id', '=', $goodsId],
  191. ['gc.del', '=', 0],
  192. ['gc.status', '=', 1],
  193. ])
  194. ->order('create_time', 'desc')
  195. ->findOrEmpty();
  196. if($one->isEmpty()) {
  197. $one = [];
  198. }else {
  199. $one = $one->toArray();
  200. // 头像
  201. $one['avatar'] = UrlServer::getFileUrl($one['avatar']);
  202. // 图片评价
  203. $one['image'] = GoodsCommentImage::where('goods_comment_id', $one['id'])->column('uri');
  204. foreach($one['image'] as $subKey => $subItem) {
  205. $one['image'][$subKey] = UrlServer::getFileUrl($subItem);
  206. }
  207. }
  208. $goodsDetailArr['comment']['one'] = $one;
  209. // 判断是否是拼团商品
  210. $teamActivity = (new TeamActivity())
  211. ->field(['id,people_num,team_max_price,team_min_price,sales_volume,activity_end_time,share_title,share_intro'])
  212. ->where([
  213. ['goods_id', '=', $goodsId],
  214. ['audit', '=', 1],
  215. ['status', '=', 1],
  216. ['del', '=', 0],
  217. ['activity_start_time', '<=', time()],
  218. ['activity_end_time', '>=', time()]
  219. ])->findOrEmpty()->toArray();
  220. if ($teamActivity) {
  221. // 替换为拼团销量
  222. $goodsDetailArr['sales_sum'] = $teamActivity['sales_volume'];
  223. $teamFound = (new TeamFound())->alias('TF')
  224. ->field(['TF.*', 'U.nickname,U.avatar'])
  225. ->limit(8)
  226. ->order('id desc')
  227. ->where('TF.team_activity_id', '=', $teamActivity['id'])
  228. ->where('TF.people','exp',' > TF.join ')
  229. ->where([
  230. ['status', '=', 0],
  231. ['invalid_time', '>=', time()]
  232. ])->join('user U', 'U.id=TF.user_id')
  233. ->select()->toArray();
  234. foreach ($teamFound as &$found) {
  235. unset($found['shop_id']);
  236. unset($found['team_sn']);
  237. unset($found['goods_snap']);
  238. unset($found['team_end_time']);
  239. $found['avatar'] = UrlServer::getFileUrl($found['avatar']);
  240. $found['surplus_time'] = intval($found['invalid_time'] - time());
  241. }
  242. $teamActivity['share_title'] = !empty($teamActivity['share_title']) ? $teamActivity['share_title'] : $goodsDetailArr['name'];
  243. $teamActivity['share_intro'] = !empty($teamActivity['share_intro']) ? $teamActivity['share_intro'] : $goodsDetailArr['remark'];
  244. $goodsDetailArr['activity'] = ['type'=>2, 'type_desc'=>'拼团商品', 'info'=>$teamActivity, 'found'=>$teamFound];
  245. $teamGoods = (new TeamGoods())->where(['team_id'=>$teamActivity['id']])->select()->toArray();
  246. foreach ($goodsDetailArr['goods_item'] as &$item) {
  247. foreach ($teamGoods as $team) {
  248. if ($item['id'] === $team['item_id']) {
  249. $item['team_price'] = $team['team_price'];
  250. }
  251. }
  252. }
  253. }
  254. // 预估佣金(计算出最高可得佣金)
  255. $goodsDetailArr['distribution'] = self::getDistribution($goodsId, $userId);
  256. // 虚拟浏览量
  257. $goodsDetailArr['clicks'] += $goodsDetailArr['clicks_virtual'];
  258. // 记录访问足迹
  259. event('Footprint', [
  260. 'type' => FootprintEnum::BROWSE_GOODS,
  261. 'user_id' => $userId,
  262. 'foreign_id' => $goodsId
  263. ]);
  264. Db::commit();
  265. return $goodsDetailArr;
  266. }catch(\Exception $e) {
  267. Db::rollback();
  268. self::$error = $e->getMessage();
  269. return false;
  270. }
  271. }
  272. /**
  273. * 热销榜单
  274. */
  275. public static function getHotList($get)
  276. {
  277. // 销售中商品:未删除/审核通过/已上架
  278. $where = [
  279. ['del', '=', GoodsEnum::DEL_NORMAL], // 未删除
  280. ['status', '=', GoodsEnum::STATUS_SHELVES], // 上架中
  281. ['audit_status', '=', GoodsEnum::AUDIT_STATUS_OK], // 审核通过
  282. ];
  283. $order = [
  284. 'sales_total' => 'desc', // 实际销量+虚拟销量倒序
  285. 'sales_actual' => 'desc', // 实际销量倒序
  286. 'id' => 'desc'
  287. ];
  288. return self::getGoodsListTemplate($where, $order, $get);
  289. }
  290. /**
  291. * 商品列表
  292. */
  293. public static function getGoodsList($get)
  294. {
  295. // 销售中商品:未删除/审核通过/已上架
  296. $where = [
  297. ['del', '=', GoodsEnum::DEL_NORMAL], // 未删除
  298. ['status', '=', GoodsEnum::STATUS_SHELVES], // 上架中
  299. ['audit_status', '=', GoodsEnum::AUDIT_STATUS_OK], // 审核通过
  300. ];
  301. $order = [
  302. 'sort_weight' => 'asc', // 商品权重,数字越小权重越大
  303. 'sort' => 'asc',
  304. 'id' => 'desc'
  305. ];
  306. return self::getGoodsListTemplate($where, $order, $get);
  307. }
  308. /**
  309. * 商品列表模板
  310. * 作用:代码复用
  311. */
  312. public static function getGoodsListTemplate($where, $order, $get)
  313. {
  314. if (!empty(self::filterShopsIds())) {
  315. // 过滤已删除、已冻结、已暂停营业、已到期的店铺
  316. $where[] = ['shop_id', 'not in', self::filterShopsIds()];
  317. }
  318. // 平台分类
  319. if(isset($get['platform_cate_id']) && !empty($get['platform_cate_id']) && filter_var($get['platform_cate_id'], FILTER_VALIDATE_INT)) {
  320. $where[] = ['first_cate_id|second_cate_id|third_cate_id', '=', $get['platform_cate_id']];
  321. }
  322. // 品牌
  323. if(isset($get['brand_id']) && !empty($get['brand_id']) && filter_var($get['brand_id'], FILTER_VALIDATE_INT)) {
  324. $where[] = ['brand_id', '=', $get['brand_id']];
  325. }
  326. // 关键词
  327. if(isset($get['keyword']) && !empty($get['keyword'])) {
  328. $where[] = ['name', 'like', '%'.trim($get['keyword']).'%'];
  329. if($get['user_id']) { // 记录关键词
  330. self::recordKeyword(trim($get['keyword']), $get['user_id']);
  331. }
  332. }
  333. // 店铺id
  334. if(isset($get['shop_id']) && !empty($get['shop_id']) && filter_var($get['shop_id'], FILTER_VALIDATE_INT)) {
  335. $where[] = ['shop_id', '=', $get['shop_id']];
  336. }
  337. // 店铺推荐
  338. if (Validate::must($get['is_recommend'] ?? '')) {
  339. $where[] = [ 'is_recommend', '=', $get['is_recommend'] ];
  340. }
  341. // 店铺分类
  342. if(isset($get['shop_cate_id']) && !empty($get['shop_cate_id']) && filter_var($get['shop_cate_id'], FILTER_VALIDATE_INT)) {
  343. $where[] = ['shop_cate_id', '=', $get['shop_cate_id']];
  344. }
  345. // 销量排序(实际销量 + 虚拟销量)
  346. if(isset($get['sort_by_sales']) && !empty($get['sort_by_sales'])) {
  347. $elt = ['sales_total'=> trim($get['sort_by_sales'])];
  348. $order = array_merge($elt, $order);
  349. }
  350. // 价格排序
  351. if(isset($get['sort_by_price']) && !empty($get['sort_by_price'])) {
  352. $elt = ['min_price'=> trim($get['sort_by_price'])];
  353. $order = array_merge($elt, $order);
  354. }
  355. // 新品排序
  356. if(isset($get['sort_by_create']) && !empty($get['sort_by_create'])) {
  357. $elt = ['create_time'=> trim($get['sort_by_create'])];
  358. $order = array_merge($elt, $order);
  359. }
  360. $field = 'id,image,name,min_price,market_price,sales_actual,first_cate_id,
  361. second_cate_id,third_cate_id,sort_weight,brand_id,shop_id,sales_virtual,
  362. (sales_actual + sales_virtual) as sales_total';
  363. $list = Goods::with(['shop'])
  364. ->field($field)
  365. ->where($where)
  366. ->order($order)
  367. ->page($get['page_no'], $get['page_size'])
  368. ->select();
  369. foreach ($list as $item) {
  370. $item['shop_type'] = $item['shop']['type'];
  371. unset($item['shop']);
  372. }
  373. $count = Goods::where($where)->count();
  374. $list = $list ? $list->toArray() : [];
  375. $more = is_more($count, $get['page_no'], $get['page_size']);
  376. $data = [
  377. 'lists' => $list,
  378. 'page_no' => $get['page_no'],
  379. 'page_size' => $get['page_size'],
  380. 'count' => $count,
  381. 'more' => $more
  382. ];
  383. return $data;
  384. }
  385. /**
  386. * 根据商品栏目获取商品列表
  387. */
  388. public static function getGoodsListByColumnId($columnId, $page_no, $page_size)
  389. {
  390. // 销售中商品:未删除/审核通过/已上架
  391. $onSaleWhere = [
  392. ['del', '=', GoodsEnum::DEL_NORMAL],
  393. ['status', '=', GoodsEnum::STATUS_SHELVES],
  394. ['audit_status', '=', GoodsEnum::AUDIT_STATUS_OK],
  395. ];
  396. if (!empty(self::filterShopsIds())) {
  397. // 过滤已删除、已冻结、已暂停营业、已到期的店铺
  398. $onSaleWhere[] = ['shop_id', 'not in', self::filterShopsIds()];
  399. }
  400. $order = [
  401. 'sort_weight' => 'asc', // 数字越小,权重越大
  402. 'sales_actual' => 'desc',
  403. 'id' => 'desc'
  404. ];
  405. $list = Goods::field('id,name,image,market_price,min_price,sales_actual,column_ids,sort_weight,sales_virtual,(sales_actual + sales_virtual) as sales_total')
  406. ->where($onSaleWhere)
  407. ->whereFindInSet('column_ids', $columnId)
  408. ->order($order)
  409. ->page($page_no, $page_size)
  410. ->select();
  411. $count = Goods::where($onSaleWhere)
  412. ->whereFindInSet('column_ids', $columnId)
  413. ->count();
  414. $list = $list ? $list->toArray() : [];
  415. $more = is_more($count, $page_no, $page_size);
  416. $data = [
  417. 'lists' => $list,
  418. 'page_no' => $page_no,
  419. 'page_size' => $page_size,
  420. 'count' => $count,
  421. 'more' => $more
  422. ];
  423. return $data;
  424. }
  425. /**
  426. * @notes 获取已删除、已冻结、已暂停营业、已到期店铺的id
  427. * @return array
  428. * @throws \think\db\exception\DataNotFoundException
  429. * @throws \think\db\exception\DbException
  430. * @throws \think\db\exception\ModelNotFoundException
  431. * @author Tab
  432. * @date 2021/7/20 14:29
  433. */
  434. public static function filterShopsIds()
  435. {
  436. // 已删除、已冻结、已暂停营业的店铺
  437. $invalidShops = Shop::field('id,name')->whereOr([
  438. ['del', '=', 1], // 已删除
  439. ['is_freeze', '=', 1], // 已冻结
  440. ['is_run', '=', 0] // 暂停营业
  441. ])->select()->toArray();
  442. // 已过期的店铺
  443. $expiredShops = Shop::field('id,name')->where([
  444. ['expire_time', '<>', 0],
  445. ['expire_time', '<=', time()],
  446. ])->select()->toArray();
  447. $filterShops = array_merge($invalidShops, $expiredShops);
  448. $filterShopsIds = array_column($filterShops, 'id');
  449. return $filterShopsIds;
  450. }
  451. /**
  452. * 记录关键词
  453. */
  454. public static function recordKeyword($keyword, $user_id)
  455. {
  456. $record = SearchRecord::where(['user_id'=>$user_id,'keyword'=>$keyword,'del'=>0])->find();
  457. if($record){
  458. // 有该关键词记录, 更新
  459. return SearchRecord::where(['id'=>$record['id']])->update(['count'=>Db::raw('count+1'),'update_time'=>time()]);
  460. }
  461. // 无该关键词记录 > 新增
  462. return SearchRecord::create([
  463. 'user_id'=>$user_id,
  464. 'keyword'=>$keyword,
  465. 'count' => 1,
  466. 'update_time' => time(),
  467. 'del' => 0
  468. ]);
  469. }
  470. //检查商品是否正在参加活动
  471. public static function checkActivity($goods){
  472. // 获取正在秒杀的时段
  473. $seckill_time = SeckillGoodsLogic::getSeckillTimeIng();
  474. if($seckill_time === false) {
  475. // 不在秒杀时段,直接返回
  476. return $goods;
  477. }
  478. // 判断是否是秒杀中的商品
  479. $seckill_goods = SeckillGoods::where([
  480. ['del', '=', 0],
  481. ['seckill_id', '=', $seckill_time['id']],
  482. ['goods_id', '=', $goods['id']],
  483. ['review_status', '=', 1],
  484. ])->select()->toArray();
  485. if(!$seckill_goods) {
  486. // 不是秒杀商品
  487. return $goods;
  488. }
  489. // 判断参与日期是否包含今天
  490. $flag = false;
  491. $now = time();
  492. foreach($seckill_goods as $item) {
  493. $start_date_time = strtotime($item['start_date'].' ' . $seckill_time['start_time']);
  494. $end_date_time = strtotime($item['end_date'].' ' . $seckill_time['end_time']);
  495. if($start_date_time < $now && $end_date_time > $now) {
  496. $flag = true;
  497. // 获取该商品的秒杀信息
  498. $seckill_goods_info = SeckillGoods::where([
  499. 'goods_id' => $goods['id'],
  500. 'seckill_id' => $seckill_time['id'],
  501. 'start_date' => $item['start_date'],
  502. 'end_date' => $item['end_date'],
  503. ])->column('goods_id,item_id,price,sales_sum', 'item_id');
  504. break;
  505. }
  506. }
  507. if($flag === false) {
  508. // 参与日期不在今天
  509. return $goods;
  510. }
  511. // 确定是秒杀中的商品
  512. // 先将商品市场价换成原SKU最小价
  513. $goods['market_price'] = $goods['min_price'];
  514. // 替换秒杀销量
  515. $goods['sales_sum'] = array_sum(array_column($seckill_goods_info ?? [], 'sales_sum'));
  516. // 替换活动价
  517. foreach($goods['goods_item'] as &$item) {
  518. // 商品价格替换为最小的秒杀价
  519. if($goods['min_price'] > $seckill_goods_info[$item['id']]['price']) {
  520. $goods['min_price'] = $seckill_goods_info[$item['id']]['price'];
  521. }
  522. // 原市场价替换为原SKU售价
  523. $item['market_price'] = $item['price'];
  524. // SKU替换秒杀价
  525. $item['price'] = $seckill_goods_info[$item['id']]['price'];
  526. }
  527. $today_date = date('Y-m-d');
  528. $goods['activity'] = [
  529. 'type' => 1,
  530. 'type_desc' => '秒杀商品',
  531. 'end_time' => strtotime($today_date.' '.$seckill_time['end_time'])
  532. ];
  533. return $goods;
  534. }
  535. /**
  536. * @notes 获取商品分销信息
  537. * @param $goodsId
  538. * @param $userId
  539. * @return array
  540. * @throws \think\db\exception\DataNotFoundException
  541. * @throws \think\db\exception\DbException
  542. * @throws \think\db\exception\ModelNotFoundException
  543. * @author Tab
  544. * @date 2021/9/6 18:48
  545. */
  546. public static function getDistribution($goodsId, $userId)
  547. {
  548. $earnings = 0;
  549. $goods = Goods::findOrEmpty($goodsId)->toArray();
  550. $distributionGoods = DistributionGoods::where('goods_id', $goodsId)->select()->toArray();
  551. if(!empty($distributionGoods) && $distributionGoods[0]['is_distribution'] && $distributionGoods[0]['rule'] == 2) {
  552. foreach($distributionGoods as $item) {
  553. $earnings = max($earnings, round($goods['max_price'] * $item['first_ratio'] / 100, 2));
  554. $earnings = max($earnings, round($goods['max_price'] * $item['second_ratio'] / 100, 2));
  555. }
  556. }
  557. if(!empty($distributionGoods) && $distributionGoods[0]['is_distribution'] && $distributionGoods[0]['rule'] == 1) {
  558. $levels = DistributionLevel::select()->toArray();
  559. foreach($levels as $item) {
  560. $earnings = max($earnings, round($goods['max_price'] * $item['first_ratio'] / 100, 2));
  561. $earnings = max($earnings, round($goods['max_price'] * $item['second_ratio'] / 100, 2));
  562. }
  563. }
  564. // 详情页是否显示佣金
  565. $isShow = ConfigServer::get('distribution', 'is_show_earnings', 0);
  566. // 系统总分销开关
  567. $distributionOpen = ConfigServer::get('distribution', 'is_open', 0);
  568. // 商家信息-获取商家是否被禁用分销功能(is_distribution)
  569. $shop = Shop::findOrEmpty($goods['shop_id'])->toArray();
  570. if ($distributionOpen && $shop['is_distribution'] && $isShow) {
  571. //详情页佣金可见用户 0-全部用户 1-分销商
  572. $scope = ConfigServer::get('distribution', 'show_earnings_scope', 0);
  573. $user = Distribution::where(['user_id' => $userId])->findOrEmpty()->toArray();
  574. if ($scope && empty($user['is_distribution'])) {
  575. $isShow = 0;
  576. }
  577. } else {
  578. $isShow = 0;
  579. }
  580. return [
  581. 'is_show' => $isShow,
  582. 'earnings' => $earnings
  583. ];
  584. }
  585. }