BookController.php 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560
  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. if($isAuth){
  168. $books = (new RecoBannerService)->getByType($reco_banner_type, 2);
  169. $books->transform(function ($item) {
  170. $result = $this->getBidCidFromUrl($item->redirect_url);
  171. $item->bid = $result['bid'];
  172. $item->cid = $result['cid'];
  173. if ($result['cid']) {
  174. $item->redirect_url = "views/Reader";
  175. } else {
  176. $item->redirect_url = "views/Detail";
  177. }
  178. return $item;
  179. });
  180. }else{
  181. $home = config('home.ycsd');
  182. $banner = $home['reco_banner'];
  183. $books = $banner[$sex];
  184. }
  185. //新判断: 根据包名来获取对应所需的bid
  186. $qapp_package = QappPackage::getPackageByPackage($package);
  187. if($qapp_package){
  188. $package_id = $qapp_package->id;
  189. }else{
  190. $package_id = 0;
  191. }
  192. \Log::info('un_send_order_book:package_id1:'.$package_id.' $package:'.$package);
  193. $user = (new QappUserService)->getGolableUser();
  194. if(isset($user->uid) && !empty($user->uid)){
  195. if(!$this->send_order_id || $this->send_order_id == 0 ){
  196. \Log::info('un_send_order_book:uid:'.$this->uid.' $package:'.$package);
  197. $result = $this->getCheckBids($channel,$books,$package_id);
  198. if(isset($result[1]['books']) && count((array)$result[1]['books']) > 1){
  199. return response()->success($result);
  200. }
  201. $result = $this->getCheckBids($channel,$books,0);
  202. return response()->success($result);
  203. }else{
  204. $package_id = 0;
  205. }
  206. \Log::info('un_send_order_book:package_id2:'.$package_id.' $package:'.$package);
  207. }else{
  208. $package_id = 0;
  209. }
  210. \Log::info('un_send_order_book:package_id3:'.$package_id.' $package:'.$package);
  211. $result = $this->getCheckBids($channel,$books,$package_id);
  212. return response()->success($result);
  213. }
  214. /**
  215. * 根据包名
  216. * @param $channel
  217. * @param $books
  218. * @param $package_id
  219. * @return array
  220. */
  221. private function getCheckBids($channel,$books,$package_id)
  222. {
  223. $isAuthor = check_qapp_auth($package_id,1);
  224. $hotBids = $this->getCheckBooks(QappRecommendService::getRecommendByPacketId($channel, 'hot',$package_id),$channel,$isAuthor);
  225. $liveBids = $this->getCheckBooks(QappRecommendService::getRecommendByPacketId($channel, 'live',$package_id),$channel,$isAuthor);
  226. $recomBids = $this->getCheckBooks(QappRecommendService::getRecommendByPacketId($channel, 'recom',$package_id),$channel,$isAuthor);
  227. $newBids = $this->getCheckBooks(QappRecommendService::getRecommendByPacketId($channel, 'new_recom',$package_id),$channel,$isAuthor);
  228. return array_filter([
  229. ['type' => 'reco_banner', 'lable' => '首页banner', 'books' => $books],
  230. ['type' => 'hot', 'lable' => '热门书单', 'books' => collectionTransform(new BookTransformer, BookConfigService::getBooksByIds($hotBids))],
  231. ['type' => 'zhibo', 'lable' => '神书直播', 'books' => collectionTransform(new BookTransformer, BookConfigService::getBooksByIds($liveBids))],
  232. ['type' => 'recom', 'lable' => '小编精选', 'books' => collectionTransform(new BookTransformer, BookConfigService::getBooksByIds($recomBids))],
  233. ['type' => 'new_recom', 'lable' => '人气新书', 'books' => collectionTransform(new BookTransformer, BookConfigService::getBooksByIds($newBids))]
  234. ]);
  235. }
  236. /**
  237. * 检测并补充不满足条件的书籍id
  238. * @param $bid_list
  239. * @param $channel : 频道
  240. * @return array
  241. */
  242. private function getCheckBooks($bid_list,$channel,$is_author)
  243. {
  244. //获取书本数量
  245. $count = count($bid_list);
  246. if (!$is_author){
  247. $where = [
  248. ['book_configs.charge_type','!=','BOOK'],
  249. ['book_configs.cp_source','=','ycsd'],
  250. ];
  251. }else{
  252. $where = [
  253. ['book_configs.charge_type','!=','BOOK'],
  254. ];
  255. }
  256. //获取当前有效书本数量
  257. $book_count = BookConfig::join('books', 'book_configs.bid', '=', 'books.id')
  258. ->leftjoin('book_categories', 'books.category_id', 'book_categories.id')
  259. ->whereIn('book_configs.bid',$bid_list)
  260. ->where('book_configs.is_on_shelf',2)
  261. ->where('book_configs.charge_type','!=','BOOK')
  262. ->whereNotIn('book_configs.cp_source',getHiddenCp())
  263. ->where($where)
  264. ->where('book_categories.pid',$channel)
  265. ->count();
  266. if($count == $book_count){
  267. return $bid_list;
  268. }
  269. //获取需要补充的书籍数量
  270. $supplement_count = (($count - $book_count) > 0) ? $count - $book_count : 0;
  271. if($supplement_count <= 0){
  272. return $bid_list;
  273. }
  274. //获取书籍交集bid,过滤掉不符合要求的书
  275. $bids = BookConfig::join('books', 'book_configs.bid', '=', 'books.id')
  276. ->leftjoin('book_categories', 'books.category_id', 'book_categories.id')
  277. ->whereIn('book_configs.bid',$bid_list)
  278. ->where('book_configs.is_on_shelf',2)
  279. ->where($where)
  280. ->whereNotIn('book_configs.cp_source',getHiddenCp())
  281. ->where('book_categories.pid',$channel)
  282. ->pluck('book_configs.bid')->all();
  283. $bid_list = array_intersect($bid_list,$bids);
  284. //获取随机的有效的书籍bid
  285. $rand_bid = BookConfig::join('books', 'book_configs.bid', '=', 'books.id')
  286. ->leftjoin('book_categories', 'books.category_id', 'book_categories.id')
  287. ->where('book_configs.is_on_shelf',2)
  288. // ->where('book_configs.charge_type','!=','BOOK')
  289. ->where($where)
  290. ->whereNotIn('book_configs.cp_source',getHiddenCp())
  291. ->where('book_categories.pid',$channel)
  292. ->inRandomOrder()
  293. ->limit($supplement_count)
  294. ->get()->pluck('bid')->toArray();
  295. return array_filter(array_merge($bid_list,$rand_bid));
  296. }
  297. private function getBidCidFromUrl(string $url)
  298. {
  299. if (preg_match('/\?bid=(\w+)\S+cid=(\w+)/', $url, $matches) || preg_match('/\?id=(\w+)/', $url, $matches)) {
  300. return [
  301. 'bid' => $matches[1],
  302. 'cid' => isset($matches[2]) ? $matches[2] : 0,
  303. ];
  304. } else {
  305. return [
  306. 'bid' => '',
  307. 'cid' => 0,
  308. ];
  309. }
  310. }
  311. public function library(Request $request)
  312. {
  313. $where = [];
  314. $order = [];
  315. $where['is_on_shelf'] = [2];
  316. $category_id = $request->input('category_id');
  317. if ($category_id) {
  318. if ($category_id == 1) {
  319. $where['channel_name'] = '男频';
  320. } elseif ($category_id == 2) {
  321. $where['channel_name'] = '女频';
  322. } else {
  323. $where['category_id'] = $category_id;
  324. }
  325. }
  326. $key = $request->input('key');
  327. $uid = $request->input('uid', 0);
  328. if ($key && $uid && is_numeric($uid)) {
  329. BookConfigService::saveUserSearchLog($key, $uid);
  330. }
  331. $where['key'] = $key;
  332. $order_field = $request->input('order_field');
  333. $order_seq = $request->input('order_seq');
  334. if ($order_field != '' && in_array($order_field, ['recommend_index', 'click_count', 'update', 'size', 'create'])) {
  335. if ($order_field == 'update') {
  336. $order = ['book_configs.updated_at', 'desc'];
  337. } elseif ($order_field == 'create') {
  338. $order = ['book_configs.created_at', 'desc'];
  339. } else {
  340. $order = [$order_field, 'desc'];
  341. }
  342. if ($order_seq == 'asc') {
  343. $order = [$order_field, 'asc'];
  344. }
  345. if ($order_seq == 'desc') {
  346. $order = [$order_field, 'desc'];
  347. }
  348. }
  349. // 审核状态默认值
  350. $package = $request->header('x-package', '');
  351. $brand = $request->header('x-nbrand', '');
  352. $codeVersion = $request->header('x-codeversion', '');
  353. if ($order_field === 'recommend_index' && Utils::checkIsAudit($package, $brand, $codeVersion)) {
  354. $order = ['book_configs.bid', 'desc'];
  355. }
  356. // 是否只使用原创书殿的书
  357. $isAuth = check_qapp_auth($package ,0);
  358. if (!$isAuth){
  359. $where['cp_source'] = "ycsd";
  360. }
  361. $status = $request->input('status');
  362. if ($status != '') {
  363. $where['status'] = $status;
  364. }
  365. // 搜索关键词的情况下,屏蔽书籍完本状态
  366. if ($key && isset($where['status'])) {
  367. unset($where['status']);
  368. }
  369. $page_size = $request->input('page_size', 15);
  370. $books = BookConfigService::getBooks($where, $order, $page_size);
  371. return response()->pagination(new BookTransformer, $books);
  372. }
  373. public function hotWords(Request $request)
  374. {
  375. $result = BookConfigService::findBookKeywords();
  376. return response()->pagination(new KeywordTransformer, $result);
  377. }
  378. public function similarRecom(Request $request)
  379. {
  380. $package = $request->header('x-package', '');
  381. $category_id = $request->input('category_id');
  382. $bid = $request->input('bid');
  383. if (empty($bid) || (empty($category_id) && $category_id != 0)) {
  384. return response()->error('PARAM_ERROR');
  385. }
  386. $isAuth = check_qapp_auth($package ,0);
  387. $bid = BookService::decodeBidStatic($bid);
  388. $where = ['category_id' => $category_id, 'is_on_shelf' => [2]];
  389. if (!$isAuth){
  390. $where['cp_source'] = "ycsd";
  391. }
  392. $books = BookConfigService::getBooks($where, [], 4);
  393. $data = [];
  394. foreach ($books as $v) {
  395. if ($v->bid != $bid && count($data) < 3) {
  396. $data[] = $v;
  397. }
  398. }
  399. return response()->collection(new BookTransformer(), $data);
  400. }
  401. public function readOverRecommend(Request $request)
  402. {
  403. $bid = $request->input('bid');
  404. if (empty($bid)) {
  405. return response()->error('PARAM_ERROR');
  406. }
  407. $bid = BookService::decodeBidStatic($bid);
  408. $book_info = BookConfigService::getBookById($bid);
  409. $res = BookConfigService::getRecommendBooks($bid, $book_info->channel_name);
  410. $urge_status = 0;
  411. if ($book_info->status == 0 && !BookUrgeUpdateService::isHadUrged($this->uid, $bid)) {
  412. $urge_status = 1;
  413. }
  414. $recommend_result = collectionTransform(new BookTransformer(), $res);
  415. $book_status = [
  416. 'status' => $book_info->status,
  417. 'urge_status' => $urge_status
  418. ];
  419. $data = [
  420. 'recommend_result' => $recommend_result,
  421. 'book_status' => $book_status
  422. ];
  423. return response()->success($data);
  424. }
  425. public function rank(Request $request)
  426. {
  427. // 1:男频,2:女频
  428. $sex = (int)$request->input('sex');
  429. if (!in_array($sex, [1, 2], true)) {
  430. return response()->error('PARAM_ERROR');
  431. }
  432. // 默认
  433. $bids = [11529, 11941, 12720, 11990, 11988, 11976, 11977, 4183, 12717, 11833,
  434. 7287,14297,12716,14312,14000,13577,16712,13002,12717,15103,13928,14793,
  435. 12708,13286];
  436. if ($sex === 2) {
  437. $bids = [8469, 11660, 9117, 7891, 12281, 12470, 8167, 11661, 11670, 8476, 8557, 11662,
  438. 11680, 11926, 12462, 7836, 11681, 11664, 11928, 8631];
  439. }
  440. /**
  441. * pid:1为男频 2为女频
  442. SELECT
  443. CONCAT( books.id, ',' )
  444. FROM
  445. books
  446. LEFT JOIN book_configs ON books.id = book_configs.bid
  447. WHERE
  448. book_configs.is_on_shelf = 2
  449. AND books.category_id IN ( SELECT id FROM book_categories WHERE pid = 2 )
  450. ORDER BY
  451. book_configs.recommend_index DESC
  452. LIMIT 10;
  453. */
  454. // 根据包名、平台、版本号判断是否审核
  455. $package = $request->header('x-package', '');
  456. $brand = $request->header('x-nbrand', '');
  457. $codeVersion = $request->header('x-codeversion', '');
  458. if (Utils::checkIsAudit($package, $brand, $codeVersion)) {
  459. $bids = [2266, 3838, 9700, 10175, 10301, 3422, 1166, 4546, 9163, 2509,
  460. 7287,14297,12716,14312,14000,13577,16712,13002,12717,15103,13928,
  461. 14793,12708,13286,13336,13275,13073,15121,13929,12693,13254,3526,
  462. 10313,3483,13278,14004,4098,10378,14072,21376,21139,21757,19449];
  463. if ($sex === 2) {
  464. $bids = [159, 2439, 6276, 10074, 5409, 9379, 10323, 9078, 3603, 487];
  465. }
  466. }
  467. $isAuth = check_qapp_auth($package,0);
  468. if (!$isAuth){
  469. $rank = config('home.rank');
  470. $bids = $rank['male'];
  471. if ($sex === 2) {
  472. $bids = $rank['female'];;
  473. }
  474. }
  475. $books = collectionTransform(new BookTransformer, BookConfigService::getBooksByIds($bids));
  476. return response()->success($books);
  477. }
  478. /**
  479. * 推荐书
  480. */
  481. public function recommen()
  482. {
  483. $reco_banner_type = ['FEMALE', 'PUBLIC'];
  484. $books = (new RecoBannerService)->getByType($reco_banner_type, 2);
  485. $books->transform(function ($item) {
  486. $result = $this->getBidCidFromUrl($item->redirect_url);
  487. $item->bid = $result['bid'];
  488. $item->cid = $result['cid'];
  489. if ($result['cid']) {
  490. $item->redirect_url = "views/Reader";
  491. } else {
  492. $item->redirect_url = "views/Detail";
  493. }
  494. return $item;
  495. });
  496. return response()->success($books);
  497. }
  498. /**
  499. * 限免
  500. */
  501. public function free(int $sex)
  502. {
  503. $result = BookConfigService::findFreeBooks($sex);
  504. return response()->success($result);
  505. }
  506. }