misuzu/public-legacy/forum/topic.php

347 lines
12 KiB
PHP
Raw Permalink Normal View History

2022-09-13 13:14:49 +00:00
<?php
namespace Misuzu;
2023-08-28 01:17:34 +00:00
use stdClass;
use RuntimeException;
2024-10-05 02:40:29 +00:00
$urls = $msz->getUrls();
$forumCtx = $msz->getForumContext();
$forumCategories = $forumCtx->getCategories();
$forumTopics = $forumCtx->getTopics();
$forumTopicRedirects = $forumCtx->getTopicRedirects();
$forumPosts = $forumCtx->getPosts();
$usersCtx = $msz->getUsersContext();
2023-08-28 01:17:34 +00:00
2022-09-13 13:14:49 +00:00
$postId = !empty($_GET['p']) && is_string($_GET['p']) ? (int)$_GET['p'] : 0;
$topicId = !empty($_GET['t']) && is_string($_GET['t']) ? (int)$_GET['t'] : 0;
2023-08-28 01:17:34 +00:00
$categoryId = null;
2022-09-13 13:14:49 +00:00
$moderationMode = !empty($_GET['m']) && is_string($_GET['m']) ? (string)$_GET['m'] : '';
$submissionConfirmed = !empty($_GET['confirm']) && is_string($_GET['confirm']) && $_GET['confirm'] === '1';
$authInfo = $msz->getAuthInfo();
$currentUser = $authInfo->getUserInfo();
2023-08-28 01:17:34 +00:00
$currentUserId = $currentUser === null ? '0' : $currentUser->getId();
2022-09-13 13:14:49 +00:00
if($topicId < 1 && $postId > 0) {
2023-08-28 01:17:34 +00:00
try {
$postInfo = $forumPosts->getPost(postId: $postId);
2023-08-28 01:17:34 +00:00
} catch(RuntimeException $ex) {
Template::throwError(404);
2023-08-28 01:17:34 +00:00
}
$categoryId = $postInfo->getCategoryId();
$perms = $authInfo->getPerms('forum', $postInfo->getCategoryId());
2023-08-30 22:37:21 +00:00
$canDeleteAny = $perms->check(Perm::F_POST_DELETE_ANY);
2022-09-13 13:14:49 +00:00
if($postInfo->isDeleted() && !$canDeleteAny)
Template::throwError(404);
2023-08-28 01:17:34 +00:00
$topicId = $postInfo->getTopicId();
$preceedingPostCount = $forumPosts->countPosts(
2023-08-28 01:17:34 +00:00
topicInfo: $topicId,
upToPostInfo: $postInfo,
deleted: $canDeleteAny ? null : false
);
2022-09-13 13:14:49 +00:00
}
2023-08-28 01:17:34 +00:00
try {
$topicIsNuked = $topicIsDeleted = $canDeleteAny = false;
$topicInfo = $forumTopics->getTopic(topicId: $topicId);
2023-08-28 01:17:34 +00:00
} catch(RuntimeException $ex) {
$topicIsNuked = true;
}
2022-09-13 13:14:49 +00:00
2023-08-28 01:17:34 +00:00
if(!$topicIsNuked) {
$topicIsDeleted = $topicInfo->isDeleted();
2022-09-13 13:14:49 +00:00
2023-08-28 01:17:34 +00:00
if($categoryId !== (int)$topicInfo->getCategoryId()) {
$categoryId = (int)$topicInfo->getCategoryId();
$perms = $authInfo->getPerms('forum', $topicInfo->getCategoryId());
2023-08-28 01:17:34 +00:00
}
2022-09-13 13:14:49 +00:00
if($usersCtx->hasActiveBan($currentUser))
2023-08-30 22:37:21 +00:00
$perms = $perms->apply(fn($calc) => $calc & (Perm::F_CATEGORY_LIST | Perm::F_CATEGORY_VIEW));
2023-08-28 01:17:34 +00:00
2023-08-30 22:37:21 +00:00
$canDeleteAny = $perms->check(Perm::F_POST_DELETE_ANY);
2023-08-28 01:17:34 +00:00
}
if($topicIsNuked || $topicIsDeleted) {
if($forumTopicRedirects->hasTopicRedirect($topicId)) {
$topicRedirectInfo = $forumTopicRedirects->getTopicRedirect($topicId);
Template::set('topic_redir_info', $topicRedirectInfo);
if($topicIsNuked || !$canDeleteAny) {
header('Location: ' . $topicRedirectInfo->getLinkTarget());
return;
}
}
if(empty($topicRedirectInfo))
Template::throwError(404);
2022-09-13 13:14:49 +00:00
}
if(!$perms->check(Perm::F_CATEGORY_VIEW))
Template::throwError(403);
2022-09-13 13:14:49 +00:00
2023-08-28 01:17:34 +00:00
// Maximum amount of posts a topic may contain to still be deletable by the author
// this should be in the config
$deletePostThreshold = 1;
$categoryInfo = $forumCategories->getCategory(topicInfo: $topicInfo);
2023-08-28 01:17:34 +00:00
$topicIsLocked = $topicInfo->isLocked();
$topicIsArchived = $categoryInfo->isArchived();
$topicPostsTotal = $topicInfo->getTotalPostsCount();
2022-09-13 13:14:49 +00:00
$topicIsFrozen = $topicIsArchived || $topicIsDeleted;
2023-08-30 22:37:21 +00:00
$canDeleteOwn = !$topicIsFrozen && !$topicIsLocked && $perms->check(Perm::F_POST_DELETE_OWN);
$canBumpTopic = !$topicIsFrozen && $perms->check(Perm::F_TOPIC_BUMP);
$canLockTopic = !$topicIsFrozen && $perms->check(Perm::F_TOPIC_LOCK);
2022-09-13 13:14:49 +00:00
$canNukeOrRestore = $canDeleteAny && $topicIsDeleted;
$canDelete = !$topicIsDeleted && (
$canDeleteAny || (
$topicPostsTotal > 0
2023-08-28 01:17:34 +00:00
&& $topicPostsTotal <= $deletePostThreshold
2022-09-13 13:14:49 +00:00
&& $canDeleteOwn
2023-08-28 01:17:34 +00:00
&& $topicInfo->getUserId() === (string)$currentUserId
2022-09-13 13:14:49 +00:00
)
);
$validModerationModes = [
'delete', 'restore', 'nuke',
'bump', 'lock', 'unlock',
];
if(in_array($moderationMode, $validModerationModes, true)) {
if(!CSRF::validateRequest())
Template::displayInfo("Couldn't verify this request, please refresh the page and try again.", 403);
2022-09-13 13:14:49 +00:00
$authInfo = $authInfo;
if(!$authInfo->isLoggedIn())
Template::displayInfo('You must be logged in to manage posts.', 401);
2022-09-13 13:14:49 +00:00
if($usersCtx->hasActiveBan($currentUser))
Template::displayInfo('You have been banned, check your profile for more information.', 403);
2022-09-13 13:14:49 +00:00
switch($moderationMode) {
case 'delete':
2023-08-28 01:17:34 +00:00
if($canDeleteAny) {
if($topicInfo->isDeleted())
Template::displayInfo('This topic has already been marked as deleted.', 404);
2023-08-28 01:17:34 +00:00
} else {
if($topicInfo->isDeleted())
Template::throwError(404);
2023-08-28 01:17:34 +00:00
if(!$canDeleteOwn)
Template::displayInfo("You aren't allowed to delete topics.", 403);
2023-08-28 01:17:34 +00:00
if($topicInfo->getUserId() !== $currentUser->getId())
Template::displayInfo('You can only delete your own topics.', 403);
2023-08-28 01:17:34 +00:00
// topics may only be deleted within a day of creation, this should be a config value
$deleteTimeFrame = 60 * 60 * 24;
if($topicInfo->getCreatedTime() < time() - $deleteTimeFrame)
Template::displayInfo('This topic has existed for too long. Ask a moderator to remove if it absolutely necessary.', 403);
2023-08-28 01:17:34 +00:00
// deleted posts are intentionally included
$topicPostCount = $forumPosts->countPosts(topicInfo: $topicInfo);
if($topicPostCount > $deletePostThreshold)
Template::displayInfo('This topic already has replies, you may no longer delete it. Ask a moderator to remove if it absolutely necessary.', 403);
2022-09-13 13:14:49 +00:00
}
2023-01-02 23:12:23 +00:00
if(!isset($_GET['confirm'])) {
Template::render('forum.confirm', [
'title' => 'Confirm topic deletion',
'class' => 'far fa-trash-alt',
2023-08-28 01:17:34 +00:00
'message' => sprintf('You are about to delete topic #%d. Are you sure about that?', $topicInfo->getId()),
2023-01-02 23:12:23 +00:00
'params' => [
2023-08-28 01:17:34 +00:00
't' => $topicInfo->getId(),
2023-01-02 23:12:23 +00:00
'm' => 'delete',
],
]);
break;
} elseif(!$submissionConfirmed) {
2023-09-08 20:40:48 +00:00
Tools::redirect($urls->format(
2023-01-02 23:12:23 +00:00
'forum-topic',
2023-08-28 01:17:34 +00:00
['topic' => $topicInfo->getId()]
2023-09-08 20:40:48 +00:00
));
2023-01-02 23:12:23 +00:00
break;
2022-09-13 13:14:49 +00:00
}
$forumTopics->deleteTopic($topicInfo->getId());
2023-08-28 01:17:34 +00:00
$msz->createAuditLog('FORUM_TOPIC_DELETE', [$topicInfo->getId()]);
2022-09-13 13:14:49 +00:00
2023-09-08 20:40:48 +00:00
Tools::redirect($urls->format('forum-category', [
2023-08-28 01:17:34 +00:00
'forum' => $categoryInfo->getId(),
2023-09-08 20:40:48 +00:00
]));
2022-09-13 13:14:49 +00:00
break;
case 'restore':
if(!$canNukeOrRestore)
Template::throwError(403);
2022-09-13 13:14:49 +00:00
2023-01-02 23:12:23 +00:00
if(!isset($_GET['confirm'])) {
Template::render('forum.confirm', [
'title' => 'Confirm topic restore',
'class' => 'fas fa-magic',
2023-08-28 01:17:34 +00:00
'message' => sprintf('You are about to restore topic #%d. Are you sure about that?', $topicInfo->getId()),
2023-01-02 23:12:23 +00:00
'params' => [
2023-08-28 01:17:34 +00:00
't' => $topicInfo->getId(),
2023-01-02 23:12:23 +00:00
'm' => 'restore',
],
]);
break;
} elseif(!$submissionConfirmed) {
2023-09-08 20:40:48 +00:00
Tools::redirect($urls->format('forum-topic', [
2023-08-28 01:17:34 +00:00
'topic' => $topicInfo->getId(),
2023-09-08 20:40:48 +00:00
]));
2023-01-02 23:12:23 +00:00
break;
2022-09-13 13:14:49 +00:00
}
$forumTopics->restoreTopic($topicInfo->getId());
2023-08-28 01:17:34 +00:00
$msz->createAuditLog('FORUM_TOPIC_RESTORE', [$topicInfo->getId()]);
2022-09-13 13:14:49 +00:00
2023-09-08 20:40:48 +00:00
Tools::redirect($urls->format('forum-category', [
2023-08-28 01:17:34 +00:00
'forum' => $categoryInfo->getId(),
2023-09-08 20:40:48 +00:00
]));
2022-09-13 13:14:49 +00:00
break;
case 'nuke':
if(!$canNukeOrRestore)
Template::throwError(403);
2022-09-13 13:14:49 +00:00
2023-01-02 23:12:23 +00:00
if(!isset($_GET['confirm'])) {
Template::render('forum.confirm', [
'title' => 'Confirm topic nuke',
'class' => 'fas fa-radiation',
2023-08-28 01:17:34 +00:00
'message' => sprintf('You are about to PERMANENTLY DELETE topic #%d. Are you sure about that?', $topicInfo->getId()),
2023-01-02 23:12:23 +00:00
'params' => [
2023-08-28 01:17:34 +00:00
't' => $topicInfo->getId(),
2023-01-02 23:12:23 +00:00
'm' => 'nuke',
],
]);
break;
} elseif(!$submissionConfirmed) {
2023-09-08 20:40:48 +00:00
Tools::redirect($urls->format('forum-topic', [
2023-08-28 01:17:34 +00:00
'topic' => $topicInfo->getId(),
2023-09-08 20:40:48 +00:00
]));
2023-01-02 23:12:23 +00:00
break;
2022-09-13 13:14:49 +00:00
}
$forumTopics->nukeTopic($topicInfo->getId());
2023-08-28 01:17:34 +00:00
$msz->createAuditLog('FORUM_TOPIC_NUKE', [$topicInfo->getId()]);
2022-09-13 13:14:49 +00:00
2023-09-08 20:40:48 +00:00
Tools::redirect($urls->format('forum-category', [
2023-08-28 01:17:34 +00:00
'forum' => $categoryInfo->getId(),
2023-09-08 20:40:48 +00:00
]));
2022-09-13 13:14:49 +00:00
break;
case 'bump':
2023-08-28 01:17:34 +00:00
if($canBumpTopic) {
$forumTopics->bumpTopic($topicInfo->getId());
2023-08-28 01:17:34 +00:00
$msz->createAuditLog('FORUM_TOPIC_BUMP', [$topicInfo->getId()]);
2022-09-13 13:14:49 +00:00
}
2023-09-08 20:40:48 +00:00
Tools::redirect($urls->format('forum-topic', [
2023-08-28 01:17:34 +00:00
'topic' => $topicInfo->getId(),
2023-09-08 20:40:48 +00:00
]));
2022-09-13 13:14:49 +00:00
break;
case 'lock':
2023-08-28 01:17:34 +00:00
if($canLockTopic && !$topicIsLocked) {
$forumTopics->lockTopic($topicInfo->getId());
2023-08-28 01:17:34 +00:00
$msz->createAuditLog('FORUM_TOPIC_LOCK', [$topicInfo->getId()]);
2022-09-13 13:14:49 +00:00
}
2023-09-08 20:40:48 +00:00
Tools::redirect($urls->format('forum-topic', [
2023-08-28 01:17:34 +00:00
'topic' => $topicInfo->getId(),
2023-09-08 20:40:48 +00:00
]));
2022-09-13 13:14:49 +00:00
break;
case 'unlock':
2023-08-28 01:17:34 +00:00
if($canLockTopic && $topicIsLocked) {
$forumTopics->unlockTopic($topicInfo->getId());
2023-08-28 01:17:34 +00:00
$msz->createAuditLog('FORUM_TOPIC_UNLOCK', [$topicInfo->getId()]);
2022-09-13 13:14:49 +00:00
}
2023-09-08 20:40:48 +00:00
Tools::redirect($urls->format('forum-topic', [
2023-08-28 01:17:34 +00:00
'topic' => $topicInfo->getId(),
2023-09-08 20:40:48 +00:00
]));
2022-09-13 13:14:49 +00:00
break;
}
return;
}
2023-08-28 01:17:34 +00:00
$topicPosts = $topicInfo->getPostsCount();
if($canDeleteAny)
$topicPosts += $topicInfo->getDeletedPostsCount();
2022-09-13 13:14:49 +00:00
2023-08-28 01:17:34 +00:00
$topicPagination = new Pagination($topicPosts, 10, 'page');
2022-09-13 13:14:49 +00:00
2023-08-28 01:17:34 +00:00
if(isset($preceedingPostCount))
$topicPagination->setPage(floor($preceedingPostCount / $topicPagination->getRange()), true);
2022-09-13 13:14:49 +00:00
if(!$topicPagination->hasValidOffset())
Template::throwError(404);
2022-09-13 13:14:49 +00:00
$postInfos = $forumPosts->getPosts(
2023-08-28 01:17:34 +00:00
topicInfo: $topicInfo,
2023-08-30 22:37:21 +00:00
deleted: $perms->check(Perm::F_POST_DELETE_ANY) ? null : false,
2023-08-28 01:17:34 +00:00
pagination: $topicPagination,
2022-09-13 13:14:49 +00:00
);
if(empty($postInfos))
Template::throwError(404);
2022-09-13 13:14:49 +00:00
try {
$originalPostInfo = $forumPosts->getPost(topicInfo: $topicInfo);
} catch(RuntimeException $ex) {
Template::throwError(404);
}
2023-08-28 01:17:34 +00:00
$posts = [];
foreach($postInfos as $postInfo) {
$posts[] = $post = new stdClass;
$post->info = $postInfo;
if($postInfo->hasUserId()) {
$post->user = $usersCtx->getUserInfo($postInfo->getUserId());
$post->colour = $usersCtx->getUserColour($post->user);
$post->postsCount = $forumCtx->countTotalUserPosts($post->user);
2023-08-28 01:17:34 +00:00
}
$post->isOriginalPost = $originalPostInfo->getId() == $postInfo->getId();
$post->isOriginalPoster = $originalPostInfo->hasUserId() && $postInfo->hasUserId()
&& $originalPostInfo->getUserId() === $postInfo->getUserId();
}
2023-08-30 22:37:21 +00:00
$canReply = !$topicIsArchived && !$topicIsLocked && !$topicIsDeleted && $perms->check(Perm::F_POST_CREATE);
2022-09-13 13:14:49 +00:00
if(!$forumTopics->checkUserHasReadTopic($currentUser, $topicInfo))
$forumTopics->incrementTopicViews($topicInfo);
2023-08-28 01:17:34 +00:00
$forumTopics->updateUserReadTopic($currentUser, $topicInfo);
2023-08-28 01:17:34 +00:00
2023-08-30 22:37:21 +00:00
$perms = $perms->checkMany([
'can_create_post' => Perm::F_POST_CREATE,
'can_edit_post' => Perm::F_POST_EDIT_OWN,
'can_edit_any_post' => Perm::F_POST_EDIT_ANY,
'can_delete_post' => Perm::F_POST_DELETE_OWN,
'can_delete_any_post' => Perm::F_POST_DELETE_ANY,
2023-08-28 01:17:34 +00:00
]);
2022-09-13 13:14:49 +00:00
Template::render('forum.topic', [
2024-02-08 15:20:44 +00:00
'topic_breadcrumbs' => iterator_to_array($forumCategories->getCategoryAncestry($topicInfo)),
'global_accent_colour' => $forumCategories->getCategoryColour($topicInfo),
2023-08-28 01:17:34 +00:00
'topic_info' => $topicInfo,
'category_info' => $categoryInfo,
2022-09-13 13:14:49 +00:00
'topic_posts' => $posts,
'can_reply' => $canReply,
'topic_pagination' => $topicPagination,
'topic_can_delete' => $canDelete,
'topic_can_nuke_or_restore' => $canNukeOrRestore,
'topic_can_bump' => $canBumpTopic,
'topic_can_lock' => $canLockTopic,
2023-08-28 01:17:34 +00:00
'topic_user_id' => $currentUserId,
'topic_perms' => $perms,
2022-09-13 13:14:49 +00:00
]);