BookController.php 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564
  1. <?php
  2. namespace App\Http\Controllers\QuickApp\Book;
  3. use App\Libs\Utils;
  4. use App\Modules\Book\Services\BookAuditService;
  5. use App\Modules\RecommendBook\Services\QappRecommendService;
  6. use App\Modules\RecommendBook\Services\RecommendService;
  7. use App\Modules\Book\Services\RecoBannerService;
  8. use App\Modules\User\Models\QappPackage;
  9. use App\Modules\User\Services\QappUserService;
  10. use Hashids\Hashids;
  11. use Illuminate\Http\Request;
  12. use App\Http\Controllers\QuickApp\BaseController;
  13. use App\Http\Controllers\QuickApp\Book\Transformers\BookTransformer;
  14. use App\Http\Controllers\QuickApp\Book\Transformers\KeywordTransformer;
  15. use App\Modules\Book\Models\BookConfig;
  16. use App\Modules\Book\Services\BookConfigService;
  17. use App\Modules\Book\Services\BookService;
  18. use App\Modules\Book\Services\BookUrgeUpdateService;
  19. use App\Modules\Book\Services\UserShelfBooksService;
  20. use App\Modules\Book\Services\ChapterService;
  21. use App\Modules\Subscribe\Services\BookOrderService;
  22. use App\Modules\Subscribe\Services\ChapterOrderService;
  23. use App\Modules\Subscribe\Services\YearOrderService;
  24. use App\Modules\User\Services\ReadRecordService;
  25. class BookController extends BaseController
  26. {
  27. public function index(Request $request, $bid)
  28. {
  29. $bid = BookService::decodeBidStatic($bid);
  30. $book_info = BookConfigService::getBookById($bid);
  31. if (!$book_info) {
  32. return response()->error('QAPP_SYS_ERROR');
  33. }
  34. //yuyuedu、xinghe 快应用这两个cp的书屏蔽下
  35. if(in_array($book_info->cp_source,getHiddenCp())){
  36. return response()->error('QAPP_SYS_ERROR');
  37. }
  38. if($this->distribution_channel_id == 7477 && $bid == 13765){
  39. $book_info->is_on_shelf = 2;
  40. }
  41. if($bid == 58886){
  42. $book_info->is_on_shelf = 0;
  43. }
  44. if (!in_array($book_info->is_on_shelf, [2])) {
  45. return response()->error('QAPP_OFF_SHELF');
  46. }
  47. $is_on_shelf = UserShelfBooksService::getUserShelfBooksListByUidAndBid($this->uid, $bid);
  48. $book_info['is_on_user_shelf'] = 0;
  49. if ($is_on_shelf) {
  50. $book_info['is_on_user_shelf'] = 1;
  51. }
  52. $last_chapter = ChapterService::getChapterNameById($book_info['last_cid'], $bid);
  53. $book_info->last_chapter = $last_chapter['name'];
  54. list($is_split,$is_change_chapter_name) = BookService::splitContent($bid);
  55. if($is_split && ($book_info->channel_name == '男频' || $is_change_chapter_name) ){
  56. $book_info->last_chapter = '第'.$book_info->chapter_count.'章';
  57. }
  58. $book_info['last_chapter_is_vip'] = $last_chapter['is_vip'];
  59. $book_info['is_need_charge'] = $this->isNeedCharge($bid, $last_chapter, $book_info);
  60. $record = ReadRecordService::getBookReadRecordStatic($this->uid, $bid);
  61. if ($record) {
  62. $book_info['record_chapter_id'] = $record['record_chapter_id'];
  63. $book_info['record_chapter_name'] = $record['record_chapter_name'];
  64. }
  65. return response()->item(new BookTransformer(), $book_info);
  66. }
  67. /**
  68. * 获取订购记录
  69. * @param $book_info
  70. * @param $chapter_id
  71. * @return bool
  72. */
  73. protected function getOrderRecord($bid, $chapter_id)
  74. {
  75. //包年记录
  76. $uid = $this->uid;
  77. $res = YearOrderService::getRecord($uid);
  78. if ($res) return true;
  79. $res = null;
  80. //单本订购记录
  81. $res = BookOrderService::getRecordByuidBid($uid, $bid);
  82. if ($res) return true;
  83. $res = null;
  84. //章节订购记录
  85. $chapterOrder = new ChapterOrderService();
  86. if ($chapterOrder->checkIsOrdered($uid, $bid, $chapter_id)) return true;
  87. return false;
  88. }
  89. /**
  90. * 判断是否需要充值
  91. */
  92. private function isBookNeedCharge(int $bid, float $price)
  93. {
  94. $book_order = $this->getOrderRecord($bid, 0);
  95. if ($book_order) {
  96. return false;
  97. } else {
  98. $user_info = $this->user_info;
  99. return $user_info['balance'] < $price;
  100. }
  101. }
  102. /**
  103. * 判断章节是否需要充值
  104. */
  105. private function isChapterNeedCharge(int $bid, int $cid, float $price)
  106. {
  107. $book_order = $this->getOrderRecord($bid, $cid);
  108. if ($book_order) {
  109. return false;
  110. } else {
  111. $user_info = $this->user_info;
  112. return $user_info['balance'] < $price;
  113. }
  114. }
  115. /**
  116. * 判断是否需要充值
  117. */
  118. private function isNeedCharge(int $bid, $last_chapter, $book_info)
  119. {
  120. $is_free = BookConfigService::judgeBookIsFree($bid);
  121. if ($is_free) {
  122. return false;
  123. }
  124. switch ($book_info->charge_type) {
  125. case 'BOOK':
  126. $price = $this->getPrice($book_info);
  127. return $this->isBookNeedCharge($bid, $price);
  128. default:
  129. $price = $last_chapter->is_vip ? $this->getPrice($book_info, $last_chapter->size) : 0;
  130. return $last_chapter->is_vip ? $this->isChapterNeedCharge($bid, $last_chapter->id, $price) : false;
  131. }
  132. }
  133. /**
  134. * 计算价格
  135. * @param $book_info
  136. * @param $chapter_size
  137. * @return float
  138. */
  139. protected function getPrice($book_info, $chapter_size = 0)
  140. {
  141. if ($book_info->charge_type == 'BOOK')
  142. return $book_info->price * 100;
  143. return ceil($chapter_size / 100);
  144. }
  145. /**
  146. * 首页
  147. */
  148. public function getBookLists(Request $request, $sex)
  149. {
  150. // 获取基本数据
  151. $package = $request->header('x-package', '');
  152. $brand = $request->header('x-nbrand', '');
  153. $codeVersion = $request->header('x-codeversion', '');
  154. $isAuth = check_qapp_auth($package ,0);
  155. // 根据包名、平台、版本号判断是否审核
  156. if (Utils::checkIsAudit($package, $brand, $codeVersion) || $isAuth == false) {
  157. $result = BookAuditService::getHomeBooksData($sex, $package,$isAuth);
  158. return response()->success($result);
  159. }
  160. if ($sex == 'male') {
  161. $channel = 1;
  162. $reco_banner_type = ['MALE', 'PUBLIC'];
  163. } else {
  164. $reco_banner_type = ['FEMALE', 'PUBLIC'];
  165. $channel = 2;
  166. }
  167. $books = (new RecoBannerService)->getByType($reco_banner_type, 2);
  168. $books->transform(function ($item) {
  169. $result = $this->getBidCidFromUrl($item->redirect_url);
  170. $item->bid = $result['bid'];
  171. $item->cid = $result['cid'];
  172. if ($result['cid']) {
  173. $item->redirect_url = "views/Reader";
  174. } else {
  175. $item->redirect_url = "views/Detail";
  176. }
  177. return $item;
  178. });
  179. //新判断: 根据包名来获取对应所需的bid
  180. $qapp_package = QappPackage::getPackageByPackage($package);
  181. if($qapp_package){
  182. $package_id = $qapp_package->id;
  183. }else{
  184. $package_id = 0;
  185. }
  186. if($isAuth == false && !$books->isEmpty()){
  187. // 未授权快应
  188. $bookIds = array_column($books->toArray(),'bid');
  189. $bookIds = BookConfigService::checkBookId($bookIds);
  190. foreach ($books as &$value){
  191. if (!in_array($value->bid,$bookIds)){
  192. $value->redirect_url = "#";
  193. }
  194. }
  195. unset($value);
  196. }
  197. \Log::info('un_send_order_book:package_id1:'.$package_id.' $package:'.$package);
  198. $user = (new QappUserService)->getGolableUser();
  199. if(isset($user->uid) && !empty($user->uid)){
  200. if(!$this->send_order_id || $this->send_order_id == 0 ){
  201. \Log::info('un_send_order_book:uid:'.$this->uid.' $package:'.$package);
  202. $result = $this->getCheckBids($channel,$books,$package_id);
  203. if(isset($result[1]['books']) && count((array)$result[1]['books']) > 1){
  204. return response()->success($result);
  205. }
  206. $result = $this->getCheckBids($channel,$books,0);
  207. return response()->success($result);
  208. }else{
  209. $package_id = 0;
  210. }
  211. \Log::info('un_send_order_book:package_id2:'.$package_id.' $package:'.$package);
  212. }else{
  213. $package_id = 0;
  214. }
  215. \Log::info('un_send_order_book:package_id3:'.$package_id.' $package:'.$package);
  216. $result = $this->getCheckBids($channel,$books,$package_id);
  217. return response()->success($result);
  218. }
  219. /**
  220. * 根据包名
  221. * @param $channel
  222. * @param $books
  223. * @param $package_id
  224. * @return array
  225. */
  226. private function getCheckBids($channel,$books,$package_id)
  227. {
  228. $isAuthor = check_qapp_auth($package_id,1);
  229. $hotBids = $this->getCheckBooks(QappRecommendService::getRecommendByPacketId($channel, 'hot',$package_id),$channel,$isAuthor);
  230. $liveBids = $this->getCheckBooks(QappRecommendService::getRecommendByPacketId($channel, 'live',$package_id),$channel,$isAuthor);
  231. $recomBids = $this->getCheckBooks(QappRecommendService::getRecommendByPacketId($channel, 'recom',$package_id),$channel,$isAuthor);
  232. $newBids = $this->getCheckBooks(QappRecommendService::getRecommendByPacketId($channel, 'new_recom',$package_id),$channel,$isAuthor);
  233. return array_filter([
  234. ['type' => 'reco_banner', 'lable' => '首页banner', 'books' => $books],
  235. ['type' => 'hot', 'lable' => '热门书单', 'books' => collectionTransform(new BookTransformer, BookConfigService::getBooksByIds($hotBids))],
  236. ['type' => 'zhibo', 'lable' => '神书直播', 'books' => collectionTransform(new BookTransformer, BookConfigService::getBooksByIds($liveBids))],
  237. ['type' => 'recom', 'lable' => '小编精选', 'books' => collectionTransform(new BookTransformer, BookConfigService::getBooksByIds($recomBids))],
  238. ['type' => 'new_recom', 'lable' => '人气新书', 'books' => collectionTransform(new BookTransformer, BookConfigService::getBooksByIds($newBids))]
  239. ]);
  240. }
  241. /**
  242. * 检测并补充不满足条件的书籍id
  243. * @param $bid_list
  244. * @param $channel : 频道
  245. * @return array
  246. */
  247. private function getCheckBooks($bid_list,$channel,$is_author)
  248. {
  249. //获取书本数量
  250. $count = count($bid_list);
  251. if (!$is_author){
  252. $where = [
  253. ['book_configs.charge_type','!=','BOOK'],
  254. ['book_configs.cp_source','=','ycsd'],
  255. ];
  256. }else{
  257. $where = [
  258. ['book_configs.charge_type','!=','BOOK'],
  259. ];
  260. }
  261. //获取当前有效书本数量
  262. $book_count = BookConfig::join('books', 'book_configs.bid', '=', 'books.id')
  263. ->leftjoin('book_categories', 'books.category_id', 'book_categories.id')
  264. ->whereIn('book_configs.bid',$bid_list)
  265. ->where('book_configs.is_on_shelf',2)
  266. ->where('book_configs.charge_type','!=','BOOK')
  267. ->whereNotIn('book_configs.cp_source',getHiddenCp())
  268. ->where($where)
  269. ->where('book_categories.pid',$channel)
  270. ->count();
  271. if($count == $book_count){
  272. return $bid_list;
  273. }
  274. //获取需要补充的书籍数量
  275. $supplement_count = (($count - $book_count) > 0) ? $count - $book_count : 0;
  276. if($supplement_count <= 0){
  277. return $bid_list;
  278. }
  279. //获取书籍交集bid,过滤掉不符合要求的书
  280. $bids = BookConfig::join('books', 'book_configs.bid', '=', 'books.id')
  281. ->leftjoin('book_categories', 'books.category_id', 'book_categories.id')
  282. ->whereIn('book_configs.bid',$bid_list)
  283. ->where('book_configs.is_on_shelf',2)
  284. ->where($where)
  285. ->whereNotIn('book_configs.cp_source',getHiddenCp())
  286. ->where('book_categories.pid',$channel)
  287. ->pluck('book_configs.bid')->all();
  288. $bid_list = array_intersect($bid_list,$bids);
  289. //获取随机的有效的书籍bid
  290. $rand_bid = BookConfig::join('books', 'book_configs.bid', '=', 'books.id')
  291. ->leftjoin('book_categories', 'books.category_id', 'book_categories.id')
  292. ->where('book_configs.is_on_shelf',2)
  293. // ->where('book_configs.charge_type','!=','BOOK')
  294. ->where($where)
  295. ->whereNotIn('book_configs.cp_source',getHiddenCp())
  296. ->where('book_categories.pid',$channel)
  297. ->inRandomOrder()
  298. ->limit($supplement_count)
  299. ->get()->pluck('bid')->toArray();
  300. return array_filter(array_merge($bid_list,$rand_bid));
  301. }
  302. private function getBidCidFromUrl(string $url)
  303. {
  304. if (preg_match('/\?bid=(\w+)\S+cid=(\w+)/', $url, $matches) || preg_match('/\?id=(\w+)/', $url, $matches)) {
  305. return [
  306. 'bid' => $matches[1],
  307. 'cid' => isset($matches[2]) ? $matches[2] : 0,
  308. ];
  309. } else {
  310. return [
  311. 'bid' => '',
  312. 'cid' => 0,
  313. ];
  314. }
  315. }
  316. public function library(Request $request)
  317. {
  318. $where = [];
  319. $order = [];
  320. $where['is_on_shelf'] = [2];
  321. $category_id = $request->input('category_id');
  322. if ($category_id) {
  323. if ($category_id == 1) {
  324. $where['channel_name'] = '男频';
  325. } elseif ($category_id == 2) {
  326. $where['channel_name'] = '女频';
  327. } else {
  328. $where['category_id'] = $category_id;
  329. }
  330. }
  331. $key = $request->input('key');
  332. $uid = $request->input('uid', 0);
  333. if ($key && $uid && is_numeric($uid)) {
  334. BookConfigService::saveUserSearchLog($key, $uid);
  335. }
  336. $where['key'] = $key;
  337. $order_field = $request->input('order_field');
  338. $order_seq = $request->input('order_seq');
  339. if ($order_field != '' && in_array($order_field, ['recommend_index', 'click_count', 'update', 'size', 'create'])) {
  340. if ($order_field == 'update') {
  341. $order = ['book_configs.updated_at', 'desc'];
  342. } elseif ($order_field == 'create') {
  343. $order = ['book_configs.created_at', 'desc'];
  344. } else {
  345. $order = [$order_field, 'desc'];
  346. }
  347. if ($order_seq == 'asc') {
  348. $order = [$order_field, 'asc'];
  349. }
  350. if ($order_seq == 'desc') {
  351. $order = [$order_field, 'desc'];
  352. }
  353. }
  354. // 审核状态默认值
  355. $package = $request->header('x-package', '');
  356. $brand = $request->header('x-nbrand', '');
  357. $codeVersion = $request->header('x-codeversion', '');
  358. if ($order_field === 'recommend_index' && Utils::checkIsAudit($package, $brand, $codeVersion)) {
  359. $order = ['book_configs.bid', 'desc'];
  360. }
  361. // 是否只使用原创书殿的书
  362. $isAuth = check_qapp_auth($package ,0);
  363. if (!$isAuth){
  364. $where['cp_source'] = "ycsd";
  365. }
  366. $status = $request->input('status');
  367. if ($status != '') {
  368. $where['status'] = $status;
  369. }
  370. // 搜索关键词的情况下,屏蔽书籍完本状态
  371. if ($key && isset($where['status'])) {
  372. unset($where['status']);
  373. }
  374. $page_size = $request->input('page_size', 15);
  375. $books = BookConfigService::getBooks($where, $order, $page_size);
  376. return response()->pagination(new BookTransformer, $books);
  377. }
  378. public function hotWords(Request $request)
  379. {
  380. $result = BookConfigService::findBookKeywords();
  381. return response()->pagination(new KeywordTransformer, $result);
  382. }
  383. public function similarRecom(Request $request)
  384. {
  385. $package = $request->header('x-package', '');
  386. $category_id = $request->input('category_id');
  387. $bid = $request->input('bid');
  388. if (empty($bid) || (empty($category_id) && $category_id != 0)) {
  389. return response()->error('PARAM_ERROR');
  390. }
  391. $isAuth = check_qapp_auth($package ,0);
  392. $bid = BookService::decodeBidStatic($bid);
  393. $where = ['category_id' => $category_id, 'is_on_shelf' => [2]];
  394. if (!$isAuth){
  395. $where['cp_source'] = "ycsd";
  396. }
  397. $books = BookConfigService::getBooks($where, [], 4);
  398. $data = [];
  399. foreach ($books as $v) {
  400. if ($v->bid != $bid && count($data) < 3) {
  401. $data[] = $v;
  402. }
  403. }
  404. return response()->collection(new BookTransformer(), $data);
  405. }
  406. public function readOverRecommend(Request $request)
  407. {
  408. $bid = $request->input('bid');
  409. if (empty($bid)) {
  410. return response()->error('PARAM_ERROR');
  411. }
  412. $bid = BookService::decodeBidStatic($bid);
  413. $book_info = BookConfigService::getBookById($bid);
  414. $res = BookConfigService::getRecommendBooks($bid, $book_info->channel_name);
  415. $urge_status = 0;
  416. if ($book_info->status == 0 && !BookUrgeUpdateService::isHadUrged($this->uid, $bid)) {
  417. $urge_status = 1;
  418. }
  419. $recommend_result = collectionTransform(new BookTransformer(), $res);
  420. $book_status = [
  421. 'status' => $book_info->status,
  422. 'urge_status' => $urge_status
  423. ];
  424. $data = [
  425. 'recommend_result' => $recommend_result,
  426. 'book_status' => $book_status
  427. ];
  428. return response()->success($data);
  429. }
  430. public function rank(Request $request)
  431. {
  432. // 1:男频,2:女频
  433. $sex = (int)$request->input('sex');
  434. if (!in_array($sex, [1, 2], true)) {
  435. return response()->error('PARAM_ERROR');
  436. }
  437. // 默认
  438. $bids = [11529, 11941, 12720, 11990, 11988, 11976, 11977, 4183, 12717, 11833,
  439. 7287,14297,12716,14312,14000,13577,16712,13002,12717,15103,13928,14793,
  440. 12708,13286];
  441. if ($sex === 2) {
  442. $bids = [8469, 11660, 9117, 7891, 12281, 12470, 8167, 11661, 11670, 8476, 8557, 11662,
  443. 11680, 11926, 12462, 7836, 11681, 11664, 11928, 8631];
  444. }
  445. /**
  446. * pid:1为男频 2为女频
  447. SELECT
  448. CONCAT( books.id, ',' )
  449. FROM
  450. books
  451. LEFT JOIN book_configs ON books.id = book_configs.bid
  452. WHERE
  453. book_configs.is_on_shelf = 2
  454. AND books.category_id IN ( SELECT id FROM book_categories WHERE pid = 2 )
  455. ORDER BY
  456. book_configs.recommend_index DESC
  457. LIMIT 10;
  458. */
  459. // 根据包名、平台、版本号判断是否审核
  460. $package = $request->header('x-package', '');
  461. $brand = $request->header('x-nbrand', '');
  462. $codeVersion = $request->header('x-codeversion', '');
  463. if (Utils::checkIsAudit($package, $brand, $codeVersion)) {
  464. $bids = [2266, 3838, 9700, 10175, 10301, 3422, 1166, 4546, 9163, 2509,
  465. 7287,14297,12716,14312,14000,13577,16712,13002,12717,15103,13928,
  466. 14793,12708,13286,13336,13275,13073,15121,13929,12693,13254,3526,
  467. 10313,3483,13278,14004,4098,10378,14072,21376,21139,21757,19449];
  468. if ($sex === 2) {
  469. $bids = [159, 2439, 6276, 10074, 5409, 9379, 10323, 9078, 3603, 487];
  470. }
  471. }
  472. $isAuth = check_qapp_auth($package,0);
  473. if (!$isAuth){
  474. $rank = config('home.rank');
  475. $bids = $rank['male'];
  476. if ($sex === 2) {
  477. $bids = $rank['female'];;
  478. }
  479. }
  480. $books = collectionTransform(new BookTransformer, BookConfigService::getBooksByIds($bids));
  481. return response()->success($books);
  482. }
  483. /**
  484. * 推荐书
  485. */
  486. public function recommen()
  487. {
  488. $reco_banner_type = ['FEMALE', 'PUBLIC'];
  489. $books = (new RecoBannerService)->getByType($reco_banner_type, 2);
  490. $books->transform(function ($item) {
  491. $result = $this->getBidCidFromUrl($item->redirect_url);
  492. $item->bid = $result['bid'];
  493. $item->cid = $result['cid'];
  494. if ($result['cid']) {
  495. $item->redirect_url = "views/Reader";
  496. } else {
  497. $item->redirect_url = "views/Detail";
  498. }
  499. return $item;
  500. });
  501. return response()->success($books);
  502. }
  503. /**
  504. * 限免
  505. */
  506. public function free(int $sex)
  507. {
  508. $result = BookConfigService::findFreeBooks($sex);
  509. return response()->success($result);
  510. }
  511. }