截流自动化的商城平台
Nevar pievienot vairāk kā 25 tēmas Tēmai ir jāsākas ar burtu vai ciparu, tā var saturēt domu zīmes ('-') un var būt līdz 35 simboliem gara.

UserLogic.php 20KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549
  1. <?php
  2. namespace app\admin\logic\user;
  3. use app\admin\validate\user\UserValidate;
  4. use app\api\cache\TokenCache;
  5. use app\common\basics\Logic;
  6. use app\common\enum\ClientEnum;
  7. use app\common\enum\OrderEnum;
  8. use app\common\logic\AccountLogLogic;
  9. use app\common\model\AccountLog;
  10. use app\common\model\Session;
  11. use app\common\model\user\User;
  12. use app\common\model\user\UserLevel;
  13. use app\common\model\user\UserTag;
  14. use app\common\server\UrlServer;
  15. use app\common\model\order\Order;
  16. use think\facade\Db;
  17. class UserLogic extends Logic
  18. {
  19. public static function lists($get)
  20. {
  21. $where[] = ['del','=', '0'];
  22. //查询
  23. if(isset($get['keyword']) && $get['keyword']){
  24. $where[] = [$get['keyword_type'],'like', '%'. trim($get['keyword']) . '%'];
  25. }
  26. //用户状态
  27. if(isset($get['disable']) && $get['disable'] !== ''){
  28. $where[] = ['disable', '=', $get['disable']];
  29. }
  30. //等级查询
  31. if(isset($get['level']) && $get['level'] !== ''){
  32. $where[] = ['level','=',$get['level']];
  33. }
  34. // 标签查询
  35. if(isset($get['tag']) && $get['tag']){
  36. $where[] = ['tag_ids','find in set',$get['tag']];
  37. }
  38. //注册来源
  39. if(isset($get['client']) && $get['client']){
  40. $where[] = ['client','=',$get['client']];
  41. }
  42. //消费金额
  43. if(isset($get['total_amount_start']) && $get['total_amount_start']){
  44. $where[] = ['total_order_amount','>=',$get['total_amount_start']];
  45. }
  46. if(isset($get['total_amount_end']) && $get['total_amount_end']){
  47. $where[] = ['total_order_amount','<=',$get['total_amount_end']];
  48. }
  49. //注册时间
  50. if(isset($get['start_time']) && $get['start_time']!=''){
  51. $where[] = ['create_time','>=',strtotime($get['start_time'])];
  52. }
  53. if(isset($get['end_time']) && $get['end_time']!=''){
  54. $where[] = ['create_time','<=',strtotime($get['end_time'])];
  55. }
  56. $user_count = User::where($where)->count();
  57. $user_list = User::where($where)
  58. ->field('id,sn,nickname,avatar,level,total_order_amount,tag_ids,client,login_time,create_time,user_growth,user_money,earnings,first_leader,disable,user_delete')
  59. ->page($get['page'],$get['limit'])
  60. ->order('id desc')
  61. ->select()
  62. ->toArray();
  63. //会员等级
  64. $user_level = UserLevel::where(['del'=>0])->column('name','id');
  65. // 会员标签
  66. $user_tag = UserTag::where(['del'=>0])->column('name', 'id');
  67. // 注册来源
  68. $client_list = ClientEnum::getClient(true);
  69. foreach ($user_list as &$item){
  70. // $item['nickname'] = htmlspecialchars($item['nickname']);
  71. // 可提现金额
  72. $item['earnings'] = empty($item['earnings']) ? 0 : $item['earnings'];
  73. // 总资产
  74. $item['total_amount'] = $item['user_money'] + $item['earnings'];
  75. // 会员等级
  76. $item['level_name'] = '暂无等级';
  77. if(isset($user_level[$item['level']])){
  78. $item['level_name'] = $user_level[$item['level']];
  79. }
  80. // 头像
  81. if ($item['avatar'] != '/static/common/image/default/user.png') {
  82. $item['abs_avatar'] = $item['avatar'] ? UrlServer::getFileUrl($item['avatar']) : '';
  83. } else {
  84. $item['abs_avatar'] = '/static/common/image/default/user.png';
  85. }
  86. // 会员标签
  87. $item['tag_str'] = '';
  88. if(!empty($item['tag_ids'])) {
  89. $tempArr = explode(',',$item['tag_ids']);
  90. foreach($tempArr as $v) {
  91. $item['tag_str'] .= $user_tag[$v] . ',';
  92. }
  93. $item['tag_str'] = trim( $item['tag_str'], ',');
  94. }
  95. // 注册来源
  96. $item['client_desc'] = $client_list[$item['client']];
  97. // 上级推荐人
  98. $item['first_leader_info'] = User::getUserInfo($item['first_leader']);
  99. //推荐下级人数
  100. $item['fans'] = User::where([
  101. ['first_leader|second_leader', '=', $item['id']],
  102. ['del', '=', 0]
  103. ])->count();
  104. }
  105. return ['count'=>$user_count , 'lists'=>$user_list];
  106. }
  107. public static function setTag($post)
  108. {
  109. try{
  110. User::where([
  111. ['id', 'in', $post['user_ids']]
  112. ])->update([
  113. 'tag_ids' => $post['tag_ids'],
  114. 'update_time' => time()
  115. ]);
  116. return true;
  117. }catch(\Exception $e) {
  118. self::$error = $e->getMessage();
  119. return false;
  120. }
  121. }
  122. public static function getUser($id)
  123. {
  124. $field = [
  125. 'id', 'sn','nickname','avatar','mobile','sex','birthday','tag_ids',
  126. 'remark','user_money','user_growth','user_integral','earnings', 'disable'
  127. ];
  128. $user = User::field($field)->where(['del' => 0, 'id' => $id])->findOrEmpty();
  129. if($user->isEmpty()) {
  130. return [];
  131. }
  132. // 头像
  133. if($user['avatar']) {
  134. $user['avatar'] = UrlServer::getFileUrl($user['avatar']);
  135. }
  136. // 会员标签
  137. $user['tag_ids'] = json_encode(explode(',', $user['tag_ids']));
  138. return $user->toArray();
  139. }
  140. public static function edit($post)
  141. {
  142. try{
  143. $data = [
  144. 'id' => $post['id'],
  145. 'nickname' => $post['nickname'],
  146. 'avatar' => clearDomain($post['avatar']),
  147. 'mobile' => $post['mobile'],
  148. 'birthday' => strtotime($post['birthday']),
  149. 'tag_ids' => $post['select'],
  150. 'remark' => $post['remark'],
  151. 'disable' => $post['disable'],
  152. 'update_time' => time()
  153. ];
  154. User::update($data);
  155. if ($post['disable']) {
  156. $tokens = Session::where(['user_id' => $post['id']])->select()->toArray();
  157. if(count($tokens) > 0) {
  158. foreach ($tokens as $item) {
  159. (new TokenCache($item['token']))->del();
  160. }
  161. Session::where(['user_id' => $post['id']])->update([
  162. 'expire_time' => time(),
  163. 'update_time' => time(),
  164. ]);
  165. }
  166. }
  167. return true;
  168. }catch(\Exception $e) {
  169. self::$error = $e->getMessage();
  170. return false;
  171. }
  172. }
  173. public static function getInfo($id)
  174. {
  175. $field = [
  176. 'id', 'sn', 'nickname', 'avatar', 'birthday', 'sex', 'mobile', 'client',
  177. 'create_time','login_time', 'user_money', 'tag_ids', 'user_growth', 'earnings',
  178. 'first_leader', 'disable', 'user_delete'
  179. ];
  180. $user = User::field($field)->findOrEmpty($id);
  181. if($user->isEmpty()) {
  182. return [];
  183. }
  184. $user =$user->toArray();
  185. $orderWhere = [
  186. 'user_id' => $id,
  187. 'del' => 0,
  188. 'pay_status' => OrderEnum::PAY_STATUS_PAID
  189. ];
  190. // 上级推荐人
  191. $user['first_leader_info'] = User::getUserInfo($user['first_leader']);
  192. // 推荐下级人数
  193. $user['fans'] = User::where([
  194. ['first_leader|second_leader', '=', $user['id']],
  195. ['del', '=', 0],
  196. ])->count();
  197. // 总资产
  198. $user['assets'] = $user['user_money'] + $user['earnings'];
  199. // 总订单数
  200. $user['order_num'] = Order::where($orderWhere)->count();
  201. // 总消费金额
  202. $user['total_amount'] = Order::where($orderWhere)->sum('order_amount');
  203. $user['total_amount'] = round($user['total_amount'] ,2);
  204. // 平均消费单价
  205. $user['avg_amount'] = Order::where($orderWhere)->avg('order_amount');
  206. $user['avg_amount'] = round($user['avg_amount'], 2);
  207. // 头像
  208. $user['avatar'] = UrlServer::getFileUrl($user['avatar']);
  209. // 客户端
  210. $user['client_desc'] = ClientEnum::getClient($user['client']);
  211. // 会员标签
  212. $user_tag = UserTag::where(['del'=>0])->column('name', 'id');
  213. $user['tag_str'] = '';
  214. if(!empty($user['tag_ids'])) {
  215. $tempArr = explode(',',$user['tag_ids']);
  216. foreach($tempArr as $v) {
  217. $user['tag_str'] .= $user_tag[$v] . ',';
  218. }
  219. $user['tag_str'] = trim( $user['tag_str'], ',');
  220. }
  221. return $user;
  222. }
  223. public static function adjustAccount($post)
  224. {
  225. Db::startTrans();
  226. try{
  227. $user = User::findOrEmpty($post['id']);
  228. if($user->isEmpty()) {
  229. throw new \Exception('用户不存在');
  230. }
  231. // 余额调整
  232. if($post['type'] == 'money') {
  233. if(empty($post['money'])) {
  234. throw new \Exception('请输入调整的余额');
  235. }
  236. if(empty($post['money_remark'])) {
  237. throw new \Exception('请输入余额备注');
  238. }
  239. if($post['money_handle'] == 1) { // 增加
  240. $user->user_money = $user->user_money + $post['money'];
  241. $user->save();
  242. AccountLogLogic::AccountRecord($user['id'], $post['money'],1,AccountLog::admin_add_money, $post['money_remark']);
  243. }else{
  244. $user->user_money = $user->user_money - $post['money'];
  245. $user->save();
  246. AccountLogLogic::AccountRecord($user['id'], $post['money'],0,AccountLog::admin_reduce_money, $post['money_remark']);
  247. }
  248. }
  249. // 成长值调整
  250. if($post['type'] == 'growth') {
  251. if(empty($post['growth'])) {
  252. throw new \Exception('请输入调整的成长值');
  253. }
  254. if(empty($post['growth_remark'])) {
  255. throw new \Exception('请输入成长值备注');
  256. }
  257. if($post['growth_handle'] == 1) { // 增加
  258. $user->user_growth = $user->user_growth + $post['growth'];
  259. $user->save();
  260. AccountLogLogic::AccountRecord($user['id'], $post['growth'],1,AccountLog::admin_add_growth, $post['growth_remark']);
  261. }else{
  262. $user->user_growth = $user->user_growth - $post['growth'];
  263. $user->save();
  264. AccountLogLogic::AccountRecord($user['id'], $post['growth'],0,AccountLog::admin_reduce_growth, $post['growth_remark']);
  265. }
  266. }
  267. // 积分调整
  268. if($post['type'] == 'integral') {
  269. if(empty($post['integral'])) {
  270. throw new \Exception('请输入调整的积分');
  271. }
  272. if(empty($post['integral_remark'])) {
  273. throw new \Exception('请输入积分调整备注');
  274. }
  275. if($post['integral_handle'] == 1) { // 增加
  276. $user->user_integral = $user->user_integral + $post['integral'];
  277. $user->save();
  278. AccountLogLogic::AccountRecord($user['id'], $post['integral'],1,AccountLog::admin_add_integral, $post['integral_remark']);
  279. }else{
  280. $user->user_integral = $user->user_integral - $post['integral'];
  281. $user->save();
  282. AccountLogLogic::AccountRecord($user['id'], $post['integral'],0,AccountLog::admin_reduce_integral, $post['integral_remark']);
  283. }
  284. }
  285. // 佣金调整
  286. if($post['type'] == 'earnings') {
  287. if(empty($post['earnings'])) {
  288. throw new \Exception('请输入调整的佣金');
  289. }
  290. if(empty($post['earnings_remark'])) {
  291. throw new \Exception('请输入佣金调整备注');
  292. }
  293. if($post['earnings_handle'] == 1) { // 增加
  294. $user->earnings = $user->earnings + $post['earnings'];
  295. $user->save();
  296. AccountLogLogic::AccountRecord($user['id'], $post['earnings'],1,AccountLog::admin_inc_earnings, $post['earnings_remark']);
  297. }else{
  298. $user->earnings = $user->earnings - $post['earnings'];
  299. $user->save();
  300. AccountLogLogic::AccountRecord($user['id'], $post['earnings'],0,AccountLog::admin_reduce_earnings, $post['earnings_remark']);
  301. }
  302. }
  303. Db::commit();
  304. return true;
  305. }catch(\Exception $e) {
  306. Db::rollback();
  307. self::$error = $e->getMessage();
  308. return false;
  309. }
  310. }
  311. public static function fans($params)
  312. {
  313. $where = [];
  314. // 一级
  315. if ($params['type'] == 'one') {
  316. $where[] = ['first_leader', '=', $params['id']];
  317. }
  318. // 二级粉丝
  319. if ($params['type'] == 'two') {
  320. $where[] = ['second_leader', '=', $params['id']];
  321. }
  322. if(isset($params['keyword']) && !empty($params['keyword'])) {
  323. $where[] = [$params['field'], 'like', '%'. $params['keyword'] . '%'];
  324. }
  325. $lists = User::field('id,sn,nickname,avatar,first_leader')
  326. ->where($where)
  327. ->page($params['page'], $params['limit'])
  328. ->select()
  329. ->toArray();
  330. $count = User::field('id,sn,nickname,avatar,first_leader')
  331. ->where($where)
  332. ->count();
  333. foreach($lists as &$item) {
  334. $item['avatar'] = empty($item['avatar']) ? '' : UrlServer::getFileUrl($item['avatar']);
  335. $item['first_leader_info'] = User::getUserInfo($item['first_leader']);
  336. $item['fans'] = User::where([
  337. ['first_leader|second_leader', '=', $item['id']],
  338. ['del', '=', 0]
  339. ])->count();
  340. }
  341. return [
  342. 'lists' => $lists,
  343. 'count' => $count,
  344. ];
  345. }
  346. public static function adjustLevel($params)
  347. {
  348. try {
  349. $user = User::findOrEmpty($params['id']);
  350. if ($user->isEmpty()) {
  351. throw new \Exception('用户不存在');
  352. }
  353. if (User::UserIsDelete($params['id'])) {
  354. throw new \Exception('用户已注销');
  355. }
  356. $user->level = $params['level'];
  357. $user->remark = $params['remark'];
  358. $user->save();
  359. return true;
  360. } catch(\Exception $e) {
  361. self::$error = $e->getMessage();
  362. return false;
  363. }
  364. }
  365. public static function adjustFirstLeader($params)
  366. {
  367. Db::startTrans();
  368. try {
  369. if (User::UserIsDelete($params['id'])) {
  370. throw new \Exception('用户已注销');
  371. }
  372. switch($params['type']) {
  373. // 指定推荐人
  374. case 'assign':
  375. $formatData = self::assignFirstLeader($params);
  376. break;
  377. // 设置推荐人为系统,即清空上级
  378. case 'system':
  379. $formatData = self::clearFirstLeader($params);
  380. break;
  381. }
  382. $user = User::findOrEmpty($params['id']);
  383. // 旧关系链
  384. if (!empty($user->ancestor_relation)) {
  385. $old_ancestor_relation = $user->id . ',' .$user->ancestor_relation;
  386. } else {
  387. $old_ancestor_relation = $user->id;
  388. }
  389. // 更新当前用户的分销关系
  390. User::where(['id' => $params['id']])->update($formatData);
  391. //更新当前用户下级的分销关系
  392. $data = [
  393. 'second_leader' => $formatData['first_leader'],
  394. 'third_leader' => $formatData['second_leader'],
  395. 'update_time' => time()
  396. ];
  397. User::where(['first_leader' => $params['id']])->update($data);
  398. //更新当前用户下下级的分销关系
  399. $data = [
  400. 'third_leader' => $formatData['first_leader'],
  401. 'update_time' => time()
  402. ];
  403. User::where(['second_leader' => $params['id']])->update($data);
  404. //更新当前用户所有后代的关系链
  405. $posterityArr = User::field('id,ancestor_relation')
  406. ->whereFindInSet('ancestor_relation', $params['id'])
  407. ->select()
  408. ->toArray();
  409. $updateData = [];
  410. $replace_ancestor_relation = $params['id'] . ','. $formatData['ancestor_relation'];
  411. foreach($posterityArr as $item) {
  412. $updateData[] = [
  413. 'id' => $item['id'],
  414. 'ancestor_relation' => trim(str_replace($old_ancestor_relation, $replace_ancestor_relation, $item['ancestor_relation']), ',')
  415. ];
  416. }
  417. // 批量更新
  418. (new User())->saveAll($updateData);
  419. Db::commit();
  420. return true;
  421. } catch(\Exception $e) {
  422. Db::rollback();
  423. self::$error = $e->getMessage();
  424. return false;
  425. }
  426. }
  427. public static function assignFirstLeader($params)
  428. {
  429. if (empty($params['first_id'])) {
  430. throw new \think\Exception('请选择推荐人');
  431. }
  432. $firstLeader = User::field(['id', 'first_leader', 'second_leader', 'third_leader', 'ancestor_relation'])
  433. ->where('id', $params['first_id'])
  434. ->findOrEmpty()
  435. ->toArray();
  436. if(empty($firstLeader)) {
  437. throw new \think\Exception('推荐人不存在');
  438. }
  439. if ($params['first_id'] == $params['id']) {
  440. throw new \think\Exception('不能指定上级是自己');
  441. }
  442. $ancestorArr = explode(',', trim($firstLeader['ancestor_relation']));
  443. if(!empty($ancestorArr) && in_array($params['id'], $ancestorArr)) {
  444. throw new \think\Exception('不能指定推荐人为自己的下级');
  445. }
  446. // 上级
  447. $first_leader_id = $firstLeader['id'];
  448. // 上上级
  449. $second_leader_id = $firstLeader['first_leader'];
  450. // 上上上级
  451. $third_leader_id = $firstLeader['second_leader'];
  452. // 拼接关系链
  453. $firstLeader['ancestor_relation'] = $firstLeader['ancestor_relation'] ?? ''; // 清空null值及0
  454. $my_ancestor_relation = $first_leader_id. ',' . $firstLeader['ancestor_relation'];
  455. // 去除两端逗号
  456. $my_ancestor_relation = trim($my_ancestor_relation, ',');
  457. $data = [
  458. 'first_leader' => $first_leader_id,
  459. 'second_leader' => $second_leader_id,
  460. 'third_leader' => $third_leader_id,
  461. 'ancestor_relation' => $my_ancestor_relation,
  462. 'update_time' => time()
  463. ];
  464. return $data;
  465. }
  466. public static function clearFirstLeader($params)
  467. {
  468. $data = [
  469. 'first_leader' => 0,
  470. 'second_leader' => 0,
  471. 'third_leader' => 0,
  472. 'ancestor_relation' => '',
  473. 'update_time' => time()
  474. ];
  475. return $data;
  476. }
  477. public static function userLists()
  478. {
  479. $where[] = ['del', '=', 0];
  480. $where[] = ['user_delete', '=', 0];
  481. // 用户信息
  482. if (isset($params['keyword']) && !empty($params['keyword'])) {
  483. $where[] = ['sn|nickname', 'like', '%'. $params['keyword'] .'%'];
  484. }
  485. $lists = User::field('id,sn,nickname,id as distribution,user_delete')
  486. ->where($where)
  487. ->page($params['page'], $params['limit'])
  488. ->select()
  489. ->toArray();
  490. $count = User::where($where)->count();
  491. return [
  492. 'count' => $count,
  493. 'lists' => $lists,
  494. ];
  495. }
  496. }