19 declare(strict_types=1);
44 private string $replQuote1 =
'<blockquote class="ilForumQuote">';
60 $this->error = $DIC[
'ilErr'];
61 $this->
lng = $DIC->language();
62 $this->db = $DIC->database();
63 $this->
user = $DIC->user();
65 $this->
event = $DIC->event();
70 $this->
id = $a_obj_id;
75 $this->ref_id = $a_ref_id;
90 if ($dbTable ===
'') {
91 die($this->className .
'::setDbTable(): No database table given.');
104 $this->mdb2Query = $query_string;
105 $this->mdb2DataValue = $data_value;
106 $this->mdb2DataType = $data_type;
113 return $this->mdb2Query ?:
'';
118 return $this->mdb2DataValue ?: [];
123 return $this->mdb2DataType ?: [];
146 $query =
'SELECT * FROM frm_data WHERE ';
155 $res = $this->db->queryF(
$query, $data_type, $data_value);
156 $row = $this->db->fetchAssoc(
$res);
158 if (!is_array($row) || $row === []) {
170 $query =
'SELECT * FROM frm_threads WHERE ';
179 $sql_res = $this->db->queryF(
$query, $data_type, $data_value);
180 $result = $this->db->fetchAssoc($sql_res);
181 $result[
'thr_subject'] = trim((
string) ($result[
'thr_subject'] ??
''));
184 $thread_obj->assignData($result);
189 public function generatePost(
193 int $display_user_id,
197 string $subject =
'',
198 ?
string $alias = null,
201 bool $send_activation_mail =
false 204 $objNewPost->setForumId($forum_id);
205 $objNewPost->setThreadId($thread_id);
206 $objNewPost->setSubject($subject);
207 $objNewPost->setMessage($message);
208 $objNewPost->setDisplayUserId($display_user_id);
209 $objNewPost->setUserAlias($alias);
210 $objNewPost->setPosAuthorId($author_id);
214 $is_moderator =
false;
215 if ($frm_settings->getMarkModeratorPosts() && self::_isModerator($this->
getForumRefId(), $author_id)) {
216 $is_moderator =
true;
218 $objNewPost->setIsAuthorModerator($is_moderator);
221 $objNewPost->setCreateDate(date(
'Y-m-d H:i:s'));
222 } elseif (strpos($date,
'-') > 0) {
223 $objNewPost->setCreateDate($date);
227 $objNewPost->setPostActivationDate($objNewPost->getCreateDate());
231 $objNewPost->setNotification($notify);
232 $objNewPost->setStatus($status);
233 $objNewPost->insert();
235 if ($parent_pos === 0) {
236 $this->
addPostTree($objNewPost->getThreadId(), $objNewPost->getId(), $objNewPost->getCreateDate());
239 $objNewPost->getId(),
241 $objNewPost->getThreadId(),
242 $objNewPost->getCreateDate()
246 $lastPost = $objNewPost->getForumId() .
'#' . $objNewPost->getThreadId() .
'#' . $objNewPost->getId();
248 $this->db->manipulateF(
249 'UPDATE frm_threads SET thr_num_posts = thr_num_posts + 1, thr_last_post = %s WHERE thr_pk = %s',
251 [$lastPost, $objNewPost->getThreadId()]
254 $this->db->manipulateF(
255 'UPDATE frm_data SET top_num_posts = top_num_posts + 1, top_last_post = %s WHERE top_pk = %s',
257 [$lastPost, $objNewPost->getForumId()]
262 $forum_obj->markPostRead($objNewPost->getPosAuthorId(), $objNewPost->getThreadId(), $objNewPost->getId());
264 if ($status && $parent_pos > 0) {
266 $news_item->setContext($forum_obj->getId(),
'frm', $objNewPost->getId(),
'pos');
268 $news_item->setTitle($objNewPost->getSubject());
273 if ($objNewPost->getMessage() !== strip_tags($objNewPost->getMessage())) {
274 $news_item->setContentHtml(
true);
277 $news_item->setUserId($display_user_id);
279 $news_item->create();
282 return $objNewPost->getId();
291 bool $withFirstVisibleEntry =
true 304 $this->db->manipulateF(
305 'UPDATE frm_data SET top_num_threads = top_num_threads + 1 WHERE top_pk = %s',
310 $rootNodeId = $this->generatePost(
325 if (!$withFirstVisibleEntry) {
329 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()]
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();
535 if (is_numeric($postIdOrRecord)) {
538 $p_node = $postIdOrRecord;
543 $is_deleted_thread = ($p_node[
"parent"] == 0) ?
true :
false;
544 $num_visible_active_posts = 0;
545 if ($is_deleted_thread) {
547 SELECT COUNT(*) AS cnt 549 INNER JOIN frm_posts_tree ON pos_pk = pos_fk 550 WHERE frm_posts_tree.parent_pos != 0 551 AND pos_thr_fk = ' . $this->db->quote($p_node[
'pos_thr_fk'],
'integer') .
' 552 AND pos_status = ' . $this->db->quote(1,
'integer');
554 $row = $this->db->fetchAssoc(
$res);
555 $num_visible_active_posts = (
int) ($row[
'cnt'] ?? 0);
560 'beforePostDeletion',
565 'thread_deleted' => $is_deleted_thread,
566 'num_visible_active_posts' => $num_visible_active_posts
571 $affected_user_ids[] =
$post->getPosAuthorId();
575 $obj_history->deleteHistoryByPostIds($deleted_post_ids);
578 $obj_draft->deleteDraftsByPostIds($deleted_post_ids);
580 foreach ($deleted_post_ids as $post_id) {
586 $dead_pos = count($deleted_post_ids);
589 if ((
int) $p_node[
'parent'] === 0) {
592 $dead_thr = (
int) $p_node[
'tree'];
594 $this->db->manipulateF(
'DELETE FROM frm_threads WHERE thr_pk = %s', [
'integer'], [$dead_thr]);
595 $this->db->manipulateF(
596 'UPDATE frm_data SET top_num_threads = top_num_threads - 1 WHERE top_frm_fk = %s',
601 $posset = $this->db->queryF(
'SELECT * FROM frm_posts WHERE pos_thr_fk = %s', [
'integer'], [$dead_thr]);
602 while ($posrec = $this->db->fetchAssoc($posset)) {
606 (
int) $posrec[
'pos_pk'],
611 $news_item->delete();
616 foreach (
$mobs as $mob) {
625 $affected_user_ids[] = (
int) $posrec[
'pos_author_id'];
628 $this->db->manipulateF(
'DELETE FROM frm_posts WHERE pos_thr_fk = %s', [
'integer'], [$p_node[
'tree']]);
630 for (
$i = 0;
$i < $dead_pos;
$i++) {
631 $this->db->manipulateF(
'DELETE FROM frm_posts WHERE pos_pk = %s', [
'integer'], [$deleted_post_ids[
$i]]);
636 $deleted_post_ids[$i],
641 $news_item->delete();
646 foreach (
$mobs as $mob) {
657 $this->db->manipulateF(
658 'UPDATE frm_threads SET thr_num_posts = thr_num_posts - %s WHERE thr_pk = %s',
659 [
'integer',
'integer'],
660 [$dead_pos, $p_node[
'tree']]
663 $res1 = $this->db->queryF(
664 'SELECT * FROM frm_posts WHERE pos_thr_fk = %s ORDER BY pos_date DESC',
670 if ($res1->numRows() > 0) {
673 while ($selData = $this->db->fetchAssoc($res1)) {
678 $lastPost_thr = $selData[
'pos_top_fk'] .
'#' . $selData[
'pos_thr_fk'] .
'#' . $selData[
'pos_pk'];
683 $this->db->manipulateF(
684 'UPDATE frm_threads SET thr_last_post = %s WHERE thr_pk = %s',
686 [$lastPost_thr, $p_node[
'tree']]
690 $this->db->manipulateF(
691 'UPDATE frm_data SET top_num_posts = top_num_posts - %s WHERE top_frm_fk = %s',
692 [
'integer',
'integer'],
693 [$dead_pos, $this->
id]
696 $res2 = $this->db->queryF(
697 'SELECT * FROM frm_posts, frm_data WHERE pos_top_fk = top_pk AND top_frm_fk = %s ORDER BY pos_date DESC',
703 if ($res2->numRows() > 0) {
706 while ($selData = $this->db->fetchAssoc($res2)) {
711 $lastPost_top = $selData[
'pos_top_fk'] .
'#' . $selData[
'pos_thr_fk'] .
'#' . $selData[
'pos_pk'];
716 $this->db->manipulateF(
717 'UPDATE frm_data SET top_last_post = %s WHERE top_frm_fk = %s',
719 [$lastPost_top, $this->
id]
729 'user_ids' => $affected_user_ids
746 $frm_overview_setting = (
int) (
new ilSetting(
'frma'))->
get(
751 $is_post_activation_enabled = $frm_props->isPostActivationEnabled();
753 $user_id = $this->
user->getId();
755 $excluded_ids_condition =
'';
756 if (isset(
$params[
'excluded_ids']) && is_array(
$params[
'excluded_ids']) &&
$params[
'excluded_ids']) {
757 $excluded_ids_condition =
' AND ' . $this->db->in(
'thr_pk',
$params[
'excluded_ids'],
true,
'integer') .
' ';
760 if (!isset(
$params[
'order_column']) || !in_array(
761 strtolower(
$params[
'order_column']),
762 [
'lp_date',
'rating',
'thr_subject',
'num_posts',
'num_visit']
764 $params[
'order_column'] =
'post_date';
766 if (!isset(
$params[
'order_direction']) || !in_array(strtolower(
$params[
'order_direction']), [
'asc',
'desc'])) {
767 $params[
'order_direction'] =
'desc';
770 $cnt_active_pos_query =
'';
771 $cnt_join_type =
'LEFT';
772 if ($is_post_activation_enabled && !
$params[
'is_moderator']) {
773 $cnt_active_pos_query =
" AND (pos_status = {$this->db->quote(1, 'integer')} OR pos_author_id = {$this->db->quote($user_id, 'integer')}) ";
774 $cnt_join_type =
"INNER";
777 "SELECT COUNT(DISTINCT(thr_pk)) cnt 779 $cnt_join_type JOIN frm_posts 780 ON pos_thr_fk = thr_pk $cnt_active_pos_query 781 WHERE thr_top_fk = %s $excluded_ids_condition 783 $res = $this->db->queryF(
$query, [
'integer'], [$a_topic_id]);
784 $cntData = $this->db->fetchAssoc(
$res);
785 $cnt = (
int) $cntData[
'cnt'];
788 $active_inner_query =
'';
790 if ($is_post_activation_enabled && !
$params[
'is_moderator']) {
791 $active_query =
' AND (pos_status = %s OR pos_author_id = %s) ';
792 $active_inner_query =
' AND (ipos.pos_status = %s OR ipos.pos_author_id = %s) ';
793 $having =
' HAVING num_posts > 0';
800 $optional_fields =
'';
801 if ($frm_props->isIsThreadRatingEnabled()) {
802 $optional_fields =
',avg_rating';
804 if ($frm_props->getThreadSorting() === 1) {
805 $optional_fields =
',thread_sorting';
808 $additional_sort =
'';
809 if ($frm_props->getThreadSorting()) {
810 $additional_sort .=
' , thread_sorting ASC ';
813 if (
$params[
'order_column'] ===
'thr_subject') {
814 $dynamic_columns = [
', thr_subject ' .
$params[
'order_direction']];
815 } elseif (
$params[
'order_column'] ===
'num_posts') {
816 $dynamic_columns = [
', num_posts ' .
$params[
'order_direction']];
817 } elseif (
$params[
'order_column'] ===
'num_visit') {
818 $dynamic_columns = [
', visits ' .
$params[
'order_direction']];
820 $dynamic_columns = [
', post_date ' .
$params[
'order_direction']];
823 if ($frm_props->isIsThreadRatingEnabled()) {
824 $dynamic_columns[] =
' ,avg_rating ' .
$params[
'order_direction'];
826 if (
'rating' === strtolower(
$params[
'order_column'])) {
827 $dynamic_columns = array_reverse($dynamic_columns);
829 $additional_sort .= implode(
' ', $dynamic_columns);
831 $new_deadline_condition = $this->db->quote(date(
839 if (!$this->
user->isAnonymous()) {
841 (CASE WHEN COUNT(DISTINCT(notification_id)) > 0 THEN 1 ELSE 0 END) usr_notification_is_enabled, 842 MAX(pos_date) post_date, 843 SUM(tree1.parent_pos != 0) num_posts, 844 SUM(tree1.parent_pos != 0) - SUM(tree1.parent_pos != 0 AND postread.post_id IS NOT NULL) num_unread_posts, ";
849 (SELECT COUNT(DISTINCT(ipos.pos_pk)) 851 INNER JOIN frm_posts_tree treenew 852 ON treenew.pos_fk = ipos.pos_pk 853 LEFT JOIN frm_user_read iread ON iread.post_id = ipos.pos_pk AND iread.usr_id = %s 854 LEFT JOIN frm_thread_access iacc ON (iacc.thread_id = ipos.pos_thr_fk AND iacc.usr_id = %s) 855 WHERE ipos.pos_thr_fk = thr_pk 856 AND treenew.parent_pos != 0 857 AND (ipos.pos_update > iacc.access_old_ts 859 (iacc.access_old IS NULL AND (ipos.pos_update > " . $new_deadline_condition .
")) 862 AND ipos.pos_author_id != %s 863 AND iread.usr_id IS NULL $active_inner_query 867 $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 871 LEFT JOIN frm_notification 872 ON frm_notification.thread_id = thr_pk 873 AND frm_notification.user_id = %s 876 ON pos_thr_fk = thr_pk $active_query 877 LEFT JOIN frm_posts_tree tree1 878 ON tree1.pos_fk = frm_posts.pos_pk 879 LEFT JOIN frm_user_read postread 880 ON postread.post_id = pos_pk 881 AND postread.usr_id = %s";
883 $query .=
" WHERE thr_top_fk = %s 884 $excluded_ids_condition 885 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 888 ORDER BY is_sticky DESC $additional_sort, thr_date DESC";
892 $data_types[] =
'integer';
893 $data_types[] =
'integer';
894 $data_types[] =
'integer';
895 if ($is_post_activation_enabled && !
$params[
'is_moderator']) {
896 array_push($data_types,
'integer',
'integer');
899 $data_types[] =
'integer';
900 if ($is_post_activation_enabled && !
$params[
'is_moderator']) {
901 array_push($data_types,
'integer',
'integer');
903 $data_types[] =
'integer';
904 $data_types[] =
'integer';
911 if ($is_post_activation_enabled && !
$params[
'is_moderator']) {
912 array_push(
$data, 1, $user_id);
916 if ($is_post_activation_enabled && !
$params[
'is_moderator']) {
917 array_push(
$data, 1, $user_id);
922 0 usr_notification_is_enabled, 923 MAX(pos_date) post_date, 924 COUNT(DISTINCT(tree1.pos_fk)) num_posts, 925 COUNT(DISTINCT(tree1.pos_fk)) num_unread_posts, 926 COUNT(DISTINCT(tree1.pos_fk)) num_new_posts, 927 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 932 ON pos_thr_fk = thr_pk $active_query 933 LEFT JOIN frm_posts_tree tree1 934 ON tree1.pos_fk = frm_posts.pos_pk AND tree1.parent_pos != 0 937 $query .=
" WHERE thr_top_fk = %s 938 $excluded_ids_condition 939 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 942 ORDER BY is_sticky DESC $additional_sort, thr_date DESC";
944 if ($is_post_activation_enabled && !
$params[
'is_moderator']) {
945 array_push($data_types,
'integer',
'integer');
947 $data_types[] =
'integer';
948 if ($is_post_activation_enabled && !
$params[
'is_moderator']) {
949 array_push(
$data, 1, $user_id);
952 $data[] = $a_topic_id;
954 if ($limit || $offset) {
955 $this->db->setLimit($limit, $offset);
960 while ($row = $this->db->fetchAssoc(
$res)) {
962 $thread->assignData($row);
963 $threads[(
int) $row[
'thr_pk']] = $thread;
964 $threadIds[] = (
int) $row[
'thr_pk'];
967 $inner_last_active_post_condition =
'';
968 if ($is_post_activation_enabled && !
$params[
'is_moderator']) {
969 $inner_last_active_post_condition = sprintf(
970 ' AND (iposts.pos_status = %s OR (iposts.pos_status = %s AND iposts.pos_author_id = %s)) ',
971 $this->db->quote(1,
'integer'),
972 $this->db->quote(0,
'integer'),
973 $this->db->quote($this->
user->getId(),
'integer')
977 $post_res = $this->db->query(
981 SELECT pos_thr_fk, MAX(iposts.pos_date) i_pos_date 982 FROM frm_posts iposts 983 WHERE ' . $this->db->in(
'iposts.pos_thr_fk', $threadIds,
false,
'integer') .
' 984 ' . $inner_last_active_post_condition .
' 986 ) opost ON frm_posts.pos_thr_fk = opost.pos_thr_fk AND frm_posts.pos_date = opost.i_pos_date' 988 while ($post_row = $this->db->fetchAssoc($post_res)) {
989 $tmp_obj =
new ilForumPost((
int) $post_row[
'pos_pk'], (
bool)
$params[
'is_moderator'],
true);
990 $tmp_obj->setPosAuthorId((
int) $post_row[
'pos_author_id']);
991 $tmp_obj->setDisplayUserId((
int) $post_row[
'pos_display_user_id']);
992 $tmp_obj->setUserAlias((
string) $post_row[
'pos_usr_alias']);
993 $tmp_obj->setImportName((
string) $post_row[
'import_name']);
994 $tmp_obj->setId((
int) $post_row[
'pos_pk']);
995 $tmp_obj->setCreateDate((
string) $post_row[
'pos_date']);
997 $threads[(
int) $post_row[
'pos_thr_fk']]->setLastPostForThreadOverview($tmp_obj);
1001 'items' => $threads,
1010 SUM(IF(f.pos_cens = %s, 1, 0)) cnt 1012 INNER JOIN frm_posts_tree t ON f.pos_pk = t.pos_fk AND t.parent_pos != %s 1013 INNER JOIN frm_threads th ON t.thr_fk = th.thr_pk 1014 INNER JOIN frm_data d ON d.top_pk = f.pos_top_fk AND d.top_frm_fk = %s 1015 WHERE f.pos_author_id = %s 1018 if ($post_activation_required) {
1019 $query .=
' AND f.pos_status = ' . $this->db->quote(1,
'integer');
1022 $res = $this->db->queryF(
1024 [
'integer',
'integer',
'integer',
'integer'],
1027 $row = $this->db->fetchAssoc(
$res);
1028 if (is_array($row)) {
1029 return (
int) $row[
'cnt'];
1047 u.login, u.lastname, u.firstname, f.pos_author_id, u.usr_id, 1048 p.value public_profile, 1049 SUM(IF(f.pos_cens = %s, 1, 0)) num_postings 1051 INNER JOIN frm_posts_tree t ON f.pos_pk = t.pos_fk 1052 INNER JOIN frm_threads th ON t.thr_fk = th.thr_pk 1053 INNER JOIN usr_data u ON u.usr_id = f.pos_author_id 1054 INNER JOIN frm_data d ON d.top_pk = f.pos_top_fk 1055 LEFT JOIN usr_pref p ON p.usr_id = u.usr_id AND p.keyword = %s 1056 WHERE t.parent_pos != %s 1059 $data_types[] =
'integer';
1060 $data_types[] =
'text';
1061 $data_types[] =
'integer';
1063 $data[] =
'public_profile';
1066 if ($post_activation_required) {
1067 $query .=
' AND pos_status = %s';
1068 $data_types[] =
'integer';
1073 AND d.top_frm_fk = %s 1074 GROUP BY u.login, p.value,u.lastname, u.firstname, f.pos_author_id 1077 $data_types[] =
'integer';
1081 while ($row = $this->db->fetchAssoc(
$res)) {
1083 !in_array($row[
'public_profile'], [
1088 $row[
'lastname'] =
'';
1089 $row[
'firstname'] =
'';
1092 $row[
'usr_id'] = (
int) $row[
'usr_id'];
1093 $row[
'pos_author_id'] = (
int) $row[
'pos_author_id'];
1094 $row[
'num_postings'] = (
int) $row[
'num_postings'];
1096 $statistic[] = $row;
1104 $res = $this->db->queryF(
1105 'SELECT pos_fk FROM frm_posts_tree WHERE thr_fk = %s AND parent_pos = %s',
1106 [
'integer',
'integer'],
1110 $row = $this->db->fetchObject(
$res);
1112 return (
int) $row->pos_fk;
1134 $rbacreview = $DIC->rbac()->review();
1136 $role_arr = $rbacreview->getRolesOfRoleFolder($a_ref_id);
1137 foreach ($role_arr as $role_id) {
1139 return array_map(
'intval', $rbacreview->assignedUsers($role_id));
1148 if (!isset(self::$moderators_by_ref_id_map[$a_ref_id])) {
1149 self::$moderators_by_ref_id_map[$a_ref_id] = self::_getModerators($a_ref_id);
1152 return in_array($a_usr_id, self::$moderators_by_ref_id_map[$a_ref_id],
true);
1157 $res = $this->db->queryF(
1158 'SELECT * FROM frm_data 1159 INNER JOIN frm_posts ON pos_top_fk = top_pk 1160 INNER JOIN frm_posts_tree tree1 ON tree1.pos_fk = frm_posts.pos_pk AND tree1.parent_pos != 0 1161 WHERE top_frm_fk = %s 1162 AND pos_author_id = %s',
1163 [
'integer',
'integer'],
1167 return $res->numRows();
1172 $res = $this->db->queryF(
1173 'SELECT * FROM frm_data 1174 INNER JOIN frm_posts ON pos_top_fk = top_pk 1175 INNER JOIN frm_posts_tree tree1 ON tree1.pos_fk = frm_posts.pos_pk AND tree1.parent_pos != 0 1176 WHERE top_frm_fk = %s 1177 AND (pos_status = %s OR (pos_status = %s AND pos_author_id = %s)) 1178 AND pos_author_id = %s',
1179 [
'integer',
'integer',
'integer',
'integer',
'integer'],
1183 return $res->numRows();
1191 public function addPostTree(
int $a_tree_id,
int $a_node_id = -1,
string $a_date =
''): bool
1193 $a_date = $a_date ?: date(
'Y-m-d H:i:s');
1195 if ($a_node_id <= 0) {
1196 $a_node_id = $a_tree_id;
1199 $nextId = $this->db->nextId(
'frm_posts_tree');
1201 $this->db->manipulateF(
1203 INSERT INTO frm_posts_tree 1213 VALUES(%s, %s, %s, %s, %s, %s, %s, %s )',
1214 [
'integer',
'integer',
'integer',
'integer',
'integer',
'integer',
'integer',
'timestamp'],
1215 [$nextId, $a_tree_id, $a_node_id, 0, 1, 2, 1, $a_date]
1224 public function insertPostNode(
int $a_node_id,
int $a_parent_id,
int $tree_id,
string $a_date =
''): void
1226 $a_date = $a_date ?: date(
'Y-m-d H:i:s');
1229 $res = $this->db->queryF(
1230 'SELECT lft FROM frm_posts_tree WHERE pos_fk = %s AND thr_fk = %s',
1231 [
'integer',
'integer'],
1232 [$a_parent_id, $tree_id]
1234 $row = $this->db->fetchObject(
$res);
1236 $left = (
int) $row->lft;
1242 $this->db->manipulateF(
1244 UPDATE frm_posts_tree 1256 [
'integer',
'integer',
'integer'],
1257 [$left, $left, $tree_id]
1260 $depth = $this->
getPostDepth($a_parent_id, $tree_id) + 1;
1262 $nextId = $this->db->nextId(
'frm_posts_tree');
1263 $this->db->manipulateF(
1265 INSERT INTO frm_posts_tree 1275 VALUES(%s,%s,%s, %s, %s, %s,%s, %s)',
1276 [
'integer',
'integer',
'integer',
'integer',
'integer',
'integer',
'integer',
'timestamp'],
1293 $res = $this->db->queryF(
1294 'SELECT depth FROM frm_posts_tree WHERE pos_fk = %s AND thr_fk = %s',
1295 [
'integer',
'integer'],
1296 [$a_node_id, $tree_id]
1299 $row = $this->db->fetchObject(
$res);
1301 return (
int) $row->depth;
1310 $res = $this->db->queryF(
1311 'SELECT * FROM frm_posts, frm_posts_tree WHERE pos_pk = pos_fk AND parent_pos = %s AND thr_fk = %s',
1312 [
'integer',
'integer'],
1316 if ($row = $this->db->fetchObject(
$res)) {
1325 $res = $this->db->queryF(
1326 'SELECT * FROM frm_posts, frm_posts_tree WHERE pos_pk = pos_fk AND pos_pk = %s',
1331 if ($row = $this->db->fetchObject(
$res)) {
1344 $tmp_user =
new ilObjUser((
int) $a_row->pos_display_user_id);
1345 $fullname = $tmp_user->getFullname();
1346 $loginname = $tmp_user->getLogin();
1349 if ($fullname ===
'') {
1350 $fullname = $this->
lng->txt(
'unknown');
1351 if ($a_row->import_name) {
1352 $fullname = $a_row->import_name;
1358 'pos_pk' => (
int) $a_row->pos_pk,
1359 'pos_thr_fk' => (
int) $a_row->pos_thr_fk,
1360 'child' => (
int) $a_row->pos_pk,
1361 'author' => (
int) $a_row->pos_display_user_id,
1362 'alias' => (string) $a_row->pos_usr_alias,
1363 'title' => $fullname,
1364 'loginname' => $loginname,
1365 'message' => (
string) $a_row->pos_message,
1366 'subject' => (string) $a_row->pos_subject,
1367 'pos_cens_com' => (
string) $a_row->pos_cens_com,
1368 'pos_cens' => (
int) $a_row->pos_cens,
1369 'date' => $a_row->fpt_date,
1370 'create_date' => $a_row->pos_date,
1371 'update' => $a_row->pos_update,
1372 'update_user' => (
int) $a_row->update_user,
1373 'tree' => (
int) $a_row->thr_fk,
1374 'parent' => (
int) $a_row->parent_pos,
1375 'lft' => (
int) $a_row->lft,
1376 'rgt' => (
int) $a_row->rgt,
1377 'depth' => (
int) $a_row->depth,
1378 'id' => (
int) $a_row->fpt_pk,
1379 'notify' => (
int) $a_row->notify,
1380 'import_name' => $a_row->import_name,
1381 'pos_status' => (
int) $a_row->pos_status
1391 $res = $this->db->queryF(
1392 'SELECT lft, rgt FROM frm_posts_tree WHERE thr_fk = %s AND pos_fk = %s AND parent_pos = %s',
1393 [
'integer',
'integer',
'integer'],
1394 [$a_node[
'tree'], $a_node[
'pos_pk'], $a_node[
'parent']]
1397 while ($row = $this->db->fetchObject(
$res)) {
1398 $a_node[
'lft'] = (
int) $row->lft;
1399 $a_node[
'rgt'] = (
int) $row->rgt;
1402 $diff = $a_node[
'rgt'] - $a_node[
'lft'] + 1;
1404 $res = $this->db->queryF(
1405 'SELECT pos_fk FROM frm_posts_tree WHERE lft BETWEEN %s AND %s AND thr_fk = %s',
1406 [
'integer',
'integer',
'integer'],
1407 [$a_node[
'lft'], $a_node[
'rgt'], $a_node[
'tree']]
1410 $deleted_post_ids = [];
1411 while ($treeData = $this->db->fetchAssoc(
$res)) {
1412 $deleted_post_ids[] = (
int) $treeData[
'pos_fk'];
1415 $this->db->manipulateF(
1416 'DELETE FROM frm_posts_tree WHERE lft BETWEEN %s AND %s AND thr_fk = %s',
1417 [
'integer',
'integer',
'integer'],
1418 [$a_node[
'lft'], $a_node[
'rgt'], $a_node[
'tree']]
1421 $this->db->manipulateF(
1423 UPDATE frm_posts_tree 1435 [
'integer',
'integer',
'integer',
'integer',
'integer'],
1436 [$a_node[
'lft'], $diff, $a_node[
'lft'], $diff, $a_node[
'tree']]
1439 return $deleted_post_ids;
1444 $checkTime = time() - (60 * 60);
1445 $session_key =
'frm_visit_' . $this->dbTable .
'_' . $ID;
1449 $query =
'UPDATE ' . $this->dbTable .
' SET visits = visits + 1 WHERE ';
1458 $this->db->manipulateF(
$query, $data_type, $data_value);
1463 public function prepareText(
string $text,
int $edit = 0,
string $quote_user =
'',
string $type =
''): string
1465 if (
$type ===
'export') {
1466 $this->replQuote1 =
"<blockquote class=\"quote\"><hr size=\"1\" color=\"#000000\">";
1467 $this->replQuote2 =
"<hr size=\"1\" color=\"#000000\"/></blockquote>";
1472 if ($quote_user !==
'') {
1473 $lname =
'="' . $quote_user .
'"';
1476 $text =
"[quote$lname]" . $text .
"[/quote]";
1479 $startZ = substr_count($text,
"[quote");
1480 $endZ = substr_count($text,
"[/quote]");
1482 if ($startZ > 0 || $endZ > 0) {
1484 if ($startZ > $endZ) {
1485 $diff = $startZ - $endZ;
1487 for (
$i = 0;
$i < $diff;
$i++) {
1488 if (
$type ===
'export') {
1491 $text .=
"[/quote]";
1494 } elseif ($startZ < $endZ) {
1495 $diff = $endZ - $startZ;
1497 for (
$i = 0;
$i < $diff;
$i++) {
1498 if (
$type ===
'export') {
1499 $text = $this->txtQuote1 . $text;
1501 $text =
"[quote]" . $text;
1507 $text = preg_replace(
1508 '@\[(quote\s*?=\s*?"([^"]*?)"\s*?)\]@i',
1509 $this->replQuote1 .
'<div class="ilForumQuoteHead">' . $this->
lng->txt(
'quote') .
' ($2)</div>',
1513 $text = str_replace(
1514 [
"[quote]",
"[/quote]"],
1516 $this->replQuote1 .
'<div class="ilForumQuoteHead">' . $this->
lng->txt(
'quote') .
'</div>',
1525 if (
$type !==
'export') {
1534 $text = str_replace([
"{",
"}"], [
"{",
"}"], $text);
1546 foreach ($a_ids as $pos_id) {
1547 $forumFiles->setPosId($pos_id);
1548 $files = $forumFiles->getFilesOfPost();
1549 foreach ($files as $file) {
1550 $forumFiles->unlinkFile($file[
'name']);
1562 $this->import_name = $a_import_name;
1569 $res = $this->db->queryF(
1571 SELECT frm_notification.thread_id FROM frm_data, frm_notification, frm_threads 1572 WHERE frm_notification.user_id = %s 1573 AND frm_notification.thread_id = frm_threads.thr_pk 1574 AND frm_threads.thr_top_fk = frm_data.top_pk 1575 AND frm_data.top_frm_fk = %s 1576 GROUP BY frm_notification.thread_id',
1577 [
'integer',
'integer'],
1578 [$user_id, $this->
id]
1581 if (
$res->numRows() > 0) {
1583 $thread_data_types = [];
1585 $query =
' DELETE FROM frm_notification WHERE user_id = %s AND thread_id IN (';
1586 $thread_data[] = $user_id;
1587 $thread_data_types[] =
'integer';
1590 while ($row = $this->db->fetchAssoc(
$res)) {
1591 if ($counter < $res->numRows()) {
1593 $thread_data[] = $row[
'thread_id'];
1594 $thread_data_types[] =
'integer';
1597 if ($counter ===
$res->numRows()) {
1599 $thread_data[] = $row[
'thread_id'];
1600 $thread_data_types[] =
'integer';
1605 $this->db->manipulateF(
$query, $thread_data_types, $thread_data);
1609 $nextId = $this->db->nextId(
'frm_notification');
1610 $this->db->manipulateF(
1611 'INSERT INTO frm_notification (notification_id, user_id, frm_id) VALUES(%s, %s, %s)',
1612 [
'integer',
'integer',
'integer'],
1613 [$nextId, $user_id, $this->
id]
1622 $this->db->manipulateF(
1623 'DELETE FROM frm_notification WHERE user_id = %s AND frm_id = %s',
1624 [
'integer',
'integer'],
1625 [$user_id, $this->
id]
1633 $res = $this->db->queryF(
1634 'SELECT COUNT(*) cnt FROM frm_notification WHERE user_id = %s AND frm_id = %s',
1635 [
'integer',
'integer'],
1636 [$user_id, $this->
id]
1639 if ($row = $this->db->fetchAssoc(
$res)) {
1640 return (
int) $row[
'cnt'] > 0;
1648 $res = $this->db->queryF(
1649 'SELECT COUNT(*) cnt FROM frm_notification WHERE user_id = %s AND thread_id = %s',
1650 [
'integer',
'integer'],
1651 [$user_id, $thread_id]
1654 if ($row = $this->db->fetchAssoc(
$res)) {
1655 return (
int) $row[
'cnt'] > 0;
1670 switch ($a_sort_mode) {
1671 case self::SORT_DATE:
1675 case self::SORT_TITLE:
1677 $sort =
'thr_subject';
1681 $res = $DIC->database()->queryF(
1682 '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',
1683 [
'integer',
'text'],
1688 while ($row = $DIC->database()->fetchObject(
$res)) {
1689 $threads[(
int) $row->thr_pk] = $row->thr_subject;
1699 $res = $DIC->database()->queryF(
'SELECT top_frm_fk FROM frm_data WHERE top_pk = %s', [
'integer'], [$a_for_id]);
1700 if ($row = $DIC->database()->fetchAssoc(
$res)) {
1701 return (
int) $row[
'top_frm_fk'];
1713 if ($sourceThread->getForumId() !== $targetThread->getForumId()) {
1714 throw new ilException(
'not_allowed_to_merge_into_another_forum');
1718 if ($sourceThread->getCreateDate() > $targetThread->getCreateDate()) {
1719 $sourceThreadForMerge = $sourceThread;
1720 $targetThreadForMerge = $targetThread;
1722 $sourceThreadForMerge = $targetThread;
1723 $targetThreadForMerge = $sourceThread;
1726 $threadSubject = $targetThreadForMerge->getSubject();
1728 $targetWasClosedBeforeMerge = $targetThreadForMerge->isClosed();
1729 $sourceThreadForMerge->close();
1731 if (
false === $targetWasClosedBeforeMerge) {
1732 $targetThreadForMerge->close();
1735 $allSourcePostings = $sourceThreadForMerge->getAllPostIds();
1736 $sourceThreadRootNode = $sourceThreadForMerge->getPostRootNode();
1737 $targetThreadRootNode = $targetThreadForMerge->getPostRootNode();
1739 $sourceThreadRootArray = $this->
getPostNode($sourceThreadRootNode->getId());
1741 $ilAtomQuery = $this->db->buildAtomQuery();
1742 $ilAtomQuery->addTableLock(
'frm_posts');
1743 $ilAtomQuery->addTableLock(
'frm_posts_tree');
1744 $ilAtomQuery->addTableLock(
'frm_threads');
1745 $ilAtomQuery->addTableLock(
'frm_data');
1748 $targetThreadForMerge,
1749 $sourceThreadForMerge,
1750 $targetThreadRootNode,
1751 $sourceThreadRootNode,
1754 $targetRootNodeRgt = $targetThreadRootNode->getRgt();
1755 $targetRootNodeId = $targetThreadRootNode->getId();
1759 $targetThreadRootNode->getId(),
1760 ($targetThreadRootNode->getRgt() + $sourceThreadRootNode->getRgt() - 2)
1764 foreach ($allSourcePostings as $pos_pk) {
1767 if ($post_obj->getId() === $sourceThreadRootNode->getId()) {
1773 $tree->setPosFk($pos_pk);
1775 if ($post_obj->getParentId() === $sourceThreadRootNode->getId()) {
1776 $tree->setParentPos($targetRootNodeId);
1778 $tree->setParentPos($post_obj->getParentId());
1781 $tree->setLft(($post_obj->getLft() + $targetRootNodeRgt) - 2);
1782 $tree->setRgt(($post_obj->getRgt() + $targetRootNodeRgt) - 2);
1784 $tree->setDepth($post_obj->getDepth());
1785 $tree->setTargetThreadId($targetThreadForMerge->getId());
1786 $tree->setSourceThreadId($sourceThreadForMerge->getId());
1792 $sourceThreadForMerge->getId(),
1793 $targetThreadForMerge->getId(),
1794 [$sourceThreadRootNode->getId()]
1797 $ilAtomQuery->run();
1803 $lastPostString = $targetThreadForMerge->getLastPostString();
1804 $exp = explode(
'#', $lastPostString);
1805 if (array_key_exists(2, $exp)) {
1807 $exp[2] = $targetThreadForMerge->getLastPost()->getId();
1808 $lastPostString = implode(
'#', $exp);
1810 $lastPostString = null;
1815 $frm_topic_obj->setNumPosts($sourceThreadForMerge->getNumPosts() + $targetThreadForMerge->getNumPosts());
1816 $frm_topic_obj->setVisits($sourceThreadForMerge->getVisits() + $targetThreadForMerge->getVisits());
1817 $frm_topic_obj->setLastPostString($lastPostString);
1818 $frm_topic_obj->setSubject($threadSubject);
1819 $frm_topic_obj->setId($targetThreadForMerge->getId());
1820 $frm_topic_obj->updateMergedThread();
1822 if (!$targetWasClosedBeforeMerge) {
1823 $targetThreadForMerge->reopen();
1826 $this->
event->raise(
1831 'source_thread_id' => $sourceThreadForMerge->getId(),
1832 'target_thread_id' => $targetThreadForMerge->getId()
1836 $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)
const PROFILE_ENABLED_LOGGED_IN_USERS
Class Forum core functions for forum.
static array $moderators_by_ref_id_map
const NEWS_NEW_CONSIDERATION_WEEKS
static _deleteAccessEntries(int $a_thread_id)
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)
static formatDate(ilDateTime $date, bool $a_skip_day=false, bool $a_include_wd=false, bool $include_seconds=false)
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)
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)
prepareText(string $text, int $edit=0, string $quote_user='', string $type='')
static _lookupObjIdForForumId(int $a_for_id)
const NEWS_NOTICE
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
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()
getAllThreads(int $a_topic_id, array $params=[], int $limit=0, int $offset=0)
const PROFILE_ENABLED_GLOBAL
enableForumNotification(int $user_id)
generateThread(ilForumTopic $thread, string $message, bool $notify, bool $notify_posts, bool $status=true, bool $withFirstVisibleEntry=true)
const FORUM_OVERVIEW_WITH_NEW_POSTS
setPageHits(int $pageHits)
static _isModerator(int $a_ref_id, int $a_usr_id)
A news item can be created by different sources.
Error Handling & global info handling uses PEAR error class.
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.
This class handles all operations on files for the forum object.
deletePost($postIdOrRecord, bool $raiseEvents=true)
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)