UserBrokerageServices.php 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543
  1. <?php
  2. // +----------------------------------------------------------------------
  3. // | CRMEB [ CRMEB赋能开发者,助力企业发展 ]
  4. // +----------------------------------------------------------------------
  5. // | Copyright (c) 2016~2023 https://www.crmeb.com All rights reserved.
  6. // +----------------------------------------------------------------------
  7. // | Licensed CRMEB并不是自由软件,未经许可不能去掉CRMEB相关版权
  8. // +----------------------------------------------------------------------
  9. // | Author: CRMEB Team <admin@crmeb.com>
  10. // +----------------------------------------------------------------------
  11. namespace app\services\user;
  12. use app\dao\user\UserBrokerageDao;
  13. use app\dao\user\UserDao;
  14. use app\services\BaseServices;
  15. use crmeb\exceptions\ApiException;
  16. /**
  17. * 用户佣金
  18. * Class UserBrokerageServices
  19. * @package app\services\user
  20. * @method getUserFrozenPrice(int $uid) 获取某个用户冻结的佣金
  21. */
  22. class UserBrokerageServices extends BaseServices
  23. {
  24. /**
  25. * 用户记录模板
  26. * @var array[]
  27. */
  28. protected $incomeData = [
  29. 'get_self_member_brokerage' => [
  30. 'title' => '获得自购付费会员佣金',
  31. 'type' => 'self_member_brokerage',
  32. 'mark' => '您成功消费{%pay_price%}元,奖励自购佣金{%number%}',
  33. 'status' => 1,
  34. 'pm' => 1
  35. ],
  36. 'get_member_brokerage' => [
  37. 'title' => '获得下级购买付费会员佣金',
  38. 'type' => 'one_member_brokerage',
  39. 'mark' => '{%nickname%}成功消费{%pay_price%}元,奖励推广佣金{%number%}',
  40. 'status' => 1,
  41. 'pm' => 1
  42. ],
  43. 'get_two_member_brokerage' => [
  44. 'title' => '获得二级购买付费会员佣金',
  45. 'type' => 'two_member_brokerage',
  46. 'mark' => '二级推广人{%nickname%}成功消费{%pay_price%}元,奖励推广佣金{%number%}',
  47. 'status' => 1,
  48. 'pm' => 1
  49. ],
  50. 'get_self_brokerage' => [
  51. 'title' => '获得自购订单佣金',
  52. 'type' => 'self_brokerage',
  53. 'mark' => '您成功消费{%pay_price%}元,奖励自购佣金{%number%}',
  54. 'status' => 1,
  55. 'pm' => 1
  56. ],
  57. 'get_brokerage' => [
  58. 'title' => '获得下级推广订单佣金',
  59. 'type' => 'one_brokerage',
  60. 'mark' => '{%nickname%}成功消费{%pay_price%}元,奖励推广佣金{%number%}',
  61. 'status' => 1,
  62. 'pm' => 1
  63. ],
  64. 'get_two_brokerage' => [
  65. 'title' => '获得二级推广订单佣金',
  66. 'type' => 'two_brokerage',
  67. 'mark' => '二级推广人{%nickname%}成功消费{%pay_price%}元,奖励推广佣金{%number%}',
  68. 'status' => 1,
  69. 'pm' => 1
  70. ],
  71. 'get_user_brokerage' => [
  72. 'title' => '获得推广用户佣金',
  73. 'type' => 'brokerage_user',
  74. 'mark' => '成功推广用户:{%nickname%},奖励推广佣金{%number%}',
  75. 'status' => 1,
  76. 'pm' => 1
  77. ],
  78. 'extract' => [
  79. 'title' => '佣金提现',
  80. 'type' => 'extract',
  81. 'mark' => '{%mark%}',
  82. 'status' => 1,
  83. 'pm' => 0
  84. ],
  85. 'extract_fail' => [
  86. 'title' => '提现失败',
  87. 'type' => 'extract_fail',
  88. 'mark' => '提现失败,退回佣金{%number%}元',
  89. 'status' => 1,
  90. 'pm' => 1
  91. ],
  92. 'brokerage_to_nowMoney' => [
  93. 'title' => '佣金提现到余额',
  94. 'type' => 'extract_money',
  95. 'mark' => '佣金提现到余额{%number%}元',
  96. 'status' => 1,
  97. 'pm' => 0
  98. ],
  99. 'brokerage_refund' => [
  100. 'title' => '退款退佣金',
  101. 'type' => 'refund',
  102. 'mark' => '订单退款扣除佣金{%number%}元',
  103. 'status' => 1,
  104. 'pm' => 0
  105. ],
  106. 'get_staff_brokerage' => [
  107. 'title' => '获得员工推广订单佣金',
  108. 'type' => 'staff_brokerage',
  109. 'mark' => '{%nickname%}成功消费{%pay_price%}元,奖励推广佣金{%number%}',
  110. 'status' => 1,
  111. 'pm' => 1
  112. ],
  113. 'get_agent_brokerage' => [
  114. 'title' => '获得代理推广订单佣金',
  115. 'type' => 'agent_brokerage',
  116. 'mark' => '{%nickname%}成功消费{%pay_price%}元,奖励推广佣金{%number%}',
  117. 'status' => 1,
  118. 'pm' => 1
  119. ],
  120. 'get_division_brokerage' => [
  121. 'title' => '获得事业部推广订单佣金',
  122. 'type' => 'division_brokerage',
  123. 'mark' => '{%nickname%}成功消费{%pay_price%}元,奖励推广佣金{%number%}',
  124. 'status' => 1,
  125. 'pm' => 1
  126. ],
  127. 'get_pink_master_brokerage' => [
  128. 'title' => '获得拼团团长佣金',
  129. 'type' => 'pink_master_brokerage',
  130. 'mark' => '开团成功,奖励团长佣金{%number%}',
  131. 'status' => 1,
  132. 'pm' => 1
  133. ],
  134. 'get_verific_gift_brokerage' => [
  135. 'title' => '获得核销礼包奖励佣金',
  136. 'type' => 'verific_gift_brokerage',
  137. 'mark' => '核销礼包订单,奖励佣金{%number%}元',
  138. 'status' => 1,
  139. 'pm' => 1
  140. ],
  141. 'get_verific_repeat_brokerage' => [
  142. 'title' => '获得核销复购奖励佣金',
  143. 'type' => 'verific_repeat_brokerage',
  144. 'mark' => '核销复购订单,奖励佣金{%number%}元',
  145. 'status' => 1,
  146. 'pm' => 1
  147. ],
  148. 'line_two_reward' => [
  149. 'title' => '获得两条核销线路达标奖励',
  150. 'type' => 'line_two_reward',
  151. 'mark' => '两条核销线路达标(各200单),奖励佣金{%number%}元',
  152. 'status' => 1,
  153. 'pm' => 1
  154. ],
  155. 'line_more_reward' => [
  156. 'title' => '获得多条核销线路达标奖励',
  157. 'type' => 'line_more_reward',
  158. 'mark' => '第三条及以上核销线路达标(200单),奖励佣金{%number%}元',
  159. 'status' => 1,
  160. 'pm' => 1
  161. ],
  162. 'get_same_level_reward' => [
  163. 'title' => '获得平级奖励',
  164. 'type' => 'same_level_reward',
  165. 'mark' => '平级奖励,奖励佣金{%number%}元',
  166. 'status' => 1,
  167. 'pm' => 1
  168. ],
  169. 'service_gift' => [
  170. 'title' => '获得礼包服务费',
  171. 'type' => 'service_gift',
  172. 'mark' => '{%nickname%}成功购买礼包商品{%pay_price%}元,奖励服务费{%number%}元',
  173. 'status' => 1,
  174. 'pm' => 1
  175. ],
  176. 'service_repeat' => [
  177. 'title' => '获得复购服务费',
  178. 'type' => 'service_repeat',
  179. 'mark' => '{%nickname%}成功购买复购商品{%pay_price%}元,奖励服务费{%number%}元',
  180. 'status' => 1,
  181. 'pm' => 1
  182. ],
  183. ];
  184. /**
  185. * UserBrokerageServices constructor.
  186. * @param UserBrokerageDao $dao
  187. */
  188. public function __construct(UserBrokerageDao $dao)
  189. {
  190. $this->dao = $dao;
  191. }
  192. /**
  193. * 写入佣金记录
  194. * @param string $type 写入类型
  195. * @param int $uid
  196. * @param int|string|array $number
  197. * @param int|string $balance
  198. * @param $linkId
  199. * @return bool|mixed
  200. */
  201. public function income(string $type, int $uid, $number, $balance, $linkId)
  202. {
  203. $data = $this->incomeData[$type] ?? null;
  204. if (!$data) {
  205. return true;
  206. }
  207. $data['uid'] = $uid;
  208. $data['balance'] = $balance ?? 0;
  209. $data['link_id'] = $linkId;
  210. if (is_array($number)) {
  211. $key = array_keys($number);
  212. $key = array_map(function ($item) {
  213. return '{%' . $item . '%}';
  214. }, $key);
  215. $value = array_values($number);
  216. $data['number'] = $number['number'] ?? 0;
  217. $data['frozen_time'] = $number['frozen_time'] ?? 0;
  218. $data['mark'] = str_replace($key, $value, $data['mark']);
  219. } else {
  220. $data['number'] = $number;
  221. $data['mark'] = str_replace(['{%number%}'], $number, $data['mark']);
  222. }
  223. $data['add_time'] = time();
  224. return $this->dao->save($data);
  225. }
  226. /**
  227. * 某个用户佣金总和
  228. * @param int $uid
  229. * @param array|string[] $type
  230. * @param string $time
  231. * @return float
  232. * @throws \ReflectionException
  233. */
  234. public function getUserBrokerageSum(int $uid, array $type = ['one_brokerage', 'two_brokerage', 'brokerage_user'], $time = '')
  235. {
  236. $where = ['uid' => $uid];
  237. if ($type) $where['type'] = $type;
  238. if ($time) $where['time'] = $time;
  239. return $this->dao->sum($where, 'number', true);
  240. }
  241. /**
  242. * 退佣金
  243. * @param $order
  244. * @return bool
  245. * @throws \think\db\exception\DataNotFoundException
  246. * @throws \think\db\exception\DbException
  247. * @throws \think\db\exception\ModelNotFoundException
  248. */
  249. public function orderRefundBrokerageBack($order)
  250. {
  251. $id = (int)$order['id'];
  252. $where = [
  253. 'uid' => [$order['spread_uid'], $order['spread_two_uid'], $order['staff_id'], $order['agent_id'], $order['division_id']],
  254. 'type' => ['self_brokerage', 'one_brokerage', 'two_brokerage', 'staff_brokerage', 'agent_brokerage', 'division_brokerage', 'pink_master_brokerage'],
  255. 'link_id' => $id,
  256. 'pm' => 1
  257. ];
  258. $brokerageList = $this->dao->getUserBrokerageList($where);
  259. //子订单
  260. if (!$brokerageList && $order['pid']) {
  261. $where['link_id'] = $order['pid'];
  262. $p_brokerageList = $this->dao->getUserBrokerageList($where);
  263. //主订单已分佣 子订单按订单拆分后计算结果回退
  264. if ($p_brokerageList) {
  265. $brokerageList = [
  266. ['uid' => $order['spread_uid'], 'number' => $order['one_brokerage']],
  267. ['uid' => $order['spread_two_uid'], 'number' => $order['two_brokerage']],
  268. ];
  269. }
  270. }
  271. $res = true;
  272. if ($brokerageList) {
  273. /** @var UserServices $userServices */
  274. $userServices = app()->make(UserServices::class);
  275. $brokerages = $userServices->getColumn([['uid', 'in', array_column($brokerageList, 'uid')]], 'brokerage_price', 'uid');
  276. $brokerageData = [];
  277. foreach ($brokerageList as $item) {
  278. if (!$item['uid']) continue;
  279. $usermoney = $brokerages[$item['uid']] ?? 0;
  280. if ($item['number'] > $usermoney) {
  281. $item['number'] = $usermoney;
  282. }
  283. $res = $res && $userServices->bcDec($item['uid'], 'brokerage_price', (string)$item['number'], 'uid');
  284. $brokerageData[] = [
  285. 'title' => '退款退佣金',
  286. 'uid' => $item['uid'],
  287. 'pm' => 0,
  288. 'add_time' => time(),
  289. 'type' => 'refund',
  290. 'number' => $item['number'],
  291. 'link_id' => $id,
  292. 'balance' => bcsub((string)$usermoney, (string)$item['number'], 2),
  293. 'mark' => '订单退款扣除佣金' . floatval($item['number']) . '元'
  294. ];
  295. }
  296. if ($brokerageData) {
  297. $res = $res && $this->dao->saveAll($brokerageData);
  298. }
  299. //修改佣金冻结时间
  300. $this->dao->update($where, ['frozen_time' => 0]);
  301. }
  302. return $res;
  303. }
  304. /**
  305. * 佣金排行
  306. * @param string $time
  307. * @return array
  308. * @throws \think\db\exception\DataNotFoundException
  309. * @throws \think\db\exception\DbException
  310. * @throws \think\db\exception\ModelNotFoundException
  311. */
  312. public function brokerageRankList(string $time = 'week')
  313. {
  314. $where = ['pm' => 1];
  315. if ($time) {
  316. $where['time'] = $time;
  317. }
  318. [$page, $limit] = $this->getPageValue();
  319. $list = $this->dao->brokerageRankList($where, $page, $limit);
  320. foreach ($list as $key => &$item) {
  321. if (!isset($item['user']) || !$item['user'] || $item['brokerage_price'] <= 0) {
  322. unset($list[$key]);
  323. continue;
  324. }
  325. $item['nickname'] = $item['user']['nickname'] ?? '';
  326. $item['avatar'] = $item['user']['avatar'] ?? '';
  327. if ($item['brokerage_price'] == '0.00' || $item['brokerage_price'] == 0 || !$item['brokerage_price']) {
  328. unset($list[$key]);
  329. }
  330. unset($item['user']);
  331. }
  332. return array_merge($list);
  333. }
  334. /**
  335. * 获取用户排名
  336. * @param int $uid
  337. * @param string $time
  338. * @return false|int|string
  339. * @throws \think\db\exception\DataNotFoundException
  340. * @throws \think\db\exception\DbException
  341. * @throws \think\db\exception\ModelNotFoundException
  342. */
  343. public function getUserBrokerageRank(int $uid, string $time = 'week')
  344. {
  345. $where = ['pm' => 1];
  346. if ($time) {
  347. $where['time'] = $time;
  348. }
  349. $list = $this->dao->brokerageRankList($where);
  350. foreach ($list as $key => &$item) {
  351. if (!isset($item['user']) || !$item['user'] || $item['brokerage_price'] <= 0) {
  352. unset($list[$key]);
  353. }
  354. }
  355. $position_tmp_one = array_column($list, 'uid');
  356. $position_tmp_two = array_column($list, 'brokerage_price', 'uid');
  357. if (!in_array($uid, $position_tmp_one)) {
  358. $position = 0;
  359. } else {
  360. if ($position_tmp_two[$uid] == 0.00) {
  361. $position = 0;
  362. } else {
  363. $position = array_search($uid, $position_tmp_one) + 1;
  364. }
  365. }
  366. return $position;
  367. }
  368. /**
  369. * 推广数据 昨天的佣金 累计提现金额 当前佣金
  370. * @param int $uid
  371. * @return mixed
  372. * @throws \think\db\exception\DataNotFoundException
  373. * @throws \think\db\exception\DbException
  374. * @throws \think\db\exception\ModelNotFoundException
  375. */
  376. public function commission(int $uid)
  377. {
  378. /** @var UserServices $userServices */
  379. $userServices = app()->make(UserServices::class);
  380. if (!$userServices->getUserInfo($uid)) {
  381. throw new ApiException(100026);
  382. }
  383. /** @var UserExtractServices $userExtract */
  384. $userExtract = app()->make(UserExtractServices::class);
  385. $data = [];
  386. $data['uid'] = $uid;
  387. $data['pm'] = 1;
  388. $data['commissionSum'] = $this->getUsersBokerageSum($data);
  389. $data['pm'] = 0;
  390. $data['commissionRefund'] = $this->getUsersBokerageSum($data);
  391. $data['commissionCount'] = $data['commissionSum'] > $data['commissionRefund'] ? bcsub((string)$data['commissionSum'], (string)$data['commissionRefund'], 2) : 0.00;
  392. $data['lastDayCount'] = $this->getUsersBokerageSum($data, 'yesterday');//昨天的佣金
  393. $data['extractCount'] = $userExtract->getUserExtract($uid);//累计提现金额
  394. return $data;
  395. }
  396. /**
  397. * 计算佣金
  398. * @param array $where
  399. * @param int $time
  400. * @return mixed
  401. * @throws \ReflectionException
  402. */
  403. public function getUsersBokerageSum(array $where, $time = 0)
  404. {
  405. $where_data = [
  406. 'status' => 1,
  407. 'pm' => $where['pm'] ?? '',
  408. 'uid' => $where['uid'] ?? '',
  409. 'time' => $where['time'] ?? 0
  410. ];
  411. if ($time) $where_data['time'] = $time;
  412. return $this->dao->getBrokerageSumColumn($where_data);
  413. }
  414. /**
  415. * 佣金明细
  416. * @param $uid
  417. * @param $type
  418. * @return array
  419. * @throws \ReflectionException
  420. * @throws \think\db\exception\DataNotFoundException
  421. * @throws \think\db\exception\DbException
  422. * @throws \think\db\exception\ModelNotFoundException
  423. */
  424. public function getBrokerageList($uid, $type)
  425. {
  426. $where = [];
  427. $where['uid'] = $uid;
  428. [$page, $limit] = $this->getPageValue();
  429. if ($type == 4) {
  430. $where['type'] = ['extract', 'extract_money', 'extract_fail'];
  431. }
  432. /** @var UserExtractServices $userExtractService */
  433. $userExtractService = app()->make(UserExtractServices::class);
  434. $userExtract = $userExtractService->getColumn(['uid' => $uid], 'fail_msg,extract_type', 'id');
  435. $list = $this->dao->getList($where, '*', $page, $limit);
  436. $count = $this->dao->count($where);
  437. $times = [];
  438. if ($list) {
  439. foreach ($list as &$item) {
  440. $item['time'] = $item['time_key'] = $item['add_time'] ? date('Y-m', (int)$item['add_time']) : '';
  441. $item['add_time'] = $item['add_time'] ? date('Y-m-d H:i', (int)$item['add_time']) : '';
  442. $item['fail_msg'] = $item['type'] == 'extract_fail' ? $userExtract[$item['link_id']]['fail_msg'] : '';
  443. if ($type == 4) {
  444. $extract_type = $userExtract[$item['link_id']]['extract_type'] ?? '';
  445. if ($extract_type == 'alipay') {
  446. $item['extract_type'] = '支付宝';
  447. } elseif ($extract_type == 'weixin') {
  448. $item['extract_type'] = '微信';
  449. } elseif ($extract_type == 'bank') {
  450. $item['extract_type'] = '银行卡';
  451. } else {
  452. $item['extract_type'] = '余额';
  453. }
  454. } else {
  455. $item['extract_type'] = '';
  456. }
  457. }
  458. $times = array_merge(array_unique(array_column($list, 'time_key')));
  459. }
  460. return ['list' => $list, 'time' => $times, 'count' => $count];
  461. }
  462. /**
  463. * 前端佣金排行页面数据
  464. * @param int $uid
  465. * @param $type
  466. * @return array
  467. * @throws \think\db\exception\DataNotFoundException
  468. * @throws \think\db\exception\DbException
  469. * @throws \think\db\exception\ModelNotFoundException
  470. */
  471. public function brokerageRank(int $uid, $type)
  472. {
  473. /** @var UserServices $userService */
  474. $userService = app()->make(UserServices::class);
  475. if (!$userService->getUserInfo($uid)) {
  476. throw new ApiException(100026);
  477. }
  478. return [
  479. 'rank' => $this->brokerageRankList($type),
  480. 'position' => $this->getUserBrokerageRank($uid, $type)
  481. ];
  482. }
  483. /**
  484. * 获取某个用户的佣金记录列表
  485. * @param int $uid 用户ID
  486. * @param array $where 查询条件
  487. * @return array
  488. * @throws \ReflectionException
  489. * @throws \think\db\exception\DataNotFoundException
  490. * @throws \think\db\exception\DbException
  491. * @throws \think\db\exception\ModelNotFoundException
  492. */
  493. public function getBrokerageOneList(int $uid, array $where)
  494. {
  495. $where['uid'] = $uid;
  496. // 获取分页参数
  497. $page = $where['page'] ?? 1;
  498. $limit = $where['limit'] ?? 20;
  499. // 查询列表
  500. $list = $this->dao->getList($where, '*', $page, $limit);
  501. $count = $this->dao->count($where);
  502. // 格式化时间
  503. if ($list) {
  504. foreach ($list as &$item) {
  505. $item['add_time'] = $item['add_time'] ? date('Y-m-d H:i:s', (int)$item['add_time']) : '';
  506. if (isset($item['user'])) {
  507. $item['user_nickname'] = $item['user']['nickname'] ?? '';
  508. $item['user_phone'] = $item['user']['phone'] ?? '';
  509. } else {
  510. $item['user_nickname'] = '';
  511. $item['user_phone'] = '';
  512. }
  513. }
  514. }
  515. return [
  516. 'count' => $count,
  517. 'limit' => $limit,
  518. 'page' => $page,
  519. 'data' => $list
  520. ];
  521. }
  522. }