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