xref: /webtrees/app/Http/RequestHandlers/ReportSetupPage.php (revision 4db266ad654d7eeb6d0e1278b63275d1cccf1b05)
1<?php
2
3/**
4 * webtrees: online genealogy
5 * Copyright (C) 2019 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 <http://www.gnu.org/licenses/>.
16 */
17
18declare(strict_types=1);
19
20namespace Fisharebest\Webtrees\Http\RequestHandlers;
21
22use Fisharebest\Webtrees\Auth;
23use Fisharebest\Webtrees\Contracts\UserInterface;
24use Fisharebest\Webtrees\Family;
25use Fisharebest\Webtrees\Html;
26use Fisharebest\Webtrees\Http\ViewResponseTrait;
27use Fisharebest\Webtrees\I18N;
28use Fisharebest\Webtrees\Individual;
29use Fisharebest\Webtrees\Module\ModuleReportInterface;
30use Fisharebest\Webtrees\Report\ReportParserSetup;
31use Fisharebest\Webtrees\Services\ModuleService;
32use Fisharebest\Webtrees\Source;
33use Fisharebest\Webtrees\Tree;
34use InvalidArgumentException;
35use Psr\Http\Message\ResponseInterface;
36use Psr\Http\Message\ServerRequestInterface;
37use Psr\Http\Server\RequestHandlerInterface;
38
39use function assert;
40use function redirect;
41use function route;
42
43/**
44 * Get parameters for a report.
45 */
46class ReportSetupPage implements RequestHandlerInterface
47{
48    use ViewResponseTrait;
49
50    /**
51     * @var ModuleService
52     */
53    private $module_service;
54
55    /**
56     * ReportEngineController constructor.
57     *
58     * @param ModuleService $module_service
59     */
60    public function __construct(ModuleService $module_service)
61    {
62        $this->module_service = $module_service;
63    }
64
65    /**
66     * @param ServerRequestInterface $request
67     *
68     * @return ResponseInterface
69     */
70    public function handle(ServerRequestInterface $request): ResponseInterface
71    {
72        $tree = $request->getAttribute('tree');
73        assert($tree instanceof Tree, new InvalidArgumentException());
74
75        $user = $request->getAttribute('user');
76        assert($user instanceof UserInterface, new InvalidArgumentException());
77
78        $report = $request->getAttribute('report');
79        $module = $this->module_service->findByName($report);
80
81        if (!$module instanceof ModuleReportInterface) {
82            return redirect(route(ReportListPage::class, ['tree' => $tree->name()]));
83        }
84
85        Auth::checkComponentAccess($module, 'report', $tree, $user);
86
87        $xref = $request->getQueryParams()['xref'] ?? '';
88
89        $xml_filename = $module->resourcesFolder() . $module->xmlFilename();
90
91        $report_array = (new ReportParserSetup($xml_filename))->reportProperties();
92        $description  = $report_array['description'];
93        $title        = $report_array['title'];
94
95        $inputs = [];
96
97        foreach ($report_array['inputs'] ?? [] as $n => $input) {
98            $input += [
99                'type'    => 'text',
100                'default' => '',
101                'lookup'  => '',
102                'extra'   => '',
103            ];
104
105            $attributes = [
106                'id'   => 'input-' . $n,
107                'name' => 'vars[' . $input['name'] . ']',
108            ];
109
110            switch ($input['lookup']) {
111                case 'INDI':
112                    $input['control'] = view('components/select-individual', [
113                        'id'         => 'input-' . $n,
114                        'name'       => 'vars[' . $input['name'] . ']',
115                        'individual' => Individual::getInstance($xref, $tree),
116                        'tree'       => $tree,
117                        'required'   => true,
118                    ]);
119                    break;
120
121                case 'FAM':
122                    $input['control'] = view('components/select-family', [
123                        'id'       => 'input-' . $n,
124                        'name'     => 'vars[' . $input['name'] . ']',
125                        'family'   => Family::getInstance($xref, $tree),
126                        'tree'     => $tree,
127                        'required' => true,
128                    ]);
129                    break;
130
131                case 'SOUR':
132                    $input['control'] = view('components/select-source', [
133                        'id'       => 'input-' . $n,
134                        'name'     => 'vars[' . $input['name'] . ']',
135                        'family'   => Source::getInstance($xref, $tree),
136                        'tree'     => $tree,
137                        'required' => true,
138                    ]);
139                    break;
140
141                case 'DATE':
142                    $attributes += [
143                        'type'  => 'text',
144                        'value' => $input['default'],
145                        'dir'   => 'ltr',
146                    ];
147                    $input['control'] = '<input ' . Html::attributes($attributes) . '>';
148                    $input['extra']   = '<a href="#" title="' . I18N::translate('Select a date') . '" class ="btn btn-link" onclick="' . e('return calendarWidget("calendar-widget-' . $n . '", "input-' . $n . '");') . '">' . view('icons/calendar') . '</a>' .
149                        '<div id="calendar-widget-' . $n . '" style="position:absolute;visibility:hidden;background-color:white;z-index:1000;"></div>';
150                    break;
151
152                default:
153                    switch ($input['type']) {
154                        case 'text':
155                            $attributes += [
156                                'type'  => 'text',
157                                'value' => $input['default'],
158                            ];
159                            $input['control'] = '<input ' . Html::attributes($attributes) . '>';
160                            break;
161
162                        case 'checkbox':
163                            $attributes += [
164                                'type'    => 'checkbox',
165                                'checked' => (bool) $input['default'],
166                            ];
167                            $input['control'] = '<input ' . Html::attributes($attributes) . '>';
168                            break;
169
170                        case 'select':
171                            $options = [];
172                            foreach (explode('|', $input['options']) as $option) {
173                                [$key, $value] = explode('=>', $option);
174                                if (preg_match('/^I18N::number\((.+?)(,([\d+]))?\)$/', $value, $match)) {
175                                    $number        = (float) $match[1];
176                                    $precision     = (int) ($match[3] ?? 0);
177                                    $options[$key] = I18N::number($number, $precision);
178                                } elseif (preg_match('/^I18N::translate\(\'(.+)\'\)$/', $value, $match)) {
179                                    $options[$key] = I18N::translate($match[1]);
180                                } elseif (preg_match('/^I18N::translateContext\(\'(.+)\', *\'(.+)\'\)$/', $value, $match)) {
181                                    $options[$key] = I18N::translateContext($match[1], $match[2]);
182                                }
183                            }
184                            $input['control'] = view('components/select', ['name' => 'vars[' . $input['name'] . ']', 'id' => 'input-' . $n, 'selected' => $input['default'], 'options' => $options]);
185                            break;
186                    }
187            }
188
189            $inputs[] = $input;
190        }
191
192        return $this->viewResponse('report-setup-page', [
193            'description' => $description,
194            'inputs'      => $inputs,
195            'report'      => $report,
196            'title'       => $title,
197            'tree'        => $tree,
198        ]);
199    }
200}
201