ILIAS  trunk Revision v11.0_alpha-1689-g66c127b4ae8
All Data Structures Namespaces Files Functions Variables Enumerations Enumerator Modules Pages
class.ilQuestionPageParser.php
Go to the documentation of this file.
1 <?php
2 
20 
28 {
32  protected bool $in_properties;
36  protected bool $in_glossary_definition = false;
37  protected array $media_meta_cache = [];
38  protected bool $media_meta_start = false;
39  protected array $pg_mapping = [];
40  protected array $mobs_with_int_links = [];
41  protected bool $inside_code = false;
42  protected string $content_type = '';
43  public ilTree $tree;
44  public array $cnt = []; // counts open elements
45  public array $current_element = []; // store current element type
46  public ?ilObjLearningModule $learning_module = null; // current learning module
47  public ?ilPageObject $page_object = null; // current page object
49  public array $structure_objects = []; // array of current structure objects
51  public ?object $current_object = null; // at the time a LearningModule, PageObject or StructureObject
52  public ?ilLMTree $lm_tree = null;
53  public array $pg_into_tree = [];
54  public array $st_into_tree = [];
55  public array $container = [];
56  public bool $in_page_object = false; // are we currently within a PageObject? true/false
57  public bool $in_meta_data = false; // are we currently within MetaData? true/false
58  public bool $in_media_object = false;
59  public bool $in_file_item = false;
60  public bool $in_glossary = false;
61  public bool $in_map_area = false;
64  public array $pages_to_parse = [];
65  public array $mob_mapping = [];
66  public array $file_item_mapping = [];
67  public ?ilMediaItem $media_item = null; // current media item
68  public string $loc_type = ''; // current location type
69  public ?ilMapArea $map_area = null; // current map area
70  public array $link_targets = []; // stores all objects by import id
71  public array $qst_mapping = [];
72  public bool $metadata_parsing_disabled = false;
73  public bool $in_meta_meta_data = false;
74  protected array $glossary_term_map = [];
75  protected ilLogger $log;
78  protected string $cur_qid = '';
79  protected string $chr_data = '';
80  protected bool $in_media_item = false;
81 
82  public function __construct(
83  private readonly ilObject $content_object,
84  string $xml_file,
85  private readonly string $importdir
86  ) {
87  global $DIC;
88  $lng = $DIC->language();
89  $tree = $DIC->repositoryTree();
90 
91  $this->log = ilLoggerFactory::getLogger('lm');
92 
93  parent::__construct($xml_file);
94  $this->cnt = [];
95  $this->current_element = [];
96  $this->structure_objects = [];
97  $this->st_into_tree = [];
98  $this->pg_into_tree = [];
99  $this->pages_to_parse = [];
100  $this->mobs_with_int_links = [];
101  $this->mob_mapping = [];
102  $this->file_item_mapping = [];
103  $this->pg_mapping = [];
104  $this->link_targets = [];
105  $this->lng = $lng;
106  $this->tree = $tree;
107  $this->inside_code = false;
108  $this->qst_mapping = [];
109  $this->content_type = $this->content_object->getType();
110  $this->metadata_parsing_disabled = false;
111 
112  if ($this->content_type !== 'tst' && $this->content_type !== 'qpl') {
113  $this->lm_tree = new ilLMTree($this->content_object->getId());
114  }
115  }
116 
120  public function setHandlers($xml_parser): void
121  {
122  xml_set_element_handler($xml_parser, $this->handlerBeginTag(...), $this->handlerEndTag(...));
123  xml_set_character_data_handler($xml_parser, $this->handlerCharacterData(...));
124  }
125 
126  public function setImportMapping(?ilImportMapping $mapping = null): void
127  {
128  $this->mapping = $mapping;
129  }
130 
131  public function startParsing(): void
132  {
133  $this->log->debug('start');
134 
135  parent::startParsing();
136  $this->storeTree();
137  $this->processPagesToParse();
138  $this->copyFileItems();
139  }
140 
144  public function storeTree(): void
145  {
146  $ilLog = $this->log;
147 
148  foreach ($this->st_into_tree as $st) {
149  $this->lm_tree->insertNode($st['id'], $st['parent']);
150  if (is_array($this->pg_into_tree[$st['id']])) {
151  foreach ($this->pg_into_tree[$st['id']] as $pg) {
152  $pg_id = 0;
153  switch ($pg['type']) {
154  case 'pg_alias':
155  if ($this->pg_mapping[$pg['id']] == '') {
156  $ilLog->write('LM Import: No PageObject for PageAlias ' .
157  $pg['id'] . ' found! (Please update export installation to ILIAS 3.3.0)');
158 
159  // Jump two levels up. First level is switch
160  continue 2;
161  }
162  $pg_id = $this->pg_mapping[$pg['id']];
163  break;
164 
165  case 'pg':
166  $pg_id = $pg['id'];
167  break;
168  }
169  if (!$this->lm_tree->isInTree($pg_id)) {
170  $this->lm_tree->insertNode($pg_id, $st['id']);
171  }
172  }
173  }
174  }
175  }
176 
180  public function processPagesToParse(): void
181  {
182  // outgoin internal links
183  foreach ($this->pages_to_parse as $page_id) {
184  $page_arr = explode(':', $page_id);
185  $page_obj = null;
186  switch ($page_arr[0]) {
187  case 'lm':
188  switch ($this->content_object->getType()) {
189  case 'lm':
190  $page_obj = new ilLMPage($page_arr[1]);
191  break;
192 
193  default:
194  die('Unknown content type ' . $this->content_object->getType());
195  }
196 
197  break;
198 
199  case 'term':
200  $page_obj = new ilGlossaryDefPage($page_arr[1]);
201  break;
202 
203  case 'qpl':
204  $page_obj = new ilAssQuestionPage($page_arr[1]);
205  break;
206  }
207  $page_obj->buildDom();
208  $page_obj->resolveIntLinks();
209  $page_obj->resolveIIMMediaAliases($this->mob_mapping);
210  if ($this->content_type == 'lm') {
211  $page_obj->resolveQuestionReferences($this->qst_mapping);
212  }
213  $page_obj->update(false);
214 
215  if ($page_arr[0] == 'term') {
216  $term = new ilGlossaryTerm($page_arr[1]);
217  $term->updateShortText();
218  }
219 
220  unset($page_obj);
221  }
222 
223  foreach ($this->mobs_with_int_links as $mob_id) {
225  }
226 
227  $done = [];
228  foreach ($this->link_targets as $link_target) {
229  $parsed = $this->parseLinkTarget((string) $link_target);
230  if (!isset($parsed)) {
231  continue;
232  }
233  //echo "doin link target:".$link_target.":<br>";
235  $parsed['target_type'],
236  $parsed['target_id'],
237  $parsed['target_inst']
238  );
239  foreach ($sources as $key => $source) {
240  if (in_array($key, $done)) {
241  continue;
242  }
243  $type_arr = explode(':', $source['type']);
244 
245  // content object pages
246  if ($type_arr[1] == 'pg') {
247  if (ilPageObject::_exists($type_arr[0], $source['id'])) {
248  $page_object = ilPageObjectFactory::getInstance($type_arr[0], $source['id']);
249  $page_object->buildDom();
250  $page_object->resolveIntLinks();
251  $page_object->update();
252  unset($page_object);
253  }
254  }
255 
256  // eventually correct links in questions to learning modules
257  if ($type_arr[0] === 'qst') {
258  assQuestion::instantiateQuestion($source['id'])->resolveSuggestedSolutionLinks();
259  }
260  $done[$key] = $key;
261  }
262  }
263  }
264 
268  private function copyFileItems(): void
269  {
270  foreach ($this->file_item_mapping as $origin_id => $file_id) {
271  if (empty($origin_id)) {
272  continue;
273  }
274  $obj_dir = $origin_id;
275  $source_dir = $this->importdir . DIRECTORY_SEPARATOR . 'objects' . DIRECTORY_SEPARATOR . $obj_dir;
276 
277  $file_obj = new ilObjFile($file_id, false);
278  if (is_dir($source_dir)) {
279  $files = scandir($source_dir, SCANDIR_SORT_DESCENDING);
280  if ($files !== false && $files !== [] && is_file($source_dir . '/' . $files[0])) {
281  $file = fopen($source_dir . '/' . $files[0], 'rb');
282  $file_stream = Streams::ofResource($file);
283  $file_obj->appendStream($file_stream, $files[0]);
284  }
285  }
286  $file_obj->update();
287  }
288  }
289 
294  public function setQuestionMapping(array $a_map): void
295  {
296  $this->qst_mapping = $a_map;
297  }
298 
299  public function beginElement(string $a_name): void
300  {
301  if (!isset($this->status['$a_name'])) {
302  $this->cnt[$a_name] = 1;
303  } else {
304  $this->cnt[$a_name]++;
305  }
306  $this->current_element[count($this->current_element)] = $a_name;
307  }
308 
309  public function endElement(string $a_name): void
310  {
311  $this->cnt[$a_name]--;
312  unset($this->current_element[count($this->current_element) - 1]);
313  }
314 
315  public function getCurrentElement(): string
316  {
317  return ($this->current_element[count($this->current_element) - 1] ?? '');
318  }
319 
320  public function getOpenCount(string $a_name): int
321  {
322  if (isset($this->cnt[$a_name])) {
323  return $this->cnt[$a_name];
324  }
325  return 0;
326  }
327 
328  public function buildTag(
329  string $type,
330  string $name,
331  array $attr = []
332  ): string {
333  $tag = '<';
334 
335  if ($type == 'end') {
336  $tag .= '/';
337  }
338 
339  $tag .= $name;
340 
341  if (is_array($attr)) {
342  foreach ($attr as $k => $v) {
343  $tag .= ' ' . $k . "='$v'";
344  }
345  }
346 
347  $tag .= '>';
348 
349  return $tag;
350  }
351 
352  public function handlerBeginTag($a_xml_parser, string $a_name, array $a_attribs): void
353  {
354  switch ($a_name) {
355  case 'ContentObject':
356  $this->current_object = $this->content_object;
357  if ($a_attribs['Type'] == 'Glossary') {
358  $this->glossary_object = $this->content_object;
359  }
360  break;
361 
362  case 'StructureObject':
364  $lm = $this->content_object;
365  $this->structure_objects[count($this->structure_objects)]
366  = new ilStructureObject($lm);
367  $this->current_object = $this->structure_objects[count($this->structure_objects) - 1];
368  $this->current_object->setLMId($this->content_object->getId());
369  // new meta data handling: we create the structure
370  // object already here, this should also create a
371  // md entry
372  $this->current_object->create(true);
373  break;
374 
375  case 'PageObject':
376  $this->in_page_object = true;
377  $this->cur_qid = '';
378  if ($this->content_type !== 'tst' && $this->content_type !== 'qpl') {
380  $lm = $this->content_object;
381  $this->lm_page_object = new ilLMPageObject($lm);
382  $this->page_object = new ilLMPage();
383  $this->lm_page_object->setLMId($this->content_object->getId());
384  $this->lm_page_object->assignPageObject($this->page_object);
385  $this->current_object = $this->lm_page_object;
386  } else {
387  $this->page_object = new ilAssQuestionPage();
388  }
389  break;
390 
391  case 'PageAlias':
392  throw new ilLMException('Page Alias not supported.');
393 
394  case 'MediaObject':
395  case 'InteractiveImage':
396  if ($a_name == 'MediaObject') {
397  $this->in_media_object = true;
398  }
399  $this->media_meta_start = true;
400  $this->media_meta_cache = [];
401  $this->media_object = new ilObjMediaObject();
402  $this->media_object->create(true, false);
403  break;
404 
405  case 'MediaAlias':
406  $this->media_object->setAlias(true);
407  $this->media_object->setImportId($a_attribs['OriginId']);
408  $this->mob_mapping[$this->media_object->getImportId()] = $this->media_object->getId();
409  if (is_object($this->page_object)) {
410  $this->page_object->needsImportParsing(true);
411  }
412  break;
413 
414  case 'MediaItem':
415  case 'MediaAliasItem':
416  $this->in_media_item = true;
417  $this->media_item = new ilMediaItem();
418  $this->media_item->setPurpose($a_attribs['Purpose']);
419  break;
420 
421  case 'Layout':
422  if (is_object($this->media_object) && $this->in_media_object) {
423  $this->media_item->setWidth($a_attribs['Width'] ?? '');
424  $this->media_item->setHeight($a_attribs['Height'] ?? '');
425  $this->media_item->setHAlign($a_attribs['HorizontalAlign'] ?? '');
426  }
427  break;
428 
429  case 'Parameter':
430  if (is_object($this->media_object) && $this->in_media_object) {
431  $this->media_item->setParameter($a_attribs['Name'], $a_attribs['Value']);
432  }
433  break;
434 
435  case 'MapArea':
436  $this->in_map_area = true;
437  $this->map_area = new ilMapArea();
438  $this->map_area->setShape($a_attribs['Shape']);
439  $this->map_area->setCoords($a_attribs['Coords']);
440  $this->map_area->setHighlightMode($a_attribs['HighlightMode']);
441  $this->map_area->setHighlightClass($a_attribs['HighlightClass']);
442  break;
443 
444  case 'Glossary':
445  $this->in_glossary = true;
446  if ($this->content_object->getType() != 'glo') {
447  $this->glossary_object = new ilObjGlossary();
448  $this->glossary_object->setTitle('');
449  $this->glossary_object->setDescription('');
450  $this->glossary_object->create(true);
451  $this->glossary_object->createReference();
452  $parent = $this->tree->getParentNodeData($this->content_object->getRefId());
453  $this->glossary_object->putInTree($parent['child']);
454  $this->glossary_object->setPermissions($parent['child']);
455  }
456  $this->current_object = $this->glossary_object;
457  break;
458 
459  case 'GlossaryItem':
460  $this->glossary_term = new ilGlossaryTerm();
461  $this->glossary_term->setGlossaryId($this->glossary_object->getId());
462  $this->glossary_term->setLanguage($a_attribs['Language']);
463  $this->glossary_term->setImportId($a_attribs['Id']);
464  $parsed = $this->parseLinkTarget((string) $a_attribs['Id']);
465  if (isset($parsed)) {
466  $this->link_targets[$a_attribs["Id"]] = $a_attribs['Id'];
467  }
468 
469  // no break
470  case 'Definition':
471  $this->in_glossary_definition = true;
472  $this->page_object = new ilGlossaryDefPage();
473  $this->page_object->setParentId($this->glossary_term->getGlossaryId());
474  $this->glossary_term->assignPageObject($this->page_object);
475  $this->current_object = $this->glossary_term;
476  // see bug #12465, we need to clear xml after creation, since it will be <PageObject></PageObject>
477  // otherwise, and appendXML calls will lead to '<PageObject></PageObject><PageObject>....</PageObject>'
478  $this->page_object->setXMLContent('');
479  break;
480 
481  case 'FileItem':
482  $this->in_file_item = true;
483  $this->file_item = new ilObjFile();
484  $this->file_item->setTitle('dummy');
485  $this->file_item->setMode('filelist');
486  if (is_object($this->page_object)) {
487  $this->page_object->needsImportParsing(true);
488  }
489  break;
490 
491  case 'Paragraph':
492  if ($a_attribs['Characteristic'] == 'Code') {
493  $this->inside_code = true;
494  }
495  break;
496 
497  case 'Properties':
498  $this->in_properties = true;
499  break;
500 
501  case 'Property':
502  if ($this->content_object->getType() == 'lm') {
503  switch ($a_attribs['Name']) {
504  case 'Layout':
505  $this->content_object->setLayout($a_attribs['Value']);
506  break;
507 
508  case 'PageHeader':
509  $this->content_object->setPageHeader($a_attribs['Value']);
510  break;
511 
512  case 'TOCMode':
513  $this->content_object->setTOCMode($a_attribs['Value']);
514  break;
515 
516  case 'ActiveLMMenu':
517  $this->content_object->setActiveLMMenu(
518  ilUtil::yn2tf($a_attribs['Value'])
519  );
520  break;
521 
522  case 'ActiveNumbering':
523  $this->content_object->setActiveNumbering(
524  ilUtil::yn2tf($a_attribs['Value'])
525  );
526  break;
527 
528  case 'ActiveTOC':
529  $this->content_object->setActiveTOC(
530  ilUtil::yn2tf($a_attribs['Value'])
531  );
532  break;
533 
534  case 'ActivePrintView':
535  $this->content_object->setActivePrintView(
536  ilUtil::yn2tf($a_attribs['Value'])
537  );
538  break;
539 
540  case 'CleanFrames':
541  $this->content_object->setCleanFrames(
542  ilUtil::yn2tf($a_attribs['Value'])
543  );
544  break;
545 
546  case 'PublicNotes':
547  $this->content_object->setPublicNotes(
548  ilUtil::yn2tf($a_attribs['Value'])
549  );
550  break;
551 
552  case 'HistoryUserComments':
553  $this->content_object->setHistoryUserComments(
554  ilUtil::yn2tf($a_attribs['Value'])
555  );
556  break;
557 
558  case 'Rating':
559  $this->content_object->setRating(
560  ilUtil::yn2tf($a_attribs['Value'])
561  );
562  break;
563 
564  case 'RatingPages':
565  $this->content_object->setRatingPages(
566  ilUtil::yn2tf($a_attribs['Value'])
567  );
568  break;
569 
570  case 'HeaderPage':
571  if ($a_attribs['Value'] != '') {
572  if ($this->pg_mapping[$a_attribs['Value']] > 0) {
573  $this->content_object->setHeaderPage(
574  $this->pg_mapping[$a_attribs['Value']]
575  );
576  }
577  }
578  break;
579 
580  case 'FooterPage':
581  if ($a_attribs['Value'] != '') {
582  if ($this->pg_mapping[$a_attribs['Value']] > 0) {
583  $this->content_object->setFooterPage(
584  $this->pg_mapping[$a_attribs['Value']]
585  );
586  }
587  }
588  break;
589 
590  case 'LayoutPerPage':
591  $this->content_object->setLayoutPerPage($a_attribs['Value']);
592  break;
593 
594  case 'ProgressIcons':
595  $this->content_object->setProgressIcons($a_attribs['Value']);
596  break;
597 
598  case 'StoreTries':
599  $this->content_object->setStoreTries($a_attribs['Value']);
600  break;
601 
602  case 'RestrictForwardNavigation':
603  $this->content_object->setRestrictForwardNavigation($a_attribs['Value']);
604  break;
605 
606  case 'DisableDefaultFeedback':
607  $this->content_object->setDisableDefaultFeedback($a_attribs['Value']);
608  break;
609  }
610  }
611  break;
612 
616  case 'MetaData':
617  $this->in_meta_data = true;
618  // media obejct meta data handling
619  // is done in the 'Identifier' begin tag processing
620  // the rest is done here
621  if (!$this->in_media_object) {
622  if ($this->content_type != 'tst' && $this->content_type != 'qpl') {
623  // type pg/st
624  if ($this->current_object->getType() == 'st'
625  || $this->current_object->getType() == 'pg') {
626  // late creation of page object
627  if ($this->current_object->getType() == 'pg') {
628  $this->lm_page_object->create(true);
629  }
630  $this->md = new ilMD(
631  $this->content_object->getId(),
632  $this->current_object->getId(),
633  $this->current_object->getType()
634  );
635  }
636  // type term
637  elseif ($this->current_object->getType() == 'term') {
638  $this->md = new ilMD(
639  $this->glossary_object->getId(),
640  $this->current_object->getId(),
641  $this->current_object->getType()
642  );
643  }
644  // type lm, dbk, glo
645  else {
646  if ($this->processMeta()) {
647  $this->md = new ilMD(
648  $this->current_object->getId(),
649  0,
650  $this->current_object->getType()
651  );
652  }
653  }
654  } else {
655  // type qpl or tst
656  $this->md = new ilMD(
657  $this->content_object->getId(),
658  0,
659  $this->current_object->getType()
660  );
661  if ($this->md->getGeneral() != false) {
662  $this->metadata_parsing_disabled = true;
663  $this->enableMDParsing(false);
664  }
665  }
666  }
667  break;
668 
669  // Identifier
670  case 'Identifier':
671  // begin-patch optes_lok_export
672  if ($this->in_meta_data && $this->current_object instanceof ilStructureObject) {
673  if ($this->mapping instanceof ilImportMapping) {
674  $import_id_parsed = ilUtil::parseImportId($a_attribs['Entry']);
675  if ($import_id_parsed['type'] == 'st') {
676  $this->mapping->addMapping(
677  'components/ILIAS/LearningModule',
678  'lm_tree',
679  $import_id_parsed['id'],
680  $this->current_object->getId()
681  );
682  }
683  }
684  }
685  // end-patch optes_lok_export
686 
687  // please note: Meta-Metadata and MetaData are different tags!
688  if (!$this->in_meta_meta_data) {
689  if ($this->in_meta_data && !$this->in_glossary_definition) {
690  if (!$this->in_media_object) {
691  $this->current_object->setImportId($a_attribs['Entry']);
692  }
693  // #40680 add a link target only if it is an internal ILIAS link
694  // Export from IMS UCAN sets something like 'IMSm-i1e79762'
695  $parsed = $this->parseLinkTarget($a_attribs["Entry"]);
696  if (isset($parsed)) {
697  $this->link_targets[$a_attribs['Entry']] = $a_attribs['Entry'];
698  }
699  }
700  if ($this->in_file_item) {
701  if (!isset($this->file_item_mapping[$a_attribs['Entry']])
702  || $this->file_item_mapping[$a_attribs['Entry']] === '') {
703  $this->file_item->create();
704  $this->file_item->setImportId($a_attribs['Entry']);
705  $this->file_item_mapping[$a_attribs['Entry']] = $this->file_item->getId();
706  }
707  }
708  if ($this->in_meta_data && $this->in_media_object) {
709  $mob_id = $this->mob_mapping[$a_attribs['Entry']];
710 
711  // within learning module import, usually a media object
712  // has already been created with a media alias tag
713  if ($mob_id > 0) {
714  $this->media_object = new ilObjMediaObject($mob_id);
715  } else { // in glossaries the media objects precede the definitions
716  // so we don't have an object already
717  $this->media_object = new ilObjMediaObject();
718  $this->media_object->create(true, false);
719  $this->mob_mapping[$a_attribs['Entry']]
720  = $this->media_object->getId();
721  }
722  $this->media_object->setImportId($a_attribs['Entry']);
723  $this->md = new ilMD(
724  0,
725  $this->media_object->getId(),
726  'mob'
727  );
728  $this->emptyMediaMetaCache($a_xml_parser);
729  }
730  }
731  break;
732 
733  case 'Meta-Metadata':
734  $this->in_meta_meta_data = true;
735  break;
736 
737  // Internal Link
738  case 'IntLink':
739  if (is_object($this->page_object)) {
740  $this->page_object->setContainsIntLink(true);
741  }
742  if ($this->in_map_area) {
743  $this->map_area->setLinkType(IL_INT_LINK);
744  $this->map_area->setTarget($a_attribs['Target']);
745  $this->map_area->setType($a_attribs['Type']);
746  $this->map_area->setTargetFrame($a_attribs['TargetFrame']);
747  if (is_object($this->media_object)) {
748  $this->media_object->setContainsIntLink(true);
749  }
750  }
751  break;
752 
753  // External Link
754  case 'ExtLink':
755  if ($this->in_map_area) {
756  $this->map_area->setLinkType(IL_EXT_LINK);
757  $this->map_area->setHref($a_attribs['Href']);
758  $this->map_area->setExtTitle($a_attribs['Title']);
759  }
760  break;
761 
762  // Question
763  case 'Question':
764  $this->cur_qid = $a_attribs['QRef'];
765  $this->page_object->setContainsQuestion(true);
766  break;
767 
768  case 'Location':
769  $this->loc_type = $a_attribs['Type'];
770  break;
771  }
772  $this->beginElement($a_name);
773 
774  // append content to page xml content
775  if (($this->in_page_object || $this->in_glossary_definition)
776  && !$this->in_meta_data && !$this->in_media_object) {
777  if ($a_name == 'Definition') {
778  $app_name = 'PageObject';
779  $app_attribs = [];
780  } else {
781  $app_name = $a_name;
782  $app_attribs = $a_attribs;
783  }
784 
785  // change identifier entry of file items to new local file id
786  if ($this->in_file_item && $app_name == 'Identifier') {
787  $app_attribs['Entry'] = 'il__file_' . $this->file_item_mapping[$a_attribs['Entry']];
788  }
789 
790  $this->page_object->appendXMLContent($this->buildTag('start', $app_name, $app_attribs));
791  }
792 
793 
794  // call meta data handler
795  if ($this->in_meta_data && $this->processMeta()) {
796  // cache beginning of meta data within media object tags
797  // (we need to know the id at the begin of meta elements within
798  // media objects, after the 'Identifier' tag has been processed
799  // we send the cached data to the meta xml handler)
800  if ($this->in_media_object && $this->media_meta_start) {
801  $this->media_meta_cache[] =
802  ['type' => 'handlerBeginTag', 'par1' => $a_name, 'par2' => $a_attribs];
803  } else {
804  if ($a_name == 'Identifier') {
805  if (!$this->in_media_object) {
806  $a_attribs['Entry'] = 'il__' . $this->current_object->getType() .
807  '_' . $this->current_object->getId();
808  } else {
809  $a_attribs['Entry'] = 'il__mob' .
810  '_' . $this->media_object->getId();
811  }
812  $a_attribs['Catalog'] = 'ILIAS';
813  }
814 
815  parent::handlerBeginTag($a_xml_parser, $a_name, $a_attribs);
816  }
817  }
818  }
819 
820  public function processMeta(): bool
821  {
822  // do not process second meta block in (ilias3) glossaries
823  // which comes right after the 'Glossary' tag
824  if ($this->content_object->getType() == 'glo' &&
825  $this->in_glossary && !$this->in_media_object
827  return false;
828  }
829 
830  return true;
831  }
832 
833 
834  public function handlerEndTag($a_xml_parser, string $a_name): void
835  {
836  // call meta data handler
837  if ($this->in_meta_data && $this->processMeta()) {
838  // cache beginning of meta data within media object tags
839  // (we need to know the id, after that we send the cached data
840  // to the meta xml handler)
841  if ($this->in_media_object && $this->media_meta_start) {
842  $this->media_meta_cache[] =
843  ['type' => 'handlerEndTag', 'par1' => $a_name];
844  } else {
845  parent::handlerEndTag($a_xml_parser, $a_name);
846  }
847  }
848 
849  // append content to page xml content
850  if (($this->in_page_object || $this->in_glossary_definition)
851  && !$this->in_meta_data && !$this->in_media_object) {
852  $app_name = ($a_name == 'Definition')
853  ? 'PageObject'
854  : $a_name;
855  $this->page_object->appendXMLContent($this->buildTag('end', $app_name));
856  }
857 
858  switch ($a_name) {
859  case 'StructureObject':
860  unset($this->structure_objects[count($this->structure_objects) - 1]);
861  break;
862 
863  case 'PageObject':
864  $this->in_page_object = false;
865  if ($this->content_type != 'tst' && $this->content_type != 'qpl') {
866  //if (!$this->lm_page_object->isAlias()) {
867  $this->page_object->updateFromXML();
868  $this->pg_mapping[$this->lm_page_object->getImportId()]
869  = $this->lm_page_object->getId();
870 
871  if ($this->mapping instanceof ilImportMapping) {
872  $import_id_parsed = ilUtil::parseImportId($this->lm_page_object->getImportId());
873  if ($import_id_parsed['type'] == 'pg') {
874  $this->mapping->addMapping(
875  'components/ILIAS/LearningModule',
876  'pg',
877  $import_id_parsed['id'],
878  $this->lm_page_object->getId()
879  );
880  }
881  }
882 
883  // collect pages with internal links
884  if ($this->page_object->containsIntLink()) {
885  $this->pages_to_parse['lm:' . $this->page_object->getId()] = 'lm:' . $this->page_object->getId();
886  }
887 
888  // collect pages with mobs or files
889  if ($this->page_object->needsImportParsing()) {
890  $this->pages_to_parse['lm:' . $this->page_object->getId()] = 'lm:' . $this->page_object->getId();
891  }
892 
893  // collect pages with questions
894  if ($this->page_object->getContainsQuestion()) {
895  $this->pages_to_parse['lm:' . $this->page_object->getId()] = 'lm:' . $this->page_object->getId();
896  }
897  //}
898  } else {
899  $xml = $this->page_object->getXMLContent();
900  if ($this->cur_qid != '') {
901  $ids = $this->qst_mapping[$this->cur_qid] ?? ['pool' => 0, 'test' => 0];
902  if ($ids['pool'] > 0) {
903  // question pool question
904  $page = new ilAssQuestionPage($ids['pool']);
905  $xmlcontent = str_replace(
906  $this->cur_qid,
907  'il__qst_' . $ids['pool'],
908  $xml
909  );
910  $page->setXMLContent($xmlcontent);
911  $page->updateFromXML();
912  if ($this->page_object->needsImportParsing()) {
913  $this->pages_to_parse['qpl:' . $page->getId()] = 'qpl:' . $page->getId();
914  }
915  unset($page);
916  }
917  if ($ids['test'] > 0) {
918  // test question
919  $page = new ilAssQuestionPage($ids['test']);
920  $xmlcontent = str_replace(
921  $this->cur_qid,
922  'il__qst_' . $ids['test'],
923  $xml
924  );
925  $page->setXMLContent($xmlcontent);
926  $page->updateFromXML();
927  if ($this->page_object->needsImportParsing()) {
928  $this->pages_to_parse['qpl:' . $page->getId()] = 'qpl:' . $page->getId();
929  }
930  unset($page);
931  }
932  }
933  }
934 
935  // if we are within a structure object: put page in tree
936  $cnt = count($this->structure_objects);
937  if ($cnt > 0) {
938  $parent_id = $this->structure_objects[$cnt - 1]->getId();
939  $this->pg_into_tree[$parent_id][] = ['type' => 'pg', 'id' => $this->lm_page_object->getId()];
940  }
941 
942  unset($this->page_object);
943  unset($this->lm_page_object);
944  unset($this->container[count($this->container) - 1]);
945  break;
946 
947  case 'MediaObject':
948  case 'InteractiveImage':
949  if ($a_name == 'MediaObject') {
950  $this->in_media_object = false;
951  }
952 
953  if (empty($this->mob_mapping[$this->media_object->getImportId()])) {
954  // create media object
955  // media items are saves for mobs outside of
956  // pages only
957  $this->media_object->create(true, false);
958 
959  // collect mobs with internal links
960  if ($this->media_object->containsIntLink()) {
961  $this->mobs_with_int_links[] = $this->media_object->getId();
962  }
963 
964  $this->mob_mapping[$this->media_object->getImportId()]
965  = $this->media_object->getId();
966  } else {
967  // get the id from mapping
968  $this->media_object->setId($this->mob_mapping[$this->media_object->getImportId()]);
969 
970  // update 'real' (no alias) media object
971  // (note: we overwrite any data from the media object
972  // created by an MediaAlias, only the data of the real
973  // object is stored in db separately; data of the
974  // MediaAliases are within the page XML
975  if (!$this->media_object->isAlias()) {
976  // now the media items are saved within the db
977  $this->media_object->update();
978 
979  // collect mobs with internal links
980  if ($this->media_object->containsIntLink()) {
981  $this->mobs_with_int_links[] = $this->media_object->getId();
982  }
983  }
984  }
985 
986  // append media alias to page, if we are in a page
987  if ($this->in_page_object || $this->in_glossary_definition) {
988  if ($a_name != 'InteractiveImage') {
989  $this->page_object->appendXMLContent($this->media_object->getXML(IL_MODE_ALIAS));
990  }
991  }
992 
993  break;
994 
995  case 'MediaAliasItem':
996  $this->in_media_item = false;
997  $this->media_object->addMediaItem($this->media_item);
998  break;
999  case 'MediaItem':
1000  $this->in_media_item = false;
1001  $import_dir = $this->importdir . DIRECTORY_SEPARATOR . 'objects' . DIRECTORY_SEPARATOR . $this->media_object->getImportId();
1002  if (!file_exists($import_dir)
1003  || !is_dir($import_dir)) {
1004  $this->media_object->addMediaItem($this->media_item);
1005  break;
1006  }
1007 
1008  $dir_handle = opendir($import_dir);
1009  while (($file = readdir($dir_handle)) !== false) {
1010  if ($file !== $this->media_item->getLocation()) {
1011  continue;
1012  }
1013 
1014  $this->media_object->addMediaItemFromLocalFile(
1015  $this->media_item->getPurpose(),
1016  $import_dir . DIRECTORY_SEPARATOR . $file,
1017  $file
1018  );
1019  break;
1020  }
1021  closedir($dir_handle);
1022  break;
1023 
1024  case 'MapArea':
1025  $this->in_map_area = false;
1026  $this->media_item->addMapArea($this->map_area);
1027  break;
1028 
1029  case 'Properties':
1030  $this->in_properties = false;
1031  if ($this->content_object->getType() == 'lm') {
1032  $this->content_object->update();
1033  }
1034  break;
1035 
1036  case 'MetaData':
1037  $this->in_meta_data = false;
1038  if (strtolower(get_class($this->current_object)) == 'illmpageobject' && !$this->in_media_object) {
1039  // Metadaten eines PageObjects sichern in NestedSet
1040  if (is_object($this->lm_page_object)) {
1041  // update title/description of page object
1042  $this->current_object->MDUpdateListener('General');
1044  $this->current_object->getId(),
1045  $this->current_object->getImportId()
1046  );
1047  }
1048  } elseif ((strtolower(get_class($this->current_object)) == 'ilobjquestionpool' ||
1049  strtolower(get_class($this->current_object)) == 'ilobjtest') &&
1050  !$this->in_media_object) {
1051  // !$this->in_media_object && !$this->in_page_object)
1052  // changed for imports of ILIAS 2 Tests where PageObjects could have
1053  // Metadata sections (Helmut Schottmüller, 2005-12-02)
1054  if ($this->metadata_parsing_disabled) {
1055  $this->enableMDParsing(true);
1056  } else {
1057  if ($this->in_page_object && !is_null($this->page_object)) {
1058  /*
1059  $this->page_object->MDUpdateListener('General');
1060  ilLMObject::_writeImportId(
1061  $this->page_object->getId(),
1062  $this->page_object->getImportId()
1063  );*/
1064  } else {
1065  $this->current_object->MDUpdateListener('General');
1067  $this->current_object->getId(),
1068  $this->current_object->getImportId()
1069  );
1070  }
1071  }
1072  } elseif (strtolower(get_class($this->current_object)) == 'ilstructureobject') { // save structure object at the end of its meta block
1073  // determine parent
1074  $cnt = count($this->structure_objects);
1075  if ($cnt > 1) {
1076  $parent_id = $this->structure_objects[$cnt - 2]->getId();
1077  } else {
1078  $parent_id = $this->lm_tree->getRootId();
1079  }
1080 
1081  $this->st_into_tree[] = ['id' => $this->current_object->getId(),
1082  'parent' => $parent_id];
1083 
1084  // update title/description of structure object
1085  $this->current_object->MDUpdateListener('General');
1087  $this->current_object->getId(),
1088  $this->current_object->getImportId()
1089  );
1090  } elseif (strtolower(get_class($this->current_object)) == 'ilobjlearningmodule' ||
1091  strtolower(get_class($this->current_object)) == 'ilobjcontentobject' ||
1092  (strtolower(get_class($this->current_object)) == 'ilobjglossary' && $this->in_glossary)) {
1093  // todo: saving of md? getting title/descr and
1094  // set it for current object
1095  } elseif (strtolower(get_class($this->current_object)) == 'ilglossaryterm' && !$this->in_media_object) {
1096  // now on top
1097 
1098  $this->page_object->setId($this->glossary_term->getId());
1099  $this->page_object->updateFromXML();
1100 
1101  // todo: saving of md? getting title/descr and
1102  // set it for current object
1103  }
1104 
1105 
1106  if (strtolower(get_class($this->current_object)) == 'ilobjlearningmodule' ||
1107  strtolower(get_class($this->current_object)) == 'ilobjglossary') {
1108  if (strtolower(get_class($this->current_object)) == 'ilobjglossary' &&
1109  $this->content_object->getType() != 'glo') {
1110  $this->current_object->setTitle($this->content_object->getTitle() . ' - ' .
1111  $this->lng->txt('glossary'));
1112  }
1113 
1114  $this->current_object->MDUpdateListener('General');
1115  }
1116 
1117  if ($this->in_media_object) {
1118  $this->media_object->MDUpdateListener('General');
1119  }
1120 
1121  break;
1122 
1123  case 'Meta-Metadata':
1124  $this->in_meta_meta_data = false;
1125  break;
1126 
1127  case 'FileItem':
1128  $this->in_file_item = false;
1129  // only update new file items
1130  if ($this->file_item->getImportId()) {
1131  $this->file_item->update();
1132  }
1133  break;
1134 
1135 
1136  case 'Table':
1137  unset($this->container[count($this->container) - 1]);
1138  break;
1139 
1140  case 'Glossary':
1141  $this->in_glossary = false;
1142  break;
1143 
1144  case 'GlossaryTerm':
1145  $term = trim($this->chr_data);
1146  $term = str_replace('&lt;', '<', $term);
1147  $term = str_replace('&gt;', '>', $term);
1148  $this->glossary_term->setTerm($term);
1149  $this->glossary_term->create();
1150  $iia = explode('_', $this->glossary_term->getImportId());
1151  $this->glossary_term_map[(int) $iia[count($iia) - 1]] = $this->glossary_term->getId();
1152  break;
1153 
1154  case 'Paragraph':
1155  $this->inside_code = false;
1156  break;
1157 
1158  case 'Definition':
1159  $this->in_glossary_definition = false;
1160  $this->page_object->updateFromXML();
1161  $this->page_object->buildDom();
1162  $this->glossary_term->setShortText($this->page_object->getFirstParagraphText());
1163  $this->glossary_term->update();
1164  if ($this->page_object->containsIntLink()) {
1165  $this->pages_to_parse['term:' . $this->page_object->getId()] = 'term:' . $this->page_object->getId();
1166  }
1167  if ($this->page_object->needsImportParsing()) {
1168  $this->pages_to_parse['term:' . $this->page_object->getId()] = 'term:' . $this->page_object->getId();
1169  }
1170  break;
1171 
1172  case 'Format':
1173  if ($this->in_media_item) {
1174  $this->media_item->setFormat(trim($this->chr_data));
1175  }
1176  break;
1177 
1178  case 'Title':
1179  if ($this->in_meta_data && !$this->in_media_object) {
1180  $this->current_object->setTitle(trim($this->chr_data));
1181  }
1182  if ($this->in_media_object) {
1183  $this->media_object->setTitle(trim($this->chr_data));
1184  }
1185  break;
1186 
1187  case 'Description':
1188  case 'Language':
1189  break;
1190 
1191  case 'Caption':
1192  if ($this->in_media_object) {
1193  $this->media_item->setCaption(trim($this->chr_data));
1194  }
1195  break;
1196 
1197  case 'TextRepresentation':
1198  if ($this->in_media_object) {
1199  $this->media_item->setTextRepresentation(trim($this->chr_data));
1200  }
1201  break;
1202 
1203  // Location
1204  case 'Location':
1205  // TODO: adapt for files in 'real' subdirectories
1206  if ($this->in_media_item) {
1207  $this->media_item->setLocationType($this->loc_type);
1208  if ($this->loc_type == 'Reference') {
1209  $this->media_item->setLocation(str_replace('&', '&amp;', trim($this->chr_data)));
1210  } else {
1211  $this->media_item->setLocation(trim($this->chr_data));
1212  }
1213  }
1214  if ($this->in_file_item) {
1215  // set file name from xml file
1216  $this->file_item->setFileName(trim($this->chr_data));
1217 
1218  // special handling for file names with special characters
1219  // (e.g. '&gt;')
1220  if ($this->file_item->getType() == 'file' &&
1221  is_int(strpos($this->chr_data, '&')) &&
1222  is_int(strpos($this->chr_data, ';'))) {
1223  $source_dir = $this->importdir . DIRECTORY_SEPARATOR . 'objects' . DIRECTORY_SEPARATOR .
1224  $this->file_item->getImportId();
1225 
1226  // read 'physical' file name from directory
1227  if ($dir = opendir($source_dir)) {
1228  while (false !== ($file = readdir($dir))) {
1229  if ($file != '.' && $file != '..') {
1230  $this->file_item->setFileName($file);
1231  }
1232  }
1233  closedir($dir);
1234  }
1235  }
1236 
1237  // set file item title
1238  $this->file_item->setTitle(trim($this->chr_data));
1239  }
1240  break;
1241  }
1242  $this->endElement($a_name);
1243  $this->chr_data = '';
1244  }
1245 
1246  public function handlerCharacterData($a_xml_parser, string $a_data): void
1247  {
1248  // call meta data handler
1249  if ($this->in_meta_data && $this->processMeta()) {
1250  // cache beginning of meta data within media object tags
1251  // (we need to know the id, after that we send the cached data
1252  // to the meta xml handler)
1253  if ($this->in_media_object && $this->media_meta_start) {
1254  $this->media_meta_cache[] =
1255  ['type' => 'handlerCharacterData', 'par1' => $a_data];
1256  } else {
1257  parent::handlerCharacterData($a_xml_parser, $a_data);
1258  }
1259  }
1260 
1261  // the parser converts '&gt;' to '>' and '&lt;' to '<'
1262  // in character data, but we don't want that, because it's the
1263  // way we mask user html in our content, so we convert back...
1264 
1265  $a_data = str_replace('<', '&lt;', $a_data);
1266  $a_data = str_replace('>', '&gt;', $a_data);
1267 
1268 
1269  // DELETE WHITESPACES AND NEWLINES OF CHARACTER DATA
1270  $a_data = preg_replace('/\n/', '', $a_data);
1271  if (!$this->inside_code) {
1272  $a_data = preg_replace('/\t+/', '', $a_data);
1273  }
1274 
1275  $this->chr_data .= $a_data;
1276 
1277  if (!empty($a_data) || $a_data === '0') {
1278  // append all data to page, if we are within PageObject,
1279  // but not within MetaData or MediaObject
1280  if (($this->in_page_object || $this->in_glossary_definition)
1281  && !$this->in_meta_data && !$this->in_media_object) {
1282  $this->page_object->appendXMLContent($a_data);
1283  }
1284 
1285  switch ($this->getCurrentElement()) {
1286  case 'IntLink':
1287  case 'ExtLink':
1288  if ($this->in_map_area) {
1289  $this->map_area->appendTitle($a_data);
1290  }
1291  break;
1292  }
1293  }
1294  }
1295 
1299  public function emptyMediaMetaCache($a_xml_parser): void
1300  {
1301  foreach ($this->media_meta_cache as $cache_entry) {
1302  switch ($cache_entry['type']) {
1303  case 'handlerBeginTag':
1304  parent::handlerBeginTag(
1305  $a_xml_parser,
1306  $cache_entry['par1'],
1307  $cache_entry['par2']
1308  );
1309  break;
1310 
1311  case 'handlerEndTag':
1312  parent::handlerEndTag(
1313  $a_xml_parser,
1314  $cache_entry['par1']
1315  );
1316  break;
1317 
1318  case 'handlerCharacterData':
1319  parent::handlerCharacterData(
1320  $a_xml_parser,
1321  $cache_entry['par1']
1322  );
1323  break;
1324  }
1325  }
1326 
1327  $this->media_meta_start = false;
1328  $this->media_meta_cache[] = [];
1329  }
1330 
1334  public function getGlossaryTermMap(): array
1335  {
1336  return $this->glossary_term_map;
1337  }
1338 
1344  private function parseLinkTarget(string $identifier): ?array
1345  {
1346  $link_arr = explode('_', $identifier);
1347 
1348  if (count($link_arr) !== 4
1349  || $link_arr[0] !== 'il'
1350  || !is_numeric($link_arr[1])
1351  || !is_numeric($link_arr[3])
1352  ) {
1353  return null;
1354  }
1355 
1356  return [
1357  'target_inst' => (int) $link_arr[1],
1358  'target_type' => (string) $link_arr[2],
1359  'target_id' => (int) $link_arr[3]
1360  ];
1361  }
1362 }
processPagesToParse()
parse pages that contain files, mobs and/or internal links
enableMDParsing(bool $a_status)
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
buildDom(bool $a_force=false)
static getLogger(string $a_component_id)
Get component logger.
buildTag(string $type, string $name, array $attr=[])
handlerEndTag($a_xml_parser, string $a_name)
write(string $message, $level=ilLogLevel::INFO, array $context=[])
write log message
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
handlerBeginTag($a_xml_parser, string $a_name, array $a_attribs)
update(bool $a_validate=true, bool $a_no_history=false)
update complete page content in db (dom xml content is used)
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
ilLanguage $lng
parseLinkTarget(string $identifier)
Parse a string the get the elements of a link target Return null if the string is not a link target...
while($session_entry=$r->fetchRow(ilDBConstants::FETCHMODE_ASSOC)) return null
static instantiateQuestion(int $question_id)
setQuestionMapping(array $a_map)
set question import ident to pool/test question id mapping
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
const IL_INT_LINK
handlerCharacterData($a_xml_parser, string $a_data)
static _exists(string $a_parent_type, int $a_id, string $a_lang="", bool $a_no_cache=false)
Checks whether page exists.
Class ilPageObject Handles PageObjects of ILIAS Learning Modules (see ILIAS DTD)
Class ilObjFile.
global $DIC
Definition: shib_login.php:22
Class ilMediaItem Media Item, component of a media object (file or reference)
Legacy Content Object Parser.
resolveIntLinks(?array $a_link_map=null)
Resolves all internal link targets of the page, if targets are available (after import) ...
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
static parseImportId(string $a_import_id)
Parse an ilias import id Typically of type il_[IL_INST_ID]_[OBJ_TYPE]_[OBJ_ID] returns array( &#39;orig&#39; ...
const IL_MODE_ALIAS
static _writeImportId(int $a_id, string $a_import_id)
static _resolveMapAreaLinks(int $a_mob_id)
resolve internal links of all media items of a media object
ilObjLearningModule $learning_module
copyFileItems()
copy files of file items
const IL_EXT_LINK
Class ilMapArea.
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
__construct(Container $dic, ilPlugin $plugin)
static yn2tf(string $a_yn)
setImportMapping(?ilImportMapping $mapping=null)
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
__construct(private readonly ilObject $content_object, string $xml_file, private readonly string $importdir)
static getInstance(string $a_parent_type, int $a_id=0, int $a_old_nr=0, string $a_lang="-")
Get page object instance.
storeTree()
insert StructureObjects and PageObjects into tree