ILIAS  trunk Revision v11.0_alpha-1702-gfd3ecb7f852
All Data Structures Namespaces Files Functions Variables Enumerations Enumerator Modules Pages
class.ilObjForum.php
Go to the documentation of this file.
1 <?php
2 
19 declare(strict_types=1);
20 
26 class ilObjForum extends ilObject
27 {
28  public ilForum $Forum;
30  private static array $obj_id_to_forum_id_cache = [];
32  private static array $ref_id_to_forum_id_cache = [];
34  private static array $forum_statistics_cache = [];
36  private static array $forum_last_post_cache = [];
37  private readonly \ILIAS\DI\RBACServices $rbac;
38  private readonly ilLogger $logger;
39  private readonly ilSetting $settings;
40 
41  public function __construct(int $a_id = 0, bool $a_call_by_reference = true)
42  {
43  global $DIC;
44 
45  $this->type = 'frm';
46  parent::__construct($a_id, $a_call_by_reference);
47 
48  $this->rbac = $DIC->rbac();
49  $this->logger = $DIC->logger()->root();
50 
51  $this->settings = $DIC->settings();
52  $this->Forum = new ilForum();
53  }
54 
55  public function create(): int
56  {
57  $id = parent::create();
58 
59  $properties = ilForumProperties::getInstance($this->getId());
60  $properties->setDefaultView(
61  (int) $this->settings->get('forum_default_view', (string) ilForumProperties::VIEW_DATE_ASC)
62  );
63  $properties->setAnonymisation(false);
64  $properties->setStatisticsStatus(false);
65  $properties->setPostActivation(false);
66  $properties->insert();
67 
68  $this->createSettings();
69 
70  $this->setOfflineStatus(true);
71  $this->update();
72  $this->saveData();
73 
74  return $id;
75  }
76 
77  public function setPermissions(int $parent_ref_id): void
78  {
79  parent::setPermissions($parent_ref_id);
80 
81  $roles = [self::_lookupModeratorRole($this->getRefId())];
82  $this->rbac->admin()->assignUser($roles[0], $this->getOwner());
83  $this->updateModeratorRole($roles[0]);
84  }
85 
86  public function updateModeratorRole(int $role_id): void
87  {
88  $this->db->manipulate(
89  'UPDATE frm_data SET top_mods = ' . $this->db->quote(
90  $role_id,
91  'integer'
92  ) . ' WHERE top_frm_fk = ' . $this->db->quote($this->getId(), 'integer')
93  );
94  }
95 
96  public static function _lookupThreadSubject(int $a_thread_id): string
97  {
98  global $DIC;
99 
100  $ilDB = $DIC->database();
101 
102  $res = $ilDB->queryF('SELECT thr_subject FROM frm_threads WHERE thr_pk = %s', ['integer'], [$a_thread_id]);
103  while ($row = $ilDB->fetchObject($res)) {
104  return $row->thr_subject ?? '';
105  }
106 
107  return '';
108  }
109 
110  public function getCountUnread(int $a_usr_id, int $a_thread_id = 0, bool $ignoreRoot = false): int
111  {
112  $a_frm_id = $this->getId();
113  $topic_id = 0;
114  $num_posts = 0;
115  $count_read = 0;
116 
117  if ($a_thread_id === 0) {
118  $res = $this->db->queryF('SELECT top_pk FROM frm_data WHERE top_frm_fk = %s', ['integer'], [$a_frm_id]);
119  while ($row = $this->db->fetchObject($res)) {
120  $topic_id = (int) $row->top_pk;
121  }
122 
123  $res = $this->db->queryF(
124  '
125  SELECT COUNT(pos_pk) num_posts
126  FROM frm_posts
127  LEFT JOIN frm_posts_tree ON frm_posts_tree.pos_fk = pos_pk
128  WHERE pos_top_fk = %s' . ($ignoreRoot ? ' AND parent_pos != 0 ' : ''),
129  ['integer'],
130  [$topic_id]
131  );
132 
133  while ($row = $this->db->fetchObject($res)) {
134  $num_posts = (int) $row->num_posts;
135  }
136 
137  $res = $this->db->queryF(
138  'SELECT COUNT(post_id) count_read FROM frm_user_read WHERE obj_id = %s AND usr_id = %s',
139  ['integer', 'integer'],
140  [$a_frm_id, $a_usr_id]
141  );
142 
143  while ($row = $this->db->fetchObject($res)) {
144  $count_read = (int) $row->count_read;
145  }
146  } else {
147  $res = $this->db->queryF(
148  '
149  SELECT COUNT(pos_pk) num_posts FROM frm_posts
150  LEFT JOIN frm_posts_tree ON frm_posts_tree.pos_fk = pos_pk
151  WHERE pos_thr_fk = %s' . ($ignoreRoot ? ' AND parent_pos != 0 ' : ''),
152  ['integer'],
153  [$a_thread_id]
154  );
155 
156  $row = $this->db->fetchObject($res);
157  $num_posts = (int) $row->num_posts;
158 
159  $res = $this->db->queryF(
160  '
161  SELECT COUNT(post_id) count_read FROM frm_user_read
162  WHERE obj_id = %s
163  AND usr_id = %s
164  AND thread_id = %s',
165  ['integer', 'integer', 'integer'],
166  [$a_frm_id, $a_frm_id, $a_thread_id]
167  );
168 
169  $row = $this->db->fetchObject($res);
170  $count_read = (int) $row->count_read;
171  }
172  $unread = $num_posts - $count_read;
173 
174  return max($unread, 0);
175  }
176 
177  public function markThreadRead(int $a_usr_id, int $a_thread_id): bool
178  {
179  $res = $this->db->queryF('SELECT pos_pk FROM frm_posts WHERE pos_thr_fk = %s', ['integer'], [$a_thread_id]);
180  while ($row = $this->db->fetchObject($res)) {
181  $this->markPostRead($a_usr_id, $a_thread_id, (int) $row->pos_pk);
182  }
183 
184  return true;
185  }
186 
187  public function markAllThreadsRead(int $a_usr_id): void
188  {
189  $res = $this->db->queryF(
190  'SELECT thr_pk FROM frm_data, frm_threads WHERE top_frm_fk = %s AND top_pk = thr_top_fk',
191  ['integer'],
192  [$this->getId()]
193  );
194 
195  while ($row = $this->db->fetchObject($res)) {
196  $this->markThreadRead($a_usr_id, (int) $row->thr_pk);
197  }
198  }
199 
200  public function markPostRead(int $a_usr_id, int $a_thread_id, int $a_post_id): void
201  {
202  $res = $this->db->queryF(
203  '
204  SELECT thread_id FROM frm_user_read
205  WHERE usr_id = %s
206  AND obj_id = %s
207  AND thread_id = %s
208  AND post_id = %s',
209  ['integer', 'integer', 'integer', 'integer'],
210  [$a_usr_id, $this->getId(), $a_thread_id, $a_post_id]
211  );
212 
213  if ($this->db->numRows($res) === 0) {
214  $this->db->manipulateF(
215  '
216  INSERT INTO frm_user_read
217  ( usr_id,
218  obj_id,
219  thread_id,
220  post_id
221  )
222  VALUES (%s,%s,%s,%s)',
223  ['integer', 'integer', 'integer', 'integer'],
224  [$a_usr_id, $this->getId(), $a_thread_id, $a_post_id]
225  );
226  }
227  }
228 
229  public function markPostUnread(int $a_user_id, int $a_post_id): void
230  {
231  $this->db->manipulateF(
232  'DELETE FROM frm_user_read WHERE usr_id = %s AND post_id = %s',
233  ['integer', 'integer'],
234  [$a_user_id, $a_post_id]
235  );
236  }
237 
238  public function isRead($a_usr_id, $a_post_id): bool
239  {
240  $res = $this->db->queryF(
241  'SELECT * FROM frm_user_read WHERE usr_id = %s AND post_id = %s',
242  ['integer', 'integer'],
243  [$a_usr_id, $a_post_id]
244  );
245 
246  return (bool) $this->db->numRows($res);
247  }
248 
249  public static function _deleteUser(int $a_usr_id): void
250  {
251  global $DIC;
252 
253  $data = [$a_usr_id];
254 
255  $DIC->database()->manipulateF('DELETE FROM frm_user_read WHERE usr_id = %s', ['integer'], $data);
256  $DIC->database()->manipulateF('DELETE FROM frm_notification WHERE user_id = %s', ['integer'], $data);
257  }
258 
259  public static function _deleteReadEntries(int $a_post_id): void
260  {
261  global $DIC;
262 
263  $DIC->database()->manipulateF('DELETE FROM frm_user_read WHERE post_id = %s', ['integer'], [$a_post_id]);
264  }
265 
266  public function updateModificationUserId(int $usr_id): void
267  {
268  $this->db->manipulateF(
269  'UPDATE frm_data SET update_user = %s WHERE top_frm_fk = %s',
270  ['integer', 'integer'],
271  [$usr_id, $this->getId()],
272  );
273  }
274 
275  public function update(): bool
276  {
277  if (parent::update()) {
278  $this->db->manipulateF(
279  'UPDATE frm_data SET top_name = %s, top_description = %s, top_update = %s, update_user = %s WHERE top_frm_fk = %s',
280  ['text', 'text', 'timestamp', 'integer', 'integer'],
281  [
282  $this->getTitle(),
283  $this->getDescription(),
284  date("Y-m-d H:i:s"),
285  $this->user->getId(),
286  $this->getId()
287  ]
288  );
289 
290  return true;
291  }
292 
293  return false;
294  }
295 
296  public function cloneObject(int $target_id, int $copy_id = 0, bool $omit_tree = false): ?ilObject
297  {
299  $new_obj = parent::cloneObject($target_id, $copy_id, $omit_tree);
300  $this->cloneAutoGeneratedRoles($new_obj);
301 
302  ilForumProperties::getInstance($this->getId())->copy($new_obj->getId());
303  $this->Forum->setMDB2WhereCondition('top_frm_fk = %s ', ['integer'], [$this->getId()]);
304  $topData = $this->Forum->getOneTopic();
305 
306  $this->db->update('frm_data', [
307  'top_name' => ['text', $topData->getTopName()],
308  'top_description' => ['text', $topData->getTopDescription()],
309  'top_num_posts' => ['integer', $topData->getTopNumPosts()],
310  'top_num_threads' => ['integer', $topData->getTopNumThreads()],
311  'top_last_post' => ['text', $topData->getTopLastPost()],
312  'top_date' => ['timestamp', $topData->getTopDate()],
313  'visits' => ['integer', $topData->getVisits()],
314  'top_update' => ['timestamp', $topData->getTopUpdate()],
315  'update_user' => ['integer', $topData->getUpdateUser()],
316  'top_usr_id' => ['integer', $topData->getTopUsrId()]
317  ], [
318  'top_frm_fk' => ['integer', $new_obj->getId()]
319  ]);
320 
321  $cwo = ilCopyWizardOptions::_getInstance($copy_id);
322  $options = $cwo->getOptions($this->getRefId());
323 
324  $options['threads'] = $this->Forum::getSortedThreadSubjects($this->getId());
325 
326  $new_frm = $new_obj->Forum;
327  $new_frm->setMDB2WhereCondition('top_frm_fk = %s ', ['integer'], [$new_obj->getId()]);
328 
329  $new_frm->setForumId($new_obj->getId());
330  $new_frm->setForumRefId($new_obj->getRefId());
331 
332  $new_topic = $new_frm->getOneTopic();
333  foreach (array_keys($options['threads']) as $thread_id) {
334  $this->Forum->setMDB2WhereCondition('thr_pk = %s ', ['integer'], [$thread_id]);
335 
336  $old_thread = $this->Forum->getOneThread();
337 
338  $old_post_id = $this->Forum->getRootPostIdByThread($old_thread->getId());
339 
340  $newThread = new ilForumTopic(0, true, true);
341  $newThread->setSticky($old_thread->isSticky());
342  $newThread->setForumId($new_topic->getTopPk());
343  $newThread->setThrAuthorId($old_thread->getThrAuthorId());
344  $newThread->setDisplayUserId($old_thread->getDisplayUserId());
345  $newThread->setSubject($old_thread->getSubject());
346  $newThread->setUserAlias($old_thread->getUserAlias());
347  $newThread->setCreateDate($old_thread->getCreateDate());
348 
349  try {
350  $top_pos = $old_thread->getFirstVisiblePostNode();
351  } catch (OutOfBoundsException) {
352  $top_pos = new ilForumPost($old_post_id);
353  }
354 
355  $newPostId = $new_frm->generateThread(
356  $newThread,
357  $top_pos->getMessage(),
358  $top_pos->isNotificationEnabled(),
359  false,
360  true,
361  (bool) ($old_thread->getNumPosts() - 1)
362  );
363 
364  $old_forum_files = new ilFileDataForum($this->getId(), $top_pos->getId());
365  $old_forum_files->ilClone($new_obj->getId(), $newPostId);
366  }
367 
368  $sourceRefId = $this->getRefId();
369  $targetRefId = $new_obj->getRefId();
370 
371  if (
372  $sourceRefId > 0 && $targetRefId > 0 &&
373  $this->tree->getParentId($sourceRefId) === $this->tree->getParentId($targetRefId)
374  ) {
375  $grpRefId = $this->tree->checkForParentType($targetRefId, 'grp');
376  $crsRefId = $this->tree->checkForParentType($targetRefId, 'crs');
377 
378  if ($grpRefId > 0 || $crsRefId > 0) {
379  $notifications = new ilForumNotification($targetRefId);
380  $notifications->cloneFromSource($sourceRefId);
381  }
382  }
383 
384  if (ilForumPage::_exists($this->getType(), $this->getId())) {
385  $translations = ilContentPagePage::lookupTranslations($this->getType(), $this->getId());
386  foreach ($translations as $language) {
387  $originalPageObject = new ilForumPage($this->getId(), 0, $language);
388  $copiedXML = $originalPageObject->copyXmlContent();
389 
390  $duplicatePageObject = new ilForumPage();
391  $duplicatePageObject->setId($new_obj->getId());
392  $duplicatePageObject->setParentId($new_obj->getId());
393  $duplicatePageObject->setLanguage($language);
394  $duplicatePageObject->setXMLContent($copiedXML);
395  $duplicatePageObject->createFromXML();
396  }
397  }
398 
399  $cwo = ilCopyWizardOptions::_getInstance($copy_id);
400  //copy online status if object is not the root copy object
401  if (!$cwo->isRootNode($this->getRefId())) {
402  $new_obj->setOfflineStatus($this->getOfflineStatus());
403  } else {
404  $new_obj->setOfflineStatus(true);
405  }
406  $new_obj->update();
407 
408  return $new_obj;
409  }
410 
411  public function cloneAutoGeneratedRoles(self $new_obj): void
412  {
413  $src_moderator_role_id = self::_lookupModeratorRole($this->getRefId());
414  $new_moderator_role_id = self::_lookupModeratorRole($new_obj->getRefId());
415 
416  if (
417  0 === $src_moderator_role_id ||
418  0 === $new_moderator_role_id ||
419  0 === $this->getRefId() ||
420  0 === $new_obj->getRefId()
421  ) {
422  $this->logger->write(__METHOD__ . ' : Error cloning auto generated role: il_frm_moderator');
423  }
424 
425  $this->rbac->admin()->copyRolePermissions(
426  $src_moderator_role_id,
427  $this->getRefId(),
428  $new_obj->getRefId(),
429  $new_moderator_role_id,
430  true
431  );
432 
433  $this->logger->write(__METHOD__ . ' : Finished copying of role il_frm_moderator.');
434 
435  $moderators = new ilForumModerators($this->getRefId());
436  $src_moderator_usr_ids = $moderators->getCurrentModerators();
437  foreach ($src_moderator_usr_ids as $usr_id) {
438  // The object owner is already member of the moderator role when this method is called
439  // Since the new static caches are introduced with ILIAS 5.0, a database error occurs if we try to assign the user here.
440  if ($this->getOwner() !== $usr_id) {
441  $this->rbac->admin()->assignUser($new_moderator_role_id, $usr_id);
442  }
443  }
444  }
445 
446  public function delete(): bool
447  {
448  $this->Forum->setForumId($this->getId());
449 
450  if (!parent::delete()) {
451  return false;
452  }
453 
454  if (ilForumPage::_exists($this->getType(), $this->getId())) {
455  $originalPageObject = new ilForumPage($this->getId());
456  $originalPageObject->delete();
457  }
458 
459  $this->Forum->setMDB2WhereCondition('top_frm_fk = %s ', [ilDBConstants::T_INTEGER], [$this->getId()]);
460 
461  $topData = $this->Forum->getOneTopic();
462 
463  $threads = $this->Forum->getAllThreads($topData->getTopPk(), [
464  'is_moderator' => true,
465  ]);
466  $thread_ids_to_delete = [];
467  foreach ($threads['items'] as $thread) {
468  $thread_ids_to_delete[$thread->getId()] = $thread->getId();
469  }
470 
471  // Get All posting IDs
472  $posting_ids = [];
473  $res = $this->db->query(
474  'SELECT pos_pk FROM frm_posts WHERE '
475  . $this->db->in('pos_thr_fk', $thread_ids_to_delete, false, ilDBConstants::T_INTEGER)
476  );
477 
478  while ($row = $res->fetchObject()) {
479  $posting_ids[] = (int) $row->pos_pk;
480  }
481 
482  $tmp_file_obj = new ilFileDataForum($this->getId());
483  $tmp_file_obj->delete($posting_ids);
484 
485  // Get All draft IDs
486  $draft_ids = [];
487  $res = $this->db->query(
488  'SELECT draft_id FROM frm_posts_drafts WHERE '
489  . $this->db->in('thread_id', $thread_ids_to_delete, false, ilDBConstants::T_INTEGER)
490  );
491 
492  while ($row = $res->fetchObject()) {
493  $draft_ids[] = (int) $row->draft_id;
494  }
495 
496  $tmp_file_obj = new ilFileDataForumDrafts($this->getId());
497  $tmp_file_obj->delete($draft_ids);
498 
499  $this->db->manipulate(
500  'DELETE FROM frm_posts_tree WHERE ' . $this->db->in(
501  'thr_fk',
502  $thread_ids_to_delete,
503  false,
505  )
506  );
507  $this->db->manipulate(
508  'DELETE FROM frm_posts WHERE ' . $this->db->in(
509  'pos_thr_fk',
510  $thread_ids_to_delete,
511  false,
513  )
514  );
515  $this->db->manipulate(
516  'DELETE FROM frm_threads WHERE ' . $this->db->in(
517  'thr_pk',
518  $thread_ids_to_delete,
519  false,
521  )
522  );
523 
524  $obj_id = [$this->getId()];
525 
526  $this->db->manipulateF('DELETE FROM frm_data WHERE top_frm_fk = %s', [ilDBConstants::T_INTEGER], $obj_id);
527  $this->db->manipulateF('DELETE FROM frm_settings WHERE obj_id = %s', [ilDBConstants::T_INTEGER], $obj_id);
528  $this->db->manipulateF('DELETE FROM frm_user_read WHERE obj_id = %s', [ilDBConstants::T_INTEGER], $obj_id);
529  $this->db->manipulate(
530  'DELETE FROM frm_notification WHERE ' . $this->db->in(
531  'thread_id',
532  $thread_ids_to_delete,
533  false,
534  'integer'
535  )
536  );
537  $this->db->manipulateF('DELETE FROM frm_notification WHERE frm_id = %s', [ilDBConstants::T_INTEGER], $obj_id);
538  $this->db->manipulateF('DELETE FROM frm_posts_deleted WHERE obj_id = %s', [ilDBConstants::T_INTEGER], $obj_id);
539  $this->deleteDraftsByForumId($topData->getTopPk());
540 
541  return true;
542  }
543 
544  private function deleteDraftsByForumId(int $forum_id): void
545  {
546  $res = $this->db->queryF(
547  'SELECT draft_id FROM frm_posts_drafts WHERE forum_id = %s',
548  ['integer'],
549  [$forum_id]
550  );
551 
552  $draft_ids = [];
553  while ($row = $this->db->fetchAssoc($res)) {
554  $draft_ids[] = (int) $row['draft_id'];
555  }
556 
557  if ($draft_ids !== []) {
558  $historyObj = new ilForumDraftsHistory();
559  $historyObj->deleteHistoryByDraftIds($draft_ids);
560 
561  $draftObj = new ilForumPostDraft();
562  $draftObj->deleteDraftsByDraftIds($draft_ids);
563  }
564  }
565 
566  public function initDefaultRoles(): void
567  {
569  'il_frm_moderator_' . $this->getRefId(),
570  "Moderator of forum obj_no." . $this->getId(),
571  'il_frm_moderator',
572  $this->getRefId()
573  );
574  }
575 
576  public static function _lookupModeratorRole(int $a_ref_id): int
577  {
578  global $DIC;
579 
580  $ilDB = $DIC->database();
581 
582  $mod_title = 'il_frm_moderator_' . $a_ref_id;
583 
584  $res = $ilDB->queryF('SELECT obj_id FROM object_data WHERE title = %s', ['text'], [$mod_title]);
585  while ($row = $ilDB->fetchObject($res)) {
586  return (int) $row->obj_id;
587  }
588 
589  return 0;
590  }
591 
592  public function createSettings(): void
593  {
594  global $DIC;
595 
596  $ref_id = 0;
597  if ($DIC->http()->wrapper()->query()->has('ref_id')) {
598  $ref_id = $DIC->http()->wrapper()->query()->retrieve(
599  'ref_id',
600  $DIC->refinery()->kindlyTo()->int()
601  );
602  }
603 
604  // news settings (public notifications yes/no)
605  $default_visibility = ilNewsItem::_getDefaultVisibilityForRefId($ref_id);
606  if ($default_visibility === 'public') {
607  ilBlockSetting::_write('news', 'public_notifications', '1', 0, $this->getId());
608  }
609  }
610 
611  public function saveData(): void
612  {
613  $nextId = $this->db->nextId('frm_data');
614 
615  $top_data = [
616  'top_frm_fk' => $this->getId(),
617  'top_name' => $this->getTitle(),
618  'top_description' => $this->getDescription(),
619  'top_num_posts' => 0,
620  'top_num_threads' => 0,
621  'top_last_post' => null,
622  'top_mods' => 0,
623  'top_usr_id' => $this->user->getId(),
624  'top_date' => ilUtil::now()
625  ];
626 
627  $this->db->manipulateF(
628  '
629  INSERT INTO frm_data
630  (
631  top_pk,
632  top_frm_fk,
633  top_name,
634  top_description,
635  top_num_posts,
636  top_num_threads,
637  top_last_post,
638  top_mods,
639  top_date,
640  top_usr_id
641  )
642  VALUES(%s, %s, %s, %s, %s, %s, %s, %s, %s, %s)',
643  [
644  'integer',
645  'integer',
646  'text',
647  'text',
648  'integer',
649  'integer',
650  'text',
651  'integer',
652  'timestamp',
653  'integer'
654  ],
655  [
656  $nextId,
657  $top_data['top_frm_fk'],
658  $top_data['top_name'],
659  $top_data['top_description'],
660  $top_data['top_num_posts'],
661  $top_data['top_num_threads'],
662  $top_data['top_last_post'],
663  $top_data['top_mods'],
664  $top_data['top_date'],
665  $top_data['top_usr_id']
666  ]
667  );
668  }
669 
670  public static function lookupForumIdByObjId(int $obj_id): int
671  {
672  if (array_key_exists($obj_id, self::$obj_id_to_forum_id_cache)) {
673  return self::$obj_id_to_forum_id_cache[$obj_id];
674  }
675 
676  self::preloadForumIdsByObjIds([$obj_id]);
677 
678  return self::$obj_id_to_forum_id_cache[$obj_id];
679  }
680 
681  public static function lookupForumIdByRefId(int $ref_id): int
682  {
683  if (array_key_exists($ref_id, self::$ref_id_to_forum_id_cache)) {
684  return self::$ref_id_to_forum_id_cache[$ref_id];
685  }
686 
687  self::preloadForumIdsByRefIds([$ref_id]);
688 
689  return self::$ref_id_to_forum_id_cache[$ref_id];
690  }
691 
695  public static function preloadForumIdsByObjIds(array $obj_ids): void
696  {
697  global $DIC;
698 
699  $ilDB = $DIC->database();
700 
701  if (count($obj_ids) === 1) {
702  $in = ' objr.obj_id = ' . $ilDB->quote(current($obj_ids), 'integer') . ' ';
703  } else {
704  $in = $ilDB->in('objr.obj_id', $obj_ids, false, 'integer');
705  }
706  $query = "
707  SELECT frmd.top_pk, objr.ref_id, objr.obj_id
708  FROM object_reference objr
709  INNER JOIN frm_data frmd ON frmd.top_frm_fk = objr.obj_id
710  WHERE $in
711  ";
712  $res = $ilDB->query($query);
713 
714  // Prepare cache array
715  foreach ($obj_ids as $obj_id) {
716  self::$obj_id_to_forum_id_cache[$obj_id] = 0;
717  }
718 
719  while ($row = $ilDB->fetchAssoc($res)) {
720  self::$obj_id_to_forum_id_cache[(int) $row['obj_id']] = (int) $row['top_pk'];
721  self::$ref_id_to_forum_id_cache[(int) $row['ref_id']] = (int) $row['top_pk'];
722  }
723  }
724 
728  public static function preloadForumIdsByRefIds(array $ref_ids): void
729  {
730  global $DIC;
731 
732  $ilDB = $DIC->database();
733 
734  if (count($ref_ids) === 1) {
735  $in = " objr.ref_id = " . $ilDB->quote(current($ref_ids), 'integer') . " ";
736  } else {
737  $in = $ilDB->in('objr.ref_id', $ref_ids, false, 'integer');
738  }
739  $query = "
740  SELECT frmd.top_pk, objr.ref_id, objr.obj_id
741  FROM object_reference objr
742  INNER JOIN frm_data frmd ON frmd.top_frm_fk = objr.obj_id
743  WHERE $in
744  ";
745  $res = $ilDB->query($query);
746 
747  // Prepare cache array
748  foreach ($ref_ids as $ref_id) {
749  if (!array_key_exists($ref_id, self::$ref_id_to_forum_id_cache)) {
750  self::$ref_id_to_forum_id_cache[$ref_id] = 0;
751  }
752  }
753 
754  while ($row = $ilDB->fetchAssoc($res)) {
755  self::$obj_id_to_forum_id_cache[(int) $row['obj_id']] = (int) $row['top_pk'];
756  self::$ref_id_to_forum_id_cache[(int) $row['ref_id']] = (int) $row['top_pk'];
757  }
758  }
759 
763  public static function lookupStatisticsByRefId(int $ref_id): array
764  {
765  global $DIC;
766 
767  $ilAccess = $DIC->access();
768  $ilUser = $DIC->user();
769  $ilDB = $DIC->database();
770  $ilSetting = $DIC->settings();
771 
772  if (isset(self::$forum_statistics_cache[$ref_id])) {
773  return self::$forum_statistics_cache[$ref_id];
774  }
775 
776  $statistics = [
777  'num_posts' => 0,
778  'num_unread_posts' => 0,
779  ];
780 
781  $forumId = self::lookupForumIdByRefId($ref_id);
782  if ($forumId === 0) {
783  self::$forum_statistics_cache[$ref_id] = $statistics;
784  return self::$forum_statistics_cache[$ref_id];
785  }
786 
788  $is_post_activation_enabled = $objProperties->isPostActivationEnabled();
789 
790  $act_clause = '';
791 
792  if ($is_post_activation_enabled && !$ilAccess->checkAccess('moderate_frm', '', $ref_id)) {
793  $act_clause .= ' AND (frm_posts.pos_status = ' . $ilDB->quote(
794  1,
795  'integer'
796  ) . ' OR frm_posts.pos_author_id = ' . $ilDB->quote($ilUser->getId(), 'integer') . ') ';
797  }
798 
799  if (!$ilUser->isAnonymous()) {
800  $query = "
801  (SELECT COUNT(frm_posts.pos_pk) cnt
802  FROM frm_posts
803  INNER JOIN frm_posts_tree tree1
804  ON tree1.pos_fk = frm_posts.pos_pk
805  AND tree1.parent_pos != 0
806  INNER JOIN frm_threads ON frm_posts.pos_thr_fk = frm_threads.thr_pk
807  WHERE frm_threads.thr_top_fk = %s $act_clause)
808 
809  UNION ALL
810 
811  (SELECT COUNT(DISTINCT(frm_user_read.post_id)) cnt
812  FROM frm_user_read
813  INNER JOIN frm_posts ON frm_user_read.post_id = frm_posts.pos_pk
814  INNER JOIN frm_posts_tree tree1
815  ON tree1.pos_fk = frm_posts.pos_pk
816  AND tree1.parent_pos != 0
817  INNER JOIN frm_threads ON frm_threads.thr_pk = frm_posts.pos_thr_fk
818  WHERE frm_user_read.usr_id = %s AND frm_posts.pos_top_fk = %s $act_clause)
819  ";
820 
821  $types = ['integer', 'integer', 'integer'];
822  $values = [$forumId, $ilUser->getId(), $forumId];
823 
824  $mapping = array_keys($statistics);
825  $res = $ilDB->queryF(
826  $query,
827  $types,
828  $values
829  );
830  for ($i = 0; $i < 2; $i++) {
831  $row = $ilDB->fetchAssoc($res);
832 
833  $statistics[$mapping[$i]] = (int) ((is_array($row) ? $row['cnt'] : 0));
834 
835  if ($i === 1) {
836  // unread = all - read
837  $statistics[$mapping[$i]] = $statistics[$mapping[$i - 1]] - $statistics[$mapping[$i]];
838  }
839  }
840  } else {
841  $query = "
842  SELECT COUNT(frm_posts.pos_pk) cnt
843  FROM frm_posts
844  INNER JOIN frm_posts_tree tree1
845  ON tree1.pos_fk = frm_posts.pos_pk
846  AND tree1.parent_pos != 0
847  INNER JOIN frm_threads ON frm_posts.pos_thr_fk = frm_threads.thr_pk
848  WHERE frm_threads.thr_top_fk = %s $act_clause
849  ";
850  $types = ['integer'];
851  $values = [$forumId];
852  $res = $ilDB->queryF(
853  $query,
854  $types,
855  $values
856  );
857  $row = $ilDB->fetchAssoc($res);
858 
859  $statistics = [
860  'num_posts' => (int) $row['cnt'],
861  'num_unread_posts' => (int) $row['cnt'],
862  ];
863  }
864 
865  self::$forum_statistics_cache[$ref_id] = $statistics;
866 
867  return self::$forum_statistics_cache[$ref_id];
868  }
869 
870  public static function lookupLastPostByRefId(int $ref_id): ?array
871  {
872  global $DIC;
873 
874  $ilAccess = $DIC->access();
875  $ilUser = $DIC->user();
876  $ilDB = $DIC->database();
877 
878  if (array_key_exists($ref_id, self::$forum_last_post_cache)) {
879  return self::$forum_last_post_cache[$ref_id];
880  }
881 
882  $forumId = self::lookupForumIdByRefId($ref_id);
883  if ($forumId === 0) {
884  self::$forum_last_post_cache[$ref_id] = null;
885  return self::$forum_last_post_cache[$ref_id];
886  }
887 
888  $act_clause = '';
889  if (!$ilAccess->checkAccess('moderate_frm', '', $ref_id)) {
890  $act_clause .= ' AND (frm_posts.pos_status = ' . $ilDB->quote(
891  1,
892  'integer'
893  ) . ' OR frm_posts.pos_author_id = ' . $ilDB->quote($ilUser->getId(), 'integer') . ') ';
894  }
895 
896  $ilDB->setLimit(1, 0);
897  $query = "
898  SELECT *
899  FROM frm_posts
900  INNER JOIN frm_posts_tree tree1
901  ON tree1.pos_fk = frm_posts.pos_pk
902  AND tree1.parent_pos != 0
903  WHERE pos_top_fk = %s $act_clause
904  ORDER BY pos_date DESC
905  ";
906  $res = $ilDB->queryF(
907  $query,
908  ['integer'],
909  [$forumId]
910  );
911 
912  $data = $ilDB->fetchAssoc($res);
913  if (!is_array($data) || empty($data)) {
914  self::$forum_last_post_cache[$ref_id] = null;
915  return self::$forum_last_post_cache[$ref_id];
916  }
917 
918  $casted_data = [];
919  $casted_data['pos_pk'] = (int) $data['pos_pk'];
920  $casted_data['pos_top_fk'] = (int) $data['pos_top_fk'];
921  $casted_data['pos_thr_fk'] = (int) $data['pos_thr_fk'];
922  $casted_data['pos_usr_alias'] = (string) $data['pos_usr_alias'];
923  $casted_data['pos_subject'] = (string) $data['pos_subject'];
924  $casted_data['pos_date'] = (string) $data['pos_date'];
925  $casted_data['pos_update'] = (string) $data['pos_update'];
926  $casted_data['update_user'] = (int) $data['update_user'];
927  $casted_data['pos_cens'] = (int) $data['pos_cens'];
928  $casted_data['pos_cens_com'] = (string) $data['pos_cens_com'];
929  $casted_data['notify'] = (int) $data['notify'];
930  $casted_data['import_name'] = (string) $data['import_name'];
931  $casted_data['pos_status'] = (int) $data['pos_status'];
932  $casted_data['pos_message'] = (string) $data['pos_message'];
933  $casted_data['pos_author_id'] = (int) $data['pos_author_id'];
934  $casted_data['pos_display_user_id'] = (int) $data['pos_display_user_id'];
935  $casted_data['is_author_moderator'] = (int) $data['is_author_moderator'];
936  $casted_data['pos_cens_date'] = (string) $data['pos_cens_date'];
937  $casted_data['pos_activation_date'] = (string) $data['pos_activation_date'];
938 
939  self::$forum_last_post_cache[$ref_id] = $casted_data;
940 
941  return self::$forum_last_post_cache[$ref_id];
942  }
943 
948  public static function getUserIdsOfLastPostsByRefIdAndThreadIds(int $ref_id, array $thread_ids): array
949  {
950  global $DIC;
951 
952  $ilAccess = $DIC->access();
953  $ilUser = $DIC->user();
954  $ilDB = $DIC->database();
955 
956  $act_clause = '';
957  $act_inner_clause = '';
958  if (!$ilAccess->checkAccess('moderate_frm', '', $ref_id)) {
959  $act_clause .= " AND (t1.pos_status = " . $ilDB->quote(
960  1,
961  "integer"
962  ) . " OR t1.pos_author_id = " . $ilDB->quote($ilUser->getId(), "integer") . ") ";
963  $act_inner_clause .= " AND (t3.pos_status = " . $ilDB->quote(
964  1,
965  "integer"
966  ) . " OR t3.pos_author_id = " . $ilDB->quote($ilUser->getId(), "integer") . ") ";
967  }
968 
969  $in = $ilDB->in("t1.pos_thr_fk", $thread_ids, false, 'integer');
970  $inner_in = $ilDB->in("t3.pos_thr_fk", $thread_ids, false, 'integer');
971 
972  $query = "
973  SELECT t1.pos_display_user_id, t1.update_user
974  FROM frm_posts t1
975  INNER JOIN frm_posts_tree tree1 ON tree1.pos_fk = t1.pos_pk AND tree1.parent_pos != 0
976  INNER JOIN (
977  SELECT t3.pos_thr_fk, MAX(t3.pos_date) pos_date
978  FROM frm_posts t3
979  INNER JOIN frm_posts_tree tree2 ON tree2.pos_fk = t3.pos_pk AND tree2.parent_pos != 0
980  WHERE $inner_in $act_inner_clause
981  GROUP BY t3.pos_thr_fk
982  ) t2 ON t2.pos_thr_fk = t1.pos_thr_fk AND t2.pos_date = t1.pos_date
983  WHERE $in $act_clause
984  GROUP BY t1.pos_thr_fk, t1.pos_display_user_id, t1.update_user
985  ";
986 
987  $usr_ids = [];
988 
989  $res = $ilDB->query($query);
990  while ($row = $ilDB->fetchAssoc($res)) {
991  if ((int) $row['pos_display_user_id'] !== 0) {
992  $usr_ids[] = (int) $row['pos_display_user_id'];
993  }
994  if ((int) $row['update_user'] !== 0) {
995  $usr_ids[] = (int) $row['update_user'];
996  }
997  }
998 
999  return array_unique($usr_ids);
1000  }
1001 
1002  public static function mergeForumUserRead(int $merge_source_thread_id, int $merge_target_thread_id): void
1003  {
1004  global $DIC;
1005 
1006  $DIC->database()->update(
1007  'frm_user_read',
1008  ['thread_id' => ['integer', $merge_target_thread_id]],
1009  ['thread_id' => ['integer', $merge_source_thread_id]]
1010  );
1011  }
1012 
1013  public function getNumStickyThreads(): int
1014  {
1015  $res = $this->db->query(
1016  'SELECT COUNT(is_sticky) num_sticky FROM frm_threads
1017  INNER JOIN frm_data ON top_pk = thr_top_fk
1018  WHERE frm_data.top_frm_fk = ' . $this->db->quote($this->getId(), 'integer') . '
1019  AND is_sticky = ' . $this->db->quote(1, 'integer')
1020  );
1021  if ($row = $this->db->fetchAssoc($res)) {
1022  return (int) $row['num_sticky'];
1023  }
1024 
1025  return 0;
1026  }
1027 
1031  public function getPageObjIds(): array
1032  {
1033  $pageObjIds = [];
1034 
1035  $sql = 'SELECT DISTINCT page_id FROM page_object WHERE parent_id = %s AND parent_type = %s';
1036  $res = $this->db->queryF(
1037  $sql,
1038  ['integer', 'text'],
1039  [$this->getId(), $this->getType()]
1040  );
1041 
1042  while ($row = $this->db->fetchAssoc($res)) {
1043  $pageObjIds[] = (int) $row['page_id'];
1044  }
1045 
1046  return $pageObjIds;
1047  }
1048 }
ilClone(int $new_obj_id, int $new_posting_id)
__construct(int $a_id=0, bool $a_call_by_reference=true)
$res
Definition: ltiservices.php:66
static mergeForumUserRead(int $merge_source_thread_id, int $merge_target_thread_id)
static _lookupThreadSubject(int $a_thread_id)
static array $forum_statistics_cache
updateModificationUserId(int $usr_id)
static _getDefaultVisibilityForRefId(int $a_ref_id)
Get default visibility for reference id.
static lookupForumIdByObjId(int $obj_id)
static _lookupModeratorRole(int $a_ref_id)
updateModeratorRole(int $role_id)
static getUserIdsOfLastPostsByRefIdAndThreadIds(int $ref_id, array $thread_ids)
deleteDraftsByForumId(int $forum_id)
readonly ilLogger $logger
static createDefaultRole(string $a_title, string $a_description, string $a_tpl_name, int $a_ref_id)
static preloadForumIdsByObjIds(array $obj_ids)
Class ilForumNotification.
static lookupTranslations(string $a_parent_type, int $a_id)
Lookup translations.
static lookupForumIdByRefId(int $ref_id)
static now()
Return current timestamp in Y-m-d H:i:s format.
markAllThreadsRead(int $a_usr_id)
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
static _deleteUser(int $a_usr_id)
static lookupStatisticsByRefId(int $ref_id)
static _exists(string $a_parent_type, int $a_id, string $a_lang="", bool $a_no_cache=false)
Checks whether page exists.
global $DIC
Definition: shib_login.php:22
static _write(string $a_type, string $a_setting, string $a_value, int $a_user=0, int $a_block_id=0)
Write setting to database.
static _lookupObjectId(int $ref_id)
readonly ilSetting $settings
static array $obj_id_to_forum_id_cache
setOfflineStatus(bool $status)
static array $forum_last_post_cache
getCountUnread(int $a_usr_id, int $a_thread_id=0, bool $ignoreRoot=false)
static array $ref_id_to_forum_id_cache
static lookupLastPostByRefId(int $ref_id)
isRead($a_usr_id, $a_post_id)
global $ilSetting
Definition: privfeed.php:31
$id
plugin.php for ilComponentBuildPluginInfoObjectiveTest::testAddPlugins
Definition: plugin.php:23
__construct(Container $dic, ilPlugin $plugin)
readonly ILIAS DI RBACServices $rbac
static _getInstance(int $a_copy_id)
markPostUnread(int $a_user_id, int $a_post_id)
setPermissions(int $parent_ref_id)
markThreadRead(int $a_usr_id, int $a_thread_id)
static preloadForumIdsByRefIds(array $ref_ids)
markPostRead(int $a_usr_id, int $a_thread_id, int $a_post_id)
cloneAutoGeneratedRoles(self $new_obj)