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\Module; 21 22use Fisharebest\Webtrees\Auth; 23use Fisharebest\Webtrees\Registry; 24use Fisharebest\Webtrees\Http\RequestHandlers\ControlPanel; 25use Fisharebest\Webtrees\I18N; 26use Fisharebest\Webtrees\Individual; 27use Fisharebest\Webtrees\Menu; 28use Fisharebest\Webtrees\Services\HtmlService; 29use Fisharebest\Webtrees\Services\TreeService; 30use Fisharebest\Webtrees\Tree; 31use Fisharebest\Webtrees\Validator; 32use Illuminate\Database\Capsule\Manager as DB; 33use Psr\Http\Message\ResponseInterface; 34use Psr\Http\Message\ServerRequestInterface; 35 36use function redirect; 37use function route; 38 39/** 40 * Class StoriesModule 41 */ 42class StoriesModule extends AbstractModule implements ModuleConfigInterface, ModuleMenuInterface, ModuleTabInterface 43{ 44 use ModuleTabTrait; 45 use ModuleConfigTrait; 46 use ModuleMenuTrait; 47 48 private HtmlService $html_service; 49 50 private TreeService $tree_service; 51 52 /** 53 * StoriesModule constructor. 54 * 55 * @param HtmlService $html_service 56 * @param TreeService $tree_service 57 */ 58 public function __construct(HtmlService $html_service, TreeService $tree_service) 59 { 60 $this->html_service = $html_service; 61 $this->tree_service = $tree_service; 62 } 63 64 /** @var int The default access level for this module. It can be changed in the control panel. */ 65 protected int $access_level = Auth::PRIV_HIDE; 66 67 /** 68 * A sentence describing what this module does. 69 * 70 * @return string 71 */ 72 public function description(): string 73 { 74 /* I18N: Description of the “Stories” module */ 75 return I18N::translate('Add narrative stories to individuals in the family tree.'); 76 } 77 78 /** 79 * The default position for this menu. It can be changed in the control panel. 80 * 81 * @return int 82 */ 83 public function defaultMenuOrder(): int 84 { 85 return 7; 86 } 87 88 /** 89 * The default position for this tab. It can be changed in the control panel. 90 * 91 * @return int 92 */ 93 public function defaultTabOrder(): int 94 { 95 return 9; 96 } 97 98 /** 99 * Generate the HTML content of this tab. 100 * 101 * @param Individual $individual 102 * 103 * @return string 104 */ 105 public function getTabContent(Individual $individual): string 106 { 107 return view('modules/stories/tab', [ 108 'is_admin' => Auth::isAdmin(), 109 'individual' => $individual, 110 'stories' => $this->getStoriesForIndividual($individual), 111 'tree' => $individual->tree(), 112 ]); 113 } 114 115 /** 116 * @param Individual $individual 117 * 118 * @return array<object> 119 */ 120 private function getStoriesForIndividual(Individual $individual): array 121 { 122 $block_ids = DB::table('block') 123 ->where('module_name', '=', $this->name()) 124 ->where('xref', '=', $individual->xref()) 125 ->where('gedcom_id', '=', $individual->tree()->id()) 126 ->pluck('block_id'); 127 128 $stories = []; 129 foreach ($block_ids as $block_id) { 130 $block_id = (int) $block_id; 131 132 // Only show this block for certain languages 133 $languages = $this->getBlockSetting($block_id, 'languages'); 134 if ($languages === '' || in_array(I18N::languageTag(), explode(',', $languages), true)) { 135 $stories[] = (object) [ 136 'block_id' => $block_id, 137 'title' => $this->getBlockSetting($block_id, 'title'), 138 'story_body' => $this->getBlockSetting($block_id, 'story_body'), 139 ]; 140 } 141 } 142 143 return $stories; 144 } 145 146 /** 147 * Is this tab empty? If so, we don't always need to display it. 148 * 149 * @param Individual $individual 150 * 151 * @return bool 152 */ 153 public function hasTabContent(Individual $individual): bool 154 { 155 return Auth::isManager($individual->tree()) || $this->getStoriesForIndividual($individual) !== []; 156 } 157 158 /** 159 * A greyed out tab has no actual content, but may perhaps have 160 * options to create content. 161 * 162 * @param Individual $individual 163 * 164 * @return bool 165 */ 166 public function isGrayedOut(Individual $individual): bool 167 { 168 return $this->getStoriesForIndividual($individual) === []; 169 } 170 171 /** 172 * Can this tab load asynchronously? 173 * 174 * @return bool 175 */ 176 public function canLoadAjax(): bool 177 { 178 return false; 179 } 180 181 /** 182 * A menu, to be added to the main application menu. 183 * 184 * @param Tree $tree 185 * 186 * @return Menu|null 187 */ 188 public function getMenu(Tree $tree): ?Menu 189 { 190 return new Menu($this->title(), route('module', [ 191 'module' => $this->name(), 192 'action' => 'ShowList', 193 'tree' => $tree->name(), 194 ]), 'menu-story'); 195 } 196 197 /** 198 * How should this module be identified in the control panel, etc.? 199 * 200 * @return string 201 */ 202 public function title(): string 203 { 204 /* I18N: Name of a module */ 205 return I18N::translate('Stories'); 206 } 207 208 /** 209 * @param ServerRequestInterface $request 210 * 211 * @return ResponseInterface 212 */ 213 public function getAdminAction(ServerRequestInterface $request): ResponseInterface 214 { 215 $this->layout = 'layouts/administration'; 216 217 // This module can't run without a tree 218 $tree = Validator::attributes($request)->treeOptional(); 219 220 if (!$tree instanceof Tree) { 221 $tree = $this->tree_service->all()->first(); 222 if ($tree instanceof Tree) { 223 return redirect(route('module', ['module' => $this->name(), 'action' => 'Admin', 'tree' => $tree->name()])); 224 } 225 226 return redirect(route(ControlPanel::class)); 227 } 228 229 $stories = DB::table('block') 230 ->where('module_name', '=', $this->name()) 231 ->where('gedcom_id', '=', $tree->id()) 232 ->orderBy('xref') 233 ->get(); 234 235 foreach ($stories as $story) { 236 $block_id = (int) $story->block_id; 237 $xref = (string) $story->xref; 238 239 $story->individual = Registry::individualFactory()->make($xref, $tree); 240 $story->title = $this->getBlockSetting($block_id, 'title'); 241 $story->languages = $this->getBlockSetting($block_id, 'languages'); 242 } 243 244 $tree_names = $this->tree_service->all()->map(static function (Tree $tree): string { 245 return $tree->title(); 246 }); 247 248 return $this->viewResponse('modules/stories/config', [ 249 'module' => $this->name(), 250 'stories' => $stories, 251 'title' => $this->title() . ' — ' . $tree->title(), 252 'tree' => $tree, 253 'tree_names' => $tree_names, 254 ]); 255 } 256 257 /** 258 * @param ServerRequestInterface $request 259 * 260 * @return ResponseInterface 261 */ 262 public function postAdminAction(ServerRequestInterface $request): ResponseInterface 263 { 264 $params = (array) $request->getParsedBody(); 265 266 return redirect(route('module', [ 267 'module' => $this->name(), 268 'action' => 'Admin', 269 'tree' => $params['tree'] ?? '', 270 ])); 271 } 272 273 /** 274 * @param ServerRequestInterface $request 275 * 276 * @return ResponseInterface 277 */ 278 public function getAdminEditAction(ServerRequestInterface $request): ResponseInterface 279 { 280 $this->layout = 'layouts/administration'; 281 282 $tree = Validator::attributes($request)->tree(); 283 $block_id = (int) ($request->getQueryParams()['block_id'] ?? 0); 284 $url = $request->getQueryParams()['url'] ?? ''; 285 286 if ($block_id === 0) { 287 // Creating a new story 288 $story_title = ''; 289 $story_body = ''; 290 $languages = []; 291 $xref = $request->getQueryParams()['xref'] ?? ''; 292 $title = I18N::translate('Add a story') . ' — ' . e($tree->title()); 293 } else { 294 // Editing an existing story 295 $xref = (string) DB::table('block') 296 ->where('block_id', '=', $block_id) 297 ->value('xref'); 298 299 $story_title = $this->getBlockSetting($block_id, 'title'); 300 $story_body = $this->getBlockSetting($block_id, 'story_body'); 301 $languages = explode(',', $this->getBlockSetting($block_id, 'languages')); 302 $title = I18N::translate('Edit the story') . ' — ' . e($tree->title()); 303 } 304 305 $individual = Registry::individualFactory()->make($xref, $tree); 306 307 return $this->viewResponse('modules/stories/edit', [ 308 'block_id' => $block_id, 309 'languages' => $languages, 310 'story_body' => $story_body, 311 'story_title' => $story_title, 312 'title' => $title, 313 'tree' => $tree, 314 'url' => $url, 315 'individual' => $individual, 316 ]); 317 } 318 319 /** 320 * @param ServerRequestInterface $request 321 * 322 * @return ResponseInterface 323 */ 324 public function postAdminEditAction(ServerRequestInterface $request): ResponseInterface 325 { 326 $tree = Validator::attributes($request)->tree(); 327 $block_id = (int) ($request->getQueryParams()['block_id'] ?? 0); 328 $params = (array) $request->getParsedBody(); 329 330 $xref = $params['xref']; 331 $story_body = $params['story_body']; 332 $story_title = $params['story_title']; 333 $languages = $params['languages'] ?? []; 334 $url = $params['url'] ?? ''; 335 336 $story_body = $this->html_service->sanitize($story_body); 337 338 if ($block_id !== 0) { 339 DB::table('block') 340 ->where('block_id', '=', $block_id) 341 ->update([ 342 'gedcom_id' => $tree->id(), 343 'xref' => $xref, 344 ]); 345 } else { 346 DB::table('block')->insert([ 347 'gedcom_id' => $tree->id(), 348 'xref' => $xref, 349 'module_name' => $this->name(), 350 'block_order' => 0, 351 ]); 352 353 $block_id = (int) DB::connection()->getPdo()->lastInsertId(); 354 } 355 356 $this->setBlockSetting($block_id, 'story_body', $story_body); 357 $this->setBlockSetting($block_id, 'title', $story_title); 358 $this->setBlockSetting($block_id, 'languages', implode(',', $languages)); 359 360 $url = $url ?: route('module', [ 361 'module' => $this->name(), 362 'action' => 'Admin', 363 'tree' => $tree->name(), 364 ]); 365 366 return redirect($url); 367 } 368 369 /** 370 * @param ServerRequestInterface $request 371 * 372 * @return ResponseInterface 373 */ 374 public function postAdminDeleteAction(ServerRequestInterface $request): ResponseInterface 375 { 376 $tree = Validator::attributes($request)->tree(); 377 $block_id = $request->getQueryParams()['block_id']; 378 379 DB::table('block_setting') 380 ->where('block_id', '=', $block_id) 381 ->delete(); 382 383 DB::table('block') 384 ->where('block_id', '=', $block_id) 385 ->delete(); 386 387 $url = route('module', [ 388 'module' => $this->name(), 389 'action' => 'Admin', 390 'tree' => $tree->name(), 391 ]); 392 393 return redirect($url); 394 } 395 396 /** 397 * @param ServerRequestInterface $request 398 * 399 * @return ResponseInterface 400 */ 401 public function getShowListAction(ServerRequestInterface $request): ResponseInterface 402 { 403 $tree = Validator::attributes($request)->tree(); 404 405 $stories = DB::table('block') 406 ->where('module_name', '=', $this->name()) 407 ->where('gedcom_id', '=', $tree->id()) 408 ->get() 409 ->map(function (object $story) use ($tree): object { 410 $block_id = (int) $story->block_id; 411 $xref = (string) $story->xref; 412 413 $story->individual = Registry::individualFactory()->make($xref, $tree); 414 $story->title = $this->getBlockSetting($block_id, 'title'); 415 $story->languages = $this->getBlockSetting($block_id, 'languages'); 416 417 return $story; 418 })->filter(static function (object $story): bool { 419 // Filter non-existent and private individuals. 420 return $story->individual instanceof Individual && $story->individual->canShow(); 421 })->filter(static function (object $story): bool { 422 // Filter foreign languages. 423 return $story->languages === '' || in_array(I18N::languageTag(), explode(',', $story->languages), true); 424 }); 425 426 return $this->viewResponse('modules/stories/list', [ 427 'stories' => $stories, 428 'title' => $this->title(), 429 'tree' => $tree, 430 ]); 431 } 432} 433