xref: /webtrees/app/Statistics/Google/ChartDistribution.php (revision 202c018b592d5a516e4a465dc6dc515f3be37399)
1<?php
2
3/**
4 * webtrees: online genealogy
5 * Copyright (C) 2023 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\Statistics\Google;
21
22use Fisharebest\Webtrees\DB;
23use Fisharebest\Webtrees\I18N;
24use Fisharebest\Webtrees\Statistics\Repository\Interfaces\IndividualRepositoryInterface;
25use Fisharebest\Webtrees\Statistics\Service\CountryService;
26use Fisharebest\Webtrees\Tree;
27use Illuminate\Database\Query\Builder;
28use Illuminate\Database\Query\Expression;
29use Illuminate\Database\Query\JoinClause;
30
31use function preg_match;
32use function preg_quote;
33use function view;
34
35/**
36 * A chart showing the distribution of different events on a map.
37 */
38class ChartDistribution
39{
40    private Tree $tree;
41
42    private CountryService $country_service;
43
44    private IndividualRepositoryInterface $individual_repository;
45
46    /**
47     * @var array<string>
48     */
49    private array $country_to_iso3166;
50
51    /**
52     * @param Tree                          $tree
53     * @param CountryService                $country_service
54     * @param IndividualRepositoryInterface $individual_repository
55     */
56    public function __construct(
57        Tree $tree,
58        CountryService $country_service,
59        IndividualRepositoryInterface $individual_repository
60    ) {
61        $this->tree                  = $tree;
62        $this->country_service       = $country_service;
63        $this->individual_repository = $individual_repository;
64
65        // Get the country names for each language
66        $this->country_to_iso3166 = $this->getIso3166Countries();
67    }
68
69    /**
70     * Returns the country names for each language.
71     *
72     * @return array<string>
73     */
74    private function getIso3166Countries(): array
75    {
76        // Get the country names for each language
77        $country_to_iso3166 = [];
78
79        $current_language = I18N::languageTag();
80
81        foreach (I18N::activeLocales() as $locale) {
82            I18N::init($locale->languageTag());
83
84            $countries = $this->country_service->getAllCountries();
85
86            foreach ($this->country_service->iso3166() as $three => $two) {
87                $country_to_iso3166[$three]             = $two;
88                $country_to_iso3166[$countries[$three]] = $two;
89            }
90        }
91
92        I18N::init($current_language);
93
94        return $country_to_iso3166;
95    }
96
97    /**
98     * Returns the data structure required by google geochart.
99     *
100     * @param array<int> $places
101     *
102     * @return array<int,array<int|string|array<string,string>>>
103     */
104    private function createChartData(array $places): array
105    {
106        $data = [
107            [
108                I18N::translate('Country'),
109                I18N::translate('Total'),
110            ],
111        ];
112
113        // webtrees uses 3-letter country codes and localised country names, but google uses 2 letter codes.
114        foreach ($places as $country => $count) {
115            $data[] = [
116                [
117                    'v' => $country,
118                    'f' => $this->country_service->mapTwoLetterToName($country),
119                ],
120                $count
121            ];
122        }
123
124        return $data;
125    }
126
127    /**
128     * @param Tree   $tree
129     *
130     * @return array<int>
131     */
132    private function countIndividualsByCountry(Tree $tree): array
133    {
134        $rows = DB::table('places')
135            ->where('p_file', '=', $tree->id())
136            ->where('p_parent_id', '=', 0)
137            ->join('placelinks', static function (JoinClause $join): void {
138                $join
139                    ->on('pl_file', '=', 'p_file')
140                    ->on('pl_p_id', '=', 'p_id');
141            })
142            ->join('individuals', static function (JoinClause $join): void {
143                $join
144                    ->on('pl_file', '=', 'i_file')
145                    ->on('pl_gid', '=', 'i_id');
146            })
147            ->groupBy('p_place')
148            ->pluck(new Expression('COUNT(*) AS total'), 'p_place');
149
150        $totals = [];
151
152        foreach ($rows as $country => $count) {
153            $country_code = $this->country_to_iso3166[$country] ?? null;
154
155            if ($country_code !== null) {
156                $totals[$country_code] = $count + ($totals[$country_code] ?? 0);
157            }
158        }
159
160        return $totals;
161    }
162
163    /**
164     * @param Tree   $tree
165     * @param string $surname
166     *
167     * @return array<int>
168     */
169    private function countSurnamesByCountry(Tree $tree, string $surname): array
170    {
171        $rows =
172            DB::table('places')
173                ->where('p_file', '=', $tree->id())
174                ->where('p_parent_id', '=', 0)
175                ->join('placelinks', static function (JoinClause $join): void {
176                    $join
177                        ->on('pl_file', '=', 'p_file')
178                        ->on('pl_p_id', '=', 'p_id');
179                })
180                ->join('name', static function (JoinClause $join): void {
181                    $join
182                        ->on('n_file', '=', 'pl_file')
183                        ->on('n_id', '=', 'pl_gid');
184                })
185                ->where('n_surn', '=', $surname)
186                ->groupBy('p_place')
187                ->pluck(new Expression('COUNT(*) AS total'), 'p_place');
188
189        $totals = [];
190
191        foreach ($rows as $country => $count) {
192            $country_code = $this->country_to_iso3166[$country] ?? null;
193
194            if ($country_code !== null) {
195                $totals[$country_code] = $count + ($totals[$country_code] ?? 0);
196            }
197        }
198
199        return $totals;
200    }
201
202    /**
203     * @param Tree   $tree
204     * @param string $fact
205     *
206     * @return array<int>
207     */
208    private function countFamilyEventsByCountry(Tree $tree, string $fact): array
209    {
210        $query = DB::table('places')
211            ->where('p_file', '=', $tree->id())
212            ->where('p_parent_id', '=', 0)
213            ->join('placelinks', static function (JoinClause $join): void {
214                $join
215                    ->on('pl_file', '=', 'p_file')
216                    ->on('pl_p_id', '=', 'p_id');
217            })
218            ->join('families', static function (JoinClause $join): void {
219                $join
220                    ->on('pl_file', '=', 'f_file')
221                    ->on('pl_gid', '=', 'f_id');
222            })
223            ->select(['p_place AS place', 'f_gedcom AS gedcom']);
224
225        return $this->filterEventPlaces($query, $fact);
226    }
227
228    /**
229     * @param Tree   $tree
230     * @param string $fact
231     *
232     * @return array<int>
233     */
234    private function countIndividualEventsByCountry(Tree $tree, string $fact): array
235    {
236        $query = DB::table('places')
237            ->where('p_file', '=', $tree->id())
238            ->where('p_parent_id', '=', 0)
239            ->join('placelinks', static function (JoinClause $join): void {
240                $join
241                    ->on('pl_file', '=', 'p_file')
242                    ->on('pl_p_id', '=', 'p_id');
243            })
244            ->join('individuals', static function (JoinClause $join): void {
245                $join
246                    ->on('pl_file', '=', 'i_file')
247                    ->on('pl_gid', '=', 'i_id');
248            })
249            ->select(['p_place AS place', 'i_gedcom AS gedcom']);
250
251        return $this->filterEventPlaces($query, $fact);
252    }
253
254    /**
255     * @param Builder $query
256     * @param string  $fact
257     *
258     * @return array<int>
259     */
260    private function filterEventPlaces(Builder $query, string $fact): array
261    {
262        $totals = [];
263
264        foreach ($query->cursor() as $row) {
265            $country_code = $this->country_to_iso3166[$row->place] ?? null;
266
267            if ($country_code !== null) {
268                $place_regex = '/\n1 ' . $fact . '(?:\n[2-9].*)*\n2 PLAC.*[, ]' . preg_quote($row->place, '(?:\n|$)/i') . '\n/';
269
270                if (preg_match($place_regex, $row->gedcom) === 1) {
271                    $totals[$country_code] = 1 + ($totals[$country_code] ?? 0);
272                }
273            }
274        }
275
276        return $totals;
277    }
278
279    /**
280     * Create a chart showing where events occurred.
281     *
282     * @param string $chart_shows The type of chart map to show
283     * @param string $chart_type  The type of chart to show
284     * @param string $surname     The surname for surname based distribution chart
285     *
286     * @return string
287     */
288    public function chartDistribution(
289        string $chart_shows = 'world',
290        string $chart_type = '',
291        string $surname = ''
292    ): string {
293        switch ($chart_type) {
294            case 'surname_distribution_chart':
295                $chart_title = I18N::translate('Surname distribution chart') . ': ' . $surname;
296                $surname     = $surname ?: $this->individual_repository->getCommonSurname();
297                $data        = $this->createChartData($this->countSurnamesByCountry($this->tree, $surname));
298                break;
299
300            case 'birth_distribution_chart':
301                $chart_title = I18N::translate('Birth by country');
302                $data        = $this->createChartData($this->countIndividualEventsByCountry($this->tree, 'BIRT'));
303                break;
304
305            case 'death_distribution_chart':
306                $chart_title = I18N::translate('Death by country');
307                $data        = $this->createChartData($this->countIndividualEventsByCountry($this->tree, 'DEAT'));
308                break;
309
310            case 'marriage_distribution_chart':
311                $chart_title = I18N::translate('Marriage by country');
312                $data        = $this->createChartData($this->countFamilyEventsByCountry($this->tree, 'MARR'));
313                break;
314
315            case 'indi_distribution_chart':
316            default:
317                $chart_title = I18N::translate('Individual distribution chart');
318                $data        = $this->createChartData($this->countIndividualsByCountry($this->tree));
319                break;
320        }
321
322        return view('statistics/other/charts/geo', [
323            'chart_title'  => $chart_title,
324            'chart_color2' => '84beff',
325            'chart_color3' => 'c3dfff',
326            'region'       => $chart_shows,
327            'data'         => $data,
328            'language'     => I18N::languageTag(),
329        ]);
330    }
331}
332