ILIAS  trunk Revision v11.0_alpha-1702-gfd3ecb7f852
All Data Structures Namespaces Files Functions Variables Enumerations Enumerator Modules Pages
class.ilAssQuestionList.php
Go to the documentation of this file.
1 <?php
2 
23 
31 {
32  private array $parentObjIdsFilter = [];
33  private ?int $parentObjId = null;
34  private string $parentObjType = 'qpl';
35  private array $availableTaxonomyIds = [];
36  private array $fieldFilters = [];
37  private array $taxFilters = [];
39  private array $taxParentIds = [];
40  private array $taxParentTypes = [];
41  private ?int $answerStatusActiveId = null;
42  protected bool $join_obj_data = true;
43 
47  public const QUESTION_ANSWER_STATUS_NON_ANSWERED = 'nonAnswered';
48  public const QUESTION_ANSWER_STATUS_WRONG_ANSWERED = 'wrongAnswered';
49  public const QUESTION_ANSWER_STATUS_CORRECT_ANSWERED = 'correctAnswered';
50 
54  public const ANSWER_STATUS_FILTER_ALL_NON_CORRECT = 'allNonCorrect';
55  public const ANSWER_STATUS_FILTER_NON_ANSWERED_ONLY = 'nonAnswered';
56  public const ANSWER_STATUS_FILTER_WRONG_ANSWERED_ONLY = 'wrongAnswered';
57 
58  private string $answerStatusFilter = '';
59 
60  public const QUESTION_INSTANCE_TYPE_ORIGINALS = 'QST_INSTANCE_TYPE_ORIGINALS';
61  public const QUESTION_INSTANCE_TYPE_DUPLICATES = 'QST_INSTANCE_TYPE_DUPLICATES';
62  public const QUESTION_INSTANCE_TYPE_ALL = 'QST_INSTANCE_TYPE_ALL';
63  private string $questionInstanceTypeFilter = self::QUESTION_INSTANCE_TYPE_ORIGINALS;
64 
65  private array $includeQuestionIdsFilter = [];
66  private array $excludeQuestionIdsFilter = [];
67 
68  public const QUESTION_COMMENTED_ONLY = '1';
69  public const QUESTION_COMMENTED_EXCLUDED = '2';
70  protected ?string $filter_comments = null;
71 
72  protected array $questions = [];
73 
74  private ?Order $order = null;
75  private ?Range $range = null;
76 
77  public function __construct(
78  private ilDBInterface $db,
79  private ilLanguage $lng,
80  private Refinery $refinery,
81  private ilComponentRepository $component_repository,
82  private ?NotesService $notes_service = null
83  ) {
84  }
85 
86  public function setOrder(?Order $order = null): void
87  {
88  $this->order = $order;
89  }
90 
91  public function setRange(?Range $range = null): void
92  {
93  $this->range = $range;
94  }
95 
96  public function getParentObjId(): ?int
97  {
98  return $this->parentObjId;
99  }
100 
101  public function setParentObjId(?int $parentObjId): void
102  {
103  $this->parentObjId = $parentObjId;
104  }
105 
106  public function setParentObjectType(string $parentObjType): void
107  {
108  $this->parentObjType = $parentObjType;
109  }
110 
111  public function setParentObjIdsFilter(array $parentObjIdsFilter): void
112  {
113  $this->parentObjIdsFilter = $parentObjIdsFilter;
114  }
115 
116  public function setQuestionInstanceTypeFilter(?string $questionInstanceTypeFilter): void
117  {
118  $this->questionInstanceTypeFilter = (string) $questionInstanceTypeFilter;
119  }
120 
121  public function setIncludeQuestionIdsFilter(array $questionIdsFilter): void
122  {
123  $this->includeQuestionIdsFilter = $questionIdsFilter;
124  }
125 
126  public function setExcludeQuestionIdsFilter(array $excludeQuestionIdsFilter): void
127  {
128  $this->excludeQuestionIdsFilter = $excludeQuestionIdsFilter;
129  }
130 
131  public function addFieldFilter(string $fieldName, mixed $fieldValue): void
132  {
133  $this->fieldFilters[$fieldName] = $fieldValue;
134  }
135 
136  public function addTaxonomyFilter($taxId, $taxNodes, $parentObjId, $parentObjType): void
137  {
138  $this->taxFilters[$taxId] = $taxNodes;
139  $this->taxParentIds[$taxId] = $parentObjId;
140  $this->taxParentTypes[$taxId] = $parentObjType;
141  }
142 
143  public function addTaxonomyFilterNoTaxonomySet(bool $flag): void
144  {
145  $this->taxFiltersExcludeAnyObjectsWithTaxonomies = $flag;
146  }
147 
148  public function setAvailableTaxonomyIds(array $availableTaxonomyIds): void
149  {
150  $this->availableTaxonomyIds = $availableTaxonomyIds;
151  }
152 
153  public function setAnswerStatusActiveId(?int $answerStatusActiveId): void
154  {
155  $this->answerStatusActiveId = $answerStatusActiveId;
156  }
157 
158  public function setAnswerStatusFilter(string $answerStatusFilter): void
159  {
160  $this->answerStatusFilter = $answerStatusFilter;
161  }
162 
166  public function setJoinObjectData(bool $a_val): void
167  {
168  $this->join_obj_data = $a_val;
169  }
170 
171  private function getParentObjFilterExpression(): ?string
172  {
173  if ($this->parentObjId) {
174  return "qpl_questions.obj_fi = {$this->db->quote($this->parentObjId, ilDBConstants::T_INTEGER)}";
175  }
176 
177  if (!empty($this->parentObjIdsFilter)) {
178  return $this->db->in('qpl_questions.obj_fi', $this->parentObjIdsFilter, false, ilDBConstants::T_INTEGER);
179  }
180 
181  return null;
182  }
183 
184  private function getFieldFilterExpressions(): array
185  {
186  $expressions = [];
187 
188  foreach ($this->fieldFilters as $fieldName => $fieldValue) {
189  switch ($fieldName) {
190  case 'title':
191  case 'description':
192  case 'author':
193  case 'lifecycle':
194  $expressions[] = $this->db->like("qpl_questions.$fieldName", ilDBConstants::T_TEXT, "%%$fieldValue%%");
195  break;
196  case 'type':
197  $expressions[] = "qpl_qst_type.type_tag = {$this->db->quote($fieldValue, ilDBConstants::T_TEXT)}";
198  break;
199  case 'question_id':
200  if ($fieldValue !== '' && !is_array($fieldValue)) {
201  $fieldValue = [$fieldValue];
202  }
203  $expressions[] = $this->db->in('qpl_questions.question_id', $fieldValue, false, ilDBConstants::T_INTEGER);
204  break;
205  case 'parent_title':
206  if ($this->join_obj_data) {
207  $expressions[] = $this->db->like('object_data.title', ilDBConstants::T_TEXT, "%%$fieldValue%%");
208  }
209  break;
210  }
211  }
212 
213  return $expressions;
214  }
215 
216  private function handleFeedbackJoin(string $tableJoin): string
217  {
218  $feedback_join = match ($this->fieldFilters['feedback'] ?? null) {
219  'true' => 'INNER',
220  'false' => 'LEFT',
221  default => null
222  };
223 
224  if (isset($feedback_join)) {
225  $SQL = "$feedback_join JOIN qpl_fb_generic ON qpl_fb_generic.question_fi = qpl_questions.question_id ";
226  $tableJoin .= !str_contains($tableJoin, $SQL) ? $SQL : '';
227  }
228 
229  return $tableJoin;
230  }
231 
232  private function handleHintJoin(string $tableJoin): string
233  {
234  $feedback_join = match ($this->fieldFilters['hints'] ?? null) {
235  'true' => 'INNER',
236  'false' => 'LEFT',
237  default => null
238  };
239 
240  if (isset($feedback_join)) {
241  $SQL = "$feedback_join JOIN qpl_hints ON qpl_hints.qht_question_fi = qpl_questions.question_id ";
242  $tableJoin .= !str_contains($tableJoin, $SQL) ? $SQL : '';
243  }
244 
245  return $tableJoin;
246  }
247 
248  private function getTaxonomyFilterExpressions(): array
249  {
250  $expressions = $this->getFilterByAssignedTaxonomyIdsExpression();
251 
252  $taxonomy_title = $this->fieldFilters['taxonomy_title'] ?? '';
253  $taxonomy_node_title = $this->fieldFilters['taxonomy_node_title'] ?? '';
254 
255  if ($taxonomy_title === '' && $taxonomy_node_title === '') {
256  return $expressions;
257  }
258 
259  $base = 'SELECT DISTINCT item_id FROM tax_node_assignment';
260 
261  $like_taxonomy_title = $taxonomy_title !== ''
262  ? "AND {$this->db->like('object_data.title', ilDBConstants::T_TEXT, "%$taxonomy_title%", false)}"
263  : '';
264  $like_taxonomy_node_title = $taxonomy_node_title !== ''
265  ? "AND {$this->db->like('tax_node.title', ilDBConstants::T_TEXT, "%$taxonomy_node_title%", false)}"
266  : '';
267 
268  $inner_join_object_data = "INNER JOIN object_data ON (object_data.obj_id = tax_node_assignment.tax_id AND object_data.type = 'tax' $like_taxonomy_title)";
269  $inner_join_tax_node = "INNER JOIN tax_node ON (tax_node.tax_id = tax_node_assignment.tax_id AND tax_node.type = 'taxn' AND tax_node_assignment.node_id = tax_node.obj_id $like_taxonomy_node_title)";
270 
271  $expressions[] = "qpl_questions.question_id IN ($base $inner_join_object_data $inner_join_tax_node)";
272 
273  return $expressions;
274  }
275 
277  {
278  if ($this->taxFiltersExcludeAnyObjectsWithTaxonomies) {
279  return ['question_id NOT IN (SELECT DISTINCT item_id FROM tax_node_assignment)'];
280  }
281 
282  $expressions = [];
283  foreach ($this->taxFilters as $tax_id => $tax_nodes) {
284  $question_ids = [];
285 
286  if ($tax_nodes === []) {
287  continue;
288  }
289 
290  foreach ($tax_nodes as $tax_node) {
291  $tax_items_by_tax_parent = $this->getTaxItems(
292  $this->taxParentTypes[$tax_id],
293  $this->taxParentIds[$tax_id],
294  $tax_id,
295  $tax_node
296  );
297 
298  $tax_items_by_parent = $this->getTaxItems(
299  $this->parentObjType,
300  $this->parentObjId,
301  $tax_id,
302  $tax_node
303  );
304 
305  $tax_items = array_merge($tax_items_by_tax_parent, $tax_items_by_parent);
306  foreach ($tax_items as $tax_item) {
307  $question_ids[$tax_item['item_id']] = $tax_item['item_id'];
308  }
309  }
310 
311  $expressions[] = $this->db->in('question_id', $question_ids, false, ilDBConstants::T_INTEGER);
312  }
313 
314  return $expressions;
315  }
316 
317  protected function getTaxItems(string $parentType, int $parentObjId, int $taxId, int $taxNode): array
318  {
319  $taxTree = new ilTaxonomyTree($taxId);
320 
321  $taxNodeAssignment = new ilTaxNodeAssignment(
322  $parentType,
323  $parentObjId,
324  'quest',
325  $taxId
326  );
327 
328  $subNodes = $taxTree->getSubTreeIds($taxNode);
329  $subNodes[] = $taxNode;
330 
331  return $taxNodeAssignment->getAssignmentsOfNode($subNodes);
332  }
333 
334  private function getQuestionInstanceTypeFilterExpression(): ?string
335  {
336  return match ($this->questionInstanceTypeFilter) {
337  self::QUESTION_INSTANCE_TYPE_ORIGINALS => 'qpl_questions.original_id IS NULL',
338  self::QUESTION_INSTANCE_TYPE_DUPLICATES => 'qpl_questions.original_id IS NOT NULL',
339  default => null
340  };
341  }
342 
343  private function getQuestionIdsFilterExpressions(): array
344  {
345  $expressions = [];
346 
347  if (!empty($this->includeQuestionIdsFilter)) {
348  $expressions[] = $this->db->in(
349  'qpl_questions.question_id',
350  $this->includeQuestionIdsFilter,
351  false,
353  );
354  }
355 
356  if (!empty($this->excludeQuestionIdsFilter)) {
357  $IN = $this->db->in(
358  'qpl_questions.question_id',
359  $this->excludeQuestionIdsFilter,
360  true,
362  );
363 
364  $expressions[] = $IN === ' 1=2 ' ? ' 1=1 ' : $IN; // required for ILIAS < 5.0
365  }
366 
367  return $expressions;
368  }
369 
370  private function getAnswerStatusFilterExpressions(): array
371  {
372  return match ($this->answerStatusFilter) {
373  self::ANSWER_STATUS_FILTER_ALL_NON_CORRECT => ['
374  (tst_test_result.question_fi IS NULL OR tst_test_result.points < qpl_questions.points)
375  '],
376  self::ANSWER_STATUS_FILTER_NON_ANSWERED_ONLY => ['tst_test_result.question_fi IS NULL'],
377  self::ANSWER_STATUS_FILTER_WRONG_ANSWERED_ONLY => [
378  'tst_test_result.question_fi IS NOT NULL',
379  'tst_test_result.points < qpl_questions.points'
380  ],
381  default => [],
382  };
383  }
384 
385  private function getTableJoinExpression(): string
386  {
387  $tableJoin = '
388  INNER JOIN qpl_qst_type
389  ON qpl_qst_type.question_type_id = qpl_questions.question_type_fi
390  ';
391 
392  if ($this->join_obj_data) {
393  $tableJoin .= '
394  INNER JOIN object_data
395  ON object_data.obj_id = qpl_questions.obj_fi
396  ';
397  }
398 
399  if (
400  $this->parentObjType === 'tst'
401  && $this->questionInstanceTypeFilter === self::QUESTION_INSTANCE_TYPE_ALL
402  ) {
403  $tableJoin .= 'INNER JOIN tst_test_question tstquest ON tstquest.question_fi = qpl_questions.question_id';
404  }
405 
406  $tableJoin = $this->handleFeedbackJoin($tableJoin);
407  $tableJoin = $this->handleHintJoin($tableJoin);
408 
409  if ($this->answerStatusActiveId) {
410  $tableJoin .= "
411  LEFT JOIN tst_test_result
412  ON tst_test_result.question_fi = qpl_questions.question_id
413  AND tst_test_result.active_fi = {$this->db->quote($this->answerStatusActiveId, ilDBConstants::T_INTEGER)}
414  ";
415  }
416 
417  return $tableJoin;
418  }
419 
420  private function getConditionalFilterExpression(): string
421  {
422  $conditions = [];
423 
425  $conditions[] = $this->getQuestionInstanceTypeFilterExpression();
426  }
427 
428  if ($this->getParentObjFilterExpression() !== null) {
429  $conditions[] = $this->getParentObjFilterExpression();
430  }
431 
432  $conditions = array_merge(
433  $conditions,
435  $this->getFieldFilterExpressions(),
438  );
439 
440  $conditions = implode(' AND ', $conditions);
441  return $conditions !== '' ? "AND $conditions" : '';
442  }
443 
444  private function getSelectFieldsExpression(): string
445  {
446  $select_fields = [
447  'qpl_questions.*',
448  'qpl_qst_type.type_tag',
449  'qpl_qst_type.plugin',
450  'qpl_qst_type.plugin_name',
451  'qpl_questions.points max_points'
452  ];
453 
454  if ($this->join_obj_data) {
455  $select_fields[] = 'object_data.title parent_title';
456  }
457 
458  if ($this->answerStatusActiveId) {
459  $select_fields[] = 'tst_test_result.points reached_points';
460  $select_fields[] = "CASE
461  WHEN tst_test_result.points IS NULL THEN '" . self::QUESTION_ANSWER_STATUS_NON_ANSWERED . "'
462  WHEN tst_test_result.points < qpl_questions.points THEN '" . self::QUESTION_ANSWER_STATUS_WRONG_ANSWERED . "'
463  ELSE '" . self::QUESTION_ANSWER_STATUS_CORRECT_ANSWERED . "'
464  END question_answer_status
465  ";
466  }
467 
468  $select_fields[] = $this->generateFeedbackSubquery();
469  $select_fields[] = $this->generateHintSubquery();
470  $select_fields[] = $this->generateTaxonomySubquery();
471 
472  $select_fields = implode(', ', $select_fields);
473  return "SELECT DISTINCT $select_fields";
474  }
475 
476  private function generateFeedbackSubquery(): string
477  {
478  $cases = [];
479  $tables = ['qpl_fb_generic', 'qpl_fb_specific'];
480 
481  foreach ($tables as $table) {
482  $subquery = "SELECT 1 FROM $table WHERE $table.question_fi = qpl_questions.question_id AND $table.feedback <> ''";
483  $cases[] = "WHEN EXISTS ($subquery) THEN TRUE";
484  }
485 
486  $page_object_table = 'page_object';
487  foreach ($tables as $table) {
488  $subquery = sprintf(
489  "SELECT 1 FROM $table JOIN $page_object_table ON $page_object_table.page_id = $table.feedback_id WHERE $page_object_table.parent_type IN ('%s', '%s') AND $page_object_table.is_empty <> 1 AND $table.question_fi = qpl_questions.question_id",
492  );
493  $cases[] = "WHEN EXISTS ($subquery) THEN TRUE";
494  }
495 
496  $feedback_case_subquery = implode(' ', $cases);
497  return "CASE $feedback_case_subquery ELSE FALSE END AS feedback";
498  }
499 
500  private function generateHintSubquery(): string
501  {
502  $hint_subquery = 'SELECT 1 FROM qpl_hints WHERE qpl_hints.qht_question_fi = qpl_questions.question_id';
503  return "CASE WHEN EXISTS ($hint_subquery) THEN TRUE ELSE FALSE END AS hints";
504  }
505 
506  private function generateTaxonomySubquery(): string
507  {
508  $tax_node_assignment_table = 'tax_node_assignment';
509  $tax_subquery = "SELECT 1 FROM $tax_node_assignment_table WHERE $tax_node_assignment_table.item_id = qpl_questions.question_id AND $tax_node_assignment_table.item_type = 'quest'";
510  return "CASE WHEN EXISTS ($tax_subquery) THEN TRUE ELSE FALSE END AS taxonomies";
511  }
512 
513  private function buildBasicQuery(): string
514  {
515  return "{$this->getSelectFieldsExpression()} FROM qpl_questions {$this->getTableJoinExpression()} WHERE qpl_questions.tstamp > 0";
516  }
517 
518  private function getHavingFilterExpression(): string
519  {
520  $expressions = [];
521 
522  foreach ($this->fieldFilters as $fieldName => $fieldValue) {
523  if ($fieldName === 'feedback') {
524  $fieldValue = strtoupper($fieldValue);
525  if (in_array($fieldValue, ['TRUE', 'FALSE'], true)) {
526  $expressions[] = "feedback IS $fieldValue";
527  }
528  continue;
529 
530  }
531 
532  if ($fieldName === 'hints') {
533  $fieldValue = strtoupper($fieldValue);
534  if (in_array($fieldValue, ['TRUE', 'FALSE'], true)) {
535  $expressions[] = "hints IS $fieldValue";
536  }
537  }
538  }
539 
540  $having = implode(' AND ', $expressions);
541  return $having !== '' ? "HAVING $having" : '';
542  }
543 
544  private function buildOrderQueryExpression(): string
545  {
546  $order = $this->order;
547  if ($order === null) {
548  return '';
549  }
550 
551  [$order_field, $order_direction] = $order->join(
552  '',
553  static fn(string $index, string $key, string $value): array => [$key, $value]
554  );
555 
556  $order_direction = strtoupper($order_direction);
557  if (!in_array($order_direction, [Order::ASC, Order::DESC], true)) {
558  $order_direction = Order::ASC;
559  }
560 
561  return " ORDER BY `$order_field` $order_direction";
562  }
563 
564  private function buildLimitQueryExpression(): string
565  {
566  $range = $this->range;
567  if ($range === null) {
568  return '';
569  }
570 
571  $limit = max($range->getLength(), 0);
572  $offset = max($range->getStart(), 0);
573 
574  return " LIMIT $limit OFFSET $offset";
575  }
576 
577  private function buildQuery(): string
578  {
579  return implode(PHP_EOL, array_filter([
580  $this->buildBasicQuery(),
582  $this->getHavingFilterExpression(),
583  $this->buildOrderQueryExpression(),
584  $this->buildLimitQueryExpression(),
585  ]));
586  }
587 
588  public function load(): void
589  {
590  $this->checkFilters();
591 
592  $tags_trafo = $this->refinery->encode()->htmlSpecialCharsAsEntities();
593 
594  $res = $this->db->query($this->buildQuery());
595  while ($row = $this->db->fetchAssoc($res)) {
597 
598  if (!$this->isActiveQuestionType($row)) {
599  continue;
600  }
601 
602  $row['title'] = $tags_trafo->transform($row['title'] ?? '&nbsp;');
603  $row['description'] = $tags_trafo->transform($row['description'] !== '' && $row['description'] !== null ? $row['description'] : '&nbsp;');
604  $row['author'] = $tags_trafo->transform($row['author']);
605  $row['taxonomies'] = $this->loadTaxonomyAssignmentData($row['obj_fi'], $row['question_id']);
606  $row['ttype'] = $this->lng->txt($row['type_tag']);
607  $row['feedback'] = $row['feedback'] === 1;
608  $row['hints'] = $row['hints'] === 1;
609  $row['comments'] = $this->getNumberOfCommentsForQuestion($row['question_id']);
610 
611  if (
612  $this->filter_comments === self::QUESTION_COMMENTED_ONLY && $row['comments'] === 0
613  || $this->filter_comments === self::QUESTION_COMMENTED_EXCLUDED && $row['comments'] > 0
614  ) {
615  continue;
616  }
617 
618  $this->questions[$row['question_id']] = $row;
619  }
620  }
621 
622  public function getTotalRowCount(?array $filter_data, ?array $additional_parameters): ?int
623  {
624  $this->checkFilters();
625 
626  $count = 'COUNT(*)';
627  $query = "SELECT $count FROM qpl_questions {$this->getTableJoinExpression()} WHERE qpl_questions.tstamp > 0 {$this->getConditionalFilterExpression()}";
628 
629  return (int) ($this->db->query($query)->fetch()[$count] ?? 0);
630  }
631 
632  protected function getNumberOfCommentsForQuestion(int $question_id): int
633  {
634  if ($this->notes_service === null) {
635  return 0;
636  }
637  $notes_context = $this->notes_service->data()->context(
638  $this->getParentObjId(),
639  $question_id,
640  'quest'
641  );
642  return $this->notes_service->domain()->getNrOfCommentsForContext($notes_context);
643  }
644 
645  public function setCommentFilter(?int $commented = null): void
646  {
647  $this->filter_comments = $commented;
648  }
649 
650  private function loadTaxonomyAssignmentData(
651  int $parent_obj_id,
652  int $question_id
653  ): array {
654  $tax_assignment_data = [];
655  foreach ($this->availableTaxonomyIds as $tax_id) {
656  $tax_tree = new ilTaxonomyTree($tax_id);
657 
658  $tax_assignment = new ilTaxNodeAssignment('qpl', $parent_obj_id, 'quest', $tax_id);
659  $assignments = $tax_assignment->getAssignmentsOfItem($question_id);
660 
661  foreach ($assignments as $ass_data) {
662  if (!isset($tax_assignment_data[$ass_data['tax_id']])) {
663  $tax_assignment_data[$ass_data['tax_id']] = [];
664  }
665 
666  $ass_data['node_lft'] = $tax_tree->getNodeData($ass_data['node_id']);
667 
668  $tax_assignment_data[$ass_data['tax_id']][$ass_data['node_id']] = $ass_data;
669  }
670  }
671 
672  return $tax_assignment_data;
673  }
674 
675  private function isActiveQuestionType(array $questionData): bool
676  {
677  if (!isset($questionData['plugin'])) {
678  return false;
679  }
680 
681  if (!$questionData['plugin']) {
682  return true;
683  }
684 
685  if (
686  !isset($questionData['plugin_name'])
687  || !$this->component_repository->getComponentByTypeAndName(
689  'TestQuestionPool'
690  )->getPluginSlotById('qst')->hasPluginName($questionData['plugin_name'])
691  ) {
692  return false;
693  }
694 
695  return $this->component_repository
696  ->getComponentByTypeAndName(ilComponentInfo::TYPE_MODULES, 'TestQuestionPool')
697  ->getPluginSlotById('qst')
698  ->getPluginByName($questionData['plugin_name'])
699  ->isActive();
700  }
701 
702  public function getDataArrayForQuestionId(int $questionId)
703  {
704  return $this->questions[$questionId];
705  }
706 
707  public function getQuestionDataArray(): array
708  {
709  return $this->questions;
710  }
711 
712  public function isInList(int $questionId): bool
713  {
714  return isset($this->questions[$questionId]);
715  }
716 
726  public function getTitle(string $a_comp_id, string $a_item_type, int $a_item_id): string
727  {
728  if ($a_comp_id !== 'qpl' || $a_item_type !== 'quest' || !$a_item_id) {
729  return '';
730  }
731 
732  return $this->questions[$a_item_id]['title'] ?? '';
733  }
734 
735  private function checkFilters(): void
736  {
737  if ($this->answerStatusFilter !== '' && !$this->answerStatusActiveId) {
738  throw new ilTestQuestionPoolException(
739  'No active id given! You cannot use the answer status filter without giving an active id.'
740  );
741  }
742  }
743 }
static completeMissingPluginName(array $question_type_data)
setExcludeQuestionIdsFilter(array $excludeQuestionIdsFilter)
$res
Definition: ltiservices.php:66
Readable part of repository interface to ilComponentDataDB.
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
setQuestionInstanceTypeFilter(?string $questionInstanceTypeFilter)
join($init, callable $fn)
Definition: Order.php:75
isActiveQuestionType(array $questionData)
setAvailableTaxonomyIds(array $availableTaxonomyIds)
setJoinObjectData(bool $a_val)
Set if object data table should be joined.
handleHintJoin(string $tableJoin)
setOrder(?Order $order=null)
setParentObjIdsFilter(array $parentObjIdsFilter)
setParentObjectType(string $parentObjType)
setIncludeQuestionIdsFilter(array $questionIdsFilter)
getTitle(string $a_comp_id, string $a_item_type, int $a_item_id)
Get title of an assigned item.
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
setRange(?Range $range=null)
Both the subject and the direction need to be specified when expressing an order. ...
Definition: Order.php:28
addTaxonomyFilterNoTaxonomySet(bool $flag)
__construct(private ilDBInterface $db, private ilLanguage $lng, private Refinery $refinery, private ilComponentRepository $component_repository, private ?NotesService $notes_service=null)
while($session_entry=$r->fetchRow(ilDBConstants::FETCHMODE_ASSOC)) return null
getDataArrayForQuestionId(int $questionId)
addFieldFilter(string $fieldName, mixed $fieldValue)
getTotalRowCount(?array $filter_data, ?array $additional_parameters)
const PAGE_OBJECT_TYPE_GENERIC_FEEDBACK
type for generic feedback page objects
loadTaxonomyAssignmentData(int $parent_obj_id, int $question_id)
addTaxonomyFilter($taxId, $taxNodes, $parentObjId, $parentObjType)
getNumberOfCommentsForQuestion(int $question_id)
const ANSWER_STATUS_FILTER_ALL_NON_CORRECT
answer status filter value domain
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
setParentObjId(?int $parentObjId)
global $lng
Definition: privfeed.php:31
setAnswerStatusFilter(string $answerStatusFilter)
handleFeedbackJoin(string $tableJoin)
A simple class to express a naive range of whole positive numbers.
Definition: Range.php:28
setCommentFilter(?int $commented=null)
setAnswerStatusActiveId(?int $answerStatusActiveId)
const QUESTION_ANSWER_STATUS_NON_ANSWERED
answer status domain for single questions
const PAGE_OBJECT_TYPE_SPECIFIC_FEEDBACK
type for specific feedback page objects
getTaxItems(string $parentType, int $parentObjId, int $taxId, int $taxNode)