19 declare(strict_types=1);
91 private readonly \
ilObjUser $current_user,
94 $this->
lng->loadLanguageModule(
'dateplaner');
99 return $this->ui_factory->table()->data(
101 $this->
lng->txt(
'history'),
118 if ($this->filter_data ===
null) {
119 $this->filter_data = $this->ui_service->filter()->getData($this->
filter) ?? [];
125 $field_factory = $this->ui_factory->input()->field();
127 self::FILTER_FIELD_PERIOD => $field_factory->duration($this->
lng->txt(
'cal_period'))
129 ->withFormat($this->log_viewer->buildUserDateTimeFormat())
131 if ($this->ref_id ===
null) {
132 $filter_inputs[self::FILTER_FIELD_TEST_TITLE] = $field_factory->text($this->
lng->txt(
'test'));
136 self::FILTER_FIELD_ADMIN => $field_factory->text($this->
lng->txt(
'author')),
137 self::FILTER_FIELD_PARTICIPANT => $field_factory->text($this->lng->txt(
'tst_participant')),
138 self::FILTER_FIELD_IP => $field_factory->text($this->lng->txt(
'client_ip')),
139 self::FILTER_FIELD_QUESTION_TITLE => $field_factory->text($this->lng->txt(
'question_title')),
140 self::FILTER_FIELD_LOG_ENTRY_TYPE => $field_factory->multiSelect(
141 $this->lng->txt(
'log_entry_type'),
144 self::FILTER_FIELD_INTERACTION_TYPE => $field_factory->multiSelect(
145 $this->lng->txt(
'interaction_type'),
150 $active = array_fill(0, count($filter_inputs),
true);
152 $this->
filter = $this->ui_service->filter()->standard(
153 'log_table_filter_id',
165 $f = $this->ui_factory->table()->column();
168 self::COLUMN_DATE_TIME =>
$f->date($this->
lng->txt(
'date_time'), $this->log_viewer->buildUserDateTimeFormat()),
169 self::COLUMN_CORRESPONDING_TEST =>
$f->link($this->lng->txt(
'test'))->withIsOptional(
true,
true),
170 self::COLUMN_ADMIN =>
$f->text($this->
lng->txt(
'author'))->withIsOptional(
true,
true),
171 self::COLUMN_PARTICIPANT =>
$f->text($this->
lng->txt(
'tst_participant'))->withIsOptional(
true,
true)
174 if ($this->
logger->isIPLoggingEnabled()) {
175 $columns[self::COLUMN_SOURCE_IP] =
$f->text($this->
lng->txt(
'client_ip'))->withIsOptional(
true,
true);
179 self::COLUMN_QUESTION =>
$f->link($this->
lng->txt(
'question'))->withIsOptional(
true,
true),
180 self::COLUMN_LOG_ENTRY_TYPE =>
$f->text($this->
lng->txt(
'log_entry_type'))->withIsOptional(
true,
true),
181 self::COLUMN_INTERACTION_TYPE =>
$f->text($this->
lng->txt(
'interaction_type'))->withIsOptional(
true,
true)
187 array $visible_column_ids,
191 ?array $additional_parameters
201 $log_entry_type_filter,
202 $interaction_type_filter
206 'timezone' => new \DateTimeZone($this->current_user->getTimeZone()),
207 'date_format' => $this->log_viewer->buildUserDateTimeFormat()->toString()
210 foreach ($this->logging_repository->getLogs(
211 $this->logger->getInteractionTypes(),
221 $log_entry_type_filter,
222 $interaction_type_filter
224 yield $interaction->getLogEntryAsDataTableRow(
226 $this->title_builder,
235 ?array $additional_parameters
245 $log_entry_type_filter,
246 $interaction_type_filter
249 return $this->logging_repository->getLogsCount(
250 $this->
logger->getInteractionTypes(),
258 $log_entry_type_filter,
259 $interaction_type_filter
265 array $affected_items
277 $log = $this->logging_repository->getLog($affected_item);
283 'timezone' => new \DateTimeZone($this->current_user->getTimeZone()),
284 'date_format' => $this->log_viewer->buildUserDateTimeFormat()->toString()
287 echo $this->ui_renderer->renderAsync(
288 $this->ui_factory->modal()->roundtrip(
289 $this->
lng->txt(
'additional_info'),
290 $log->getParsedAdditionalInformation(
291 $this->
logger->getAdditionalInformationGenerator(),
302 if ($affected_items === []) {
306 echo $this->ui_renderer->renderAsync(
307 $this->ui_factory->modal()->interruptive(
308 $this->
lng->txt(
'confirmation'),
309 $this->
lng->txt(
'confirm_log_deletion'),
311 ->withParameter($this->action_parameter_token, self::ACTION_DELETE)
312 ->withParameter($this->row_id_token, $affected_items)
313 ->buildURI()->__toString())
321 if ($this->ref_id !==
null) {
322 $this->tpl->setOnScreenMessage(
'failure', $this->
lng->txt(
'log_deletion_not_allowed'));
326 $this->logging_repository->deleteLogs($affected_items);
327 $this->tpl->setOnScreenMessage(
'success', $this->
lng->txt(
'logs_deleted'));
332 if ($affected_items === []) {
333 $this->tpl->setOnScreenMessage(
'info', $this->
lng->txt(
'no_checkbox'));
337 $this->log_viewer->buildExcelWorkbookForLogs(
339 )->sendToClient(date(
'Y-m-d') . self::EXPORT_FILE_NAME);
344 if ($affected_items[0] !==
'ALL_OBJECTS') {
345 return $this->logging_repository->getLogsByUniqueIdentifiers($affected_items);
357 $log_entry_type_filter,
358 $interaction_type_filter
360 return $this->logging_repository->getLogs(
361 $this->
logger->getInteractionTypes(),
371 $log_entry_type_filter,
372 $interaction_type_filter
378 $af = $this->ui_factory->table()->action();
380 self::ACTION_ID_SHOW_ADDITIONAL_INFO => $af->single(
381 $this->
lng->txt(
'additional_info'),
382 $this->url_builder->withParameter(
383 $this->action_parameter_token,
384 self::ACTION_ADDITIONAL_INFORMATION
388 self::ACTION_ID_EXPORT => $af->multi(
389 $this->
lng->txt(
'export'),
390 $this->url_builder->withParameter(
391 $this->action_parameter_token,
392 self::ACTION_EXPORT_AS_CSV
397 if ($this->ref_id !==
null) {
401 self::ACTION_ID_DELETE => $af->standard(
402 $this->
lng->txt(
'delete'),
403 $this->url_builder->withParameter(
404 $this->action_parameter_token,
405 self::ACTION_CONFIRM_DELETE
418 $log_entry_types = $this->
logger->getLogEntryTypes();
419 $log_entry_options = [];
420 foreach ($log_entry_types as $log_entry_type) {
421 $log_entry_options [$log_entry_type] = $this->
lng->txt($lang_prefix . $log_entry_type);
423 asort($log_entry_options);
424 return $log_entry_options;
433 $interaction_types = array_reduce(
434 $this->
logger->getInteractionTypes(),
435 fn(array $et, array $it): array => [...$et, ...$it],
439 $interaction_options = [];
440 foreach ($interaction_types as $interaction_type) {
441 $interaction_options[$interaction_type] = $this->
lng->txt($lang_prefix . $interaction_type);
443 asort($interaction_options);
444 return $interaction_options;
453 $admin_filter =
null;
454 $question_filter =
null;
456 if (!empty($filter_array[self::FILTER_FIELD_PERIOD][0])) {
457 $from_filter = (new \DateTimeImmutable(
458 $filter_array[self::FILTER_FIELD_PERIOD][0],
463 if (!empty($filter_array[self::FILTER_FIELD_PERIOD][1])) {
464 $to_filter = (new \DateTimeImmutable(
465 $filter_array[self::FILTER_FIELD_PERIOD][1],
470 if (!empty($filter_array[self::FILTER_FIELD_TEST_TITLE])) {
471 $test_filter = array_reduce(
473 static fn(array $ref_ids,
int $obj_id) => array_merge(
481 if (!empty($filter_array[self::FILTER_FIELD_ADMIN])) {
482 $admin_query = new \ilUserQuery();
483 $admin_query->setTextFilter($filter_array[self::FILTER_FIELD_ADMIN]);
485 $admin_query->query()
489 if (!empty($filter_array[self::FILTER_FIELD_PARTICIPANT])) {
490 $pax_query = new \ilUserQuery();
491 $pax_query->setTextFilter($filter_array[self::FILTER_FIELD_PARTICIPANT]);
497 if (!empty($filter_array[self::FILTER_FIELD_QUESTION_TITLE])) {
498 $question_filter = $this->question_repo->searchQuestionIdsByTitle(
499 $filter_array[self::FILTER_FIELD_QUESTION_TITLE]
510 !empty($filter_array[self::FILTER_FIELD_IP]) ? $filter_array[self::FILTER_FIELD_IP] :
null,
511 $filter_array[self::FILTER_FIELD_LOG_ENTRY_TYPE] ??
null,
512 $filter_array[self::FILTER_FIELD_INTERACTION_TYPE] ??
null 518 echo $this->ui_renderer->renderAsync(
519 $this->ui_factory->modal()->roundtrip(
520 $this->
lng->txt(
'error'),
521 $this->ui_factory->messageBox()->failure($message)
529 if (!isset($response[
'set'])) {
534 static fn(array $v):
int => $v[
'usr_id'],
545 preg_match(
'/cmdNode=([A-Za-z0-9]+%3)+[A-Za-z0-9]+&/i', $url, $matches);
546 if (empty($matches[0])) {
549 $replacement = str_replace(
'%3',
':', $matches[0]);
550 return str_replace($matches[0], $replacement, $url);
getRows(Table\DataRowBuilder $row_builder, array $visible_column_ids, Range $range, Order $order, ?array $filter_data, ?array $additional_parameters)
executeAction(string $action, array $affected_items)
const COLUMN_INTERACTION_TYPE
exportTestUserInteractions(array $affected_items)
buildInteractionTypesOptionsForFilter()
const ACTION_ADDITIONAL_INFORMATION
buildLogsFromAffectedItems(array $affected_items)
static _getAllReferences(int $id)
get all reference ids for object ID
__construct(private readonly TestLoggingRepository $logging_repository, private readonly TestLogger $logger, private readonly TestLogViewer $log_viewer, private readonly TitleColumnsBuilder $title_builder, private readonly GeneralQuestionPropertiesRepository $question_repo, private readonly \ilUIService $ui_service, private readonly UIFactory $ui_factory, private readonly UIRenderer $ui_renderer, private readonly \ilLanguage $lng, private \ilGlobalTemplateInterface $tpl, private readonly URLBuilder $url_builder, private readonly URLBuilderToken $action_parameter_token, private readonly URLBuilderToken $row_id_token, private readonly StreamDelivery $stream_delivery, private readonly \ilObjUser $current_user, private readonly ?int $ref_id=null)
const QUERY_PARAMETER_NAME_SPACE
showAdditionalDetails(string $affected_item)
const COLUMN_LOG_ENTRY_TYPE
const FILTER_FIELD_LOG_ENTRY_TYPE
prepareFilterData(array $filter_array)
Both the subject and the direction need to be specified when expressing an order. ...
const FILTER_FIELD_QUESTION_TITLE
while($session_entry=$r->fetchRow(ilDBConstants::FETCHMODE_ASSOC)) return null
buildLogEntryTypesOptionsForFilter()
const ACTION_CONFIRM_DELETE
const COLUMN_CORRESPONDING_TEST
showErrorModal(string $message)
deleteTestUserInteractions(array $affected_items)
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
static _getIdsForTitle(string $title, string $type='', bool $partial_match=false)
showConfirmTestUserInteractionsDeletion(array $affected_items)
initializeFilterAndData()
getTotalRowCount(?array $filter_data, ?array $additional_parameters)
const ACTION_TOKEN_STRING
const FILTER_FIELD_INTERACTION_TYPE
const FILTER_FIELD_TEST_TITLE
filter(string $filter_id, $class_path, string $cmd, bool $activated=true, bool $expanded=true)
const ACTION_ID_SHOW_ADDITIONAL_INFO
const ACTION_EXPORT_AS_CSV
const FILTER_FIELD_PARTICIPANT
A simple class to express a naive range of whole positive numbers.
const FILTER_FIELD_PERIOD
exit
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
unmaskCmdNodesFromBuilder(string $url)
2024-05-07 skergomard: This is a workaround as I didn't find another way
extractIdsFromUserQuery(array $response)