1<?php 2 3/** 4 * webtrees: online genealogy 5 * Copyright (C) 2021 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 Aura\Router\Route; 23use Fisharebest\Webtrees\Auth; 24use Fisharebest\Webtrees\Family; 25use Fisharebest\Webtrees\Gedcom; 26use Fisharebest\Webtrees\GedcomRecord; 27use Fisharebest\Webtrees\Http\RequestHandlers\FamilyPage; 28use Fisharebest\Webtrees\Http\RequestHandlers\IndividualPage; 29use Fisharebest\Webtrees\Http\RequestHandlers\LocationPage; 30use Fisharebest\Webtrees\Http\RequestHandlers\MediaPage; 31use Fisharebest\Webtrees\Http\RequestHandlers\NotePage; 32use Fisharebest\Webtrees\Http\RequestHandlers\RepositoryPage; 33use Fisharebest\Webtrees\Http\RequestHandlers\SourcePage; 34use Fisharebest\Webtrees\Http\RequestHandlers\SubmitterPage; 35use Fisharebest\Webtrees\I18N; 36use Fisharebest\Webtrees\Individual; 37use Fisharebest\Webtrees\Location; 38use Fisharebest\Webtrees\Media; 39use Fisharebest\Webtrees\Menu; 40use Fisharebest\Webtrees\Note; 41use Fisharebest\Webtrees\Registry; 42use Fisharebest\Webtrees\Repository; 43use Fisharebest\Webtrees\Services\GedcomExportService; 44use Fisharebest\Webtrees\Session; 45use Fisharebest\Webtrees\Source; 46use Fisharebest\Webtrees\Submitter; 47use Fisharebest\Webtrees\Tree; 48use Illuminate\Support\Collection; 49use League\Flysystem\Filesystem; 50use League\Flysystem\FilesystemException; 51use League\Flysystem\ZipArchive\FilesystemZipArchiveProvider; 52use League\Flysystem\ZipArchive\ZipArchiveAdapter; 53use Psr\Http\Message\ResponseFactoryInterface; 54use Psr\Http\Message\ResponseInterface; 55use Psr\Http\Message\ServerRequestInterface; 56use Psr\Http\Message\StreamFactoryInterface; 57 58use function app; 59use function array_filter; 60use function array_keys; 61use function array_map; 62use function array_search; 63use function assert; 64use function fclose; 65use function in_array; 66use function is_array; 67use function is_string; 68use function preg_match_all; 69use function redirect; 70use function route; 71use function str_replace; 72use function stream_get_meta_data; 73use function tmpfile; 74use function uasort; 75use function view; 76 77use const PREG_SET_ORDER; 78 79/** 80 * Class ClippingsCartModule 81 */ 82class ClippingsCartModule extends AbstractModule implements ModuleMenuInterface 83{ 84 use ModuleMenuTrait; 85 86 // What to add to the cart? 87 private const ADD_RECORD_ONLY = 'record'; 88 private const ADD_CHILDREN = 'children'; 89 private const ADD_DESCENDANTS = 'descendants'; 90 private const ADD_PARENT_FAMILIES = 'parents'; 91 private const ADD_SPOUSE_FAMILIES = 'spouses'; 92 private const ADD_ANCESTORS = 'ancestors'; 93 private const ADD_ANCESTOR_FAMILIES = 'families'; 94 private const ADD_LINKED_INDIVIDUALS = 'linked'; 95 96 // Routes that have a record which can be added to the clipboard 97 private const ROUTES_WITH_RECORDS = [ 98 'Family' => FamilyPage::class, 99 'Individual' => IndividualPage::class, 100 'Media' => MediaPage::class, 101 'Location' => LocationPage::class, 102 'Note' => NotePage::class, 103 'Repository' => RepositoryPage::class, 104 'Source' => SourcePage::class, 105 'Submitter' => SubmitterPage::class, 106 ]; 107 108 /** @var int The default access level for this module. It can be changed in the control panel. */ 109 protected int $access_level = Auth::PRIV_USER; 110 111 private GedcomExportService $gedcom_export_service; 112 113 private ResponseFactoryInterface $response_factory; 114 115 private StreamFactoryInterface $stream_factory; 116 117 /** 118 * ClippingsCartModule constructor. 119 * 120 * @param GedcomExportService $gedcom_export_service 121 * @param ResponseFactoryInterface $response_factory 122 * @param StreamFactoryInterface $stream_factory 123 */ 124 public function __construct( 125 GedcomExportService $gedcom_export_service, 126 ResponseFactoryInterface $response_factory, 127 StreamFactoryInterface $stream_factory 128 ) { 129 $this->gedcom_export_service = $gedcom_export_service; 130 $this->response_factory = $response_factory; 131 $this->stream_factory = $stream_factory; 132 } 133 134 /** 135 * A sentence describing what this module does. 136 * 137 * @return string 138 */ 139 public function description(): string 140 { 141 /* I18N: Description of the “Clippings cart” module */ 142 return I18N::translate('Select records from your family tree and save them as a GEDCOM file.'); 143 } 144 145 /** 146 * The default position for this menu. It can be changed in the control panel. 147 * 148 * @return int 149 */ 150 public function defaultMenuOrder(): int 151 { 152 return 6; 153 } 154 155 /** 156 * A menu, to be added to the main application menu. 157 * 158 * @param Tree $tree 159 * 160 * @return Menu|null 161 */ 162 public function getMenu(Tree $tree): ?Menu 163 { 164 /** @var ServerRequestInterface $request */ 165 $request = app(ServerRequestInterface::class); 166 167 $route = $request->getAttribute('route'); 168 assert($route instanceof Route); 169 170 $cart = Session::get('cart'); 171 $cart = is_array($cart) ? $cart : []; 172 $count = count($cart[$tree->name()] ?? []); 173 $badge = view('components/badge', ['count' => $count]); 174 175 $submenus = [ 176 new Menu($this->title() . ' ' . $badge, route('module', [ 177 'module' => $this->name(), 178 'action' => 'Show', 179 'tree' => $tree->name(), 180 ]), 'menu-clippings-cart', ['rel' => 'nofollow']), 181 ]; 182 183 $action = array_search($route->name, self::ROUTES_WITH_RECORDS, true); 184 if ($action !== false) { 185 $xref = $route->attributes['xref']; 186 assert(is_string($xref)); 187 188 $add_route = route('module', [ 189 'module' => $this->name(), 190 'action' => 'Add' . $action, 191 'xref' => $xref, 192 'tree' => $tree->name(), 193 ]); 194 195 $submenus[] = new Menu(I18N::translate('Add to the clippings cart'), $add_route, 'menu-clippings-add', ['rel' => 'nofollow']); 196 } 197 198 if (!$this->isCartEmpty($tree)) { 199 $submenus[] = new Menu(I18N::translate('Empty the clippings cart'), route('module', [ 200 'module' => $this->name(), 201 'action' => 'Empty', 202 'tree' => $tree->name(), 203 ]), 'menu-clippings-empty', ['rel' => 'nofollow']); 204 205 $submenus[] = new Menu(I18N::translate('Download'), route('module', [ 206 'module' => $this->name(), 207 'action' => 'DownloadForm', 208 'tree' => $tree->name(), 209 ]), 'menu-clippings-download', ['rel' => 'nofollow']); 210 } 211 212 return new Menu($this->title(), '#', 'menu-clippings', ['rel' => 'nofollow'], $submenus); 213 } 214 215 /** 216 * How should this module be identified in the control panel, etc.? 217 * 218 * @return string 219 */ 220 public function title(): string 221 { 222 /* I18N: Name of a module */ 223 return I18N::translate('Clippings cart'); 224 } 225 226 /** 227 * @param Tree $tree 228 * 229 * @return bool 230 */ 231 private function isCartEmpty(Tree $tree): bool 232 { 233 $cart = Session::get('cart'); 234 $cart = is_array($cart) ? $cart : []; 235 $contents = $cart[$tree->name()] ?? []; 236 237 return $contents === []; 238 } 239 240 /** 241 * @param ServerRequestInterface $request 242 * 243 * @return ResponseInterface 244 */ 245 public function getDownloadFormAction(ServerRequestInterface $request): ResponseInterface 246 { 247 $tree = $request->getAttribute('tree'); 248 assert($tree instanceof Tree); 249 250 $user = $request->getAttribute('user'); 251 $title = I18N::translate('Family tree clippings cart') . ' — ' . I18N::translate('Download'); 252 253 return $this->viewResponse('modules/clippings/download', [ 254 'is_manager' => Auth::isManager($tree, $user), 255 'is_member' => Auth::isMember($tree, $user), 256 'module' => $this->name(), 257 'title' => $title, 258 'tree' => $tree, 259 ]); 260 } 261 262 /** 263 * @param ServerRequestInterface $request 264 * 265 * @return ResponseInterface 266 * @throws FilesystemException 267 */ 268 public function postDownloadAction(ServerRequestInterface $request): ResponseInterface 269 { 270 $tree = $request->getAttribute('tree'); 271 assert($tree instanceof Tree); 272 273 $data_filesystem = Registry::filesystem()->data(); 274 275 $params = (array) $request->getParsedBody(); 276 277 $privatize_export = $params['privatize_export'] ?? 'none'; 278 279 if ($privatize_export === 'none' && !Auth::isManager($tree)) { 280 $privatize_export = 'member'; 281 } 282 283 if ($privatize_export === 'gedadmin' && !Auth::isManager($tree)) { 284 $privatize_export = 'member'; 285 } 286 287 if ($privatize_export === 'user' && !Auth::isMember($tree)) { 288 $privatize_export = 'visitor'; 289 } 290 291 $convert = (bool) ($params['convert'] ?? false); 292 293 $cart = Session::get('cart'); 294 $cart = is_array($cart) ? $cart : []; 295 296 $xrefs = array_keys($cart[$tree->name()] ?? []); 297 $xrefs = array_map('strval', $xrefs); // PHP converts numeric keys to integers. 298 299 // Create a new/empty .ZIP file 300 $temp_zip_file = stream_get_meta_data(tmpfile())['uri']; 301 $zip_provider = new FilesystemZipArchiveProvider($temp_zip_file, 0755); 302 $zip_adapter = new ZipArchiveAdapter($zip_provider); 303 $zip_filesystem = new Filesystem($zip_adapter); 304 305 $media_filesystem = $tree->mediaFilesystem($data_filesystem); 306 307 // Media file prefix 308 $path = $tree->getPreference('MEDIA_DIRECTORY'); 309 310 $encoding = $convert ? 'ANSI' : 'UTF-8'; 311 312 $records = new Collection(); 313 314 switch ($privatize_export) { 315 case 'gedadmin': 316 $access_level = Auth::PRIV_NONE; 317 break; 318 case 'user': 319 $access_level = Auth::PRIV_USER; 320 break; 321 case 'visitor': 322 $access_level = Auth::PRIV_PRIVATE; 323 break; 324 case 'none': 325 default: 326 $access_level = Auth::PRIV_HIDE; 327 break; 328 } 329 330 foreach ($xrefs as $xref) { 331 $object = Registry::gedcomRecordFactory()->make($xref, $tree); 332 // The object may have been deleted since we added it to the cart.... 333 if ($object instanceof GedcomRecord) { 334 $record = $object->privatizeGedcom($access_level); 335 // Remove links to objects that aren't in the cart 336 preg_match_all('/\n1 ' . Gedcom::REGEX_TAG . ' @(' . Gedcom::REGEX_XREF . ')@(\n[2-9].*)*/', $record, $matches, PREG_SET_ORDER); 337 foreach ($matches as $match) { 338 if (!in_array($match[1], $xrefs, true)) { 339 $record = str_replace($match[0], '', $record); 340 } 341 } 342 preg_match_all('/\n2 ' . Gedcom::REGEX_TAG . ' @(' . Gedcom::REGEX_XREF . ')@(\n[3-9].*)*/', $record, $matches, PREG_SET_ORDER); 343 foreach ($matches as $match) { 344 if (!in_array($match[1], $xrefs, true)) { 345 $record = str_replace($match[0], '', $record); 346 } 347 } 348 preg_match_all('/\n3 ' . Gedcom::REGEX_TAG . ' @(' . Gedcom::REGEX_XREF . ')@(\n[4-9].*)*/', $record, $matches, PREG_SET_ORDER); 349 foreach ($matches as $match) { 350 if (!in_array($match[1], $xrefs, true)) { 351 $record = str_replace($match[0], '', $record); 352 } 353 } 354 355 $records->add($record); 356 357 if ($object instanceof Media) { 358 // Add the media files to the archive 359 foreach ($object->mediaFiles() as $media_file) { 360 $from = $media_file->filename(); 361 $to = $path . $media_file->filename(); 362 if (!$media_file->isExternal() && $media_filesystem->fileExists($from)) { 363 $zip_filesystem->writeStream($to, $media_filesystem->readStream($from)); 364 } 365 } 366 } 367 } 368 } 369 370 // We have already applied privacy filtering, so do not do it again. 371 $resource = $this->gedcom_export_service->export($tree, false, $encoding, Auth::PRIV_HIDE, $path, $records); 372 373 // Finally add the GEDCOM file to the .ZIP file. 374 $zip_filesystem->writeStream('clippings.ged', $resource); 375 fclose($resource); 376 377 // Use a stream, so that we do not have to load the entire file into memory. 378 $resource = $this->stream_factory->createStreamFromFile($temp_zip_file); 379 380 return $this->response_factory->createResponse() 381 ->withBody($resource) 382 ->withHeader('Content-Type', 'application/zip') 383 ->withHeader('Content-Disposition', 'attachment; filename="clippings.zip'); 384 } 385 386 /** 387 * @param ServerRequestInterface $request 388 * 389 * @return ResponseInterface 390 */ 391 public function getEmptyAction(ServerRequestInterface $request): ResponseInterface 392 { 393 $tree = $request->getAttribute('tree'); 394 assert($tree instanceof Tree); 395 396 $cart = Session::get('cart'); 397 $cart = is_array($cart) ? $cart : []; 398 399 $cart[$tree->name()] = []; 400 Session::put('cart', $cart); 401 402 $url = route('module', [ 403 'module' => $this->name(), 404 'action' => 'Show', 405 'tree' => $tree->name(), 406 ]); 407 408 return redirect($url); 409 } 410 411 /** 412 * @param ServerRequestInterface $request 413 * 414 * @return ResponseInterface 415 */ 416 public function postRemoveAction(ServerRequestInterface $request): ResponseInterface 417 { 418 $tree = $request->getAttribute('tree'); 419 assert($tree instanceof Tree); 420 421 $xref = $request->getQueryParams()['xref'] ?? ''; 422 423 $cart = Session::get('cart'); 424 $cart = is_array($cart) ? $cart : []; 425 426 unset($cart[$tree->name()][$xref]); 427 Session::put('cart', $cart); 428 429 $url = route('module', [ 430 'module' => $this->name(), 431 'action' => 'Show', 432 'tree' => $tree->name(), 433 ]); 434 435 return redirect($url); 436 } 437 438 /** 439 * @param ServerRequestInterface $request 440 * 441 * @return ResponseInterface 442 */ 443 public function getShowAction(ServerRequestInterface $request): ResponseInterface 444 { 445 $tree = $request->getAttribute('tree'); 446 assert($tree instanceof Tree); 447 448 return $this->viewResponse('modules/clippings/show', [ 449 'module' => $this->name(), 450 'records' => $this->allRecordsInCart($tree), 451 'title' => I18N::translate('Family tree clippings cart'), 452 'tree' => $tree, 453 ]); 454 } 455 456 /** 457 * Get all the records in the cart. 458 * 459 * @param Tree $tree 460 * 461 * @return array<GedcomRecord> 462 */ 463 private function allRecordsInCart(Tree $tree): array 464 { 465 $cart = Session::get('cart'); 466 $cart = is_array($cart) ? $cart : []; 467 468 $xrefs = array_keys($cart[$tree->name()] ?? []); 469 $xrefs = array_map('strval', $xrefs); // PHP converts numeric keys to integers. 470 471 // Fetch all the records in the cart. 472 $records = array_map(static function (string $xref) use ($tree): ?GedcomRecord { 473 return Registry::gedcomRecordFactory()->make($xref, $tree); 474 }, $xrefs); 475 476 // Some records may have been deleted after they were added to the cart. 477 $records = array_filter($records); 478 479 // Group and sort. 480 uasort($records, static function (GedcomRecord $x, GedcomRecord $y): int { 481 return $x->tag() <=> $y->tag() ?: GedcomRecord::nameComparator()($x, $y); 482 }); 483 484 return $records; 485 } 486 487 /** 488 * @param ServerRequestInterface $request 489 * 490 * @return ResponseInterface 491 */ 492 public function getAddFamilyAction(ServerRequestInterface $request): ResponseInterface 493 { 494 $tree = $request->getAttribute('tree'); 495 assert($tree instanceof Tree); 496 497 $xref = $request->getQueryParams()['xref'] ?? ''; 498 499 $family = Registry::familyFactory()->make($xref, $tree); 500 $family = Auth::checkFamilyAccess($family); 501 $name = $family->fullName(); 502 503 $options = [ 504 self::ADD_RECORD_ONLY => $name, 505 /* I18N: %s is a family (husband + wife) */ 506 self::ADD_CHILDREN => I18N::translate('%s and their children', $name), 507 /* I18N: %s is a family (husband + wife) */ 508 self::ADD_DESCENDANTS => I18N::translate('%s and their descendants', $name), 509 ]; 510 511 $title = I18N::translate('Add %s to the clippings cart', $name); 512 513 return $this->viewResponse('modules/clippings/add-options', [ 514 'options' => $options, 515 'record' => $family, 516 'title' => $title, 517 'tree' => $tree, 518 ]); 519 } 520 521 /** 522 * @param ServerRequestInterface $request 523 * 524 * @return ResponseInterface 525 */ 526 public function postAddFamilyAction(ServerRequestInterface $request): ResponseInterface 527 { 528 $tree = $request->getAttribute('tree'); 529 assert($tree instanceof Tree); 530 531 $params = (array) $request->getParsedBody(); 532 533 $xref = $params['xref'] ?? ''; 534 $option = $params['option'] ?? ''; 535 536 $family = Registry::familyFactory()->make($xref, $tree); 537 $family = Auth::checkFamilyAccess($family); 538 539 switch ($option) { 540 case self::ADD_RECORD_ONLY: 541 $this->addFamilyToCart($family); 542 break; 543 544 case self::ADD_CHILDREN: 545 $this->addFamilyAndChildrenToCart($family); 546 break; 547 548 case self::ADD_DESCENDANTS: 549 $this->addFamilyAndDescendantsToCart($family); 550 break; 551 } 552 553 return redirect($family->url()); 554 } 555 556 557 /** 558 * @param Family $family 559 * 560 * @return void 561 */ 562 protected function addFamilyAndChildrenToCart(Family $family): void 563 { 564 $this->addFamilyToCart($family); 565 566 foreach ($family->children() as $child) { 567 $this->addIndividualToCart($child); 568 } 569 } 570 571 /** 572 * @param Family $family 573 * 574 * @return void 575 */ 576 protected function addFamilyAndDescendantsToCart(Family $family): void 577 { 578 $this->addFamilyAndChildrenToCart($family); 579 580 foreach ($family->children() as $child) { 581 foreach ($child->spouseFamilies() as $child_family) { 582 $this->addFamilyAndDescendantsToCart($child_family); 583 } 584 } 585 } 586 587 /** 588 * @param ServerRequestInterface $request 589 * 590 * @return ResponseInterface 591 */ 592 public function getAddIndividualAction(ServerRequestInterface $request): ResponseInterface 593 { 594 $tree = $request->getAttribute('tree'); 595 assert($tree instanceof Tree); 596 597 $xref = $request->getQueryParams()['xref'] ?? ''; 598 599 $individual = Registry::individualFactory()->make($xref, $tree); 600 $individual = Auth::checkIndividualAccess($individual); 601 $name = $individual->fullName(); 602 603 if ($individual->sex() === 'F') { 604 $options = [ 605 self::ADD_RECORD_ONLY => $name, 606 self::ADD_PARENT_FAMILIES => I18N::translate('%s, her parents and siblings', $name), 607 self::ADD_SPOUSE_FAMILIES => I18N::translate('%s, her spouses and children', $name), 608 self::ADD_ANCESTORS => I18N::translate('%s and her ancestors', $name), 609 self::ADD_ANCESTOR_FAMILIES => I18N::translate('%s, her ancestors and their families', $name), 610 self::ADD_DESCENDANTS => I18N::translate('%s, her spouses and descendants', $name), 611 ]; 612 } else { 613 $options = [ 614 self::ADD_RECORD_ONLY => $name, 615 self::ADD_PARENT_FAMILIES => I18N::translate('%s, his parents and siblings', $name), 616 self::ADD_SPOUSE_FAMILIES => I18N::translate('%s, his spouses and children', $name), 617 self::ADD_ANCESTORS => I18N::translate('%s and his ancestors', $name), 618 self::ADD_ANCESTOR_FAMILIES => I18N::translate('%s, his ancestors and their families', $name), 619 self::ADD_DESCENDANTS => I18N::translate('%s, his spouses and descendants', $name), 620 ]; 621 } 622 623 $title = I18N::translate('Add %s to the clippings cart', $name); 624 625 return $this->viewResponse('modules/clippings/add-options', [ 626 'options' => $options, 627 'record' => $individual, 628 'title' => $title, 629 'tree' => $tree, 630 ]); 631 } 632 633 /** 634 * @param ServerRequestInterface $request 635 * 636 * @return ResponseInterface 637 */ 638 public function postAddIndividualAction(ServerRequestInterface $request): ResponseInterface 639 { 640 $tree = $request->getAttribute('tree'); 641 assert($tree instanceof Tree); 642 643 $params = (array) $request->getParsedBody(); 644 645 $xref = $params['xref'] ?? ''; 646 $option = $params['option'] ?? ''; 647 648 $individual = Registry::individualFactory()->make($xref, $tree); 649 $individual = Auth::checkIndividualAccess($individual); 650 651 switch ($option) { 652 case self::ADD_RECORD_ONLY: 653 $this->addIndividualToCart($individual); 654 break; 655 656 case self::ADD_PARENT_FAMILIES: 657 foreach ($individual->childFamilies() as $family) { 658 $this->addFamilyAndChildrenToCart($family); 659 } 660 break; 661 662 case self::ADD_SPOUSE_FAMILIES: 663 foreach ($individual->spouseFamilies() as $family) { 664 $this->addFamilyAndChildrenToCart($family); 665 } 666 break; 667 668 case self::ADD_ANCESTORS: 669 $this->addAncestorsToCart($individual); 670 break; 671 672 case self::ADD_ANCESTOR_FAMILIES: 673 $this->addAncestorFamiliesToCart($individual); 674 break; 675 676 case self::ADD_DESCENDANTS: 677 foreach ($individual->spouseFamilies() as $family) { 678 $this->addFamilyAndDescendantsToCart($family); 679 } 680 break; 681 } 682 683 return redirect($individual->url()); 684 } 685 686 /** 687 * @param Individual $individual 688 * 689 * @return void 690 */ 691 protected function addAncestorsToCart(Individual $individual): void 692 { 693 $this->addIndividualToCart($individual); 694 695 foreach ($individual->childFamilies() as $family) { 696 $this->addFamilyToCart($family); 697 698 foreach ($family->spouses() as $parent) { 699 $this->addAncestorsToCart($parent); 700 } 701 } 702 } 703 704 /** 705 * @param Individual $individual 706 * 707 * @return void 708 */ 709 protected function addAncestorFamiliesToCart(Individual $individual): void 710 { 711 foreach ($individual->childFamilies() as $family) { 712 $this->addFamilyAndChildrenToCart($family); 713 714 foreach ($family->spouses() as $parent) { 715 $this->addAncestorFamiliesToCart($parent); 716 } 717 } 718 } 719 720 /** 721 * @param ServerRequestInterface $request 722 * 723 * @return ResponseInterface 724 */ 725 public function getAddLocationAction(ServerRequestInterface $request): ResponseInterface 726 { 727 $tree = $request->getAttribute('tree'); 728 assert($tree instanceof Tree); 729 730 $xref = $request->getQueryParams()['xref'] ?? ''; 731 732 $location = Registry::locationFactory()->make($xref, $tree); 733 $location = Auth::checkLocationAccess($location); 734 $name = $location->fullName(); 735 736 $options = [ 737 self::ADD_RECORD_ONLY => $name, 738 ]; 739 740 $title = I18N::translate('Add %s to the clippings cart', $name); 741 742 return $this->viewResponse('modules/clippings/add-options', [ 743 'options' => $options, 744 'record' => $location, 745 'title' => $title, 746 'tree' => $tree, 747 ]); 748 } 749 750 /** 751 * @param ServerRequestInterface $request 752 * 753 * @return ResponseInterface 754 */ 755 public function postAddLocationAction(ServerRequestInterface $request): ResponseInterface 756 { 757 $tree = $request->getAttribute('tree'); 758 assert($tree instanceof Tree); 759 760 $xref = $request->getQueryParams()['xref'] ?? ''; 761 762 $location = Registry::locationFactory()->make($xref, $tree); 763 $location = Auth::checkLocationAccess($location); 764 765 $this->addLocationToCart($location); 766 767 return redirect($location->url()); 768 } 769 770 /** 771 * @param ServerRequestInterface $request 772 * 773 * @return ResponseInterface 774 */ 775 public function getAddMediaAction(ServerRequestInterface $request): ResponseInterface 776 { 777 $tree = $request->getAttribute('tree'); 778 assert($tree instanceof Tree); 779 780 $xref = $request->getQueryParams()['xref'] ?? ''; 781 782 $media = Registry::mediaFactory()->make($xref, $tree); 783 $media = Auth::checkMediaAccess($media); 784 $name = $media->fullName(); 785 786 $options = [ 787 self::ADD_RECORD_ONLY => $name, 788 ]; 789 790 $title = I18N::translate('Add %s to the clippings cart', $name); 791 792 return $this->viewResponse('modules/clippings/add-options', [ 793 'options' => $options, 794 'record' => $media, 795 'title' => $title, 796 'tree' => $tree, 797 ]); 798 } 799 800 /** 801 * @param ServerRequestInterface $request 802 * 803 * @return ResponseInterface 804 */ 805 public function postAddMediaAction(ServerRequestInterface $request): ResponseInterface 806 { 807 $tree = $request->getAttribute('tree'); 808 assert($tree instanceof Tree); 809 810 $xref = $request->getQueryParams()['xref'] ?? ''; 811 812 $media = Registry::mediaFactory()->make($xref, $tree); 813 $media = Auth::checkMediaAccess($media); 814 815 $this->addMediaToCart($media); 816 817 return redirect($media->url()); 818 } 819 820 /** 821 * @param ServerRequestInterface $request 822 * 823 * @return ResponseInterface 824 */ 825 public function getAddNoteAction(ServerRequestInterface $request): ResponseInterface 826 { 827 $tree = $request->getAttribute('tree'); 828 assert($tree instanceof Tree); 829 830 $xref = $request->getQueryParams()['xref'] ?? ''; 831 832 $note = Registry::noteFactory()->make($xref, $tree); 833 $note = Auth::checkNoteAccess($note); 834 $name = $note->fullName(); 835 836 $options = [ 837 self::ADD_RECORD_ONLY => $name, 838 ]; 839 840 $title = I18N::translate('Add %s to the clippings cart', $name); 841 842 return $this->viewResponse('modules/clippings/add-options', [ 843 'options' => $options, 844 'record' => $note, 845 'title' => $title, 846 'tree' => $tree, 847 ]); 848 } 849 850 /** 851 * @param ServerRequestInterface $request 852 * 853 * @return ResponseInterface 854 */ 855 public function postAddNoteAction(ServerRequestInterface $request): ResponseInterface 856 { 857 $tree = $request->getAttribute('tree'); 858 assert($tree instanceof Tree); 859 860 $xref = $request->getQueryParams()['xref'] ?? ''; 861 862 $note = Registry::noteFactory()->make($xref, $tree); 863 $note = Auth::checkNoteAccess($note); 864 865 $this->addNoteToCart($note); 866 867 return redirect($note->url()); 868 } 869 870 /** 871 * @param ServerRequestInterface $request 872 * 873 * @return ResponseInterface 874 */ 875 public function getAddRepositoryAction(ServerRequestInterface $request): ResponseInterface 876 { 877 $tree = $request->getAttribute('tree'); 878 assert($tree instanceof Tree); 879 880 $xref = $request->getQueryParams()['xref'] ?? ''; 881 882 $repository = Registry::repositoryFactory()->make($xref, $tree); 883 $repository = Auth::checkRepositoryAccess($repository); 884 $name = $repository->fullName(); 885 886 $options = [ 887 self::ADD_RECORD_ONLY => $name, 888 ]; 889 890 $title = I18N::translate('Add %s to the clippings cart', $name); 891 892 return $this->viewResponse('modules/clippings/add-options', [ 893 'options' => $options, 894 'record' => $repository, 895 'title' => $title, 896 'tree' => $tree, 897 ]); 898 } 899 900 /** 901 * @param ServerRequestInterface $request 902 * 903 * @return ResponseInterface 904 */ 905 public function postAddRepositoryAction(ServerRequestInterface $request): ResponseInterface 906 { 907 $tree = $request->getAttribute('tree'); 908 assert($tree instanceof Tree); 909 910 $xref = $request->getQueryParams()['xref'] ?? ''; 911 912 $repository = Registry::repositoryFactory()->make($xref, $tree); 913 $repository = Auth::checkRepositoryAccess($repository); 914 915 $this->addRepositoryToCart($repository); 916 917 foreach ($repository->linkedSources('REPO') as $source) { 918 $this->addSourceToCart($source); 919 } 920 921 return redirect($repository->url()); 922 } 923 924 /** 925 * @param ServerRequestInterface $request 926 * 927 * @return ResponseInterface 928 */ 929 public function getAddSourceAction(ServerRequestInterface $request): ResponseInterface 930 { 931 $tree = $request->getAttribute('tree'); 932 assert($tree instanceof Tree); 933 934 $xref = $request->getQueryParams()['xref'] ?? ''; 935 936 $source = Registry::sourceFactory()->make($xref, $tree); 937 $source = Auth::checkSourceAccess($source); 938 $name = $source->fullName(); 939 940 $options = [ 941 self::ADD_RECORD_ONLY => $name, 942 self::ADD_LINKED_INDIVIDUALS => I18N::translate('%s and the individuals that reference it.', $name), 943 ]; 944 945 $title = I18N::translate('Add %s to the clippings cart', $name); 946 947 return $this->viewResponse('modules/clippings/add-options', [ 948 'options' => $options, 949 'record' => $source, 950 'title' => $title, 951 'tree' => $tree, 952 ]); 953 } 954 955 /** 956 * @param ServerRequestInterface $request 957 * 958 * @return ResponseInterface 959 */ 960 public function postAddSourceAction(ServerRequestInterface $request): ResponseInterface 961 { 962 $tree = $request->getAttribute('tree'); 963 assert($tree instanceof Tree); 964 965 $params = (array) $request->getParsedBody(); 966 967 $xref = $params['xref'] ?? ''; 968 $option = $params['option'] ?? ''; 969 970 $source = Registry::sourceFactory()->make($xref, $tree); 971 $source = Auth::checkSourceAccess($source); 972 973 $this->addSourceToCart($source); 974 975 if ($option === self::ADD_LINKED_INDIVIDUALS) { 976 foreach ($source->linkedIndividuals('SOUR') as $individual) { 977 $this->addIndividualToCart($individual); 978 } 979 foreach ($source->linkedFamilies('SOUR') as $family) { 980 $this->addFamilyToCart($family); 981 } 982 } 983 984 return redirect($source->url()); 985 } 986 987 /** 988 * @param ServerRequestInterface $request 989 * 990 * @return ResponseInterface 991 */ 992 public function getAddSubmitterAction(ServerRequestInterface $request): ResponseInterface 993 { 994 $tree = $request->getAttribute('tree'); 995 assert($tree instanceof Tree); 996 997 $xref = $request->getQueryParams()['xref'] ?? ''; 998 999 $submitter = Registry::submitterFactory()->make($xref, $tree); 1000 $submitter = Auth::checkSubmitterAccess($submitter); 1001 $name = $submitter->fullName(); 1002 1003 $options = [ 1004 self::ADD_RECORD_ONLY => $name, 1005 ]; 1006 1007 $title = I18N::translate('Add %s to the clippings cart', $name); 1008 1009 return $this->viewResponse('modules/clippings/add-options', [ 1010 'options' => $options, 1011 'record' => $submitter, 1012 'title' => $title, 1013 'tree' => $tree, 1014 ]); 1015 } 1016 1017 /** 1018 * @param ServerRequestInterface $request 1019 * 1020 * @return ResponseInterface 1021 */ 1022 public function postAddSubmitterAction(ServerRequestInterface $request): ResponseInterface 1023 { 1024 $tree = $request->getAttribute('tree'); 1025 assert($tree instanceof Tree); 1026 1027 $xref = $request->getQueryParams()['xref'] ?? ''; 1028 1029 $submitter = Registry::submitterFactory()->make($xref, $tree); 1030 $submitter = Auth::checkSubmitterAccess($submitter); 1031 1032 $this->addSubmitterToCart($submitter); 1033 1034 return redirect($submitter->url()); 1035 } 1036 1037 /** 1038 * @param Family $family 1039 */ 1040 protected function addFamilyToCart(Family $family): void 1041 { 1042 $cart = Session::get('cart'); 1043 $cart = is_array($cart) ? $cart : []; 1044 1045 $tree = $family->tree()->name(); 1046 $xref = $family->xref(); 1047 1048 if (($cart[$tree][$xref] ?? false) === false) { 1049 $cart[$tree][$xref] = true; 1050 1051 Session::put('cart', $cart); 1052 1053 foreach ($family->spouses() as $spouse) { 1054 $this->addIndividualToCart($spouse); 1055 } 1056 1057 $this->addLocationLinksToCart($family); 1058 $this->addMediaLinksToCart($family); 1059 $this->addNoteLinksToCart($family); 1060 $this->addSourceLinksToCart($family); 1061 $this->addSubmitterLinksToCart($family); 1062 } 1063 } 1064 1065 /** 1066 * @param Individual $individual 1067 */ 1068 protected function addIndividualToCart(Individual $individual): void 1069 { 1070 $cart = Session::get('cart'); 1071 $cart = is_array($cart) ? $cart : []; 1072 1073 $tree = $individual->tree()->name(); 1074 $xref = $individual->xref(); 1075 1076 if (($cart[$tree][$xref] ?? false) === false) { 1077 $cart[$tree][$xref] = true; 1078 1079 Session::put('cart', $cart); 1080 1081 $this->addLocationLinksToCart($individual); 1082 $this->addMediaLinksToCart($individual); 1083 $this->addNoteLinksToCart($individual); 1084 $this->addSourceLinksToCart($individual); 1085 } 1086 } 1087 1088 /** 1089 * @param Location $location 1090 */ 1091 protected function addLocationToCart(Location $location): void 1092 { 1093 $cart = Session::get('cart'); 1094 $cart = is_array($cart) ? $cart : []; 1095 1096 $tree = $location->tree()->name(); 1097 $xref = $location->xref(); 1098 1099 if (($cart[$tree][$xref] ?? false) === false) { 1100 $cart[$tree][$xref] = true; 1101 1102 Session::put('cart', $cart); 1103 1104 $this->addLocationLinksToCart($location); 1105 $this->addMediaLinksToCart($location); 1106 $this->addNoteLinksToCart($location); 1107 $this->addSourceLinksToCart($location); 1108 } 1109 } 1110 1111 /** 1112 * @param GedcomRecord $record 1113 */ 1114 protected function addLocationLinksToCart(GedcomRecord $record): void 1115 { 1116 preg_match_all('/\n\d _LOC @(' . Gedcom::REGEX_XREF . ')@/', $record->gedcom(), $matches); 1117 1118 foreach ($matches[1] as $xref) { 1119 $location = Registry::locationFactory()->make($xref, $record->tree()); 1120 1121 if ($location instanceof Location && $location->canShow()) { 1122 $this->addLocationToCart($location); 1123 } 1124 } 1125 } 1126 1127 /** 1128 * @param Media $media 1129 */ 1130 protected function addMediaToCart(Media $media): void 1131 { 1132 $cart = Session::get('cart'); 1133 $cart = is_array($cart) ? $cart : []; 1134 1135 $tree = $media->tree()->name(); 1136 $xref = $media->xref(); 1137 1138 if (($cart[$tree][$xref] ?? false) === false) { 1139 $cart[$tree][$xref] = true; 1140 1141 Session::put('cart', $cart); 1142 1143 $this->addNoteLinksToCart($media); 1144 } 1145 } 1146 1147 /** 1148 * @param GedcomRecord $record 1149 */ 1150 protected function addMediaLinksToCart(GedcomRecord $record): void 1151 { 1152 preg_match_all('/\n\d OBJE @(' . Gedcom::REGEX_XREF . ')@/', $record->gedcom(), $matches); 1153 1154 foreach ($matches[1] as $xref) { 1155 $media = Registry::mediaFactory()->make($xref, $record->tree()); 1156 1157 if ($media instanceof Media && $media->canShow()) { 1158 $this->addMediaToCart($media); 1159 } 1160 } 1161 } 1162 1163 /** 1164 * @param Note $note 1165 */ 1166 protected function addNoteToCart(Note $note): void 1167 { 1168 $cart = Session::get('cart'); 1169 $cart = is_array($cart) ? $cart : []; 1170 1171 $tree = $note->tree()->name(); 1172 $xref = $note->xref(); 1173 1174 if (($cart[$tree][$xref] ?? false) === false) { 1175 $cart[$tree][$xref] = true; 1176 1177 Session::put('cart', $cart); 1178 } 1179 } 1180 1181 /** 1182 * @param GedcomRecord $record 1183 */ 1184 protected function addNoteLinksToCart(GedcomRecord $record): void 1185 { 1186 preg_match_all('/\n\d NOTE @(' . Gedcom::REGEX_XREF . ')@/', $record->gedcom(), $matches); 1187 1188 foreach ($matches[1] as $xref) { 1189 $note = Registry::noteFactory()->make($xref, $record->tree()); 1190 1191 if ($note instanceof Note && $note->canShow()) { 1192 $this->addNoteToCart($note); 1193 } 1194 } 1195 } 1196 1197 /** 1198 * @param Source $source 1199 */ 1200 protected function addSourceToCart(Source $source): void 1201 { 1202 $cart = Session::get('cart'); 1203 $cart = is_array($cart) ? $cart : []; 1204 1205 $tree = $source->tree()->name(); 1206 $xref = $source->xref(); 1207 1208 if (($cart[$tree][$xref] ?? false) === false) { 1209 $cart[$tree][$xref] = true; 1210 1211 Session::put('cart', $cart); 1212 1213 $this->addNoteLinksToCart($source); 1214 $this->addRepositoryLinksToCart($source); 1215 } 1216 } 1217 1218 /** 1219 * @param GedcomRecord $record 1220 */ 1221 protected function addSourceLinksToCart(GedcomRecord $record): void 1222 { 1223 preg_match_all('/\n\d SOUR @(' . Gedcom::REGEX_XREF . ')@/', $record->gedcom(), $matches); 1224 1225 foreach ($matches[1] as $xref) { 1226 $source = Registry::sourceFactory()->make($xref, $record->tree()); 1227 1228 if ($source instanceof Source && $source->canShow()) { 1229 $this->addSourceToCart($source); 1230 } 1231 } 1232 } 1233 1234 /** 1235 * @param Repository $repository 1236 */ 1237 protected function addRepositoryToCart(Repository $repository): void 1238 { 1239 $cart = Session::get('cart'); 1240 $cart = is_array($cart) ? $cart : []; 1241 1242 $tree = $repository->tree()->name(); 1243 $xref = $repository->xref(); 1244 1245 if (($cart[$tree][$xref] ?? false) === false) { 1246 $cart[$tree][$xref] = true; 1247 1248 Session::put('cart', $cart); 1249 1250 $this->addNoteLinksToCart($repository); 1251 } 1252 } 1253 1254 /** 1255 * @param GedcomRecord $record 1256 */ 1257 protected function addRepositoryLinksToCart(GedcomRecord $record): void 1258 { 1259 preg_match_all('/\n\d REPO @(' . Gedcom::REGEX_XREF . '@)/', $record->gedcom(), $matches); 1260 1261 foreach ($matches[1] as $xref) { 1262 $repository = Registry::repositoryFactory()->make($xref, $record->tree()); 1263 1264 if ($repository instanceof Repository && $repository->canShow()) { 1265 $this->addRepositoryToCart($repository); 1266 } 1267 } 1268 } 1269 1270 /** 1271 * @param Submitter $submitter 1272 */ 1273 protected function addSubmitterToCart(Submitter $submitter): void 1274 { 1275 $cart = Session::get('cart'); 1276 $cart = is_array($cart) ? $cart : []; 1277 $tree = $submitter->tree()->name(); 1278 $xref = $submitter->xref(); 1279 1280 if (($cart[$tree][$xref] ?? false) === false) { 1281 $cart[$tree][$xref] = true; 1282 1283 Session::put('cart', $cart); 1284 1285 $this->addNoteLinksToCart($submitter); 1286 } 1287 } 1288 1289 /** 1290 * @param GedcomRecord $record 1291 */ 1292 protected function addSubmitterLinksToCart(GedcomRecord $record): void 1293 { 1294 preg_match_all('/\n\d SUBM @(' . Gedcom::REGEX_XREF . ')@/', $record->gedcom(), $matches); 1295 1296 foreach ($matches[1] as $xref) { 1297 $submitter = Registry::submitterFactory()->make($xref, $record->tree()); 1298 1299 if ($submitter instanceof Submitter && $submitter->canShow()) { 1300 $this->addSubmitterToCart($submitter); 1301 } 1302 } 1303 } 1304} 1305