User.php 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359
  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\wap\model\user;
  12. use app\admin\model\wechat\WechatQrcode;
  13. use basic\ModelBasic;
  14. use service\SystemConfigService;
  15. use think\Cookie;
  16. use think\Request;
  17. use think\response\Redirect;
  18. use think\Session;
  19. use think\Url;
  20. use traits\ModelTrait;
  21. use app\wap\model\user\WechatUser;
  22. use service\WechatTemplateService;
  23. class User extends ModelBasic
  24. {
  25. use ModelTrait;
  26. protected $insert = ['add_time', 'add_ip', 'last_time', 'last_ip'];
  27. protected function setAddTimeAttr($value)
  28. {
  29. return time();
  30. }
  31. protected function setAddIpAttr($value)
  32. {
  33. return Request::instance()->ip();
  34. }
  35. protected function setLastTimeAttr($value)
  36. {
  37. return time();
  38. }
  39. protected function setLastIpAttr($value)
  40. {
  41. return Request::instance()->ip();
  42. }
  43. public static function ResetSpread($openid)
  44. {
  45. $uid = WechatUser::openidToUid($openid);
  46. if (self::be(['uid' => $uid, 'is_promoter' => 0])) self::where('uid', $uid)->update(['spread_uid' => 0]);
  47. }
  48. /**
  49. * 绑定用户手机号码修改手机号码用户购买的专题和其他数据
  50. * @param $bindingPhone 绑定手机号码
  51. * @param $uid 当前用户id
  52. * @param $newUid 切换用户id
  53. * @param bool $isDel 是否删除
  54. * @param int $qcodeId 扫码id
  55. * @return bool
  56. * @throws \think\exception\PDOException
  57. */
  58. public static function setUserRelationInfos($bindingPhone, $uid, $newUid, $isDel = true, $qcodeId = 0)
  59. {
  60. self::startTrans();
  61. try {
  62. //修改下级推广人关系
  63. self::where('spread_uid', $uid)->update(['spread_uid' => $newUid]);
  64. //修改用户金额变动记录表
  65. self::getDb('user_bill')->where('uid', $uid)->update(['uid' => $newUid]);
  66. //修改提现记录用户
  67. self::getDb('user_extract')->where('uid', $uid)->update(['uid' => $newUid]);
  68. //修改专题购买记录表
  69. self::getDb('special_buy')->where('uid', $uid)->update(['uid' => $newUid]);
  70. //修改购物车记录表
  71. self::getDb('store_cart')->where('uid', $uid)->update(['uid' => $newUid]);
  72. //修改用户订单记录
  73. self::getDb('store_order')->where('uid', $uid)->update(['uid' => $newUid]);
  74. //修改拼团用户记录
  75. self::getDb('store_pink')->where('uid', $uid)->update(['uid' => $newUid]);
  76. //修改手机用户表记录
  77. self::getDb('phone_user')->where('uid', $uid)->update(['uid' => $newUid]);
  78. //修改会员记录表记录
  79. self::getDb('member_record')->where('uid', $uid)->update(['uid' => $newUid]);
  80. //修改搜索记录表记录
  81. self::getDb('search_history')->where('uid', $uid)->update(['uid' => $newUid]);
  82. //修改用户报名表记录
  83. self::getDb('event_sign_up')->where('uid', $uid)->update(['uid' => $newUid]);
  84. //删除用户表H5用户记录
  85. $user = self::where('uid', $uid)->find();
  86. if ($isDel) self::where('uid', $uid)->delete();
  87. //修改上级推广关系和绑定手机号码
  88. self::where('uid', $newUid)->update(['phone' => $bindingPhone,
  89. 'spread_uid' => $user['spread_uid'],
  90. 'valid_time' => $user['valid_time'],
  91. 'now_money' => $user['now_money'],
  92. 'gold_num' => $user['gold_num'],
  93. 'brokerage_price' => $user['brokerage_price'],
  94. 'is_permanent' => $user['is_permanent'],
  95. 'member_time' => $user['member_time'],
  96. 'overdue_time' => $user['overdue_time']
  97. ]);
  98. if ($qcodeId) WechatQrcode::where('id', $qcodeId)->update(['scan_id' => $newUid]);
  99. self::commit();
  100. \think\Session::clear('wap');
  101. \think\Session::set('loginUid', $newUid, 'wap');
  102. return true;
  103. } catch (\Exception $e) {
  104. self::rollback();
  105. return self::setErrorInfo($e->getMessage());
  106. }
  107. }
  108. /**
  109. * 保存微信用户信息
  110. * @param $wechatUser 用户信息
  111. * @param int $spread_uid 上级用户uid
  112. * @return mixed
  113. */
  114. public static function setWechatUser($wechatUser, $spread_uid = 0)
  115. {
  116. $where = ['nickname' => $wechatUser['nickname'], 'avatar' => $wechatUser['headimgurl'], 'user_type' => 'wechat'];
  117. if (self::be($where)) {
  118. $wechatUser['uid'] = (int)self::where($where)->value('uid');
  119. return $wechatUser;
  120. }
  121. if (isset($wechatUser['uid']) && $wechatUser['uid'] == $spread_uid) $spread_uid = 0;
  122. $data = [
  123. 'account' => 'wx' . date('YmdHis'),
  124. 'pwd' => md5(123456),
  125. 'nickname' => $wechatUser['nickname'] ?: '',
  126. 'avatar' => $wechatUser['headimgurl'] ?: '',
  127. 'user_type' => 'wechat'
  128. ];
  129. //处理推广关系
  130. if ($spread_uid) $data = self::manageSpread($spread_uid, $data);
  131. $res = self::set($data);
  132. if ($res) $wechatUser['uid'] = (int)$res['uid'];
  133. return $wechatUser;
  134. }
  135. /**
  136. * 设置上下级推广人关系
  137. * 普通推广人星级关系由字段 is_promoter 区分, is_promoter = 1 为 0 星, is_promoter = 2 为 1 星,依次类推
  138. * @param $spread_uid 上级推广人
  139. * @param array $data 更新字段
  140. * @param bool $isForever
  141. * @return array|bool
  142. * @throws \think\db\exception\DataNotFoundException
  143. * @throws \think\db\exception\ModelNotFoundException
  144. * @throws \think\exception\DbException
  145. */
  146. public static function manageSpread($spread_uid, $data = [], $isForever = false)
  147. {
  148. $data['spread_uid'] = $spread_uid;
  149. $data['spread_time'] = time();
  150. return $data;
  151. }
  152. /**
  153. * 更新用户数据并绑定上下级关系
  154. * @param $wechatUser
  155. * @param $uid
  156. * @return bool
  157. * @throws \think\db\exception\DataNotFoundException
  158. * @throws \think\db\exception\ModelNotFoundException
  159. * @throws \think\exception\DbException
  160. */
  161. public static function updateWechatUser($wechatUser, $uid)
  162. {
  163. $name = '__login_phone_num' . $uid;
  164. $userinfo = self::where('uid', $uid)->find();
  165. //检查是否有此字段
  166. $spread_uid = isset($wechatUser['spread_uid']) ? $wechatUser['spread_uid'] : 0;
  167. //自己不能成为自己的下线
  168. $spread_uid = $spread_uid == $userinfo->uid ? 0 : $spread_uid;
  169. //手机号码存在直接登陆
  170. if ($userinfo['phone']) {
  171. Cookie::set('__login_phone', 1);
  172. Session::set($name, $userinfo['phone'], 'wap');
  173. Session::set('__login_phone_number', $userinfo['phone'], 'wap');
  174. }
  175. //有推广人直接更新
  176. $editData = [];
  177. //不是推广人,并且有上级id绑定关系
  178. if (!$userinfo->is_promoter && $spread_uid && !$userinfo->spread_uid && $spread_uid != $uid) $editData = self::manageSpread($spread_uid, $editData);
  179. return self::edit($editData, $uid, 'uid');
  180. }
  181. public static function setSpreadUid($uid, $spreadUid)
  182. {
  183. return self::where('uid', $uid)->update(['spread_uid' => $spreadUid]);
  184. }
  185. public static function getUserInfo($uid)
  186. {
  187. $userInfo = self::where('uid', $uid)->find();
  188. if (!Session::has('__login_phone_num' . $uid) && $userInfo['phone']) {
  189. Cookie::set('__login_phone', 1);
  190. Session::set('__login_phone_num' . $uid, $userInfo['phone'], 'wap');
  191. }
  192. if (!$userInfo) {
  193. Session::delete(['loginUid', 'loginOpenid']);
  194. throw new \Exception('未查询到此用户');
  195. }
  196. return $userInfo->toArray();
  197. }
  198. /**
  199. * 获得当前登陆用户UID
  200. * @return int $uid
  201. */
  202. public static function getActiveUid()
  203. {
  204. $uid = null;
  205. if (Session::has('loginUid', 'wap')) $uid = Session::get('loginUid', 'wap');
  206. if (!$uid && Session::has('loginOpenid', 'wap') && ($openid = Session::get('loginOpenid', 'wap')))
  207. $uid = WechatUser::openidToUid($openid);
  208. if (!$uid) exit(exception('请登陆!'));
  209. return $uid;
  210. }
  211. /**
  212. * 一级推广
  213. * @param $orderInfo
  214. * @return bool
  215. */
  216. public static function backOrderBrokerage($orderInfo)
  217. {
  218. $userInfo = User::getUserInfo($orderInfo['uid']);
  219. if (!$userInfo || !$userInfo['spread_uid']) return true;
  220. $storeBrokerageStatu = SystemConfigService::get('store_brokerage_statu') ?: 1;//获取后台分销类型
  221. if ($storeBrokerageStatu == 1) {
  222. if (!User::be(['uid' => $userInfo['spread_uid'], 'is_promoter' => 1])) return true;
  223. }
  224. $brokerageRatio = (SystemConfigService::get('store_brokerage_ratio') ?: 0) / 100;
  225. if ($brokerageRatio <= 0) return true;
  226. $brokeragePrice = bcmul($orderInfo['pay_price'], $brokerageRatio, 2);
  227. if ($brokeragePrice <= 0) return true;
  228. $mark = '一级推广人' .$userInfo['nickname'] . '成功消费' . floatval($orderInfo['pay_price']) . '元,奖励推广佣金' . floatval($brokeragePrice);
  229. self::beginTrans();
  230. $res1 = UserBill::income('获得推广佣金', $userInfo['spread_uid'], 'now_money', 'brokerage', $brokeragePrice, $orderInfo['id'], 0, $mark);
  231. $res2 = self::bcInc($userInfo['spread_uid'], 'brokerage_price', $brokeragePrice, 'uid');
  232. $User = User::getUserInfo($userInfo['spread_uid']);
  233. if ($openid = WechatUser::where('uid', $userInfo['spread_uid'])->value('openid')) {
  234. WechatTemplateService::sendTemplate($openid, WechatTemplateService::USER_BALANCE_CHANGE, [
  235. 'first' => '叮!您收到一笔专题返佣,真是太优秀了!',
  236. 'keyword1' => '返佣金额',
  237. 'keyword2' => $brokeragePrice,
  238. 'keyword3' => date('Y-m-d H:i:s', time()),
  239. 'keyword4' => $User['brokerage_price'],
  240. 'remark' => '点击查看详情'
  241. ], Url::build('wap/spread/commission', [], true, true));
  242. }
  243. $res = $res1 && $res2;
  244. self::checkTrans($res);
  245. if ($res) self::backOrderBrokerageTwo($orderInfo);
  246. return $res;
  247. }
  248. /**
  249. * 二级推广
  250. * @param $orderInfo
  251. * @return bool
  252. */
  253. public static function backOrderBrokerageTwo($orderInfo)
  254. {
  255. $userInfo = User::getUserInfo($orderInfo['uid']);
  256. $userInfoTwo = User::getUserInfo($userInfo['spread_uid']);
  257. if (!$userInfoTwo || !$userInfoTwo['spread_uid']) return true;
  258. $storeBrokerageStatu = SystemConfigService::get('store_brokerage_statu') ?: 1;//获取后台分销类型
  259. if ($storeBrokerageStatu == 1) {
  260. if (!User::be(['uid' => $userInfoTwo['spread_uid'], 'is_promoter' => 1])) return true;
  261. }
  262. $brokerageRatio = bcdiv(SystemConfigService::get('store_brokerage_two'),100,2);
  263. if ($brokerageRatio <= 0) return true;
  264. $brokeragePrice = bcmul($orderInfo['pay_price'], $brokerageRatio, 2);
  265. if ($brokeragePrice <= 0) return true;
  266. $mark = '二级推广人' . $userInfo['nickname'] . '成功消费' . floatval($orderInfo['pay_price']) . '元,奖励推广佣金' . floatval($brokeragePrice);
  267. self::beginTrans();
  268. $res1 = UserBill::income('获得推广佣金', $userInfoTwo['spread_uid'], 'now_money', 'brokerage', $brokeragePrice, $orderInfo['id'], 0, $mark);
  269. $res2 = self::bcInc($userInfoTwo['spread_uid'], 'brokerage_price', $brokeragePrice, 'uid');
  270. $User = User::getUserInfo($userInfoTwo['spread_uid']);
  271. if ($openid = WechatUser::where('uid', $userInfoTwo['spread_uid'])->value('openid')) {
  272. WechatTemplateService::sendTemplate($openid, WechatTemplateService::USER_BALANCE_CHANGE, [
  273. 'first' => '叮!您收到一笔专题返佣,真是太优秀了!',
  274. 'keyword1' => '返佣金额',
  275. 'keyword2' => $brokeragePrice,
  276. 'keyword3' => date('Y-m-d H:i:s', time()),
  277. 'keyword4' => $User['brokerage_price'],
  278. 'remark' => '点击查看详情'
  279. ], Url::build('wap/spread/commission', [], true, true));
  280. }
  281. $res = $res1 && $res2;
  282. self::checkTrans($res);
  283. return $res;
  284. }
  285. /**
  286. * 获取登陆的手机号码
  287. * @param int $uid 用户id
  288. * @param string $phone 用户号码
  289. * @return string
  290. * */
  291. public static function getLogPhone($uid, $phone = null)
  292. {
  293. $name = '__login_phone_num' . $uid;
  294. if (!Cookie::get('__login_phone')) return null;
  295. if (Session::has($name, 'wap')) $phone = Session::get($name, 'wap');
  296. if (is_null($phone)) {
  297. if (Session::has('__login_phone_number', 'wap')) $phone = Session::get('__login_phone_number', 'wap');
  298. }
  299. return $phone;
  300. }
  301. /**
  302. * 获取推广人列表
  303. * @param $where array 查询条件
  304. * @param $uid int 用户uid
  305. * @return array
  306. * */
  307. public static function GetSpreadList($where, $uid)
  308. {
  309. $uids = self::getSpeadUids($uid, true);
  310. if (!count($uids)) return ['list' => [], 'page' => 2];
  311. $model = self::where('uid', 'in', $uids)->field(['nickname', 'phone', 'uid']);
  312. if ($where['search']) $model = $model->where('nickname|uid|phone', 'like', "%$where[search]%");
  313. $list = $model->page((int)$where['page'], (int)$where['limit'])->select();
  314. $list = count($list) ? $list->toArray() : [];
  315. $page = $where['page'] + 1;
  316. foreach ($list as $key => &$item) {
  317. $item['sellout_count'] = UserBill::where(['a.paid' => 1, 'a.is_del' => 0])->where('u.category', 'now_money')->where('u.type', 'brokerage')->alias('u')->join('__STORE_ORDER__ a', 'a.id=u.link_id')
  318. ->where('u.uid', $item['uid'])->count();
  319. $item['sellout_money'] = UserBill::where(['a.paid' => 1, 'a.is_del' => 0])->where('u.category', 'now_money')->where('u.type', 'brokerage')->alias('u')->join('__STORE_ORDER__ a', 'a.id=u.link_id')
  320. ->where('u.uid', $item['uid'])->sum('a.total_price');
  321. }
  322. return compact('list', 'page');
  323. }
  324. /**
  325. * 获取当前用户的下两级
  326. * @param int $uid 用户uid
  327. * @return array
  328. * */
  329. public static function getSpeadUids($uid, $isOne = false)
  330. {
  331. $uids = User::where('spread_uid', $uid)->column('uid');
  332. if ($isOne) return $uids;
  333. $two_uids = count($uids) ? User::where('spread_uid', 'in', $uids)->column('uid') : [];
  334. return array_merge($uids, $two_uids);
  335. }
  336. }