ILIAS  trunk Revision v11.0_alpha-1689-g66c127b4ae8
All Data Structures Namespaces Files Functions Variables Enumerations Enumerator Modules Pages
class.ilExAssignment.php
Go to the documentation of this file.
1 <?php
2 
28 
35 {
39  public const TYPE_UPLOAD = 1;
40  public const TYPE_BLOG = 2;
41  public const TYPE_PORTFOLIO = 3;
42  public const TYPE_UPLOAD_TEAM = 4;
43  public const TYPE_TEXT = 5;
44  public const TYPE_WIKI_TEAM = 6;
45 
46  public const FEEDBACK_DATE_DEADLINE = 1;
47  public const FEEDBACK_DATE_SUBMISSION = 2;
48  public const FEEDBACK_DATE_CUSTOM = 3;
49 
50  public const PEER_REVIEW_VALID_NONE = 1;
51  public const PEER_REVIEW_VALID_ONE = 2;
52  public const PEER_REVIEW_VALID_ALL = 3;
53 
54  public const TEAMS_FORMED_BY_PARTICIPANTS = 0;
55  public const TEAMS_FORMED_BY_TUTOR = 1;
56  public const TEAMS_FORMED_BY_RANDOM = 2;
57  public const TEAMS_FORMED_BY_ASSIGNMENT = 3;
58 
59  public const DEADLINE_ABSOLUTE = 0;
60  public const DEADLINE_RELATIVE = 1;
61  public const DEADLINE_ABSOLUTE_INDIVIDUAL = 2;
62  protected \ILIAS\Exercise\InternalDomainService $domain;
64 
65  protected ilDBInterface $db;
66  protected ilLanguage $lng;
67  protected ilObjUser $user;
70 
71  protected int $id = 0;
72  protected int $exc_id = 0;
73  protected int $type = 0;
74  protected ?int $start_time = null;
75  protected ?int $deadline = null;
76  protected ?int $deadline2 = null;
77  protected string $instruction = "";
78  protected string $title = "";
79  protected bool $mandatory = false;
80  protected int $order_nr = 0;
81  protected bool $peer = false; // peer review activated
82  protected int $peer_min = 0;
83  protected int $peer_unlock = 0;
84  protected int $peer_dl = 0;
85  protected int $peer_valid; // passed after submission, one or all peer feedbacks
86  protected bool $peer_file = false;
87  protected bool $peer_personal = false; // personalised peer review
88  protected ?int $peer_char = null; // minimun number of characters for peer review
89  protected bool $peer_text = false;
90  protected bool $peer_rating = false;
91  protected int $peer_crit_cat = 0;
92  protected ?string $feedback_file = null;
93  protected bool $feedback_cron = false;
94  protected int $feedback_date = 0;
95  protected int $feedback_date_custom = 0;
96  protected bool $team_tutor = false;
97  protected ?int $max_file = null;
98  protected int $portfolio_template = 0;
99  protected int $min_char_limit = 0;
100  protected int $max_char_limit = 0;
103  protected int $deadline_mode = 0;
104  protected int $relative_deadline = 0;
105  protected int $rel_deadline_last_subm = 0;
106  protected array $member_status = [];
107  protected ilLogger $log;
108  protected ?int $crit_cat = 0;
109 
114  public function __construct($a_id = 0)
115  {
116  global $DIC;
117 
118  $this->domain = $DIC->exercise()->internal()->domain();
119  $this->db = $DIC->database();
120  $this->lng = $DIC->language();
121  $this->user = $DIC->user();
122  $this->app_event_handler = $DIC["ilAppEventHandler"];
123  $this->types = ilExAssignmentTypes::getInstance();
124  $this->access = $DIC->access();
125  $this->domain = $DIC->exercise()->internal()->domain();
126 
127  $this->setType(self::TYPE_UPLOAD);
128  $this->setFeedbackDate(self::FEEDBACK_DATE_DEADLINE);
129 
130  $this->log = ilLoggerFactory::getLogger("exc");
131 
132  if ($a_id > 0) {
133  $this->setId($a_id);
134  $this->read();
135  }
136  $this->string_transform = $DIC->refinery()
137  ->string();
138  }
139 
145  public static function getInstancesByExercise(int $a_exc_id): array
146  {
147  global $DIC;
148 
149  $ilDB = $DIC->database();
150 
151  $set = $ilDB->query("SELECT * FROM exc_assignment " .
152  " WHERE exc_id = " . $ilDB->quote($a_exc_id, "integer") .
153  " ORDER BY order_nr");
154  $data = array();
155 
156  $order_val = 10;
157  while ($rec = $ilDB->fetchAssoc($set)) {
158  // ???
159  $rec["order_val"] = $order_val;
160 
161  $ass = new self();
162  $ass->initFromDB($rec);
163  $data[] = $ass;
164 
165  $order_val += 10;
166  }
167 
168  return $data;
169  }
170 
176  public static function instructionFileGetFileOrderData(
177  array $a_file_data,
178  int $a_ass_id
179  ): array {
180  global $DIC;
181 
182  $db = $DIC->database();
183  $db->setLimit(1, 0);
184 
185  $result_order_val = $db->query("
186  SELECT id, order_nr
187  FROM exc_ass_file_order
188  WHERE assignment_id = {$db->quote($a_ass_id, 'integer')}
189  AND filename = {$db->quote($a_file_data['entry'], 'text')}
190  ");
191 
192  $order_val = 0;
193  $order_id = 0;
194  while ($row = $db->fetchAssoc($result_order_val)) {
195  $order_val = (int) $row['order_nr'];
196  $order_id = (int) $row['id'];
197  }
198  return array($order_val, $order_id);
199  }
200 
201  public function hasTeam(): bool
202  {
203  return $this->ass_type->usesTeams();
204  }
205 
206  public function setId(int $a_val): void
207  {
208  $this->id = $a_val;
209  }
210 
211  public function getId(): int
212  {
213  return $this->id;
214  }
215 
216  public function setExerciseId(int $a_val): void
217  {
218  $this->exc_id = $a_val;
219  }
220 
221  public function getExerciseId(): int
222  {
223  return $this->exc_id;
224  }
225 
226  public function setStartTime(?int $a_val): void
227  {
228  $this->start_time = $a_val;
229  }
230 
231  public function getStartTime(): ?int
232  {
233  return $this->start_time;
234  }
235 
236  public function setDeadline(?int $a_val): void
237  {
238  $this->deadline = $a_val;
239  }
240 
241  public function getDeadline(): ?int
242  {
243  return $this->deadline;
244  }
245 
250  public function setDeadlineMode(int $a_val): void
251  {
252  $this->deadline_mode = $a_val;
253  }
254 
255  public function getDeadlineMode(): int
256  {
257  return $this->deadline_mode;
258  }
259 
260  public function setRelativeDeadline(int $a_val): void
261  {
262  $this->relative_deadline = $a_val;
263  }
264 
265  public function getRelativeDeadline(): int
266  {
268  }
269 
270  public function setRelDeadlineLastSubmission(int $a_val): void
271  {
272  $this->rel_deadline_last_subm = $a_val;
273  }
274 
276  {
278  }
279 
280 
281  // Get individual deadline (max of common or idl (team) deadline = Official Deadline)
282  public function getPersonalDeadline(int $a_user_id): int
283  {
284  $ilDB = $this->db;
285 
286  $is_team = false;
287  if ($this->ass_type->usesTeams()) {
288  $team_id = ilExAssignmentTeam::getTeamId($this->getId(), $a_user_id);
289  if (!$team_id) {
290  // #0021043
291  return $this->getDeadline();
292  }
293  $a_user_id = $team_id;
294  $is_team = true;
295  }
296 
297  $set = $ilDB->query("SELECT tstamp FROM exc_idl" .
298  " WHERE ass_id = " . $ilDB->quote($this->getId(), "integer") .
299  " AND member_id = " . $ilDB->quote($a_user_id, "integer") .
300  " AND is_team = " . $ilDB->quote($is_team, "integer"));
301  $row = $ilDB->fetchAssoc($set);
302 
303  // use assignment deadline if no direct personal
304  return max(($row["tstamp"] ?? 0), $this->getDeadline());
305  }
306 
307  // Get last/final personal deadline (of assignment)
308  public function getLastPersonalDeadline(): int
309  {
310  $ilDB = $this->db;
311 
312  $set = $ilDB->query("SELECT MAX(tstamp) FROM exc_idl" .
313  " WHERE ass_id = " . $ilDB->quote($this->getId(), "integer"));
314  $row = $ilDB->fetchAssoc($set);
315  return $row["tstamp"] ?? 0;
316  }
317 
318  // Set extended deadline (timestamp)
319  public function setExtendedDeadline(?int $a_val): void
320  {
321  $this->deadline2 = $a_val;
322  }
323 
324  public function getExtendedDeadline(): ?int
325  {
326  return $this->deadline2;
327  }
328 
329  public function setInstruction(string $a_val): void
330  {
331  $this->instruction = $a_val;
332  }
333 
334  public function getInstruction(): string
335  {
336  return $this->instruction;
337  }
338 
339  public function getInstructionPresentation(): string
340  {
341  $inst = $this->getInstruction();
342  if (trim($inst)) {
343  $is_html = (strlen($inst) != strlen(strip_tags($inst)));
344  if (!$is_html) {
345  $inst = nl2br(
346  $this->string_transform->makeClickable()->transform($inst)
347  );
348  }
349  }
350  return $inst;
351  }
352 
353  public function setTitle(string $a_val): void
354  {
355  $this->title = $a_val;
356  }
357 
358  public function getTitle(): string
359  {
360  return $this->title;
361  }
362 
363  public function setMandatory(bool $a_val): void
364  {
365  $this->mandatory = $a_val;
366  }
367 
368  public function getMandatory(): bool
369  {
370  return $this->mandatory;
371  }
372 
373  public function setOrderNr(int $a_val): void
374  {
375  $this->order_nr = $a_val;
376  }
377 
378  public function getOrderNr(): int
379  {
380  return $this->order_nr;
381  }
382 
388  public function setType(int $a_value): void
389  {
390  if ($this->isValidType($a_value)) {
391  $this->type = $a_value;
392 
393  $this->ass_type = $this->types->getById($a_value);
394 
395  if ($this->ass_type->usesTeams()) {
396  $this->setPeerReview(false);
397  }
398  }
399  }
400 
402  {
403  return $this->ass_type;
404  }
405 
406 
411  public function getType(): int
412  {
413  return $this->type;
414  }
415 
416  public function isValidType(int $a_value): bool
417  {
418  return $this->types->isValidId($a_value);
419  }
420 
421  public function setPeerReview(bool $a_value): void
422  {
423  $this->peer = $a_value;
424  }
425 
426  public function getPeerReview(): bool
427  {
428  return $this->peer;
429  }
430 
431  public function setPeerReviewMin(int $a_value): void
432  {
433  $this->peer_min = $a_value;
434  }
435 
436  public function getPeerReviewMin(): int
437  {
438  return $this->peer_min;
439  }
440 
441  public function setPeerReviewSimpleUnlock(int $a_value)
442  {
443  $this->peer_unlock = $a_value;
444  }
445 
446  public function getPeerReviewSimpleUnlock(): int
447  {
448  return $this->peer_unlock;
449  }
450 
454  public function setPeerReviewDeadline(int $a_val): void
455  {
456  $this->peer_dl = $a_val;
457  }
458 
459  public function getPeerReviewDeadline(): int
460  {
461  return $this->peer_dl;
462  }
463 
469  public function setPeerReviewValid(int $a_value): void
470  {
471  $this->peer_valid = $a_value;
472  }
473 
474  public function getPeerReviewValid(): int
475  {
476  return $this->peer_valid;
477  }
478 
479  public function setPeerReviewRating(bool $a_val): void
480  {
481  $this->peer_rating = $a_val;
482  }
483 
484  public function hasPeerReviewRating(): bool
485  {
486  return $this->peer_rating;
487  }
488 
489  public function setPeerReviewText(bool $a_val): void
490  {
491  $this->peer_text = $a_val;
492  }
493 
494  public function hasPeerReviewText(): bool
495  {
496  return $this->peer_text;
497  }
498 
499  public function setPeerReviewFileUpload(bool $a_val): void
500  {
501  $this->peer_file = $a_val;
502  }
503 
504  public function hasPeerReviewFileUpload(): bool
505  {
506  return $this->peer_file;
507  }
508 
509  public function setPeerReviewPersonalized(bool $a_val): void
510  {
511  $this->peer_personal = $a_val;
512  }
513 
514  public function hasPeerReviewPersonalized(): bool
515  {
516  return $this->peer_personal;
517  }
518 
519  public function setPeerReviewChars(?int $a_value): void
520  {
521  $a_value = (is_numeric($a_value) && (int) $a_value > 0)
522  ? (int) $a_value
523  : null;
524  $this->peer_char = $a_value;
525  }
526 
527  public function getPeerReviewChars(): ?int
528  {
529  return $this->peer_char;
530  }
531 
532  public function setPeerReviewCriteriaCatalogue(?int $a_value): void
533  {
534  $this->crit_cat = $a_value;
535  }
536 
538  {
539  return $this->crit_cat;
540  }
541 
542  public function getPeerReviewCriteriaCatalogueItems(): array
543  {
544  if ($this->crit_cat) {
545  return ilExcCriteria::getInstancesByParentId($this->crit_cat);
546  } else {
547  $res = array();
548 
549  if ($this->peer_rating) {
551  }
552 
553  if ($this->peer_text) {
555  $crit = ilExcCriteria::getInstanceByType("text");
556  if ($this->peer_char) {
557  $crit->setMinChars($this->peer_char);
558  }
559  $res[] = $crit;
560  }
561 
562  if ($this->peer_file) {
564  }
565 
566  return $res;
567  }
568  }
569 
570  public function setFeedbackFile(?string $a_value): void
571  {
572  $this->feedback_file = $a_value;
573  }
574 
575  public function getFeedbackFile(): ?string
576  {
577  return $this->feedback_file;
578  }
579 
583  public function setFeedbackCron(bool $a_value): void
584  {
585  $this->feedback_cron = $a_value;
586  }
587 
588  public function hasFeedbackCron(): bool
589  {
590  return $this->feedback_cron;
591  }
592 
593  // Set (global) feedback file availability date
594  public function setFeedbackDate(int $a_value): void
595  {
596  $this->feedback_date = $a_value;
597  }
598 
599  public function getFeedbackDate(): int
600  {
601  return $this->feedback_date;
602  }
603 
608  public function setFeedbackDateCustom(int $a_value): void
609  {
610  $this->feedback_date_custom = $a_value;
611  }
612 
613  public function getFeedbackDateCustom(): int
614  {
616  }
617 
618  // Set team management by tutor
619  public function setTeamTutor(bool $a_value): void
620  {
621  $this->team_tutor = $a_value;
622  }
623 
624  public function getTeamTutor(): bool
625  {
626  return $this->team_tutor;
627  }
628 
629  // Set max number of uploads
630  public function setMaxFile(?int $a_value): void
631  {
632  $this->max_file = $a_value;
633  }
634 
635  public function getMaxFile(): ?int
636  {
637  return $this->max_file;
638  }
639 
640  // Set portfolio template id
641  public function setPortfolioTemplateId(int $a_val): void
642  {
643  $this->portfolio_template = $a_val;
644  }
645 
646  public function getPortfolioTemplateId(): int
647  {
649  }
650 
654  public function read(): void
655  {
656  $ilDB = $this->db;
657 
658  $set = $ilDB->query(
659  "SELECT * FROM exc_assignment " .
660  " WHERE id = " . $ilDB->quote($this->getId(), "integer")
661  );
662  $rec = $ilDB->fetchAssoc($set);
663 
664  // #16172 - might be deleted
665  if (is_array($rec)) {
666  $this->initFromDB($rec);
667  }
668  }
669 
675  protected function initFromDB(array $a_set): void
676  {
677  $this->setId((int) $a_set["id"]);
678  $this->setExerciseId((int) $a_set["exc_id"]);
679  $this->setDeadline((int) $a_set["time_stamp"]);
680  $this->setExtendedDeadline((int) $a_set["deadline2"]);
681  $this->setInstruction((string) $a_set["instruction"]);
682  $this->setTitle((string) $a_set["title"]);
683  $this->setStartTime((int) $a_set["start_time"]);
684  $this->setOrderNr((int) $a_set["order_nr"]);
685  $this->setMandatory((bool) $a_set["mandatory"]);
686  $this->setType((int) $a_set["type"]);
687  $this->setPeerReview((bool) $a_set["peer"]);
688  $this->setPeerReviewMin((int) $a_set["peer_min"]);
689  $this->setPeerReviewSimpleUnlock((int) $a_set["peer_unlock"]);
690  $this->setPeerReviewDeadline((int) $a_set["peer_dl"]);
691  $this->setPeerReviewValid((int) $a_set["peer_valid"]);
692  $this->setPeerReviewFileUpload((bool) $a_set["peer_file"]);
693  $this->setPeerReviewPersonalized((bool) $a_set["peer_prsl"]);
694  $this->setPeerReviewChars((int) $a_set["peer_char"]);
695  $this->setPeerReviewText((bool) $a_set["peer_text"]);
696  $this->setPeerReviewRating((bool) $a_set["peer_rating"]);
697  $this->setPeerReviewCriteriaCatalogue((int) $a_set["peer_crit_cat"]);
698  $this->setFeedbackFile((string) $a_set["fb_file"]);
699  $this->setFeedbackDate((int) $a_set["fb_date"]);
700  $this->setFeedbackDateCustom((int) $a_set["fb_date_custom"]);
701  $this->setFeedbackCron((bool) $a_set["fb_cron"]);
702  $this->setTeamTutor((bool) $a_set["team_tutor"]);
703  $this->setMaxFile((int) $a_set["max_file"]);
704  $this->setPortfolioTemplateId((int) $a_set["portfolio_template"]);
705  $this->setMinCharLimit((int) $a_set["min_char_limit"]);
706  $this->setMaxCharLimit((int) $a_set["max_char_limit"]);
707  $this->setDeadlineMode((int) $a_set["deadline_mode"]);
708  $this->setRelativeDeadline((int) $a_set["relative_deadline"]);
709  $this->setRelDeadlineLastSubmission((int) $a_set["rel_deadline_last_subm"]);
710  }
711 
715  public function save(): void
716  {
717  $ilDB = $this->db;
718 
719  if ($this->getOrderNr() == 0) {
720  $this->setOrderNr(
721  self::lookupMaxOrderNrForEx($this->getExerciseId())
722  + 10
723  );
724  }
725 
726  $next_id = $ilDB->nextId("exc_assignment");
727  $ilDB->insert("exc_assignment", array(
728  "id" => array("integer", $next_id),
729  "exc_id" => array("integer", $this->getExerciseId()),
730  "time_stamp" => array("integer", $this->getDeadline()),
731  "deadline2" => array("integer", $this->getExtendedDeadline()),
732  "instruction" => array("clob", $this->getInstruction()),
733  "title" => array("text", $this->getTitle()),
734  "start_time" => array("integer", $this->getStartTime()),
735  "order_nr" => array("integer", $this->getOrderNr()),
736  "mandatory" => array("integer", $this->getMandatory()),
737  "type" => array("integer", $this->getType()),
738  "peer" => array("integer", $this->getPeerReview()),
739  "peer_min" => array("integer", $this->getPeerReviewMin()),
740  "peer_unlock" => array("integer", $this->getPeerReviewSimpleUnlock()),
741  "peer_dl" => array("integer", $this->getPeerReviewDeadline()),
742  "peer_valid" => array("integer", $this->getPeerReviewValid()),
743  "peer_file" => array("integer", $this->hasPeerReviewFileUpload()),
744  "peer_prsl" => array("integer", $this->hasPeerReviewPersonalized()),
745  "peer_char" => array("integer", $this->getPeerReviewChars()),
746  "peer_text" => array("integer", (int) $this->hasPeerReviewText()),
747  "peer_rating" => array("integer", (int) $this->hasPeerReviewRating()),
748  "peer_crit_cat" => array("integer", $this->getPeerReviewCriteriaCatalogue()),
749  "fb_file" => array("text", $this->getFeedbackFile()),
750  "fb_date" => array("integer", $this->getFeedbackDate()),
751  "fb_date_custom" => array("integer", $this->getFeedbackDateCustom()),
752  "fb_cron" => array("integer", $this->hasFeedbackCron()),
753  "team_tutor" => array("integer", $this->getTeamTutor()),
754  "max_file" => array("integer", $this->getMaxFile()),
755  "portfolio_template" => array("integer", $this->getPortfolioTemplateId()),
756  "min_char_limit" => array("integer", $this->getMinCharLimit()),
757  "max_char_limit" => array("integer", $this->getMaxCharLimit()),
758  "relative_deadline" => array("integer", $this->getRelativeDeadline()),
759  "rel_deadline_last_subm" => array("integer", $this->getRelDeadlineLastSubmission()),
760  "deadline_mode" => array("integer", $this->getDeadlineMode())
761  ));
762  $this->setId($next_id);
763  $exc = new ilObjExercise($this->getExerciseId(), false);
764  $exc->updateAllUsersStatus();
765 
766  $this->domain->assignment()->instructionFiles($next_id)->createCollection();
767 
768  self::createNewAssignmentRecords($next_id, $exc);
770  $this->handleCalendarEntries("create", $exc);
771  }
772 
776  public function update(): void
777  {
778  $ilDB = $this->db;
779 
780  $ilDB->update(
781  "exc_assignment",
782  array(
783  "exc_id" => array("integer", $this->getExerciseId()),
784  "time_stamp" => array("integer", $this->getDeadline()),
785  "deadline2" => array("integer", $this->getExtendedDeadline()),
786  "instruction" => array("clob", $this->getInstruction()),
787  "title" => array("text", $this->getTitle()),
788  "start_time" => array("integer", $this->getStartTime()),
789  "order_nr" => array("integer", $this->getOrderNr()),
790  "mandatory" => array("integer", $this->getMandatory()),
791  "type" => array("integer", $this->getType()),
792  "peer" => array("integer", $this->getPeerReview()),
793  "peer_min" => array("integer", $this->getPeerReviewMin()),
794  "peer_unlock" => array("integer", $this->getPeerReviewSimpleUnlock()),
795  "peer_dl" => array("integer", $this->getPeerReviewDeadline()),
796  "peer_valid" => array("integer", $this->getPeerReviewValid()),
797  "peer_file" => array("integer", $this->hasPeerReviewFileUpload()),
798  "peer_prsl" => array("integer", $this->hasPeerReviewPersonalized()),
799  "peer_char" => array("integer", $this->getPeerReviewChars()),
800  "peer_text" => array("integer", (int) $this->hasPeerReviewText()),
801  "peer_rating" => array("integer", (int) $this->hasPeerReviewRating()),
802  "peer_crit_cat" => array("integer", $this->getPeerReviewCriteriaCatalogue()),
803  "fb_file" => array("text", $this->getFeedbackFile()),
804  "fb_date" => array("integer", $this->getFeedbackDate()),
805  "fb_date_custom" => array("integer", $this->getFeedbackDateCustom()),
806  "fb_cron" => array("integer", $this->hasFeedbackCron()),
807  "team_tutor" => array("integer", $this->getTeamTutor()),
808  "max_file" => array("integer", $this->getMaxFile()),
809  "portfolio_template" => array("integer", $this->getPortfolioTemplateId()),
810  "min_char_limit" => array("integer", $this->getMinCharLimit()),
811  "max_char_limit" => array("integer", $this->getMaxCharLimit()),
812  "deadline_mode" => array("integer", $this->getDeadlineMode()),
813  "relative_deadline" => array("integer", $this->getRelativeDeadline()),
814  "rel_deadline_last_subm" => array("integer", $this->getRelDeadlineLastSubmission())
815  ),
816  array(
817  "id" => array("integer", $this->getId()),
818  )
819  );
820  $exc = new ilObjExercise($this->getExerciseId(), false);
821  $exc->updateAllUsersStatus();
823  $this->handleCalendarEntries("update", $exc);
824  }
825 
829  public function delete(
830  ilObjExercise $exc,
831  bool $update_user_status = true
832  ): void {
833  $ilDB = $this->db;
834 
835  // delete submissions
836  $exc_members = new ilExerciseMembers($exc);
837  foreach ($exc_members->getMembers() as $mem) {
838  $submission = new ilExSubmission($this, $mem);
839  $submission->deleteAllFiles();
840  }
841 
842  $ilDB->manipulateF(
843  "DELETE FROM exc_usr_tutor " .
844  "WHERE ass_id = %s",
845  array("integer"),
846  array($this->getId())
847  );
848 
849  // remove peer review data
850  if ($this->getPeerReview()) {
851  $peer_review = new ilExPeerReview($this);
852  $peer_review->resetPeerReviews();
853  }
854 
855  $ilDB->manipulate(
856  "DELETE FROM exc_ass_file_order" .
857  " WHERE assignment_id = " . $ilDB->quote($this->getId(), 'integer')
858  );
859 
860  $ilDB->manipulate(
861  "DELETE FROM exc_mem_ass_status" .
862  " WHERE ass_id = " . $ilDB->quote($this->getId(), 'integer')
863  );
864 
865  $ilDB->manipulate(
866  "DELETE FROM exc_assignment WHERE " .
867  " id = " . $ilDB->quote($this->getId(), "integer")
868  );
869 
870  if ($update_user_status) {
871  $exc->updateAllUsersStatus();
872  }
873 
874  $this->handleCalendarEntries("delete", $exc);
875 
877 
878  $reminder = new ilExAssignmentReminder();
879  $reminder->deleteReminders($this->getId());
880 
881  // delete teams
882  $this->domain->team()->deleteTeamsOfAssignment($this->getId());
883 
884  // delete resource collections and resources
885  $this->domain->assignment()->instructionFiles($this->getId())
886  ->deleteCollection();
887  }
888 
889 
890  // Get assignments data of an exercise in an array
891  public static function getAssignmentDataOfExercise(int $a_exc_id): array
892  {
893  global $DIC;
894 
895  $ilDB = $DIC->database();
896 
897  // should be changed to self::getInstancesByExerciseId()
898 
899  $set = $ilDB->query("SELECT * FROM exc_assignment " .
900  " WHERE exc_id = " . $ilDB->quote($a_exc_id, "integer") .
901  " ORDER BY order_nr");
902  $data = array();
903 
904  $order_val = 10;
905  while ($rec = $ilDB->fetchAssoc($set)) {
906  $data[] = array(
907  "id" => (int) $rec["id"],
908  "exc_id" => (int) $rec["exc_id"],
909  "deadline" => (int) $rec["time_stamp"],
910  "deadline2" => (int) $rec["deadline2"],
911  "instruction" => (string) $rec["instruction"],
912  "title" => (string) $rec["title"],
913  "start_time" => (int) $rec["start_time"],
914  "order_val" => $order_val,
915  "mandatory" => (bool) $rec["mandatory"],
916  "type" => (int) $rec["type"],
917  "peer" => (bool) $rec["peer"],
918  "peer_min" => (int) $rec["peer_min"],
919  "peer_dl" => (int) $rec["peer_dl"],
920  "peer_file" => (bool) $rec["peer_file"],
921  "peer_prsl" => (bool) $rec["peer_prsl"],
922  "fb_file" => (string) $rec["fb_file"],
923  "fb_date" => (int) $rec["fb_date"],
924  "fb_cron" => (bool) $rec["fb_cron"],
925  "deadline_mode" => (int) $rec["deadline_mode"],
926  "relative_deadline" => (int) $rec["relative_deadline"],
927  "rel_deadline_last_subm" => (int) $rec["rel_deadline_last_subm"]
928  );
929  $order_val += 10;
930  }
931  return $data;
932  }
933 
941  public static function cloneAssignmentsOfExercise(
942  int $a_old_exc_id,
943  int $a_new_exc_id,
944  array $a_crit_cat_map
945  ): void {
946  global $DIC;
947 
948  $ass_domain = $DIC->exercise()->internal()->domain()->assignment();
949  $ass_data = self::getInstancesByExercise($a_old_exc_id);
950  foreach ($ass_data as $d) {
951  // clone assignment
952  $new_ass = new ilExAssignment();
953  $new_ass->setExerciseId($a_new_exc_id);
954  $new_ass->setTitle($d->getTitle());
955  $new_ass->setDeadline($d->getDeadline());
956  $new_ass->setExtendedDeadline($d->getExtendedDeadline());
957  $new_ass->setInstruction($d->getInstruction());
958  $new_ass->setMandatory($d->getMandatory());
959  $new_ass->setOrderNr($d->getOrderNr());
960  $new_ass->setStartTime($d->getStartTime());
961  $new_ass->setType($d->getType());
962  $new_ass->setPeerReview($d->getPeerReview());
963  $new_ass->setPeerReviewMin($d->getPeerReviewMin());
964  $new_ass->setPeerReviewDeadline($d->getPeerReviewDeadline());
965  $new_ass->setPeerReviewFileUpload($d->hasPeerReviewFileUpload());
966  $new_ass->setPeerReviewPersonalized($d->hasPeerReviewPersonalized());
967  $new_ass->setPeerReviewValid($d->getPeerReviewValid());
968  $new_ass->setPeerReviewChars($d->getPeerReviewChars());
969  $new_ass->setPeerReviewText($d->hasPeerReviewText());
970  $new_ass->setPeerReviewRating($d->hasPeerReviewRating());
971  $new_ass->setPeerReviewCriteriaCatalogue($d->getPeerReviewCriteriaCatalogue());
972  $new_ass->setPeerReviewSimpleUnlock($d->getPeerReviewSimpleUnlock());
973  $new_ass->setFeedbackFile($d->getFeedbackFile());
974  $new_ass->setFeedbackDate($d->getFeedbackDate());
975  $new_ass->setFeedbackDateCustom($d->getFeedbackDateCustom());
976  $new_ass->setFeedbackCron($d->hasFeedbackCron()); // #16295
977  $new_ass->setTeamTutor($d->getTeamTutor());
978  $new_ass->setMaxFile($d->getMaxFile());
979  $new_ass->setMinCharLimit($d->getMinCharLimit());
980  $new_ass->setMaxCharLimit($d->getMaxCharLimit());
981  $new_ass->setPortfolioTemplateId($d->getPortfolioTemplateId());
982  $new_ass->setDeadlineMode($d->getDeadlineMode());
983  $new_ass->setRelativeDeadline($d->getRelativeDeadline());
984  $new_ass->setRelDeadlineLastSubmission($d->getRelDeadlineLastSubmission());
985 
986  // criteria catalogue(s)
987  if ($d->getPeerReviewCriteriaCatalogue() &&
988  array_key_exists($d->getPeerReviewCriteriaCatalogue(), $a_crit_cat_map)) {
989  $new_ass->setPeerReviewCriteriaCatalogue($a_crit_cat_map[$d->getPeerReviewCriteriaCatalogue()]);
990  }
991 
992  $new_ass->save();
993 
994  // clone assignment files
995  $ass_domain->instructionFiles($d->getId())->cloneTo($new_ass->getId());
996 
997  // clone global feedback file
998  $ass_domain->sampleSolution($d->getId())->cloneTo($new_ass->getId());
999 
1000  // clone reminders
1004  $rmd_sub = new ilExAssignmentReminder($a_old_exc_id, $d->getId(), $rem_type);
1005  if ($rmd_sub->getReminderStatus()) {
1006  $new_rmd_sub = new ilExAssignmentReminder($a_new_exc_id, $new_ass->getId(), $rem_type);
1007  $new_rmd_sub->setReminderStatus($rmd_sub->getReminderStatus());
1008  $new_rmd_sub->setReminderStart($rmd_sub->getReminderStart());
1009  $new_rmd_sub->setReminderEnd($rmd_sub->getReminderEnd());
1010  $new_rmd_sub->setReminderFrequency($rmd_sub->getReminderFrequency());
1011  $new_rmd_sub->setReminderMailTemplate($rmd_sub->getReminderMailTemplate());
1012  $new_rmd_sub->save();
1013  }
1014  }
1015 
1016 
1017  // type specific properties
1018  $ass_type = $d->getAssignmentType();
1019  $ass_type->cloneSpecificProperties($d, $new_ass);
1020  }
1021  }
1022 
1023  public function getFiles(): array
1024  {
1025  return $this->domain->assignment()->instructionFiles($this->getId())->getFiles();
1026  }
1027 
1028  public function getInstructionFilesOrder(): array
1029  {
1030  // TODO IRSS: currently the ilResourceCollectionGUI does not support the order of files done by the component,
1031  // this should be implemented as well there and will follow.
1032  $ilDB = $this->db;
1033 
1034  $set = $ilDB->query(
1035  "SELECT filename, order_nr, id FROM exc_ass_file_order " .
1036  " WHERE assignment_id = " . $ilDB->quote($this->getId(), "integer")
1037  );
1038 
1039  $data = array();
1040  while ($rec = $ilDB->fetchAssoc($set)) {
1041  $data[$rec['filename']] = $rec;
1042  }
1043 
1044  return $data;
1045  }
1046 
1047  // Select the maximum order nr for an exercise
1048  public static function lookupMaxOrderNrForEx(int $a_exc_id): int
1049  {
1050  // TODO IRSS: currently the ilResourceCollectionGUI does not support the order of files done by the component,
1051  // this should be implemented as well there and will follow.
1052  global $DIC;
1053 
1054  $ilDB = $DIC->database();
1055 
1056  $set = $ilDB->query(
1057  "SELECT MAX(order_nr) mnr FROM exc_assignment " .
1058  " WHERE exc_id = " . $ilDB->quote($a_exc_id, "integer")
1059  );
1060  if ($rec = $ilDB->fetchAssoc($set)) {
1061  return (int) $rec["mnr"];
1062  }
1063  return 0;
1064  }
1065 
1066  public static function lookupAssignmentOnline(int $a_ass_id): bool
1067  {
1068  global $DIC;
1069 
1070  $ilDB = $DIC->database();
1071 
1072  $query = "SELECT id FROM exc_assignment " .
1073  "WHERE start_time <= " . $ilDB->quote(time(), 'integer') . ' ' .
1074  "AND time_stamp >= " . $ilDB->quote(time(), 'integer') . ' ' .
1075  "AND id = " . $ilDB->quote($a_ass_id, 'integer');
1076  $res = $ilDB->query($query);
1077 
1078  return (bool) $res->numRows();
1079  }
1080 
1081  public static function lookupExerciseId(int $a_ass_id): int
1082  {
1083  global $DIC;
1084 
1085  $ilDB = $DIC->database();
1086  $query = "SELECT exc_id FROM exc_assignment " .
1087  "WHERE id = " . $ilDB->quote($a_ass_id, 'integer');
1088  $res = $ilDB->fetchAssoc($ilDB->query($query));
1089 
1090  return (int) ($res["exc_id"] ?? 0);
1091  }
1092 
1093  private static function lookup(int $a_id, string $a_field): string
1094  {
1095  global $DIC;
1096 
1097  $ilDB = $DIC->database();
1098 
1099  $set = $ilDB->query(
1100  "SELECT " . $a_field . " FROM exc_assignment " .
1101  " WHERE id = " . $ilDB->quote($a_id, "integer")
1102  );
1103 
1104  $rec = $ilDB->fetchAssoc($set);
1105 
1106  return $rec[$a_field] ?? "";
1107  }
1108 
1109  public static function lookupTitle(int $a_id): string
1110  {
1111  return self::lookup($a_id, "title");
1112  }
1113 
1114  public static function lookupType(int $a_id): int
1115  {
1116  return (int) self::lookup($a_id, "type");
1117  }
1118 
1119  // Save ordering of all assignments of an exercise
1120  public static function saveAssOrderOfExercise(int $a_ex_id, array $a_order): void
1121  {
1122  global $DIC;
1123 
1124  $ilDB = $DIC->database();
1125 
1126  asort($a_order);
1127  $nr = 10;
1128  foreach ($a_order as $k => $v) {
1129  // the check for exc_id is for security reasons. ass ids are unique.
1130  $ilDB->manipulate(
1131  "UPDATE exc_assignment SET " .
1132  " order_nr = " . $ilDB->quote($nr, "integer") .
1133  " WHERE id = " . $ilDB->quote((int) $k, "integer") .
1134  " AND exc_id = " . $ilDB->quote($a_ex_id, "integer")
1135  );
1136  $nr += 10;
1137  }
1138  }
1139 
1140  // Order assignments by deadline date
1141  public static function orderAssByDeadline(int $a_ex_id): void
1142  {
1143  global $DIC;
1144  $ilDB = $DIC->database();
1145 
1146  $set = $ilDB->query(
1147  "SELECT id FROM exc_assignment " .
1148  " WHERE exc_id = " . $ilDB->quote($a_ex_id, "integer") .
1149  " ORDER BY time_stamp"
1150  );
1151  $nr = 10;
1152  while ($rec = $ilDB->fetchAssoc($set)) {
1153  $ilDB->manipulate(
1154  "UPDATE exc_assignment SET " .
1155  " order_nr = " . $ilDB->quote($nr, "integer") .
1156  " WHERE id = " . $ilDB->quote($rec["id"], "integer")
1157  );
1158  $nr += 10;
1159  }
1160  }
1161 
1162  // Count the number of mandatory assignments
1163  public static function countMandatory(int $a_ex_id): int
1164  {
1165  global $DIC;
1166 
1167  $ilDB = $DIC->database();
1168 
1169  $set = $ilDB->query(
1170  "SELECT count(*) cntm FROM exc_assignment " .
1171  " WHERE exc_id = " . $ilDB->quote($a_ex_id, "integer") .
1172  " AND mandatory = " . $ilDB->quote(1, "integer")
1173  );
1174  $rec = $ilDB->fetchAssoc($set);
1175  return (int) $rec["cntm"];
1176  }
1177 
1178  // Count assignments
1179  public static function count(int $a_ex_id): int
1180  {
1181  global $DIC;
1182 
1183  $ilDB = $DIC->database();
1184 
1185  $set = $ilDB->query(
1186  "SELECT count(*) cntm FROM exc_assignment " .
1187  " WHERE exc_id = " . $ilDB->quote($a_ex_id, "integer")
1188  );
1189  $rec = $ilDB->fetchAssoc($set);
1190  return $rec["cntm"];
1191  }
1192 
1193  // Is assignment in exercise?
1194  public static function isInExercise(int $a_ass_id, int $a_ex_id): bool
1195  {
1196  global $DIC;
1197 
1198  $ilDB = $DIC->database();
1199 
1200  $set = $ilDB->query(
1201  "SELECT * FROM exc_assignment " .
1202  " WHERE exc_id = " . $ilDB->quote($a_ex_id, "integer") .
1203  " AND id = " . $ilDB->quote($a_ass_id, "integer")
1204  );
1205  if ($ilDB->fetchAssoc($set)) {
1206  return true;
1207  }
1208  return false;
1209  }
1210 
1211  public function getMemberListData(): array
1212  {
1213  $ilDB = $this->db;
1214 
1215  $mem = array();
1216 
1217  // first get list of members from member table
1218  $set = $ilDB->query("SELECT ud.usr_id, ud.lastname, ud.firstname, ud.login" .
1219  " FROM exc_members excm" .
1220  " JOIN usr_data ud ON (ud.usr_id = excm.usr_id)" .
1221  " WHERE excm.obj_id = " . $ilDB->quote($this->getExerciseId(), "integer"));
1222  while ($rec = $ilDB->fetchAssoc($set)) {
1223  $mem[$rec["usr_id"]] =
1224  array(
1225  "name" => $rec["lastname"] . ", " . $rec["firstname"],
1226  "login" => $rec["login"],
1227  "usr_id" => $rec["usr_id"],
1228  "lastname" => $rec["lastname"],
1229  "firstname" => $rec["firstname"]
1230  );
1231  }
1232 
1233  // users in idl may already exist before occuring in the members db table
1234  // if user starts an assignment with relative deadline
1235  $idl = $this->getIndividualDeadlines();
1236  if (!$this->ass_type->usesTeams()) {
1237  foreach ($idl as $user_id => $v) {
1238  if (!isset($mem[$user_id])) {
1239  if (ilObjUser::_exists((int) $user_id)) {
1240  $name = ilObjUser::_lookupName($user_id);
1241  $mem[$user_id] =
1242  array(
1243  "name" => $name["lastname"] . ", " . $name["firstname"],
1244  "login" => $name["login"],
1245  "usr_id" => $user_id,
1246  "lastname" => $name["lastname"],
1247  "firstname" => $name["firstname"]
1248  );
1249  }
1250  }
1251  }
1252  }
1253 
1254  $q = "SELECT * FROM exc_mem_ass_status " .
1255  "WHERE ass_id = " . $ilDB->quote($this->getId(), "integer");
1256  $set = $ilDB->query($q);
1257  while ($rec = $ilDB->fetchAssoc($set)) {
1258  if (isset($mem[$rec["usr_id"]])) {
1259  $sub = new ilExSubmission($this, $rec["usr_id"]);
1260 
1261  $mem[$rec["usr_id"]]["sent_time"] = $rec["sent_time"];
1262  $mem[$rec["usr_id"]]["submission"] = $sub->getLastSubmission();
1263  $mem[$rec["usr_id"]]["status_time"] = $rec["status_time"];
1264  $mem[$rec["usr_id"]]["feedback_time"] = $rec["feedback_time"];
1265  $mem[$rec["usr_id"]]["notice"] = $rec["notice"];
1266  $mem[$rec["usr_id"]]["status"] = $rec["status"];
1267  $mem[$rec["usr_id"]]["mark"] = $rec["mark"];
1268  $mem[$rec["usr_id"]]["comment"] = $rec["u_comment"];
1269  }
1270  }
1271  return $mem;
1272  }
1273 
1279  int $a_user_id,
1280  string $a_grade = ""
1281  ): ?array {
1282  global $DIC;
1283  $ilDB = $DIC->database();
1284 
1285  $and_grade = "";
1286  if (in_array($a_grade, array("notgraded", "passed", "failed"))) {
1287  $and_grade = " AND status = " . $ilDB->quote($a_grade, "text");
1288  }
1289 
1290  $q = "SELECT * FROM exc_mem_ass_status " .
1291  "WHERE ass_id = " . $ilDB->quote($this->getId(), "integer") .
1292  " AND usr_id = " . $ilDB->quote($a_user_id, "integer") .
1293  $and_grade;
1294 
1295  $set = $ilDB->query($q);
1296 
1297  $data = null;
1298  while ($rec = $ilDB->fetchAssoc($set)) {
1299  $sub = new ilExSubmission($this, $a_user_id);
1300 
1301  $data["sent_time"] = $rec["sent_time"];
1302  $data["submission"] = $sub->getLastSubmission();
1303  $data["status_time"] = $rec["status_time"];
1304  $data["feedback_time"] = $rec["feedback_time"];
1305  $data["notice"] = $rec["notice"];
1306  $data["status"] = $rec["status"];
1307  $data["mark"] = $rec["mark"];
1308  $data["comment"] = $rec["u_comment"];
1309  }
1310 
1311  return $data;
1312  }
1313 
1314  // Create member status record for a new participant for all assignments
1315  public static function createNewUserRecords(
1316  int $a_user_id,
1317  int $a_exc_id
1318  ): void {
1319  global $DIC;
1320 
1321  $ilDB = $DIC->database();
1322  $ass_domain = $DIC->exercise()->internal()->domain()->assignment();
1323 
1324  $ass_data = self::getAssignmentDataOfExercise($a_exc_id);
1325  foreach ($ass_data as $ass) {
1326  //echo "-".$ass["id"]."-".$a_user_id."-";
1327  $ilDB->replace("exc_mem_ass_status", array(
1328  "ass_id" => array("integer", $ass["id"]),
1329  "usr_id" => array("integer", $a_user_id)
1330  ), array(
1331  "status" => array("text", "notgraded")
1332  ));
1333  if (!$ass_domain->tutorFeedbackFile((int) $ass["id"])->hasCollection($a_user_id)) {
1334  $ass_domain->tutorFeedbackFile((int) $ass["id"])->createCollection($a_user_id);
1335  }
1336  }
1337  }
1338 
1339  // Create member status record for a new assignment for all participants
1340  public static function createNewAssignmentRecords(
1341  int $a_ass_id,
1342  ilObjExercise $a_exc
1343  ): void {
1344  global $DIC;
1345 
1346  $ilDB = $DIC->database();
1347  $ass_domain = $DIC->exercise()->internal()->domain()->assignment();
1348 
1349  $exmem = new ilExerciseMembers($a_exc);
1350  $mems = $exmem->getMembers();
1351 
1352  foreach ($mems as $mem) {
1353  $ilDB->replace("exc_mem_ass_status", array(
1354  "ass_id" => array("integer", $a_ass_id),
1355  "usr_id" => array("integer", $mem)
1356  ), array(
1357  "status" => array("text", "notgraded")
1358  ));
1359  if (!$ass_domain->tutorFeedbackFile($a_ass_id)->hasCollection($mem)) {
1360  $ass_domain->tutorFeedbackFile($a_ass_id)->createCollection($mem);
1361  }
1362  }
1363  }
1364 
1365 
1370  protected function handleCalendarEntries(
1371  string $a_event,
1372  ilObjExercise $exc
1373  ): void {
1374  $ilAppEventHandler = $this->app_event_handler;
1375 
1376  $dl_id = $this->getId() . "0";
1377  $fbdl_id = $this->getId() . "1";
1378 
1379  $context_ids = array($dl_id, $fbdl_id);
1380  $apps = array();
1381 
1382  if ($a_event != "delete") {
1383  // deadline or relative deadline given
1384  if ($this->getDeadline() || $this->getDeadlineMode() == ilExAssignment::DEADLINE_RELATIVE) {
1385  $app = new ilCalendarAppointmentTemplate($dl_id);
1386  $app->setTranslationType(ilCalendarEntry::TRANSLATION_SYSTEM);
1387  $app->setSubtitle("cal_exc_deadline");
1388  $app->setTitle($this->getTitle());
1389  $app->setFullday(false);
1390  // note: in the case of a relative deadline this will be set to 0 / 1970...)
1391  // see ilCalendarScheduleFilterExercise for appointment modification
1392  $app->setStart(new ilDateTime($this->getDeadline(), IL_CAL_UNIX));
1393 
1394  $apps[] = $app;
1395  }
1396 
1397  if ($this->getPeerReview() &&
1398  $this->getPeerReviewDeadline()) {
1399  $app = new ilCalendarAppointmentTemplate($fbdl_id);
1400  $app->setTranslationType(ilCalendarEntry::TRANSLATION_SYSTEM);
1401  $app->setSubtitle("cal_exc_peer_review_deadline");
1402  $app->setTitle($this->getTitle());
1403  $app->setFullday(false);
1404  $app->setStart(new ilDateTime($this->getPeerReviewDeadline(), IL_CAL_UNIX));
1405 
1406  $apps[] = $app;
1407  }
1408  }
1409 
1410  $ilAppEventHandler->raise(
1411  'components/ILIAS/Exercise',
1412  $a_event . 'Assignment',
1413  array(
1414  'object' => $exc,
1415  'obj_id' => $exc->getId(),
1416  'context_ids' => $context_ids,
1417  'appointments' => $apps)
1418  );
1419  }
1420 
1421  public static function getPendingFeedbackNotifications(): array
1422  {
1423  global $DIC;
1424 
1425  $log = ilLoggerFactory::getLogger("exc");
1426  $log->debug("Get feedback notifications.");
1427 
1428  $ilDB = $DIC->database();
1429 
1430  $res = array();
1431 
1432  $set = $ilDB->query("SELECT id,fb_file,time_stamp,deadline2,fb_date FROM exc_assignment" .
1433  " WHERE fb_cron = " . $ilDB->quote(1, "integer") .
1434  " AND (fb_date = " . $ilDB->quote(self::FEEDBACK_DATE_DEADLINE, "integer") .
1435  " AND time_stamp IS NOT NULL" .
1436  " AND time_stamp > " . $ilDB->quote(0, "integer") .
1437  " AND time_stamp < " . $ilDB->quote(time(), "integer") .
1438  " AND fb_cron_done = " . $ilDB->quote(0, "integer") .
1439  ") OR (fb_date = " . $ilDB->quote(self::FEEDBACK_DATE_CUSTOM, "integer") .
1440  " AND fb_date_custom IS NOT NULL" .
1441  " AND fb_date_custom > " . $ilDB->quote(0, "integer") .
1442  " AND fb_date_custom < " . $ilDB->quote(time(), "integer") .
1443  " AND fb_cron_done = " . $ilDB->quote(0, "integer") . ")");
1444 
1445 
1446 
1447  while ($row = $ilDB->fetchAssoc($set)) {
1448  $log->debug("check assignment " . $row['id'] . ", fb_file " . $row["fb_file"]);
1449  if ($row['fb_date'] == self::FEEDBACK_DATE_DEADLINE) {
1450  $max = max($row['time_stamp'], $row['deadline2']);
1451  if (trim($row["fb_file"]) && $max <= time()) {
1452  $log->debug("...adding(1)");
1453  $res[] = $row["id"];
1454  }
1455  } elseif ($row['fb_date'] == self::FEEDBACK_DATE_CUSTOM) {
1456  if (trim($row["fb_file"] ?? "") && ($row['fb_date_custom'] ?? 0) <= time()) {
1457  $log->debug("...adding(2)");
1458  $res[] = $row["id"];
1459  }
1460  }
1461  }
1462 
1463  return $res;
1464  }
1465 
1469  public static function sendFeedbackNotifications(
1470  int $a_ass_id,
1471  ?int $a_user_id = null
1472  ): bool {
1473  global $DIC;
1474 
1475  $ilDB = $DIC->database();
1476  $log = ilLoggerFactory::getLogger("exc");
1477 
1478  $ass = new self($a_ass_id);
1479 
1480  // valid assignment?
1481  if (!$ass->hasFeedbackCron() || !$ass->getFeedbackFile()) {
1482  $log->debug("return(1)");
1483  return false;
1484  }
1485 
1486  if (!$a_user_id) {
1487  // already done?
1488  $set = $ilDB->query("SELECT fb_cron_done" .
1489  " FROM exc_assignment" .
1490  " WHERE id = " . $ilDB->quote($a_ass_id, "integer"));
1491  $row = $ilDB->fetchAssoc($set);
1492  if ($row["fb_cron_done"]) {
1493  $log->debug("return(2)");
1494  return false;
1495  }
1496  }
1497 
1498  $ntf = new ilSystemNotification();
1499  $ntf->setLangModules(array("exc"));
1500  $ntf->setObjId($ass->getExerciseId());
1501  $ntf->setSubjectLangId("exc_feedback_notification_subject");
1502  $ntf->setIntroductionLangId("exc_feedback_notification_body");
1503  $ntf->addAdditionalInfo("exc_assignment", $ass->getTitle());
1504  $ntf->setGotoLangId("exc_feedback_notification_link");
1505  $ntf->setReasonLangId("exc_feedback_notification_reason");
1506 
1507  if (!$a_user_id) {
1508  $log->debug("send to members, cnt: " . count(ilExerciseMembers::_getMembers($ass->getExerciseId())));
1509  $ntf->sendMailAndReturnRecipients(ilExerciseMembers::_getMembers($ass->getExerciseId()));
1510 
1511  $ilDB->manipulate("UPDATE exc_assignment" .
1512  " SET fb_cron_done = " . $ilDB->quote(1, "integer") .
1513  " WHERE id = " . $ilDB->quote($a_ass_id, "integer"));
1514  } else {
1515  $log->debug("send to user: " . $a_user_id);
1516  $ntf->sendMailAndReturnRecipients(array($a_user_id));
1517  }
1518 
1519  return true;
1520  }
1521 
1522 
1523  // status
1524 
1525  // like: after effective deadline (for single user), no deadline: true
1526  public function afterDeadline(): bool
1527  {
1528  $ilUser = $this->user;
1529 
1530  // :TODO: always current user?
1531  $idl = $this->getPersonalDeadline($ilUser->getId()); // official deadline
1532 
1533  // no deadline === true
1534  $deadline = max($this->deadline, $this->deadline2, $idl); // includes grace period
1535  return ($deadline - time() <= 0);
1536  }
1537 
1538  public function afterDeadlineStrict(bool $a_include_personal = true): bool
1539  {
1540  // :TODO: this means that peer feedback, global feedback is available
1541  // after LAST personal deadline
1542  // team management is currently ignoring personal deadlines
1543  $idl = $a_include_personal
1544  ? $this->getLastPersonalDeadline()
1545  : null;
1546 
1547  // no deadline === false
1548  $deadline = max($this->deadline, $this->deadline2, $idl);
1549 
1550  // #18271 - afterDeadline() does not handle last personal deadline
1551  // after effective deadline of all users
1552  if ($idl && $deadline == $idl) {
1553  return ($deadline - time() <= 0);
1554  }
1555 
1556  // like: after effective deadline (for single user), except: no deadline false
1557  return ($deadline > 0 &&
1558  $this->afterDeadline());
1559  }
1560 
1564  public function afterCustomDate(): bool
1565  {
1566  $date_custom = $this->getFeedbackDateCustom();
1567  //if the solution will be displayed only after reach all the deadlines.
1568  //$final_deadline = $this->afterDeadlineStrict();
1569  //$dl = max($final_deadline, time());
1570  //return ($date_custom - $dl <= 0);
1571  return ($date_custom - time() <= 0);
1572  }
1573 
1574  // like: before effective deadline (for all users), no deadline: true
1575  public function beforeDeadline(): bool
1576  {
1577  // no deadline === true
1578  return !$this->afterDeadlineStrict();
1579  }
1580 
1581  public function notStartedYet(): bool
1582  {
1583  return (time() - $this->start_time <= 0);
1584  }
1585 
1586 
1587  //
1588  // FEEDBACK FILES
1589  //
1590 
1594  public function handleGlobalFeedbackFileUpload(int $ass_id, array $a_file): bool
1595  {
1596  $rcid = $this->domain->assignment()->sampleSolution($ass_id)->importFromLegacyUpload($a_file);
1597  $this->setFeedbackFile($a_file["name"]);
1598  return ($rcid !== "");
1599  }
1600 
1601 
1602  public function getMemberStatus(?int $a_user_id = null): ilExAssignmentMemberStatus
1603  {
1604  $ilUser = $this->user;
1605 
1606  if (!$a_user_id) {
1607  $a_user_id = $ilUser->getId();
1608  }
1609  if (!array_key_exists($a_user_id, $this->member_status)) {
1610  $this->member_status[$a_user_id] = new ilExAssignmentMemberStatus($this->getId(), $a_user_id);
1611  }
1612  return $this->member_status[$a_user_id];
1613  }
1614 
1615  //
1616  // individual deadlines
1617  //
1618 
1619  public function setIndividualDeadline(
1620  string $id,
1621  ilDateTime $date
1622  ): void {
1623  $is_team = false;
1624  if (!is_numeric($id)) {
1625  $id = substr($id, 1);
1626  $is_team = true;
1627  }
1628 
1629  $idl = ilExcIndividualDeadline::getInstance($this->getId(), (int) $id, $is_team);
1630  $idl->setIndividualDeadline($date->get(IL_CAL_UNIX));
1631  $idl->save();
1632  }
1633 
1634  public function getIndividualDeadlines(): array
1635  {
1636  $ilDB = $this->db;
1637 
1638  $res = array();
1639 
1640  $set = $ilDB->query("SELECT * FROM exc_idl" .
1641  " WHERE ass_id = " . $ilDB->quote($this->getId(), "integer"));
1642  while ($row = $ilDB->fetchAssoc($set)) {
1643  if ($row["is_team"]) {
1644  $row["member_id"] = "t" . $row["member_id"];
1645  }
1646 
1647  $res[$row["member_id"]] = $row["tstamp"];
1648  }
1649 
1650  return $res;
1651  }
1652 
1653  public function getRequestedDeadlines(): array
1654  {
1655  $ilDB = $this->db;
1656 
1657  $res = array();
1658 
1659  $set = $ilDB->query("SELECT * FROM exc_idl" .
1660  " WHERE ass_id = " . $ilDB->quote($this->getId(), "integer") .
1661  " AND requested = " . $ilDB->quote(1, "integer"));
1662  while ($row = $ilDB->fetchAssoc($set)) {
1663  if ($row["is_team"]) {
1664  $row["member_id"] = "t" . $row["member_id"];
1665  }
1666 
1667  $res[$row["member_id"]] = $row["requested"];
1668  }
1669 
1670  return $res;
1671  }
1672 
1673  public function hasActiveIDl(): bool
1674  {
1675  return (bool) ($this->getDeadline() || $this->getDeadlineMode() === self::DEADLINE_ABSOLUTE_INDIVIDUAL);
1676  }
1677 
1678  public function hasReadOnlyIDl(): bool
1679  {
1680  if (!$this->ass_type->usesTeams() &&
1681  $this->getPeerReview()) {
1682  // all deadlines are read-only if we have peer feedback
1683  $peer_review = new ilExPeerReview($this);
1684  if ($peer_review->hasPeerReviewGroups()) {
1685  return true;
1686  }
1687  }
1688 
1689  return false;
1690  }
1691 
1693  int $a_ass_id,
1694  array $a_order
1695  ): void {
1696  global $DIC;
1697 
1698  $db = $DIC->database();
1699 
1700  asort($a_order, SORT_NUMERIC);
1701 
1702  $nr = 10;
1703  foreach (array_keys($a_order) as $k) {
1704  // the check for exc_id is for security reasons. ass ids are unique.
1705  $db->manipulate(
1706  "UPDATE exc_ass_file_order SET " .
1707  " order_nr = " . $db->quote($nr, "integer") .
1708  " WHERE id = " . $db->quote((int) $k, "integer") .
1709  " AND assignment_id = " . $db->quote($a_ass_id, "integer")
1710  );
1711  $nr += 10;
1712  }
1713  }
1714 
1715  public static function insertFileOrderNr(
1716  int $a_ass_id,
1717  string $a_filename,
1718  int $a_order_nr
1719  ): void {
1720  global $DIC;
1721  $db = $DIC->database();
1722  $id = $db->nextId("exc_ass_file_order");
1723  $db->insert(
1724  "exc_ass_file_order",
1725  [
1726  "id" => ["integer", $id],
1727  "order_nr" => ["integer", $a_order_nr],
1728  "assignment_id" => ["integer", $a_ass_id],
1729  "filename" => ["text", $a_filename]
1730  ]
1731  );
1732  }
1733 
1734  // Store the order nr of a file in the database
1735  public static function instructionFileInsertOrder(
1736  string $a_filename,
1737  int $a_ass_id,
1738  int $a_order_nr = 0
1739  ): void {
1740  global $DIC;
1741 
1742  $db = $DIC->database();
1743 
1744  if ($a_ass_id) {
1745  //first of all check the suffix and change if necessary
1746  $filename = ilFileUtils::getSafeFilename($a_filename);
1747 
1748  if (self::instructionFileExistsInDb($filename, $a_ass_id) == 0) {
1749  if ($a_order_nr == 0) {
1750  $order_val = self::instructionFileOrderGetMax($a_ass_id);
1751  $order = $order_val + 10;
1752  } else {
1753  $order = $a_order_nr;
1754  }
1755 
1756  $id = $db->nextID('exc_ass_file_order');
1757  $db->manipulate("INSERT INTO exc_ass_file_order " .
1758  "(id, assignment_id, filename, order_nr) VALUES (" .
1759  $db->quote($id, "integer") . "," .
1760  $db->quote($a_ass_id, "integer") . "," .
1761  $db->quote($filename, "text") . "," .
1762  $db->quote($order, "integer") .
1763  ")");
1764  }
1765  }
1766  }
1767 
1771  public static function instructionFileDeleteOrder(
1772  int $a_ass_id,
1773  array $a_file
1774  ): void {
1775  global $DIC;
1776 
1777  $db = $DIC->database();
1778 
1779  //now its done by filename. We need to figure how to get the order id in the confirmdelete method
1780  foreach ($a_file as $v) {
1781  $db->manipulate(
1782  "DELETE FROM exc_ass_file_order " .
1783  "WHERE filename = " . $db->quote($v, 'text') .
1784  " AND assignment_id = " . $db->quote($a_ass_id, 'integer')
1785  );
1786  }
1787  }
1788 
1789  public static function renameInstructionFile(
1790  string $a_old_name,
1791  string $a_new_name,
1792  int $a_ass_id
1793  ): void {
1794  global $DIC;
1795 
1796  $db = $DIC->database();
1797 
1798  if ($a_ass_id) {
1799  $db->manipulate(
1800  "DELETE FROM exc_ass_file_order" .
1801  " WHERE assignment_id = " . $db->quote($a_ass_id, 'integer') .
1802  " AND filename = " . $db->quote($a_new_name, 'text')
1803  );
1804 
1805  $db->manipulate(
1806  "UPDATE exc_ass_file_order SET" .
1807  " filename = " . $db->quote($a_new_name, 'text') .
1808  " WHERE assignment_id = " . $db->quote($a_ass_id, 'integer') .
1809  " AND filename = " . $db->quote($a_old_name, 'text')
1810  );
1811  }
1812  }
1813 
1814  public static function instructionFileExistsInDb(
1815  string $a_filename,
1816  int $a_ass_id
1817  ): int {
1818  global $DIC;
1819 
1820  $db = $DIC->database();
1821 
1822  if ($a_ass_id) {
1823  $result = $db->query(
1824  "SELECT id FROM exc_ass_file_order" .
1825  " WHERE assignment_id = " . $db->quote($a_ass_id, 'integer') .
1826  " AND filename = " . $db->quote($a_filename, 'text')
1827  );
1828 
1829  return $db->numRows($result);
1830  }
1831 
1832  return 0;
1833  }
1834 
1835  public function fixInstructionFileOrdering(): void
1836  {
1837  $db = $this->db;
1838 
1839  $files = array_map(function ($v) {
1840  return $v["name"];
1841  }, $this->getFiles());
1842 
1843  $set = $db->query("SELECT * FROM exc_ass_file_order " .
1844  " WHERE assignment_id = " . $db->quote($this->getId(), "integer") .
1845  " ORDER BY order_nr");
1846  $order_nr = 10;
1847  $numbered_files = array();
1848  while ($rec = $db->fetchAssoc($set)) {
1849  // file exists, set correct order nr
1850  if (in_array($rec["filename"], $files)) {
1851  $db->manipulate(
1852  "UPDATE exc_ass_file_order SET " .
1853  " order_nr = " . $db->quote($order_nr, "integer") .
1854  " WHERE assignment_id = " . $db->quote($this->getId(), "integer") .
1855  " AND id = " . $db->quote($rec["id"], "integer")
1856  );
1857  $order_nr += 10;
1858  $numbered_files[] = $rec["filename"];
1859  } else { // file does not exist, delete entry
1860  $db->manipulate(
1861  "DELETE FROM exc_ass_file_order " .
1862  " WHERE assignment_id = " . $db->quote($this->getId(), "integer") .
1863  " AND id = " . $db->quote($rec["id"], "integer")
1864  );
1865  }
1866  }
1867  foreach ($files as $f) {
1868  if (!in_array($f, $numbered_files)) {
1869  self::instructionFileInsertOrder($f, $this->getId());
1870  }
1871  }
1872  }
1873 
1874  public function fileAddOrder(
1875  array $a_entries = array()
1876  ): array {
1877  $this->fixInstructionFileOrdering();
1878 
1879  $order = $this->getInstructionFilesOrder();
1880  foreach ($a_entries as $k => $e) {
1881  $a_entries[$k]["order_val"] = $order[$e["file"]]["order_nr"] ?? 0;
1882  $a_entries[$k]["order_id"] = $order[$e["file"]]["id"] ?? "";
1883  }
1884 
1885  return $a_entries;
1886  }
1887 
1888  public static function instructionFileOrderGetMax(int $a_ass_id): int
1889  {
1890  global $DIC;
1891 
1892  $db = $DIC->database();
1893 
1894  //get max order number
1895  $result = $db->queryF(
1896  "SELECT max(order_nr) as max_order FROM exc_ass_file_order WHERE assignment_id = %s",
1897  array('integer'),
1898  array($db->quote($a_ass_id, 'integer'))
1899  );
1900 
1901  $order_val = 0;
1902  while ($row = $db->fetchAssoc($result)) {
1903  $order_val = (int) $row['max_order'];
1904  }
1905  return $order_val;
1906  }
1907 
1908 
1909  // Set limit minimum characters
1910  public function setMinCharLimit(int $a_val): void
1911  {
1912  $this->min_char_limit = $a_val;
1913  }
1914 
1915  public function getMinCharLimit(): int
1916  {
1917  return $this->min_char_limit;
1918  }
1919 
1920  // Set limit maximum characters
1921  public function setMaxCharLimit(int $a_val): void
1922  {
1923  $this->max_char_limit = $a_val;
1924  }
1925 
1926  public function getMaxCharLimit(): int
1927  {
1928  return $this->max_char_limit;
1929  }
1930 
1939  public function getCalculatedDeadlines(): array
1940  {
1941  $calculated_deadlines = array(
1942  "user" => array(),
1943  "team" => array()
1944  );
1945 
1946  if ($this->getRelativeDeadline() && $this->getDeadlineMode() == self::DEADLINE_RELATIVE) {
1947  foreach (ilExcIndividualDeadline::getStartingTimestamps($this->getId()) as $ts) {
1948  $type = $ts["is_team"]
1949  ? "team"
1950  : "user";
1951 
1952  $calculated_deadlines[$type][$ts["member_id"]] = array(
1953  "calculated_deadline" => $ts["starting_ts"] + ($this->getRelativeDeadline() * 24 * 60 * 60)
1954  );
1955  }
1956  }
1957  return $calculated_deadlines;
1958  }
1959 
1960  // see bug #36253
1961  public function canParticipantReceiveFeedback(int $part_id): bool
1962  {
1963  if ($this->hasTeam()) {
1964  if (!ilExAssignmentTeam::getTeamId($this->getId(), $part_id)) {
1965  return false;
1966  }
1967  }
1968  return true;
1969  }
1970 }
setStartTime(?int $a_val)
static cloneAssignmentsOfExercise(int $a_old_exc_id, int $a_new_exc_id, array $a_crit_cat_map)
Clone assignments of exercise.
setRelativeDeadline(int $a_val)
get(int $a_format, string $a_format_str='', string $a_tz='')
get formatted date
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
ILIAS Refinery String Group $string_transform
$res
Definition: ltiservices.php:66
Global event handler.
ilAppEventHandler $app_event_handler
static instructionFileOrderGetMax(int $a_ass_id)
getType()
Get type this will most probably become an non public function in the future (or become obsolete) ...
Exercise assignment.
numRows(ilDBStatement $statement)
insert(string $table_name, array $values)
static getLogger(string $a_component_id)
Get component logger.
static createNewUserRecords(int $a_user_id, int $a_exc_id)
canParticipantReceiveFeedback(int $part_id)
setFeedbackFile(?string $a_value)
isValidType(int $a_value)
setPeerReviewValid(int $a_value)
Set peer review validation.
fetchAssoc(ilDBStatement $statement)
setType(int $a_value)
Set type this will most probably become an non public function in the future (or become obsolete) ...
setFeedbackDateCustom(int $a_value)
Set (global) feedback file availability using a custom date.
static getPendingFeedbackNotifications()
static getInstancesByParentId(int $a_parent_id)
handleGlobalFeedbackFileUpload(int $ass_id, array $a_file)
static count(int $a_ex_id)
ilExAssignmentTypes $types
getExerciseMemberAssignmentData(int $a_user_id, string $a_grade="")
Get submission data for an specific user,exercise and assignment.
static createNewAssignmentRecords(int $a_ass_id, ilObjExercise $a_exc)
initFromDB(array $a_set)
Import DB record.
setPeerReviewMin(int $a_value)
getMemberStatus(?int $a_user_id=null)
static countMandatory(int $a_ex_id)
setPeerReviewFileUpload(bool $a_val)
afterDeadlineStrict(bool $a_include_personal=true)
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
static lookupTitle(int $a_id)
static saveAssOrderOfExercise(int $a_ex_id, array $a_order)
static lookupType(int $a_id)
static _lookupName(int $a_user_id)
lookup user name
setDeadlineMode(int $a_val)
Set deadline mode.
Apointment templates are used for automatic generated apointments.
quote($value, string $type)
setFeedbackCron(bool $a_value)
Toggle (global) feedback file cron.
static orderAssByDeadline(int $a_ex_id)
static getInstanceByType(string $a_type)
const IL_CAL_UNIX
const TYPE_UPLOAD
direct checks against const should be avoided, use type objects instead
setLimit(int $limit, int $offset=0)
setInstruction(string $a_val)
static getSafeFilename(string $a_initial_filename)
static lookupMaxOrderNrForEx(int $a_exc_id)
while($session_entry=$r->fetchRow(ilDBConstants::FETCHMODE_ASSOC)) return null
static sendFeedbackNotifications(int $a_ass_id, ?int $a_user_id=null)
setTeamTutor(bool $a_value)
updateAllUsersStatus()
Update status of all users.
static getAssignmentDataOfExercise(int $a_exc_id)
cloneSpecificProperties(ilExAssignment $source, ilExAssignment $target)
Exercise peer review.
static _exists(int $id, bool $reference=false, ?string $type=null)
checks if an object exists in object_data
Class ilObjExercise.
setPeerReviewRating(bool $a_val)
setExtendedDeadline(?int $a_val)
static instructionFileGetFileOrderData(array $a_file_data, int $a_ass_id)
static renameInstructionFile(string $a_old_name, string $a_new_name, int $a_ass_id)
getCalculatedDeadlines()
Get calculated deadlines for user/team members.
nextId(string $table_name)
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
static raiseContentChanged(int $obj_id)
static instructionFileExistsInDb(string $a_filename, int $a_ass_id)
global $DIC
Definition: shib_login.php:22
getPersonalDeadline(int $a_user_id)
setMandatory(bool $a_val)
setFeedbackDate(int $a_value)
query(string $query)
Run a (read-only) Query on the database.
static getTeamId(int $a_assignment_id, int $a_user_id, bool $a_create_on_demand=false)
setPortfolioTemplateId(int $a_val)
setPeerReviewDeadline(int $a_val)
setRelDeadlineLastSubmission(int $a_val)
static instructionFileDeleteOrder(int $a_ass_id, array $a_file)
static instructionFileInsertOrder(string $a_filename, int $a_ass_id, int $a_order_nr=0)
queryF(string $query, array $types, array $values)
setIndividualDeadline(string $id, ilDateTime $date)
$filename
Definition: buildRTE.php:78
setPeerReviewPersonalized(bool $a_val)
static insertFileOrderNr(int $a_ass_id, string $a_filename, int $a_order_nr)
static getInstancesByExercise(int $a_exc_id)
static saveInstructionFilesOrderOfAssignment(int $a_ass_id, array $a_order)
ilAccessHandler $access
setPeerReviewSimpleUnlock(int $a_value)
$q
Definition: shib_logout.php:21
handleCalendarEntries(string $a_event, ilObjExercise $exc)
Handle calendar entries for deadline(s)
setPeerReviewChars(?int $a_value)
setPeerReviewText(bool $a_val)
Exercise submission //TODO: This class has many static methods related to delivered "files"...
static _getMembers(int $a_obj_id)
static getStartingTimestamps(int $a_ass_id)
Get starting timestamp data for an assignment.
setMaxFile(?int $a_value)
debug(string $message, array $context=[])
fileAddOrder(array $a_entries=array())
manipulate(string $query)
Run a (write) Query on the database.
setTitle(string $a_val)
static isInExercise(int $a_ass_id, int $a_ex_id)
ilExAssignmentTypeInterface $ass_type
setPeerReview(bool $a_value)
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
static lookupAssignmentOnline(int $a_ass_id)
setReminderStatus(?bool $a_status)
Set reminder for users without submission.
static lookup(int $a_id, string $a_field)
setPeerReviewCriteriaCatalogue(?int $a_value)
setDeadline(?int $a_val)
static lookupExerciseId(int $a_ass_id)
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
__construct($a_id=0)
Constructor.
ILIAS Exercise InternalDomainService $domain
static getInstance(int $a_ass_id, int $a_participant_id, bool $a_is_team=false)