ILIAS  trunk Revision v11.0_alpha-1689-g66c127b4ae8
All Data Structures Namespaces Files Functions Variables Enumerations Enumerator Modules Pages
ExtractPages.php
Go to the documentation of this file.
1 <?php
18 declare(strict_types=1);
19 
21 
34 
40 {
42 
43  public const ID = 'extract_pages';
44  private bool $high_quality = false;
45 
46  public function getId(): string
47  {
48  return self::ID;
49  }
50 
51  public function dependsOnEngine(): ?string
52  {
53  return ImagickEngineWithOptionalFFMpeg::class;
54  }
55 
56  public function canHandleDefinition(FlavourDefinition $definition): bool
57  {
58  return $definition instanceof PagesToExtract;
59  }
60 
61  public function processStream(
62  FileInformation $information,
63  FileStream $stream,
64  FlavourDefinition $for_definition
65  ): \Generator {
66  if (!$for_definition instanceof PagesToExtract) {
67  throw new \InvalidArgumentException('Invalid definition');
68  }
69 
70  if (!class_exists(\Imagick::class)) {
71  return;
72  }
73 
74  // Create target image
75  $img = new \Imagick();
76 
77  // Quality Settings
78  $this->high_quality = $for_definition->useMaxQuality();
79  if ($this->high_quality) {
80  $quality = 100;
81  $img->setResolution(300, 300); // way better previews for PDFs. Must be set before reading the file
82  } else {
83  $quality = $for_definition->getQuality();
84  }
85 
86  $extractor = new General();
87 
88  $mime_type = $information->getMimeType();
89  switch (true) {
90  case ($mime_type === 'image/svg+xml' || $mime_type === 'image/svg'):
91  $extractor = new SVG();
92  break;
93  case (str_contains($mime_type, 'video')):
94  $extractor = new Video();
95  break;
96  case ($mime_type === 'application/pdf'):
97  $extractor = new PDF();
98  break;
99  }
100 
101  $target_format = $extractor->getTargetFormat();
102  $target_background = $extractor->getBackground();
103  $alpha_channel = $extractor->getAlphaChannel();
104  $remove_color = $extractor->getRemoveColor();
105  $resolution = $extractor->getResolution();
106 
107  // General Image Settings
108  $img->setResolution($resolution, $resolution);
109  $img->setBackgroundColor($target_background);
110 
111  // Read source image
112  try {
113  $img = $extractor->readImage($img, $stream, $for_definition);
114  } catch (\ImagickException) {
115  // due to possible security risks, gs disabled access to files, see e.g. https://en.linuxportal.info/tutorials/troubleshooting/how-to-fix-errors-from-imagemagick-imagick-conversion-system-security-policy
116  return;
117  }
118 
119  // Size Settings
120  $max_size = $for_definition->getMaxSize();
121  $img->resetIterator();
122 
123  // create gif if needed
124  $gif = $target_format === 'gif' ? new \Imagick() : null;
125  if ($gif !== null) {
126  $gif->setFormat('GIF');
127  }
128 
129  for ($x = 0; ($x < $for_definition->getMaxPages() && $x < $img->getNumberImages()); $x++) {
130  $img->setIteratorIndex($x);
131  $img->setImageAlphaChannel($alpha_channel);
132  $img->setImageBackgroundColor($target_background);
133  $img->setImageFormat($target_format);
134 
135  if ($remove_color !== null) {
136  $img->transparentPaintImage($remove_color, 0, 0, false);
137  }
138 
139  [$columns, $rows] = $this->calculateWidthHeightFromImage($img, $max_size);
140 
141  if (!$this->high_quality) {
142  $yield = $img->thumbnailImage(
143  (int) $columns,
144  (int) $rows,
145  true,
146  $for_definition->isFill()
147  );
148  } else {
149  $img->setImageResolution($resolution, $resolution);
150  $img->setImageCompression(\Imagick::COMPRESSION_JPEG);
151  $img->setImageCompressionQuality($quality);
152  $img->stripImage();
153 
154  $yield = $img->resizeImage(
155  (int) $columns,
156  (int) $rows,
157  \Imagick::FILTER_MITCHELL,
158  1
159  );
160  }
161 
162  if ($yield && $gif === null) {
163  yield new Result(
164  $for_definition,
165  Streams::ofString($img->getImageBlob()),
166  $x,
167  $for_definition->persist()
168  );
169  } elseif ($yield && $gif !== null) {
170  $gif->addImage($img->getImage());
171  $gif->setImageDelay(50);
172  }
173  }
174 
175  if ($gif !== null) {
176  $gif->setImageFormat('gif');
177  $gif->setIteratorIndex(0);
178  $gif->setImageIterations(0); // 0 means infinite loop
179  [$columns, $rows] = $this->calculateWidthHeightFromImage($gif, $max_size);
180  $gif->thumbnailImage(
181  (int) $columns,
182  (int) $rows,
183  true,
184  $for_definition->isFill()
185  );
186 
187  yield new Result(
188  $for_definition,
189  Streams::ofString($gif->getImagesBlob()),
190  1,
191  $for_definition->persist()
192  );
193  $gif->destroy();
194  }
195  $img->destroy();
196  }
197 }
persist()
Define whether the generated flavor and the respective streams should be persisted, or whether they should only be generated and used in-memory.
while($session_entry=$r->fetchRow(ilDBConstants::FETCHMODE_ASSOC)) return null
dependsOnEngine()
Return the class name of the Engine that is required for this Machine to work.
static ofString(string $string)
Creates a new stream with an initial value.
Definition: Streams.php:41
processStream(FileInformation $information, FileStream $stream, FlavourDefinition $for_definition)
The base interface for all filesystem streams.
Definition: FileStream.php:31
canHandleDefinition(FlavourDefinition $definition)
Check if a corresponding configuration can be processed by this Machine.