UserLevelOrder.php 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419
  1. <?php
  2. namespace app\models\user;
  3. use app\admin\model\user\User as UserModel;
  4. use app\admin\model\user\UserBill;
  5. use app\admin\model\user\UserBill as UserBillAdmin;
  6. use app\admin\model\user\UserLevel;
  7. use app\admin\model\user\UserTaskFinish;
  8. use app\models\store\StoreBargain;
  9. use app\models\store\StoreCart;
  10. use app\models\store\StoreCombination;
  11. use app\models\store\StoreCouponUser;
  12. use app\models\store\StoreOrderCartInfo;
  13. use app\models\store\StoreOrderStatus;
  14. use app\models\store\StorePink;
  15. use app\models\store\StoreProduct;
  16. use app\models\store\StoreSeckill;
  17. use app\models\system\SystemStore;
  18. use app\models\system\SystemUserLevel;
  19. use app\models\system\SystemUserTask;
  20. use crmeb\basic\BaseModel;
  21. use crmeb\repositories\PaymentRepositories;
  22. use crmeb\repositories\UserRepository;
  23. use crmeb\services\JsonService as Json;
  24. use crmeb\services\UtilService as Util;
  25. use crmeb\traits\ModelTrait;
  26. use think\db\exception\DataNotFoundException;
  27. use think\db\exception\DbException;
  28. use think\db\exception\ModelNotFoundException;
  29. use think\Model;
  30. /**
  31. * TODO 会员等级Model
  32. * Class UserLevel
  33. * @package app\models\user
  34. */
  35. class UserLevelOrder extends BaseModel
  36. {
  37. /**
  38. * 数据表主键
  39. * @var string
  40. */
  41. protected $pk = 'id';
  42. /**
  43. * 模型名称
  44. * @var string
  45. */
  46. protected $name = 'user_level_order';
  47. use ModelTrait;
  48. /**
  49. * 获取用户等级人数
  50. * @param $uids
  51. * @return int
  52. */
  53. public static function setUserLevelOrderCount($uids)
  54. {
  55. $model = new self();
  56. if (is_array($uids)) $model = $model->where('uid', 'in', $uids);
  57. else $model = $model->where('uid', $uids);
  58. return $model->count();
  59. }
  60. /**
  61. * 设置查询初始化条件
  62. * @param string $alias 表别名
  63. * @param null $model 模型实例化对象
  64. * @return UserLevel
  65. */
  66. public static function valiWhere($alias = '', $model = null)
  67. {
  68. $model = is_null($model) ? new self() : $model;
  69. if ($alias) {
  70. $model = $model->alias($alias);
  71. $alias .= '.';
  72. }
  73. return $model->where("{$alias}status", 1)->where("{$alias}is_del", 0);
  74. }
  75. /**
  76. * 设置会员等级
  77. * @param int $uid 用户uid
  78. * @param int $level_id 等级id
  79. * @return UserLevel|bool|Model
  80. * @throws DataNotFoundException
  81. * @throws ModelNotFoundException
  82. * @throws \think\exception\DbException
  83. */
  84. public static function setUserLevel($uid, $level_id)
  85. {
  86. $vipinfo = SystemUserLevel::get($level_id);
  87. if (!$vipinfo) return false;
  88. $userinfo = User::find($uid);
  89. if (!$userinfo) return false;
  90. $add_valid_time = (int)$vipinfo->valid_date * 86400;
  91. $uservipinfo = self::valiWhere()->where('uid', $uid)->where('level_id', $level_id)->find();
  92. //检查是否购买过
  93. if ($uservipinfo) {
  94. $stay = 0;
  95. //剩余时间
  96. if (time() < $uservipinfo->valid_time) $stay = $uservipinfo->valid_time - time();
  97. //如果购买过当前等级的会员过期了.从当前时间开始计算
  98. //过期时效: 剩余时间+当前会员等级时间+当前time
  99. $add_valid_time = $stay + $add_valid_time + time();
  100. $data['is_forever'] = $vipinfo->is_forever;
  101. $data['valid_time'] = $add_valid_time;
  102. User::where('uid', $uid)->update(['level' => $level_id]);
  103. return self::where('uid', $uid)->where('level_id', $level_id)->update($data);
  104. } else {
  105. $data = [
  106. 'is_forever' => $vipinfo->is_forever,
  107. 'status' => 1,
  108. 'is_del' => 0,
  109. 'grade' => $vipinfo->grade,
  110. 'uid' => $uid,
  111. 'add_time' => time(),
  112. 'level_id' => $level_id,
  113. 'discount' => $vipinfo->discount,
  114. ];
  115. if ($data['is_forever'])
  116. $data['valid_time'] = 0;
  117. else
  118. $data['valid_time'] = $add_valid_time + time();
  119. $data['mark'] = '尊敬的用户' . $userinfo['nickname'] . '在' . date('Y-m-d H:i:s', time()) . '成为了' . $vipinfo['name'];
  120. $res = self::create($data);
  121. if (!$res) return false;
  122. User::where('uid', $uid)->update(['level' => $level_id]);
  123. return $res;
  124. }
  125. }
  126. /**
  127. * 获取当前用户会员等级返回当前用户等级id
  128. * @param int $uid 用户uid
  129. * @param int $grade 会员id
  130. * @return bool|mixed
  131. * @throws DataNotFoundException
  132. * @throws ModelNotFoundException
  133. * @throws \think\exception\DbException
  134. */
  135. public static function getUserLevel($uid, $grade = 0)
  136. {
  137. $model = self::valiWhere();
  138. if ($grade) $model = $model->where('grade', '<', $grade);
  139. $level = $model->where('uid', $uid)->order('grade desc')->field('level_id,is_forever,valid_time,id,status,grade')->find();
  140. if (!$level) return false;
  141. if ($level->is_forever) return $level->id;
  142. //会员已经过期
  143. if (time() > $level->valid_time) {
  144. if ($level->status == 1) {
  145. $level->status = 0;
  146. $level->save();
  147. }
  148. return self::getUserLevel($uid, $level->grade);
  149. } else
  150. //会员没有过期
  151. return $level->id;
  152. }
  153. /**
  154. * 获取会员详细信息
  155. * @param $id 会员记录id
  156. * @param string $keyName 字段名
  157. * @return array|mixed|string|Model|null
  158. * @throws DataNotFoundException
  159. * @throws ModelNotFoundException
  160. * @throws \think\exception\DbException
  161. */
  162. public static function getUserLevelInfo($id, $keyName = '')
  163. {
  164. $vipinfo = self::valiWhere('a')->where('a.id', $id)->field('l.id,a.add_time,l.discount,a.level_id,l.name,l.money,l.icon,l.is_pay,l.grade')
  165. ->join('system_user_level l', 'l.id=a.level_id')->find();
  166. if ($keyName) if (isset($vipinfo[$keyName])) return $vipinfo[$keyName]; else return '';
  167. return $vipinfo;
  168. }
  169. /**
  170. * 获取当前用户已成为的vip id
  171. * @param $uid 用户id
  172. * @return array
  173. */
  174. public static function getUserLevelIds($uid)
  175. {
  176. return self::valiWhere()->group('level_id')->where('uid', $uid)->order('grade asc')->column('level_id', 'level_id');
  177. }
  178. /**
  179. * 余额支付
  180. * @param $order_id
  181. * @param $uid
  182. * @param string $formId
  183. * @return bool
  184. * @throws \think\Exception
  185. * @throws DataNotFoundException
  186. * @throws ModelNotFoundException
  187. * @throws \think\exception\DbException
  188. */
  189. public static function yuePay($order_id, $uid, $formId = '')
  190. {
  191. $orderInfo = self::where('uid', $uid)->where('order_id', $order_id)->where('is_del', 0)->find();
  192. if (!$orderInfo) return self::setErrorInfo('订单不存在!');
  193. if ($orderInfo['paid']) return self::setErrorInfo('该订单已支付!');
  194. // if($orderInfo['pay_type'] != 'yue') return self::setErrorInfo('该订单不能使用余额支付!');
  195. $userInfo = User::getUserInfo($uid);
  196. if ($userInfo['now_money'] < $orderInfo['pay_price'])
  197. return self::setErrorInfo(['status' => 'pay_deficiency', 'msg' => '余额不足' . floatval($orderInfo['pay_price'])]);
  198. self::beginTrans();
  199. $res1 = false !== User::bcDec($uid, 'now_money', $orderInfo['pay_price'], 'uid');
  200. $res2 = UserBill::expend('购买会员', $uid, 'now_money', 'pay_level', $orderInfo['pay_price'], $orderInfo['id'], $userInfo['now_money'], '余额支付' . floatval($orderInfo['pay_price']) . '元购买会员');
  201. $res3 = self::paySuccess($order_id, 'yue', $formId);//余额支付成功
  202. try {
  203. PaymentRepositories::yuePayLevel($userInfo, $orderInfo);
  204. } catch (\Exception $e) {
  205. self::rollbackTrans();
  206. return self::setErrorInfo($e->getMessage());
  207. }
  208. $res = $res1 && $res2 && $res3;
  209. self::checkTrans($res);
  210. return $res;
  211. }
  212. /**
  213. * 微信支付 为 0元时
  214. * @param $order_id
  215. * @param $uid
  216. * @param string $formId
  217. * @return bool
  218. * @throws \think\Exception
  219. * @throws DataNotFoundException
  220. * @throws ModelNotFoundException
  221. * @throws \think\exception\DbException
  222. */
  223. public static function jsPayPrice($order_id, $uid, $formId = '')
  224. {
  225. $orderInfo = self::where('uid', $uid)->where('order_id', $order_id)->where('is_del', 0)->find();
  226. if (!$orderInfo) return self::setErrorInfo('订单不存在!');
  227. if ($orderInfo['paid']) return self::setErrorInfo('该订单已支付!');
  228. $userInfo = User::getUserInfo($uid);
  229. self::beginTrans();
  230. $res1 = UserBill::expend('购买会员', $uid, 'now_money', 'pay_level', $orderInfo['pay_price'], $orderInfo['id'], $userInfo['now_money'], '微信支付' . floatval($orderInfo['pay_price']) . '元购买会员');
  231. $res2 = self::paySuccess($order_id, 'weixin', $formId);//微信支付为0时
  232. $res = $res1 && $res2;
  233. self::checkTrans($res);
  234. return $res;
  235. }
  236. /**
  237. * //TODO 支付成功后
  238. * @param $orderId
  239. * @param string $paytype
  240. * @param string $formId
  241. * @return bool
  242. * @throws DataNotFoundException
  243. * @throws DbException
  244. * @throws ModelNotFoundException
  245. */
  246. public static function paySuccess($orderId, $paytype = 'weixin', $formId = '')
  247. {
  248. $order = self::where('order_id', $orderId)->find();
  249. $resPink = true;
  250. $res1 = self::where('order_id', $orderId)->update(['paid' => 1, 'pay_type' => $paytype, 'pay_time' => time(), 'is_del' => 0]);//订单改为支付
  251. // if ($order->combination_id && $res1 && !$order->refund_status) $resPink = StorePink::createPink($order);//创建拼团
  252. $oid = self::where('order_id', $orderId)->value('id');
  253. // StoreOrderStatus::status($oid, 'pay_success', '用户付款成功');
  254. $now_money = User::where('uid', $order['uid'])->value('now_money');
  255. UserBill::expend('购买会员', $order['uid'], 'now_money', 'pay_money', $order['pay_price'], $order['id'], $now_money, '支付' . floatval($order['pay_price']) . '元购买会员');
  256. //支付成功后
  257. // event('OrderPaySuccess', [$order, $formId]);
  258. self::save_give_level($order['uid'], $order['level_id']);
  259. // 邀请返利
  260. self::rebates($order['uid'], $order['level_id'],$order['id']);
  261. $res = $res1;
  262. // $res = $res1 && $resPink && UserSpread::setSpreadSure($order['uid']) && User::backOrderBrokerage($order);
  263. return false !== $res;
  264. }
  265. /*
  266. * 购买获得会员等级
  267. * @paran int $uid
  268. * @return json
  269. * */
  270. public static function save_give_level($uid = 0,$level_id = 0)
  271. {
  272. if (!$uid) return Json::fail('缺少参数');
  273. //查询当前选择的会员等级
  274. $systemLevel = \app\admin\model\system\SystemUserLevel::where(['is_show' => 1, 'is_del' => 0, 'id' => $level_id])->find();
  275. if (!$systemLevel) return Json::fail('您选择赠送的会员等级不存在!');
  276. //检查是否拥有此会员等级
  277. $level = UserLevel::where(['uid' => $uid, 'level_id' => $level_id, 'is_del' => 0])->field('valid_time,is_forever')->find();
  278. // if ($level) if (!$level['is_forever'] && time() < $level['valid_time']) return Json::fail('此用户已有该会员等级,无法再次赠送');
  279. //设置会员过期时间
  280. $add_valid_time = (int)$systemLevel->valid_date * 86400;
  281. UserModel::commitTrans();
  282. try {
  283. //保存会员信息
  284. $res = UserLevel::create([
  285. 'is_forever' => $systemLevel->is_forever,
  286. 'status' => 1,
  287. 'is_del' => 0,
  288. 'grade' => $systemLevel->grade,
  289. 'uid' => $uid,
  290. 'add_time' => time(),
  291. 'level_id' => $level_id,
  292. 'discount' => $systemLevel->discount,
  293. 'valid_time' => $systemLevel->discount ? $add_valid_time + time() : 0,
  294. 'mark' => '尊敬的用户【' . UserModel::where('uid', $uid)->value('nickname') . '】在' . date('Y-m-d H:i:s', time()) . '购买会员等级成为' . $systemLevel['name'] . '会员',
  295. ]);
  296. //提取等级任务并记录完成情况
  297. $levelIds = [$level_id];
  298. $lowGradeLevelIds = SystemUserLevel::where('grade', '<', $systemLevel->grade)->where(['is_show' => 1, 'is_del' => 0])->column('id', 'id');
  299. if (count($lowGradeLevelIds)) $levelIds = array_merge($levelIds, $lowGradeLevelIds);
  300. $taskIds = \app\admin\model\system\SystemUserTask::where('level_id', 'in', $levelIds)->column('id', 'id');
  301. $inserValue = [];
  302. foreach ($taskIds as $id) {
  303. $inserValue[] = ['uid' => $uid, 'task_id' => $id, 'status' => 1, 'add_time' => time()];
  304. }
  305. $res = $res && UserModel::where('uid', $uid)->update(['level' => $level_id]);
  306. if ($inserValue) $res && UserTaskFinish::insertAll($inserValue);
  307. $user = UserModel::where('uid', $uid)->find();
  308. if ($systemLevel['integral']>0) {//赠送积分
  309. $edit['integral'] = bcadd($user['integral'], $systemLevel['integral'], 2);
  310. $res2 = UserBillAdmin::income('购买会员增加积分', $user['uid'], 'integral', 'system_add', $systemLevel['integral'], 0, $edit['integral'], '购买会员增加了' . floatval($systemLevel['integral']) . '积分');
  311. try {
  312. UserRepository::adminAddIntegral($user, $systemLevel['integral']);
  313. } catch (\Exception $e) {
  314. // BaseModel::rollbackTrans();
  315. return Json::fail($e->getMessage());
  316. }
  317. }
  318. if ($res) {
  319. UserModel::commitTrans();
  320. return Json::successful('购买成功');
  321. } else {
  322. UserModel::rollbackTrans();
  323. return Json::successful('购买失败');
  324. }
  325. } catch (\Exception $e) {
  326. UserModel::rollbackTrans();
  327. return Json::fail('购买失败');
  328. }
  329. }
  330. // 邀请返利
  331. public static function rebates($uid, $level_id,$order_id){
  332. // $spread_id = User::where('uid', $uid)->value('spread_id');
  333. $userInfo = User::getUserInfo($uid);
  334. $level_info=SystemUserLevel::where('id', $level_id)->find();
  335. if ($level_info['rebate']>0){
  336. $brokerage = bcmul($level_info['rebate'], $level_info['money'], 2);
  337. $balance = bcadd($userInfo['now_money'], $brokerage, 2);
  338. $mark = $userInfo['nickname'] . '成功消费' . floatval($level_info['money']) . '元,购买会员,奖励推广佣金' . floatval($brokerage);
  339. $res1 = UserBill::income('获得推广佣金', $userInfo['spread_uid'], 'now_money', 'brokerage', $brokerage, $order_id, $balance, $mark);
  340. $res2 = self::bcInc($userInfo['spread_uid'], 'now_money', $brokerage, 'uid');
  341. }
  342. return true;
  343. }
  344. /**
  345. * 生成订单唯一id
  346. * @param $uid 用户uid
  347. * @return string
  348. */
  349. public static function getNewOrderId()
  350. {
  351. do {
  352. list($msec, $sec) = explode(' ', microtime());
  353. $msectime = number_format((floatval($msec) + floatval($sec)) * 1000, 0, '', '');
  354. $orderId = 'hy' . $msectime . mt_rand(10000, 99999);
  355. } while (self::be(['order_id' => $orderId]));// $orderId = 'wx' . $msectime . mt_rand(10000, 99999);
  356. return $orderId;
  357. }
  358. // 创建会员订单
  359. public static function CreateOrder($uid, $level_id,$recharge_type)
  360. {
  361. self::beginTrans();
  362. try {
  363. // `grade` int(10) unsigned zerofill NOT NULL COMMENT '会员等级',
  364. // `price` decimal(8,2) unsigned NOT NULL DEFAULT '0.00' COMMENT '充值金额',
  365. // `give_price` decimal(8,2) NOT NULL DEFAULT '0.00' COMMENT '购买赠送金额',
  366. // `recharge_type` varchar(32) NOT NULL COMMENT '充值类型',
  367. // `paid` tinyint(3) unsigned NOT NULL DEFAULT '0' COMMENT '是否支付0未支付1已支付',
  368. // `pay_time` timestamp NULL DEFAULT NULL COMMENT '充值支付时间',
  369. // `create_time` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT '充值时间',
  370. // `refund_price` decimal(10,2) unsigned DEFAULT '0.00' COMMENT '退款金额',
  371. $levelInfo = SystemUserLevel::get($level_id);
  372. if ($levelInfo['is_spread']==1){ //限定邀请
  373. $userSpread = \app\admin\model\user\User::where('uid', $uid)->value('spread_id');
  374. if ($userSpread<=0){
  375. return self::setErrorInfo('只有受邀请用户才能购买该等级的会员', true);
  376. }
  377. }
  378. $orderInfo = [
  379. 'uid' => $uid,
  380. 'order_id' => self::getNewOrderId(),
  381. 'level_id' =>$level_id,
  382. 'grade' => $levelInfo['grade'],
  383. 'give_price' => $levelInfo['give_price'],
  384. ];
  385. $order = self::create($orderInfo);
  386. if (!$order) return self::setErrorInfo('订单生成失败!', true);
  387. self::commitTrans();
  388. // StoreOrderStatus::status($order['id'], 'cache_key_create_order', '订单生成');
  389. return $order;
  390. } catch (\PDOException $e) {
  391. self::rollbackTrans();
  392. return self::setErrorInfo('生成订单时SQL执行错误错误原因:' . $e->getMessage());
  393. } catch (\Exception $e) {
  394. self::rollbackTrans();
  395. return self::setErrorInfo('生成订单时系统错误错误原因:' . $e->getMessage());
  396. }
  397. }
  398. }