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