19 declare(strict_types=1);
46 private string $replQuote1 =
'<blockquote class="ilForumQuote">';
62 $this->error = $DIC[
'ilErr'];
63 $this->
lng = $DIC->language();
64 $this->db = $DIC->database();
65 $this->
user = $DIC->user();
67 $this->
event = $DIC->event();
72 $this->
id = $a_obj_id;
77 $this->ref_id = $a_ref_id;
92 if ($dbTable ===
'') {
93 die($this->className .
'::setDbTable(): No database table given.');
106 $this->mdb2Query = $query_string;
107 $this->mdb2DataValue = $data_value;
108 $this->mdb2DataType = $data_type;
115 return $this->mdb2Query ?:
'';
120 return $this->mdb2DataValue ?: [];
125 return $this->mdb2DataType ?: [];
148 $query =
'SELECT * FROM frm_data WHERE ';
157 $res = $this->db->queryF($query, $data_type, $data_value);
158 $row = $this->db->fetchAssoc(
$res);
160 if (!is_array($row) || $row === []) {
172 $query =
'SELECT * FROM frm_threads WHERE ';
181 $sql_res = $this->db->queryF($query, $data_type, $data_value);
182 $result = $this->db->fetchAssoc($sql_res);
183 $result[
'thr_subject'] = trim((
string) ($result[
'thr_subject'] ??
''));
186 $thread_obj->assignData($result);
191 public function generatePost(
195 int $display_user_id,
199 string $subject =
'',
200 ?
string $alias =
null,
203 bool $send_activation_mail =
false 206 $objNewPost->setForumId($forum_id);
207 $objNewPost->setThreadId($thread_id);
208 $objNewPost->setSubject($subject);
209 $objNewPost->setMessage($message);
210 $objNewPost->setDisplayUserId($display_user_id);
211 $objNewPost->setUserAlias($alias);
212 $objNewPost->setPosAuthorId($author_id);
216 $is_moderator =
false;
217 if ($frm_settings->getMarkModeratorPosts() && self::_isModerator($this->
getForumRefId(), $author_id)) {
218 $is_moderator =
true;
220 $objNewPost->setIsAuthorModerator($is_moderator);
223 $objNewPost->setCreateDate(date(
'Y-m-d H:i:s'));
224 } elseif (strpos($date,
'-') > 0) {
225 $objNewPost->setCreateDate($date);
229 $objNewPost->setPostActivationDate($objNewPost->getCreateDate());
233 $objNewPost->setNotification($notify);
234 $objNewPost->setStatus($status);
235 $objNewPost->insert();
237 if ($parent_pos === 0) {
238 $this->
addPostTree($objNewPost->getThreadId(), $objNewPost->getId(), $objNewPost->getCreateDate());
241 $objNewPost->getId(),
243 $objNewPost->getThreadId(),
244 $objNewPost->getCreateDate()
248 $lastPost = $objNewPost->getForumId() .
'#' . $objNewPost->getThreadId() .
'#' . $objNewPost->getId();
250 $this->db->manipulateF(
251 'UPDATE frm_threads SET thr_num_posts = thr_num_posts + 1, thr_last_post = %s WHERE thr_pk = %s',
253 [$lastPost, $objNewPost->getThreadId()]
256 $this->db->manipulateF(
257 'UPDATE frm_data SET top_num_posts = top_num_posts + 1, top_last_post = %s WHERE top_pk = %s',
259 [$lastPost, $objNewPost->getForumId()]
264 $forum_obj->markPostRead($objNewPost->getPosAuthorId(), $objNewPost->getThreadId(), $objNewPost->getId());
266 if ($status && $parent_pos > 0) {
268 $news_item->setContext($forum_obj->getId(),
'frm', $objNewPost->getId(),
'pos');
270 $news_item->setTitle($objNewPost->getSubject());
275 if ($objNewPost->getMessage() !== strip_tags($objNewPost->getMessage())) {
276 $news_item->setContentHtml(
true);
279 $news_item->setUserId($display_user_id);
281 $news_item->create();
284 return $objNewPost->getId();
293 bool $withFirstVisibleEntry =
true 306 $this->db->manipulateF(
307 'UPDATE frm_data SET top_num_threads = top_num_threads + 1 WHERE top_pk = %s',
312 $rootNodeId = $this->generatePost(
327 if (!$withFirstVisibleEntry) {
331 return $this->generatePost(
355 if ($target_obj_id <= 0 || $src_forum->
getId() <= 0) {
356 return $errorMessages;
365 if (!$oldFrmData->getTopPk() || !$newFrmData->getTopPk()) {
366 return $errorMessages;
369 $num_moved_posts = 0;
370 $num_moved_threads = 0;
373 foreach ($thread_ids as $id) {
377 $numPosts = $objTmpThread->movePosts(
379 $oldFrmData->getTopPk(),
381 $newFrmData->getTopPk()
384 if (($last_post_string = ($objTmpThread->getLastPostString() ??
'')) !==
'') {
385 $last_post_string = explode(
'#', $last_post_string);
386 $last_post_string[0] = $newFrmData->getTopPk();
387 $last_post_string = implode(
'#', $last_post_string);
388 $objTmpThread->setLastPostString($last_post_string);
391 $num_visits += $objTmpThread->getVisits();
392 $num_moved_posts += $numPosts;
393 ++$num_moved_threads;
395 $objTmpThread->setForumId($newFrmData->getTopPk());
396 $objTmpThread->update();
398 $errorMessages[] = sprintf($this->
lng->txt(
'frm_move_invalid_file_type'), $objTmpThread->getSubject());
403 if (0 === max($num_moved_threads, $num_moved_posts, $num_visits)) {
404 return $errorMessages;
407 $this->db->setLimit(1, 0);
408 $res = $this->db->queryF(
409 'SELECT pos_thr_fk, pos_pk FROM frm_posts WHERE pos_top_fk = %s ORDER BY pos_date DESC',
411 [$oldFrmData->getTopPk()]
415 $row = $this->db->fetchObject(
$res);
417 $last_post_src = $oldFrmData->getTopPk() .
'#' . $row->pos_thr_fk .
'#' . $row->pos_pk;
420 $this->db->manipulateF(
422 'SET top_num_posts = top_num_posts - %s, top_num_threads = top_num_threads - %s, visits = visits - %s, ' .
423 'top_last_post = %s WHERE top_pk = %s',
424 [
'integer',
'integer',
'integer',
'text',
'integer'],
430 $oldFrmData->getTopPk()
434 $this->db->setLimit(1, 0);
435 $res = $this->db->queryF(
436 'SELECT pos_thr_fk, pos_pk FROM frm_posts WHERE pos_top_fk = %s ORDER BY pos_date DESC',
438 [$newFrmData->getTopPk()]
441 $last_post_dest =
'';
442 $row = $this->db->fetchObject(
$res);
444 $last_post_dest = $newFrmData->getTopPk() .
'#' . $row->pos_thr_fk .
'#' . $row->pos_pk;
447 $this->db->manipulateF(
448 'UPDATE frm_data SET top_num_posts = top_num_posts + %s, top_num_threads = top_num_threads + %s, ' .
449 'visits = visits + %s, top_last_post = %s WHERE top_pk = %s',
450 [
'integer',
'integer',
'integer',
'text',
'integer'],
451 [$num_moved_posts, $num_moved_threads, $num_visits, $last_post_dest, $newFrmData->getTopPk()]
455 'components/ILIAS/Forum',
458 'source_ref_id' => $src_forum->
getId(),
459 'target_ref_id' => $src_forum->
getId(),
460 'thread_ids' => $src_forum->
getId(),
461 'source_frm_obj_id' => $src_forum->
getId(),
462 'target_frm_obj_id' => $target_obj_id
466 return $errorMessages;
471 $cens_date = date(
'Y-m-d H:i:s');
473 $this->db->manipulateF(
475 SET pos_cens_com = %s, 480 [
'text',
'timestamp',
'integer',
'integer',
'integer'],
484 $this->
user->getId(),
498 $news_item->setContent(nl2br($this->
prepareText($message, 0)));
499 $news_item->setContentHtml(
false);
500 if ($message !== strip_tags($message)) {
501 $news_item->setContentHtml(
true);
504 $res = $this->db->queryF(
'SELECT pos_message FROM frm_posts WHERE pos_pk = %s', [
'integer'], [$pos_pk]);
505 $rec = $this->db->fetchAssoc(
$res);
508 $news_item->setContent(nl2br($this->
prepareText($rec[
'pos_message'], 0)));
509 $news_item->setContentHtml(
false);
510 if ($rec[
'pos_message'] !== strip_tags($rec[
'pos_message'])) {
511 $news_item->setContentHtml(
true);
514 $news_item->update();
518 'components/ILIAS/Forum',
533 $p_node = $postIdOrRecord;
534 if (is_numeric($postIdOrRecord)) {
540 $is_deleted_thread = (
$post->getParentId() == 0) ?
true :
false;
541 $num_visible_active_posts = 0;
542 if ($is_deleted_thread) {
544 SELECT COUNT(*) AS cnt 546 INNER JOIN frm_posts_tree ON pos_pk = pos_fk 547 WHERE frm_posts_tree.parent_pos != 0 548 AND pos_thr_fk = ' . $this->db->quote(
$post->getThreadId(),
'integer') .
' 549 AND pos_status = ' . $this->db->quote(1,
'integer');
550 $res = $this->db->query($query);
551 $row = $this->db->fetchAssoc(
$res);
552 $num_visible_active_posts = (
int) ($row[
'cnt'] ?? 0);
556 'components/ILIAS/Forum',
557 'beforePostDeletion',
562 'thread_deleted' => $is_deleted_thread,
563 'num_visible_active_posts' => $num_visible_active_posts
573 $affected_user_ids[] =
$post->getPosAuthorId();
577 $obj_history->deleteHistoryByPostIds($deleted_post_ids);
580 $obj_draft->deleteDraftsByPostIds($deleted_post_ids);
582 foreach ($deleted_post_ids as $post_id) {
586 $dead_pos = count($deleted_post_ids);
589 if ((
int)
$post->getParentId() === 0) {
590 $dead_thr =
$post->getThreadId();
592 $this->db->manipulateF(
'DELETE FROM frm_threads WHERE thr_pk = %s', [
'integer'], [$dead_thr]);
593 $this->db->manipulateF(
594 'UPDATE frm_data SET top_num_threads = top_num_threads - 1 WHERE top_frm_fk = %s',
599 $posset = $this->db->queryF(
'SELECT * FROM frm_posts WHERE pos_thr_fk = %s', [
'integer'], [$dead_thr]);
600 while ($posrec = $this->db->fetchAssoc($posset)) {
604 (
int) $posrec[
'pos_pk'],
609 $news_item->delete();
614 foreach ($mobs as $mob) {
623 $affected_user_ids[] = (
int) $posrec[
'pos_author_id'];
626 $this->db->manipulateF(
'DELETE FROM frm_posts WHERE pos_thr_fk = %s', [
'integer'], [
$post->getThreadId()]);
628 for ($i = 0; $i < $dead_pos; $i++) {
629 $this->db->manipulateF(
'DELETE FROM frm_posts WHERE pos_pk = %s', [
'integer'], [$deleted_post_ids[$i]]);
634 $deleted_post_ids[$i],
639 $news_item->delete();
644 foreach ($mobs as $mob) {
655 $this->db->manipulateF(
656 'UPDATE frm_threads SET thr_num_posts = thr_num_posts - %s WHERE thr_pk = %s',
657 [
'integer',
'integer'],
658 [$dead_pos,
$post->getTreeId()]
661 $res1 = $this->db->queryF(
662 'SELECT * FROM frm_posts WHERE pos_thr_fk = %s ORDER BY pos_date DESC',
668 if ($res1->numRows() > 0) {
671 while ($selData = $this->db->fetchAssoc($res1)) {
676 $lastPost_thr = $selData[
'pos_top_fk'] .
'#' . $selData[
'pos_thr_fk'] .
'#' . $selData[
'pos_pk'];
681 $this->db->manipulateF(
682 'UPDATE frm_threads SET thr_last_post = %s WHERE thr_pk = %s',
684 [$lastPost_thr,
$post->getTreeId()]
688 $this->db->manipulateF(
689 'UPDATE frm_data SET top_num_posts = top_num_posts - %s WHERE top_frm_fk = %s',
690 [
'integer',
'integer'],
691 [$dead_pos, $this->
id]
694 $res2 = $this->db->queryF(
695 'SELECT * FROM frm_posts, frm_data WHERE pos_top_fk = top_pk AND top_frm_fk = %s ORDER BY pos_date DESC',
701 if ($res2->numRows() > 0) {
704 while ($selData = $this->db->fetchAssoc($res2)) {
709 $lastPost_top = $selData[
'pos_top_fk'] .
'#' . $selData[
'pos_thr_fk'] .
'#' . $selData[
'pos_pk'];
714 $this->db->manipulateF(
715 'UPDATE frm_data SET top_last_post = %s WHERE top_frm_fk = %s',
717 [$lastPost_top, $this->
id]
722 'components/ILIAS/Forum',
727 'user_ids' => $affected_user_ids
741 $is_post_activation_enabled = $frm_props->isPostActivationEnabled();
745 $excluded_ids_condition =
'';
746 if (isset(
$params[
'excluded_ids']) && is_array(
$params[
'excluded_ids']) &&
$params[
'excluded_ids']) {
747 $excluded_ids_condition =
' AND ' . $this->db->in(
'thr_pk',
$params[
'excluded_ids'],
true,
'integer') .
' ';
750 if (!isset(
$params[
'order_column']) || !in_array(
751 strtolower(
$params[
'order_column']),
752 [
'lp_date',
'rating',
'thr_subject',
'num_posts',
'num_visit']
754 $params[
'order_column'] =
'post_date';
756 if (!isset(
$params[
'order_direction']) || !in_array(strtolower(
$params[
'order_direction']), [
'asc',
'desc'])) {
757 $params[
'order_direction'] =
'desc';
760 $cnt_active_pos_query =
'';
761 $cnt_join_type =
'LEFT';
762 if ($is_post_activation_enabled && !
$params[
'is_moderator']) {
763 $cnt_active_pos_query =
" AND (pos_status = {$this->db->quote(1, 'integer')} OR pos_author_id = {$this->db->quote($user_id, 'integer')}) ";
764 $cnt_join_type =
"INNER";
767 "SELECT COUNT(DISTINCT(thr_pk)) cnt 769 $cnt_join_type JOIN frm_posts 770 ON pos_thr_fk = thr_pk $cnt_active_pos_query 771 WHERE thr_top_fk = %s $excluded_ids_condition 773 $res = $this->db->queryF($query, [
'integer'], [$a_topic_id]);
774 $cntData = $this->db->fetchAssoc(
$res);
775 $cnt = (
int) $cntData[
'cnt'];
779 if ($is_post_activation_enabled && !
$params[
'is_moderator']) {
780 $active_query =
' AND (pos_status = %s OR pos_author_id = %s) ';
781 $having =
' HAVING num_posts > 0';
788 $optional_fields =
'';
789 if ($frm_props->isIsThreadRatingEnabled()) {
790 $optional_fields =
', avg_rating';
793 $additional_sort =
'';
795 if (
$params[
'order_column'] ===
'thr_subject') {
796 $dynamic_columns = [
', thr_subject ' .
$params[
'order_direction']];
797 } elseif (
$params[
'order_column'] ===
'num_posts') {
798 $dynamic_columns = [
', num_posts ' .
$params[
'order_direction']];
799 } elseif (
$params[
'order_column'] ===
'num_visit') {
800 $dynamic_columns = [
', visits ' .
$params[
'order_direction']];
802 $dynamic_columns = [
', post_date ' .
$params[
'order_direction']];
805 if ($frm_props->isIsThreadRatingEnabled()) {
806 $dynamic_columns[] =
' ,avg_rating ' .
$params[
'order_direction'];
808 if (
'rating' === strtolower(
$params[
'order_column'])) {
809 $dynamic_columns = array_reverse($dynamic_columns);
811 $additional_sort .= implode(
' ', $dynamic_columns);
814 if (!$this->
user->isAnonymous()) {
816 (CASE WHEN COUNT(DISTINCT(notification_id)) > 0 THEN 1 ELSE 0 END) usr_notification_is_enabled, 817 MAX(pos_date) post_date, 818 SUM(tree1.parent_pos != 0) num_posts, 819 SUM(tree1.parent_pos != 0) - SUM(tree1.parent_pos != 0 AND postread.post_id IS NOT NULL) num_unread_posts, ";
821 $query .=
" thr_pk, thr_top_fk, thr_subject, thr_author_id, thr_display_user_id, thr_usr_alias, thr_num_posts, thr_last_post, thr_date, thr_update, visits, frm_threads.import_name, is_sticky, is_closed 825 LEFT JOIN frm_notification 826 ON frm_notification.thread_id = thr_pk 827 AND frm_notification.user_id = %s 830 ON pos_thr_fk = thr_pk $active_query 831 LEFT JOIN frm_posts_tree tree1 832 ON tree1.pos_fk = frm_posts.pos_pk 833 LEFT JOIN frm_user_read postread 834 ON postread.post_id = pos_pk 835 AND postread.usr_id = %s";
837 $query .=
" WHERE thr_top_fk = %s 838 $excluded_ids_condition 839 GROUP BY thr_pk, thr_top_fk, thr_subject, thr_author_id, thr_display_user_id, thr_usr_alias, thr_num_posts, thr_last_post, thr_date, thr_update, visits, frm_threads.import_name, is_sticky, is_closed 842 ORDER BY is_sticky DESC $additional_sort, thr_date DESC";
845 $data_types[] =
'integer';
846 if ($is_post_activation_enabled && !
$params[
'is_moderator']) {
847 $data_types[] =
'integer';
848 $data_types[] =
'integer';
850 $data_types[] =
'integer';
851 $data_types[] =
'integer';
855 if ($is_post_activation_enabled && !
$params[
'is_moderator']) {
862 0 usr_notification_is_enabled, 863 MAX(pos_date) post_date, 864 COUNT(DISTINCT(tree1.pos_fk)) num_posts, 865 COUNT(DISTINCT(tree1.pos_fk)) num_unread_posts, 866 thr_pk, thr_top_fk, thr_subject, thr_author_id, thr_display_user_id, thr_usr_alias, thr_num_posts, thr_last_post, thr_date, thr_update, visits, frm_threads.import_name, is_sticky, is_closed 871 ON pos_thr_fk = thr_pk $active_query 872 LEFT JOIN frm_posts_tree tree1 873 ON tree1.pos_fk = frm_posts.pos_pk AND tree1.parent_pos != 0 876 $query .=
" WHERE thr_top_fk = %s 877 $excluded_ids_condition 878 GROUP BY thr_pk, thr_top_fk, thr_subject, thr_author_id, thr_display_user_id, thr_usr_alias, thr_num_posts, thr_last_post, thr_date, thr_update, visits, frm_threads.import_name, is_sticky, is_closed 881 ORDER BY is_sticky DESC $additional_sort, thr_date DESC";
883 if ($is_post_activation_enabled && !
$params[
'is_moderator']) {
884 $data_types[] =
'integer';
885 $data_types[] =
'integer';
887 $data_types[] =
'integer';
888 if ($is_post_activation_enabled && !
$params[
'is_moderator']) {
893 $data[] = $a_topic_id;
895 if ($limit || $offset) {
896 $this->db->setLimit($limit, $offset);
900 $res = $this->db->queryF($query, $data_types,
$data);
901 while ($row = $this->db->fetchAssoc(
$res)) {
903 $thread->assignData($row);
904 $threads[(
int) $row[
'thr_pk']] = $thread;
905 $threadIds[] = (
int) $row[
'thr_pk'];
908 $inner_last_active_post_condition =
'';
909 if ($is_post_activation_enabled && !
$params[
'is_moderator']) {
910 $inner_last_active_post_condition = sprintf(
911 ' AND (iposts.pos_status = %s OR (iposts.pos_status = %s AND iposts.pos_author_id = %s)) ',
912 $this->db->quote(1,
'integer'),
913 $this->db->quote(0,
'integer'),
914 $this->db->quote($this->
user->getId(),
'integer')
918 $post_res = $this->db->query(
922 SELECT pos_thr_fk, MAX(iposts.pos_date) i_pos_date 923 FROM frm_posts iposts 924 WHERE ' . $this->db->in(
'iposts.pos_thr_fk', $threadIds,
false,
'integer') .
' 925 ' . $inner_last_active_post_condition .
' 927 ) opost ON frm_posts.pos_thr_fk = opost.pos_thr_fk AND frm_posts.pos_date = opost.i_pos_date' 929 while ($post_row = $this->db->fetchAssoc($post_res)) {
930 $tmp_obj =
new ilForumPost((
int) $post_row[
'pos_pk'], (
bool)
$params[
'is_moderator'],
true);
931 $tmp_obj->setPosAuthorId((
int) $post_row[
'pos_author_id']);
932 $tmp_obj->setDisplayUserId((
int) $post_row[
'pos_display_user_id']);
933 $tmp_obj->setUserAlias((
string) $post_row[
'pos_usr_alias']);
934 $tmp_obj->setImportName((
string) $post_row[
'import_name']);
935 $tmp_obj->setId((
int) $post_row[
'pos_pk']);
936 $tmp_obj->setCreateDate((
string) $post_row[
'pos_date']);
938 $threads[(
int) $post_row[
'pos_thr_fk']]->setLastPostForThreadOverview($tmp_obj);
951 SUM(IF(f.pos_cens = %s, 1, 0)) cnt 953 INNER JOIN frm_posts_tree t ON f.pos_pk = t.pos_fk AND t.parent_pos != %s 954 INNER JOIN frm_threads th ON t.thr_fk = th.thr_pk 955 INNER JOIN frm_data d ON d.top_pk = f.pos_top_fk AND d.top_frm_fk = %s 956 WHERE f.pos_author_id = %s 959 if ($post_activation_required) {
960 $query .=
' AND f.pos_status = ' . $this->db->quote(1,
'integer');
963 $res = $this->db->queryF(
965 [
'integer',
'integer',
'integer',
'integer'],
968 $row = $this->db->fetchAssoc(
$res);
969 if (is_array($row)) {
970 return (
int) $row[
'cnt'];
987 u.login, u.lastname, u.firstname, f.pos_author_id, u.usr_id, 988 p.value public_profile, 989 SUM(IF(f.pos_cens = %s, 1, 0)) num_postings 991 INNER JOIN frm_posts_tree t ON f.pos_pk = t.pos_fk 992 INNER JOIN frm_threads th ON t.thr_fk = th.thr_pk 993 INNER JOIN usr_data u ON u.usr_id = f.pos_author_id 994 INNER JOIN frm_data d ON d.top_pk = f.pos_top_fk 995 LEFT JOIN usr_pref p ON p.usr_id = u.usr_id AND p.keyword = %s 996 WHERE t.parent_pos != %s 999 $data_types[] =
'integer';
1000 $data_types[] =
'text';
1001 $data_types[] =
'integer';
1003 $data[] =
'public_profile';
1006 if ($post_activation_required) {
1007 $query .=
' AND pos_status = %s';
1008 $data_types[] =
'integer';
1013 AND d.top_frm_fk = %s 1014 GROUP BY u.login, p.value,u.lastname, u.firstname, f.pos_author_id 1017 $data_types[] =
'integer';
1020 $res = $this->db->queryF($query, $data_types,
$data);
1021 while ($row = $this->db->fetchAssoc(
$res)) {
1023 !in_array($row[
'public_profile'], [
1024 PersonalProfileMode::PROFILE_ENABLED_LOGGED_IN_USERS,
1025 PersonalProfileMode::PROFILE_ENABLED_GLOBAL],
true)
1026 || ($this->
user->isAnonymous() && $row[
'public_profile'] !== PersonalProfileMode::PROFILE_ENABLED_GLOBAL)
1028 $row[
'lastname'] =
'';
1029 $row[
'firstname'] =
'';
1032 $row[
'usr_id'] = (
int) $row[
'usr_id'];
1033 $row[
'pos_author_id'] = (
int) $row[
'pos_author_id'];
1034 $row[
'num_postings'] = (
int) $row[
'num_postings'];
1036 $statistic[] = $row;
1044 $res = $this->db->queryF(
1045 'SELECT pos_fk FROM frm_posts_tree WHERE thr_fk = %s AND parent_pos = %s',
1046 [
'integer',
'integer'],
1050 $row = $this->db->fetchObject(
$res);
1052 return (
int) $row->pos_fk;
1073 $rbacreview = $DIC->rbac()->review();
1075 $role_arr = $rbacreview->getRolesOfRoleFolder($a_ref_id);
1076 foreach ($role_arr as $role_id) {
1078 return array_map(
'intval', $rbacreview->assignedUsers($role_id));
1087 if (!isset(self::$moderators_by_ref_id_map[$a_ref_id])) {
1088 self::$moderators_by_ref_id_map[$a_ref_id] = self::_getModerators($a_ref_id);
1091 return in_array($a_usr_id, self::$moderators_by_ref_id_map[$a_ref_id],
true);
1096 $res = $this->db->queryF(
1097 'SELECT * FROM frm_data 1098 INNER JOIN frm_posts ON pos_top_fk = top_pk 1099 INNER JOIN frm_posts_tree tree1 ON tree1.pos_fk = frm_posts.pos_pk AND tree1.parent_pos != 0 1100 WHERE top_frm_fk = %s 1101 AND pos_author_id = %s',
1102 [
'integer',
'integer'],
1106 return $res->numRows();
1111 $res = $this->db->queryF(
1112 'SELECT * FROM frm_data 1113 INNER JOIN frm_posts ON pos_top_fk = top_pk 1114 INNER JOIN frm_posts_tree tree1 ON tree1.pos_fk = frm_posts.pos_pk AND tree1.parent_pos != 0 1115 WHERE top_frm_fk = %s 1116 AND (pos_status = %s OR (pos_status = %s AND pos_author_id = %s)) 1117 AND pos_author_id = %s',
1118 [
'integer',
'integer',
'integer',
'integer',
'integer'],
1122 return $res->numRows();
1130 public function addPostTree(
int $a_tree_id,
int $a_node_id = -1,
string $a_date =
''): bool
1132 $a_date = $a_date ?: date(
'Y-m-d H:i:s');
1134 if ($a_node_id <= 0) {
1135 $a_node_id = $a_tree_id;
1138 $nextId = $this->db->nextId(
'frm_posts_tree');
1140 $this->db->manipulateF(
1142 INSERT INTO frm_posts_tree 1152 VALUES(%s, %s, %s, %s, %s, %s, %s, %s )',
1153 [
'integer',
'integer',
'integer',
'integer',
'integer',
'integer',
'integer',
'timestamp'],
1154 [$nextId, $a_tree_id, $a_node_id, 0, 1, 2, 1, $a_date]
1163 public function insertPostNode(
int $a_node_id,
int $a_parent_id,
int $tree_id,
string $a_date =
''): void
1165 $a_date = $a_date ?: date(
'Y-m-d H:i:s');
1168 $res = $this->db->queryF(
1169 'SELECT lft FROM frm_posts_tree WHERE pos_fk = %s AND thr_fk = %s',
1170 [
'integer',
'integer'],
1171 [$a_parent_id, $tree_id]
1173 $row = $this->db->fetchObject(
$res);
1175 $left = (
int) $row->lft;
1181 $this->db->manipulateF(
1183 UPDATE frm_posts_tree 1195 [
'integer',
'integer',
'integer'],
1196 [$left, $left, $tree_id]
1199 $depth = $this->
getPostDepth($a_parent_id, $tree_id) + 1;
1201 $nextId = $this->db->nextId(
'frm_posts_tree');
1202 $this->db->manipulateF(
1204 INSERT INTO frm_posts_tree 1214 VALUES(%s,%s,%s, %s, %s, %s,%s, %s)',
1215 [
'integer',
'integer',
'integer',
'integer',
'integer',
'integer',
'integer',
'timestamp'],
1231 if ($tree_id !== 0) {
1232 $res = $this->db->queryF(
1233 'SELECT depth FROM frm_posts_tree WHERE pos_fk = %s AND thr_fk = %s',
1234 [
'integer',
'integer'],
1235 [$a_node_id, $tree_id]
1238 $row = $this->db->fetchObject(
$res);
1240 return (
int) $row->depth;
1249 $res = $this->db->queryF(
1250 'SELECT * FROM frm_posts, frm_posts_tree WHERE pos_pk = pos_fk AND parent_pos = %s AND thr_fk = %s',
1251 [
'integer',
'integer'],
1255 if (($row = $this->db->fetchObject(
$res)) !==
null) {
1264 $res = $this->db->queryF(
1265 'SELECT * FROM frm_posts, frm_posts_tree WHERE pos_pk = pos_fk AND pos_pk = %s',
1270 if (($row = $this->db->fetchObject(
$res)) !==
null) {
1286 $tmp_user =
new ilObjUser((
int) $a_row->pos_display_user_id);
1287 $fullname = $tmp_user->getFullname();
1288 $loginname = $tmp_user->getLogin();
1291 if ($fullname ===
'') {
1292 $fullname = $this->
lng->txt(
'unknown');
1293 if ($a_row->import_name) {
1294 $fullname = $a_row->import_name;
1300 'pos_pk' => (
int) $a_row->pos_pk,
1301 'pos_thr_fk' => (
int) $a_row->pos_thr_fk,
1302 'child' => (
int) $a_row->pos_pk,
1303 'author' => (
int) $a_row->pos_display_user_id,
1304 'alias' => (string) $a_row->pos_usr_alias,
1305 'title' => $fullname,
1306 'loginname' => $loginname,
1307 'message' => (
string) $a_row->pos_message,
1308 'subject' => (string) $a_row->pos_subject,
1309 'pos_cens_com' => (
string) $a_row->pos_cens_com,
1310 'pos_cens' => (
int) $a_row->pos_cens,
1311 'date' => $a_row->fpt_date,
1312 'create_date' => $a_row->pos_date,
1313 'update' => $a_row->pos_update,
1314 'update_user' => (
int) $a_row->update_user,
1315 'tree' => (
int) $a_row->thr_fk,
1316 'parent' => (
int) $a_row->parent_pos,
1317 'lft' => (
int) $a_row->lft,
1318 'rgt' => (
int) $a_row->rgt,
1319 'depth' => (
int) $a_row->depth,
1320 'id' => (
int) $a_row->fpt_pk,
1321 'notify' => (
int) $a_row->notify,
1322 'import_name' => $a_row->import_name,
1323 'pos_status' => (
int) $a_row->pos_status
1332 $res = $this->db->queryF(
1333 'SELECT pos_fk FROM frm_posts_tree WHERE lft BETWEEN %s AND %s AND thr_fk = %s',
1339 while ($post_tree_data = $this->db->fetchAssoc(
$res)) {
1340 $post_ids[] = (
int) $post_tree_data[
'pos_fk'];
1351 $res = $this->db->queryF(
1352 'SELECT lft, rgt FROM frm_posts_tree WHERE thr_fk = %s AND pos_fk = %s AND parent_pos = %s',
1353 [
'integer',
'integer',
'integer'],
1354 [$a_node[
'tree'], $a_node[
'pos_pk'], $a_node[
'parent']]
1357 while ($row = $this->db->fetchObject(
$res)) {
1358 $a_node[
'lft'] = (
int) $row->lft;
1359 $a_node[
'rgt'] = (
int) $row->rgt;
1362 $diff = $a_node[
'rgt'] - $a_node[
'lft'] + 1;
1364 $res = $this->db->queryF(
1365 'SELECT pos_fk FROM frm_posts_tree WHERE lft BETWEEN %s AND %s AND thr_fk = %s',
1366 [
'integer',
'integer',
'integer'],
1367 [$a_node[
'lft'], $a_node[
'rgt'], $a_node[
'tree']]
1370 $deleted_post_ids = [];
1371 while ($treeData = $this->db->fetchAssoc(
$res)) {
1372 $deleted_post_ids[] = (
int) $treeData[
'pos_fk'];
1375 $this->db->manipulateF(
1376 'DELETE FROM frm_posts_tree WHERE lft BETWEEN %s AND %s AND thr_fk = %s',
1377 [
'integer',
'integer',
'integer'],
1378 [$a_node[
'lft'], $a_node[
'rgt'], $a_node[
'tree']]
1381 $this->db->manipulateF(
1383 UPDATE frm_posts_tree 1395 [
'integer',
'integer',
'integer',
'integer',
'integer'],
1396 [$a_node[
'lft'], $diff, $a_node[
'lft'], $diff, $a_node[
'tree']]
1399 return $deleted_post_ids;
1404 $checkTime = time() - (60 * 60);
1405 $session_key =
'frm_visit_' . $this->dbTable .
'_' . $ID;
1409 $query =
'UPDATE ' . $this->dbTable .
' SET visits = visits + 1 WHERE ';
1418 $this->db->manipulateF($query, $data_type, $data_value);
1423 public function prepareText(
string $text,
int $edit = 0,
string $quote_user =
'',
string $type =
''): string
1425 if ($type ===
'export') {
1426 $this->replQuote1 =
"<blockquote class=\"quote\"><hr size=\"1\" color=\"#000000\">";
1427 $this->replQuote2 =
"<hr size=\"1\" color=\"#000000\"/></blockquote>";
1432 if ($quote_user !==
'') {
1433 $lname =
'="' . $quote_user .
'"';
1436 $text =
"[quote$lname]" . $text .
"[/quote]";
1439 $startZ = substr_count($text,
"[quote");
1440 $endZ = substr_count($text,
"[/quote]");
1442 if ($startZ > 0 || $endZ > 0) {
1444 if ($startZ > $endZ) {
1445 $diff = $startZ - $endZ;
1447 for ($i = 0; $i < $diff; $i++) {
1448 if ($type ===
'export') {
1451 $text .=
"[/quote]";
1454 } elseif ($startZ < $endZ) {
1455 $diff = $endZ - $startZ;
1457 for ($i = 0; $i < $diff; $i++) {
1458 if ($type ===
'export') {
1459 $text = $this->txtQuote1 . $text;
1461 $text =
"[quote]" . $text;
1467 $text = preg_replace(
1468 '@\[(quote\s*?=\s*?"([^"]*?)"\s*?)\]@i',
1469 $this->replQuote1 .
'<div class="ilForumQuoteHead">' . $this->
lng->txt(
'quote') .
' ($2)</div>',
1473 $text = str_replace(
1474 [
"[quote]",
"[/quote]"],
1476 $this->replQuote1 .
'<div class="ilForumQuoteHead">' . $this->
lng->txt(
'quote') .
'</div>',
1485 if ($type !==
'export') {
1494 $text = str_replace([
"{",
"}"], [
"{",
"}"], $text);
1505 foreach ($a_ids as $pos_id) {
1507 $files = $forumFiles->getFilesOfPost();
1508 foreach ($files as $file) {
1509 $forumFiles->unlinkFile($file[
'name']);
1521 $this->import_name = $a_import_name;
1528 $res = $this->db->queryF(
1530 SELECT frm_notification.thread_id FROM frm_data, frm_notification, frm_threads 1531 WHERE frm_notification.user_id = %s 1532 AND frm_notification.thread_id = frm_threads.thr_pk 1533 AND frm_threads.thr_top_fk = frm_data.top_pk 1534 AND frm_data.top_frm_fk = %s 1535 GROUP BY frm_notification.thread_id',
1536 [
'integer',
'integer'],
1537 [$user_id, $this->
id]
1540 if (
$res->numRows() > 0) {
1542 $thread_data_types = [];
1544 $query =
' DELETE FROM frm_notification WHERE user_id = %s AND thread_id IN (';
1546 $thread_data_types[] =
'integer';
1549 while ($row = $this->db->fetchAssoc(
$res)) {
1550 if ($counter < $res->numRows()) {
1552 $thread_data[] = $row[
'thread_id'];
1553 $thread_data_types[] =
'integer';
1556 if ($counter ===
$res->numRows()) {
1558 $thread_data[] = $row[
'thread_id'];
1559 $thread_data_types[] =
'integer';
1564 $this->db->manipulateF($query, $thread_data_types, $thread_data);
1568 $nextId = $this->db->nextId(
'frm_notification');
1569 $this->db->manipulateF(
1570 'INSERT INTO frm_notification (notification_id, user_id, frm_id) VALUES(%s, %s, %s)',
1571 [
'integer',
'integer',
'integer'],
1572 [$nextId, $user_id, $this->
id]
1581 $this->db->manipulateF(
1582 'DELETE FROM frm_notification WHERE user_id = %s AND frm_id = %s',
1583 [
'integer',
'integer'],
1584 [$user_id, $this->
id]
1592 $res = $this->db->queryF(
1593 'SELECT COUNT(*) cnt FROM frm_notification WHERE user_id = %s AND frm_id = %s',
1594 [
'integer',
'integer'],
1595 [$user_id, $this->
id]
1598 if ($row = $this->db->fetchAssoc(
$res)) {
1599 return (
int) $row[
'cnt'] > 0;
1607 $res = $this->db->queryF(
1608 'SELECT COUNT(*) cnt FROM frm_notification WHERE user_id = %s AND thread_id = %s',
1609 [
'integer',
'integer'],
1610 [$user_id, $thread_id]
1613 if ($row = $this->db->fetchAssoc(
$res)) {
1614 return (
int) $row[
'cnt'] > 0;
1627 $sort = match ($a_sort_mode) {
1628 self::SORT_DATE =>
'thr_date',
1629 default =>
'thr_subject',
1632 $res = $DIC->database()->queryF(
1633 'SELECT thr_pk, thr_subject FROM frm_threads INNER JOIN frm_data ON top_pk = thr_top_fk WHERE top_frm_fk = %s ORDER BY %s',
1634 [
'integer',
'text'],
1639 while ($row = $DIC->database()->fetchObject(
$res)) {
1640 $threads[(
int) $row->thr_pk] = $row->thr_subject;
1650 $res = $DIC->database()->queryF(
'SELECT top_frm_fk FROM frm_data WHERE top_pk = %s', [
'integer'], [$a_for_id]);
1651 if ($row = $DIC->database()->fetchAssoc(
$res)) {
1652 return (
int) $row[
'top_frm_fk'];
1664 if ($sourceThread->getForumId() !== $targetThread->getForumId()) {
1665 throw new ilException(
'not_allowed_to_merge_into_another_forum');
1669 if ($sourceThread->getCreateDate() > $targetThread->getCreateDate()) {
1670 $sourceThreadForMerge = $sourceThread;
1671 $targetThreadForMerge = $targetThread;
1673 $sourceThreadForMerge = $targetThread;
1674 $targetThreadForMerge = $sourceThread;
1677 $threadSubject = $targetThreadForMerge->getSubject();
1679 $targetWasClosedBeforeMerge = $targetThreadForMerge->isClosed();
1680 $sourceThreadForMerge->close();
1682 if (!$targetWasClosedBeforeMerge) {
1683 $targetThreadForMerge->close();
1686 $allSourcePostings = $sourceThreadForMerge->getAllPostIds();
1687 $sourceThreadRootNode = $sourceThreadForMerge->getPostRootNode();
1688 $targetThreadRootNode = $targetThreadForMerge->getPostRootNode();
1690 $sourceThreadRootArray = $this->
getPostNode($sourceThreadRootNode->getId());
1692 $ilAtomQuery = $this->db->buildAtomQuery();
1693 $ilAtomQuery->addTableLock(
'frm_posts');
1694 $ilAtomQuery->addTableLock(
'frm_posts_tree');
1695 $ilAtomQuery->addTableLock(
'frm_threads');
1696 $ilAtomQuery->addTableLock(
'frm_data');
1699 $targetThreadForMerge,
1700 $sourceThreadForMerge,
1701 $targetThreadRootNode,
1702 $sourceThreadRootNode,
1705 $targetRootNodeRgt = $targetThreadRootNode->getRgt();
1706 $targetRootNodeId = $targetThreadRootNode->getId();
1710 $targetThreadRootNode->getId(),
1711 ($targetThreadRootNode->getRgt() + $sourceThreadRootNode->getRgt() - 2)
1715 foreach ($allSourcePostings as $pos_pk) {
1718 if ($post_obj->getId() === $sourceThreadRootNode->getId()) {
1724 $tree->setPosFk($pos_pk);
1726 if ($post_obj->getParentId() === $sourceThreadRootNode->getId()) {
1727 $tree->setParentPos($targetRootNodeId);
1729 $tree->setParentPos($post_obj->getParentId());
1732 $tree->setLft(($post_obj->getLft() + $targetRootNodeRgt) - 2);
1733 $tree->setRgt(($post_obj->getRgt() + $targetRootNodeRgt) - 2);
1735 $tree->setDepth($post_obj->getDepth());
1736 $tree->setTargetThreadId($targetThreadForMerge->getId());
1737 $tree->setSourceThreadId($sourceThreadForMerge->getId());
1743 $sourceThreadForMerge->getId(),
1744 $targetThreadForMerge->getId(),
1745 [$sourceThreadRootNode->getId()]
1748 $ilAtomQuery->run();
1753 $lastPostString = $targetThreadForMerge->getLastPostString() ??
'';
1754 $exp = explode(
'#', $lastPostString);
1755 if (array_key_exists(2, $exp)) {
1757 $exp[2] = $targetThreadForMerge->getLastPost()->getId();
1758 $lastPostString = implode(
'#', $exp);
1760 $lastPostString =
null;
1765 $frm_topic_obj->setNumPosts($sourceThreadForMerge->getNumPosts() + $targetThreadForMerge->getNumPosts());
1766 $frm_topic_obj->setVisits($sourceThreadForMerge->getVisits() + $targetThreadForMerge->getVisits());
1767 $frm_topic_obj->setLastPostString($lastPostString);
1768 $frm_topic_obj->setSubject($threadSubject);
1769 $frm_topic_obj->setId($targetThreadForMerge->getId());
1770 $frm_topic_obj->updateMergedThread();
1772 if (!$targetWasClosedBeforeMerge) {
1773 $targetThreadForMerge->reopen();
1776 $this->
event->raise(
1777 'components/ILIAS/Forum',
1781 'source_thread_id' => $sourceThreadForMerge->getId(),
1782 'target_thread_id' => $targetThreadForMerge->getId()
1786 $this->
deletePost($sourceThreadRootArray,
false);
static _replaceMediaObjectImageSrc(string $a_text, int $a_direction=0, string $nic='')
Replaces image source from mob image urls with the mob id or replaces mob id with the correct image s...
addPostTree(int $a_tree_id, int $a_node_id=-1, string $a_date='')
getFirstPostNode(int $tree_id)
static get(string $a_var)
isForumNotificationEnabled(int $user_id)
getPostNode(int $post_id)
static mergeForumUserRead(int $merge_source_thread_id, int $merge_target_thread_id)
static array $moderators_by_ref_id_map
Personal profile publishing mode of a user.
getRootPostIdByThread(int $a_thread_id)
setCreateDate(?string $a_createdate)
if(! $DIC->user() ->getId()||!ilLTIConsumerAccess::hasCustomProviderCreationAccess()) $params
setMDB2WhereCondition(string $query_string, array $data_type, array $data_value)
getUserStatistics(bool $post_activation_required)
disableForumNotification(int $user_id)
deletePostFiles(array $a_ids)
enableNotification(int $a_user_id)
getPostDepth(int $a_node_id, int $tree_id)
setDbTable(string $dbTable)
static mergePosts(int $sourceThreadId, int $targetThreadId, array $excludedPostIds=[])
moveThreads(array $thread_ids, ilObjForum $src_forum, int $target_obj_id)
setForumRefId(int $a_ref_id)
convertDate(string $date)
static _deleteReadEntries(int $a_post_id)
Class ilForumDraftHistory.
static getInstance(int $a_obj_id=0)
while($session_entry=$r->fetchRow(ilDBConstants::FETCHMODE_ASSOC)) return null
setImportName(?string $a_import_name)
mergeThreads(int $source_id, int $target_id)
insertPostNode(int $a_node_id, int $a_parent_id, int $tree_id, string $a_date='')
insert node under parent node
getNumberOfPublishedUserPostings(int $usr_id, bool $post_activation_required)
fetchPostNodeData(stdClass $a_row)
static _exists(int $id, bool $reference=false, ?string $type=null)
checks if an object exists in object_data
static getInstanceFromArray(array $record)
static updateTargetRootRgt(int $root_node_id, int $rgt)
static _lookupTitle(int $obj_id)
readonly ilAppEventHandler $event
prepareText(string $text, int $edit=0, string $quote_user='', string $type='')
static _lookupObjIdForForumId(int $a_for_id)
static getInstanceByRefId(int $ref_id, bool $stop_on_error=true)
get an instance of an Ilias object by reference id
countUserArticles(int $a_user_id)
static getEmptyInstance()
getSubPathIdsForNode(ilForumPost $post)
getAllThreads(int $a_topic_id, array $params=[], int $limit=0, int $offset=0)
enableForumNotification(int $user_id)
generateThread(ilForumTopic $thread, string $message, bool $notify, bool $notify_posts, bool $status=true, bool $withFirstVisibleEntry=true)
setPageHits(int $pageHits)
static _isModerator(int $a_ref_id, int $a_usr_id)
A news item can be created by different sources.
countActiveUserArticles(int $a_user_id)
static getInstance()
Singleton: get instance for use in ILIAS requests with a config loaded from the settings.
static getFirstNewsIdForContext(int $a_context_obj_id, string $a_context_obj_type, int $a_context_sub_obj_id=0, string $a_context_sub_obj_type="")
Get first new id of news set related to a certain context.
deletePost($postIdOrRecord, bool $raiseEvents=true)
static formatDate(ilDateTime $date, bool $a_skip_day=false, bool $a_include_wd=false, bool $include_seconds=false, ?ilObjUser $user=null,)
postCensorship(ilObjForum $forum, string $message, int $pos_pk, int $cens=0)
static _getModerators(int $a_ref_id)
setImportName(string $a_import_name)
static getSortedThreadSubjects(int $a_obj_id, int $a_sort_mode=self::SORT_DATE)
setForumId(int $a_obj_id)
deletePostTree(array $a_node)
static mergeThreadNotifications($merge_source_thread_id, $merge_target_thread_id)
static set(string $a_var, $a_val)
Set a value.
isThreadNotificationEnabled(int $user_id, int $thread_id)
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...