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