UserSignService.php 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523
  1. <?php
  2. namespace App\Modules\User\Services;
  3. use App\Modules\Activity\Services\ActivitySwitchService;
  4. use App\Modules\Subscribe\Models\Order;
  5. use App\Modules\User\Models\User;
  6. use App\Modules\User\Models\UserSign;
  7. use App\Modules\OfficialAccount\Models\ForceSubscribeUsers;
  8. use Hashids;
  9. use App\Modules\Book\Services\BookConfigService;
  10. use DB;
  11. use Redis;
  12. use App\Modules\User\Services\UserService;
  13. use App\Modules\Activity\Services\ActivityService;
  14. class UserSignService
  15. {
  16. protected $table = 'user_sign';
  17. protected $fillable = ['uid', 'price', 'day', 'sign_time'];
  18. /**
  19. * 用户是否已签到
  20. * @param $uid
  21. * @param $day
  22. * @return mixed
  23. */
  24. public static function isSign($uid)
  25. {
  26. //return UserSign::isSign($uid, $day);
  27. $sign_day = ReadRecordService::getSignDay($uid);
  28. //异常
  29. if ($sign_day == -1) {
  30. return true;
  31. }
  32. if ($sign_day && $sign_day == date('Y-m-d')) {
  33. return true;
  34. }
  35. return false;
  36. }
  37. /**
  38. * 新用户首次签到获得书币调整为30书币,连续签到2天后,赠送书币量提升至50,断签后重新按30开始
  39. * 用户是否已签到(升级版)
  40. * @param $uid
  41. * @param $day
  42. * @return mixed
  43. */
  44. public static function isSigni($uid, $day)
  45. {
  46. $day_2 = date('Y-m-d', strtotime($day) - 86400 * 2);
  47. $sign_stat = UserSign::where('uid', $uid)->where('day', '>=', $day_2)->select('day')->orderBy('id', 'desc')->limit(3)->get();
  48. if (!$sign_stat->isEmpty()) {
  49. $day_1 = date('Y-m-d', strtotime($day) - 86400);
  50. $sign_day = [];
  51. foreach ($sign_stat as $v) {
  52. $sign_day[] = $v->day;
  53. }
  54. if (in_array($day, $sign_day)) {
  55. return 0;
  56. }
  57. if (in_array($day_1, $sign_day) && in_array($day_2, $sign_day)) {
  58. return 50;
  59. }
  60. return 30;
  61. }
  62. return 30;
  63. }
  64. /**
  65. * 用户签到记录
  66. */
  67. public static function getUserSignRecord($uid)
  68. {
  69. return UserSign::getUserSignRecord($uid);
  70. }
  71. /**
  72. * 签到
  73. * @param $uid
  74. * @param $day
  75. * @return mixed
  76. */
  77. public static function sign($uid, $day)
  78. {
  79. \Log::info('sign:uid:' . $uid . ' day:' . $day);
  80. //查看签到日期
  81. $sign_day = ReadRecordService::getSignDay($uid);
  82. if ($sign_day == -1) {
  83. return false;
  84. }
  85. //已经签过到
  86. if ($sign_day == $day) {
  87. return false;
  88. }
  89. $count = ReadRecordService::getSignCountSimple($uid);
  90. //记录签到日期
  91. if ($sign_day && $sign_day == date('Y-m-d', time() - 86400)) {
  92. $continue = true;
  93. //昨天有签过到
  94. ReadRecordService::sign((int)$uid, true);
  95. $count += 1;
  96. } else {
  97. $continue = false;
  98. //昨天没有签过到
  99. ReadRecordService::sign((int)$uid, false);
  100. $count = 1;
  101. }
  102. $return_fee = $fee = 30;
  103. //连续签到两天 50书币
  104. if ($continue && $count >= 3) {
  105. $return_fee = $fee = 50;
  106. }
  107. if ($count % 15 == 7) {
  108. $fee += 100;
  109. }
  110. if ($count % 15 == 0) {
  111. $fee += 150;
  112. }
  113. UserService::addBalance($uid, $fee, 0, $fee);
  114. // 先扔到redis里面,异步更新user_sign表
  115. $use_redis_user_sign = true;
  116. if ($use_redis_user_sign) {
  117. $sign_data = ['uid' => $uid, 'price' => $fee, 'day' => $day, 'sign_time' => time(), 'created_at' => date('Y-m-d H:i:s'), 'updated_at' => date('Y-m-d H:i:s')];
  118. Redis::sadd('user_sign:uid', $uid);
  119. Redis::hset('user_sign:uid:info', $uid, json_encode($sign_data));
  120. ReadRecordService::setSignInfo($uid, json_encode($sign_data));
  121. } else {
  122. UserSign::sign($uid, $day, $fee);
  123. }
  124. return $return_fee;
  125. }
  126. public static function signV2($uid, $day)
  127. {
  128. \Log::info('signv2:uid:' . $uid . ' day:' . $day);
  129. //查看签到日期
  130. $sign_day = ReadRecordService::getSignDay($uid);
  131. if ($sign_day == -1) {
  132. return false;
  133. }
  134. //已经签过到
  135. if ($sign_day == $day) {
  136. return false;
  137. }
  138. $count = ReadRecordService::getSignCountSimple($uid);
  139. //记录签到日期
  140. if ($sign_day && $sign_day == date('Y-m-d', time() - 86400)) {
  141. $continue = true;
  142. //昨天有签过到
  143. ReadRecordService::sign((int)$uid, true);
  144. $count += 1;
  145. } else {
  146. $continue = false;
  147. //昨天没有签过到
  148. ReadRecordService::sign((int)$uid, false);
  149. $count = 1;
  150. }
  151. $fee = 30;
  152. if ($count % 7 == 1 && $count<=7) {
  153. $fee = 30;
  154. } elseif ($count % 7 == 3) {
  155. $fee = 120;
  156. } elseif ($count % 7 == 0) {
  157. $fee = 150;
  158. } else {
  159. $fee = 50;
  160. }
  161. UserService::addBalance($uid, $fee, 0, $fee);
  162. // 先扔到redis里面,异步更新user_sign表
  163. $use_redis_user_sign = true;
  164. if ($use_redis_user_sign) {
  165. $sign_data = ['uid' => $uid, 'price' => $fee, 'day' => $day, 'sign_time' => time(), 'created_at' => date('Y-m-d H:i:s'), 'updated_at' => date('Y-m-d H:i:s')];
  166. Redis::sadd('user_sign:uid', $uid);
  167. Redis::hset('user_sign:uid:info', $uid, json_encode($sign_data));
  168. ReadRecordService::setSignInfo($uid, json_encode($sign_data));
  169. } else {
  170. UserSign::sign($uid, $day, $fee);
  171. }
  172. return $fee;
  173. }
  174. public static function getUserSignVersion($uid)
  175. {
  176. list($version, $sign_day, $count) = ReadRecordService::getByMultiField($uid, 'sign_version', 'sign_day', 'sign_counts');
  177. if ($version == 'v2') {
  178. return 'v2';
  179. }
  180. if ($version == 'v1') {
  181. if ($sign_day == date('Y-m-d')) {
  182. return $version;
  183. }
  184. if ($sign_day == date('Y-m-d', time() - 86400)) {
  185. if ($count % 15 == 0 ) {
  186. ReadRecordService::setByMultiField($uid, ['sign_counts' => 0, 'sign_version' => 'v2']);
  187. return 'v2';
  188. }
  189. return $version;
  190. }
  191. ReadRecordService::setByMultiField($uid, ['sign_version' => 'v2']);
  192. return 'v2';
  193. }
  194. if (!$count || !$sign_day) {
  195. ReadRecordService::setByMultiField($uid, ['sign_version' => 'v2']);
  196. return 'v2';
  197. }
  198. if ($sign_day == date('Y-m-d')) {
  199. self::setUserSignVersion($uid, 'v1');
  200. return 'v1';
  201. }
  202. if ($sign_day == date('Y-m-d', time() - 86400)) {
  203. if ($count % 15 == 0) {
  204. ReadRecordService::setByMultiField($uid, ['sign_counts' => 0, 'sign_version' => 'v2']);
  205. return 'v2';
  206. }
  207. self::setUserSignVersion($uid, 'v1');
  208. return 'v1';
  209. }
  210. self::setUserSignVersion($uid, 'v2');
  211. return 'v2';
  212. }
  213. public static function setUserSignVersion($uid, $version)
  214. {
  215. ReadRecordService::setByField($uid, 'sign_version', $version);
  216. }
  217. public static function signToday($uid,$version='')
  218. {
  219. if(!$version){
  220. $version = self::getUserSignVersion($uid);
  221. }
  222. if($version == 'v1'){
  223. return self::sign($uid, date('Y-m-d'));
  224. }
  225. if($version == 'v2'){
  226. return self::signV2($uid, date('Y-m-d'));
  227. }
  228. return 0;
  229. }
  230. /**
  231. * 新签到回复
  232. * @param $openid
  233. * @param bool $check_sign
  234. * @param int $price
  235. * @return string
  236. */
  237. public static function userSignReturnContent3($openid, $distribution_channel_id = '')
  238. {
  239. $content = '';
  240. $day = date('Y-m-d');
  241. $user = ForceSubscribeUsers::getOneForceSubscribeUsersByOpenid($openid);
  242. if ($user) {
  243. $user_wechat = User::where('id', $user->uid)->first();
  244. $new_user_activity_content = '';
  245. if (!Order::where('uid', $user->uid)->where('status', 'PAID')->select('id')->first()) {
  246. $new_user_activity_content = self::newUserActivity($user);
  247. }
  248. $encode_distribution_channel_id = encodeDistributionChannelId($user->distribution_channel_id);
  249. $attach_content = self::signCallBackPushActivityInfo($user->uid, $user->distribution_channel_id);
  250. $continueReadUrl = env('PROTOCOL') . '://site' . $encode_distribution_channel_id . '.' . env('WECHAT_CUSTOM_HOST') . '.com/continue';
  251. $sign_url = env('PROTOCOL') . '://site' . $encode_distribution_channel_id . '.' . env('WECHAT_CUSTOM_HOST') . '.com/sign';
  252. $sign_stat = self::isSign($user->uid);
  253. if ($sign_stat) {
  254. $content = '今日已经签到过了,明日继续签到得书币哦~';
  255. //$content .= '<a href=' . '"' . $continueReadUrl . '"' . '> ☞ 点我继续上次阅读</a>';
  256. } else {
  257. $content = '尊敬的会员:' . ($user_wechat ? $user_wechat->nickname : '') . "\n\n" . "<a href='" . $sign_url . "'>💰点击此处签到领书币</a>";
  258. //$content .= '继续阅读\n\n<a href=' . '"' . $continueReadUrl . '"' . '> ☞ 点我继续上次阅读</a>';
  259. }
  260. $res = ReadRecordService::getReadRecord($user->uid);
  261. // foreach ($res as $key => $record) {
  262. // if ($key == 1) break;
  263. // $url = env('PROTOCOL') . '://site' . $encode_distribution_channel_id . '.' . env('CUSTOM_HOST') . '.com/reader?bid=' . Hashids::encode($record['bid']) . '&cid=' . $record['cid'];
  264. // $content .= "\n\n" . '<a href="' . $url . '"> ☞ 《' . $record['book_name'] . '》</a>';
  265. // }
  266. $read_bid_arr = [];
  267. $read_bid_arr[] = -1;
  268. foreach ($res as $vbook) {
  269. $read_bid_arr[] = $vbook['bid'];
  270. }
  271. $user_detail = UserService::getById($user->uid);
  272. $sign_recomm_bid_key = '男频';
  273. if ($user_detail && isset($user_detail->sex)) {
  274. if ($user_detail->sex == 2) {
  275. $sign_recomm_bid_key = '女频';
  276. }
  277. }
  278. $hot_book_num = 3;
  279. $recomm_books = BookConfigService::getSignRecommendBooks($read_bid_arr, $sign_recomm_bid_key, $hot_book_num);
  280. $content .= "\n\n" . '热门书籍推荐';
  281. if ($recomm_books) {
  282. foreach ($recomm_books as $book) {
  283. $url = env('PROTOCOL') . '://site' . $encode_distribution_channel_id . '.' . env('WECHAT_CUSTOM_HOST') . '.com/reader?bid=' . Hashids::encode($book->bid) . '&cid=' . $book->first_cid;
  284. $content .= "\n\n" . '<a href="' . $url . '"> ☞ 《' . $book->book_name . '》</a>';
  285. }
  286. }
  287. //$content .= "\n\n" . '为方便下次阅读,请置顶公众号';
  288. $content .= "\n\n" . '为方便下次阅读,请<a href="' . 'https://help.' . env('WECHAT_CUSTOM_HOST') . '.com/top.html"' . '>置顶公众号</a>';
  289. if ($attach_content) {
  290. $content .= "\n\n" . $attach_content;
  291. }
  292. if ($new_user_activity_content) {
  293. $content .= $new_user_activity_content;
  294. }
  295. } // 空用户推默认的文案
  296. else {
  297. $encode_distribution_channel_id = encodeDistributionChannelId($distribution_channel_id);
  298. $url = env('PROTOCOL') . '://site' . $encode_distribution_channel_id . '.' . env('WECHAT_CUSTOM_HOST') . '.com/sign';
  299. $content = "尊敬的会员:\n\n" . '<a href="' . $url . '"> 💰点击此处签到领书币</a>';
  300. }
  301. return $content;
  302. }
  303. private static function newUserActivity($user)
  304. {
  305. $content = '';
  306. $status = self::newUserActivityStatus($user->uid);
  307. $record = [];
  308. //新关未付费用户42小时后充推送活动 68元的活动 文案:全年免费看书
  309. if (strtotime($user->created_at) + 42 * 3600 < time() &&
  310. $user->distribution_channel_id == 123 &&
  311. !in_array(1, $status)
  312. ) {
  313. $url = env('PROTOCOL') . '://site' . encodeDistributionChannelId($user->distribution_channel_id) . '.' . env('CUSTOM_HOST') . '.com/sale/seYearActivity?fromtype=signcallback_forever&send_time=' . time();
  314. $content .= "\n\n" . '<a href="' . $url . '"> 💰全年免费看书</a>';
  315. $record[] = ['uid' => $user->uid, 'type' => 1, 'created_at' => date('Y-m-d H:i:s'), 'updated_at' => date('Y-m-d H:i:s')];
  316. }
  317. //20小时的 文字链的文案:书币充值特惠
  318. if (strtotime($user->created_at) + 20 * 3600 < time() && !in_array(2, $status)) {
  319. $url = env('PROTOCOL') . '://site' . encodeDistributionChannelId($user->distribution_channel_id) . '.' . env('CUSTOM_HOST') . '.com/sale/newUserSale?fromtype=signcallback_newUserSale&send_time=' . time();
  320. $content .= "\n\n" . '<a href="' . $url . '"> 💰书币充值特惠</a>';
  321. $record[] = ['uid' => $user->uid, 'type' => 2, 'created_at' => date('Y-m-d H:i:s'), 'updated_at' => date('Y-m-d H:i:s')];
  322. }
  323. //36小时的。文案:充9.9送2000书币
  324. if (strtotime($user->created_at) + 36 * 3600 < time() && !in_array(3, $status)) {
  325. $url = env('PROTOCOL') . '://site' . encodeDistributionChannelId($user->distribution_channel_id) . '.' . env('CUSTOM_HOST') . '.com/sale/newUserActivity?fromtype=signcallback_newUserActivity&send_time=' . time();
  326. $content .= "\n\n" . '<a href="' . $url . '"> 💰充9.9得2000书币</a>';
  327. $record[] = ['uid' => $user->uid, 'type' => 3, 'created_at' => date('Y-m-d H:i:s'), 'updated_at' => date('Y-m-d H:i:s')];
  328. }
  329. self::recordnewUserActivityPush($record);
  330. return $content;
  331. }
  332. private static function newUserActivityStatus($uid)
  333. {
  334. $data = [-1];
  335. if (!$uid) {
  336. return $data;
  337. }
  338. $result = DB::table('user_sign_push_activity')->where('uid', $uid)->select('type')->get();
  339. if ($result) {
  340. foreach ($result as $v) {
  341. array_push($data, $v->type);
  342. }
  343. }
  344. return $data;
  345. }
  346. private static function recordnewUserActivityPush($data)
  347. {
  348. if ($data) {
  349. try {
  350. DB::table('user_sign_push_activity')->insert($data);
  351. } catch (\Exception $e) {
  352. }
  353. }
  354. }
  355. /**
  356. *
  357. * @param $openid
  358. * @return string
  359. */
  360. public static function userSignReturnContent($openid)
  361. {
  362. $content = '';
  363. $day = date('Y-m-d');
  364. $user = ForceSubscribeUsers::getOneForceSubscribeUsersByOpenid($openid);
  365. if ($user) {
  366. $encode_distribution_channel_id = encodeDistributionChannelId($user->distribution_channel_id);
  367. $attach_content = self::signCallBackPushActivityInfo($user->uid);
  368. $continueReadUrl = env('PROTOCOL') . '://site' . $encode_distribution_channel_id . '.' . env('CUSTOM_HOST') . '.com/continue';
  369. $is_sign = self::isSign($user->uid);
  370. if ($is_sign) {
  371. $content = '今日已经签到过了,明日继续签到得书币哦~' . "\n\n" . '<a href=' . '"' . $continueReadUrl . '"' . '> >>点我继续上次阅读</a>';
  372. /*if($attach_content){
  373. $content .= "\n\n" .$attach_content;
  374. }*/
  375. //return $content;
  376. } else {
  377. $sign_stat = self::sign($user->uid, $day);
  378. $content = '今日签到成功,赠送' . $sign_stat . '书币,明日继续签到得书币哦~' . "\n\n" . '<a href=' . '"' . $continueReadUrl . '"' . '> >>点我继续上次阅读</a>' . "\n\n" . '阅读记录:';
  379. if ($sign_stat == 30) {
  380. $content = '今日签到成功,赠送30书币,连续签到2日后,赠送书币增加至50哦~' . "\n\n" . '<a href=' . '"' . $continueReadUrl . '"' . '> >>点我继续上次阅读</a>' . "\n\n" . '阅读记录:';
  381. }
  382. }
  383. $new_user_activity_content = '';
  384. if (!Order::where('uid', $user->uid)->where('status', 'PAID')->select('id')->first()) {
  385. $new_user_activity_content = self::newUserActivity($user);
  386. }
  387. $res = ReadRecordService::getReadRecord($user->uid);
  388. foreach ($res as $key => $record) {
  389. if ($key == 3) break;
  390. $url = env('PROTOCOL') . '://site' . $encode_distribution_channel_id . '.' . env('CUSTOM_HOST') . '.com/reader?bid=' . Hashids::encode($record['bid']) . '&cid=' . $record['cid'];
  391. $content .= "\n\n" . '<a href="' . $url . '"> >>《' . $record['book_name'] . '》</a>';
  392. }
  393. $read_bid_arr = [];
  394. $read_bid_arr[] = -1;
  395. foreach ($res as $vbook) {
  396. $read_bid_arr[] = $vbook['bid'];
  397. }
  398. $user_detail = UserService::getById($user->uid);
  399. $sign_recomm_bid_key = '男频';
  400. if ($user_detail && isset($user_detail->sex)) {
  401. if ($user_detail->sex == 2) {
  402. $sign_recomm_bid_key = '女频';
  403. }
  404. }
  405. $recomm_books = BookConfigService::getSignRecommendBooks($read_bid_arr, $sign_recomm_bid_key);
  406. $content .= "\n\n" . '热门书籍推荐';
  407. if ($recomm_books) {
  408. foreach ($recomm_books as $book) {
  409. $url = env('PROTOCOL') . '://site' . $encode_distribution_channel_id . '.' . env('CUSTOM_HOST') . '.com/reader?bid=' . Hashids::encode($book->bid) . '&cid=' . $book->first_cid;
  410. $content .= "\n\n" . '<a href="' . $url . '"> >>《' . $book->book_name . '》</a>';
  411. }
  412. }
  413. //$content .= "\n\n" . '为方便下次阅读,请置顶公众号';
  414. $content .= "\n\n" . '为方便下次阅读,请<a href="' . 'https://help.leyuee.com/top.html"' . '>置顶公众号</a>';
  415. if ($attach_content) {
  416. $content .= "\n\n" . $attach_content;
  417. }
  418. if ($new_user_activity_content) {
  419. $content .= "\n\n" . $new_user_activity_content;
  420. }
  421. }
  422. return $content;
  423. }
  424. public static function signCallBackPushActivityInfo($uid, $distribution_channel_id)
  425. {
  426. $activity_setting = ActivityService::getActivitySetting();
  427. if (!$activity_setting)
  428. return false;
  429. $acyivity_id = isset($activity_setting['activity_id']) ? $activity_setting['activity_id'] : 0;
  430. $other = env('OTHER_ACTIVITY_ID', 0);
  431. if ($acyivity_id == $other && !in_array($distribution_channel_id, explode(',', env('OTHER_ACTIVITY_CHANNEL', '1')))) {
  432. return false;
  433. }
  434. if (!$acyivity_id)
  435. return false;
  436. $activity_title = isset($activity_setting['sign_call_back_text']) ? $activity_setting['sign_call_back_text'] : '';;
  437. if (empty($activity_title))
  438. return false;
  439. $activity_info = ActivityService::getById($acyivity_id);
  440. if (empty($activity_info))
  441. return false;
  442. if (time() < strtotime($activity_info->start_time) || time() > strtotime($activity_info->end_time)) {
  443. return false;
  444. }
  445. $user = UserService::getById($uid);
  446. if (empty($user))
  447. return false;
  448. if (!ActivitySwitchService::isShowInPage($acyivity_id, $distribution_channel_id, 'sign')) {
  449. return false;
  450. }
  451. /*
  452. $no_participate_activity = env('no_participate_activity','');
  453. if($no_participate_activity && in_array($user->distribution_channel_id, explode(',',$no_participate_activity))){
  454. return false;
  455. }*/
  456. if ($user && isset($user->created_at) && (time() - strtotime($user->created_at)) >= 86400 * 2) {
  457. $url = env('PROTOCOL') . '://site' . encodeDistributionChannelId($distribution_channel_id) . '.' . env('CUSTOM_HOST') . '.com' . $activity_info->activity_page . '&fromtype=signcallback';
  458. return '<a href="' . $url . '"> ' . $activity_title . '</a>';
  459. }
  460. return false;
  461. }
  462. /**
  463. * 签到统计
  464. */
  465. public static function getSignTotalByChnnelId(array $data = [])
  466. {
  467. $search_obj = UserSign::join('users', 'user_sign.uid', '=', 'users.id');
  468. if (isset($params['distribution_channel_id'])) $search_obj->where('users.distribution_channel_id', $params['distribution_channel_id']);
  469. if (isset($params['begin_time']) && $params['begin_time']) $search_obj->where('user_sign.created_at', '>=', $params['begin_time']);
  470. if (isset($params['end_time']) && $params['end_time']) $search_obj->where('user_sign.created_at', '<=', $params['end_time']);
  471. return $search_obj->groupBy('user_sign.day', 'users.distribution_channel_id')->select(DB::raw('count(*) as sign_num'), DB::raw('user_sign.day as day'), 'users.distribution_channel_id')->get();
  472. }
  473. }