ILIAS  trunk Revision v11.0_alpha-1689-g66c127b4ae8
All Data Structures Namespaces Files Functions Variables Enumerations Enumerator Modules Pages
class.ilMailAttachmentGUI.php
Go to the documentation of this file.
1 <?php
2 
19 declare(strict_types=1);
20 
29 
31 {
32  private readonly ilGlobalTemplateInterface $tpl;
33  private readonly ilLanguage $lng;
34  private readonly ilObjUser $user;
35  private readonly ilFormatMail $umail;
36  private readonly ilFileDataMail $mfile;
37  private readonly Refinery $refinery;
38  private readonly \ILIAS\UI\Factory $ui_factory;
39  private readonly \ILIAS\UI\Renderer $ui_renderer;
40  private readonly ilTabsGUI $tabs;
41  private AttachmentManagement $mode = AttachmentManagement::MANAGE;
42 
43  public function __construct()
44  {
45  global $DIC;
46 
48  $this->tpl = $DIC->ui()->mainTemplate();
49  $this->lng = $DIC->language();
50  $this->user = $DIC->user();
51  $this->tabs = $DIC->tabs();
52  $this->refinery = $DIC->refinery();
53  $this->ui_factory = $DIC->ui()->factory();
54  $this->ui_renderer = $DIC->ui()->renderer();
55 
56  $this->ctrl->saveParameter($this, 'mobj_id');
57 
58  $this->umail = new ilFormatMail($DIC->user()->getId());
59  $this->mfile = new ilFileDataMail($DIC->user()->getId());
60  }
61 
62  public function manage(): self
63  {
64  $this->mode = AttachmentManagement::MANAGE;
65  return $this;
66  }
67 
68  public function consume(): self
69  {
70  $this->mode = AttachmentManagement::CONSUME;
71  return $this;
72  }
73 
74  public function executeCommand(): void
75  {
76  if (!($cmd = $this->ctrl->getCmd())) {
77  $cmd = 'showAttachments';
78  }
79 
80  match ($cmd) {
81  AbstractCtrlAwareUploadHandler::CMD_UPLOAD,
82  AbstractCtrlAwareUploadHandler::CMD_INFO,
83  AbstractCtrlAwareUploadHandler::CMD_REMOVE => parent::executeCommand(),
84  default => $this->$cmd()
85  };
86  }
87 
88  public function saveAttachments(): void
89  {
90  $files = [];
91 
92  // Important: Do not check for uploaded files here,
93  // otherwise it is no more possible to remove files (please ignore bug reports like 10137)
94 
95  $sizeOfSelectedFiles = 0;
96  $filesOfRequest = $this->http->wrapper()->query()->retrieve(
97  'mail_attachments_filename',
98  $this->refinery->byTrying([
99  $this->refinery->kindlyTo()->listOf($this->refinery->kindlyTo()->string()),
100  $this->refinery->always([])
101  ])
102  );
103 
104  if ($filesOfRequest !== [] && $filesOfRequest[0] === 'ALL_OBJECTS') {
105  $filesOfRequest = array_map(static function (array $file): string {
106  return $file['name'];
107  }, $this->mfile->getUserFilesData());
108  }
109 
110  foreach ($filesOfRequest as $file) {
111  if (is_file($this->mfile->getMailPath() . '/' . basename($this->user->getId() . '_' . urldecode($file)))) {
112  $files[] = urldecode($file);
113  $sizeOfSelectedFiles += filesize(
114  $this->mfile->getMailPath() . '/' .
115  basename($this->user->getId() . '_' . urldecode($file))
116  );
117  }
118  }
119 
120  if ($files !== [] &&
121  null !== $this->mfile->getAttachmentsTotalSizeLimit() &&
122  $sizeOfSelectedFiles > $this->mfile->getAttachmentsTotalSizeLimit()) {
123  $this->tpl->setOnScreenMessage(
124  'failure',
125  $this->lng->txt('mail_max_size_attachments_total_error') . ' ' .
126  ilUtil::formatSize((int) $this->mfile->getAttachmentsTotalSizeLimit())
127  );
128  $this->showAttachments();
129  return;
130  }
131 
132  $this->umail->saveAttachments($files);
133 
134  $this->ctrl->returnToParent($this);
135  }
136 
137  public function cancelSaveAttachments(): void
138  {
139  $this->ctrl->setParameter($this, 'type', ilMailFormGUI::MAIL_FORM_TYPE_ATTACH);
140  $this->ctrl->returnToParent($this);
141  }
142 
143  public function deleteAttachments(): void
144  {
145  $files = $this->http->wrapper()->query()->retrieve(
146  'mail_attachments_filename',
147  $this->refinery->byTrying([
148  $this->refinery->kindlyTo()->listOf($this->refinery->kindlyTo()->string()),
149  $this->refinery->always([])
150  ])
151  );
152 
153  if ($files !== [] && $files[0] === 'ALL_OBJECTS') {
154  $files = array_map(static function (array $file): string {
155  return $file['name'];
156  }, $this->mfile->getUserFilesData());
157  }
158 
159  if ($files === []) {
160  $this->tpl->setOnScreenMessage('info', $this->lng->txt('select_one'), true);
161  $this->ctrl->redirect($this);
162  }
163 
164  $this->tpl->setTitle($this->lng->txt('mail'));
165 
166  $confirmation = new ilConfirmationGUI();
167  $confirmation->setFormAction($this->ctrl->getFormAction($this, 'confirmDeleteAttachments'));
168  $confirmation->setConfirm($this->lng->txt('confirm'), 'confirmDeleteAttachments');
169  $confirmation->setCancel($this->lng->txt('cancel'), 'showAttachments');
170  $confirmation->setHeaderText($this->lng->txt('mail_sure_delete_file'));
171 
172  foreach ($files as $filename) {
173  $confirmation->addItem(
174  'filename[]',
175  ilUtil::stripSlashes($filename),
176  ilUtil::stripSlashes(urldecode($filename))
177  );
178  }
179 
180  $this->tpl->setContent($confirmation->getHTML());
181  $this->tpl->printToStdout();
182  }
183 
184  public function confirmDeleteAttachments(): void
185  {
186  $files = $this->http->wrapper()->post()->retrieve(
187  'filename',
188  $this->refinery->byTrying([
189  $this->refinery->kindlyTo()->listOf($this->refinery->kindlyTo()->string()),
190  $this->refinery->always([])
191  ])
192  );
193 
194  if ($files === []) {
195  $this->tpl->setOnScreenMessage('info', $this->lng->txt('mail_select_one_mail'));
196  $this->showAttachments();
197  return;
198  }
199 
200  $decodedFiles = [];
201  foreach ($files as $value) {
202  $decodedFiles[] = urldecode($value);
203  }
204 
205  $error = $this->mfile->unlinkFiles($decodedFiles);
206  if ($error !== '') {
207  $this->tpl->setOnScreenMessage('failure', $this->lng->txt('mail_error_delete_file') . ' ' . $error, true);
208  } else {
209  $mail_data = $this->umail->retrieveFromStage();
210  if (is_array($mail_data['attachments'])) {
211  $tmp = [];
212  foreach ($mail_data['attachments'] as $attachment) {
213  if (!in_array($attachment, $decodedFiles, true)) {
214  $tmp[] = $attachment;
215  }
216  }
217  $this->umail->saveAttachments($tmp);
218  }
219 
220  $this->tpl->setOnScreenMessage('success', $this->lng->txt('mail_files_deleted'), true);
221  }
222 
223  $this->ctrl->redirect($this);
224  }
225 
226  public function showAttachments(): void
227  {
228  $this->tpl->setTitle($this->lng->txt('mail'));
229 
230  if ($this->mode === AttachmentManagement::CONSUME) {
231  $this->tabs->clearTargets();
232  $this->tabs->setBackTarget(
233  $this->lng->txt('mail_manage_attachments_back_to_compose'),
234  $this->ctrl->getLinkTarget($this, 'cancelSaveAttachments')
235  );
236  }
237 
238  $components = [];
239  if ($this->mode === AttachmentManagement::MANAGE) {
240  $dropzone = $this->ui_factory
241  ->dropzone()
242  ->file()
243  ->standard(
244  $this->lng->txt('mail_manage_attachments'),
245  $this->lng->txt('mail_manage_attachments_drop_files_msg'),
246  '#',
247  $this->ui_factory->input()->field()->file(
248  $this,
249  $this->lng->txt('file')
250  )->withMaxFiles(42) // The answer to life, universe and the rest
251  )
252  ->withBulky(true)
253  ->withUploadButton(
254  $this->ui_factory->button()->shy(
255  $this->lng->txt('upload'),
256  '#'
257  )
258  );
259  $components[] = $dropzone;
260  }
261 
262  $mail_data = $this->umail->retrieveFromStage();
263  $files = $this->mfile->getUserFilesData();
264  $records = [];
265  $checked_items = [];
266  foreach ($files as $file) {
267  if (is_array($mail_data['attachments']) && in_array($file['name'], $mail_data['attachments'], true)) {
268  $checked_items[] = urlencode($file['name']);
269  }
270 
271  $records[] = [
272  'filename' => $file['name'],
273  'filesize' => (int) $file['size'],
274  'filecreatedate' => (int) $file['ctime'],
275  ];
276  }
277 
278  $table = new MailAttachmentTableGUI(
279  $this,
280  $this->user,
281  $records,
282  $this->ui_factory,
283  $this->ui_renderer,
284  $this->lng,
285  $this->ctrl,
286  $this->http->request(),
287  new ILIAS\Data\Factory(),
288  'handleTableActions',
290  );
291  $components[] = $table->get();
292 
293  $this->tpl->setContent($this->ui_renderer->render($components));
294 
295  if ($this->mode === AttachmentManagement::CONSUME) {
296  // The table above has to be rendered first, because it deselects all checkboxes
297  $this->tpl->addOnLoadCode('
298  const checked_items = ' . json_encode($checked_items, JSON_THROW_ON_ERROR) . ';
299  for (const item of checked_items) {
300  const checkbox = document.querySelector("input[type=\'checkbox\'][value=\'" + item + "\']");
301  if (checkbox) {
302  checkbox.checked = true;
303  }
304  }
305  ');
306  }
307 
308  $this->tpl->printToStdout();
309  }
310 
311  private function handleTableActions(): void
312  {
313  $query = $this->http->wrapper()->query();
314  if (!$query->has('mail_attachments_table_action')) {
315  return;
316  }
317 
318  $action = $query->retrieve('mail_attachments_table_action', $this->refinery->to()->string());
319  match ($action) {
320  'saveAttachments' => $this->saveAttachments(),
321  'deleteAttachments' => $this->deleteAttachments(),
322  default => $this->ctrl->redirect($this),
323  };
324  }
325 
326  protected function getUploadResult(): HandlerResult
327  {
328  $this->upload->process();
329  $array = $this->upload->getResults();
330  $result = end($array);
331 
332  if ($result instanceof UploadResult && $result->isOK()) {
333  $identifier = $this->mfile->storeUploadedFile($result);
334  $status = HandlerResult::STATUS_OK;
335  $message = $this->lng->txt('saved_successfully');
336  $this->tpl->setOnScreenMessage('success', $this->lng->txt('saved_successfully'), true);
337  } else {
338  $status = HandlerResult::STATUS_FAILED;
339  $identifier = '';
340  $message = $result->getStatus()->getMessage();
341  }
342 
343  return new BasicHandlerResult($this->getFileIdentifierParameterName(), $status, $identifier, $message);
344  }
345 
346  protected function getRemoveResult(string $identifier): HandlerResult
347  {
348  throw new DomainException('Not necessary for this handler');
349  }
350 
351  public function getInfoResult(string $identifier): ?FileInfoResult
352  {
353  throw new DomainException('Not necessary for this handler');
354  }
355 
356  public function getInfoForExistingFiles(array $file_ids): array
357  {
358  throw new DomainException('Not necessary for this handler');
359  }
360 
361  public function getFileIdentifierParameterName(): string
362  {
363  return 'userfile';
364  }
365 }
AttachmentManagement $mode
readonly ilFileDataMail $mfile
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
This class handles all operations on files (attachments) in directory ilias_data/mail.
static stripSlashes(string $a_str, bool $a_strip_html=true, string $a_allow="")
readonly ILIAS UI Renderer $ui_renderer
$components
while($session_entry=$r->fetchRow(ilDBConstants::FETCHMODE_ASSOC)) return null
readonly ilFormatMail $umail
executeCommand()
Since this is a ilCtrl aware UploadHandler executeCommand MUST be implemented.
getInfoResult(string $identifier)
static http()
Fetches the global http state from ILIAS.
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
static formatSize(int $size, string $a_mode='short', ?ilLanguage $a_lng=null)
Returns the specified file size value in a human friendly form.
global $DIC
Definition: shib_login.php:22
$filename
Definition: buildRTE.php:78
final const MAIL_FORM_TYPE_ATTACH
getInfoForExistingFiles(array $file_ids)
__construct(Container $dic, ilPlugin $plugin)
readonly ILIAS UI Factory $ui_factory
$message
Definition: xapiexit.php:31
getRemoveResult(string $identifier)
readonly ilGlobalTemplateInterface $tpl