xref: /webtrees/app/Module/ClippingsCartModule.php (revision dae8440ab1203b4eb5e333de33e187c1dfc5b759)
1<?php
2/**
3 * webtrees: online genealogy
4 * Copyright (C) 2019 webtrees development team
5 * This program is free software: you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License as published by
7 * the Free Software Foundation, either version 3 of the License, or
8 * (at your option) any later version.
9 * This program is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 * GNU General Public License for more details.
13 * You should have received a copy of the GNU General Public License
14 * along with this program. If not, see <http://www.gnu.org/licenses/>.
15 */
16declare(strict_types=1);
17
18namespace Fisharebest\Webtrees\Module;
19
20use Fisharebest\Webtrees\Auth;
21use Fisharebest\Webtrees\Exceptions\FamilyNotFoundException;
22use Fisharebest\Webtrees\Exceptions\IndividualNotFoundException;
23use Fisharebest\Webtrees\Exceptions\MediaNotFoundException;
24use Fisharebest\Webtrees\Exceptions\NoteNotFoundException;
25use Fisharebest\Webtrees\Exceptions\RepositoryNotFoundException;
26use Fisharebest\Webtrees\Exceptions\SourceNotFoundException;
27use Fisharebest\Webtrees\Family;
28use Fisharebest\Webtrees\Functions\FunctionsExport;
29use Fisharebest\Webtrees\Gedcom;
30use Fisharebest\Webtrees\GedcomRecord;
31use Fisharebest\Webtrees\I18N;
32use Fisharebest\Webtrees\Individual;
33use Fisharebest\Webtrees\Media;
34use Fisharebest\Webtrees\Menu;
35use Fisharebest\Webtrees\Module;
36use Fisharebest\Webtrees\Note;
37use Fisharebest\Webtrees\Repository;
38use Fisharebest\Webtrees\Session;
39use Fisharebest\Webtrees\Source;
40use Fisharebest\Webtrees\Tree;
41use Fisharebest\Webtrees\User;
42use League\Flysystem\Filesystem;
43use League\Flysystem\ZipArchive\ZipArchiveAdapter;
44use function str_replace;
45use Symfony\Component\HttpFoundation\BinaryFileResponse;
46use Symfony\Component\HttpFoundation\RedirectResponse;
47use Symfony\Component\HttpFoundation\Request;
48use Symfony\Component\HttpFoundation\Response;
49use Symfony\Component\HttpFoundation\ResponseHeaderBag;
50use Symfony\Component\HttpKernel\Exception\NotFoundHttpException;
51
52/**
53 * Class ClippingsCartModule
54 */
55class ClippingsCartModule extends AbstractModule implements ModuleMenuInterface
56{
57    // Routes that have a record which can be added to the clipboard
58    private const ROUTES_WITH_RECORDS = [
59        'family',
60        'individual',
61        'media',
62        'note',
63        'repository',
64        'source',
65    ];
66
67    /** {@inheritdoc} */
68    public function getTitle(): string
69    {
70        /* I18N: Name of a module */
71        return I18N::translate('Clippings cart');
72    }
73
74    /** {@inheritdoc} */
75    public function getDescription(): string
76    {
77        /* I18N: Description of the “Clippings cart” module */
78        return I18N::translate('Select records from your family tree and save them as a GEDCOM file.');
79    }
80
81    /**
82     * What is the default access level for this module?
83     *
84     * Some modules are aimed at admins or managers, and are not generally shown to users.
85     *
86     * @return int
87     */
88    public function defaultAccessLevel(): int
89    {
90        return Auth::PRIV_USER;
91    }
92
93    /**
94     * The user can re-order menus. Until they do, they are shown in this order.
95     *
96     * @return int
97     */
98    public function defaultMenuOrder(): int
99    {
100        return 20;
101    }
102
103    /**
104     * A menu, to be added to the main application menu.
105     *
106     * @param Tree $tree
107     *
108     * @return Menu|null
109     */
110    public function getMenu(Tree $tree)
111    {
112        $request = Request::createFromGlobals();
113
114        $route = $request->get('route', '');
115
116        $submenus = [
117            new Menu($this->getTitle(), route('module', [
118                'module' => 'clippings',
119                'action' => 'Show',
120                'ged'    => $tree->name(),
121            ]), 'menu-clippings-cart', ['rel' => 'nofollow']),
122        ];
123
124        if (in_array($route, self::ROUTES_WITH_RECORDS)) {
125            $xref      = $request->get('xref', '');
126            $action    = 'Add' . ucfirst($route);
127            $add_route = route('module', [
128                'module' => 'clippings',
129                'action' => $action,
130                'xref'   => $xref,
131                'ged'    => $tree->name(),
132            ]);
133
134            $submenus[] = new Menu(I18N::translate('Add to the clippings cart'), $add_route, 'menu-clippings-add', ['rel' => 'nofollow']);
135        }
136
137        if (!$this->isCartEmpty($tree)) {
138            $submenus[] = new Menu(I18N::translate('Empty the clippings cart'), route('module', [
139                'module' => 'clippings',
140                'action' => 'Empty',
141                'ged'    => $tree->name(),
142            ]), 'menu-clippings-empty', ['rel' => 'nofollow']);
143            $submenus[] = new Menu(I18N::translate('Download'), route('module', [
144                'module' => 'clippings',
145                'action' => 'DownloadForm',
146                'ged'    => $tree->name(),
147            ]), 'menu-clippings-download', ['rel' => 'nofollow']);
148        }
149
150        return new Menu($this->getTitle(), '#', 'menu-clippings', ['rel' => 'nofollow'], $submenus);
151    }
152
153    /**
154     * @param Request $request
155     * @param Tree    $tree
156     *
157     * @return BinaryFileResponse
158     */
159    public function getDownloadAction(Request $request, Tree $tree): BinaryFileResponse
160    {
161        $this->checkModuleAccess($tree);
162
163        $privatize_export = $request->get('privatize_export', '');
164        $convert          = (bool) $request->get('convert');
165
166        $cart = Session::get('cart', []);
167
168        $xrefs = array_keys($cart[$tree->name()] ?? []);
169
170        // Create a new/empty .ZIP file
171        $temp_zip_file  = tempnam(sys_get_temp_dir(), 'webtrees-zip-');
172        $zip_filesystem = new Filesystem(new ZipArchiveAdapter($temp_zip_file));
173
174        // Media file prefix
175        $path = $tree->getPreference('MEDIA_DIRECTORY');
176
177        // GEDCOM file header
178        $filetext = FunctionsExport::gedcomHeader($tree, $convert ? 'ANSI' : 'UTF-8');
179
180        switch ($privatize_export) {
181            case 'gedadmin':
182                $access_level = Auth::PRIV_NONE;
183                break;
184            case 'user':
185                $access_level = Auth::PRIV_USER;
186                break;
187            case 'visitor':
188                $access_level = Auth::PRIV_PRIVATE;
189                break;
190            case 'none':
191            default:
192                $access_level = Auth::PRIV_HIDE;
193                break;
194        }
195
196        foreach ($xrefs as $xref) {
197            $object = GedcomRecord::getInstance($xref, $tree);
198            // The object may have been deleted since we added it to the cart....
199            if ($object) {
200                $record = $object->privatizeGedcom($access_level);
201                // Remove links to objects that aren't in the cart
202                preg_match_all('/\n1 ' . Gedcom::REGEX_TAG . ' @(' . Gedcom::REGEX_XREF . ')@(\n[2-9].*)*/', $record, $matches, PREG_SET_ORDER);
203                foreach ($matches as $match) {
204                    if (!array_key_exists($match[1], $xrefs)) {
205                        $record = str_replace($match[0], '', $record);
206                    }
207                }
208                preg_match_all('/\n2 ' . Gedcom::REGEX_TAG . ' @(' . Gedcom::REGEX_XREF . ')@(\n[3-9].*)*/', $record, $matches, PREG_SET_ORDER);
209                foreach ($matches as $match) {
210                    if (!array_key_exists($match[1], $xrefs)) {
211                        $record = str_replace($match[0], '', $record);
212                    }
213                }
214                preg_match_all('/\n3 ' . Gedcom::REGEX_TAG . ' @(' . Gedcom::REGEX_XREF . ')@(\n[4-9].*)*/', $record, $matches, PREG_SET_ORDER);
215                foreach ($matches as $match) {
216                    if (!array_key_exists($match[1], $xrefs)) {
217                        $record = str_replace($match[0], '', $record);
218                    }
219                }
220
221                if ($object instanceof Individual || $object instanceof Family) {
222                    $filetext .= $record . "\n";
223                    $filetext .= "1 SOUR @WEBTREES@\n";
224                    $filetext .= '2 PAGE ' . $object->url() . "\n";
225                } elseif ($object instanceof Source) {
226                    $filetext .= $record . "\n";
227                    $filetext .= '1 NOTE ' . $object->url() . "\n";
228                } elseif ($object instanceof Media) {
229                    // Add the media files to the archive
230                    foreach ($object->mediaFiles() as $media_file) {
231                        if (file_exists($media_file->getServerFilename())) {
232                            $fp = fopen($media_file->getServerFilename(), 'r');
233                            $zip_filesystem->writeStream($path . $media_file->filename(), $fp);
234                            fclose($fp);
235                        }
236                    }
237                    $filetext .= $record . "\n";
238                } else {
239                    $filetext .= $record . "\n";
240                }
241            }
242        }
243
244        // Create a source, to indicate the source of the data.
245        $filetext .= "0 @WEBTREES@ SOUR\n1 TITL " . WT_BASE_URL . "\n";
246        $author = User::find((int) $tree->getPreference('CONTACT_USER_ID'));
247        if ($author !== null) {
248            $filetext .= '1 AUTH ' . $author->getRealName() . "\n";
249        }
250        $filetext .= "0 TRLR\n";
251
252        // Make sure the preferred line endings are used
253        $filetext = str_replace('\n', Gedcom::EOL, $filetext);
254
255        if ($convert) {
256            $filetext = utf8_decode($filetext);
257        }
258
259        // Finally add the GEDCOM file to the .ZIP file.
260        $zip_filesystem->write('clippings.ged', $filetext);
261
262        // Need to force-close the filesystem
263        unset($zip_filesystem);
264
265        $response = new BinaryFileResponse($temp_zip_file);
266        $response->deleteFileAfterSend(true);
267
268        $response->headers->set('Content-Type', 'application/zip');
269        $response->setContentDisposition(
270            ResponseHeaderBag::DISPOSITION_ATTACHMENT,
271            'clippings.zip'
272        );
273
274        return $response;
275    }
276
277    /**
278     * @param Tree $tree
279     * @param User $user
280     *
281     * @return Response
282     */
283    public function getDownloadFormAction(Tree $tree, User $user): Response
284    {
285        $title = I18N::translate('Family tree clippings cart') . ' — ' . I18N::translate('Download');
286
287        return $this->viewResponse('modules/clippings/download', [
288            'is_manager' => Auth::isManager($tree, $user),
289            'is_member'  => Auth::isMember($tree, $user),
290            'title'      => $title,
291        ]);
292    }
293
294    /**
295     * @param Tree $tree
296     *
297     * @return RedirectResponse
298     */
299    public function getEmptyAction(Tree $tree): RedirectResponse
300    {
301        $cart                = Session::get('cart', []);
302        $cart[$tree->name()] = [];
303        Session::put('cart', $cart);
304
305        $url = route('module', [
306            'module' => 'clippings',
307            'action' => 'Show',
308            'ged'    => $tree->name(),
309        ]);
310
311        return new RedirectResponse($url);
312    }
313
314    /**
315     * @param Request $request
316     * @param Tree    $tree
317     *
318     * @return RedirectResponse
319     */
320    public function postRemoveAction(Request $request, Tree $tree): RedirectResponse
321    {
322        $xref = $request->get('xref', '');
323
324        $cart = Session::get('cart', []);
325        unset($cart[$tree->name()][$xref]);
326        Session::put('cart', $cart);
327
328        $url = route('module', [
329            'module' => 'clippings',
330            'action' => 'Show',
331            'ged'    => $tree->name(),
332        ]);
333
334        return new RedirectResponse($url);
335    }
336
337    /**
338     * @param Tree $tree
339     *
340     * @return Response
341     */
342    public function getShowAction(Tree $tree): Response
343    {
344        return $this->viewResponse('modules/clippings/show', [
345            'records' => $this->allRecordsInCart($tree),
346            'title'   => I18N::translate('Family tree clippings cart'),
347            'tree'    => $tree,
348        ]);
349    }
350
351    /**
352     * @param Request $request
353     * @param Tree    $tree
354     *
355     * @return Response
356     */
357    public function getAddFamilyAction(Request $request, Tree $tree): Response
358    {
359        $xref = $request->get('xref', '');
360
361        $family = Family::getInstance($xref, $tree);
362
363        if ($family === null) {
364            throw new FamilyNotFoundException();
365        }
366
367        $options = $this->familyOptions($family);
368
369        $title = I18N::translate('Add %s to the clippings cart', $family->getFullName());
370
371        return $this->viewResponse('modules/clippings/add-options', [
372            'options' => $options,
373            'default' => key($options),
374            'record'  => $family,
375            'title'   => $title,
376            'tree'    => $tree,
377        ]);
378    }
379
380    /**
381     * @param Family $family
382     *
383     * @return string[]
384     */
385    private function familyOptions(Family $family): array
386    {
387        $name = strip_tags($family->getFullName());
388
389        return [
390            'parents'     => $name,
391            /* I18N: %s is a family (husband + wife) */
392            'members'     => I18N::translate('%s and their children', $name),
393            /* I18N: %s is a family (husband + wife) */
394            'descendants' => I18N::translate('%s and their descendants', $name),
395        ];
396    }
397
398    /**
399     * @param Request $request
400     * @param Tree    $tree
401     *
402     * @return RedirectResponse
403     */
404    public function postAddFamilyAction(Request $request, Tree $tree): RedirectResponse
405    {
406        $xref   = $request->get('xref', '');
407        $option = $request->get('option', '');
408
409        $family = Family::getInstance($xref, $tree);
410
411        if ($family === null) {
412            throw new FamilyNotFoundException();
413        }
414
415        switch ($option) {
416            case 'parents':
417                $this->addFamilyToCart($family);
418                break;
419
420            case 'members':
421                $this->addFamilyAndChildrenToCart($family);
422                break;
423
424            case 'descendants':
425                $this->addFamilyAndDescendantsToCart($family);
426                break;
427        }
428
429        return new RedirectResponse($family->url());
430    }
431
432    /**
433     * @param Family $family
434     *
435     * @return void
436     */
437    private function addFamilyToCart(Family $family)
438    {
439        $this->addRecordToCart($family);
440
441        foreach ($family->getSpouses() as $spouse) {
442            $this->addRecordToCart($spouse);
443        }
444    }
445
446    /**
447     * @param Family $family
448     *
449     * @return void
450     */
451    private function addFamilyAndChildrenToCart(Family $family)
452    {
453        $this->addRecordToCart($family);
454
455        foreach ($family->getSpouses() as $spouse) {
456            $this->addRecordToCart($spouse);
457        }
458        foreach ($family->getChildren() as $child) {
459            $this->addRecordToCart($child);
460        }
461    }
462
463    /**
464     * @param Family $family
465     *
466     * @return void
467     */
468    private function addFamilyAndDescendantsToCart(Family $family)
469    {
470        $this->addRecordToCart($family);
471
472        foreach ($family->getSpouses() as $spouse) {
473            $this->addRecordToCart($spouse);
474        }
475        foreach ($family->getChildren() as $child) {
476            $this->addRecordToCart($child);
477            foreach ($child->getSpouseFamilies() as $child_family) {
478                $this->addFamilyAndDescendantsToCart($child_family);
479            }
480        }
481    }
482
483    /**
484     * @param Request $request
485     * @param Tree    $tree
486     *
487     * @return Response
488     */
489    public function getAddIndividualAction(Request $request, Tree $tree): Response
490    {
491        $xref = $request->get('xref', '');
492
493        $individual = Individual::getInstance($xref, $tree);
494
495        if ($individual === null) {
496            throw new IndividualNotFoundException();
497        }
498
499        $options = $this->individualOptions($individual);
500
501        $title = I18N::translate('Add %s to the clippings cart', $individual->getFullName());
502
503        return $this->viewResponse('modules/clippings/add-options', [
504            'options' => $options,
505            'default' => key($options),
506            'record'  => $individual,
507            'title'   => $title,
508            'tree'    => $tree,
509        ]);
510    }
511
512    /**
513     * @param Individual $individual
514     *
515     * @return string[]
516     */
517    private function individualOptions(Individual $individual): array
518    {
519        $name = strip_tags($individual->getFullName());
520
521        if ($individual->getSex() === 'F') {
522            return [
523                'self'              => $name,
524                'parents'           => I18N::translate('%s, her parents and siblings', $name),
525                'spouses'           => I18N::translate('%s, her spouses and children', $name),
526                'ancestors'         => I18N::translate('%s and her ancestors', $name),
527                'ancestor_families' => I18N::translate('%s, her ancestors and their families', $name),
528                'descendants'       => I18N::translate('%s, her spouses and descendants', $name),
529            ];
530        }
531
532        return [
533            'self'              => $name,
534            'parents'           => I18N::translate('%s, his parents and siblings', $name),
535            'spouses'           => I18N::translate('%s, his spouses and children', $name),
536            'ancestors'         => I18N::translate('%s and his ancestors', $name),
537            'ancestor_families' => I18N::translate('%s, his ancestors and their families', $name),
538            'descendants'       => I18N::translate('%s, his spouses and descendants', $name),
539        ];
540    }
541
542    /**
543     * @param Request $request
544     * @param Tree    $tree
545     *
546     * @return RedirectResponse
547     */
548    public function postAddIndividualAction(Request $request, Tree $tree): RedirectResponse
549    {
550        $xref   = $request->get('xref', '');
551        $option = $request->get('option', '');
552
553        $individual = Individual::getInstance($xref, $tree);
554
555        if ($individual === null) {
556            throw new IndividualNotFoundException();
557        }
558
559        switch ($option) {
560            case 'self':
561                $this->addRecordToCart($individual);
562                break;
563
564            case 'parents':
565                foreach ($individual->getChildFamilies() as $family) {
566                    $this->addFamilyAndChildrenToCart($family);
567                }
568                break;
569
570            case 'spouses':
571                foreach ($individual->getSpouseFamilies() as $family) {
572                    $this->addFamilyAndChildrenToCart($family);
573                }
574                break;
575
576            case 'ancestors':
577                $this->addAncestorsToCart($individual);
578                break;
579
580            case 'ancestor_families':
581                $this->addAncestorFamiliesToCart($individual);
582                break;
583
584            case 'descendants':
585                foreach ($individual->getSpouseFamilies() as $family) {
586                    $this->addFamilyAndDescendantsToCart($family);
587                }
588                break;
589        }
590
591        return new RedirectResponse($individual->url());
592    }
593
594    /**
595     * @param Individual $individual
596     *
597     * @return void
598     */
599    private function addAncestorsToCart(Individual $individual)
600    {
601        $this->addRecordToCart($individual);
602
603        foreach ($individual->getChildFamilies() as $family) {
604            foreach ($family->getSpouses() as $parent) {
605                $this->addAncestorsToCart($parent);
606            }
607        }
608    }
609
610    /**
611     * @param Individual $individual
612     *
613     * @return void
614     */
615    private function addAncestorFamiliesToCart(Individual $individual)
616    {
617        foreach ($individual->getChildFamilies() as $family) {
618            $this->addFamilyAndChildrenToCart($family);
619            foreach ($family->getSpouses() as $parent) {
620                $this->addAncestorsToCart($parent);
621            }
622        }
623    }
624
625    /**
626     * @param Request $request
627     * @param Tree    $tree
628     *
629     * @return Response
630     */
631    public function getAddMediaAction(Request $request, Tree $tree): Response
632    {
633        $xref = $request->get('xref', '');
634
635        $media = Media::getInstance($xref, $tree);
636
637        if ($media === null) {
638            throw new MediaNotFoundException();
639        }
640
641        $options = $this->mediaOptions($media);
642
643        $title = I18N::translate('Add %s to the clippings cart', $media->getFullName());
644
645        return $this->viewResponse('modules/clippings/add-options', [
646            'options' => $options,
647            'default' => key($options),
648            'record'  => $media,
649            'title'   => $title,
650            'tree'    => $tree,
651        ]);
652    }
653
654    /**
655     * @param Media $media
656     *
657     * @return string[]
658     */
659    private function mediaOptions(Media $media): array
660    {
661        $name = strip_tags($media->getFullName());
662
663        return [
664            'self' => $name,
665        ];
666    }
667
668    /**
669     * @param Request $request
670     * @param Tree    $tree
671     *
672     * @return RedirectResponse
673     */
674    public function postAddMediaAction(Request $request, Tree $tree): RedirectResponse
675    {
676        $xref = $request->get('xref', '');
677
678        $media = Media::getInstance($xref, $tree);
679
680        if ($media === null) {
681            throw new MediaNotFoundException();
682        }
683
684        $this->addRecordToCart($media);
685
686        return new RedirectResponse($media->url());
687    }
688
689    /**
690     * @param Request $request
691     * @param Tree    $tree
692     *
693     * @return Response
694     */
695    public function getAddNoteAction(Request $request, Tree $tree): Response
696    {
697        $xref = $request->get('xref', '');
698
699        $note = Note::getInstance($xref, $tree);
700
701        if ($note === null) {
702            throw new NoteNotFoundException();
703        }
704
705        $options = $this->noteOptions($note);
706
707        $title = I18N::translate('Add %s to the clippings cart', $note->getFullName());
708
709        return $this->viewResponse('modules/clippings/add-options', [
710            'options' => $options,
711            'default' => key($options),
712            'record'  => $note,
713            'title'   => $title,
714            'tree'    => $tree,
715        ]);
716    }
717
718    /**
719     * @param Note $note
720     *
721     * @return string[]
722     */
723    private function noteOptions(Note $note): array
724    {
725        $name = strip_tags($note->getFullName());
726
727        return [
728            'self' => $name,
729        ];
730    }
731
732    /**
733     * @param Request $request
734     * @param Tree    $tree
735     *
736     * @return RedirectResponse
737     */
738    public function postAddNoteAction(Request $request, Tree $tree): RedirectResponse
739    {
740        $xref = $request->get('xref', '');
741
742        $note = Note::getInstance($xref, $tree);
743
744        if ($note === null) {
745            throw new NoteNotFoundException();
746        }
747
748        $this->addRecordToCart($note);
749
750        return new RedirectResponse($note->url());
751    }
752
753    /**
754     * @param Request $request
755     * @param Tree    $tree
756     *
757     * @return Response
758     */
759    public function getAddRepositoryAction(Request $request, Tree $tree): Response
760    {
761        $xref = $request->get('xref', '');
762
763        $repository = Repository::getInstance($xref, $tree);
764
765        if ($repository === null) {
766            throw new RepositoryNotFoundException();
767        }
768
769        $options = $this->repositoryOptions($repository);
770
771        $title = I18N::translate('Add %s to the clippings cart', $repository->getFullName());
772
773        return $this->viewResponse('modules/clippings/add-options', [
774            'options' => $options,
775            'default' => key($options),
776            'record'  => $repository,
777            'title'   => $title,
778            'tree'    => $tree,
779        ]);
780    }
781
782    /**
783     * @param Repository $repository
784     *
785     * @return string[]
786     */
787    private function repositoryOptions(Repository $repository): array
788    {
789        $name = strip_tags($repository->getFullName());
790
791        return [
792            'self' => $name,
793        ];
794    }
795
796    /**
797     * @param Request $request
798     * @param Tree    $tree
799     *
800     * @return RedirectResponse
801     */
802    public function postAddRepositoryAction(Request $request, Tree $tree): RedirectResponse
803    {
804        $xref = $request->get('xref', '');
805
806        $repository = Repository::getInstance($xref, $tree);
807
808        if ($repository === null) {
809            throw new RepositoryNotFoundException();
810        }
811
812        $this->addRecordToCart($repository);
813
814        return new RedirectResponse($repository->url());
815    }
816
817    /**
818     * @param Request $request
819     * @param Tree    $tree
820     *
821     * @return Response
822     */
823    public function getAddSourceAction(Request $request, Tree $tree): Response
824    {
825        $xref = $request->get('xref', '');
826
827        $source = Source::getInstance($xref, $tree);
828
829        if ($source === null) {
830            throw new SourceNotFoundException();
831        }
832
833        $options = $this->sourceOptions($source);
834
835        $title = I18N::translate('Add %s to the clippings cart', $source->getFullName());
836
837        return $this->viewResponse('modules/clippings/add-options', [
838            'options' => $options,
839            'default' => key($options),
840            'record'  => $source,
841            'title'   => $title,
842            'tree'    => $tree,
843        ]);
844    }
845
846    /**
847     * @param Source $source
848     *
849     * @return string[]
850     */
851    private function sourceOptions(Source $source): array
852    {
853        $name = strip_tags($source->getFullName());
854
855        return [
856            'only'   => strip_tags($source->getFullName()),
857            'linked' => I18N::translate('%s and the individuals that reference it.', $name),
858        ];
859    }
860
861    /**
862     * @param Request $request
863     * @param Tree    $tree
864     *
865     * @return RedirectResponse
866     */
867    public function postAddSourceAction(Request $request, Tree $tree): RedirectResponse
868    {
869        $xref   = $request->get('xref', '');
870        $option = $request->get('option', '');
871
872        $source = Source::getInstance($xref, $tree);
873
874        if ($source === null) {
875            throw new SourceNotFoundException();
876        }
877
878        $this->addRecordToCart($source);
879
880        if ($option === 'linked') {
881            foreach ($source->linkedIndividuals('SOUR') as $individual) {
882                $this->addRecordToCart($individual);
883            }
884            foreach ($source->linkedFamilies('SOUR') as $family) {
885                $this->addRecordToCart($family);
886            }
887        }
888
889        return new RedirectResponse($source->url());
890    }
891
892    /**
893     * Get all the records in the cart.
894     *
895     * @param Tree $tree
896     *
897     * @return GedcomRecord[]
898     */
899    private function allRecordsInCart(Tree $tree): array
900    {
901        $cart = Session::get('cart', []);
902
903        $xrefs = array_keys($cart[$tree->name()] ?? []);
904
905        // Fetch all the records in the cart.
906        $records = array_map(function (string $xref) use ($tree): GedcomRecord {
907            return GedcomRecord::getInstance($xref, $tree);
908        }, $xrefs);
909
910        // Some records may have been deleted after they were added to the cart.
911        $records = array_filter($records);
912
913        // Group and sort.
914        uasort($records, function (GedcomRecord $x, GedcomRecord $y): int {
915            return $x::RECORD_TYPE <=> $y::RECORD_TYPE ?: GedcomRecord::nameComparator()($x, $y);
916        });
917
918        return $records;
919    }
920
921    /**
922     * Add a record (and direclty linked sources, notes, etc. to the cart.
923     *
924     * @param GedcomRecord $record
925     *
926     * @return void
927     */
928    private function addRecordToCart(GedcomRecord $record)
929    {
930        $cart = Session::get('cart', []);
931
932        $tree_name = $record->tree()->name();
933
934        // Add this record
935        $cart[$tree_name][$record->xref()] = true;
936
937        // Add directly linked media, notes, repositories and sources.
938        preg_match_all('/\n\d (?:OBJE|NOTE|SOUR|REPO) @(' . Gedcom::REGEX_XREF . ')@/', $record->gedcom(), $matches);
939
940        foreach ($matches[1] as $match) {
941            $cart[$tree_name][$match] = true;
942        }
943
944        Session::put('cart', $cart);
945    }
946
947    /**
948     * @param Tree $tree
949     *
950     * @return bool
951     */
952    private function isCartEmpty(Tree $tree): bool
953    {
954        $cart = Session::get('cart', []);
955
956        return empty($cart[$tree->name()]);
957    }
958
959    /**
960     * Only allow access to the routes/functions if the menu is active
961     *
962     * @param Tree $tree
963     *
964     * @return void
965     *
966     * @throws NoteNotFoundException
967     */
968    private function checkModuleAccess(Tree $tree)
969    {
970        if (!array_key_exists($this->getName(), Module::getActiveMenus($tree))) {
971            throw new NotFoundHttpException();
972        }
973    }
974}
975