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

DistributionLogic.php 19KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562
  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\api\logic;
  20. use app\common\enum\DistributionOrderGoodsEnum;
  21. use app\common\enum\NoticeEnum;
  22. use app\common\model\distribution\Distribution;
  23. use app\common\model\distribution\DistributionLevel;
  24. use app\common\server\ConfigServer;
  25. use app\common\model\user\UserDistribution;
  26. use app\common\model\user\User;
  27. use app\common\model\distribution\DistributionMemberApply;
  28. use app\common\model\distribution\DistributionOrderGoods;
  29. use app\common\server\AreaServer;
  30. use app\common\server\JsonServer;
  31. use app\common\server\UrlServer;
  32. use app\common\basics\Logic;
  33. use app\common\logic\AccountLogLogic;
  34. use app\common\model\AccountLog;
  35. use think\Exception;
  36. use think\facade\Db;
  37. class DistributionLogic extends Logic
  38. {
  39. /**
  40. * Notes: 根据后台设置返回当前生成用户的分销会员状态(设置了全员分销,新生成的用户即为分销会员)
  41. * @author 段誉(2021/4/7 14:48)
  42. * @return int
  43. */
  44. public static function isDistributionMember()
  45. {
  46. $is_distribution = 0;
  47. //分销会员申请--1,申请分销; 2-全员分销;
  48. $distribution = ConfigServer::get('distribution', 'member_apply', 1);
  49. if ($distribution == 2) {
  50. $is_distribution = 1;
  51. }
  52. return $is_distribution;
  53. }
  54. /**
  55. * Desc: 生成用户扩展表
  56. * @param $user_id
  57. * @return bool
  58. * @throws \think\db\exception\DataNotFoundException
  59. * @throws \think\db\exception\ModelNotFoundException
  60. * @throws \think\exception\DbException
  61. */
  62. public static function createUserDistribution($user_id)
  63. {
  64. $user_distribution = UserDistribution::where(['user_id' => $user_id])->find();
  65. if ($user_distribution) {
  66. return true;
  67. }
  68. $data = [
  69. 'user_id' => $user_id,
  70. 'distribution_order_num' => 0,
  71. 'distribution_money' => 0,
  72. 'fans' => 0,
  73. 'create_time' => time(),
  74. ];
  75. UserDistribution::create($data);
  76. return true;
  77. }
  78. /**
  79. * 申请分销会员
  80. */
  81. public static function apply($post)
  82. {
  83. $time = time();
  84. $data = [
  85. 'user_id' => $post['user_id'],
  86. 'real_name' => $post['real_name'],
  87. 'mobile' => $post['mobile'],
  88. 'province' => $post['province'],
  89. 'city' => $post['city'],
  90. 'district' => $post['district'],
  91. 'reason' => $post['reason'],
  92. 'status' => 0, // 待审核
  93. 'create_time' => $time,
  94. 'update_time' => $time,
  95. ];
  96. return DistributionMemberApply::create($data);
  97. }
  98. /**
  99. * 最新分销申请详情
  100. */
  101. public static function applyDetail($userId)
  102. {
  103. $result = DistributionMemberApply::field(['real_name','mobile','province', 'city', 'district', 'reason', 'denial_reason', 'status'])
  104. ->where('user_id', $userId)
  105. ->order('id', 'desc')
  106. ->findOrEmpty();
  107. if ($result->isEmpty()) {
  108. return [];
  109. }
  110. $result = $result->toArray();
  111. $result['province'] = AreaServer::getAddress($result['province']);
  112. $result['city'] = AreaServer::getAddress($result['city']);
  113. $result['district'] = AreaServer::getAddress($result['district']);
  114. switch ($result['status']) {
  115. case 0:
  116. $result['status_str'] = '已提交,等待客服审核...';
  117. break;
  118. case 1:
  119. $result['status_str'] = '已审核通过';
  120. break;
  121. case 2:
  122. $result['status_str'] = '审核失败,请重新提交审核';
  123. break;
  124. }
  125. return $result;
  126. }
  127. /**
  128. * 分销主页
  129. */
  130. public static function index($userId)
  131. {
  132. // 自身及上级信息
  133. $user_info = self::myLeader($userId);
  134. // 粉丝数(一级/二级)
  135. $fans = User::where([
  136. ['first_leader|second_leader', '=', $userId],
  137. ['del', '=', 0]
  138. ])->count();
  139. //今天的预估收益(待返佣)
  140. $today_earnings = DistributionOrderGoods::whereDay('create_time')
  141. ->where(['status' => 1, 'user_id' => $userId])
  142. ->sum('money');
  143. //本月预估收益(待返佣)
  144. $month_earnings = DistributionOrderGoods::whereMonth('create_time')
  145. ->where(['status' => 1, 'user_id' => $userId])
  146. ->sum('money');
  147. //累计收益(已结算)
  148. $history_earnings = DistributionOrderGoods::where(['status' => 2, 'user_id' => $userId])
  149. ->sum('money');
  150. // 用户分销会员等级
  151. $levelId = Distribution::where('user_id', $userId)->value('level_id');
  152. $levelName = DistributionLevel::getLevelNameTwo($levelId);
  153. $data = [
  154. 'user' => $user_info['user'],
  155. 'leader' => $user_info['leader'],
  156. 'fans' => $fans,
  157. 'able_withdrawal' => $user_info['user']['earnings'],//可提现佣金
  158. 'today_earnings' => round($today_earnings, 2),//今天预估收益
  159. 'month_earnings' => round($month_earnings, 2),//本月预估收益
  160. 'history_earnings' => round($history_earnings, 2),//累计收益
  161. 'level_name' => $levelName
  162. ];
  163. return $data;
  164. }
  165. /***
  166. * 获取自身及上级信息
  167. */
  168. public static function myLeader($userId)
  169. {
  170. $field = 'nickname,avatar,is_distribution,mobile,first_leader,distribution_code,earnings';
  171. $user = User::field($field)->where(['id' => $userId, 'del'=>0])->findOrEmpty();
  172. $first_leader = User::field('nickname,mobile')
  173. ->where(['id' => $user['first_leader'], 'del'=>0])
  174. ->findOrEmpty();
  175. $user['avatar'] = UrlServer::getFileUrl($user['avatar']);
  176. return [
  177. 'user' => $user,
  178. 'leader' => $first_leader,
  179. ];
  180. }
  181. /**
  182. * 填写邀请码
  183. */
  184. public static function code($post)
  185. {
  186. try {
  187. Db::startTrans();
  188. $firstLeader = User::field(['id', 'first_leader', 'second_leader', 'third_leader', 'ancestor_relation','user_integral'])
  189. ->where(['distribution_code' => $post['code']])
  190. ->findOrEmpty();
  191. if($firstLeader->isEmpty()) {
  192. throw new \think\Exception('无效的邀请码');
  193. }
  194. // 上级
  195. $first_leader_id = $firstLeader['id'];
  196. // 上上级
  197. $second_leader_id = $firstLeader['first_leader'];
  198. // 上上上级
  199. $third_leader_id = $firstLeader['second_leader'];
  200. // 拼接关系链
  201. $firstLeader['ancestor_relation'] = $firstLeader['ancestor_relation'] ? : ''; // 清空null值及0
  202. $my_ancestor_relation = $first_leader_id. ',' . $firstLeader['ancestor_relation'];
  203. // 去除两端逗号
  204. $my_ancestor_relation = trim($my_ancestor_relation, ',');
  205. $user = User::findOrEmpty($post['user_id']);
  206. // 旧关系链
  207. if (!empty($user->ancestor_relation)) {
  208. $old_ancestor_relation = $user->id . ',' .$user->ancestor_relation;
  209. } else {
  210. $old_ancestor_relation = $user->id;
  211. }
  212. $data = [
  213. 'first_leader' => $first_leader_id,
  214. 'second_leader' => $second_leader_id,
  215. 'third_leader' => $third_leader_id,
  216. 'ancestor_relation' => $my_ancestor_relation,
  217. 'update_time' => time()
  218. ];
  219. // 更新当前用户的分销关系
  220. User::where(['id' => $post['user_id']])->update($data);
  221. //更新当前用户下级的分销关系
  222. $data = [
  223. 'second_leader' => $first_leader_id,
  224. 'third_leader' => $second_leader_id,
  225. 'update_time' => time()
  226. ];
  227. User::where(['first_leader' => $post['user_id']])->update($data);
  228. //更新当前用户下下级的分销关系
  229. $data = [
  230. 'third_leader' => $first_leader_id,
  231. 'update_time' => time()
  232. ];
  233. User::where(['second_leader' => $post['user_id']])->update($data);
  234. //更新当前用户所有后代的关系链
  235. $posterityArr = User::field('id,ancestor_relation')
  236. ->whereFindInSet('ancestor_relation', $post['user_id'])
  237. ->select()
  238. ->toArray();
  239. $updateData = [];
  240. $replace_ancestor_relation = $post['user_id'] . ','. $my_ancestor_relation;
  241. foreach($posterityArr as $item) {
  242. $updateData[] = [
  243. 'id' => $item['id'],
  244. 'ancestor_relation' => trim(str_replace($old_ancestor_relation, $replace_ancestor_relation, $item['ancestor_relation']), ',')
  245. ];
  246. }
  247. // 批量更新
  248. (new User())->saveAll($updateData);
  249. //邀请会员赠送积分
  250. // $invited_award_integral = ConfigServer::get('marketing','invited_award_integral',0);
  251. // if($invited_award_integral > 0){
  252. // // 增加上级积分
  253. // $firstLeader->user_integral += (int)$invited_award_integral;
  254. // $firstLeader->save();
  255. // // 增加上级积分变动记录
  256. // AccountLogLogic::AccountRecord($firstLeader['id'],$invited_award_integral,1, AccountLog::invite_add_integral);
  257. // }
  258. //通知用户
  259. event('Notice', [
  260. 'scene' => NoticeEnum::INVITE_SUCCESS_NOTICE,
  261. 'params' => [
  262. 'user_id' => $first_leader_id,
  263. 'lower_id' => $post['user_id'],
  264. 'join_time' => date('Y-m-d H:i:s', time())
  265. ]
  266. ]);
  267. Db::commit();
  268. return true;
  269. } catch (\Exception $e) {
  270. Db::rollback();
  271. self::$error = $e->getMessage();
  272. return false;
  273. }
  274. }
  275. /**
  276. * 分销订单
  277. */
  278. public static function order($get)
  279. {
  280. $where[] = ['d.user_id', '=', $get['user_id']];
  281. if (isset($get['status']) && in_array($get['status'], [1,2,3])) {
  282. $where[] = ['d.status', '=', $get['status']];
  283. }
  284. $field = 'd.create_time, d.money, d.goods_num, d.status, d.status as statusDesc, o.order_sn, og.total_pay_price as pay_price, g.image as goods_image, g.name as goods_name, gi.spec_value_str';
  285. $count = DistributionOrderGoods::alias('d')
  286. ->where($where)
  287. ->count();
  288. $lists = DistributionOrderGoods::alias('d')
  289. ->field($field)
  290. ->leftJoin('order_goods og', 'og.id = d.order_goods_id')
  291. ->leftJoin('order o', 'o.id = og.order_id')
  292. ->leftJoin('goods g', 'og.goods_id=g.id')
  293. ->leftJoin('goods_item gi', 'og.item_id=gi.id')
  294. ->where($where)
  295. ->order('d.create_time desc')
  296. ->page($get['page_no'], $get['page_size'])
  297. ->select()
  298. ->toArray();
  299. foreach ($lists as &$item) {
  300. $item['goods_image'] = empty($item['goods_image']) ? '' : UrlServer::getFileUrl($item['goods_image']);
  301. }
  302. $data = [
  303. 'list' => $lists,
  304. 'page' => $get['page_no'],
  305. 'size' => $get['page_size'],
  306. 'count' => $count,
  307. 'more' => is_more($count, $get['page_no'], $get['page_size'])
  308. ];
  309. return $data;
  310. }
  311. /**
  312. * 月度账单
  313. */
  314. public static function monthBill($get)
  315. {
  316. $field = [
  317. "FROM_UNIXTIME(d.create_time,'%Y年%m月') as date",
  318. "FROM_UNIXTIME(d.create_time,'%Y') as year",
  319. "FROM_UNIXTIME(d.create_time,'%m') as month",
  320. 'sum(d.money) as total_money',
  321. 'count(d.id) as order_num'
  322. ];
  323. $count = DistributionOrderGoods::alias('d')
  324. ->field($field)
  325. ->leftJoin('order_goods g', 'g.id = d.order_goods_id')
  326. ->leftJoin('order o', 'o.id = g.order_id')
  327. ->where(['d.user_id' => $get['user_id']])
  328. ->where('d.status', 'in', [1, 2])
  329. ->group('date')
  330. ->count();
  331. $lists = DistributionOrderGoods::alias('d')
  332. ->field($field)
  333. ->leftJoin('order_goods g', 'g.id = d.order_goods_id')
  334. ->leftJoin('order o', 'o.id = g.order_id')
  335. ->where(['d.user_id' => $get['user_id']])
  336. ->where('d.status', 'in', [1, 2])
  337. ->order('d.create_time desc')
  338. ->page($get['page_no'], $get['page_size'])
  339. ->group('date')
  340. ->select()
  341. ->toArray();
  342. $data = [
  343. 'list' => $lists,
  344. 'page' => $get['page_no'],
  345. 'size' => $get['page_size'],
  346. 'count' => $count,
  347. 'more' => is_more($count, $get['page_no'], $get['page_size'])
  348. ];
  349. return $data;
  350. }
  351. /**
  352. * 月度明细
  353. */
  354. public static function monthDetail($get)
  355. {
  356. $where[] = ['d.user_id', '=', $get['user_id']];
  357. $monthStr = $get['year'] . '-' . str_pad($get['month'], 2, '0', STR_PAD_LEFT);
  358. $field = 'd.create_time, d.money, d.goods_num, d.status, d.status as statusDesc, o.order_sn, og.total_pay_price as pay_price, g.image as goods_image, g.name as goods_name, gi.spec_value_str';
  359. $count = DistributionOrderGoods::alias('d')
  360. ->where($where)
  361. ->whereMonth('d.create_time', $monthStr)
  362. ->count();
  363. $lists = DistributionOrderGoods::alias('d')
  364. ->field($field)
  365. ->leftJoin('order_goods og', 'og.id = d.order_goods_id')
  366. ->leftJoin('order o', 'o.id = og.order_id')
  367. ->leftJoin('goods g', 'og.goods_id=g.id')
  368. ->leftJoin('goods_item gi', 'og.item_id=gi.id')
  369. ->where($where)
  370. ->whereMonth('d.create_time', $monthStr)
  371. ->order('d.create_time desc')
  372. ->page($get['page_no'], $get['page_size'])
  373. ->select()
  374. ->toArray();
  375. $data = [
  376. 'list' => $lists,
  377. 'page' => $get['page_no'],
  378. 'size' => $get['page_size'],
  379. 'count' => $count,
  380. 'more' => is_more($count, $get['page_no'], $get['page_size'])
  381. ];
  382. return $data;
  383. }
  384. /**
  385. * Desc: 取消订单后更新分销订单为已失效
  386. * @param $order_id
  387. * @throws Exception
  388. * @throws \think\exception\PDOException
  389. */
  390. public static function setDistributionOrderFail($order_id)
  391. {
  392. //订单取消后更新分销订单为已失效状态
  393. return Db::name('distribution_order_goods d')
  394. ->join('order_goods og', 'og.id = d.order_goods_id')
  395. ->join('order o', 'o.id = og.order_id')
  396. ->where('o.id', $order_id)
  397. ->update([
  398. 'd.status' => DistributionOrderGoodsEnum::STATUS_ERROR,
  399. 'd.update_time' => time(),
  400. ]);
  401. }
  402. /**
  403. * @Notes: 分销佣金列表
  404. * @Author: 张无忌
  405. * @param $get
  406. * @param $user_id
  407. * @return bool|array
  408. */
  409. public static function commission($get, $user_id)
  410. {
  411. try {
  412. $where = [
  413. ['user_id', '=', $user_id],
  414. ['source_type', 'in', AccountLog::earnings_change]
  415. ];
  416. $model = new AccountLog();
  417. $count = $model->where($where)->count();
  418. $lists = $model->field(['id,source_type,change_amount,change_type,create_time'])
  419. ->where($where)
  420. ->order('id', 'desc')
  421. ->page($get['page_no'] ?? 1, $get['page_size'] ?? 20)
  422. ->select();
  423. foreach ($lists as &$item) {
  424. $symbol = $item['change_type'] == 1 ? '+' : '-';
  425. $item['change_amount'] = $symbol.$item['change_amount'];
  426. }
  427. return [
  428. 'list' => $lists,
  429. 'page' => $get['page_no']??1,
  430. 'size' => $get['page_size']??20,
  431. 'count' => $count,
  432. 'more' => is_more($count, $get['page_no']??1, $get['page_size']??20)
  433. ];
  434. } catch (\Exception $e) {
  435. static::$error = $e->getMessage();
  436. return false;
  437. }
  438. }
  439. public static function fixAncestorRelation()
  440. {
  441. try {
  442. $userList = User::select()->toArray();
  443. if (empty($userList)) {
  444. throw new \Exception('没有用户,无需修复');
  445. }
  446. $updateEmptyData = [];
  447. $updateData = [];
  448. foreach($userList as $user) {
  449. $my_ancestor_relation = self::myAncestorRelation($user);
  450. $updateEmptyData[] = ['id' => $user['id'], 'ancestor_relation' => ''];
  451. $updateData[] = ['id' => $user['id'], 'ancestor_relation' => $my_ancestor_relation];
  452. }
  453. // 先清除所有关系链
  454. (new User())->saveAll($updateEmptyData);
  455. // 重新设置关系链
  456. (new User())->saveAll($updateData);
  457. return true;
  458. } catch (\Exception $e) {
  459. self::$error = $e->getMessage();
  460. return false;
  461. }
  462. }
  463. public static function myAncestorRelation($user)
  464. {
  465. if (empty($user['first_leader'])) {
  466. return '';
  467. }
  468. return trim(self::findAncestorRelation($user['first_leader']), ',');
  469. }
  470. public static function findAncestorRelation($id, $flag = true)
  471. {
  472. static $ancestor_relation = '';
  473. if ($flag) {
  474. $ancestor_relation = '';
  475. }
  476. $ancestor_relation .= $id . ',';
  477. $user = User::findOrEmpty($id);
  478. if (empty($user['first_leader'])) {
  479. return $ancestor_relation;
  480. }
  481. return self::findAncestorRelation($user['first_leader'], false);
  482. }
  483. /**
  484. * @notes 获取背景海报
  485. * @return array|mixed|string|null
  486. * @author cjhao
  487. * @date 2021/11/29 11:35
  488. */
  489. public static function getPoster()
  490. {
  491. $poster = ConfigServer::get('invite', 'poster', '/images/share/share_user_bg.png');
  492. $poster = empty($poster) ? $poster : UrlServer::getFileUrl($poster);
  493. return ['poster'=>$poster];
  494. }
  495. }