xref: /webtrees/app/Module/FamilyTreeFavoritesModule.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\GedcomRecord;
22use Fisharebest\Webtrees\I18N;
23use Fisharebest\Webtrees\Tree;
24use Fisharebest\Webtrees\User;
25use Illuminate\Database\Capsule\Manager as DB;
26use stdClass;
27use Symfony\Component\HttpFoundation\RedirectResponse;
28use Symfony\Component\HttpFoundation\Request;
29
30/**
31 * Class FamilyTreeFavoritesModule
32 */
33class FamilyTreeFavoritesModule extends AbstractModule implements ModuleBlockInterface
34{
35    /**
36     * How should this module be labelled on tabs, menus, etc.?
37     *
38     * @return string
39     */
40    public function getTitle(): string
41    {
42        /* I18N: Name of a module */
43        return I18N::translate('Favorites');
44    }
45
46    /**
47     * A sentence describing what this module does.
48     *
49     * @return string
50     */
51    public function getDescription(): string
52    {
53        /* I18N: Description of the “Favorites” module */
54        return I18N::translate('Display and manage a family tree’s favorite pages.');
55    }
56
57    /**
58     * Generate the HTML content of this block.
59     *
60     * @param Tree     $tree
61     * @param int      $block_id
62     * @param string   $ctype
63     * @param string[] $cfg
64     *
65     * @return string
66     */
67    public function getBlock(Tree $tree, int $block_id, string $ctype = '', array $cfg = []): string
68    {
69        $content = view('modules/gedcom_favorites/favorites', [
70            'block_id'   => $block_id,
71            'favorites'  => $this->getFavorites($tree),
72            'is_manager' => Auth::isManager($tree),
73            'tree'       => $tree,
74        ]);
75
76        if ($ctype !== '') {
77            return view('modules/block-template', [
78                'block'      => str_replace('_', '-', $this->getName()),
79                'id'         => $block_id,
80                'config_url' => '',
81                'title'      => $this->getTitle(),
82                'content'    => $content,
83            ]);
84        }
85
86        return $content;
87    }
88
89    /**
90     * Should this block load asynchronously using AJAX?
91     *
92     * Simple blocks are faster in-line, more comples ones
93     * can be loaded later.
94     *
95     * @return bool
96     */
97    public function loadAjax(): bool
98    {
99        return false;
100    }
101
102    /**
103     * Can this block be shown on the user’s home page?
104     *
105     * @return bool
106     */
107    public function isUserBlock(): bool
108    {
109        return false;
110    }
111
112    /**
113     * Can this block be shown on the tree’s home page?
114     *
115     * @return bool
116     */
117    public function isGedcomBlock(): bool
118    {
119        return true;
120    }
121
122    /**
123     * Update the configuration for a block.
124     *
125     * @param Request $request
126     * @param int     $block_id
127     *
128     * @return void
129     */
130    public function saveBlockConfiguration(Request $request, int $block_id)
131    {
132    }
133
134    /**
135     * An HTML form to edit block settings
136     *
137     * @param Tree $tree
138     * @param int  $block_id
139     *
140     * @return void
141     */
142    public function editBlockConfiguration(Tree $tree, int $block_id)
143    {
144    }
145
146    /**
147     * Get favorites for a family tree
148     *
149     * @param Tree $tree
150     *
151     * @return stdClass[]
152     */
153    public function getFavorites(Tree $tree): array
154    {
155        return DB::table('favorite')
156            ->where('gedcom_id', '=', $tree->id())
157            ->whereNull('user_id')
158            ->get()
159            ->map(function (stdClass $row) use ($tree): stdClass {
160                if ($row->xref !== null) {
161                    $row->record = GedcomRecord::getInstance($row->xref, $tree);
162                } else {
163                    $row->record = null;
164                }
165
166                return $row;
167            })
168            ->all();
169    }
170
171    /**
172     * @param Request $request
173     * @param Tree    $tree
174     * @param User    $user
175     *
176     * @return RedirectResponse
177     */
178    public function postAddFavoriteAction(Request $request, Tree $tree, User $user): RedirectResponse
179    {
180        $note         = $request->get('note', '');
181        $title        = $request->get('title', '');
182        $url          = $request->get('url', '');
183        $xref         = $request->get('xref', '');
184        $fav_category = $request->get('fav_category', '');
185
186        $record = GedcomRecord::getInstance($xref, $tree);
187
188        if (Auth::isManager($tree, $user)) {
189            if ($fav_category === 'url' && $url !== '') {
190                $this->addUrlFavorite($tree, $url, $title ?: $url, $note);
191            }
192
193            if ($fav_category === 'record' && $record instanceof GedcomRecord && $record->canShow()) {
194                $this->addRecordFavorite($tree, $record, $note);
195            }
196        }
197
198        $url = route('tree-page', ['ged' => $tree->name()]);
199
200        return new RedirectResponse($url);
201    }
202
203    /**
204     * @param Request $request
205     * @param Tree    $tree
206     * @param User    $user
207     *
208     * @return RedirectResponse
209     */
210    public function postDeleteFavoriteAction(Request $request, Tree $tree, User $user): RedirectResponse
211    {
212        $favorite_id = (int) $request->get('favorite_id');
213
214        if (Auth::isManager($tree, $user)) {
215            DB::table('favorite')
216                ->where('favorite_id', '=', $favorite_id)
217                ->whereNull('user_id')
218                ->delete();
219        }
220
221        $url = route('tree-page', ['ged' => $tree->name()]);
222
223        return new RedirectResponse($url);
224    }
225
226    /**
227     * @param Tree   $tree
228     * @param string $url
229     * @param string $title
230     * @param string $note
231     *
232     * @return void
233     */
234    private function addUrlFavorite(Tree $tree, string $url, string $title, string $note)
235    {
236        DB::table('favorite')->updateOrInsert([
237            'gedcom_id' => $tree->id(),
238            'user_id'   => null,
239            'url'       => $url,
240        ], [
241            'favorite_type' => 'URL',
242            'note'          => $note,
243            'title'         => $title,
244        ]);
245    }
246
247    /**
248     * @param Tree         $tree
249     * @param GedcomRecord $record
250     * @param string       $note
251     *
252     * @return void
253     */
254    private function addRecordFavorite(Tree $tree, GedcomRecord $record, string $note)
255    {
256        DB::table('favorite')->updateOrInsert([
257            'gedcom_id' => $tree->id(),
258            'user_id'   => null,
259            'xref'      => $record->xref(),
260        ], [
261            'favorite_type' => $record::RECORD_TYPE,
262            'note'          => $note,
263        ]);
264    }
265}
266