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

Размер файла: 22.84Kb
  1. <?php
  2.  
  3. declare(strict_types=1);
  4.  
  5. namespace App\Http\Controllers\Forum;
  6.  
  7. use App\Http\Controllers\Controller;
  8. use App\Models\File;
  9. use App\Models\Polling;
  10. use App\Models\Post;
  11. use App\Models\Bookmark;
  12. use App\Models\Reader;
  13. use App\Models\Topic;
  14. use App\Models\User;
  15. use App\Models\Vote;
  16. use App\Models\Flood;
  17. use App\Classes\Validator;
  18. use App\Models\VoteAnswer;
  19. use Illuminate\Http\RedirectResponse;
  20. use Illuminate\Support\Facades\DB;
  21. use Illuminate\Database\Eloquent\Builder;
  22. use Illuminate\Database\Query\JoinClause;
  23. use Illuminate\Http\Request;
  24. use Illuminate\Support\Arr;
  25. use Illuminate\View\View;
  26.  
  27. class TopicController extends Controller
  28. {
  29. /**
  30. * Main page
  31. *
  32. * @param int $id
  33. *
  34. * @return View
  35. */
  36. public function index(int $id): View
  37. {
  38. $user = getUser();
  39.  
  40. $topic = Topic::query()
  41. ->where('topics.id', $id)
  42. ->when($user, static function (Builder $query) use ($user) {
  43. $query->select('topics.*', 'bookmarks.count_posts as bookmark_posts')
  44. ->leftJoin('bookmarks', static function (JoinClause $join) use ($user) {
  45. $join->on('topics.id', 'bookmarks.topic_id')
  46. ->where('bookmarks.user_id', $user->id);
  47. });
  48. })
  49. ->with('forum.parent')
  50. ->first();
  51.  
  52. if (! $topic) {
  53. abort(404, __('forums.topic_not_exist'));
  54. }
  55.  
  56. $posts = Post::query()
  57. ->where('topic_id', $topic->id)
  58. ->when($user, static function (Builder $query) use ($user) {
  59. $query->select('posts.*', 'pollings.vote')
  60. ->leftJoin('pollings', static function (JoinClause $join) use ($user) {
  61. $join->on('posts.id', 'pollings.relate_id')
  62. ->where('pollings.relate_type', Post::$morphName)
  63. ->where('pollings.user_id', $user->id);
  64. });
  65. })
  66. ->with('files', 'user', 'editUser')
  67. ->orderBy('created_at')
  68. ->paginate(setting('forumpost'));
  69.  
  70. if ($posts->onFirstPage()) {
  71. $firstPost = $posts->first();
  72. } else {
  73. $firstPost = Post::query()->where('topic_id', $topic->id)->orderBy('created_at')->first();
  74. }
  75.  
  76. if (isset($topic->bookmark_posts) && $topic->count_posts > $topic->bookmark_posts) {
  77. Bookmark::query()
  78. ->where('topic_id', $topic->id)
  79. ->where('user_id', $user->id)
  80. ->update(['count_posts' => $topic->count_posts]);
  81. }
  82.  
  83. // Curators
  84. if ($topic->moderators) {
  85. $topic->curators = User::query()->whereIn('login', explode(',', (string) $topic->moderators))->get();
  86. $topic->isModer = $user ? $topic->curators->where('id', $user->id)->isNotEmpty() : false;
  87. }
  88.  
  89. // Visits
  90. Reader::countingStat($topic);
  91.  
  92. // Votes
  93. $vote = Vote::query()->where('topic_id', $topic->id)->first();
  94.  
  95. if ($vote) {
  96. $vote->poll = $vote->pollings()
  97. ->where('user_id', $user->id ?? null)
  98. ->first();
  99.  
  100. if ($vote->answers->isNotEmpty()) {
  101. $results = Arr::pluck($vote->answers, 'result', 'answer');
  102. $max = max($results);
  103.  
  104. arsort($results);
  105.  
  106. $vote->voted = $results;
  107.  
  108. $vote->sum = ($vote->count > 0) ? $vote->count : 1;
  109. $vote->max = ($max > 0) ? $max : 1;
  110. }
  111. }
  112.  
  113. $description = $firstPost ? truncateDescription(bbCode($firstPost->text, false)) : $topic->title;
  114.  
  115. return view('forums/topic', compact('topic', 'posts', 'vote', 'description'));
  116. }
  117.  
  118. /**
  119. * Message creation
  120. *
  121. * @param int $id
  122. * @param Request $request
  123. * @param Validator $validator
  124. * @param Flood $flood
  125. *
  126. * @return RedirectResponse
  127. */
  128. public function create(int $id, Request $request, Validator $validator, Flood $flood): RedirectResponse
  129. {
  130. $msg = $request->input('msg');
  131. $files = (array) $request->file('files');
  132.  
  133. if (! $user = getUser()) {
  134. abort(403, __('main.not_authorized'));
  135. }
  136.  
  137. $topic = Topic::query()
  138. ->select('topics.*', 'forums.parent_id')
  139. ->where('topics.id', $id)
  140. ->leftJoin('forums', 'topics.forum_id', 'forums.id')
  141. ->first();
  142.  
  143. if (! $topic) {
  144. abort(404, __('forums.topic_not_exist'));
  145. }
  146.  
  147. $validator->equal($request->input('_token'), csrf_token(), ['msg' => __('validator.token')])
  148. ->empty($topic->closed, ['msg' => __('forums.topic_closed')])
  149. ->false($flood->isFlood(), ['msg' => __('validator.flood', ['sec' => $flood->getPeriod()])])
  150. ->length($msg, 5, setting('forumtextlength'), ['msg' => __('validator.text')]);
  151.  
  152. // Проверка сообщения на схожесть
  153. /** @var Post $post */
  154. $post = Post::query()->where('topic_id', $topic->id)->orderByDesc('id')->first();
  155. $validator->notEqual($msg, $post->text, ['msg' => __('forums.post_repeat')]);
  156.  
  157. if ($files && $validator->isValid()) {
  158. $validator
  159. ->lte(count($files), setting('maxfiles'), ['files' => __('validator.files_max', ['max' => setting('maxfiles')])]);
  160.  
  161. $rules = [
  162. 'maxsize' => setting('filesize'),
  163. 'extensions' => explode(',', setting('file_extensions')),
  164. ];
  165.  
  166. foreach ($files as $file) {
  167. $validator->file($file, $rules, ['files' => __('validator.file_upload_failed')]);
  168. }
  169. }
  170.  
  171. if ($validator->isValid()) {
  172. $msg = antimat($msg);
  173.  
  174. if (
  175. $post
  176. && $post->created_at + 600 > SITETIME
  177. && $user->id === $post->user_id
  178. && (utfStrlen($msg) + utfStrlen($post->text) <= setting('forumtextlength'))
  179. && count($files) + $post->files->count() <= setting('maxfiles')
  180. ) {
  181. $newpost = $post->text . PHP_EOL . PHP_EOL . '[i][size=1]' . __('forums.post_added_after', ['sec' => makeTime(SITETIME - $post->created_at)]) . '[/size][/i]' . PHP_EOL . $msg;
  182.  
  183. $post->update(['text' => $newpost]);
  184. } else {
  185. $post = Post::query()->create([
  186. 'topic_id' => $topic->id,
  187. 'user_id' => $user->id,
  188. 'text' => $msg,
  189. 'created_at' => SITETIME,
  190. 'ip' => getIp(),
  191. 'brow' => getBrowser(),
  192. ]);
  193.  
  194. $user->increment('allforum');
  195. $user->increment('point');
  196. $user->increment('money', 5);
  197.  
  198. $topic->update([
  199. 'count_posts' => DB::raw('count_posts + 1'),
  200. 'last_post_id' => $post->id,
  201. 'updated_at' => SITETIME,
  202. ]);
  203.  
  204. $topic->forum->update([
  205. 'count_posts' => DB::raw('count_posts + 1'),
  206. 'last_topic_id' => $topic->id,
  207. ]);
  208.  
  209. // Обновление родительского форума
  210. if ($topic->forum->parent_id) {
  211. $topic->forum->parent->update([
  212. 'last_topic_id' => $topic->id,
  213. ]);
  214. }
  215. }
  216.  
  217. clearCache(['statForums', 'recentTopics']);
  218. $flood->saveState();
  219. sendNotify($msg, '/topics/' . $topic->id . '/' . $post->id, $topic->title);
  220.  
  221. if ($files) {
  222. foreach ($files as $file) {
  223. $post->uploadFile($file);
  224. }
  225. }
  226.  
  227. setFlash('success', __('main.message_added_success'));
  228. } else {
  229. setInput($request->all());
  230. setFlash('danger', $validator->getErrors());
  231. }
  232.  
  233. return redirect('topics/end/' . $topic->id);
  234. }
  235.  
  236. /**
  237. * Delete messages
  238. *
  239. * @param int $id
  240. * @param Request $request
  241. * @param Validator $validator
  242. *
  243. * @return RedirectResponse
  244. */
  245. public function delete(int $id, Request $request, Validator $validator): RedirectResponse
  246. {
  247. $del = intar($request->input('del'));
  248. $page = int($request->input('page'));
  249.  
  250. if (! $user = getUser()) {
  251. abort(403, __('main.not_authorized'));
  252. }
  253.  
  254. /** @var Topic $topic */
  255. $topic = Topic::query()->find($id);
  256.  
  257. if (! $topic) {
  258. abort(404, __('forums.topic_not_exist'));
  259. }
  260.  
  261. $isModer = in_array($user->login, explode(',', (string) $topic->moderators), true);
  262.  
  263. $validator->equal($request->input('_token'), csrf_token(), __('validator.token'))
  264. ->notEmpty($del, __('validator.deletion'))
  265. ->empty($topic->closed, __('forums.topic_closed'))
  266. ->equal($isModer, true, __('forums.posts_deleted_curators'));
  267.  
  268. if ($validator->isValid()) {
  269. // ------ Удаление загруженных файлов -------//
  270. $files = File::query()
  271. ->where('relate_type', Post::$morphName)
  272. ->whereIn('relate_id', $del)
  273. ->get();
  274.  
  275. if ($files->isNotEmpty()) {
  276. foreach ($files as $file) {
  277. $file->delete();
  278. }
  279. }
  280.  
  281. $delPosts = Post::query()->whereIn('id', $del)->delete();
  282.  
  283. $topic->decrement('count_posts', $delPosts);
  284. $topic->forum->decrement('count_posts', $delPosts);
  285.  
  286. setFlash('success', __('main.messages_deleted_success'));
  287. } else {
  288. setFlash('danger', $validator->getErrors());
  289. }
  290.  
  291. return redirect('topics/' . $topic->id . '?page=' . $page);
  292. }
  293.  
  294. /**
  295. * Close topic
  296. *
  297. * @param int $id
  298. * @param Request $request
  299. * @param Validator $validator
  300. *
  301. * @return RedirectResponse
  302. */
  303. public function close(int $id, Request $request, Validator $validator): RedirectResponse
  304. {
  305. if (! $user = getUser()) {
  306. abort(403, __('main.not_authorized'));
  307. }
  308.  
  309. /** @var Topic $topic */
  310. $topic = Topic::query()->find($id);
  311.  
  312. $validator->equal($request->input('_token'), csrf_token(), __('validator.token'))
  313. ->gte($user->point, setting('editforumpoint'), __('forums.topic_edited_points', ['point' => plural(setting('editforumpoint'), setting('scorename'))]))
  314. ->notEmpty($topic, __('forums.topic_not_exist'))
  315. ->equal($topic->user_id, $user->id, __('forums.topic_not_author'))
  316. ->empty($topic->closed, __('forums.topic_closed'));
  317.  
  318. if ($validator->isValid()) {
  319. $topic->update([
  320. 'closed' => 1,
  321. 'close_user_id' => getUser('id'),
  322. ]);
  323.  
  324. if ($topic->vote) {
  325. $topic->vote->update(['closed' => 1]);
  326. $topic->vote->pollings()->delete();
  327. }
  328.  
  329. setFlash('success', __('forums.topic_success_closed'));
  330. } else {
  331. setFlash('danger', $validator->getErrors());
  332. }
  333.  
  334. return redirect('topics/' . $topic->id);
  335. }
  336.  
  337. /**
  338. * Open topic
  339. *
  340. * @param int $id
  341. * @param Request $request
  342. * @param Validator $validator
  343. *
  344. * @return RedirectResponse
  345. */
  346. public function open(int $id, Request $request, Validator $validator): RedirectResponse
  347. {
  348. if (! $user = getUser()) {
  349. abort(403, __('main.not_authorized'));
  350. }
  351.  
  352. /** @var Topic $topic */
  353. $topic = Topic::query()->find($id);
  354.  
  355. $validator->equal($request->input('_token'), csrf_token(), __('validator.token'))
  356. ->notEmpty($topic, __('forums.topic_not_exist'))
  357. ->equal($topic->user_id, $user->id, __('forums.topic_not_author'))
  358. ->equal($topic->close_user_id, $user->id, __('forums.topic_opened_author'))
  359. ->notEmpty($topic->closed, __('forums.topic_already_open'));
  360.  
  361. if ($validator->isValid()) {
  362. $topic->update([
  363. 'closed' => 0,
  364. 'close_user_id' => null,
  365. ]);
  366.  
  367. if ($topic->vote) {
  368. $topic->vote->update(['closed' => 0]);
  369. }
  370.  
  371. setFlash('success', __('forums.topic_success_opened'));
  372. } else {
  373. setFlash('danger', $validator->getErrors());
  374. }
  375.  
  376. return redirect('topics/' . $topic->id);
  377. }
  378.  
  379. /**
  380. * Topic editing
  381. *
  382. * @param int $id
  383. * @param Request $request
  384. * @param Validator $validator
  385. *
  386. * @return View|RedirectResponse
  387. */
  388. public function edit(int $id, Request $request, Validator $validator)
  389. {
  390. if (! $user = getUser()) {
  391. abort(403, __('main.not_authorized'));
  392. }
  393.  
  394. if ($user->point < setting('editforumpoint')) {
  395. abort(200, __('forums.topic_edited_points', ['point' => plural(setting('editforumpoint'), setting('scorename'))]));
  396. }
  397.  
  398. /** @var Topic $topic */
  399. $topic = Topic::query()->find($id);
  400.  
  401. if (! $topic) {
  402. abort(404, __('forums.topic_not_exist'));
  403. }
  404.  
  405. if ($topic->user_id !== $user->id) {
  406. abort(200, __('forums.topic_not_author'));
  407. }
  408.  
  409. if ($topic->closed) {
  410. abort(200, __('forums.topic_closed'));
  411. }
  412.  
  413. $post = Post::query()->where('topic_id', $topic->id)
  414. ->orderBy('id')
  415. ->first();
  416.  
  417. /** @var Vote $vote */
  418. $vote = Vote::query()->where('topic_id', $id)->first();
  419.  
  420. if ($request->isMethod('post')) {
  421. $title = $request->input('title');
  422. $msg = $request->input('msg');
  423. $question = $request->input('question');
  424. $answers = (array) $request->input('answers');
  425.  
  426. $validator->equal($request->input('_token'), csrf_token(), __('validator.token'))
  427. ->length($title, 3, 50, ['title' => __('validator.text')]);
  428.  
  429. if ($post) {
  430. $validator->length($msg, 5, setting('forumtextlength'), ['msg' => __('validator.text')]);
  431. }
  432.  
  433. if ($vote) {
  434. $validator->length($question, 5, 100, ['question' => __('validator.text')]);
  435.  
  436. if ($answers) {
  437. $validator->empty($vote->count, ['question' => __('votes.answer_changed_impossible')]);
  438.  
  439. $answers = array_unique(array_diff($answers, ['']));
  440.  
  441. foreach ($answers as $answer) {
  442. if (utfStrlen($answer) > 50) {
  443. $validator->addError(['answers' => __('votes.answer_wrong_length')]);
  444. break;
  445. }
  446. }
  447.  
  448. $validator->between(count($answers), 2, 10, ['answers' => __('votes.answer_not_enough')]);
  449. }
  450. }
  451.  
  452. if ($validator->isValid()) {
  453. $title = antimat($title);
  454. $msg = antimat($msg);
  455.  
  456. $topic->update(['title' => $title]);
  457.  
  458. if ($post) {
  459. $post->update([
  460. 'text' => $msg,
  461. 'edit_user_id' => $user->id,
  462. 'updated_at' => SITETIME,
  463. ]);
  464. }
  465.  
  466. if ($vote) {
  467. $vote->update([
  468. 'title' => $question,
  469. ]);
  470.  
  471. if ($answers) {
  472. $countAnswers = $vote->answers()->count();
  473.  
  474. foreach ($answers as $answerId => $answer) {
  475. /** @var VoteAnswer $ans */
  476. $ans = $vote->answers()->firstOrNew(['id' => $answerId]);
  477.  
  478. if ($ans->exists) {
  479. $ans->update(['answer' => $answer]);
  480. } elseif ($countAnswers < 10) {
  481. $ans->fill(['answer' => $answer])->save();
  482. $countAnswers++;
  483. }
  484. }
  485. }
  486. }
  487.  
  488. setFlash('success', __('forums.topic_success_changed'));
  489.  
  490. return redirect('topics/' . $topic->id);
  491. }
  492.  
  493. setInput($request->all());
  494. setFlash('danger', $validator->getErrors());
  495. }
  496.  
  497. if ($vote) {
  498. $vote->getAnswers = $vote->answers->pluck('answer', 'id')->all();
  499. }
  500.  
  501. return view('forums/topic_edit', compact('post', 'topic', 'vote'));
  502. }
  503.  
  504. /**
  505. * Post editing
  506. *
  507. * @param int $id
  508. * @param Request $request
  509. * @param Validator $validator
  510. *
  511. * @return View|RedirectResponse
  512. */
  513. public function editPost(int $id, Request $request, Validator $validator)
  514. {
  515. $page = int($request->input('page'));
  516.  
  517. if (! $user = getUser()) {
  518. abort(403, __('main.not_authorized'));
  519. }
  520.  
  521. /** @var Post $post */
  522. $post = Post::query()
  523. ->select('posts.*', 'moderators', 'closed')
  524. ->leftJoin('topics', 'posts.topic_id', 'topics.id')
  525. ->where('posts.id', $id)
  526. ->first();
  527.  
  528. if (! $post) {
  529. abort(404, __('forums.post_not_exist'));
  530. }
  531.  
  532. if ($post->closed) {
  533. abort(200, __('forums.topic_closed'));
  534. }
  535.  
  536. $isModer = in_array($user->login, explode(',', (string) $post->moderators), true);
  537.  
  538. if (! $isModer && $post->user_id !== $user->id) {
  539. abort(200, __('forums.posts_edited_curators'));
  540. }
  541.  
  542. if (! $isModer && $post->created_at + 600 < SITETIME) {
  543. abort(200, __('main.editing_impossible'));
  544. }
  545.  
  546. if ($request->isMethod('post')) {
  547. $msg = $request->input('msg');
  548. $delfile = intar($request->input('delfile'));
  549.  
  550. $validator->equal($request->input('_token'), csrf_token(), __('validator.token'))
  551. ->length($msg, 5, setting('forumtextlength'), ['msg' => __('validator.text')]);
  552.  
  553. if ($validator->isValid()) {
  554. $post->update([
  555. 'text' => antimat($msg),
  556. 'edit_user_id' => $user->id,
  557. 'updated_at' => SITETIME,
  558. ]);
  559.  
  560. // ------ Удаление загруженных файлов -------//
  561. if ($delfile) {
  562. $files = $post->files()
  563. ->whereIn('id', $delfile)
  564. ->get();
  565.  
  566. if ($files->isNotEmpty()) {
  567. foreach ($files as $file) {
  568. $file->delete();
  569. }
  570. }
  571. }
  572.  
  573. setFlash('success', __('main.message_edited_success'));
  574.  
  575. return redirect('topics/' . $post->topic_id . '?page=' . $page);
  576. }
  577.  
  578. setInput($request->all());
  579. setFlash('danger', $validator->getErrors());
  580. }
  581.  
  582. return view('forums/topic_edit_post', compact('post', 'page'));
  583. }
  584.  
  585. /**
  586. * Voting
  587. *
  588. * @param int $id
  589. * @param Request $request
  590. * @param Validator $validator
  591. *
  592. * @return RedirectResponse
  593. */
  594. public function vote(int $id, Request $request, Validator $validator): RedirectResponse
  595. {
  596. if (! $user = getUser()) {
  597. abort(403, __('main.not_authorized'));
  598. }
  599.  
  600. /** @var Vote $vote */
  601. $vote = Vote::query()->where('topic_id', $id)->first();
  602.  
  603. if (! $vote) {
  604. abort(404, __('votes.voting_not_found'));
  605. }
  606.  
  607. $poll = int($request->input('poll'));
  608. $page = int($request->input('page'));
  609.  
  610. $validator->equal($request->input('_token'), csrf_token(), __('validator.token'))
  611. ->notEmpty($poll, __('votes.answer_not_chosen'))
  612. ->empty($vote->closed, __('votes.voting_closed'));
  613.  
  614. if ($validator->isValid()) {
  615. $polling = $vote->pollings()
  616. ->where('user_id', $user->id)
  617. ->first();
  618. $validator->empty($polling, __('votes.voting_passed'));
  619. }
  620.  
  621. if ($validator->isValid()) {
  622. /** @var VoteAnswer $answer */
  623. $answer = $vote->answers()
  624. ->where('id', $poll)
  625. ->where('vote_id', $vote->id)
  626. ->first();
  627. $validator->notEmpty($answer, __('votes.answer_not_found'));
  628. }
  629.  
  630. if ($validator->isValid()) {
  631. $vote->increment('count');
  632. $answer->increment('result');
  633.  
  634. Polling::query()->create([
  635. 'relate_type' => Vote::$morphName,
  636. 'relate_id' => $vote->id,
  637. 'user_id' => $user->id,
  638. 'vote' => $answer->answer,
  639. 'created_at' => SITETIME,
  640. ]);
  641.  
  642. setFlash('success', __('votes.voting_success'));
  643. } else {
  644. setFlash('danger', $validator->getErrors());
  645. }
  646.  
  647. return redirect('topics/' . $vote->topic_id . '?page=' . $page);
  648. }
  649.  
  650. /**
  651. * Print topic
  652. *
  653. * @param int $id
  654. *
  655. * @return View
  656. */
  657. public function print(int $id): View
  658. {
  659. /** @var Topic $topic */
  660. $topic = Topic::query()->find($id);
  661.  
  662. if (! $topic) {
  663. abort(404, __('forums.topic_not_exist'));
  664. }
  665.  
  666. $posts = Post::query()
  667. ->where('topic_id', $topic->id)
  668. ->with('user')
  669. ->orderBy('created_at')
  670. ->get();
  671.  
  672. $description = $posts->first() ? truncateDescription(bbCode($posts->first()->text, false)) : $topic->title;
  673.  
  674. return view('forums/print', compact('topic', 'posts', 'description'));
  675. }
  676.  
  677. /**
  678. * Forward to message
  679. *
  680. * @param int $id
  681. * @param int $pid
  682. *
  683. * @return RedirectResponse
  684. */
  685. public function viewpost(int $id, int $pid): RedirectResponse
  686. {
  687. $countTopics = Post::query()
  688. ->where('id', '<=', $pid)
  689. ->where('topic_id', $id)
  690. ->count();
  691.  
  692. if (! $countTopics) {
  693. abort(404, __('forums.topic_not_exist'));
  694. }
  695.  
  696. $end = ceil($countTopics / setting('forumpost'));
  697.  
  698. return redirect('topics/' . $id . '?page=' . $end . '#post_' . $pid);
  699. }
  700.  
  701. /**
  702. * Forward to the last message
  703. *
  704. * @param int $id
  705. *
  706. * @return RedirectResponse
  707. */
  708. public function end(int $id): RedirectResponse
  709. {
  710. /** @var Topic $topic */
  711. $topic = Topic::query()->find($id);
  712.  
  713. if (! $topic) {
  714. abort(404, __('forums.topic_not_exist'));
  715. }
  716.  
  717. $end = ceil($topic->count_posts / setting('forumpost'));
  718.  
  719. return redirect('topics/' . $topic->id . '?page=' . $end);
  720. }
  721. }