19 declare(strict_types=1);
62 public function __construct(
int $a_id = 0,
bool $a_is_moderator =
false,
bool $preventImplicitRead =
false)
66 $this->is_moderator = $a_is_moderator;
67 $this->db = $DIC->database();
68 $this->
user = $DIC->user();
71 if (!$preventImplicitRead) {
78 $this->
setId((
int) $data[
'thr_pk']);
88 $this->
setSticky((
bool) $data[
'is_sticky']);
89 $this->
setClosed((
bool) $data[
'is_closed']);
90 $this->
setAverageRating(isset($data[
'avg_rating']) ? (
float) $data[
'avg_rating'] : 0);
93 if (isset($data[
'thread_sorting'])) {
98 if (isset($data[
'num_posts'])) {
101 if (isset($data[
'num_unread_posts'])) {
104 if (isset($data[
'num_new_posts'])) {
107 if (isset($data[
'usr_notification_is_enabled'])) {
114 if ($this->forum_id) {
115 $nextId = $this->db->nextId(
'frm_threads');
120 'thr_pk' => [
'integer', $nextId],
121 'thr_top_fk' => [
'integer', $this->forum_id],
122 'thr_subject' => [
'text', $this->subject],
123 'thr_display_user_id' => [
'integer', $this->display_user_id],
124 'thr_usr_alias' => [
'text', $this->user_alias],
125 'thr_num_posts' => [
'integer', $this->num_posts],
126 'thr_last_post' => [
'text', $this->last_post_string],
127 'thr_date' => [
'timestamp', $this->createdate],
128 'thr_update' => [
'timestamp', null],
129 'thread_sorting' => [
'integer', (
int) $this->order_sequence_index],
130 'import_name' => [
'text', $this->import_name],
131 'is_sticky' => [
'integer', (
int) $this->is_sticky],
132 'is_closed' => [
'integer', (
int) $this->is_closed],
133 'avg_rating' => [
'text', (
string) $this->average_rating],
134 'thr_author_id' => [
'integer', $this->thr_author_id]
149 $this->db->manipulateF(
159 [
'integer',
'text',
'timestamp',
'integer',
'text',
'text',
'integer'],
165 $this->last_post_string,
166 (
string) $this->average_rating,
180 $res = $this->db->queryF(
182 SELECT frm_threads.*, top_frm_fk frm_obj_id 184 INNER JOIN frm_data ON top_pk = thr_top_fk 192 if (is_object($row)) {
193 $this->forum_id = (
int) $row->thr_top_fk;
194 $this->display_user_id = (
int) $row->thr_display_user_id;
195 $this->user_alias = $row->thr_usr_alias;
196 $this->subject = html_entity_decode((
string) $row->thr_subject);
197 $this->createdate = $row->thr_date;
198 $this->changedate = $row->thr_update;
199 $this->import_name = $row->import_name;
200 $this->num_posts = (
int) $row->thr_num_posts;
201 $this->last_post_string = $row->thr_last_post;
202 $this->visits = (
int) $row->visits;
203 $this->is_sticky = (bool) $row->is_sticky;
204 $this->is_closed = (
bool) $row->is_closed;
205 $this->frm_obj_id = (
int) $row->frm_obj_id;
206 $this->average_rating = (
float) $row->avg_rating;
207 $this->thr_author_id = (
int) $row->thr_author_id;
208 $this->order_sequence_index = (
int) $row->thread_sorting;
221 return $this->
read();
226 $this->db->setLimit(1);
227 $res = $this->db->queryF(
228 'SELECT pos_fk FROM frm_posts_tree WHERE thr_fk = %s AND parent_pos = %s AND depth = %s ORDER BY rgt DESC',
229 [
'integer',
'integer',
'integer'],
233 if ($row = $this->db->fetchObject(
$res)) {
234 return (
int) $row->pos_fk ?: 0;
241 $this->db->setLimit(1);
242 $res = $this->db->queryF(
243 'SELECT pos_fk FROM frm_posts_tree WHERE thr_fk = %s AND parent_pos != %s AND depth = %s ORDER BY rgt DESC',
244 [
'integer',
'integer',
'integer'],
248 if ($row = $this->db->fetchObject(
$res)) {
249 return (
int) $row->pos_fk ?: 0;
256 $checkTime = time() - (60 * 60);
258 if (
ilSession::get(
'frm_visit_frm_threads_' . $this->
id) < $checkTime) {
261 $this->db->manipulateF(
262 'UPDATE frm_threads SET visits = visits + 1 WHERE thr_pk = %s',
271 $res = $this->db->queryF(
275 INNER JOIN frm_posts_tree ON frm_posts_tree.pos_fk = pos_pk 276 WHERE pos_thr_fk = %s' . ($ignoreRoot ?
' AND parent_pos != 0 ' :
''),
281 $row = $this->db->fetchAssoc(
$res);
282 if (is_array($row)) {
283 return (
int) $row[
'cnt'];
291 $res = $this->db->queryF(
295 INNER JOIN frm_posts_tree ON frm_posts_tree.pos_fk = pos_pk 296 WHERE (pos_status = %s 297 OR (pos_status = %s AND pos_display_user_id = %s)) 298 AND pos_thr_fk = %s' . ($ignoreRoot ?
' AND parent_pos != 0 ' :
''),
299 [
'integer',
'integer',
'integer',
'integer'],
303 $row = $this->db->fetchAssoc(
$res);
304 if (is_array($row)) {
305 return (
int) $row[
'cnt'];
313 $this->db->setLimit(1);
314 $res = $this->db->queryF(
318 INNER JOIN frm_posts_tree ON pos_fk = pos_pk 319 WHERE parent_pos = %s 322 [
'integer',
'integer'],
326 if ($row = $this->db->fetchAssoc(
$res)) {
327 $post =
new ilForumPost((
int) $row[
'pos_pk'], $isModerator, $preventImplicitRead);
328 $post->assignData($row);
337 $this->db->setLimit(1);
338 $res = $this->db->queryF(
342 INNER JOIN frm_posts_tree ON pos_fk = pos_pk 343 WHERE parent_pos != %s 347 [
'integer',
'integer',
'integer'],
351 if ($row = $this->db->fetchAssoc(
$res)) {
352 $post =
new ilForumPost((
int) $row[
'pos_pk'], $isModerator, $preventImplicitRead);
353 $post->assignData($row);
363 $this->db->setLimit(1);
364 $res = $this->db->queryF(
365 'SELECT pos_pk FROM frm_posts WHERE pos_thr_fk = %s ORDER BY pos_date DESC',
370 if ($row = $this->db->fetchObject(
$res)) {
381 $this->db->setLimit(1);
382 $res = $this->db->queryF(
386 WHERE pos_thr_fk = %s 387 AND (pos_status = %s OR (pos_status = %s AND pos_display_user_id = %s)) 388 ORDER BY pos_date DESC',
389 [
'integer',
'integer',
'integer',
'integer'],
390 [$this->
id,
'1',
'0', $this->
user->getId()]
393 if ($row = $this->db->fetchObject(
$res)) {
398 throw new OutOfBoundsException(sprintf(
'Could not find last active posting by id: %s', $this->
id));
409 $res = $this->db->queryF(
'SELECT pos_pk FROM frm_posts WHERE pos_thr_fk = %s', [
'integer'], [$this->
id]);
412 $posts[(
int) $row->pos_pk] = (
int) $row->pos_pk;
431 if ($a_post_node->
getLft() > 1) {
432 $dummy_root_condition =
'lft >= %s AND lft < %s';
434 $dummy_root_condition =
'lft > %s AND lft < %s';
438 SELECT is_author_moderator, pos_author_id, pos_pk, fpt_date, rgt, pos_top_fk, pos_thr_fk, 439 pos_display_user_id, pos_usr_alias, pos_subject, 440 pos_status, pos_message, pos_date, pos_update, 441 update_user, pos_cens, pos_cens_com, notify, 442 import_name, fpt_pk, parent_pos, lft, depth, 444 WHEN fur.post_id IS NULL ' .
449 firstname, lastname, title, login 457 ON pos_display_user_id = usr_id 459 LEFT JOIN frm_user_read fur 460 ON fur.thread_id = pos_thr_fk 461 AND fur.post_id = pos_pk 464 WHERE ' . $dummy_root_condition .
' 467 array_push($data_types,
'integer',
'integer',
'integer',
'integer');
470 $this->
user->getId(),
476 if ($this->orderField !==
'') {
483 while ($row = $this->db->fetchAssoc(
$res)) {
485 $post->assignData($row);
487 if (!$this->is_moderator && !
$post->isActivated() &&
$post->getPosAuthorId() !== $this->
user->getId()) {
491 if ((
int) $row[
'pos_display_user_id']) {
492 $usr_ids[(
int) $row[
'pos_display_user_id']] = (
int) $row[
'pos_display_user_id'];
494 if ((
int) $row[
'update_user']) {
495 $usr_ids[(
int) $row[
'update_user']] = (
int) $row[
'update_user'];
515 public function movePosts(
int $old_obj_id,
int $old_pk,
int $new_obj_id,
int $new_pk):
int 524 foreach ($post_ids as $post_id) {
526 $moved = $file_obj->moveFilesOfPost($new_obj_id);
528 if (
true === $moved) {
530 'from' => $old_obj_id,
532 'position_id' => $post_id
539 foreach ($postsMoved as $postedInformation) {
540 $file_obj =
new ilFileDataForum($postedInformation[
'to'], $postedInformation[
'position_id']);
541 $file_obj->moveFilesOfPost($postedInformation[
'from']);
549 $ilAtomQuery = $this->db->buildAtomQuery();
550 $ilAtomQuery->addTableLock(
'frm_user_read');
551 $ilAtomQuery->addTableLock(
'frm_thread_access');
553 $ilAtomQuery->addQueryCallable(
static function (
ilDBInterface $ilDB) use ($new_obj_id, $current_id):
void {
555 'DELETE FROM frm_user_read WHERE obj_id = %s AND thread_id =%s',
556 [
'integer',
'integer'],
557 [$new_obj_id, $current_id]
561 'UPDATE frm_user_read SET obj_id = %s WHERE thread_id = %s',
562 [
'integer',
'integer'],
563 [$new_obj_id, $current_id]
567 'DELETE FROM frm_thread_access WHERE obj_id = %s AND thread_id = %s',
568 [
'integer',
'integer'],
569 [$new_obj_id, $current_id]
573 'UPDATE frm_thread_access SET obj_id = %s WHERE thread_id =%s',
574 [
'integer',
'integer'],
575 [$new_obj_id, $current_id]
581 $this->db->manipulateF(
582 'UPDATE frm_posts SET pos_top_fk = %s WHERE pos_thr_fk = %s',
583 [
'integer',
'integer'],
587 $res = $this->db->queryF(
588 'SELECT * FROM frm_posts WHERE pos_thr_fk = %s',
596 while (
$post = $this->db->fetchAssoc(
$res)) {
600 (
int)
$post[
'pos_pk'],
604 $news_item->setContextObjId($new_obj_id);
605 $news_item->update();
608 return count($post_ids);
615 $is_post_activation_enabled = $objProperties->isPostActivationEnabled();
617 if ($pos_id !== null) {
618 $res = $this->db->queryF(
620 SELECT lft, rgt, depth 624 [
'integer',
'integer'],
641 fp.pos_display_user_id, 645 fp.is_author_moderator, 648 WHEN fur.post_id IS NULL ' .
653 COUNT(fpt2.pos_fk) children 655 FROM frm_posts_tree fpt 657 INNER JOIN frm_posts fp 658 ON fp.pos_pk = fpt.pos_fk 660 LEFT JOIN frm_posts_tree fpt2 661 ON fpt2.lft BETWEEN fpt.lft AND fpt.rgt 662 AND fpt.thr_fk = fpt2.thr_fk 663 AND fpt.pos_fk != fpt2.pos_fk ';
666 LEFT JOIN frm_user_read fur 667 ON fur.thread_id = fp.pos_thr_fk 668 AND fur.post_id = fp.pos_pk 669 AND fur.usr_id = ' . $this->db->quote($this->
user->getId(),
'integer') .
' 671 LEFT JOIN usr_data ud 672 ON ud.usr_id = fp.pos_display_user_id 674 WHERE fpt.thr_fk = ' . $this->db->quote($this->id,
'integer');
677 $query .=
' AND fpt.lft > ' . $this->db->quote(
$data[
'lft'],
'integer') .
678 ' AND fpt.lft < ' . $this->db->quote(
$data[
'rgt'],
'integer') .
' ';
680 if ($is_post_activation_enabled && !$this->is_moderator) {
681 $query .=
' AND (fp.pos_status = 1 OR fp.pos_status = 0 AND fp.pos_display_user_id = ' . $this->db->quote(
682 $this->
user->getId(),
687 if (
$data && is_numeric($num_levels)) {
688 $query .=
' AND fpt.depth <= ' . $this->db->quote((
int)
$data[
'depth'] + $num_levels,
'integer') .
' ';
691 $query .=
' GROUP BY fpt.depth, 700 fp.pos_display_user_id, 704 fp.is_author_moderator, 706 ORDER BY fpt.rgt DESC 711 FROM frm_posts_tree fpt 712 INNER JOIN frm_posts fp 713 ON fp.pos_pk = fpt.pos_fk 714 WHERE fpt.thr_fk = ' . $this->db->quote($this->
id,
'integer');
716 if ($is_post_activation_enabled && !$this->is_moderator) {
717 $queryCounter .=
' AND (fp.pos_status = 1 OR fp.pos_status = 0 AND fp.pos_display_user_id = ' . $this->db->quote(
718 $this->
user->getId(),
722 $queryCounter .=
' ORDER BY fpt.rgt DESC';
724 $resCounter = $this->db->query($queryCounter);
727 while ($row = $this->db->fetchAssoc($resCounter)) {
728 $counter[(
int) $row[
'pos_fk']] =
$i++;
734 while ($row = $this->db->fetchAssoc($res)) {
735 if ((
int) $row[
'pos_display_user_id']) {
736 $usr_ids[] = (
int) $row[
'pos_display_user_id'];
739 $row[
'counter'] = $counter[$row[
'pos_pk']];
741 $casted_row[
'depth'] = (
int) $row[
'depth'];
742 $casted_row[
'rgt'] = (
int) $row[
'rgt'];
743 $casted_row[
'parent_pos'] = (
int) $row[
'parent_pos'];
744 $casted_row[
'pos_pk'] = (
int) $row[
'pos_pk'];
745 $casted_row[
'pos_subject'] = (string) $row[
'pos_subject'];
746 $casted_row[
'pos_usr_alias'] = (string) $row[
'pos_usr_alias'];
747 $casted_row[
'pos_date'] = (string) $row[
'pos_date'];
748 $casted_row[
'pos_update'] = (string) $row[
'pos_update'];
749 $casted_row[
'pos_status'] = (
int) $row[
'pos_status'];
750 $casted_row[
'pos_display_user_id'] = (
int) $row[
'pos_display_user_id'];
751 $casted_row[
'import_name'] = (string) $row[
'import_name'];
752 $casted_row[
'pos_author_id'] = (
int) $row[
'pos_author_id'];
753 $casted_row[
'is_author_moderator'] = (
int) $row[
'is_author_moderator'];
754 $casted_row[
'post_id'] = (
int) $row[
'post_id'];
755 $casted_row[
'post_read'] = (
int) $row[
'post_read'];
756 $casted_row[
'children'] = (
int) $row[
'children'];
758 $children[] = $casted_row;
768 if ($this->
id && $a_user_id) {
769 $result = $this->db->queryF(
770 'SELECT COUNT(notification_id) cnt FROM frm_notification WHERE user_id = %s AND thread_id = %s',
771 [
'integer',
'integer'],
772 [$a_user_id, $this->
id]
775 if ($row = $this->db->fetchAssoc($result)) {
776 return (
int) $row[
'cnt'] > 0;
788 $nextId = $this->db->nextId(
'frm_notification');
789 $this->db->manipulateF(
791 INSERT INTO frm_notification 797 [
'integer',
'integer',
'integer'],
798 [$nextId, $a_user_id, $this->
id]
805 if ($this->
id && $a_user_id) {
806 $this->db->manipulateF(
807 'DELETE FROM frm_notification WHERE user_id = %s AND thread_id = %s',
808 [
'integer',
'integer'],
809 [$a_user_id, $this->
id]
816 if ($this->
id && !$this->is_sticky) {
817 $this->db->manipulateF(
818 'UPDATE frm_threads SET is_sticky = %s WHERE thr_pk = %s',
819 [
'integer',
'integer'],
823 $this->is_sticky =
true;
832 if ($this->
id && $this->is_sticky) {
833 $this->db->manipulateF(
834 'UPDATE frm_threads SET is_sticky = %s WHERE thr_pk = %s',
835 [
'integer',
'integer'],
839 $this->is_sticky =
false;
848 if ($this->
id && !$this->is_closed) {
849 $this->db->manipulateF(
850 'UPDATE frm_threads SET is_closed = %s WHERE thr_pk = %s',
851 [
'integer',
'integer'],
854 $this->is_closed =
true;
860 if ($this->
id && $this->is_closed) {
861 $this->db->manipulateF(
862 'UPDATE frm_threads SET is_closed = %s WHERE thr_pk = %s',
863 [
'integer',
'integer'],
867 $this->is_closed =
false;
881 public function setId(
int $a_id): void
893 $this->forum_id = $a_forum_id;
903 $this->display_user_id = $a_user_id;
913 $this->user_alias = $a_user_alias;
923 $this->subject = $a_subject;
933 $this->createdate = $a_createdate;
943 $this->changedate = $a_changedate;
953 $this->import_name = $a_import_name;
963 $this->last_post_string = $a_last_post;
973 $this->visits = $a_visits;
983 $this->is_sticky = $a_sticky;
1003 $this->is_closed = $a_closed;
1013 $this->orderField = $a_order_field;
1039 $ilDB = $DIC->database();
1042 'SELECT thr_subject FROM frm_threads WHERE thr_pk = %s',
1048 return (
string) $row->thr_subject;
1058 [
'thr_subject' => [
'text', $this->
getSubject()]],
1059 [
'thr_pk' => [
'integer', $this->
getId()]]
1064 $first_node->setSubject($this->
getSubject());
1065 $first_node->update();
1072 $this->num_posts = $a_num_posts;
1105 $this->user_notification_enabled = $status;
1116 if (!in_array(strtoupper($direction), self::$possibleOrderDirections,
true)) {
1117 $direction = current(self::$possibleOrderDirections);
1120 $this->orderDirection = $direction;
1132 $ilDB = $DIC->database();
1135 'SELECT thr_top_fk FROM frm_threads WHERE thr_pk = %s',
1142 return (
int) $row[
'thr_top_fk'];
1150 'thr_num_posts' => [
'integer', $this->
getNumPosts()],
1151 'visits' => [
'integer', $this->
getVisits()],
1153 'thr_subject' => [
'text', $this->
getSubject()]
1155 [
'thr_pk' => [
'integer', $this->
getId()]]
1162 $ilDB = $DIC->database();
1165 'SELECT thr_date FROM frm_threads WHERE thr_pk = %s',
1172 if (is_array($row)) {
1173 $date = $row[
'thr_date'];
1186 $this->last_post =
$post;
static get(string $a_var)
setOrderSequenceIndex(?int $order_sequence_index)
manipulateF(string $query, array $types, array $values)
setNumNewPosts(int $num_new_posts)
getNestedSetPostChildren(?int $pos_id=null, ?int $num_levels=null)
setCreateDate(?string $a_createdate)
static array $possibleOrderDirections
setUserAlias(?string $a_user_alias)
setDisplayUserId(int $a_user_id)
static lookupTitle(int $a_topic_id)
movePosts(int $old_obj_id, int $old_pk, int $new_obj_id, int $new_pk)
Moves all posts within the current thread to a new forum.
enableNotification(int $a_user_id)
setLastPostString(?string $a_last_post)
setSticky(bool $a_sticky)
static lookupCreationDate(int $thread_id)
setNumPosts(int $a_num_posts)
__construct(int $a_id=0, bool $a_is_moderator=false, bool $preventImplicitRead=false)
Returns an object of a forum topic.
getPostRootNode(bool $isModerator=false, bool $preventImplicitRead=false)
static getInstance(int $a_obj_id=0)
setImportName(?string $a_import_name)
countActivePosts(bool $ignoreRoot=false)
setForumId(int $a_forum_id)
getLastPostForThreadOverview()
setNumUnreadPosts(int $num_unread_posts)
setAverageRating(float $average_rating)
setClosed(bool $a_closed)
static _lookupObjIdForForumId(int $a_for_id)
int $order_sequence_index
isUserNotificationEnabled()
setChangeDate(?string $a_changedate)
A news item can be created by different sources.
setSubject(string $a_subject)
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.
setLastPostForThreadOverview(ilForumPost $post)
setOrderField(string $a_order_field)
This class handles all operations on files for the forum object.
setUserNotificationEnabled(bool $status)
static lookupForumIdByTopicId(int $a_topic_id)
getPostTree(ilForumPost $a_post_node)
Fetches and returns an array of posts from the post tree, starting with the node object passed by the...
isNotificationEnabled(int $a_user_id)
disableNotification(int $a_user_id)
getFirstVisiblePostNode(bool $isModerator=false, bool $preventImplicitRead=false)
countPosts(bool $ignoreRoot=false)
static set(string $a_var, $a_val)
Set a value.
setOrderDirection(string $direction)
bool $user_notification_enabled
setThrAuthorId(int $thr_author_id)