ILIAS  trunk Revision v11.0_alpha-1702-gfd3ecb7f852
All Data Structures Namespaces Files Functions Variables Enumerations Enumerator Modules Pages
class.ilObjSCORMTracking.php
Go to the documentation of this file.
1 <?php
2 
19 declare(strict_types=1);
20 
27 {
28  public static function storeJsApi(): void
29  {
30  global $DIC;
31  $obj_id = $DIC->http()->wrapper()->query()->retrieve('package_id', $DIC->refinery()->kindlyTo()->int());
32  $refId = $DIC->http()->wrapper()->query()->retrieve('ref_id', $DIC->refinery()->kindlyTo()->int());
33  $in = file_get_contents("php://input");
34  $data = json_decode($in);
35  $user_id = (int) $data->p;
36 
37  header('Content-Type: text/plain; charset=UTF-8');
38 
39  $rval = self::storeJsApiCmi($user_id, $obj_id, $data);
40  if ($rval != true) {
41  print("storeJsApiCmi failed");
42  } else {
43  $rval = self::syncGlobalStatus($user_id, $obj_id, $refId, $data, $data->now_global_status);
44  if ($rval != true) {
45  print("syncGlobalStatus failed");
46  }
47  }
48  if ($rval == true) {
49  print("ok");
50  }
51  }
52 
53  public static function storeJsApiCmi(int $user_id, int $obj_id, object $data): bool
54  {
55  global $DIC;
56  $ilLog = ilLoggerFactory::getLogger('sahs');
57  $ilDB = $DIC->database();
58 
59  $b_updateStatus = false;
60  $i_score_max = 0;
61  $i_score_raw = 0;
62 
63  $aa_data = array();
64  // if (is_array($_POST["S"])) {
65  // foreach($_POST["S"] as $key => $value) {
66  // $aa_data[] = array("sco_id" => $value, "left" => $_POST["L"][$key], "right" => $_POST["R"][$key]);
67  // }
68  // }
69  foreach ($data->cmi as $value) {
70  $aa_data[] = array("sco_id" => (int) $value[0],
71  "left" => $value[1],
72  "right" => $value[2]
73  );
74  // $aa_data[] = array("sco_id" => (int) $data->cmi[$i][0], "left" => $data->cmi[$i][1], "right" => rawurldecode($data->cmi[$i][2]));
75  }
76 
77  if ($obj_id <= 1) {
78  $ilLog->write("ScormAicc: storeJsApi: Error: No valid obj_id given.");
79  } else {
80  foreach ($aa_data as $a_data) {
81  $set = $ilDB->queryF(
82  '
83  SELECT rvalue FROM scorm_tracking
84  WHERE user_id = %s
85  AND sco_id = %s
86  AND lvalue = %s
87  AND obj_id = %s',
88  array('integer', 'integer', 'text', 'integer'),
89  array($user_id, $a_data["sco_id"], $a_data["left"], $obj_id)
90  );
91  if ($rec = $ilDB->fetchAssoc($set)) {
92  if ($a_data["left"] === 'cmi.core.lesson_status' && $a_data["right"] != $rec["rvalue"]) {
93  $b_updateStatus = true;
94  }
95  $ilDB->update(
96  'scorm_tracking',
97  array(
98  'rvalue' => array('clob', $a_data["right"]),
99  'c_timestamp' => array('timestamp', ilUtil::now())
100  ),
101  array(
102  'user_id' => array('integer', $user_id),
103  'sco_id' => array('integer', $a_data["sco_id"]),
104  'lvalue' => array('text', $a_data["left"]),
105  'obj_id' => array('integer', $obj_id)
106  )
107  );
108  $ilLog->debug("ScormAicc: storeJsApi Updated - L:" . $a_data["left"] . ",R:" .
109  $a_data["right"] . " for obj_id:" . $obj_id . ",sco_id:" . $a_data["sco_id"] . ",user_id:" . $user_id);
110  } else {
111  if ($a_data["left"] === 'cmi.core.lesson_status') {
112  $b_updateStatus = true;
113  }
114  $ilDB->insert('scorm_tracking', array(
115  'obj_id' => array('integer', $obj_id),
116  'user_id' => array('integer', $user_id),
117  'sco_id' => array('integer', $a_data["sco_id"]),
118  'lvalue' => array('text', $a_data["left"]),
119  'rvalue' => array('clob', $a_data["right"]),
120  'c_timestamp' => array('timestamp', ilUtil::now())
121  ));
122  $ilLog->debug("ScormAicc: storeJsApi Inserted - L:" . $a_data["left"] . ",R:" .
123  $a_data["right"] . " for obj_id:" . $obj_id . ",sco_id:" . $a_data["sco_id"] . ",user_id:" . $user_id);
124  }
125  if ($a_data["left"] === 'cmi.core.score.max') {
126  $i_score_max = $a_data["right"];
127  }
128  if ($a_data["left"] === 'cmi.core.score.raw') {
129  $i_score_raw = $a_data["right"];
130  }
131  }
132  // mantis #30293
133  if ($i_score_max > 0 && $i_score_raw > 0) {
134  if (count(ilSCORMObject::_lookupPresentableItems($obj_id)) == 1) {
136  $obj_id,
137  $user_id,
138  ($i_score_raw / $i_score_max) * 100
139  );
140  }
141  }
142  }
143 
144  // update status
145  // if ($b_updateStatus == true) {
146  // include_once("../components/ILIAS/Tracking/classes/class.ilLPStatusWrapper.php");
147  // ilLPStatusWrapper::_updateStatus($obj_id, $user_id);
148  // }
149 
150  return true;
151  }
152 
153  public static function syncGlobalStatus(int $userId, int $packageId, int $refId, object $data, ?int $new_global_status): bool
154  {
155  global $DIC;
156  $ilDB = $DIC->database();
157  $ilLog = ilLoggerFactory::getLogger('sahs');
158  $saved_global_status = $data->saved_global_status;
159  $ilLog->write("saved_global_status=" . $saved_global_status);
160 
161  // get attempts
162  if (!isset($data->packageAttempts)) {
163  $val_set = $ilDB->queryF(
164  'SELECT package_attempts FROM sahs_user WHERE obj_id = %s AND user_id = %s',
165  array('integer', 'integer'),
166  array($packageId, $userId)
167  );
168  $val_rec = $ilDB->fetchAssoc($val_set);
169  $attempts = $val_rec["package_attempts"];
170  } else {
171  $attempts = $data->packageAttempts;
172  }
173  if ($attempts == null) {
174  $attempts = 1;
175  }
176 
177  //update percentage_completed, sco_total_time_sec,status in sahs_user
178  $totalTime = (int) $data->totalTimeCentisec;
179  $totalTime = round($totalTime / 100);
180  $ilDB->queryF(
181  'UPDATE sahs_user SET last_visited=%s, last_access = %s, sco_total_time_sec=%s, status=%s, percentage_completed=%s, package_attempts=%s WHERE obj_id = %s AND user_id = %s',
182  array('text', 'timestamp', 'integer', 'integer', 'integer', 'integer', 'integer', 'integer'),
183  array($data->last_visited,
184  date('Y-m-d H:i:s'),
185  $totalTime,
186  $new_global_status,
187  $data->percentageCompleted,
188  $attempts,
189  $packageId,
190  $userId
191  )
192  );
193 
194  // self::ensureObjectDataCacheExistence();
195  global $DIC;
196  $ilObjDataCache = $DIC['ilObjDataCache'];
197  //workaround if $row->read_count == null TODO ERASE
198  try {
200  "sahs",
201  $refId,
202  $packageId,
203  $userId,
204  false,
205  $attempts,
206  $totalTime
207  );
208  } catch (\Exception $exception) {
210  "sahs",
211  $refId,
212  $packageId,
213  $userId,
214  false,
215  null,
216  $totalTime
217  );
218  }
219 
220  //end sync access number and time in read event table
221 
222  // update learning progress
223  if ($new_global_status != null) {
224  ilLPStatus::writeStatus($packageId, $userId, $new_global_status, $data->percentageCompleted);
225 
226  // here put code for soap to MaxCMS e.g. when if($saved_global_status != $new_global_status)
227  }
228  return true;
229  }
230 
231  public static function _insertTrackData(int $a_sahs_id, string $a_lval, string $a_rval, int $a_obj_id): void
232  {
233  global $DIC;
234  $ilDB = $DIC->database();
235  $ilUser = $DIC->user();
236 
237  $ilDB->insert('scorm_tracking', array(
238  'obj_id' => array('integer', $a_obj_id),
239  'user_id' => array('integer', $ilUser->getId()),
240  'sco_id' => array('integer', $a_sahs_id),
241  'lvalue' => array('text', $a_lval),
242  'rvalue' => array('clob', $a_rval),
243  'c_timestamp' => array('timestamp', ilUtil::now())
244  ));
245 
246  if ($a_lval === "cmi.core.lesson_status") {
247  ilLPStatusWrapper::_updateStatus($a_obj_id, $ilUser->getId());
248  }
249  }
250 
251  //erase later see ilSCORM2004StoreData
255  public static function _getCompleted(object $scorm_item_id, int $a_obj_id): array
256  {
257  global $DIC;
258  $ilDB = $DIC->database();
259 
260  $user_ids = [];
261 
262  if (is_array($scorm_item_id)) {
263  $in = $ilDB->in('sco_id', $scorm_item_id, false, 'integer');
264 
265  $res = $ilDB->queryF(
266  'SELECT DISTINCT(user_id) FROM scorm_tracking
267  WHERE ' . $in . '
268  AND obj_id = %s
269  AND lvalue = %s
270  AND (' . $ilDB->like('rvalue', 'clob', 'completed') . ' OR ' . $ilDB->like(
271  'rvalue',
272  'clob',
273  'passed'
274  ) . ')',
275  array('integer', 'text'),
276  array($a_obj_id, 'cmi.core.lesson_status')
277  );
278  } else {
279  $res = $ilDB->queryF(
280  'SELECT DISTINCT(user_id) FROM scorm_tracking
281  WHERE sco_id = %s
282  AND obj_id = %s
283  AND lvalue = %s
284  AND (' . $ilDB->like('rvalue', 'clob', 'completed') . ' OR ' . $ilDB->like(
285  'rvalue',
286  'clob',
287  'passed'
288  ) . ')',
289  array('integer', 'integer', 'text'),
290  array($scorm_item_id, $a_obj_id, 'cmi.core.lesson_status')
291  );
292  }
293 
294  while ($row = $ilDB->fetchObject($res)) {
295  $user_ids[] = $row->user_id;
296  }
297  return $user_ids;
298  }
299 
300  public static function _getCollectionStatus(?array $a_scos, int $a_obj_id, int $a_user_id): string
301  {
302  global $DIC;
303  $ilDB = $DIC->database();
304 
305  $status = "not_attempted";
306 
307  if (is_array($a_scos)) {
308  $in = $ilDB->in('sco_id', $a_scos, false, 'integer');
309 
310  $res = $ilDB->queryF(
311  'SELECT sco_id, rvalue FROM scorm_tracking
312  WHERE ' . $in . '
313  AND obj_id = %s
314  AND lvalue = %s
315  AND user_id = %s',
316  array('integer', 'text', 'integer'),
317  array($a_obj_id, 'cmi.core.lesson_status', $a_user_id)
318  );
319 
320  $cnt = 0;
321  $completed = true;
322  $failed = false;
323  while ($rec = $ilDB->fetchAssoc($res)) {
324  if ($rec["rvalue"] === "failed") {
325  $failed = true;
326  }
327  if ($rec["rvalue"] !== "completed" && $rec["rvalue"] !== "passed") {
328  $completed = false;
329  }
330  $cnt++;
331  }
332  if ($cnt > 0) {
333  $status = "in_progress";
334  }
335  if ($completed && $cnt == count($a_scos)) {
336  $status = "completed";
337  }
338  if ($failed) {
339  $status = "failed";
340  }
341  }
342  return $status;
343  }
344 
345  public static function _countCompleted(?array $a_scos, int $a_obj_id, int $a_user_id): int
346  {
347  global $DIC;
348  $ilDB = $DIC->database();
349  $cnt = 0;
350 
351  if (is_array($a_scos)) {
352  $in = $ilDB->in('sco_id', $a_scos, false, 'integer');
353 
354  $res = $ilDB->queryF(
355  'SELECT sco_id, rvalue FROM scorm_tracking
356  WHERE ' . $in . '
357  AND obj_id = %s
358  AND lvalue = %s
359  AND user_id = %s',
360  array('integer', 'text', 'integer'),
361  array($a_obj_id, 'cmi.core.lesson_status', $a_user_id)
362  );
363 
364  while ($rec = $ilDB->fetchAssoc($res)) {
365  if ($rec["rvalue"] === "completed" || $rec["rvalue"] === "passed") {
366  $cnt++;
367  }
368  }
369  }
370  return $cnt;
371  }
372 
377  public static function lookupLastAccessTimes(int $a_obj_id): array
378  {
379  global $DIC;
380  $ilDB = $DIC->database();
381  $users = array();
382 
383  $query = 'SELECT user_id, MAX(c_timestamp) tst ' .
384  'FROM scorm_tracking ' .
385  'WHERE obj_id = ' . $ilDB->quote($a_obj_id, 'integer') . ' ' .
386  'GROUP BY user_id';
387  $res = $ilDB->query($query);
388 
389  while ($row = $res->fetchRow(ilDBConstants::FETCHMODE_OBJECT)) {
390  $users[$row->user_id] = $row->tst;
391  }
392  return $users;
393  }
394 
398  public static function _getTrackedUsers(int $a_obj_id): array
399  {
400  global $DIC;
401  $ilDB = $DIC->database();
402  // $ilLog = ilLoggerFactory::getLogger('sahs');
403 
404  $res = $ilDB->queryF(
405  'SELECT DISTINCT user_id FROM scorm_tracking
406  WHERE obj_id = %s
407  AND lvalue = %s',
408  array('integer', 'text'),
409  array($a_obj_id, 'cmi.core.lesson_status')
410  );
411 
412  $users = array();
413  while ($row = $ilDB->fetchAssoc($res)) {
414  $users[] = $row["user_id"];
415  }
416  return $users;
417  }
418 
422  public static function _getFailed(object $scorm_item_id, int $a_obj_id): array
423  {
424  global $DIC;
425  $ilDB = $DIC->database();
426  $user_ids = [];
427 
428  if (is_array($scorm_item_id)) {
429  $in = $ilDB->in('sco_id', $scorm_item_id, false, 'integer');
430 
431  $res = $ilDB->queryF(
432  '
433  SELECT DISTINCT(user_id) FROM scorm_tracking
434  WHERE ' . $in . '
435  AND obj_id = %s
436  AND lvalue = %s
437  AND ' . $ilDB->like('rvalue', 'clob', 'failed') . ' ',
438  array('integer', 'text'),
439  array($a_obj_id, 'cmi.core.lesson_status')
440  );
441  } else {
442  $res = $ilDB->queryF(
443  '
444  SELECT DISTINCT(user_id) FROM scorm_tracking
445  WHERE sco_id = %s
446  AND obj_id = %s
447  AND lvalue = %s
448  AND ' . $ilDB->like('rvalue', 'clob', 'failed') . ' ',
449  array('integer', 'integer', 'text'),
450  array($scorm_item_id, $a_obj_id, 'cmi.core.lesson_status')
451  );
452  }
453 
454  while ($row = $ilDB->fetchObject($res)) {
455  $user_ids[] = $row->user_id;
456  }
457  return $user_ids;
458  }
459 
464  public static function _getCountCompletedPerUser(array $a_scorm_item_ids, int $a_obj_id): array
465  {
466  global $DIC;
467  $ilDB = $DIC->database();
468  $users = [];
469 
470  $in = $ilDB->in('sco_id', $a_scorm_item_ids, false, 'integer');
471 
472  // Why does this query use a like search against "passed" and "failed"
473  //because it's clob and we support Oracle
474  $res = $ilDB->queryF(
475  '
476  SELECT user_id, COUNT(user_id) completed FROM scorm_tracking
477  WHERE ' . $in . '
478  AND obj_id = %s
479  AND lvalue = %s
480  AND (' . $ilDB->like('rvalue', 'clob', 'completed') . ' OR ' . $ilDB->like('rvalue', 'clob', 'passed') . ')
481  GROUP BY user_id',
482  array('integer', 'text'),
483  array($a_obj_id, 'cmi.core.lesson_status')
484  );
485  while ($row = $ilDB->fetchObject($res)) {
486  $users[$row->user_id] = $row->completed;
487  }
488  return $users;
489  }
490  //not correct because of assets!
495  public static function _getProgressInfo(array $sco_item_ids, int $a_obj_id): array
496  {
497  global $DIC;
498  $ilDB = $DIC->database();
499 
500  $in = $ilDB->in('sco_id', $sco_item_ids, false, 'integer');
501 
502  $res = $ilDB->queryF(
503  '
504  SELECT * FROM scorm_tracking
505  WHERE ' . $in . '
506  AND obj_id = %s
507  AND lvalue = %s ',
508  array('integer', 'text'),
509  array($a_obj_id, 'cmi.core.lesson_status')
510  );
511 
512  $info['completed'] = array();
513  $info['failed'] = array();
514 
515  $user_ids = array();
516  while ($row = $ilDB->fetchObject($res)) {
517  switch ($row->rvalue) {
518  case 'completed':
519  case 'passed':
520  $info['completed'][$row->sco_id][] = $row->user_id;
521  $user_ids[$row->sco_id][] = $row->user_id;
522  break;
523 
524  case 'failed':
525  $info['failed'][$row->sco_id][] = $row->user_id;
526  $user_ids[$row->sco_id][] = $row->user_id;
527  break;
528  }
529  }
530  $info['in_progress'] = ilObjSCORMTracking::_getInProgress($sco_item_ids, $a_obj_id, $user_ids);
531 
532  return $info;
533  }
534 
538  public static function _getInProgress(array|int $scorm_item_id, int $a_obj_id, ?array $a_blocked_user_ids = null): array
539  {
540  global $DIC;
541  $ilDB = $DIC->database();
542 
543  if (is_array($scorm_item_id)) {
544  $in = $ilDB->in('sco_id', $scorm_item_id, false, 'integer');
545 
546  $res = $ilDB->queryF(
547  'SELECT user_id,sco_id FROM scorm_tracking
548  WHERE ' . $in . '
549  AND obj_id = %s
550  GROUP BY user_id, sco_id',
551  array('integer'),
552  array($a_obj_id)
553  );
554  } else {
555  $res = $ilDB->queryF(
556  'SELECT user_id,sco_id FROM scorm_tracking
557  WHERE sco_id = %s
558  AND obj_id = %s',
559  array('integer', 'integer'),
560  array($scorm_item_id, $a_obj_id)
561  );
562  }
563 
564  $in_progress = array();
565 
566  while ($row = $ilDB->fetchObject($res)) {
567  // #15061 - see _getProgressInfo()
568  if (!($a_blocked_user_ids &&
569  isset($a_blocked_user_ids[$row->sco_id]) &&
570  is_array($a_blocked_user_ids[$row->sco_id]) &&
571  in_array($row->user_id, $a_blocked_user_ids[$row->sco_id]))) {
572  $in_progress[$row->sco_id][] = $row->user_id;
573  }
574  }
575  return $in_progress;
576  }
577 
578  public static function scorm12PlayerUnload(): void
579  {
580  global $DIC;
581  $ilDB = $DIC->database();
582  $user_id = $DIC->http()->wrapper()->query()->retrieve('p', $DIC->refinery()->kindlyTo()->int());
583  $ref_id = $DIC->http()->wrapper()->query()->retrieve('ref_id', $DIC->refinery()->kindlyTo()->int());
584  $obj_id = $DIC->http()->wrapper()->query()->retrieve('package_id', $DIC->refinery()->kindlyTo()->int());
585  if ($obj_id <= 1) {
586  $GLOBALS['DIC']['ilLog']->write(__METHOD__ . ' no valid obj_id');
587  } else {
588  $last_visited = "";
589  if ($DIC->http()->wrapper()->query()->has('last_visited')) {
590  $last_visited = $DIC->http()->wrapper()->query()->retrieve('last_visited', $DIC->refinery()->kindlyTo()->string());
591  }
592 
593  $endDate = date(
594  'Y-m-d H:i:s',
595  mktime((int) date('H'), (int) date('i') + 5, (int) date('s'), (int) date('m'), (int) date('d'), (int) date('Y'))
596  );
597  $ilDB->manipulateF(
598  'UPDATE sahs_user
599  SET last_visited = %s, hash_end =%s, last_access = %s
600  WHERE obj_id = %s AND user_id = %s',
601  array('text', 'timestamp', 'timestamp', 'integer', 'integer'),
602  array($last_visited, $endDate, date('Y-m-d H:i:s'), $obj_id, $user_id)
603  );
604  // update time and numbers of attempts in change event
605  //NOTE: here it is correct (not count of commit with changed values); be careful to performance issues
607  }
608  header('Content-Type: text/plain; charset=UTF-8');
609  print("");
610  }
611 
612  public static function checkIfAllowed(int $packageId, int $userId, int $hash): void
613  {
614  global $DIC;
615  $ilDB = $DIC->database();
616  $res = $ilDB->queryF(
617  'select hash from sahs_user where obj_id=%s AND user_id=%s AND hash_end>%s',
618  array('integer', 'integer', 'timestamp'),
619  array($packageId, $userId, date('Y-m-d H:i:s'))
620  );
621  $rowtmp = $ilDB->fetchAssoc($res);
622  if (! ($rowtmp && $rowtmp['hash'] == $hash)) {
623  //output used by api
624  die("not allowed");
625  }
626  }
627 
628  public static function _syncReadEvent(int $a_obj_id, int $a_user_id, string $a_type, int $a_ref_id): void
629  {
630  global $DIC;
631  $ilDB = $DIC->database();
632  $ilLog = ilLoggerFactory::getLogger('sahs');
633  $val_set = $ilDB->queryF(
634  'SELECT package_attempts, total_time_sec, sco_total_time_sec, time_from_lms FROM sahs_user, sahs_lm '
635  . 'WHERE sahs_user.obj_id = %s AND sahs_user.user_id = %s AND sahs_user.obj_id = sahs_lm.id',
636  array('integer', 'integer'),
637  array($a_obj_id, $a_user_id)
638  );
639 
640  $val_rec = $ilDB->fetchAssoc($val_set);
641 
642  if ($val_rec["package_attempts"] == null) {
643  $val_rec["package_attempts"] = "";
644  }
645  $attempts = $val_rec["package_attempts"];
646 
647  $time = (int) $val_rec["sco_total_time_sec"];
648 
649  // get learning time for old ILIAS-Versions
650  if ($time == 0) {
651  $sco_set = $ilDB->queryF(
652  '
653  SELECT sco_id, rvalue FROM scorm_tracking
654  WHERE obj_id = %s
655  AND user_id = %s
656  AND lvalue = %s
657  AND sco_id <> %s',
658  array('integer', 'integer', 'text', 'integer'),
659  array($a_obj_id, $a_user_id, 'cmi.core.total_time', 0)
660  );
661 
662  while ($sco_rec = $ilDB->fetchAssoc($sco_set)) {
663  $tarr = explode(":", $sco_rec["rvalue"]);
664  $sec = (int) $tarr[2] + (int) $tarr[1] * 60 +
665  (int) substr($tarr[0], strlen($tarr[0]) - 3) * 60 * 60;
666  $time += $sec;
667  }
668  }
669  ilChangeEvent::_recordReadEvent($a_type, $a_ref_id, $a_obj_id, $a_user_id, false, $attempts, $time);
670  }
671 } // END class.ilObjSCORMTracking
static _getInProgress(array|int $scorm_item_id, int $a_obj_id, ?array $a_blocked_user_ids=null)
$res
Definition: ltiservices.php:66
static storeJsApiCmi(int $user_id, int $obj_id, object $data)
static getLogger(string $a_component_id)
Get component logger.
static _lookupPresentableItems(int $a_slm_id)
Count number of presentable SCOs/Assets of SCORM learning module.
static _getCollectionStatus(?array $a_scos, int $a_obj_id, int $a_user_id)
$packageId
Definition: storeScorm.php:24
static _getFailed(object $scorm_item_id, int $a_obj_id)
like necessary because of Oracle
Class ilObjSCORMTracking.
static _getProgressInfo(array $sco_item_ids, int $a_obj_id)
Get info about.
$refId
Definition: xapitoken.php:58
static lookupLastAccessTimes(int $a_obj_id)
Lookup last acccess time for all users of a scorm module.
static now()
Return current timestamp in Y-m-d H:i:s format.
static _insertTrackData(int $a_sahs_id, string $a_lval, string $a_rval, int $a_obj_id)
while($session_entry=$r->fetchRow(ilDBConstants::FETCHMODE_ASSOC)) return null
$ref_id
Definition: ltiauth.php:65
$GLOBALS["DIC"]
Definition: wac.php:53
static _recordReadEvent(string $a_type, int $a_ref_id, int $obj_id, int $usr_id, bool $isCatchupWriteEvents=true, $a_ext_rc=null, $a_ext_time=null)
global $DIC
Definition: shib_login.php:22
static _getTrackedUsers(int $a_obj_id)
Get all tracked users.
static syncGlobalStatus(int $userId, int $packageId, int $refId, object $data, ?int $new_global_status)
static handleOutcomeWithoutLP(int $a_obj_id, int $a_usr_id, ?float $a_percentage)
static _countCompleted(?array $a_scos, int $a_obj_id, int $a_user_id)
static _getCountCompletedPerUser(array $a_scorm_item_ids, int $a_obj_id)
Get users who have status completed or passed.
static checkIfAllowed(int $packageId, int $userId, int $hash)
static writeStatus(int $a_obj_id, int $a_user_id, int $a_status, int $a_percentage=0, bool $a_force_per=false, ?int &$a_old_status=self::LP_STATUS_NOT_ATTEMPTED_NUM)
Write status for user and object.
static _getCompleted(object $scorm_item_id, int $a_obj_id)
like necessary because of Oracle
static _syncReadEvent(int $a_obj_id, int $a_user_id, string $a_type, int $a_ref_id)
header()
expected output: > ILIAS shows the rendered Component.
Definition: header.php:29
static _updateStatus(int $a_obj_id, int $a_usr_id, ?object $a_obj=null, bool $a_percentage=false, bool $a_force_raise=false)