xref: /webtrees/app/Http/RequestHandlers/ReportSetupPage.php (revision 9abbcca9747738c136d30d39ec7be82574ab4b52)
1<?php
2
3/**
4 * webtrees: online genealogy
5 * Copyright (C) 2022 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 Fisharebest\Webtrees\Auth;
23use Fisharebest\Webtrees\Html;
24use Fisharebest\Webtrees\Http\ViewResponseTrait;
25use Fisharebest\Webtrees\I18N;
26use Fisharebest\Webtrees\Module\ModuleReportInterface;
27use Fisharebest\Webtrees\Registry;
28use Fisharebest\Webtrees\Report\ReportParserSetup;
29use Fisharebest\Webtrees\Services\LocalizationService;
30use Fisharebest\Webtrees\Services\ModuleService;
31use Fisharebest\Webtrees\Validator;
32use Psr\Http\Message\ResponseInterface;
33use Psr\Http\Message\ServerRequestInterface;
34use Psr\Http\Server\RequestHandlerInterface;
35
36use function redirect;
37use function route;
38
39/**
40 * Get parameters for a report.
41 */
42class ReportSetupPage implements RequestHandlerInterface
43{
44    use ViewResponseTrait;
45
46    private LocalizationService $localization_service;
47
48    private ModuleService $module_service;
49
50    /**
51     * ReportEngineController constructor.
52     *
53     * @param LocalizationService $localization_service
54     * @param ModuleService       $module_service
55     */
56    public function __construct(LocalizationService $localization_service, ModuleService $module_service)
57    {
58        $this->localization_service = $localization_service;
59        $this->module_service = $module_service;
60    }
61
62    /**
63     * @param ServerRequestInterface $request
64     *
65     * @return ResponseInterface
66     */
67    public function handle(ServerRequestInterface $request): ResponseInterface
68    {
69        $tree = Validator::attributes($request)->tree();
70        $user = Validator::attributes($request)->user();
71
72        $report = Validator::attributes($request)->string('report');
73        $module = $this->module_service->findByName($report);
74
75        if (!$module instanceof ModuleReportInterface) {
76            return redirect(route(ReportListPage::class, ['tree' => $tree->name()]));
77        }
78
79        Auth::checkComponentAccess($module, ModuleReportInterface::class, $tree, $user);
80
81        $xref = $request->getQueryParams()['xref'] ?? '';
82
83        $xml_filename = $module->resourcesFolder() . $module->xmlFilename();
84
85        $report_array = (new ReportParserSetup($xml_filename))->reportProperties();
86        $description  = $report_array['description'];
87        $title        = $report_array['title'];
88
89        $inputs = [];
90
91        foreach ($report_array['inputs'] ?? [] as $n => $input) {
92            $input += [
93                'type'    => 'text',
94                'default' => '',
95                'lookup'  => '',
96                'extra'   => '',
97            ];
98
99            $attributes = [
100                'id'    => 'input-' . $n,
101                'name'  => 'vars[' . $input['name'] . ']',
102                'class' => $input['type'] === 'checkbox' ? 'form-control-check' : 'form-control',
103            ];
104
105            switch ($input['lookup']) {
106                case 'INDI':
107                    $input['control'] = view('components/select-individual', [
108                        'id'         => 'input-' . $n,
109                        'name'       => 'vars[' . $input['name'] . ']',
110                        'individual' => Registry::individualFactory()->make($xref, $tree),
111                        'tree'       => $tree,
112                        'required'   => true,
113                    ]);
114                    break;
115
116                case 'FAM':
117                    $input['control'] = view('components/select-family', [
118                        'id'       => 'input-' . $n,
119                        'name'     => 'vars[' . $input['name'] . ']',
120                        'family'   => Registry::familyFactory()->make($xref, $tree),
121                        'tree'     => $tree,
122                        'required' => true,
123                    ]);
124                    break;
125
126                case 'SOUR':
127                    $input['control'] = view('components/select-source', [
128                        'id'       => 'input-' . $n,
129                        'name'     => 'vars[' . $input['name'] . ']',
130                        'family'   => Registry::sourceFactory()->make($xref, $tree),
131                        'tree'     => $tree,
132                        'required' => true,
133                    ]);
134                    break;
135
136                case 'DATE':
137                    // Need to know if the user prefers DMY/MDY/YMD so we can validate dates properly.
138                    $dmy = $this->localization_service->dateFormatToOrder(I18N::dateFormat());
139
140                    $attributes += [
141                        'type'     => 'text',
142                        'value'    => $input['default'],
143                        'dir'      => 'ltr',
144                        'onchange' => 'webtrees.reformatDate(this, "' . $dmy . '")'
145                    ];
146                    $input['control'] = '<input ' . Html::attributes($attributes) . '>';
147                    $input['extra'] = view('edit/input-addon-calendar', ['id' => 'input-' . $n]);
148                    break;
149
150                default:
151                    switch ($input['type']) {
152                        case 'text':
153                            $attributes += [
154                                'type'  => 'text',
155                                'value' => $input['default'],
156                            ];
157                            $input['control'] = '<input ' . Html::attributes($attributes) . '>';
158                            break;
159
160                        case 'checkbox':
161                            $attributes += [
162                                'type'    => 'checkbox',
163                                'checked' => (bool) $input['default'],
164                            ];
165                            $input['control'] = '<input ' . Html::attributes($attributes) . '>';
166                            break;
167
168                        case 'select':
169                            $options = [];
170                            foreach (explode('|', $input['options']) as $option) {
171                                [$key, $value] = explode('=>', $option);
172                                if (preg_match('/^I18N::number\((.+?)(,([\d+]))?\)$/', $value, $match)) {
173                                    $number        = (float) $match[1];
174                                    $precision     = (int) ($match[3] ?? 0);
175                                    $options[$key] = I18N::number($number, $precision);
176                                } elseif (preg_match('/^I18N::translate\(\'(.+)\'\)$/', $value, $match)) {
177                                    $options[$key] = I18N::translate($match[1]);
178                                } elseif (preg_match('/^I18N::translateContext\(\'(.+)\', *\'(.+)\'\)$/', $value, $match)) {
179                                    $options[$key] = I18N::translateContext($match[1], $match[2]);
180                                }
181                            }
182                            $input['control'] = view('components/select', ['name' => 'vars[' . $input['name'] . ']', 'id' => 'input-' . $n, 'selected' => $input['default'], 'options' => $options]);
183                            break;
184                    }
185            }
186
187            $inputs[] = $input;
188        }
189
190        $destination = $user->getPreference('default-report-destination', 'view');
191        $format      = $user->getPreference('default-report-format', 'PDF');
192
193        return $this->viewResponse('report-setup-page', [
194            'description' => $description,
195            'destination' => $destination,
196            'format'      => $format,
197            'inputs'      => $inputs,
198            'report'      => $report,
199            'title'       => $title,
200            'tree'        => $tree,
201        ]);
202    }
203}
204