AgentManageServices.php 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423
  1. <?php
  2. // +----------------------------------------------------------------------
  3. // | CRMEB [ CRMEB赋能开发者,助力企业发展 ]
  4. // +----------------------------------------------------------------------
  5. // | Copyright (c) 2016~2020 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\agent;
  12. use app\jobs\agent\AutoAgentJob;
  13. use app\services\BaseServices;
  14. use app\services\order\StoreOrderServices;
  15. use app\services\order\StoreOrderStatusServices;
  16. use app\services\other\QrcodeServices;
  17. use app\services\system\attachment\SystemAttachmentServices;
  18. use app\services\user\UserBrokerageServices;
  19. use app\services\user\UserExtractServices;
  20. use app\services\user\UserServices;
  21. use crmeb\exceptions\AdminException;
  22. use crmeb\services\{QrcodeService, UploadService, wechat\MiniProgram};
  23. use think\exception\ValidateException;
  24. /**
  25. * 分销员
  26. * Class AgentManageServices
  27. * @package app\services\agent
  28. */
  29. class AgentManageServices extends BaseServices
  30. {
  31. /**
  32. * @param array $where
  33. * @return array
  34. */
  35. public function agentSystemPage(array $where, $is_page = true)
  36. {
  37. /** @var UserServices $userServices */
  38. $userServices = app()->make(UserServices::class);
  39. $data = $userServices->getAgentUserList($where, '*', $is_page);
  40. /** @var UserBrokerageServices $userBrokerageServices */
  41. $userBrokerageServices = app()->make(UserBrokerageServices::class);
  42. foreach ($data['list'] as &$item) {
  43. $item['headimgurl'] = $item['avatar'];
  44. $item['extract_count_price'] = $item['extract'][0]['extract_count_price'] ?? 0;
  45. $item['extract_count_num'] = $item['extract'][0]['extract_count_num'] ?? 0;
  46. $item['spread_name'] = $item['spreadUser']['nickname'] ?? '';
  47. if ($item['spread_name']) {
  48. $item['spread_name'] .= '/' . $item['spread_uid'];
  49. }
  50. $item['spread_count'] = $item['spreadCount'][0]['spread_count'] ?? 0;
  51. $item['order_price'] = $item['order'][0]['order_price'] ?? 0;
  52. $item['order_count'] = $item['order'][0]['order_count'] ?? 0;
  53. $item['broken_commission'] = $userBrokerageServices->getUserFrozenPrice((int)$item['uid']);
  54. if ($item['broken_commission'] < 0)
  55. $item['broken_commission'] = 0;
  56. $item['brokerage_money'] = $item['brokerage'][0]['brokerage_money'] ?? 0;
  57. if ($item['brokerage_price'] > $item['broken_commission'])
  58. $item['brokerage_money'] = bcsub($item['brokerage_price'], $item['broken_commission'], 2);
  59. else
  60. $item['brokerage_money'] = 0;
  61. $item['new_money'] = $item['brokerage_price'];
  62. unset($item['extract'], $item['order'], $item['bill'], $item['spreadUser'], $item['spreadCount']);
  63. }
  64. return $data;
  65. }
  66. /**
  67. * 分销头部信息
  68. * @param $where
  69. * @return array
  70. * @throws \think\db\exception\DataNotFoundException
  71. * @throws \think\db\exception\DbException
  72. * @throws \think\db\exception\ModelNotFoundException
  73. */
  74. public function getSpreadBadge($where)
  75. {
  76. /** @var UserServices $userServices */
  77. $userServices = app()->make(UserServices::class);
  78. $uids = $userServices->getAgentUserIds($where);
  79. //分销员人数
  80. $data['uids'] = $uids;
  81. $data['sum_count'] = count($uids);
  82. $data['spread_sum'] = 0;
  83. $data['extract_price'] = 0;
  84. if ($data['sum_count']) {
  85. //发展会员人数
  86. $data['spread_sum'] = $userServices->getCount([['spread_uid', 'in', $uids]]);
  87. //获取某个用户可提现金额
  88. $data['extract_price'] = $userServices->getSumBrokerage(['uid' => $uids]);
  89. }
  90. //分销员人数
  91. $data['order_count'] = 0;
  92. $data['pay_price'] = 0;
  93. $data['pay_price'] = 0;
  94. $data['extract_count'] = 0;
  95. if ($data['sum_count']) {
  96. /** @var StoreOrderServices $storeOrder */
  97. $storeOrder = app()->make(StoreOrderServices::class);
  98. $order_where = ['uid' => $uids, 'pid' => 0, 'paid' => 1, 'refund_status' => [0, 3]];
  99. //订单总数
  100. $data['order_count'] = $storeOrder->count($order_where);
  101. //订单金额
  102. $data['pay_price'] = $storeOrder->sum($order_where, 'pay_price', true);
  103. //提现次数
  104. $data['extract_count'] = app()->make(UserExtractServices::class)->getCount([['uid', 'in', $uids], ['status', '=', 1]]);
  105. }
  106. return [
  107. [
  108. 'name' => '分销员人数(人)',
  109. 'count' => $data['sum_count'],
  110. 'className' => 'md-contacts',
  111. 'col' => 6,
  112. ],
  113. [
  114. 'name' => '发展会员人数(人)',
  115. 'count' => $data['spread_sum'],
  116. 'className' => 'md-contact',
  117. 'col' => 6,
  118. ],
  119. [
  120. 'name' => '订单数(单)',
  121. 'count' => $data['order_count'],
  122. 'className' => 'md-cart',
  123. 'col' => 6,
  124. ],
  125. [
  126. 'name' => '订单金额(元)',
  127. 'count' => $data['pay_price'],
  128. 'className' => 'md-bug',
  129. 'col' => 6,
  130. ],
  131. [
  132. 'name' => '提现次数(次)',
  133. 'count' => $data['extract_count'],
  134. 'className' => 'md-basket',
  135. 'col' => 6,
  136. ],
  137. [
  138. 'name' => '未提现金额(元)',
  139. 'count' => $data['extract_price'],
  140. 'className' => 'ios-at-outline',
  141. 'col' => 6,
  142. ],
  143. ];
  144. }
  145. /**
  146. * 推广人列表
  147. * @param array $where
  148. * @return mixed
  149. */
  150. public function getStairList(array $where)
  151. {
  152. /** @var UserServices $userServices */
  153. $userServices = app()->make(UserServices::class);
  154. $data = $userServices->getSairList($where);
  155. $store_brokerage_statu = sys_config('store_brokerage_statu');
  156. foreach ($data['list'] as &$item) {
  157. $item['spread_count'] = $item['spreadCount'][0]['spread_count'] ?? 0;
  158. $item['order_count'] = $item['order'][0]['order_count'] ?? 0;
  159. $item['promoter_name'] = $item['is_promoter'] || $store_brokerage_statu == 2 ? '是' : '否';
  160. $item['add_time'] = $item['add_time'] ? date("Y-m-d H:i:s", $item['add_time']) : '';
  161. }
  162. return $data;
  163. }
  164. /**
  165. * 推广人头部信息
  166. * @param array $where
  167. * @return array[]
  168. */
  169. public function getSairBadge(array $where)
  170. {
  171. /** @var UserServices $userServices */
  172. $userServices = app()->make(UserServices::class);
  173. $data['number'] = $userServices->getSairCount($where);
  174. $where['type'] = 1;
  175. $data['one_number'] = $userServices->getSairCount($where);
  176. $where['type'] = 2;
  177. $data['two_number'] = $userServices->getSairCount($where);
  178. $col = $data['two_number'] > 0 ? 4 : 6;
  179. return [
  180. [
  181. 'name' => '总人数(人)',
  182. 'count' => $data['number'],
  183. 'col' => $col,
  184. ],
  185. [
  186. 'name' => '一级人数(人)',
  187. 'count' => $data['one_number'],
  188. 'col' => $col,
  189. ],
  190. [
  191. 'name' => '二级人数(人)',
  192. 'count' => $data['two_number'],
  193. 'col' => $col,
  194. ],
  195. ];
  196. }
  197. /**
  198. * 推广订单
  199. * @param array $where
  200. * @return array
  201. */
  202. public function getStairOrderList(int $uid, array $where)
  203. {
  204. /** @var UserServices $userServices */
  205. $userServices = app()->make(UserServices::class);
  206. $userInfo = $userServices->getUserInfo($uid);
  207. if (!$userInfo) {
  208. return ['count' => 0, 'list' => []];
  209. }
  210. /** @var StoreOrderServices $storeOrder */
  211. $storeOrder = app()->make(StoreOrderServices::class);
  212. $data = $storeOrder->getUserStairOrderList($uid, $where);
  213. if ($data['list']) {
  214. $uids = array_unique(array_column($data['list'], 'uid'));
  215. $userList = [];
  216. if ($uids) {
  217. $userList = $userServices->getColumn([['uid', 'IN', $uids]], 'nickname,phone,avatar,real_name', 'uid');
  218. }
  219. $orderIds = array_column($data['list'], 'id');
  220. $orderChangTimes = [];
  221. if ($orderIds) {
  222. /** @var StoreOrderStatusServices $storeOrderStatus */
  223. $storeOrderStatus = app()->make(StoreOrderStatusServices::class);
  224. $orderChangTimes = $storeOrderStatus->getColumn([['oid', 'IN', $orderIds], ['change_type', '=', 'user_take_delivery']], 'change_time', 'oid');
  225. }
  226. foreach ($data['list'] as &$item) {
  227. $user = $userList[$item['uid']] ?? [];
  228. $item['user_info'] = '';
  229. $item['avatar'] = '';
  230. if (count($user)) {
  231. $item['user_info'] = $user['nickname'] . '|' . ($user['phone'] ? $user['phone'] . '|' : '') . $user['real_name'];
  232. $item['avatar'] = $user['avatar'];
  233. }
  234. $item['brokerage_price'] = $item['spread_uid'] == $uid ? $item['one_brokerage'] : $item['two_brokerage'];
  235. $item['_pay_time'] = $item['pay_time'] ? date('Y-m-d H:i:s', $item['pay_time']) : '';
  236. $item['_add_time'] = $item['add_time'] ? date('Y-m-d H:i:s', $item['add_time']) : '';
  237. $item['take_time'] = ($change_time = $orderChangTimes[$item['id']] ?? '') ? date('Y-m-d H:i:s', $change_time) : '暂无';
  238. }
  239. }
  240. return $data;
  241. }
  242. /**
  243. * 获取永久二维码
  244. * @param $type
  245. * @param $id
  246. * @return array|false|\PDOStatement|string|\think\Model
  247. */
  248. public function wechatCode(int $uid)
  249. {
  250. /** @var QrcodeServices $qrcode */
  251. $qrcode = app()->make(QrcodeServices::class);
  252. $code = $qrcode->getForeverQrcode('spread', $uid);
  253. if (!$code['ticket']) exception('永久二维码获取错误');
  254. return $code;
  255. }
  256. /**
  257. * 查看小程序推广二维码
  258. * @param string $uid
  259. */
  260. public function lookXcxCode(int $uid)
  261. {
  262. $userInfo = app()->make(UserServices::class)->getUserInfo($uid);
  263. if (!$userInfo) {
  264. throw new AdminException('数据不存在');
  265. }
  266. $name = $userInfo['uid'] . '_' . $userInfo['is_promoter'] . '_user.jpg';
  267. /** @var SystemAttachmentServices $systemAttachmentModel */
  268. $systemAttachmentModel = app()->make(SystemAttachmentServices::class);
  269. $imageInfo = $systemAttachmentModel->getInfo(['name' => $name]);
  270. if (!$imageInfo) {
  271. /** @var QrcodeServices $qrcode */
  272. $qrcode = app()->make(QrcodeServices::class);
  273. $resForever = $qrcode->qrCodeForever($uid, 'spread_routine');
  274. if ($resForever) {
  275. $data = 'id=' . $resForever->id . '&spid=0';
  276. $resCode = MiniProgram::appCodeUnlimit($data, '', 280);
  277. $res = ['res' => $resCode, 'id' => $resForever->id];
  278. } else {
  279. $res = false;
  280. }
  281. if (!$res) throw new ValidateException('二维码生成失败');
  282. $uploadType = (int)sys_config('upload_type', 1);
  283. $upload = UploadService::init($uploadType);
  284. if ($upload->to('routine/spread/code')->validate()->stream((string)$res['res'], $name) === false) {
  285. return $upload->getError();
  286. }
  287. $imageInfo = $upload->getUploadInfo();
  288. $imageInfo['image_type'] = $uploadType;
  289. $systemAttachmentModel->attachmentAdd($imageInfo['name'], $imageInfo['size'], $imageInfo['type'], $imageInfo['dir'], $imageInfo['thumb_path'], 1, $imageInfo['image_type'], $imageInfo['time'], 2);
  290. $qrcode->update($res['id'], ['status' => 1, 'time' => time(), 'qrcode_url' => $imageInfo['dir']]);
  291. $urlCode = $imageInfo['dir'];
  292. } else $urlCode = $imageInfo['att_dir'];
  293. return ['code_src' => $urlCode];
  294. }
  295. /**
  296. * 查看H5推广二维码
  297. * @param string $uid
  298. * @return mixed|string
  299. */
  300. public function lookH5Code(int $uid)
  301. {
  302. $userInfo = app()->make(UserServices::class)->getUserInfo($uid);
  303. if (!$userInfo) {
  304. throw new AdminException('数据不存在');
  305. }
  306. $name = $userInfo['uid'] . '_h5_' . $userInfo['is_promoter'] . '_user.jpg';
  307. /** @var SystemAttachmentServices $systemAttachmentModel */
  308. $systemAttachmentModel = app()->make(SystemAttachmentServices::class);
  309. $imageInfo = $systemAttachmentModel->getInfo(['name' => $name]);
  310. if (!$imageInfo) {
  311. $urlCode = QrcodeService::getWechatQrcodePath($uid . '_h5_' . $userInfo['is_promoter'] . '_user.jpg', '?spread=' . $uid);
  312. } else $urlCode = $imageInfo['att_dir'];
  313. return ['code_src' => $urlCode];
  314. }
  315. /**
  316. * 清除推广关系
  317. * @param int $uid
  318. * @return mixed
  319. */
  320. public function delSpread(int $uid)
  321. {
  322. /** @var UserServices $userServices */
  323. $userServices = app()->make(UserServices::class);
  324. if (!$userServices->userExist($uid)) {
  325. throw new AdminException('数据不存在');
  326. }
  327. if ($userServices->update($uid, ['spread_uid' => 0]) !== false)
  328. return true;
  329. else
  330. throw new AdminException('解除失败');
  331. }
  332. /**
  333. * 取消推广资格
  334. * @param int $uid
  335. * @return mixed
  336. */
  337. public function delSystemSpread(int $uid)
  338. {
  339. /** @var UserServices $userServices */
  340. $userServices = app()->make(UserServices::class);
  341. if (!$userServices->userExist($uid)) {
  342. throw new AdminException('数据不存在');
  343. }
  344. if ($userServices->update($uid, ['spread_uid' => 0, 'spread_time' => 0]) !== false)
  345. return true;
  346. else
  347. throw new AdminException('取消失败');
  348. }
  349. /**
  350. * @param $page
  351. * @param $limit
  352. * @param $where
  353. */
  354. public function startRemoveSpread($page, $limit, $where)
  355. {
  356. /** @var UserServices $userServices */
  357. $userServices = app()->make(UserServices::class);
  358. $list = $userServices->getList($where, 'uid,spread_uid,spread_time', $page, $limit);
  359. foreach ($list as $userInfo) {
  360. $userServices->update($userInfo['uid'], ['spread_uid' => 0, 'spread_time' => 0], 'uid');
  361. }
  362. return true;
  363. }
  364. /**
  365. * 取消绑定上级
  366. * @return bool
  367. */
  368. public function removeSpread()
  369. {
  370. //商城分销功能是否开启 0关闭1开启
  371. if (!sys_config('brokerage_func_status')) return true;
  372. //绑定类型
  373. $store_brokergae_binding_status = sys_config('store_brokerage_binding_status', 1);
  374. if ($store_brokergae_binding_status == 1 || $store_brokergae_binding_status == 3) {
  375. return true;
  376. } else {
  377. //分销绑定类型为时间段且没过期
  378. $store_brokerage_binding_time = (int)sys_config('store_brokerage_binding_time', 30) * 24 * 3600;
  379. $spread_time = bcsub((string)time(), (string)$store_brokerage_binding_time, 0);
  380. /** @var UserServices $userServices */
  381. $userServices = app()->make(UserServices::class);
  382. $where = ['not_spread_uid' => 0, 'status' => 1, 'time' => [0, $spread_time], 'time_key' => 'spread_time'];
  383. $count = $userServices->count($where);
  384. $pages = ceil($count / 100);
  385. for ($i = 1; $i <= $pages; $i++) {
  386. AutoAgentJob::dispatch([$i, 100, $where]);
  387. }
  388. }
  389. return true;
  390. }
  391. /**
  392. * 配置绑定类型切换重置绑定时间
  393. * @return bool
  394. */
  395. public function resetSpreadTime()
  396. {
  397. //商城分销功能是否开启 0关闭1开启
  398. if (!sys_config('brokerage_func_status')) return true;
  399. /** @var UserServices $userServices */
  400. $userServices = app()->make(UserServices::class);
  401. $userServices->update(['not_spread_uid' => 0, 'status' => 1], ['spread_time' => time()]);
  402. return true;
  403. }
  404. }