19 declare(strict_types=0);
45 $this->tree = $DIC->repositoryTree();
46 $this->db = $DIC->database();
48 $this->container_obj_id = $a_container_obj_id;
50 $this->container_ref_id = end($refs);
55 if (array_key_exists($a_container_obj_id, self::$instances)) {
56 return self::$instances[$a_container_obj_id];
58 return self::$instances[$a_container_obj_id] =
new self($a_container_obj_id);
76 public function init(): void
78 if ($this->initialized) {
83 $this->initialized =
true;
88 foreach ($this->
getItemIds() as $item_ref_id) {
94 $user_item =
new ilTimingUser($item[
'obj_id'], $a_usr_id);
95 $user_start = clone $sub_date;
98 $user_end = clone $sub_date;
99 $user_end->increment(
IL_CAL_DAY, $item[
'suggestion_end_rel']);
101 $user_item->update();
107 $query =
'DELETE FROM crs_timings_user WHERE ' . $this->db->in(
113 'AND usr_id = ' . $this->db->quote($a_usr_id,
'integer');
114 $this->db->manipulate($query);
126 foreach (self::lookupTimings($a_user_ids, $meta,
true) as $user_ids) {
131 return array_values(
$res);
139 public static function lookupTimings(array $a_user_ids, ?array &$a_meta =
null,
bool $a_only_exceeded =
true): array
143 $ilDB = $DIC->database();
144 $logger = $DIC->logger()->crs();
151 $logger->debug(
'Course membership assignments');
155 $query =
'SELECT crsi.obj_id sub_ref_id, oref.ref_id, oref.obj_id, crsi.suggestion_start' .
156 ',crsi.suggestion_end,crsi.changeable, crss.timing_mode' .
157 ' FROM crs_settings crss' .
158 ' JOIN object_reference oref ON (oref.obj_id = crss.obj_id AND oref.deleted IS NULL) ' .
159 ' JOIN crs_items crsi ON (crsi.parent_id = oref.ref_id)' .
160 ' JOIN object_reference iref ON (crsi.obj_id = iref.ref_id AND iref.deleted IS NULL) ' .
162 ' AND ' .
$ilDB->in(
'crss.obj_id', array_keys($course_members_map),
false,
'integer') .
165 $logger->debug($query);
167 $set =
$ilDB->query($query);
168 $user_relevant = $course_map = $course_parent_map = [];
169 while ($row =
$ilDB->fetchAssoc($set)) {
170 $obj_id = (
int) $row[
'obj_id'];
171 $sub_ref_id = (
int) $row[
'sub_ref_id'];
172 $mode = (
int) $row[
'timing_mode'];
175 $course_parent_map[(
int) $row[
'sub_ref_id']] = (
int) $row[
'obj_id'];
176 $course_map[(
int) $row[
'obj_id']] = (
int) $row[
'ref_id'];
179 if (is_array($a_meta)) {
182 if (in_array($user_id, $course_members_map[$obj_id])) {
183 $a_meta[
$user_id][$sub_ref_id] = array(
184 'parent' => $row[
'ref_id']
193 if (is_array($a_meta)) {
196 if (in_array($user_id, $course_members_map[$obj_id])) {
197 $a_meta[
$user_id][$sub_ref_id][
'start'] = (
int) $row[
'suggestion_start'];
198 $a_meta[
$user_id][$sub_ref_id][
'end'] = (
int) $row[
'suggestion_end'];
204 ($a_only_exceeded && ((
int) $row[
'suggestion_end'] && (
int) $row[
'suggestion_end'] < $now)) ||
205 (!$a_only_exceeded && ((
int) $row[
'suggestion_start'] && (
int) $row[
'suggestion_start'] < $now))
209 if (in_array($user_id, $course_members_map[$obj_id])) {
217 if ($row[
'changeable'] ||
219 $user_relevant[] = $sub_ref_id;
223 if ($user_relevant !== []) {
225 $query =
'SELECT * FROM crs_timings_user' .
226 ' WHERE ' .
$ilDB->in(
'usr_id', $a_user_ids,
false,
'integer') .
227 ' AND ' .
$ilDB->in(
'ref_id', $user_relevant,
false,
'integer');
228 $set =
$ilDB->query($query);
229 while ($row =
$ilDB->fetchAssoc($set)) {
234 $crs_obj_id = $course_parent_map[
$ref_id];
235 if (!in_array(
$user_id, $course_members_map[$crs_obj_id])) {
240 if (is_array($a_meta)) {
246 ($a_only_exceeded && (
int) $row[
'ssend'] && (
int) $row[
'ssend'] < $now) ||
247 (!$a_only_exceeded && (
int) $row[
'sstart'] && (
int) $row[
'sstart'] < $now)
259 if (count(
$res[$ref_id]) === 0) {
260 if (isset(
$res[
'ref_id']) && !count(
$res[
'ref_id'])) {
261 unset(
$res[$ref_id]);
270 $invalid_lp = self::getObjectsWithInactiveLP(array_keys(
$res), $obj_map);
272 foreach (array_keys(
$res) as $ref_id) {
274 if (in_array($ref_id, $invalid_lp)) {
279 if ($user_ids !== []) {
288 if (
$res[$ref_id] === []) {
289 unset(
$res[$ref_id]);
295 foreach (
$res as $ref_id => $user_ids) {
297 if ($user_ids === [] && isset(
$res[
'ref_id'])) {
298 unset(
$res[$ref_id]);
300 $crs_obj_id = $course_parent_map[
$ref_id];
301 $crs_ref_id = $course_map[$crs_obj_id];
302 if (!array_key_exists($crs_ref_id,
$res)) {
303 $res[$crs_ref_id] = $user_ids;
305 $res[$crs_ref_id] = array_unique(array_merge($user_ids,
$res[$crs_ref_id]));
320 $ilDB = $DIC->database();
323 $query =
'SELECT oref.ref_id, oref.obj_id, od.type' .
324 ' FROM object_reference oref' .
325 ' JOIN object_data od ON (oref.obj_id = od.obj_id)' .
326 ' WHERE ' .
$ilDB->in(
'oref.ref_id', $a_ref_ids,
false,
'integer');
327 $set =
$ilDB->query($query);
328 $item_map = $item_types = array();
329 while ($row =
$ilDB->fetchAssoc($set)) {
330 $item_map[(
int) $row[
'ref_id']] = (
int) $row[
'obj_id'];
331 $item_types[(
int) $row[
'obj_id']] = $row[
'type'];
334 $a_obj_map = $item_map;
339 $type_modes = array();
340 foreach ($a_ref_ids as
$ref_id) {
342 $type = $item_types[$obj_id];
350 if (array_key_exists($obj_id, $db_modes)) {
351 $mode = $db_modes[$obj_id];
354 if (!array_key_exists($type, $type_modes)) {
356 $type_modes[$type] = $type_modes[$type]->getDefaultMode();
358 $mode = $type_modes[$type];
const IL_CRS_VIEW_TIMING_ABSOLUTE
static isSupportedObjectType(string $type)
static getInstanceByContainerId(int $a_container_obj_id)
static _getAllReferences(int $id)
get all reference ids for object ID
TableGUI class for timings administration.
increment(string $a_type, int $a_count=1)
static _lookupCompletedForObject(int $a_obj_id, ?array $a_user_ids=null)
Get completed users for object.
while($session_entry=$r->fetchRow(ilDBConstants::FETCHMODE_ASSOC)) return null
handleNewMembership(int $a_usr_id, ilDateTime $sub_date)
static preloadData(array $ref_ids)
Preload data to internal cache.
__construct(int $a_container_obj_id)
Singleton constructor.
static lookupTimings(array $a_user_ids, ?array &$a_meta=null, bool $a_only_exceeded=true)
Lookup references, users with exceeded timings.
const IL_CRS_VIEW_TIMING_RELATIVE
static _lookupDBModeForObjects(array $a_obj_ids)
const LP_MODE_DEACTIVATED
static getItem(int $ref_id)
static lookupTimingsExceededByUser(array $a_user_ids)
Check if users currently exceeded ANY object.
static getObjectsWithInactiveLP(array $a_ref_ids, ?array &$a_obj_map=null)
Check object LP modes.
static getUserMembershipAssignmentsByType(array $a_user_ids, array $a_type, bool $a_only_member_roles)
Get user membership assignments by type.
handleUnsubscribe(int $a_usr_id)
static getInstance(int $obj_id)