118 $this->questionInstanceTypeFilter = (string) $questionInstanceTypeFilter;
123 $this->includeQuestionIdsFilter = $questionIdsFilter;
133 $this->fieldFilters[$fieldName] = $fieldValue;
138 $this->taxFilters[$taxId] = $taxNodes;
145 $this->taxFiltersExcludeAnyObjectsWithTaxonomies = $flag;
168 $this->join_obj_data = $a_val;
173 if ($this->parentObjId) {
174 return "qpl_questions.obj_fi = {$this->db->quote($this->parentObjId, ilDBConstants::T_INTEGER)}";
177 if (!empty($this->parentObjIdsFilter)) {
188 foreach ($this->fieldFilters as $fieldName => $fieldValue) {
189 switch ($fieldName) {
194 $expressions[] = $this->db->like(
"qpl_questions.$fieldName",
ilDBConstants::T_TEXT,
"%%$fieldValue%%");
197 $expressions[] =
"qpl_qst_type.type_tag = {$this->db->quote($fieldValue, ilDBConstants::T_TEXT)}";
200 if ($fieldValue !==
'' && !is_array($fieldValue)) {
201 $fieldValue = [$fieldValue];
206 if ($this->join_obj_data) {
218 $feedback_join = match ($this->fieldFilters[
'feedback'] ??
null) {
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 :
'';
234 $feedback_join = match ($this->fieldFilters[
'hints'] ??
null) {
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 :
'';
252 $taxonomy_title = $this->fieldFilters[
'taxonomy_title'] ??
'';
253 $taxonomy_node_title = $this->fieldFilters[
'taxonomy_node_title'] ??
'';
255 if ($taxonomy_title ===
'' && $taxonomy_node_title ===
'') {
259 $base =
'SELECT DISTINCT item_id FROM tax_node_assignment';
261 $like_taxonomy_title = $taxonomy_title !==
'' 262 ?
"AND {$this->db->like('object_data.title', ilDBConstants::T_TEXT, "%$taxonomy_title%
", false)}" 264 $like_taxonomy_node_title = $taxonomy_node_title !==
'' 265 ?
"AND {$this->db->like('tax_node.title', ilDBConstants::T_TEXT, "%$taxonomy_node_title%
", false)}" 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)";
271 $expressions[] =
"qpl_questions.question_id IN ($base $inner_join_object_data $inner_join_tax_node)";
278 if ($this->taxFiltersExcludeAnyObjectsWithTaxonomies) {
279 return [
'question_id NOT IN (SELECT DISTINCT item_id FROM tax_node_assignment)'];
283 foreach ($this->taxFilters as $tax_id => $tax_nodes) {
286 if ($tax_nodes === []) {
290 foreach ($tax_nodes as $tax_node) {
292 $this->taxParentTypes[$tax_id],
293 $this->taxParentIds[$tax_id],
299 $this->parentObjType,
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'];
317 protected function getTaxItems(
string $parentType,
int $parentObjId,
int $taxId,
int $taxNode): array
328 $subNodes = $taxTree->getSubTreeIds($taxNode);
329 $subNodes[] = $taxNode;
331 return $taxNodeAssignment->getAssignmentsOfNode($subNodes);
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',
347 if (!empty($this->includeQuestionIdsFilter)) {
348 $expressions[] = $this->db->in(
349 'qpl_questions.question_id',
350 $this->includeQuestionIdsFilter,
356 if (!empty($this->excludeQuestionIdsFilter)) {
358 'qpl_questions.question_id',
359 $this->excludeQuestionIdsFilter,
364 $expressions[] = $IN ===
' 1=2 ' ?
' 1=1 ' : $IN;
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) 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' 388 INNER JOIN qpl_qst_type 389 ON qpl_qst_type.question_type_id = qpl_questions.question_type_fi 392 if ($this->join_obj_data) {
394 INNER JOIN object_data 395 ON object_data.obj_id = qpl_questions.obj_fi 400 $this->parentObjType ===
'tst' 401 && $this->questionInstanceTypeFilter === self::QUESTION_INSTANCE_TYPE_ALL
403 $tableJoin .=
'INNER JOIN tst_test_question tstquest ON tstquest.question_fi = qpl_questions.question_id';
409 if ($this->answerStatusActiveId) {
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)} 432 $conditions = array_merge(
440 $conditions = implode(
' AND ', $conditions);
441 return $conditions !==
'' ?
"AND $conditions" :
'';
448 'qpl_qst_type.type_tag',
449 'qpl_qst_type.plugin',
450 'qpl_qst_type.plugin_name',
451 'qpl_questions.points max_points' 454 if ($this->join_obj_data) {
455 $select_fields[] =
'object_data.title parent_title';
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 472 $select_fields = implode(
', ', $select_fields);
473 return "SELECT DISTINCT $select_fields";
479 $tables = [
'qpl_fb_generic',
'qpl_fb_specific'];
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";
486 $page_object_table =
'page_object';
487 foreach ($tables as $table) {
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",
493 $cases[] =
"WHEN EXISTS ($subquery) THEN TRUE";
496 $feedback_case_subquery = implode(
' ', $cases);
497 return "CASE $feedback_case_subquery ELSE FALSE END AS feedback";
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";
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";
515 return "{$this->getSelectFieldsExpression()} FROM qpl_questions {$this->getTableJoinExpression()} WHERE qpl_questions.tstamp > 0";
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";
532 if ($fieldName ===
'hints') {
533 $fieldValue = strtoupper($fieldValue);
534 if (in_array($fieldValue, [
'TRUE',
'FALSE'],
true)) {
535 $expressions[] =
"hints IS $fieldValue";
540 $having = implode(
' AND ', $expressions);
541 return $having !==
'' ?
"HAVING $having" :
'';
547 if ($order ===
null) {
551 [$order_field, $order_direction] = $order->
join(
553 static fn(
string $index,
string $key,
string $value): array => [$key, $value]
556 $order_direction = strtoupper($order_direction);
557 if (!in_array($order_direction, [Order::ASC, Order::DESC],
true)) {
558 $order_direction = Order::ASC;
561 return " ORDER BY `$order_field` $order_direction";
567 if ($range ===
null) {
572 $offset = max($range->
getStart(), 0);
574 return " LIMIT $limit OFFSET $offset";
579 return implode(PHP_EOL, array_filter([
592 $tags_trafo = $this->
refinery->encode()->htmlSpecialCharsAsEntities();
595 while ($row = $this->db->fetchAssoc(
$res)) {
602 $row[
'title'] = $tags_trafo->transform($row[
'title'] ??
' ');
603 $row[
'description'] = $tags_trafo->transform($row[
'description'] !==
'' && $row[
'description'] !==
null ? $row[
'description'] :
' ');
604 $row[
'author'] = $tags_trafo->transform($row[
'author']);
606 $row[
'ttype'] = $this->
lng->txt($row[
'type_tag']);
607 $row[
'feedback'] = $row[
'feedback'] === 1;
608 $row[
'hints'] = $row[
'hints'] === 1;
612 $this->filter_comments === self::QUESTION_COMMENTED_ONLY && $row[
'comments'] === 0
613 || $this->filter_comments === self::QUESTION_COMMENTED_EXCLUDED && $row[
'comments'] > 0
618 $this->questions[$row[
'question_id']] = $row;
627 $query =
"SELECT $count FROM qpl_questions {$this->getTableJoinExpression()} WHERE qpl_questions.tstamp > 0 {$this->getConditionalFilterExpression()}";
629 return (
int) ($this->db->query($query)->fetch()[$count] ?? 0);
634 if ($this->notes_service ===
null) {
637 $notes_context = $this->notes_service->data()->context(
642 return $this->notes_service->domain()->getNrOfCommentsForContext($notes_context);
647 $this->filter_comments = $commented;
654 $tax_assignment_data = [];
655 foreach ($this->availableTaxonomyIds as $tax_id) {
659 $assignments = $tax_assignment->getAssignmentsOfItem($question_id);
661 foreach ($assignments as $ass_data) {
662 if (!isset($tax_assignment_data[$ass_data[
'tax_id']])) {
663 $tax_assignment_data[$ass_data[
'tax_id']] = [];
666 $ass_data[
'node_lft'] = $tax_tree->getNodeData($ass_data[
'node_id']);
668 $tax_assignment_data[$ass_data[
'tax_id']][$ass_data[
'node_id']] = $ass_data;
672 return $tax_assignment_data;
677 if (!isset($questionData[
'plugin'])) {
681 if (!$questionData[
'plugin']) {
686 !isset($questionData[
'plugin_name'])
687 || !$this->component_repository->getComponentByTypeAndName(
690 )->getPluginSlotById(
'qst')->hasPluginName($questionData[
'plugin_name'])
695 return $this->component_repository
697 ->getPluginSlotById(
'qst')
698 ->getPluginByName($questionData[
'plugin_name'])
704 return $this->questions[$questionId];
714 return isset($this->questions[$questionId]);
726 public function getTitle(
string $a_comp_id,
string $a_item_type,
int $a_item_id): string
728 if ($a_comp_id !==
'qpl' || $a_item_type !==
'quest' || !$a_item_id) {
732 return $this->questions[$a_item_id][
'title'] ??
'';
737 if ($this->answerStatusFilter !==
'' && !$this->answerStatusActiveId) {
739 'No active id given! You cannot use the answer status filter without giving an active id.'
static completeMissingPluginName(array $question_type_data)
setExcludeQuestionIdsFilter(array $excludeQuestionIdsFilter)
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)
isActiveQuestionType(array $questionData)
getSelectFieldsExpression()
int $answerStatusActiveId
setAvailableTaxonomyIds(array $availableTaxonomyIds)
setJoinObjectData(bool $a_val)
Set if object data table should be joined.
handleHintJoin(string $tableJoin)
getQuestionIdsFilterExpressions()
isInList(int $questionId)
setOrder(?Order $order=null)
setParentObjIdsFilter(array $parentObjIdsFilter)
setParentObjectType(string $parentObjType)
getParentObjFilterExpression()
setIncludeQuestionIdsFilter(array $questionIdsFilter)
const QUESTION_COMMENTED_ONLY
const QUESTION_COMMENTED_EXCLUDED
const QUESTION_INSTANCE_TYPE_DUPLICATES
array $availableTaxonomyIds
getTitle(string $a_comp_id, string $a_item_type, int $a_item_id)
Get title of an assigned item.
buildOrderQueryExpression()
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. ...
getConditionalFilterExpression()
addTaxonomyFilterNoTaxonomySet(bool $flag)
bool $taxFiltersExcludeAnyObjectsWithTaxonomies
__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
const ANSWER_STATUS_FILTER_NON_ANSWERED_ONLY
getDataArrayForQuestionId(int $questionId)
addFieldFilter(string $fieldName, mixed $fieldValue)
getTaxonomyFilterExpressions()
getFilterByAssignedTaxonomyIdsExpression()
getTotalRowCount(?array $filter_data, ?array $additional_parameters)
getAnswerStatusFilterExpressions()
array $excludeQuestionIdsFilter
string $questionInstanceTypeFilter
const PAGE_OBJECT_TYPE_GENERIC_FEEDBACK
type for generic feedback page objects
loadTaxonomyAssignmentData(int $parent_obj_id, int $question_id)
getFieldFilterExpressions()
addTaxonomyFilter($taxId, $taxNodes, $parentObjId, $parentObjType)
array $includeQuestionIdsFilter
getHavingFilterExpression()
getNumberOfCommentsForQuestion(int $question_id)
const ANSWER_STATUS_FILTER_WRONG_ANSWERED_ONLY
getQuestionInstanceTypeFilterExpression()
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...
generateFeedbackSubquery()
array $parentObjIdsFilter
const QUESTION_ANSWER_STATUS_CORRECT_ANSWERED
setParentObjId(?int $parentObjId)
const QUESTION_ANSWER_STATUS_WRONG_ANSWERED
string $answerStatusFilter
generateTaxonomySubquery()
setAnswerStatusFilter(string $answerStatusFilter)
handleFeedbackJoin(string $tableJoin)
const QUESTION_INSTANCE_TYPE_ALL
A simple class to express a naive range of whole positive numbers.
setCommentFilter(?int $commented=null)
buildLimitQueryExpression()
setAnswerStatusActiveId(?int $answerStatusActiveId)
const QUESTION_INSTANCE_TYPE_ORIGINALS
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)