<?php
namespace App\Controllers\Forum;
use App\Classes\Validator;
use App\Controllers\BaseController;
use App\Models\Flood;
use App\Models\Forum;
use App\Models\Post;
use App\Models\Topic;
use App\Models\Vote;
use App\Models\VoteAnswer;
use Illuminate\Database\Capsule\Manager as DB;
use Illuminate\Database\Query\Builder;
use Illuminate\Http\Request;
class ForumController extends BaseController
{
/**
* Главная страница
*
* @return string
*/
public function index(): string
{
$forums = Forum::query()
->where('parent_id', 0)
->with('lastTopic.lastPost.user')
->with('children')
->orderBy('sort')
->get();
if ($forums->isEmpty()) {
abort('default', 'Разделы форума еще не созданы!');
}
return view('forums/index', compact('forums'));
}
/**
* Страница списка тем
*
* @param int $id
* @return string
*/
public function forum(int $id): string
{
/** @var Forum $forum */
$forum = Forum::query()->with('parent', 'children.lastTopic.lastPost.user')->find($id);
if (! $forum) {
abort(404, 'Данного раздела не существует!');
}
$total = Topic::query()->where('forum_id', $forum->id)->count();
$page = paginate(setting('forumtem'), $total);
$topics = Topic::query()
->where('forum_id', $forum->id)
->orderBy('locked', 'desc')
->orderBy('updated_at', 'desc')
->offset($page->offset)
->limit($page->limit)
->with('lastPost.user')
->get();
return view('forums/forum', compact('forum', 'topics', 'page'));
}
/**
* Создание новой темы
*
* @param Request $request
* @param Validator $validator
* @return string
*/
public function create(Request $request, Validator $validator): string
{
$fid = int($request->input('fid'));
$forums = Forum::query()
->where('parent_id', 0)
->with('children')
->orderBy('sort')
->get();
if ($forums->isEmpty()) {
abort('default', 'Разделы форума еще не созданы!');
}
if (! $user = getUser()) {
abort(403);
}
if ($request->isMethod('post')) {
$title = check($request->input('title'));
$msg = check($request->input('msg'));
$token = check($request->input('token'));
$vote = empty($request->input('vote')) ? 0 : 1;
$question = check($request->input('question'));
$answers = check($request->input('answer'));
/** @var Forum $forum */
$forum = Forum::query()->find($fid);
$validator->equal($token, $_SESSION['token'], 'Неверный идентификатор сессии, повторите действие!')
->notEmpty($forum, ['fid' => 'Форума для новой темы не существует!'])
->equal(Flood::isFlood(), true, ['msg' => 'Антифлуд! Разрешается cоздавать темы раз в '.Flood::getPeriod().' сек!'])
->length($title, 5, 50, ['title' => 'Слишком длинное или короткое название темы!'])
->length($msg, 5, setting('forumtextlength'), ['msg' => 'Слишком длинный или короткий текст сообщения!']);
if ($forum) {
$validator->empty($forum->closed, ['fid' => 'В данном форуме запрещено создавать темы!']);
}
if ($vote) {
$validator->length($question, 5, 100, ['question' => 'Слишком длинный или короткий текст вопроса!']);
$answers = array_unique(array_diff($answers, ['']));
foreach ($answers as $answer) {
if (utfStrlen($answer) > 50) {
$validator->addError(['answer' => 'Длина вариантов ответа не должна быть более 50 символов!']);
break;
}
}
$validator->between(\count($answers), 2, 10, ['answer' => 'Недостаточное количество вариантов ответов!']);
}
/* TODO: Сделать проверку поиска похожей темы */
if ($validator->isValid()) {
$title = antimat($title);
$msg = antimat($msg);
$user->update([
'allforum' => DB::raw('allforum + 1'),
'point' => DB::raw('point + 1'),
'money' => DB::raw('money + 5'),
]);
/** @var Topic $topic */
$topic = Topic::query()->create([
'forum_id' => $forum->id,
'title' => $title,
'user_id' => getUser('id'),
'count_posts' => 1,
'created_at' => SITETIME,
'updated_at' => SITETIME,
]);
/** @var Post $post */
$post = Post::query()->create([
'topic_id' => $topic->id,
'user_id' => getUser('id'),
'text' => $msg,
'created_at' => SITETIME,
'ip' => getIp(),
'brow' => getBrowser(),
]);
Topic::query()->where('id', $topic->id)->update(['last_post_id' => $post->id]);
$forum->update([
'count_topics' => DB::raw('count_topics + 1'),
'count_posts' => DB::raw('count_posts + 1'),
'last_topic_id' => $topic->id,
]);
// Обновление родительского форума
if ($forum->parent->id) {
$forum->parent->update([
'last_topic_id' => $topic->id
]);
}
// Создание голосования
if ($vote) {
/** @var Vote $vote */
$vote = Vote::query()->create([
'title' => $question,
'topic_id' => $topic->id,
'created_at' => SITETIME,
]);
$prepareAnswers = [];
foreach ($answers as $answer) {
$prepareAnswers[] = [
'vote_id' => $vote->id,
'answer' => $answer
];
}
VoteAnswer::query()->insert($prepareAnswers);
}
setFlash('success', 'Новая тема успешно создана!');
redirect('/topics/'.$topic->id);
} else {
setInput($request->all());
setFlash('danger', $validator->getErrors());
}
}
return view('forums/forum_create', compact('forums', 'fid'));
}
/**
* Поиск
*
* @param Request $request
* @return string
*/
public function search(Request $request): ?string
{
$fid = check($request->input('fid'));
$find = check($request->input('find'));
$type = int($request->input('type'));
$where = int($request->input('where'));
$period = int($request->input('period'));
$section = int($request->input('section'));
if (! $find) {
$forums = Forum::query()
->where('parent_id', 0)
->with('children')
->orderBy('sort')
->get();
if ($forums->isEmpty()) {
abort('default', 'Разделы форума еще не созданы!');
}
return view('forums/search', compact('forums', 'fid'));
}
$find = str_replace(['@', '+', '-', '*', '~', '<', '>', '(', ')', '"', "'"], '', $find);
if (! isUtf($find)) {
$find = winToUtf($find);
}
$strlen = utfStrlen($find);
if ($strlen >= 3 && $strlen <= 50) {
$findmewords = explode(' ', utfLower($find));
$arrfind = [];
foreach ($findmewords as $val) {
if (utfStrlen($val) >= 3) {
$arrfind[] = empty($type) ? '+' . $val . '*' : $val . '*';
}
}
$findme = implode(' ', $arrfind);
if ($type === 2 && \count($findmewords) > 1) {
$findme = "\"$find\"";
}
$wheres = empty($where) ? 'topics' : 'posts';
$forumfind = ($type . $wheres . $period . $section . $find);
// Поиск в темах
if ($wheres === 'topics') {
if (empty($_SESSION['forumfindres']) || $forumfind !== $_SESSION['forumfind']) {
$searchsec = ($section > 0) ? 'forum_id = ' . $section . ' AND' : '';
$searchper = ($period > 0) ? 'updated_at > ' . strtotime('-' . $period . ' day', SITETIME) . ' AND' : '';
$result = Topic::query()
->select('id')
->whereRaw($searchsec . ' ' . $searchper . ' MATCH (`title`) AGAINST (? IN BOOLEAN MODE)', [$findme])
->limit(100)
->pluck('id')
->all();
$_SESSION['forumfind'] = $forumfind;
$_SESSION['forumfindres'] = $result;
}
$total = \count($_SESSION['forumfindres']);
if ($total > 0) {
$page = paginate(setting('forumtem'), $total);
$topics = Topic::query()
->whereIn('id', $_SESSION['forumfindres'])
->with('lastPost.user')
->orderBy('updated_at', 'desc')
->offset($page->offset)
->limit($page->limit)
->get();
return view('forums/search_topics', compact('topics', 'page', 'find', 'type', 'where', 'section', 'period'));
}
setInput($request->all());
setFlash('danger', 'По вашему запросу ничего не найдено!');
redirect('/forums/search');
}
// Поиск в сообщениях
if ($wheres === 'posts') {
if (empty($_SESSION['forumfindres']) || $forumfind !== $_SESSION['forumfind']) {
$searchsec = ($section > 0) ? 'topics.forum_id = ' . $section . ' AND' : '';
$searchper = ($period > 0) ? 'posts.created_at > ' . strtotime('-' . $period . ' day', SITETIME) . ' AND' : '';
$result = Post::query()
->select('posts.id')
->leftJoin('topics', 'posts.topic_id', 'topics.id')
->whereRaw($searchsec . ' ' . $searchper . ' MATCH (`text`) AGAINST (? IN BOOLEAN MODE)', [$findme])
->limit(100)
->pluck('id')
->all();
$_SESSION['forumfind'] = $forumfind;
$_SESSION['forumfindres'] = $result;
}
$total = \count($_SESSION['forumfindres']);
if ($total > 0) {
$page = paginate(setting('forumpost'), $total);
$posts = Post::query()
->whereIn('id', $_SESSION['forumfindres'])
->with('user', 'topic')
->orderBy('created_at', 'desc')
->offset($page->offset)
->limit($page->limit)
->get();
return view('forums/search_posts', compact('posts', 'page', 'find', 'type', 'where', 'section', 'period'));
}
setInput($request->all());
setFlash('danger', 'По вашему запросу ничего не найдено!');
redirect('/forums/search');
}
} else {
setInput($request->all());
setFlash('danger', ['find' => 'Запрос должен содержать от 3 до 50 символов!']);
redirect('/forums/search');
}
}
/**
* RSS всех топиков
*
* @return string
*/
public function rss(): string
{
$topics = Topic::query()
->where('closed', 0)
->with('lastPost.user')
->orderBy('updated_at', 'desc')
->limit(15)
->get();
if ($topics->isEmpty()) {
abort('default', 'Нет тем для отображения!');
}
return view('forums/rss', compact('topics'));
}
/**
* RSS постов
*
* @param int $id
* @return string
*/
public function rssPosts(int $id): string
{
/** @var Topic $topic */
$topic = Topic::query()->find($id);
if (empty($topic)) {
abort(404, 'Данной темы не существует!');
}
$posts = Post::query()
->where('topic_id', $topic->id)
->orderBy('created_at', 'desc')
->with('user')
->limit(15)
->get();
return view('forums/rss_posts', compact('topic', 'posts'));
}
/**
* Последние темы
*
* @return string
*/
public function topTopics(): string
{
$total = Topic::query()->count();
if ($total > 500) {
$total = 500;
}
$page = paginate(setting('forumtem'), $total);
$topics = Topic::query()
->where('closed', 0)
->orderBy('count_posts', 'desc')
->limit($page->limit)
->offset($page->offset)
->with('forum', 'user', 'lastPost.user')
->get();
return view('forums/top', compact('topics', 'page'));
}
/**
* Последние сообщения
*
* @param Request $request
* @return string
*/
public function topPosts(Request $request): string
{
$period = int($request->input('period'));
$total = Post::query()
->when($period, function (Builder $query) use ($period) {
return $query->where('created_at', '>', strtotime('-' . $period . ' day', SITETIME));
})
->count();
if ($total > 500) {
$total = 500;
}
$page = paginate(setting('forumpost'), $total);
$posts = Post::query()
->when($period, function (Builder $query) use ($period) {
return $query->where('created_at', '>', strtotime('-' . $period . ' day', SITETIME));
})
->orderBy('rating', 'desc')
->orderBy('created_at', 'desc')
->limit($page->limit)
->offset($page->offset)
->with('topic', 'user')
->get();
return view('forums/top_posts', compact('posts', 'page', 'period'));
}
}