Просмотр файла app/Http/Controllers/ArticleController.php

Размер файла: 23.24Kb
  1. <?php
  2.  
  3. declare(strict_types=1);
  4.  
  5. namespace App\Http\Controllers;
  6.  
  7. use App\Classes\Validator;
  8. use App\Models\Article;
  9. use App\Models\Blog;
  10. use App\Models\Comment;
  11. use App\Models\File;
  12. use App\Models\Flood;
  13. use App\Models\Reader;
  14. use Illuminate\Database\Query\JoinClause;
  15. use Illuminate\Http\RedirectResponse;
  16. use Illuminate\Http\Request;
  17. use Illuminate\Support\Facades\Cache;
  18. use Illuminate\View\View;
  19.  
  20. class ArticleController extends Controller
  21. {
  22. /**
  23. * Главная страница
  24. *
  25. * @return View
  26. */
  27. public function index(): View
  28. {
  29. $categories = Blog::query()
  30. ->where('parent_id', 0)
  31. ->orderBy('sort')
  32. ->with('children', 'new', 'children.new')
  33. ->get();
  34.  
  35. if ($categories->isEmpty()) {
  36. abort(200, __('blogs.categories_not_created'));
  37. }
  38.  
  39. return view('blogs/index', compact('categories'));
  40. }
  41.  
  42. /**
  43. * Список статей
  44. *
  45. * @param int $id
  46. *
  47. * @return View
  48. */
  49. public function blog(int $id): View
  50. {
  51. $category = Blog::query()->with('parent')->find($id);
  52.  
  53. if (! $category) {
  54. abort(404, __('blogs.category_not_exist'));
  55. }
  56.  
  57. $articles = Article::query()
  58. ->select('articles.*', 'pollings.vote')
  59. ->leftJoin('pollings', static function (JoinClause $join) {
  60. $join->on('articles.id', 'pollings.relate_id')
  61. ->where('pollings.relate_type', Article::$morphName)
  62. ->where('pollings.user_id', getUser('id'));
  63. })
  64. ->where('category_id', $id)
  65. ->orderByDesc('created_at')
  66. ->with('user')
  67. ->paginate(setting('blogpost'));
  68.  
  69. return view('blogs/blog', compact('articles', 'category'));
  70. }
  71.  
  72. /**
  73. * Просмотр статьи
  74. *
  75. * @param int $id
  76. *
  77. * @return View
  78. */
  79. public function view(int $id): View
  80. {
  81. /** @var Article $article */
  82. $article = Article::query()
  83. ->select('articles.*', 'pollings.vote')
  84. ->where('articles.id', $id)
  85. ->leftJoin('pollings', static function (JoinClause $join) {
  86. $join->on('articles.id', 'pollings.relate_id')
  87. ->where('pollings.relate_type', Article::$morphName)
  88. ->where('pollings.user_id', getUser('id'));
  89. })
  90. ->with('category.parent')
  91. ->first();
  92.  
  93. if (! $article) {
  94. abort(404, __('blogs.article_not_exist'));
  95. }
  96.  
  97. Reader::countingStat($article);
  98.  
  99. $tagsList = preg_split('/[\s]*[,][\s]*/', $article->tags);
  100.  
  101. $tags = [];
  102. foreach ($tagsList as $value) {
  103. $tags[] = '<a href="/blogs/tags/' . check(urlencode($value)) . '">' . check($value) . '</a>';
  104. }
  105. $tags = implode(', ', $tags);
  106.  
  107. return view('blogs/view', compact('article', 'tags'));
  108. }
  109.  
  110. /**
  111. * Редактирование статьи
  112. *
  113. * @param int $id
  114. * @param Request $request
  115. * @param Validator $validator
  116. *
  117. * @return View|RedirectResponse
  118. */
  119. public function edit(int $id, Request $request, Validator $validator)
  120. {
  121. if (! $user = getUser()) {
  122. abort(403, __('main.not_authorized'));
  123. }
  124.  
  125. /** @var Article $article */
  126. $article = Article::query()->find($id);
  127.  
  128. if (! $article) {
  129. abort(404, __('blogs.article_not_exist'));
  130. }
  131.  
  132. if ($user->id !== $article->user_id) {
  133. abort(200, __('main.article_not_author'));
  134. }
  135.  
  136. if ($request->isMethod('post')) {
  137. $cid = int($request->input('cid'));
  138. $title = $request->input('title');
  139. $text = $request->input('text');
  140. $tags = $request->input('tags');
  141.  
  142. /** @var Blog $category */
  143. $category = Blog::query()->find($cid);
  144.  
  145. $validator
  146. ->equal($request->input('_token'), csrf_token(), __('validator.token'))
  147. ->length($title, 3, 50, ['title' => __('validator.text')])
  148. ->length($text, 100, setting('maxblogpost'), ['text' => __('validator.text')])
  149. ->length($tags, 2, 50, ['tags' => __('blogs.article_error_tags')])
  150. ->notEmpty($category, ['cid' => __('blogs.category_not_exist')]);
  151.  
  152. if ($category) {
  153. $validator->empty($category->closed, ['cid' => __('blogs.category_closed')]);
  154. }
  155.  
  156. if ($validator->isValid()) {
  157. // Обновление счетчиков
  158. if ($article->category_id !== $category->id) {
  159. $category->increment('count_articles');
  160. Blog::query()->where('id', $article->category_id)->decrement('count_articles');
  161. }
  162.  
  163. $article->update([
  164. 'category_id' => $category->id,
  165. 'title' => $title,
  166. 'text' => $text,
  167. 'tags' => $tags,
  168. ]);
  169.  
  170. clearCache(['statArticles', 'recentArticles']);
  171. setFlash('success', __('blogs.article_success_edited'));
  172.  
  173. return redirect('articles/' . $article->id);
  174. }
  175.  
  176. setInput($request->all());
  177. setFlash('danger', $validator->getErrors());
  178. }
  179.  
  180. $categories = Blog::query()
  181. ->where('parent_id', 0)
  182. ->with('children')
  183. ->orderBy('sort')
  184. ->get();
  185.  
  186. return view('blogs/edit', compact('article', 'categories'));
  187. }
  188.  
  189. /**
  190. * Просмотр по категориям
  191. *
  192. * @return View
  193. */
  194. public function authors(): View
  195. {
  196. $articles = Article::query()
  197. ->select('user_id', 'login')
  198. ->selectRaw('count(*) as cnt, sum(count_comments) as count_comments')
  199. ->join('users', 'articles.user_id', 'users.id')
  200. ->groupBy('user_id')
  201. ->orderByDesc('cnt')
  202. ->paginate(setting('bloggroup'));
  203.  
  204. return view('blogs/authors', compact('articles'));
  205. }
  206.  
  207. /**
  208. * Создание статьи
  209. *
  210. * @param Request $request
  211. * @param Validator $validator
  212. * @param Flood $flood
  213. *
  214. * @return View|RedirectResponse
  215. */
  216. public function create(Request $request, Validator $validator, Flood $flood)
  217. {
  218. $cid = int($request->input('cid'));
  219.  
  220. if (! isAdmin() && ! setting('blog_create')) {
  221. abort(200, __('main.articles_closed'));
  222. }
  223.  
  224. if (! $user = getUser()) {
  225. abort(403, __('main.not_authorized'));
  226. }
  227.  
  228. $categories = Blog::query()
  229. ->where('parent_id', 0)
  230. ->with('children')
  231. ->orderBy('sort')
  232. ->get();
  233.  
  234. if (! $categories) {
  235. abort(404, __('blogs.categories_not_created'));
  236. }
  237.  
  238. if ($request->isMethod('post')) {
  239. $title = $request->input('title');
  240. $text = $request->input('text');
  241. $tags = $request->input('tags');
  242.  
  243. /** @var Blog $category */
  244. $category = Blog::query()->find($cid);
  245.  
  246. $validator
  247. ->equal($request->input('_token'), csrf_token(), __('validator.token'))
  248. ->length($title, 3, 50, ['title' => __('validator.text')])
  249. ->length($text, 100, setting('maxblogpost'), ['text' => __('validator.text')])
  250. ->length($tags, 2, 50, ['tags' => __('blogs.article_error_tags')])
  251. ->false($flood->isFlood(), ['msg' => __('validator.flood', ['sec' => $flood->getPeriod()])])
  252. ->notEmpty($category, ['cid' => __('blogs.category_not_exist')]);
  253.  
  254. if ($category) {
  255. $validator->empty($category->closed, ['cid' => __('blogs.category_closed')]);
  256. }
  257.  
  258. if ($validator->isValid()) {
  259. $text = antimat($text);
  260.  
  261. /** @var Article $article */
  262. $article = Article::query()->create([
  263. 'category_id' => $cid,
  264. 'user_id' => $user->id,
  265. 'title' => $title,
  266. 'text' => $text,
  267. 'tags' => $tags,
  268. 'created_at' => SITETIME,
  269. ]);
  270.  
  271. $category->increment('count_articles');
  272.  
  273. $user->increment('point', 5);
  274. $user->increment('money', 100);
  275.  
  276. File::query()
  277. ->where('relate_type', Article::$morphName)
  278. ->where('relate_id', 0)
  279. ->where('user_id', $user->id)
  280. ->update(['relate_id' => $article->id]);
  281.  
  282. clearCache(['statArticles', 'recentArticles']);
  283. $flood->saveState();
  284.  
  285. setFlash('success', __('blogs.article_success_created'));
  286.  
  287. return redirect('articles/' . $article->id);
  288. }
  289.  
  290. setInput($request->all());
  291. setFlash('danger', $validator->getErrors());
  292. }
  293.  
  294. $files = File::query()
  295. ->where('relate_type', Article::$morphName)
  296. ->where('relate_id', 0)
  297. ->where('user_id', $user->id)
  298. ->get();
  299.  
  300. return view('blogs/create', compact('categories', 'cid', 'files'));
  301. }
  302.  
  303. /**
  304. * Комментарии
  305. *
  306. * @param int $id
  307. * @param Request $request
  308. * @param Validator $validator
  309. * @param Flood $flood
  310. *
  311. * @return View|RedirectResponse
  312. */
  313. public function comments(int $id, Request $request, Validator $validator, Flood $flood)
  314. {
  315. /** @var Article $article */
  316. $article = Article::query()->find($id);
  317.  
  318. if (! $article) {
  319. abort(404, __('blogs.article_not_exist'));
  320. }
  321.  
  322. if ($request->isMethod('post')) {
  323. $user = getUser();
  324. $msg = $request->input('msg');
  325.  
  326. $validator
  327. ->true($user, __('main.not_authorized'))
  328. ->equal($request->input('_token'), csrf_token(), __('validator.token'))
  329. ->length($msg, 5, setting('comment_length'), ['msg' => __('validator.text')])
  330. ->false($flood->isFlood(), ['msg' => __('validator.flood', ['sec' => $flood->getPeriod()])]);
  331.  
  332. if ($validator->isValid()) {
  333. /** @var Comment $comment */
  334. $comment = $article->comments()->create([
  335. 'text' => antimat($msg),
  336. 'user_id' => $user->id,
  337. 'created_at' => SITETIME,
  338. 'ip' => getIp(),
  339. 'brow' => getBrowser(),
  340. ]);
  341.  
  342. $user->increment('allcomments');
  343. $user->increment('point');
  344. $user->increment('money', 5);
  345.  
  346. $article->increment('count_comments');
  347.  
  348. $flood->saveState();
  349. sendNotify($msg, '/articles/comment/' . $article->id . '/' . $comment->id, $article->title);
  350.  
  351. setFlash('success', __('main.comment_added_success'));
  352.  
  353. return redirect('articles/end/' . $article->id);
  354. }
  355.  
  356. setInput($request->all());
  357. setFlash('danger', $validator->getErrors());
  358. }
  359.  
  360. $comments = $article->comments()
  361. ->with('user')
  362. ->orderBy('created_at')
  363. ->paginate(setting('comments_per_page'));
  364.  
  365. return view('blogs/comments', compact('article', 'comments'));
  366. }
  367.  
  368. /**
  369. * Подготовка к редактированию комментария
  370. *
  371. * @param int $id
  372. * @param int $cid
  373. * @param Request $request
  374. * @param Validator $validator
  375. *
  376. * @return View|RedirectResponse
  377. */
  378. public function editComment(int $id, int $cid, Request $request, Validator $validator)
  379. {
  380. $page = int($request->input('page', 1));
  381.  
  382. /** @var Article $article */
  383. $article = Article::query()->find($id);
  384.  
  385. if (! $article) {
  386. abort(404, __('blogs.article_not_exist'));
  387. }
  388.  
  389. if (! $user = getUser()) {
  390. abort(403, __('main.not_authorized'));
  391. }
  392.  
  393. $comment = $article->comments()
  394. ->where('id', $cid)
  395. ->where('user_id', $user->id)
  396. ->first();
  397.  
  398. if (! $comment) {
  399. abort(200, __('main.comment_deleted'));
  400. }
  401.  
  402. if ($comment->created_at + 600 < SITETIME) {
  403. abort(200, __('main.editing_impossible'));
  404. }
  405.  
  406. if ($request->isMethod('post')) {
  407. $msg = $request->input('msg');
  408. $page = int($request->input('page', 1));
  409.  
  410. $validator
  411. ->equal($request->input('_token'), csrf_token(), __('validator.token'))
  412. ->length($msg, 5, setting('comment_length'), ['msg' => __('validator.text')]);
  413.  
  414. if ($validator->isValid()) {
  415. $msg = antimat($msg);
  416.  
  417. $comment->update([
  418. 'text' => $msg,
  419. ]);
  420.  
  421. setFlash('success', __('main.comment_edited_success'));
  422.  
  423. return redirect('articles/comments/' . $article->id . '?page=' . $page);
  424. }
  425.  
  426. setInput($request->all());
  427. setFlash('danger', $validator->getErrors());
  428. }
  429.  
  430. return view('blogs/editcomment', compact('article', 'comment', 'page'));
  431. }
  432.  
  433. /**
  434. * Переадресация на последнюю страницу
  435. *
  436. * @param int $id
  437. *
  438. * @return RedirectResponse
  439. */
  440. public function end(int $id): RedirectResponse
  441. {
  442. /** @var Article $article */
  443. $article = Article::query()->find($id);
  444.  
  445. if (! $article) {
  446. abort(404, __('blogs.article_not_exist'));
  447. }
  448.  
  449. $total = $article->comments()->count();
  450.  
  451. $end = ceil($total / setting('comments_per_page'));
  452.  
  453. return redirect('articles/comments/' . $id . '?page=' . $end);
  454. }
  455.  
  456. /**
  457. * Печать
  458. *
  459. * @param int $id
  460. *
  461. * @return View
  462. */
  463. public function print(int $id): View
  464. {
  465. /** @var Article $article */
  466. $article = Article::query()->find($id);
  467.  
  468. if (! $article) {
  469. abort(404, __('blogs.article_not_exist'));
  470. }
  471.  
  472. return view('blogs/print', compact('article'));
  473. }
  474.  
  475. /**
  476. * RSS всех блогов
  477. *
  478. * @return View
  479. */
  480. public function rss(): View
  481. {
  482. $articles = Article::query()
  483. ->orderByDesc('created_at')
  484. ->with('user')
  485. ->limit(15)
  486. ->get();
  487.  
  488. if ($articles->isEmpty()) {
  489. abort(200, __('blogs.article_not_exist'));
  490. }
  491.  
  492. return view('blogs/rss', compact('articles'));
  493. }
  494.  
  495. /**
  496. * RSS комментариев к блогу
  497. *
  498. * @param int $id
  499. *
  500. * @return View
  501. */
  502. public function rssComments(int $id): View
  503. {
  504. $article = Article::query()->where('id', $id)->with('lastComments')->first();
  505.  
  506. if (! $article) {
  507. abort(404, __('blogs.article_not_exist'));
  508. }
  509.  
  510. return view('blogs/rss_comments', compact('article'));
  511. }
  512.  
  513. /**
  514. * Вывод всех тегов
  515. *
  516. * @return View
  517. */
  518. public function tags(): View
  519. {
  520. $tags = Cache::remember('tagCloud', 3600, static function () {
  521. $allTags = Article::query()
  522. ->select('tags')
  523. ->pluck('tags')
  524. ->all();
  525.  
  526. $stingTag = implode(',', $allTags);
  527. $dumptags = preg_split('/[\s]*[,][\s]*/', $stingTag, -1, PREG_SPLIT_NO_EMPTY);
  528. $allTags = array_count_values(array_map('utfLower', $dumptags));
  529.  
  530. arsort($allTags);
  531. array_splice($allTags, 100);
  532. shuffleAssoc($allTags);
  533.  
  534. return $allTags;
  535. });
  536.  
  537. $max = max($tags);
  538. $min = min($tags);
  539.  
  540. return view('blogs/tags', compact('tags', 'max', 'min'));
  541. }
  542.  
  543. /**
  544. * Поиск по тегам
  545. *
  546. * @param string $tag
  547. * @param Request $request
  548. *
  549. * @return View|RedirectResponse
  550. */
  551. public function searchTag(string $tag, Request $request)
  552. {
  553. $tag = urldecode($tag);
  554.  
  555. if (! isUtf($tag)) {
  556. $tag = winToUtf($tag);
  557. }
  558.  
  559. if (utfStrlen($tag) < 2) {
  560. setFlash('danger', __('blogs.tag_search_rule'));
  561.  
  562. return redirect('blogs/tags');
  563. }
  564.  
  565. if ($request->session()->missing(['findresult', 'blogfind']) ||
  566. $tag !== $request->session()->get('blogfind')
  567. ) {
  568. $result = Article::query()
  569. ->select('id')
  570. ->where('tags', 'like', '%' . $tag . '%')
  571. ->limit(500)
  572. ->pluck('id')
  573. ->all();
  574.  
  575. $request->session()->put('blogfind', $tag);
  576. $request->session()->put('findresult', $result);
  577. }
  578.  
  579. $articles = Article::query()
  580. ->select('articles.*', 'blogs.name')
  581. ->whereIn('articles.id', $request->session()->get('findresult'))
  582. ->join('blogs', 'articles.category_id', 'blogs.id')
  583. ->orderByDesc('created_at')
  584. ->with('user')
  585. ->paginate(setting('blogpost'));
  586.  
  587. return view('blogs/tags_search', compact('articles', 'tag'));
  588. }
  589.  
  590. /**
  591. * Новые статьи
  592. *
  593. * @return View
  594. */
  595. public function newArticles(): View
  596. {
  597. $articles = Article::query()
  598. ->orderByDesc('created_at')
  599. ->with('user', 'category')
  600. ->paginate(setting('blogpost'));
  601.  
  602. return view('blogs/new_articles', compact('articles'));
  603. }
  604.  
  605. /**
  606. * Новые комментарии
  607. *
  608. * @return View
  609. */
  610. public function newComments(): View
  611. {
  612. $comments = Comment::query()
  613. ->select('comments.*', 'title', 'count_comments')
  614. ->where('relate_type', Article::$morphName)
  615. ->leftJoin('articles', 'comments.relate_id', 'articles.id')
  616. ->orderByDesc('comments.created_at')
  617. ->with('user', 'relate')
  618. ->paginate(setting('comments_per_page'));
  619.  
  620. return view('blogs/new_comments', compact('comments'));
  621. }
  622.  
  623. /**
  624. * Статьи пользователя
  625. *
  626. * @param Request $request
  627. *
  628. * @return View
  629. */
  630. public function userArticles(Request $request): View
  631. {
  632. $login = $request->input('user', getUser('login'));
  633. $user = getUserByLogin($login);
  634.  
  635. if (! $user) {
  636. abort(404, __('validator.user'));
  637. }
  638.  
  639. $articles = Article::query()->where('user_id', $user->id)
  640. ->orderByDesc('created_at')
  641. ->with('user')
  642. ->paginate(setting('blogpost'))
  643. ->appends(['user' => $user->login]);
  644.  
  645. return view('blogs/active_articles', compact('articles', 'user'));
  646. }
  647.  
  648. /**
  649. * Комментарии пользователя
  650. *
  651. * @param Request $request
  652. *
  653. * @return View
  654. */
  655. public function userComments(Request $request): View
  656. {
  657. $login = $request->input('user', getUser('login'));
  658. $user = getUserByLogin($login);
  659.  
  660. if (! $user) {
  661. abort(404, __('validator.user'));
  662. }
  663.  
  664. $comments = Comment::query()
  665. ->select('comments.*', 'title', 'count_comments')
  666. ->where('relate_type', Article::$morphName)
  667. ->where('comments.user_id', $user->id)
  668. ->leftJoin('articles', 'comments.relate_id', 'articles.id')
  669. ->orderByDesc('comments.created_at')
  670. ->with('user', 'relate')
  671. ->paginate(setting('comments_per_page'))
  672. ->appends(['user' => $user->login]);
  673.  
  674. return view('blogs/active_comments', compact('comments', 'user'));
  675. }
  676.  
  677. /**
  678. * Переход к сообщению
  679. *
  680. * @param int $id
  681. * @param int $cid
  682. *
  683. * @return RedirectResponse
  684. */
  685. public function viewComment(int $id, int $cid): RedirectResponse
  686. {
  687. /** @var Article $article */
  688. $article = Article::query()->find($id);
  689.  
  690. if (! $article) {
  691. abort(404, __('blogs.article_not_exist'));
  692. }
  693.  
  694. $total = $article->comments()
  695. ->where('id', '<=', $cid)
  696. ->orderBy('created_at')
  697. ->count();
  698.  
  699. $end = ceil($total / setting('comments_per_page'));
  700.  
  701. return redirect('articles/comments/' . $article->id . '?page=' . $end . '#comment_' . $cid);
  702. }
  703.  
  704. /**
  705. * Топ статей
  706. *
  707. * @param Request $request
  708. *
  709. * @return View
  710. */
  711. public function top(Request $request): View
  712. {
  713. $sort = check($request->input('sort', 'visits'));
  714.  
  715. switch ($sort) {
  716. case 'rating':
  717. $order = 'rating';
  718. break;
  719. case 'comments':
  720. $order = 'count_comments';
  721. break;
  722. default:
  723. $order = 'visits';
  724. }
  725.  
  726. $articles = Article::query()
  727. ->select('articles.*', 'blogs.name')
  728. ->leftJoin('blogs', 'articles.category_id', 'blogs.id')
  729. ->orderByDesc($order)
  730. ->with('user')
  731. ->paginate(setting('blogpost'))
  732. ->appends(['sort' => $sort]);
  733.  
  734. return view('blogs/top', compact('articles', 'order'));
  735. }
  736.  
  737. /**
  738. * Поиск
  739. *
  740. * @param Request $request
  741. * @param Validator $validator
  742. *
  743. * @return View|RedirectResponse
  744. */
  745. public function search(Request $request, Validator $validator)
  746. {
  747. $find = check($request->input('find'));
  748. $articles = collect();
  749.  
  750. if ($find) {
  751. $find = rawurldecode(trim(preg_replace('/[^\w\x7F-\xFF\s]/', ' ', $find)));
  752.  
  753. $validator->length($find, 3, 64, ['find' => __('main.request_length')]);
  754. if ($validator->isValid()) {
  755. if (config('database.default') === 'mysql') {
  756. [$sql, $bindings] = ['MATCH (title, text) AGAINST (? IN BOOLEAN MODE)', [$find . '*']];
  757. } else {
  758. [$sql, $bindings] = ['title ILIKE ? OR text ILIKE ?', ['%' . $find . '%', '%' . $find . '%']];
  759. }
  760.  
  761. $articles = Article::query()
  762. ->whereRaw($sql, $bindings)
  763. ->with('user', 'category')
  764. ->paginate(setting('blogpost'))
  765. ->appends(compact('find'));
  766.  
  767. if ($articles->isEmpty()) {
  768. setInput($request->all());
  769. setFlash('danger', __('main.empty_found'));
  770.  
  771. return redirect('loads/search');
  772. }
  773. } else {
  774. setInput($request->all());
  775. setFlash('danger', $validator->getErrors());
  776. }
  777. }
  778.  
  779. return view('blogs/search', compact('articles', 'find'));
  780. }
  781.  
  782. /**
  783. * Список всех блогов (Для вывода на главную страницу)
  784. *
  785. * @return View
  786. */
  787. public function main(): View
  788. {
  789. $articles = Article::query()
  790. ->orderByDesc('created_at')
  791. ->with('user', 'category.parent')
  792. ->paginate(setting('blogpost'));
  793.  
  794. return view('blogs/main', compact('articles'));
  795. }
  796. }