ILIAS  trunk Revision v11.0_alpha-1689-g66c127b4ae8
All Data Structures Namespaces Files Functions Variables Enumerations Enumerator Modules Pages
class.ilForumNotification.php
Go to the documentation of this file.
1 <?php
2 
19 declare(strict_types=1);
20 
28 {
30  private static array $node_data_cache = [];
32  private static array $forced_events_cache = [];
33 
34  private readonly ilDBInterface $db;
35  private readonly ilObjUser $user;
36  private int $notification_id;
37  private ?int $user_id = null;
38  private int $forum_id;
39  private int $thread_id;
40  private bool $admin_force = false;
41  private bool $user_toggle = false;
42  private int $interested_events = 0;
43  private int $user_id_noti;
44 
45  public function __construct(private int $ref_id)
46  {
47  global $DIC;
48 
49  $this->db = $DIC->database();
50  $this->user = $DIC->user();
51  $this->forum_id = $DIC['ilObjDataCache']->lookupObjId($ref_id);
52  }
53 
54  public function setNotificationId(int $a_notification_id): void
55  {
56  $this->notification_id = $a_notification_id;
57  }
58 
59  public function getNotificationId(): int
60  {
62  }
63 
64  public function setUserId(?int $a_user_id): void
65  {
66  $this->user_id = $a_user_id;
67  }
68 
69  public function getUserId(): ?int
70  {
71  return $this->user_id;
72  }
73 
74  public function setForumId(int $a_forum_id): void
75  {
76  $this->forum_id = $a_forum_id;
77  }
78 
79  public function getForumId(): int
80  {
81  return $this->forum_id;
82  }
83 
84  public function setThreadId(int $a_thread_id): void
85  {
86  $this->thread_id = $a_thread_id;
87  }
88 
89  public function getThreadId(): int
90  {
91  return $this->thread_id;
92  }
93 
94  public function setInterestedEvents(int $interested_events): void
95  {
96  $this->interested_events = $interested_events;
97  }
98 
99  public function getInterestedEvents(): int
100  {
102  }
103 
104  public function setAdminForce(bool $a_admin_force): void
105  {
106  $this->admin_force = $a_admin_force;
107  }
108 
109  public function getAdminForce(): bool
110  {
111  return $this->admin_force;
112  }
113 
114  public function setUserToggle(bool $a_user_toggle): void
115  {
116  $this->user_toggle = $a_user_toggle;
117  }
118 
119  public function getUserToggle(): bool
120  {
121  return $this->user_toggle;
122  }
123 
124  public function setForumRefId(int $a_ref_id): void
125  {
126  $this->ref_id = $a_ref_id;
127  }
128 
129  public function getForumRefId(): int
130  {
131  return $this->ref_id;
132  }
133 
134  //user_id of who sets the setting to notify members
135  public function setUserIdNoti(int $a_user_id_noti): void
136  {
137  $this->user_id_noti = $a_user_id_noti;
138  }
139 
140  //user_id of who sets the setting to notify members
141  public function getUserIdNoti(): int
142  {
143  return $this->user_id_noti;
144  }
145 
146  public function isAdminForceNotification(): bool
147  {
148  $res = $this->db->queryF(
149  '
150  SELECT admin_force_noti FROM frm_notification
151  WHERE user_id = %s
152  AND frm_id = %s
153  AND user_id_noti > %s ',
154  ['integer', 'integer', 'integer'],
155  [(int) $this->getUserId(), $this->getForumId(), 0]
156  );
157 
158  if ($row = $this->db->fetchAssoc($res)) {
159  return (bool) $row['admin_force_noti'];
160  }
161 
162  return false;
163  }
164 
165  public function isUserToggleNotification(): bool
166  {
167  $res = $this->db->queryF(
168  '
169  SELECT user_toggle_noti FROM frm_notification
170  WHERE user_id = %s
171  AND frm_id = %s
172  AND user_id_noti > %s',
173  ['integer', 'integer', 'integer'],
174  [(int) $this->getUserId(), $this->getForumId(), 0]
175  );
176 
177  if ($row = $this->db->fetchAssoc($res)) {
178  return (bool) $row['user_toggle_noti'];
179  }
180  return false;
181  }
182 
183  public function insertAdminForce(): void
184  {
185  $next_id = $this->db->nextId('frm_notification');
186  $this->setNotificationId($next_id);
187 
188  $this->db->manipulateF(
189  '
190  INSERT INTO frm_notification
191  (notification_id, user_id, frm_id, admin_force_noti, user_toggle_noti, user_id_noti)
192  VALUES(%s, %s, %s, %s, %s, %s)',
193  ['integer', 'integer', 'integer', 'integer', 'integer', 'integer'],
194  [
195  $next_id,
196  (int) $this->getUserId(),
197  $this->getForumId(),
198  $this->getAdminForce(),
199  $this->getUserToggle(),
200  $this->user->getId()
201  ]
202  );
203  }
204 
205  public function deleteAdminForce(): void
206  {
207  $this->db->manipulateF(
208  '
209  DELETE FROM frm_notification
210  WHERE user_id = %s
211  AND frm_id = %s
212  AND admin_force_noti = %s
213  AND user_id_noti > %s',
214  ['integer', 'integer', 'integer', 'integer'],
215  [(int) $this->getUserId(), $this->getForumId(), 1, 0]
216  );
217  }
218 
219  public function deleteUserToggle(): void
220  {
221  $this->db->manipulateF(
222  '
223  DELETE FROM frm_notification
224  WHERE user_id = %s
225  AND frm_id = %s
226  AND admin_force_noti = %s
227  AND user_toggle_noti = %s
228  AND user_id_noti > %s',
229  ['integer', 'integer', 'integer', 'integer', 'integer'],
230  [(int) $this->getUserId(), $this->getForumId(), 1, 1, 0]
231  );
232  }
233 
234  public function updateUserToggle(): void
235  {
236  $this->db->manipulateF(
237  'UPDATE frm_notification SET user_toggle_noti = %s WHERE user_id = %s AND frm_id = %s AND admin_force_noti = %s',
238  ['integer', 'integer', 'integer', 'integer'],
239  [$this->getUserToggle(), (int) $this->getUserId(), $this->getForumId(), 1]
240  );
241  }
242 
243  /* If a new member enters a Course or a Group, this function checks
244  * if this CRS/GRP contains a forum and a notification setting set by admin or moderator
245  * and inserts the new member into frm_notification
246  * */
247  public static function checkForumsExistsInsert(int $ref_id, int $user_id): void
248  {
249  global $DIC;
250 
251  $ilUser = $DIC->user();
252 
253  $node_data = self::getCachedNodeData($ref_id);
254 
255  foreach ($node_data as $data) {
256  //check frm_properties if frm_noti is enabled
257  $frm_noti = new ilForumNotification((int) $data['ref_id']);
258  if ($user_id !== 0) {
259  $frm_noti->setUserId($user_id);
260  } else {
261  $frm_noti->setUserId($ilUser->getId());
262  }
263 
264  $admin_force = ilForumProperties::_isAdminForceNoti((int) $data['obj_id']);
265  $frm_noti->setAdminForce($admin_force);
266 
267  $user_toggle = ilForumProperties::_isUserToggleNoti((int) $data['obj_id']);
268  if ($user_toggle) {
269  $frm_noti->setAdminForce(true);
270  }
271 
272  if ($admin_force || $user_toggle) {
273  $frm_noti->setUserToggle($user_toggle);
274  $frm_noti->setForumId((int) $data['obj_id']);
275  if (!$frm_noti->existsNotification()) {
276  $frm_noti->insertAdminForce();
277  }
278  }
279  }
280  }
281 
282  public static function checkForumsExistsDelete(int $ref_id, int $user_id): void
283  {
284  global $DIC;
285 
286  $ilUser = $DIC->user();
287 
288  $node_data = self::getCachedNodeData($ref_id);
289 
290  foreach ($node_data as $data) {
291  $frm_noti = new ilForumNotification((int) $data['ref_id']);
292  $objFrmMods = new ilForumModerators((int) $data['ref_id']);
293  $moderator_ids = $objFrmMods->getCurrentModerators();
294 
295  if ($user_id !== 0) {
296  $frm_noti->setUserId($user_id);
297  } else {
298  $frm_noti->setUserId($ilUser->getId());
299  }
300 
301  $frm_noti->setForumId((int) $data['obj_id']);
302  if (!in_array($frm_noti->getUserId(), $moderator_ids, true)) {
303  $frm_noti->deleteAdminForce();
304  }
305  }
306  }
307 
308  public static function getCachedNodeData(int $ref_id): array
309  {
310  global $DIC;
311 
312  if (!array_key_exists($ref_id, self::$node_data_cache)) {
313  $container_node = $DIC->repositoryTree()->getNodeData($ref_id);
314  if (!isset($container_node['child'])) {
315  return [];
316  }
317 
318  $node_data = $DIC->repositoryTree()->getSubTree(
319  $container_node,
320  true,
321  ['frm']
322  );
323  $node_data = array_filter($node_data, static function (array $forum_node) use ($DIC, $ref_id): bool {
324  // filter out forum if a grp lies in the path (#0027702)
325  foreach ($DIC->repositoryTree()->getNodePath((int) $forum_node['child'], $ref_id) as $path_node) {
326  $notRootNode = (int) $path_node['child'] !== $ref_id;
327  $isGroup = $path_node['type'] === 'grp';
328  if ($notRootNode && $isGroup) {
329  return false;
330  }
331  }
332 
333  return true;
334  });
335  self::$node_data_cache[$ref_id] = $node_data;
336  }
337 
338  return self::$node_data_cache[$ref_id];
339  }
340 
341  public static function _isParentNodeGrpCrs(int $a_ref_id): bool
342  {
343  global $DIC;
344 
345  $parent_ref_id = $DIC->repositoryTree()->getParentId($a_ref_id);
346  $parent_obj = ilObjectFactory::getInstanceByRefId($parent_ref_id);
347 
348  return $parent_obj->getType() === 'crs' || $parent_obj->getType() === 'grp';
349  }
350 
351  public static function _clearForcedForumNotifications(array $move_tree_event): void
352  {
353  global $DIC;
354  $ilDB = $DIC->database();
355  $ilObjDataCache = $DIC['ilObjDataCache'];
356 
357  if ($move_tree_event['tree'] !== 'tree') {
358  return;
359  }
360 
361  $ref_id = (int) $move_tree_event['source_id'];
362  $is_parent = self::_isParentNodeGrpCrs($ref_id);
363 
364  if ($is_parent) {
365  $forum_id = $ilObjDataCache->lookupObjId($ref_id);
366 
367  $ilDB->manipulateF(
368  'DELETE FROM frm_notification WHERE frm_id = %s AND admin_force_noti = %s',
369  ['integer', 'integer'],
370  [$forum_id, 1]
371  );
372  }
373  }
374 
375  public function update(): void
376  {
377  $this->db->manipulateF(
378  'UPDATE frm_notification SET admin_force_noti = %s, user_toggle_noti = %s, ' .
379  'interested_events = %s WHERE user_id = %s AND frm_id = %s',
380  ['integer', 'integer', 'integer', 'integer', 'integer'],
381  [
382  (int) $this->getAdminForce(),
383  (int) $this->getUserToggle(),
384  $this->getInterestedEvents(),
385  (int) $this->getUserId(),
386  $this->getForumId()
387  ]
388  );
389  }
390 
391  public function deleteNotificationAllUsers(): void
392  {
393  $this->db->manipulateF(
394  'DELETE FROM frm_notification WHERE frm_id = %s AND user_id_noti > %s',
395  ['integer', 'integer'],
396  [$this->getForumId(), 0]
397  );
398  }
399 
403  public function read(): array
404  {
405  $result = [];
406 
407  $res = $this->db->queryF('SELECT * FROM frm_notification WHERE frm_id = %s', ['integer'], [$this->getForumId()]);
408  while ($row = $this->db->fetchAssoc($res)) {
409  $result[(int) $row['user_id']]['notification_id'] = (int) $row['notification_id'];
410  $result[(int) $row['user_id']]['user_id'] = (int) $row['user_id'];
411  $result[(int) $row['user_id']]['frm_id'] = (int) $row['frm_id'];
412  $result[(int) $row['user_id']]['thread_id'] = (int) $row['thread_id'];
413  $result[(int) $row['user_id']]['admin_force_noti'] = (int) $row['admin_force_noti'];
414  $result[(int) $row['user_id']]['user_toggle_noti'] = (int) $row['user_toggle_noti'];
415  $result[(int) $row['user_id']]['interested_events'] = (int) $row['interested_events'];
416  $result[(int) $row['user_id']]['user_id_noti'] = (int) $row['user_id_noti'];
417  }
418 
419  return $result;
420  }
421 
422  public static function mergeThreadNotifications($merge_source_thread_id, $merge_target_thread_id): void
423  {
424  global $DIC;
425 
426  $ilDB = $DIC->database();
427 
428  $res = $ilDB->queryF(
429  'SELECT notification_id, user_id FROM frm_notification WHERE frm_id = %s AND thread_id = %s ORDER BY user_id ASC',
430  ['integer', 'integer'],
431  [0, $merge_source_thread_id]
432  );
433 
434  $res_2 = $ilDB->queryF(
435  'SELECT DISTINCT user_id FROM frm_notification WHERE frm_id = %s AND thread_id = %s ORDER BY user_id ASC',
436  ['integer', 'integer'],
437  [0, $merge_target_thread_id]
438  );
439 
440  $users_already_notified = [];
441  while ($users_row = $ilDB->fetchAssoc($res_2)) {
442  $users_already_notified[(int) $users_row['user_id']] = (int) $users_row['user_id'];
443  }
444 
445  while ($row = $ilDB->fetchAssoc($res)) {
446  if (isset($users_already_notified[(int) $row['user_id']])) {
447  $ilDB->manipulateF(
448  'DELETE FROM frm_notification WHERE notification_id = %s',
449  ['integer'],
450  [$row['notification_id']]
451  );
452  } else {
453  $ilDB->update(
454  'frm_notification',
455  ['thread_id' => ['integer', $merge_target_thread_id]],
456  ['thread_id' => ['integer', $merge_source_thread_id]]
457  );
458  }
459  }
460  }
461 
462  public function existsNotification(): bool
463  {
464  $res = $this->db->queryF(
465  'SELECT user_id FROM frm_notification WHERE user_id = %s AND frm_id = %s AND admin_force_noti = %s',
466  ['integer', 'integer', 'integer'],
467  [(int) $this->getUserId(), $this->getForumId(), (int) $this->getAdminForce()]
468  );
469 
470  return $this->db->numRows($res) > 0;
471  }
472 
473  public function cloneFromSource(int $sourceRefId): void
474  {
475  $sourceNotificationSettings = new self($sourceRefId);
476  $records = $sourceNotificationSettings->read();
477 
478  foreach ($records as $usrId => $row) {
479  $this->setUserId($usrId);
480  $this->setAdminForce((bool) $row['admin_force_noti']);
481  $this->setUserToggle((bool) $row['user_toggle_noti']);
482  $this->setUserIdNoti((int) $row['user_id_noti']);
483  $this->setInterestedEvents((int) $row['interested_events']);
484 
485  $this->insertAdminForce();
486  }
487  }
488 
489  public function updateInterestedEvents(): void
490  {
491  $this->db->manipulateF(
492  'UPDATE frm_notification SET interested_events = %s WHERE user_id = %s AND frm_id = %s',
493  ['integer', 'integer', 'integer'],
494  [$this->getInterestedEvents(), (int) $this->getUserId(), $this->getForumId()]
495  );
496  }
497 
498  public function readInterestedEvents(): int
499  {
500  $interested_events = ilForumNotificationEvents::DEACTIVATED;
501 
502  $this->db->setLimit(1);
503  $res = $this->db->queryF(
504  'SELECT interested_events FROM frm_notification WHERE user_id = %s AND frm_id = %s',
505  ['integer', 'integer'],
506  [(int) $this->getUserId(), $this->getForumId()]
507  );
508 
509  while ($row = $this->db->fetchAssoc($res)) {
510  $interested_events = (int) $row['interested_events'];
511  }
512 
513  return $interested_events;
514  }
515 
519  public function readAllForcedEvents(): array
520  {
521  $res = $this->db->queryF(
522  'SELECT * FROM frm_notification WHERE admin_force_noti = %s AND frm_id = %s',
523  ['integer', 'integer'],
524  [1, $this->forum_id]
525  );
526 
527  while ($row = $this->db->fetchAssoc($res)) {
528  $notificationConfig = new self($this->ref_id);
529  $notificationConfig->setNotificationId((int) $row['notification_id']);
530  $notificationConfig->setUserId((int) $row['user_id']);
531  $notificationConfig->setForumId((int) $row['frm_id']);
532  $notificationConfig->setAdminForce((bool) $row['admin_force_noti']);
533  $notificationConfig->setUserToggle((bool) $row['user_toggle_noti']);
534  $notificationConfig->setInterestedEvents((int) $row['interested_events']);
535 
536  self::$forced_events_cache[(int) $row['user_id']] = $notificationConfig;
537  }
538 
539  return self::$forced_events_cache;
540  }
541 
542  public function getForcedEventsObjectByUserId(int $user_id): self
543  {
544  if (!isset(self::$forced_events_cache[$user_id])) {
545  $this->readAllForcedEvents();
546  }
547 
548  if (!isset(self::$forced_events_cache[$user_id])) {
549  self::$forced_events_cache[$user_id] = $this->createMissingNotification($user_id);
550  }
551 
552  return self::$forced_events_cache[$user_id];
553  }
554 
555  private function createMissingNotification(int $user_id): self
556  {
557  $new_object = new self($this->ref_id);
558  $new_object->setUserId($user_id);
559  $new_object->setForumId($this->forum_id);
560  $new_object->insertAdminForce();
561 
562  return $new_object;
563  }
564 }
$res
Definition: ltiservices.php:66
Class ilForumNotification.
setUserToggle(bool $a_user_toggle)
while($session_entry=$r->fetchRow(ilDBConstants::FETCHMODE_ASSOC)) return null
static _isAdminForceNoti(int $a_obj_id)
setNotificationId(int $a_notification_id)
$ref_id
Definition: ltiauth.php:65
static _clearForcedForumNotifications(array $move_tree_event)
setAdminForce(bool $a_admin_force)
__construct(private int $ref_id)
static getInstanceByRefId(int $ref_id, bool $stop_on_error=true)
get an instance of an Ilias object by reference id
global $DIC
Definition: shib_login.php:22
static _isParentNodeGrpCrs(int $a_ref_id)
static getCachedNodeData(int $ref_id)
getForcedEventsObjectByUserId(int $user_id)
readonly ilDBInterface $db
setInterestedEvents(int $interested_events)
static _isUserToggleNoti(int $a_obj_id)
static mergeThreadNotifications($merge_source_thread_id, $merge_target_thread_id)
static checkForumsExistsDelete(int $ref_id, int $user_id)
static checkForumsExistsInsert(int $ref_id, int $user_id)
setUserIdNoti(int $a_user_id_noti)