xref: /webtrees/app/Http/RequestHandlers/CreateMediaObjectAction.php (revision ef3fb807042ec0187072c1155b05525722f7fa7c)
1<?php
2
3/**
4 * webtrees: online genealogy
5 * Copyright (C) 2023 webtrees development team
6 * This program is free software: you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation, either version 3 of the License, or
9 * (at your option) any later version.
10 * This program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 * GNU General Public License for more details.
14 * You should have received a copy of the GNU General Public License
15 * along with this program. If not, see <https://www.gnu.org/licenses/>.
16 */
17
18declare(strict_types=1);
19
20namespace Fisharebest\Webtrees\Http\RequestHandlers;
21
22use Fig\Http\Message\StatusCodeInterface;
23use Fisharebest\Webtrees\Exceptions\FileUploadException;
24use Fisharebest\Webtrees\I18N;
25use Fisharebest\Webtrees\Registry;
26use Fisharebest\Webtrees\Services\MediaFileService;
27use Fisharebest\Webtrees\Services\PendingChangesService;
28use Fisharebest\Webtrees\Validator;
29use Psr\Http\Message\ResponseInterface;
30use Psr\Http\Message\ServerRequestInterface;
31use Psr\Http\Server\RequestHandlerInterface;
32
33use function response;
34use function view;
35
36/**
37 * Process a form to create a new media object.
38 */
39class CreateMediaObjectAction implements RequestHandlerInterface
40{
41    private MediaFileService $media_file_service;
42
43    private PendingChangesService $pending_changes_service;
44
45    /**
46     * @param MediaFileService      $media_file_service
47     * @param PendingChangesService $pending_changes_service
48     */
49    public function __construct(MediaFileService $media_file_service, PendingChangesService $pending_changes_service)
50    {
51        $this->media_file_service      = $media_file_service;
52        $this->pending_changes_service = $pending_changes_service;
53    }
54
55    /**
56     * Process a form to create a new media object.
57     *
58     * @param ServerRequestInterface $request
59     *
60     * @return ResponseInterface
61     */
62    public function handle(ServerRequestInterface $request): ResponseInterface
63    {
64        $tree        = Validator::attributes($request)->tree();
65        $note        = Validator::parsedBody($request)->string('media-note');
66        $title       = Validator::parsedBody($request)->string('title');
67        $type        = Validator::parsedBody($request)->string('type');
68        $restriction = Validator::parsedBody($request)->string('restriction');
69
70        $note        = Registry::elementFactory()->make('OBJE:NOTE')->canonical($note);
71        $type        = Registry::elementFactory()->make('OBJE:FILE:FORM:TYPE')->canonical($type);
72        $title       = Registry::elementFactory()->make('OBJE:FILE:TITL')->canonical($title);
73        $restriction = Registry::elementFactory()->make('OBJE:RESN')->canonical($restriction);
74
75        try {
76            $file = $this->media_file_service->uploadFile($request);
77        } catch (FileUploadException $exception) {
78            return response([
79                'value' => '',
80                'text'  => '',
81                'html'  => view('components/alert-danger', [
82                    'alert' => $exception->getMessage(),
83                ]),
84            ]);
85        }
86
87        if ($file === '') {
88            return response(['error_message' => I18N::translate('There was an error uploading your file.')], StatusCodeInterface::STATUS_NOT_ACCEPTABLE);
89        }
90
91        $gedcom = "0 @@ OBJE\n" . $this->media_file_service->createMediaFileGedcom($file, $type, $title, $note);
92
93        if ($restriction !== '') {
94            $gedcom .= "\n1 RESN " . strtr($restriction, ["\n" => "\n2 CONT "]);
95        }
96
97        $record = $tree->createMediaObject($gedcom);
98
99        // Accept the new record to keep the filesystem synchronized with the genealogy.
100        $this->pending_changes_service->acceptRecord($record);
101
102        // value and text are for autocomplete
103        // html is for interactive modals
104        return response([
105            'value' => '@' . $record->xref() . '@',
106            'text'  => view('selects/media', ['media' => $record]),
107            'html'  => view('modals/record-created', [
108                'title' => I18N::translate('The media object has been created'),
109                'name'  => $record->fullName(),
110                'url'   => $record->url(),
111            ]),
112        ]);
113    }
114}
115