WechatUser.php 9.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299
  1. <?php
  2. /**
  3. *
  4. * @author: xaboy<365615158@qq.com>
  5. * @day: 2017/12/21
  6. */
  7. namespace app\models\user;
  8. use app\models\store\StoreCouponIssue;
  9. use crmeb\basic\BaseModel;
  10. use crmeb\services\WechatService;
  11. use crmeb\traits\ModelTrait;
  12. use crmeb\services\SystemConfigService;
  13. use app\models\routine\RoutineQrcode;
  14. use app\models\store\StoreCouponUser;
  15. use app\models\store\StoreCouponIssueUser;
  16. /**
  17. * TODO 微信用户Model
  18. * Class WechatUser
  19. * @package app\models\user
  20. */
  21. class WechatUser extends BaseModel
  22. {
  23. /**
  24. * 数据表主键
  25. * @var string
  26. */
  27. protected $pk = 'uid';
  28. /**
  29. * 模型名称
  30. * @var string
  31. */
  32. protected $name = 'wechat_user';
  33. use ModelTrait;
  34. /**
  35. * uid获取小程序Openid
  36. * @param string $uid
  37. * @return bool|mixed
  38. */
  39. public static function getOpenId($uid = '')
  40. {
  41. if ($uid == '') return false;
  42. return self::where('uid', $uid)->value('routine_openid');
  43. }
  44. /**
  45. * TODO 用uid获得openid
  46. * @param $uid
  47. * @param string $openidType
  48. * @return mixed
  49. * @throws \Exception
  50. */
  51. public static function uidToOpenid($uid, $openidType = 'routine_openid')
  52. {
  53. $openid = self::where('uid', $uid)->value($openidType);
  54. return $openid;
  55. }
  56. /**
  57. * TODO 用openid获得uid
  58. * @param $openid
  59. * @param string $openidType
  60. * @return mixed
  61. */
  62. public static function openidTouid($openid, $openidType = 'openid')
  63. {
  64. return self::where($openidType, $openid)->where('user_type', '<>', 'h5')->value('uid');
  65. }
  66. /**
  67. * 小程序创建用户后返回uid
  68. * @param $routineInfo
  69. * @return mixed
  70. */
  71. public static function routineOauth($routine)
  72. {
  73. $routineInfo['nickname'] = filter_emoji($routine['nickName']);//姓名
  74. $routineInfo['sex'] = $routine['gender'];//性别
  75. $routineInfo['language'] = $routine['language'];//语言
  76. $routineInfo['city'] = $routine['city'];//城市
  77. $routineInfo['province'] = $routine['province'];//省份
  78. $routineInfo['country'] = $routine['country'];//国家
  79. $routineInfo['headimgurl'] = $routine['avatarUrl'];//头像
  80. $routineInfo['routine_openid'] = $routine['openId'];//openid
  81. $routineInfo['session_key'] = $routine['session_key'];//会话密匙
  82. $routineInfo['unionid'] = $routine['unionId'];//用户在开放平台的唯一标识符
  83. $routineInfo['user_type'] = 'routine';//用户类型
  84. $routineInfo['channel'] = $routine['channel'];
  85. $spid = 0;//绑定关系uid
  86. $isByQRScan = false;
  87. //获取是否有扫码进小程序
  88. if ($routine['code']) {
  89. if ($info = RoutineQrcode::getRoutineQrcodeFindType($routine['code'])) {
  90. $spid = $info['third_id'];
  91. $isByQRScan = true;
  92. } else {
  93. $spid = $routine['spid'];
  94. }
  95. } else if ($routine['spid']) {
  96. $spid = $routine['spid'];
  97. }
  98. // 判断unionid 存在根据unionid判断
  99. if ($routineInfo['unionid'] != '' && ($uid = self::where(['unionid' => $routineInfo['unionid']])->where('user_type', '<>', 'h5')->value('uid'))) {
  100. self::edit($routineInfo, $uid, 'uid');
  101. $routineInfo['code'] = $spid;
  102. $routineInfo['isPromoter'] = $isByQRScan;
  103. if ($routine['login_type']) {
  104. $routineInfo['login_type'] = $routine['login_type'];
  105. }
  106. User::updateWechatUser($routineInfo, $uid);
  107. } else if ($uid = self::where(['routine_openid' => $routineInfo['routine_openid']])->where('user_type', '<>', 'h5')->value('uid')) { //根据小程序openid判断
  108. self::edit($routineInfo, $uid, 'uid');
  109. $routineInfo['code'] = $spid;
  110. $routineInfo['isPromoter'] = $isByQRScan;
  111. if ($routine['login_type']) {
  112. $routineInfo['login_type'] = $routine['login_type'];
  113. }
  114. User::updateWechatUser($routineInfo, $uid);
  115. } else {
  116. $routineInfo['add_time'] = time();//用户添加时间
  117. $routineInfo = self::create($routineInfo);
  118. $res = User::setRoutineUser($routineInfo, $spid);
  119. $uid = $res->uid;
  120. }
  121. return $uid;
  122. }
  123. /**
  124. * 判断是否是小程序用户
  125. * @param int $uid
  126. * @return bool|int|string
  127. */
  128. public static function isRoutineUser($uid = 0)
  129. {
  130. if (!$uid) return false;
  131. return self::where('uid', $uid)->where('user_type', 'routine')->count();
  132. }
  133. /**
  134. * @param int $uid
  135. * @return int
  136. */
  137. public static function isUserStatus($uid = 0)
  138. {
  139. if (!$uid) return 0;
  140. $user = User::getUserInfo($uid);
  141. return $user['status'];
  142. }
  143. /**
  144. * .添加新用户
  145. * @param $openid
  146. * @return object
  147. */
  148. public static function setNewUser($openid)
  149. {
  150. $userInfo = WechatService::getUserInfo($openid);
  151. if (!isset($userInfo['subscribe']) || !$userInfo['subscribe'] || !isset($userInfo['openid']))
  152. exception('请关注公众号!');
  153. $userInfo['tagid_list'] = implode(',', $userInfo['tagid_list']);
  154. //判断 unionid 是否存在
  155. if (isset($userInfo['unionid'])) {
  156. $wechatInfo = self::where('unionid', $userInfo['unionid'])->find();
  157. unset($userInfo['qr_scene'], $userInfo['qr_scene_str'], $userInfo['qr_scene_str'], $userInfo['subscribe_scene']);
  158. if ($wechatInfo) {
  159. return self::edit($userInfo->toArray(), $userInfo['unionid'], 'unionid');
  160. }
  161. }
  162. self::beginTrans();
  163. $wechatUser = self::create(is_object($userInfo) ? $userInfo->toArray() : $userInfo);
  164. if (!$wechatUser) {
  165. self::rollbackTrans();
  166. exception('用户储存失败!');
  167. }
  168. if (!User::setWechatUser($wechatUser)) {
  169. self::rollbackTrans();
  170. exception('用户信息储存失败!');
  171. }
  172. self::commitTrans();
  173. self::userFirstSubGiveCoupon($openid);
  174. return $wechatUser;
  175. }
  176. /**
  177. * TODO 关注送优惠券
  178. * @param $openid
  179. */
  180. public static function userFirstSubGiveCoupon($openid)
  181. {
  182. $couponIds = StoreCouponIssue::where('status', 1)
  183. ->where('is_give_subscribe', 1)
  184. ->where('is_del', 0)
  185. ->column('id,cid');
  186. if ($couponIds)
  187. foreach ($couponIds as $couponId)
  188. if ($couponId) {
  189. $uid = self::openidToUid($openid);
  190. StoreCouponUser::addUserCoupon($uid, $couponId['cid']);
  191. StoreCouponIssueUser::addUserIssue($uid, $couponId['id']);
  192. }
  193. }
  194. /**
  195. * 订单金额达到预设金额赠送优惠卷
  196. * @param $uid
  197. */
  198. public static function userTakeOrderGiveCoupon($uid, $total_price)
  199. {
  200. $couponIds = StoreCouponIssue::where('status', 1)
  201. ->where('is_full_give', 1)
  202. ->where('is_del', 0)
  203. ->where('status', 1)
  204. ->where('start_time', '<=', time())
  205. ->where('end_time', '>=', time())
  206. ->column('id,cid,full_reduction,remain_count,is_permanent');
  207. if ($couponIds) {
  208. $couponIssueIds = StoreCouponIssueUser::where('uid', $uid)
  209. ->whereIn('issue_coupon_id', array_column($couponIds, 'id'))
  210. ->column('issue_coupon_id');
  211. foreach ($couponIds as $couponId)
  212. if ($couponId && !in_array($couponId['id'], $couponIssueIds))
  213. if ($total_price >= $couponId['full_reduction'] && (!$couponId['is_permanent'] || $couponId['remain_count'] >= 1)) {
  214. StoreCouponUser::addUserCoupon($uid, $couponId['cid']);
  215. StoreCouponIssueUser::addUserIssue($uid, $couponId['id']);
  216. StoreCouponIssue::where('id', $couponId['id'])->update(['remain_count' => $couponId['remain_count'] - 1]);
  217. }
  218. }
  219. }
  220. /**
  221. * 更新用户信息
  222. * @param $openid
  223. * @return bool
  224. */
  225. public static function updateUser($openid)
  226. {
  227. $userInfo = WechatService::getUserInfo($openid);
  228. $userInfo['tagid_list'] = implode(',', $userInfo['tagid_list']);
  229. return self::edit($userInfo->toArray(), $openid, 'openid');
  230. }
  231. /**
  232. * 用户存在就更新 不存在就添加
  233. * @param $openid
  234. */
  235. public static function saveUser($openid)
  236. {
  237. self::be($openid, 'openid') == true ? self::updateUser($openid) : self::setNewUser($openid);
  238. }
  239. /**
  240. * 用户取消关注
  241. * @param $openid
  242. * @return bool
  243. */
  244. public static function unSubscribe($openid)
  245. {
  246. return self::edit(['subscribe' => 0], $openid, 'openid');
  247. }
  248. /**
  249. * TODO 用uid获得Unionid
  250. * @param $uid
  251. * @return mixed
  252. */
  253. public static function uidToUnionid($uid)
  254. {
  255. return self::where('uid', $uid)->value('unionid');
  256. }
  257. /**
  258. * TODO 获取用户信息
  259. * @param $openid
  260. * @param $openidType
  261. * @return array
  262. * @throws \think\db\exception\DataNotFoundException
  263. * @throws \think\db\exception\ModelNotFoundException
  264. * @throws \think\exception\DbException
  265. */
  266. public static function getWechatInfo($openid, $openidType)
  267. {
  268. if (is_numeric($openid)) $openid = self::uidToOpenid($openid);
  269. $wechatInfo = self::where($openidType, $openid)->find();
  270. if (!$wechatInfo) {
  271. self::setNewUser($openid);
  272. $wechatInfo = self::where($openidType, $openid)->find();
  273. }
  274. if (!$wechatInfo) exception('获取用户信息失败!');
  275. return $wechatInfo->toArray();
  276. }
  277. }