xref: /webtrees/app/Statistics/Repository/IndividualRepository.php (revision e2cbf57a1b80fdeccb7ccc9b79b7aa02707dbde2)
18add1155SRico Sonntag<?php
28add1155SRico Sonntag/**
38add1155SRico Sonntag * webtrees: online genealogy
48add1155SRico Sonntag * Copyright (C) 2018 webtrees development team
58add1155SRico Sonntag * This program is free software: you can redistribute it and/or modify
68add1155SRico Sonntag * it under the terms of the GNU General Public License as published by
78add1155SRico Sonntag * the Free Software Foundation, either version 3 of the License, or
88add1155SRico Sonntag * (at your option) any later version.
98add1155SRico Sonntag * This program is distributed in the hope that it will be useful,
108add1155SRico Sonntag * but WITHOUT ANY WARRANTY; without even the implied warranty of
118add1155SRico Sonntag * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
128add1155SRico Sonntag * GNU General Public License for more details.
138add1155SRico Sonntag * You should have received a copy of the GNU General Public License
148add1155SRico Sonntag * along with this program. If not, see <http://www.gnu.org/licenses/>.
158add1155SRico Sonntag */
168add1155SRico Sonntagdeclare(strict_types=1);
178add1155SRico Sonntag
188add1155SRico Sonntagnamespace Fisharebest\Webtrees\Statistics\Repository;
198add1155SRico Sonntag
208add1155SRico Sonntaguse Fisharebest\Webtrees\Auth;
218add1155SRico Sonntaguse Fisharebest\Webtrees\Database;
228add1155SRico Sonntaguse Fisharebest\Webtrees\Functions\FunctionsDate;
238add1155SRico Sonntaguse Fisharebest\Webtrees\Functions\FunctionsPrintLists;
248add1155SRico Sonntaguse Fisharebest\Webtrees\Gedcom;
258add1155SRico Sonntaguse Fisharebest\Webtrees\I18N;
268add1155SRico Sonntaguse Fisharebest\Webtrees\Individual;
278add1155SRico Sonntaguse Fisharebest\Webtrees\Statistics\Google\ChartAge;
288add1155SRico Sonntaguse Fisharebest\Webtrees\Statistics\Google\ChartBirth;
298add1155SRico Sonntaguse Fisharebest\Webtrees\Statistics\Google\ChartCommonGiven;
308add1155SRico Sonntaguse Fisharebest\Webtrees\Statistics\Google\ChartCommonSurname;
318add1155SRico Sonntaguse Fisharebest\Webtrees\Statistics\Google\ChartDeath;
328add1155SRico Sonntaguse Fisharebest\Webtrees\Statistics\Google\ChartFamilyWithSources;
338add1155SRico Sonntaguse Fisharebest\Webtrees\Statistics\Google\ChartIndividual;
348add1155SRico Sonntaguse Fisharebest\Webtrees\Statistics\Google\ChartMortality;
358add1155SRico Sonntaguse Fisharebest\Webtrees\Statistics\Google\ChartSex;
368add1155SRico Sonntaguse Fisharebest\Webtrees\Statistics\Helper\Sql;
378add1155SRico Sonntaguse Fisharebest\Webtrees\Statistics\Repository\Interfaces\IndividualRepositoryInterface;
388add1155SRico Sonntaguse Fisharebest\Webtrees\Tree;
398add1155SRico Sonntaguse Illuminate\Database\Capsule\Manager as DB;
403dc8167dSGreg Roachuse Illuminate\Database\Query\Builder;
418add1155SRico Sonntaguse Illuminate\Database\Query\JoinClause;
428add1155SRico Sonntag
438add1155SRico Sonntag/**
448add1155SRico Sonntag *
458add1155SRico Sonntag */
468add1155SRico Sonntagclass IndividualRepository implements IndividualRepositoryInterface
478add1155SRico Sonntag{
488add1155SRico Sonntag    /**
498add1155SRico Sonntag     * @var Tree
508add1155SRico Sonntag     */
518add1155SRico Sonntag    private $tree;
528add1155SRico Sonntag
538add1155SRico Sonntag    /**
548add1155SRico Sonntag     * Constructor.
558add1155SRico Sonntag     *
568add1155SRico Sonntag     * @param Tree $tree
578add1155SRico Sonntag     */
588add1155SRico Sonntag    public function __construct(Tree $tree)
598add1155SRico Sonntag    {
608add1155SRico Sonntag        $this->tree = $tree;
618add1155SRico Sonntag    }
628add1155SRico Sonntag
638add1155SRico Sonntag    /**
648add1155SRico Sonntag     * Run an SQL query and cache the result.
658add1155SRico Sonntag     *
668add1155SRico Sonntag     * @param string $sql
678add1155SRico Sonntag     *
688add1155SRico Sonntag     * @return \stdClass[]
698add1155SRico Sonntag     */
708add1155SRico Sonntag    private function runSql(string $sql): array
718add1155SRico Sonntag    {
728add1155SRico Sonntag        return Sql::runSql($sql);
738add1155SRico Sonntag    }
748add1155SRico Sonntag
758add1155SRico Sonntag    /**
768add1155SRico Sonntag     * Find common given names.
778add1155SRico Sonntag     *
788add1155SRico Sonntag     * @param string $sex
798add1155SRico Sonntag     * @param string $type
808add1155SRico Sonntag     * @param bool   $show_tot
818add1155SRico Sonntag     * @param int    $threshold
828add1155SRico Sonntag     * @param int    $maxtoshow
838add1155SRico Sonntag     *
848add1155SRico Sonntag     * @return string|int[]
858add1155SRico Sonntag     */
868add1155SRico Sonntag    private function commonGivenQuery(string $sex, string $type, bool $show_tot, int $threshold, int $maxtoshow)
878add1155SRico Sonntag    {
888add1155SRico Sonntag        switch ($sex) {
898add1155SRico Sonntag            case 'M':
908add1155SRico Sonntag                $sex_sql = "i_sex='M'";
918add1155SRico Sonntag                break;
928add1155SRico Sonntag            case 'F':
938add1155SRico Sonntag                $sex_sql = "i_sex='F'";
948add1155SRico Sonntag                break;
958add1155SRico Sonntag            case 'U':
968add1155SRico Sonntag                $sex_sql = "i_sex='U'";
978add1155SRico Sonntag                break;
988add1155SRico Sonntag            case 'B':
998add1155SRico Sonntag            default:
1008add1155SRico Sonntag                $sex_sql = "i_sex<>'U'";
1018add1155SRico Sonntag                break;
1028add1155SRico Sonntag        }
1038add1155SRico Sonntag
1048add1155SRico Sonntag        $ged_id = $this->tree->id();
1058add1155SRico Sonntag
1068add1155SRico Sonntag        $rows     = Database::prepare("SELECT n_givn, COUNT(*) AS num FROM `##name` JOIN `##individuals` ON (n_id=i_id AND n_file=i_file) WHERE n_file={$ged_id} AND n_type<>'_MARNM' AND n_givn NOT IN ('@P.N.', '') AND LENGTH(n_givn)>1 AND {$sex_sql} GROUP BY n_id, n_givn")
1078add1155SRico Sonntag            ->fetchAll();
1088add1155SRico Sonntag
1098add1155SRico Sonntag        $nameList = [];
1108add1155SRico Sonntag        foreach ($rows as $row) {
1118add1155SRico Sonntag            $row->num = (int) $row->num;
1128add1155SRico Sonntag
1138add1155SRico Sonntag            // Split “John Thomas” into “John” and “Thomas” and count against both totals
1148add1155SRico Sonntag            foreach (explode(' ', $row->n_givn) as $given) {
1158add1155SRico Sonntag                // Exclude initials and particles.
1168add1155SRico Sonntag                if (!preg_match('/^([A-Z]|[a-z]{1,3})$/', $given)) {
1178add1155SRico Sonntag                    if (\array_key_exists($given, $nameList)) {
1188add1155SRico Sonntag                        $nameList[$given] += (int) $row->num;
1198add1155SRico Sonntag                    } else {
1208add1155SRico Sonntag                        $nameList[$given] = (int) $row->num;
1218add1155SRico Sonntag                    }
1228add1155SRico Sonntag                }
1238add1155SRico Sonntag            }
1248add1155SRico Sonntag        }
1258add1155SRico Sonntag        arsort($nameList);
1268add1155SRico Sonntag        $nameList = \array_slice($nameList, 0, $maxtoshow);
1278add1155SRico Sonntag
1288add1155SRico Sonntag        foreach ($nameList as $given => $total) {
1298add1155SRico Sonntag            if ($total < $threshold) {
1308add1155SRico Sonntag                unset($nameList[$given]);
1318add1155SRico Sonntag            }
1328add1155SRico Sonntag        }
1338add1155SRico Sonntag
1348add1155SRico Sonntag        switch ($type) {
1358add1155SRico Sonntag            case 'chart':
1368add1155SRico Sonntag                return $nameList;
1378add1155SRico Sonntag
1388add1155SRico Sonntag            case 'table':
1398add1155SRico Sonntag                return view('lists/given-names-table', [
1408add1155SRico Sonntag                    'given_names' => $nameList,
1418add1155SRico Sonntag                ]);
1428add1155SRico Sonntag
1438add1155SRico Sonntag            case 'list':
1448add1155SRico Sonntag                return view('lists/given-names-list', [
1458add1155SRico Sonntag                    'given_names' => $nameList,
1468add1155SRico Sonntag                    'show_totals' => $show_tot,
1478add1155SRico Sonntag                ]);
1488add1155SRico Sonntag
1498add1155SRico Sonntag            case 'nolist':
1508add1155SRico Sonntag            default:
1518add1155SRico Sonntag                array_walk($nameList, function (int &$value, string $key) use ($show_tot): void {
1528add1155SRico Sonntag                    if ($show_tot) {
1538add1155SRico Sonntag                        $value = '<span dir="auto">' . e($key);
1548add1155SRico Sonntag                    } else {
1558add1155SRico Sonntag                        $value = '<span dir="auto">' . e($key) . ' (' . I18N::number($value) . ')';
1568add1155SRico Sonntag                    }
1578add1155SRico Sonntag                });
1588add1155SRico Sonntag
1598add1155SRico Sonntag                return implode(I18N::$list_separator, $nameList);
1608add1155SRico Sonntag        }
1618add1155SRico Sonntag    }
1628add1155SRico Sonntag
1638add1155SRico Sonntag    /**
1648add1155SRico Sonntag     * Find common give names.
1658add1155SRico Sonntag     *
1668add1155SRico Sonntag     * @param int $threshold
1678add1155SRico Sonntag     * @param int $maxtoshow
1688add1155SRico Sonntag     *
1698add1155SRico Sonntag     * @return string
1708add1155SRico Sonntag     */
1718add1155SRico Sonntag    public function commonGiven(int $threshold = 1, int $maxtoshow = 10): string
1728add1155SRico Sonntag    {
1738add1155SRico Sonntag        return $this->commonGivenQuery('B', 'nolist', false, $threshold, $maxtoshow);
1748add1155SRico Sonntag    }
1758add1155SRico Sonntag
1768add1155SRico Sonntag    /**
1778add1155SRico Sonntag     * Find common give names.
1788add1155SRico Sonntag     *
1798add1155SRico Sonntag     * @param int $threshold
1808add1155SRico Sonntag     * @param int $maxtoshow
1818add1155SRico Sonntag     *
1828add1155SRico Sonntag     * @return string
1838add1155SRico Sonntag     */
1848add1155SRico Sonntag    public function commonGivenTotals(int $threshold = 1, int $maxtoshow = 10): string
1858add1155SRico Sonntag    {
1868add1155SRico Sonntag        return $this->commonGivenQuery('B', 'nolist', true, $threshold, $maxtoshow);
1878add1155SRico Sonntag    }
1888add1155SRico Sonntag
1898add1155SRico Sonntag    /**
1908add1155SRico Sonntag     * Find common give names.
1918add1155SRico Sonntag     *
1928add1155SRico Sonntag     * @param int $threshold
1938add1155SRico Sonntag     * @param int $maxtoshow
1948add1155SRico Sonntag     *
1958add1155SRico Sonntag     * @return string
1968add1155SRico Sonntag     */
1978add1155SRico Sonntag    public function commonGivenList(int $threshold = 1, int $maxtoshow = 10): string
1988add1155SRico Sonntag    {
1998add1155SRico Sonntag        return $this->commonGivenQuery('B', 'list', false, $threshold, $maxtoshow);
2008add1155SRico Sonntag    }
2018add1155SRico Sonntag
2028add1155SRico Sonntag    /**
2038add1155SRico Sonntag     * Find common give names.
2048add1155SRico Sonntag     *
2058add1155SRico Sonntag     * @param int $threshold
2068add1155SRico Sonntag     * @param int $maxtoshow
2078add1155SRico Sonntag     *
2088add1155SRico Sonntag     * @return string
2098add1155SRico Sonntag     */
2108add1155SRico Sonntag    public function commonGivenListTotals(int $threshold = 1, int $maxtoshow = 10): string
2118add1155SRico Sonntag    {
2128add1155SRico Sonntag        return $this->commonGivenQuery('B', 'list', true, $threshold, $maxtoshow);
2138add1155SRico Sonntag    }
2148add1155SRico Sonntag
2158add1155SRico Sonntag    /**
2168add1155SRico Sonntag     * Find common give names.
2178add1155SRico Sonntag     *
2188add1155SRico Sonntag     * @param int $threshold
2198add1155SRico Sonntag     * @param int $maxtoshow
2208add1155SRico Sonntag     *
2218add1155SRico Sonntag     * @return string
2228add1155SRico Sonntag     */
2238add1155SRico Sonntag    public function commonGivenTable(int $threshold = 1, int $maxtoshow = 10): string
2248add1155SRico Sonntag    {
2258add1155SRico Sonntag        return $this->commonGivenQuery('B', 'table', false, $threshold, $maxtoshow);
2268add1155SRico Sonntag    }
2278add1155SRico Sonntag
2288add1155SRico Sonntag    /**
2298add1155SRico Sonntag     * Find common give names of females.
2308add1155SRico Sonntag     *
2318add1155SRico Sonntag     * @param int $threshold
2328add1155SRico Sonntag     * @param int $maxtoshow
2338add1155SRico Sonntag     *
2348add1155SRico Sonntag     * @return string
2358add1155SRico Sonntag     */
2368add1155SRico Sonntag    public function commonGivenFemale(int $threshold = 1, int $maxtoshow = 10): string
2378add1155SRico Sonntag    {
2388add1155SRico Sonntag        return $this->commonGivenQuery('F', 'nolist', false, $threshold, $maxtoshow);
2398add1155SRico Sonntag    }
2408add1155SRico Sonntag
2418add1155SRico Sonntag    /**
2428add1155SRico Sonntag     * Find common give names of females.
2438add1155SRico Sonntag     *
2448add1155SRico Sonntag     * @param int $threshold
2458add1155SRico Sonntag     * @param int $maxtoshow
2468add1155SRico Sonntag     *
2478add1155SRico Sonntag     * @return string
2488add1155SRico Sonntag     */
2498add1155SRico Sonntag    public function commonGivenFemaleTotals(int $threshold = 1, int $maxtoshow = 10): string
2508add1155SRico Sonntag    {
2518add1155SRico Sonntag        return $this->commonGivenQuery('F', 'nolist', true, $threshold, $maxtoshow);
2528add1155SRico Sonntag    }
2538add1155SRico Sonntag
2548add1155SRico Sonntag    /**
2558add1155SRico Sonntag     * Find common give names of females.
2568add1155SRico Sonntag     *
2578add1155SRico Sonntag     * @param int $threshold
2588add1155SRico Sonntag     * @param int $maxtoshow
2598add1155SRico Sonntag     *
2608add1155SRico Sonntag     * @return string
2618add1155SRico Sonntag     */
2628add1155SRico Sonntag    public function commonGivenFemaleList(int $threshold = 1, int $maxtoshow = 10): string
2638add1155SRico Sonntag    {
2648add1155SRico Sonntag        return $this->commonGivenQuery('F', 'list', false, $threshold, $maxtoshow);
2658add1155SRico Sonntag    }
2668add1155SRico Sonntag
2678add1155SRico Sonntag    /**
2688add1155SRico Sonntag     * Find common give names of females.
2698add1155SRico Sonntag     *
2708add1155SRico Sonntag     * @param int $threshold
2718add1155SRico Sonntag     * @param int $maxtoshow
2728add1155SRico Sonntag     *
2738add1155SRico Sonntag     * @return string
2748add1155SRico Sonntag     */
2758add1155SRico Sonntag    public function commonGivenFemaleListTotals(int $threshold = 1, int $maxtoshow = 10): string
2768add1155SRico Sonntag    {
2778add1155SRico Sonntag        return $this->commonGivenQuery('F', 'list', true, $threshold, $maxtoshow);
2788add1155SRico Sonntag    }
2798add1155SRico Sonntag
2808add1155SRico Sonntag    /**
2818add1155SRico Sonntag     * Find common give names of females.
2828add1155SRico Sonntag     *
2838add1155SRico Sonntag     * @param int $threshold
2848add1155SRico Sonntag     * @param int $maxtoshow
2858add1155SRico Sonntag     *
2868add1155SRico Sonntag     * @return string
2878add1155SRico Sonntag     */
2888add1155SRico Sonntag    public function commonGivenFemaleTable(int $threshold = 1, int $maxtoshow = 10): string
2898add1155SRico Sonntag    {
2908add1155SRico Sonntag        return $this->commonGivenQuery('F', 'table', false, $threshold, $maxtoshow);
2918add1155SRico Sonntag    }
2928add1155SRico Sonntag
2938add1155SRico Sonntag    /**
2948add1155SRico Sonntag     * Find common give names of males.
2958add1155SRico Sonntag     *
2968add1155SRico Sonntag     * @param int $threshold
2978add1155SRico Sonntag     * @param int $maxtoshow
2988add1155SRico Sonntag     *
2998add1155SRico Sonntag     * @return string
3008add1155SRico Sonntag     */
3018add1155SRico Sonntag    public function commonGivenMale(int $threshold = 1, int $maxtoshow = 10): string
3028add1155SRico Sonntag    {
3038add1155SRico Sonntag        return $this->commonGivenQuery('M', 'nolist', false, $threshold, $maxtoshow);
3048add1155SRico Sonntag    }
3058add1155SRico Sonntag
3068add1155SRico Sonntag    /**
3078add1155SRico Sonntag     * Find common give names of males.
3088add1155SRico Sonntag     *
3098add1155SRico Sonntag     * @param int $threshold
3108add1155SRico Sonntag     * @param int $maxtoshow
3118add1155SRico Sonntag     *
3128add1155SRico Sonntag     * @return string
3138add1155SRico Sonntag     */
3148add1155SRico Sonntag    public function commonGivenMaleTotals(int $threshold = 1, int $maxtoshow = 10): string
3158add1155SRico Sonntag    {
3168add1155SRico Sonntag        return $this->commonGivenQuery('M', 'nolist', true, $threshold, $maxtoshow);
3178add1155SRico Sonntag    }
3188add1155SRico Sonntag
3198add1155SRico Sonntag    /**
3208add1155SRico Sonntag     * Find common give names of males.
3218add1155SRico Sonntag     *
3228add1155SRico Sonntag     * @param int $threshold
3238add1155SRico Sonntag     * @param int $maxtoshow
3248add1155SRico Sonntag     *
3258add1155SRico Sonntag     * @return string
3268add1155SRico Sonntag     */
3278add1155SRico Sonntag    public function commonGivenMaleList(int $threshold = 1, int $maxtoshow = 10): string
3288add1155SRico Sonntag    {
3298add1155SRico Sonntag        return $this->commonGivenQuery('M', 'list', false, $threshold, $maxtoshow);
3308add1155SRico Sonntag    }
3318add1155SRico Sonntag
3328add1155SRico Sonntag    /**
3338add1155SRico Sonntag     * Find common give names of males.
3348add1155SRico Sonntag     *
3358add1155SRico Sonntag     * @param int $threshold
3368add1155SRico Sonntag     * @param int $maxtoshow
3378add1155SRico Sonntag     *
3388add1155SRico Sonntag     * @return string
3398add1155SRico Sonntag     */
3408add1155SRico Sonntag    public function commonGivenMaleListTotals(int $threshold = 1, int $maxtoshow = 10): string
3418add1155SRico Sonntag    {
3428add1155SRico Sonntag        return $this->commonGivenQuery('M', 'list', true, $threshold, $maxtoshow);
3438add1155SRico Sonntag    }
3448add1155SRico Sonntag
3458add1155SRico Sonntag    /**
3468add1155SRico Sonntag     * Find common give names of males.
3478add1155SRico Sonntag     *
3488add1155SRico Sonntag     * @param int $threshold
3498add1155SRico Sonntag     * @param int $maxtoshow
3508add1155SRico Sonntag     *
3518add1155SRico Sonntag     * @return string
3528add1155SRico Sonntag     */
3538add1155SRico Sonntag    public function commonGivenMaleTable(int $threshold = 1, int $maxtoshow = 10): string
3548add1155SRico Sonntag    {
3558add1155SRico Sonntag        return $this->commonGivenQuery('M', 'table', false, $threshold, $maxtoshow);
3568add1155SRico Sonntag    }
3578add1155SRico Sonntag
3588add1155SRico Sonntag    /**
3598add1155SRico Sonntag     * Find common give names of unknown sexes.
3608add1155SRico Sonntag     *
3618add1155SRico Sonntag     * @param int $threshold
3628add1155SRico Sonntag     * @param int $maxtoshow
3638add1155SRico Sonntag     *
3648add1155SRico Sonntag     * @return string
3658add1155SRico Sonntag     */
3668add1155SRico Sonntag    public function commonGivenUnknown(int $threshold = 1, int $maxtoshow = 10): string
3678add1155SRico Sonntag    {
3688add1155SRico Sonntag        return $this->commonGivenQuery('U', 'nolist', false, $threshold, $maxtoshow);
3698add1155SRico Sonntag    }
3708add1155SRico Sonntag
3718add1155SRico Sonntag    /**
3728add1155SRico Sonntag     * Find common give names of unknown sexes.
3738add1155SRico Sonntag     *
3748add1155SRico Sonntag     * @param int $threshold
3758add1155SRico Sonntag     * @param int $maxtoshow
3768add1155SRico Sonntag     *
3778add1155SRico Sonntag     * @return string
3788add1155SRico Sonntag     */
3798add1155SRico Sonntag    public function commonGivenUnknownTotals(int $threshold = 1, int $maxtoshow = 10): string
3808add1155SRico Sonntag    {
3818add1155SRico Sonntag        return $this->commonGivenQuery('U', 'nolist', true, $threshold, $maxtoshow);
3828add1155SRico Sonntag    }
3838add1155SRico Sonntag
3848add1155SRico Sonntag    /**
3858add1155SRico Sonntag     * Find common give names of unknown sexes.
3868add1155SRico Sonntag     *
3878add1155SRico Sonntag     * @param int $threshold
3888add1155SRico Sonntag     * @param int $maxtoshow
3898add1155SRico Sonntag     *
3908add1155SRico Sonntag     * @return string
3918add1155SRico Sonntag     */
3928add1155SRico Sonntag    public function commonGivenUnknownList(int $threshold = 1, int $maxtoshow = 10): string
3938add1155SRico Sonntag    {
3948add1155SRico Sonntag        return $this->commonGivenQuery('U', 'list', false, $threshold, $maxtoshow);
3958add1155SRico Sonntag    }
3968add1155SRico Sonntag
3978add1155SRico Sonntag    /**
3988add1155SRico Sonntag     * Find common give names of unknown sexes.
3998add1155SRico Sonntag     *
4008add1155SRico Sonntag     * @param int $threshold
4018add1155SRico Sonntag     * @param int $maxtoshow
4028add1155SRico Sonntag     *
4038add1155SRico Sonntag     * @return string
4048add1155SRico Sonntag     */
4058add1155SRico Sonntag    public function commonGivenUnknownListTotals(int $threshold = 1, int $maxtoshow = 10): string
4068add1155SRico Sonntag    {
4078add1155SRico Sonntag        return $this->commonGivenQuery('U', 'list', true, $threshold, $maxtoshow);
4088add1155SRico Sonntag    }
4098add1155SRico Sonntag
4108add1155SRico Sonntag    /**
4118add1155SRico Sonntag     * Find common give names of unknown sexes.
4128add1155SRico Sonntag     *
4138add1155SRico Sonntag     * @param int $threshold
4148add1155SRico Sonntag     * @param int $maxtoshow
4158add1155SRico Sonntag     *
4168add1155SRico Sonntag     * @return string
4178add1155SRico Sonntag     */
4188add1155SRico Sonntag    public function commonGivenUnknownTable(int $threshold = 1, int $maxtoshow = 10): string
4198add1155SRico Sonntag    {
4208add1155SRico Sonntag        return $this->commonGivenQuery('U', 'table', false, $threshold, $maxtoshow);
4218add1155SRico Sonntag    }
4228add1155SRico Sonntag
4238add1155SRico Sonntag    /**
4243dc8167dSGreg Roach     * Count the number of distinct given names (or the number of occurences of specific given names).
4258add1155SRico Sonntag     *
4263dc8167dSGreg Roach     * @param string[] ...$params
4278add1155SRico Sonntag     *
4288add1155SRico Sonntag     * @return string
4298add1155SRico Sonntag     */
4308add1155SRico Sonntag    public function totalGivennames(...$params): string
4318add1155SRico Sonntag    {
4323dc8167dSGreg Roach        $query = DB::table('name')
4333dc8167dSGreg Roach            ->where('n_file', '=', $this->tree->id());
4343dc8167dSGreg Roach
4353dc8167dSGreg Roach        if (empty($params)) {
4363dc8167dSGreg Roach            // Count number of distinct surnames
4373dc8167dSGreg Roach            $query
43844cdc21eSGreg Roach                ->whereNotIn('n_givn', ['', '@N.N.'])
43944cdc21eSGreg Roach                ->groupBy('n_givn');
4408add1155SRico Sonntag        } else {
4413dc8167dSGreg Roach            // Count number of occurences of specific surnames.
44244cdc21eSGreg Roach            $query->whereIn('n_givn', $params);
4438add1155SRico Sonntag        }
4448add1155SRico Sonntag
4453dc8167dSGreg Roach        $count = $query->count();
4463dc8167dSGreg Roach
4473dc8167dSGreg Roach        return I18N::number($count);
4488add1155SRico Sonntag    }
4498add1155SRico Sonntag
4508add1155SRico Sonntag    /**
4513dc8167dSGreg Roach     * Count the number of distinct surnames (or the number of occurences of specific surnmaes).
4528add1155SRico Sonntag     *
4533dc8167dSGreg Roach     * @param string[] ...$params
4548add1155SRico Sonntag     *
4558add1155SRico Sonntag     * @return string
4568add1155SRico Sonntag     */
4578add1155SRico Sonntag    public function totalSurnames(...$params): string
4588add1155SRico Sonntag    {
4593dc8167dSGreg Roach        $query = DB::table('name')
4603dc8167dSGreg Roach            ->where('n_file', '=', $this->tree->id());
4613dc8167dSGreg Roach
4623dc8167dSGreg Roach        if (empty($params)) {
4633dc8167dSGreg Roach            // Count number of distinct surnames
4643dc8167dSGreg Roach            $query
4653dc8167dSGreg Roach                ->whereNotIn('n_surn', ['', '@N.N.'])
4663dc8167dSGreg Roach                ->groupBy('n_surn');
4678add1155SRico Sonntag        } else {
4683dc8167dSGreg Roach            // Count number of occurences of specific surnames.
4693dc8167dSGreg Roach            $query->whereIn('n_surn', $params);
4708add1155SRico Sonntag        }
4718add1155SRico Sonntag
4723dc8167dSGreg Roach        $count = $query->count();
4738add1155SRico Sonntag
4743dc8167dSGreg Roach        return I18N::number($count);
4758add1155SRico Sonntag    }
4768add1155SRico Sonntag
4778add1155SRico Sonntag    /**
4788add1155SRico Sonntag     * @param int $number_of_surnames
4798add1155SRico Sonntag     * @param int $threshold
4808add1155SRico Sonntag     *
4818add1155SRico Sonntag     * @return \stdClass[]
4828add1155SRico Sonntag     */
4838add1155SRico Sonntag    private function topSurnames(int $number_of_surnames, int $threshold): array
4848add1155SRico Sonntag    {
4858add1155SRico Sonntag        // Use the count of base surnames.
4868add1155SRico Sonntag        $top_surnames = Database::prepare(
4878add1155SRico Sonntag            "SELECT n_surn FROM `##name`" .
4888add1155SRico Sonntag            " WHERE n_file = :tree_id AND n_type != '_MARNM' AND n_surn NOT IN ('@N.N.', '')" .
4898add1155SRico Sonntag            " GROUP BY n_surn" .
4908add1155SRico Sonntag            " ORDER BY COUNT(n_surn) DESC" .
4918add1155SRico Sonntag            " LIMIT :limit"
4928add1155SRico Sonntag        )->execute([
4938add1155SRico Sonntag            'tree_id' => $this->tree->id(),
4948add1155SRico Sonntag            'limit'   => $number_of_surnames,
4958add1155SRico Sonntag        ])->fetchOneColumn();
4968add1155SRico Sonntag
4978add1155SRico Sonntag        $surnames = [];
4988add1155SRico Sonntag        foreach ($top_surnames as $top_surname) {
4998add1155SRico Sonntag            $variants = Database::prepare(
5008add1155SRico Sonntag                "SELECT n_surname COLLATE utf8_bin, COUNT(*) FROM `##name` WHERE n_file = :tree_id AND n_surn COLLATE :collate = :surname GROUP BY 1"
5018add1155SRico Sonntag            )->execute([
5028add1155SRico Sonntag                'collate' => I18N::collation(),
5038add1155SRico Sonntag                'surname' => $top_surname,
5048add1155SRico Sonntag                'tree_id' => $this->tree->id(),
5058add1155SRico Sonntag            ])->fetchAssoc();
5068add1155SRico Sonntag
5078add1155SRico Sonntag            if (array_sum($variants) > $threshold) {
5088add1155SRico Sonntag                $surnames[$top_surname] = $variants;
5098add1155SRico Sonntag            }
5108add1155SRico Sonntag        }
5118add1155SRico Sonntag
5128add1155SRico Sonntag        return $surnames;
5138add1155SRico Sonntag    }
5148add1155SRico Sonntag
5158add1155SRico Sonntag    /**
5168add1155SRico Sonntag     * Find common surnames.
5178add1155SRico Sonntag     *
5188add1155SRico Sonntag     * @return string
5198add1155SRico Sonntag     */
5208add1155SRico Sonntag    public function getCommonSurname(): string
5218add1155SRico Sonntag    {
5228add1155SRico Sonntag        $top_surname = $this->topSurnames(1, 0);
5238add1155SRico Sonntag        return implode(', ', array_keys(array_shift($top_surname)) ?? []);
5248add1155SRico Sonntag    }
5258add1155SRico Sonntag
5268add1155SRico Sonntag    /**
5278add1155SRico Sonntag     * Find common surnames.
5288add1155SRico Sonntag     *
5298add1155SRico Sonntag     * @param string $type
5308add1155SRico Sonntag     * @param bool   $show_tot
5318add1155SRico Sonntag     * @param int    $threshold
5328add1155SRico Sonntag     * @param int    $number_of_surnames
5338add1155SRico Sonntag     * @param string $sorting
5348add1155SRico Sonntag     *
5358add1155SRico Sonntag     * @return string
5368add1155SRico Sonntag     */
5378add1155SRico Sonntag    private function commonSurnamesQuery(
5388add1155SRico Sonntag        string $type,
5398add1155SRico Sonntag        bool $show_tot,
5408add1155SRico Sonntag        int $threshold,
5418add1155SRico Sonntag        int $number_of_surnames,
5428add1155SRico Sonntag        string $sorting
5438add1155SRico Sonntag    ): string {
5448add1155SRico Sonntag        $surnames = $this->topSurnames($number_of_surnames, $threshold);
5458add1155SRico Sonntag
5468add1155SRico Sonntag        switch ($sorting) {
5478add1155SRico Sonntag            default:
5488add1155SRico Sonntag            case 'alpha':
5498add1155SRico Sonntag                uksort($surnames, [I18N::class, 'strcasecmp']);
5508add1155SRico Sonntag                break;
5518add1155SRico Sonntag            case 'count':
5528add1155SRico Sonntag                break;
5538add1155SRico Sonntag            case 'rcount':
5548add1155SRico Sonntag                $surnames = array_reverse($surnames, true);
5558add1155SRico Sonntag                break;
5568add1155SRico Sonntag        }
5578add1155SRico Sonntag
5588add1155SRico Sonntag        return FunctionsPrintLists::surnameList(
5598add1155SRico Sonntag            $surnames,
5608add1155SRico Sonntag            ($type === 'list' ? 1 : 2),
5618add1155SRico Sonntag            $show_tot,
5628add1155SRico Sonntag            'individual-list',
5638add1155SRico Sonntag            $this->tree
5648add1155SRico Sonntag        );
5658add1155SRico Sonntag    }
5668add1155SRico Sonntag
5678add1155SRico Sonntag    /**
5688add1155SRico Sonntag     * Find common surnames.
5698add1155SRico Sonntag     *
5708add1155SRico Sonntag     * @param int    $threshold
5718add1155SRico Sonntag     * @param int    $number_of_surnames
5728add1155SRico Sonntag     * @param string $sorting
5738add1155SRico Sonntag     *
5748add1155SRico Sonntag     * @return string
5758add1155SRico Sonntag     */
5768add1155SRico Sonntag    public function commonSurnames(
5778add1155SRico Sonntag        int $threshold = 1,
5788add1155SRico Sonntag        int $number_of_surnames = 10,
5798add1155SRico Sonntag        string $sorting = 'alpha'
5808add1155SRico Sonntag    ): string {
5818add1155SRico Sonntag        return $this->commonSurnamesQuery('nolist', false, $threshold, $number_of_surnames, $sorting);
5828add1155SRico Sonntag    }
5838add1155SRico Sonntag
5848add1155SRico Sonntag    /**
5858add1155SRico Sonntag     * Find common surnames.
5868add1155SRico Sonntag     *
5878add1155SRico Sonntag     * @param int    $threshold
5888add1155SRico Sonntag     * @param int    $number_of_surnames
5898add1155SRico Sonntag     * @param string $sorting
5908add1155SRico Sonntag     *
5918add1155SRico Sonntag     * @return string
5928add1155SRico Sonntag     */
5938add1155SRico Sonntag    public function commonSurnamesTotals(
5948add1155SRico Sonntag        int $threshold = 1,
5958add1155SRico Sonntag        int $number_of_surnames = 10,
5968add1155SRico Sonntag        string $sorting = 'rcount'
5978add1155SRico Sonntag    ): string {
5988add1155SRico Sonntag        return $this->commonSurnamesQuery('nolist', true, $threshold, $number_of_surnames, $sorting);
5998add1155SRico Sonntag    }
6008add1155SRico Sonntag
6018add1155SRico Sonntag    /**
6028add1155SRico Sonntag     * Find common surnames.
6038add1155SRico Sonntag     *
6048add1155SRico Sonntag     * @param int    $threshold
6058add1155SRico Sonntag     * @param int    $number_of_surnames
6068add1155SRico Sonntag     * @param string $sorting
6078add1155SRico Sonntag     *
6088add1155SRico Sonntag     * @return string
6098add1155SRico Sonntag     */
6108add1155SRico Sonntag    public function commonSurnamesList(
6118add1155SRico Sonntag        int $threshold = 1,
6128add1155SRico Sonntag        int $number_of_surnames = 10,
6138add1155SRico Sonntag        string $sorting = 'alpha'
6148add1155SRico Sonntag    ): string {
6158add1155SRico Sonntag        return $this->commonSurnamesQuery('list', false, $threshold, $number_of_surnames, $sorting);
6168add1155SRico Sonntag    }
6178add1155SRico Sonntag
6188add1155SRico Sonntag    /**
6198add1155SRico Sonntag     * Find common surnames.
6208add1155SRico Sonntag     *
6218add1155SRico Sonntag     * @param int    $threshold
6228add1155SRico Sonntag     * @param int    $number_of_surnames
6238add1155SRico Sonntag     * @param string $sorting
6248add1155SRico Sonntag     *
6258add1155SRico Sonntag     * @return string
6268add1155SRico Sonntag     */
6278add1155SRico Sonntag    public function commonSurnamesListTotals(
6288add1155SRico Sonntag        int $threshold = 1,
6298add1155SRico Sonntag        int $number_of_surnames = 10,
6308add1155SRico Sonntag        string $sorting = 'rcount'
6318add1155SRico Sonntag    ): string {
6328add1155SRico Sonntag        return $this->commonSurnamesQuery('list', true, $threshold, $number_of_surnames, $sorting);
6338add1155SRico Sonntag    }
6348add1155SRico Sonntag
6358add1155SRico Sonntag    /**
6368add1155SRico Sonntag     * Get a list of birth dates.
6378add1155SRico Sonntag     *
6388add1155SRico Sonntag     * @param bool $sex
6398add1155SRico Sonntag     * @param int  $year1
6408add1155SRico Sonntag     * @param int  $year2
6418add1155SRico Sonntag     *
6428add1155SRico Sonntag     * @return array
6438add1155SRico Sonntag     */
6448add1155SRico Sonntag    public function statsBirthQuery(bool $sex = false, int $year1 = -1, int $year2 = -1): array
6458add1155SRico Sonntag    {
6468add1155SRico Sonntag        if ($sex) {
6478add1155SRico Sonntag            $sql =
6488add1155SRico Sonntag                "SELECT d_month, i_sex, COUNT(*) AS total FROM `##dates` " .
6498add1155SRico Sonntag                "JOIN `##individuals` ON d_file = i_file AND d_gid = i_id " .
6508add1155SRico Sonntag                "WHERE " .
6518add1155SRico Sonntag                "d_file={$this->tree->id()} AND " .
6528add1155SRico Sonntag                "d_fact='BIRT' AND " .
6538add1155SRico Sonntag                "d_type IN ('@#DGREGORIAN@', '@#DJULIAN@')";
6548add1155SRico Sonntag        } else {
6558add1155SRico Sonntag            $sql =
6568add1155SRico Sonntag                "SELECT d_month, COUNT(*) AS total FROM `##dates` " .
6578add1155SRico Sonntag                "WHERE " .
6588add1155SRico Sonntag                "d_file={$this->tree->id()} AND " .
6598add1155SRico Sonntag                "d_fact='BIRT' AND " .
6608add1155SRico Sonntag                "d_type IN ('@#DGREGORIAN@', '@#DJULIAN@')";
6618add1155SRico Sonntag        }
6628add1155SRico Sonntag
6638add1155SRico Sonntag        if ($year1 >= 0 && $year2 >= 0) {
6648add1155SRico Sonntag            $sql .= " AND d_year BETWEEN '{$year1}' AND '{$year2}'";
6658add1155SRico Sonntag        }
6668add1155SRico Sonntag
6678add1155SRico Sonntag        $sql .= " GROUP BY d_month";
6688add1155SRico Sonntag
6698add1155SRico Sonntag        if ($sex) {
6708add1155SRico Sonntag            $sql .= ", i_sex";
6718add1155SRico Sonntag        }
6728add1155SRico Sonntag
6738add1155SRico Sonntag        return $this->runSql($sql);
6748add1155SRico Sonntag    }
6758add1155SRico Sonntag
6768add1155SRico Sonntag    /**
6778add1155SRico Sonntag     * General query on births.
6788add1155SRico Sonntag     *
6798add1155SRico Sonntag     * @param string|null $size
6808add1155SRico Sonntag     * @param string|null $color_from
6818add1155SRico Sonntag     * @param string|null $color_to
6828add1155SRico Sonntag     *
6838add1155SRico Sonntag     * @return string
6848add1155SRico Sonntag     */
6858add1155SRico Sonntag    public function statsBirth(string $size = null, string $color_from = null, string $color_to = null): string
6868add1155SRico Sonntag    {
6878add1155SRico Sonntag        return (new ChartBirth($this->tree))
6888add1155SRico Sonntag            ->chartBirth($size, $color_from, $color_to);
6898add1155SRico Sonntag    }
6908add1155SRico Sonntag
6918add1155SRico Sonntag    /**
6928add1155SRico Sonntag     * Get a list of death dates.
6938add1155SRico Sonntag     *
6948add1155SRico Sonntag     * @param bool $sex
6958add1155SRico Sonntag     * @param int  $year1
6968add1155SRico Sonntag     * @param int  $year2
6978add1155SRico Sonntag     *
6988add1155SRico Sonntag     * @return array
6998add1155SRico Sonntag     */
7008add1155SRico Sonntag    public function statsDeathQuery(bool $sex = false, int $year1 = -1, int $year2 = -1): array
7018add1155SRico Sonntag    {
7028add1155SRico Sonntag        if ($sex) {
7038add1155SRico Sonntag            $sql =
7048add1155SRico Sonntag                "SELECT d_month, i_sex, COUNT(*) AS total FROM `##dates` " .
7058add1155SRico Sonntag                "JOIN `##individuals` ON d_file = i_file AND d_gid = i_id " .
7068add1155SRico Sonntag                "WHERE " .
7078add1155SRico Sonntag                "d_file={$this->tree->id()} AND " .
7088add1155SRico Sonntag                "d_fact='DEAT' AND " .
7098add1155SRico Sonntag                "d_type IN ('@#DGREGORIAN@', '@#DJULIAN@')";
7108add1155SRico Sonntag        } else {
7118add1155SRico Sonntag            $sql =
7128add1155SRico Sonntag                "SELECT d_month, COUNT(*) AS total FROM `##dates` " .
7138add1155SRico Sonntag                "WHERE " .
7148add1155SRico Sonntag                "d_file={$this->tree->id()} AND " .
7158add1155SRico Sonntag                "d_fact='DEAT' AND " .
7168add1155SRico Sonntag                "d_type IN ('@#DGREGORIAN@', '@#DJULIAN@')";
7178add1155SRico Sonntag        }
7188add1155SRico Sonntag
7198add1155SRico Sonntag        if ($year1 >= 0 && $year2 >= 0) {
7208add1155SRico Sonntag            $sql .= " AND d_year BETWEEN '{$year1}' AND '{$year2}'";
7218add1155SRico Sonntag        }
7228add1155SRico Sonntag
7238add1155SRico Sonntag        $sql .= " GROUP BY d_month";
7248add1155SRico Sonntag
7258add1155SRico Sonntag        if ($sex) {
7268add1155SRico Sonntag            $sql .= ", i_sex";
7278add1155SRico Sonntag        }
7288add1155SRico Sonntag
7298add1155SRico Sonntag        return $this->runSql($sql);
7308add1155SRico Sonntag    }
7318add1155SRico Sonntag
7328add1155SRico Sonntag    /**
7338add1155SRico Sonntag     * General query on deaths.
7348add1155SRico Sonntag     *
7358add1155SRico Sonntag     * @param string|null $size
7368add1155SRico Sonntag     * @param string|null $color_from
7378add1155SRico Sonntag     * @param string|null $color_to
7388add1155SRico Sonntag     *
7398add1155SRico Sonntag     * @return string
7408add1155SRico Sonntag     */
7418add1155SRico Sonntag    public function statsDeath(string $size = null, string $color_from = null, string $color_to = null): string
7428add1155SRico Sonntag    {
7438add1155SRico Sonntag        return (new ChartDeath($this->tree))
7448add1155SRico Sonntag            ->chartDeath($size, $color_from, $color_to);
7458add1155SRico Sonntag    }
7468add1155SRico Sonntag
7478add1155SRico Sonntag    /**
7488add1155SRico Sonntag     * General query on ages.
7498add1155SRico Sonntag     *
7508add1155SRico Sonntag     * @param string $related
7518add1155SRico Sonntag     * @param string $sex
7528add1155SRico Sonntag     * @param int    $year1
7538add1155SRico Sonntag     * @param int    $year2
7548add1155SRico Sonntag     *
7558add1155SRico Sonntag     * @return array|string
7568add1155SRico Sonntag     */
7578add1155SRico Sonntag    public function statsAgeQuery(string $related = 'BIRT', string $sex = 'BOTH', int $year1 = -1, int $year2 = -1)
7588add1155SRico Sonntag    {
75944cdc21eSGreg Roach        $prefix = DB::connection()->getTablePrefix();
7608add1155SRico Sonntag
76144cdc21eSGreg Roach        $query = $this->birthAndDeathQuery($sex);
7628add1155SRico Sonntag
7638add1155SRico Sonntag        if ($year1 >= 0 && $year2 >= 0) {
76444cdc21eSGreg Roach            $query
76544cdc21eSGreg Roach                ->whereIn('birth.d_type', ['@#DGREGORIAN@', '@#DJULIAN@'])
76644cdc21eSGreg Roach                ->whereIn('death.d_type', ['@#DGREGORIAN@', '@#DJULIAN@']);
76744cdc21eSGreg Roach
7688add1155SRico Sonntag            if ($related === 'BIRT') {
76944cdc21eSGreg Roach                $query->whereBetween('birth.d_year', [$year1, $year2]);
7708add1155SRico Sonntag            } elseif ($related === 'DEAT') {
77144cdc21eSGreg Roach                $query->whereBetween('death.d_year', [$year1, $year2]);
7728add1155SRico Sonntag            }
7738add1155SRico Sonntag        }
7748add1155SRico Sonntag
77544cdc21eSGreg Roach        return $query
77644cdc21eSGreg Roach            ->select(DB::raw($prefix . 'death.d_julianday2 - ' . $prefix . 'birth.d_julianday1 AS days'))
77744cdc21eSGreg Roach            ->orderby('days', 'desc')
77844cdc21eSGreg Roach            ->get()
77944cdc21eSGreg Roach            ->all();
7808add1155SRico Sonntag    }
7818add1155SRico Sonntag
7828add1155SRico Sonntag    /**
7838add1155SRico Sonntag     * General query on ages.
7848add1155SRico Sonntag     *
7858add1155SRico Sonntag     * @param string $size
7868add1155SRico Sonntag     *
7878add1155SRico Sonntag     * @return string
7888add1155SRico Sonntag     */
7898add1155SRico Sonntag    public function statsAge(string $size = '230x250'): string
7908add1155SRico Sonntag    {
7918add1155SRico Sonntag        return (new ChartAge($this->tree))->chartAge($size);
7928add1155SRico Sonntag    }
7938add1155SRico Sonntag
7948add1155SRico Sonntag    /**
7958add1155SRico Sonntag     * Lifespan
7968add1155SRico Sonntag     *
7978add1155SRico Sonntag     * @param string $type
7988add1155SRico Sonntag     * @param string $sex
7998add1155SRico Sonntag     *
8008add1155SRico Sonntag     * @return string
8018add1155SRico Sonntag     */
8028add1155SRico Sonntag    private function longlifeQuery(string $type, string $sex): string
8038add1155SRico Sonntag    {
80444cdc21eSGreg Roach        $prefix = DB::connection()->getTablePrefix();
8058add1155SRico Sonntag
80644cdc21eSGreg Roach        $row = $this->birthAndDeathQuery($sex)
80744cdc21eSGreg Roach            ->orderBy('days', 'desc')
80844cdc21eSGreg Roach            ->select(['individuals.*', DB::raw($prefix . 'death.d_julianday2 - ' . $prefix . 'birth.d_julianday1 AS days')])
80944cdc21eSGreg Roach            ->first();
81044cdc21eSGreg Roach
81144cdc21eSGreg Roach        if ($row === null) {
8128add1155SRico Sonntag            return '';
8138add1155SRico Sonntag        }
81444cdc21eSGreg Roach
81544cdc21eSGreg Roach        /** @var Individual $individual */
81644cdc21eSGreg Roach        $individual = Individual::rowMapper()($row);
81744cdc21eSGreg Roach
81844cdc21eSGreg Roach        if (!$individual->canShow()) {
81944cdc21eSGreg Roach            return I18N::translate('This information is private and cannot be shown.');
82044cdc21eSGreg Roach        }
82144cdc21eSGreg Roach
8228add1155SRico Sonntag        switch ($type) {
8238add1155SRico Sonntag            default:
8248add1155SRico Sonntag            case 'full':
82544cdc21eSGreg Roach                return $individual->formatList();
8268add1155SRico Sonntag
82744cdc21eSGreg Roach            case 'age':
82844cdc21eSGreg Roach                return I18N::number((int) ($row->days / 365.25));
82944cdc21eSGreg Roach
83044cdc21eSGreg Roach            case 'name':
83144cdc21eSGreg Roach                return '<a href="' . e($individual->url()) . '">' . $individual->getFullName() . '</a>';
83244cdc21eSGreg Roach        }
8338add1155SRico Sonntag    }
8348add1155SRico Sonntag
8358add1155SRico Sonntag    /**
8368add1155SRico Sonntag     * Find the longest lived individual.
8378add1155SRico Sonntag     *
8388add1155SRico Sonntag     * @return string
8398add1155SRico Sonntag     */
8408add1155SRico Sonntag    public function longestLife(): string
8418add1155SRico Sonntag    {
8428add1155SRico Sonntag        return $this->longlifeQuery('full', 'BOTH');
8438add1155SRico Sonntag    }
8448add1155SRico Sonntag
8458add1155SRico Sonntag    /**
8468add1155SRico Sonntag     * Find the age of the longest lived individual.
8478add1155SRico Sonntag     *
8488add1155SRico Sonntag     * @return string
8498add1155SRico Sonntag     */
8508add1155SRico Sonntag    public function longestLifeAge(): string
8518add1155SRico Sonntag    {
8528add1155SRico Sonntag        return $this->longlifeQuery('age', 'BOTH');
8538add1155SRico Sonntag    }
8548add1155SRico Sonntag
8558add1155SRico Sonntag    /**
8568add1155SRico Sonntag     * Find the name of the longest lived individual.
8578add1155SRico Sonntag     *
8588add1155SRico Sonntag     * @return string
8598add1155SRico Sonntag     */
8608add1155SRico Sonntag    public function longestLifeName(): string
8618add1155SRico Sonntag    {
8628add1155SRico Sonntag        return $this->longlifeQuery('name', 'BOTH');
8638add1155SRico Sonntag    }
8648add1155SRico Sonntag
8658add1155SRico Sonntag    /**
8668add1155SRico Sonntag     * Find the longest lived female.
8678add1155SRico Sonntag     *
8688add1155SRico Sonntag     * @return string
8698add1155SRico Sonntag     */
8708add1155SRico Sonntag    public function longestLifeFemale(): string
8718add1155SRico Sonntag    {
8728add1155SRico Sonntag        return $this->longlifeQuery('full', 'F');
8738add1155SRico Sonntag    }
8748add1155SRico Sonntag
8758add1155SRico Sonntag    /**
8768add1155SRico Sonntag     * Find the age of the longest lived female.
8778add1155SRico Sonntag     *
8788add1155SRico Sonntag     * @return string
8798add1155SRico Sonntag     */
8808add1155SRico Sonntag    public function longestLifeFemaleAge(): string
8818add1155SRico Sonntag    {
8828add1155SRico Sonntag        return $this->longlifeQuery('age', 'F');
8838add1155SRico Sonntag    }
8848add1155SRico Sonntag
8858add1155SRico Sonntag    /**
8868add1155SRico Sonntag     * Find the name of the longest lived female.
8878add1155SRico Sonntag     *
8888add1155SRico Sonntag     * @return string
8898add1155SRico Sonntag     */
8908add1155SRico Sonntag    public function longestLifeFemaleName(): string
8918add1155SRico Sonntag    {
8928add1155SRico Sonntag        return $this->longlifeQuery('name', 'F');
8938add1155SRico Sonntag    }
8948add1155SRico Sonntag
8958add1155SRico Sonntag    /**
8968add1155SRico Sonntag     * Find the longest lived male.
8978add1155SRico Sonntag     *
8988add1155SRico Sonntag     * @return string
8998add1155SRico Sonntag     */
9008add1155SRico Sonntag    public function longestLifeMale(): string
9018add1155SRico Sonntag    {
9028add1155SRico Sonntag        return $this->longlifeQuery('full', 'M');
9038add1155SRico Sonntag    }
9048add1155SRico Sonntag
9058add1155SRico Sonntag    /**
9068add1155SRico Sonntag     * Find the age of the longest lived male.
9078add1155SRico Sonntag     *
9088add1155SRico Sonntag     * @return string
9098add1155SRico Sonntag     */
9108add1155SRico Sonntag    public function longestLifeMaleAge(): string
9118add1155SRico Sonntag    {
9128add1155SRico Sonntag        return $this->longlifeQuery('age', 'M');
9138add1155SRico Sonntag    }
9148add1155SRico Sonntag
9158add1155SRico Sonntag    /**
9168add1155SRico Sonntag     * Find the name of the longest lived male.
9178add1155SRico Sonntag     *
9188add1155SRico Sonntag     * @return string
9198add1155SRico Sonntag     */
9208add1155SRico Sonntag    public function longestLifeMaleName(): string
9218add1155SRico Sonntag    {
9228add1155SRico Sonntag        return $this->longlifeQuery('name', 'M');
9238add1155SRico Sonntag    }
9248add1155SRico Sonntag
9258add1155SRico Sonntag    /**
9268add1155SRico Sonntag     * Returns the calculated age the time of event.
9278add1155SRico Sonntag     *
9288add1155SRico Sonntag     * @param int $age The age from the database record
9298add1155SRico Sonntag     *
9308add1155SRico Sonntag     * @return string
9318add1155SRico Sonntag     */
9328add1155SRico Sonntag    private function calculateAge(int $age): string
9338add1155SRico Sonntag    {
9348add1155SRico Sonntag        if ((int) ($age / 365.25) > 0) {
9358add1155SRico Sonntag            $result = (int) ($age / 365.25) . 'y';
9368add1155SRico Sonntag        } elseif ((int) ($age / 30.4375) > 0) {
9378add1155SRico Sonntag            $result = (int) ($age / 30.4375) . 'm';
9388add1155SRico Sonntag        } else {
9398add1155SRico Sonntag            $result = $age . 'd';
9408add1155SRico Sonntag        }
9418add1155SRico Sonntag
9428add1155SRico Sonntag        return FunctionsDate::getAgeAtEvent($result);
9438add1155SRico Sonntag    }
9448add1155SRico Sonntag
9458add1155SRico Sonntag    /**
9468add1155SRico Sonntag     * Find the oldest individuals.
9478add1155SRico Sonntag     *
9488add1155SRico Sonntag     * @param string $sex
9498add1155SRico Sonntag     * @param int    $total
9508add1155SRico Sonntag     *
9518add1155SRico Sonntag     * @return array
9528add1155SRico Sonntag     */
9538add1155SRico Sonntag    private function topTenOldestQuery(string $sex, int $total): array
9548add1155SRico Sonntag    {
95544cdc21eSGreg Roach        $prefix = DB::connection()->getTablePrefix();
9568add1155SRico Sonntag
95744cdc21eSGreg Roach        $rows = $this->birthAndDeathQuery($sex)
95844cdc21eSGreg Roach            ->groupBy(['i_id', 'i_file'])
95944cdc21eSGreg Roach            ->orderBy('days', 'desc')
96044cdc21eSGreg Roach            ->select(['individuals.*', DB::raw('MAX(' . $prefix . 'death.d_julianday2 - ' . $prefix . 'birth.d_julianday1) AS days')])
96144cdc21eSGreg Roach            ->take($total)
96244cdc21eSGreg Roach            ->get();
9638add1155SRico Sonntag
9648add1155SRico Sonntag        $top10 = [];
9658add1155SRico Sonntag        foreach ($rows as $row) {
96644cdc21eSGreg Roach            /** @var Individual $individual */
96744cdc21eSGreg Roach            $individual = Individual::rowMapper()($row);
9688add1155SRico Sonntag
96944cdc21eSGreg Roach            if ($individual->canShow()) {
9708add1155SRico Sonntag                $top10[] = [
97144cdc21eSGreg Roach                    'person' => $individual,
97244cdc21eSGreg Roach                    'age'    => $this->calculateAge((int) $row->days),
9738add1155SRico Sonntag                ];
9748add1155SRico Sonntag            }
9758add1155SRico Sonntag        }
9768add1155SRico Sonntag
9778add1155SRico Sonntag        return $top10;
9788add1155SRico Sonntag    }
9798add1155SRico Sonntag
9808add1155SRico Sonntag    /**
9818add1155SRico Sonntag     * Find the oldest individuals.
9828add1155SRico Sonntag     *
9838add1155SRico Sonntag     * @param int $total
9848add1155SRico Sonntag     *
9858add1155SRico Sonntag     * @return string
9868add1155SRico Sonntag     */
9878add1155SRico Sonntag    public function topTenOldest(int $total = 10): string
9888add1155SRico Sonntag    {
9898add1155SRico Sonntag        $records = $this->topTenOldestQuery('BOTH', $total);
9908add1155SRico Sonntag
9918add1155SRico Sonntag        return view(
9928add1155SRico Sonntag            'statistics/individuals/top10-nolist',
9938add1155SRico Sonntag            [
9948add1155SRico Sonntag                'records' => $records,
9958add1155SRico Sonntag            ]
9968add1155SRico Sonntag        );
9978add1155SRico Sonntag    }
9988add1155SRico Sonntag
9998add1155SRico Sonntag    /**
10008add1155SRico Sonntag     * Find the oldest living individuals.
10018add1155SRico Sonntag     *
10028add1155SRico Sonntag     * @param int $total
10038add1155SRico Sonntag     *
10048add1155SRico Sonntag     * @return string
10058add1155SRico Sonntag     */
10068add1155SRico Sonntag    public function topTenOldestList(int $total = 10): string
10078add1155SRico Sonntag    {
10088add1155SRico Sonntag        $records = $this->topTenOldestQuery('BOTH', $total);
10098add1155SRico Sonntag
10108add1155SRico Sonntag        return view(
10118add1155SRico Sonntag            'statistics/individuals/top10-list',
10128add1155SRico Sonntag            [
10138add1155SRico Sonntag                'records' => $records,
10148add1155SRico Sonntag            ]
10158add1155SRico Sonntag        );
10168add1155SRico Sonntag    }
10178add1155SRico Sonntag
10188add1155SRico Sonntag    /**
10198add1155SRico Sonntag     * Find the oldest females.
10208add1155SRico Sonntag     *
10218add1155SRico Sonntag     * @param int $total
10228add1155SRico Sonntag     *
10238add1155SRico Sonntag     * @return string
10248add1155SRico Sonntag     */
10258add1155SRico Sonntag    public function topTenOldestFemale(int $total = 10): string
10268add1155SRico Sonntag    {
10278add1155SRico Sonntag        $records = $this->topTenOldestQuery('F', $total);
10288add1155SRico Sonntag
10298add1155SRico Sonntag        return view(
10308add1155SRico Sonntag            'statistics/individuals/top10-nolist',
10318add1155SRico Sonntag            [
10328add1155SRico Sonntag                'records' => $records,
10338add1155SRico Sonntag            ]
10348add1155SRico Sonntag        );
10358add1155SRico Sonntag    }
10368add1155SRico Sonntag
10378add1155SRico Sonntag    /**
10388add1155SRico Sonntag     * Find the oldest living females.
10398add1155SRico Sonntag     *
10408add1155SRico Sonntag     * @param int $total
10418add1155SRico Sonntag     *
10428add1155SRico Sonntag     * @return string
10438add1155SRico Sonntag     */
10448add1155SRico Sonntag    public function topTenOldestFemaleList(int $total = 10): string
10458add1155SRico Sonntag    {
10468add1155SRico Sonntag        $records = $this->topTenOldestQuery('F', $total);
10478add1155SRico Sonntag
10488add1155SRico Sonntag        return view(
10498add1155SRico Sonntag            'statistics/individuals/top10-list',
10508add1155SRico Sonntag            [
10518add1155SRico Sonntag                'records' => $records,
10528add1155SRico Sonntag            ]
10538add1155SRico Sonntag        );
10548add1155SRico Sonntag    }
10558add1155SRico Sonntag
10568add1155SRico Sonntag    /**
10578add1155SRico Sonntag     * Find the longest lived males.
10588add1155SRico Sonntag     *
10598add1155SRico Sonntag     * @param int $total
10608add1155SRico Sonntag     *
10618add1155SRico Sonntag     * @return string
10628add1155SRico Sonntag     */
10638add1155SRico Sonntag    public function topTenOldestMale(int $total = 10): string
10648add1155SRico Sonntag    {
10658add1155SRico Sonntag        $records = $this->topTenOldestQuery('M', $total);
10668add1155SRico Sonntag
10678add1155SRico Sonntag        return view(
10688add1155SRico Sonntag            'statistics/individuals/top10-nolist',
10698add1155SRico Sonntag            [
10708add1155SRico Sonntag                'records' => $records,
10718add1155SRico Sonntag            ]
10728add1155SRico Sonntag        );
10738add1155SRico Sonntag    }
10748add1155SRico Sonntag
10758add1155SRico Sonntag    /**
10768add1155SRico Sonntag     * Find the longest lived males.
10778add1155SRico Sonntag     *
10788add1155SRico Sonntag     * @param int $total
10798add1155SRico Sonntag     *
10808add1155SRico Sonntag     * @return string
10818add1155SRico Sonntag     */
10828add1155SRico Sonntag    public function topTenOldestMaleList(int $total = 10): string
10838add1155SRico Sonntag    {
10848add1155SRico Sonntag        $records = $this->topTenOldestQuery('M', $total);
10858add1155SRico Sonntag
10868add1155SRico Sonntag        return view(
10878add1155SRico Sonntag            'statistics/individuals/top10-list',
10888add1155SRico Sonntag            [
10898add1155SRico Sonntag                'records' => $records,
10908add1155SRico Sonntag            ]
10918add1155SRico Sonntag        );
10928add1155SRico Sonntag    }
10938add1155SRico Sonntag
10948add1155SRico Sonntag    /**
10958add1155SRico Sonntag     * Find the oldest living individuals.
10968add1155SRico Sonntag     *
10978add1155SRico Sonntag     * @param string $sex
10988add1155SRico Sonntag     * @param int    $total
10998add1155SRico Sonntag     *
11008add1155SRico Sonntag     * @return array
11018add1155SRico Sonntag     */
11028add1155SRico Sonntag    private function topTenOldestAliveQuery(string $sex = 'BOTH', int $total = 10): array
11038add1155SRico Sonntag    {
11048add1155SRico Sonntag        if ($sex === 'F') {
11058add1155SRico Sonntag            $sex_search = " AND i_sex='F'";
11068add1155SRico Sonntag        } elseif ($sex === 'M') {
11078add1155SRico Sonntag            $sex_search = " AND i_sex='M'";
11088add1155SRico Sonntag        } else {
11098add1155SRico Sonntag            $sex_search = '';
11108add1155SRico Sonntag        }
11118add1155SRico Sonntag
11128add1155SRico Sonntag        $rows = $this->runSql(
11138add1155SRico Sonntag            "SELECT" .
11148add1155SRico Sonntag            " birth.d_gid AS id," .
11158add1155SRico Sonntag            " MIN(birth.d_julianday1) AS age" .
11168add1155SRico Sonntag            " FROM" .
11178add1155SRico Sonntag            " `##dates` AS birth," .
11188add1155SRico Sonntag            " `##individuals` AS indi" .
11198add1155SRico Sonntag            " WHERE" .
11208add1155SRico Sonntag            " indi.i_id=birth.d_gid AND" .
11218add1155SRico Sonntag            " indi.i_gedcom NOT REGEXP '\\n1 (" . implode('|', Gedcom::DEATH_EVENTS) . ")' AND" .
11228add1155SRico Sonntag            " birth.d_file={$this->tree->id()} AND" .
11238add1155SRico Sonntag            " birth.d_fact='BIRT' AND" .
11248add1155SRico Sonntag            " birth.d_file=indi.i_file AND" .
11258add1155SRico Sonntag            " birth.d_julianday1<>0" .
11268add1155SRico Sonntag            $sex_search .
11278add1155SRico Sonntag            " GROUP BY id" .
11288add1155SRico Sonntag            " ORDER BY age" .
11298add1155SRico Sonntag            " ASC LIMIT " . $total
11308add1155SRico Sonntag        );
11318add1155SRico Sonntag
11328add1155SRico Sonntag        $top10 = [];
11338add1155SRico Sonntag
11348add1155SRico Sonntag        foreach ($rows as $row) {
11358add1155SRico Sonntag            $person = Individual::getInstance($row->id, $this->tree);
11368add1155SRico Sonntag
11378add1155SRico Sonntag            $top10[] = [
11388add1155SRico Sonntag                'person' => $person,
11398add1155SRico Sonntag                'age'    => $this->calculateAge(WT_CLIENT_JD - ((int) $row->age)),
11408add1155SRico Sonntag            ];
11418add1155SRico Sonntag        }
11428add1155SRico Sonntag
11438add1155SRico Sonntag        return $top10;
11448add1155SRico Sonntag    }
11458add1155SRico Sonntag
11468add1155SRico Sonntag    /**
11478add1155SRico Sonntag     * Find the oldest living individuals.
11488add1155SRico Sonntag     *
11498add1155SRico Sonntag     * @param int $total
11508add1155SRico Sonntag     *
11518add1155SRico Sonntag     * @return string
11528add1155SRico Sonntag     */
11538add1155SRico Sonntag    public function topTenOldestAlive(int $total = 10): string
11548add1155SRico Sonntag    {
11558add1155SRico Sonntag        if (!Auth::isMember($this->tree)) {
11568add1155SRico Sonntag            return I18N::translate('This information is private and cannot be shown.');
11578add1155SRico Sonntag        }
11588add1155SRico Sonntag
11598add1155SRico Sonntag        $records = $this->topTenOldestAliveQuery('BOTH', $total);
11608add1155SRico Sonntag
11618add1155SRico Sonntag        return view(
11628add1155SRico Sonntag            'statistics/individuals/top10-nolist',
11638add1155SRico Sonntag            [
11648add1155SRico Sonntag                'records' => $records,
11658add1155SRico Sonntag            ]
11668add1155SRico Sonntag        );
11678add1155SRico Sonntag    }
11688add1155SRico Sonntag
11698add1155SRico Sonntag    /**
11708add1155SRico Sonntag     * Find the oldest living individuals.
11718add1155SRico Sonntag     *
11728add1155SRico Sonntag     * @param int $total
11738add1155SRico Sonntag     *
11748add1155SRico Sonntag     * @return string
11758add1155SRico Sonntag     */
11768add1155SRico Sonntag    public function topTenOldestListAlive(int $total = 10): string
11778add1155SRico Sonntag    {
11788add1155SRico Sonntag        if (!Auth::isMember($this->tree)) {
11798add1155SRico Sonntag            return I18N::translate('This information is private and cannot be shown.');
11808add1155SRico Sonntag        }
11818add1155SRico Sonntag
11828add1155SRico Sonntag        $records = $this->topTenOldestAliveQuery('BOTH', $total);
11838add1155SRico Sonntag
11848add1155SRico Sonntag        return view(
11858add1155SRico Sonntag            'statistics/individuals/top10-list',
11868add1155SRico Sonntag            [
11878add1155SRico Sonntag                'records' => $records,
11888add1155SRico Sonntag            ]
11898add1155SRico Sonntag        );
11908add1155SRico Sonntag    }
11918add1155SRico Sonntag
11928add1155SRico Sonntag    /**
11938add1155SRico Sonntag     * Find the oldest living females.
11948add1155SRico Sonntag     *
11958add1155SRico Sonntag     * @param int $total
11968add1155SRico Sonntag     *
11978add1155SRico Sonntag     * @return string
11988add1155SRico Sonntag     */
11998add1155SRico Sonntag    public function topTenOldestFemaleAlive(int $total = 10): string
12008add1155SRico Sonntag    {
12018add1155SRico Sonntag        if (!Auth::isMember($this->tree)) {
12028add1155SRico Sonntag            return I18N::translate('This information is private and cannot be shown.');
12038add1155SRico Sonntag        }
12048add1155SRico Sonntag
12058add1155SRico Sonntag        $records = $this->topTenOldestAliveQuery('F', $total);
12068add1155SRico Sonntag
12078add1155SRico Sonntag        return view(
12088add1155SRico Sonntag            'statistics/individuals/top10-nolist',
12098add1155SRico Sonntag            [
12108add1155SRico Sonntag                'records' => $records,
12118add1155SRico Sonntag            ]
12128add1155SRico Sonntag        );
12138add1155SRico Sonntag    }
12148add1155SRico Sonntag
12158add1155SRico Sonntag    /**
12168add1155SRico Sonntag     * Find the oldest living females.
12178add1155SRico Sonntag     *
12188add1155SRico Sonntag     * @param int $total
12198add1155SRico Sonntag     *
12208add1155SRico Sonntag     * @return string
12218add1155SRico Sonntag     */
12228add1155SRico Sonntag    public function topTenOldestFemaleListAlive(int $total = 10): string
12238add1155SRico Sonntag    {
12248add1155SRico Sonntag        if (!Auth::isMember($this->tree)) {
12258add1155SRico Sonntag            return I18N::translate('This information is private and cannot be shown.');
12268add1155SRico Sonntag        }
12278add1155SRico Sonntag
12288add1155SRico Sonntag        $records = $this->topTenOldestAliveQuery('F', $total);
12298add1155SRico Sonntag
12308add1155SRico Sonntag        return view(
12318add1155SRico Sonntag            'statistics/individuals/top10-list',
12328add1155SRico Sonntag            [
12338add1155SRico Sonntag                'records' => $records,
12348add1155SRico Sonntag            ]
12358add1155SRico Sonntag        );
12368add1155SRico Sonntag    }
12378add1155SRico Sonntag
12388add1155SRico Sonntag    /**
12398add1155SRico Sonntag     * Find the longest lived living males.
12408add1155SRico Sonntag     *
12418add1155SRico Sonntag     * @param int $total
12428add1155SRico Sonntag     *
12438add1155SRico Sonntag     * @return string
12448add1155SRico Sonntag     */
12458add1155SRico Sonntag    public function topTenOldestMaleAlive(int $total = 10): string
12468add1155SRico Sonntag    {
12478add1155SRico Sonntag        if (!Auth::isMember($this->tree)) {
12488add1155SRico Sonntag            return I18N::translate('This information is private and cannot be shown.');
12498add1155SRico Sonntag        }
12508add1155SRico Sonntag
12518add1155SRico Sonntag        $records = $this->topTenOldestAliveQuery('M', $total);
12528add1155SRico Sonntag
12538add1155SRico Sonntag        return view(
12548add1155SRico Sonntag            'statistics/individuals/top10-nolist',
12558add1155SRico Sonntag            [
12568add1155SRico Sonntag                'records' => $records,
12578add1155SRico Sonntag            ]
12588add1155SRico Sonntag        );
12598add1155SRico Sonntag    }
12608add1155SRico Sonntag
12618add1155SRico Sonntag    /**
12628add1155SRico Sonntag     * Find the longest lived living males.
12638add1155SRico Sonntag     *
12648add1155SRico Sonntag     * @param int $total
12658add1155SRico Sonntag     *
12668add1155SRico Sonntag     * @return string
12678add1155SRico Sonntag     */
12688add1155SRico Sonntag    public function topTenOldestMaleListAlive(int $total = 10): string
12698add1155SRico Sonntag    {
12708add1155SRico Sonntag        if (!Auth::isMember($this->tree)) {
12718add1155SRico Sonntag            return I18N::translate('This information is private and cannot be shown.');
12728add1155SRico Sonntag        }
12738add1155SRico Sonntag
12748add1155SRico Sonntag        $records = $this->topTenOldestAliveQuery('M', $total);
12758add1155SRico Sonntag
12768add1155SRico Sonntag        return view(
12778add1155SRico Sonntag            'statistics/individuals/top10-list',
12788add1155SRico Sonntag            [
12798add1155SRico Sonntag                'records' => $records,
12808add1155SRico Sonntag            ]
12818add1155SRico Sonntag        );
12828add1155SRico Sonntag    }
12838add1155SRico Sonntag
12848add1155SRico Sonntag    /**
12858add1155SRico Sonntag     * Find the average lifespan.
12868add1155SRico Sonntag     *
12878add1155SRico Sonntag     * @param string $sex
12888add1155SRico Sonntag     * @param bool   $show_years
12898add1155SRico Sonntag     *
12908add1155SRico Sonntag     * @return string
12918add1155SRico Sonntag     */
12928add1155SRico Sonntag    private function averageLifespanQuery(string $sex = 'BOTH', bool $show_years = false): string
12938add1155SRico Sonntag    {
129444cdc21eSGreg Roach        $prefix = DB::connection()->getTablePrefix();
12958add1155SRico Sonntag
129644cdc21eSGreg Roach        $days = (int) $this->birthAndDeathQuery($sex)
129744cdc21eSGreg Roach            ->groupBy('i_id')
129844cdc21eSGreg Roach            ->select(DB::raw('AVG(' . $prefix . 'death.d_julianday1 - ' . $prefix . 'birth.d_julianday2) AS days'))
129944cdc21eSGreg Roach            ->value('days');
13008add1155SRico Sonntag
13018add1155SRico Sonntag        if ($show_years) {
130244cdc21eSGreg Roach            return $this->calculateAge($days);
13038add1155SRico Sonntag        }
13048add1155SRico Sonntag
130544cdc21eSGreg Roach        return I18N::number((int) ($days / 365.25));
13068add1155SRico Sonntag    }
13078add1155SRico Sonntag
13088add1155SRico Sonntag    /**
13098add1155SRico Sonntag     * Find the average lifespan.
13108add1155SRico Sonntag     *
13118add1155SRico Sonntag     * @param bool $show_years
13128add1155SRico Sonntag     *
13138add1155SRico Sonntag     * @return string
13148add1155SRico Sonntag     */
13158add1155SRico Sonntag    public function averageLifespan($show_years = false): string
13168add1155SRico Sonntag    {
13178add1155SRico Sonntag        return $this->averageLifespanQuery('BOTH', $show_years);
13188add1155SRico Sonntag    }
13198add1155SRico Sonntag
13208add1155SRico Sonntag    /**
13218add1155SRico Sonntag     * Find the average lifespan of females.
13228add1155SRico Sonntag     *
13238add1155SRico Sonntag     * @param bool $show_years
13248add1155SRico Sonntag     *
13258add1155SRico Sonntag     * @return string
13268add1155SRico Sonntag     */
13278add1155SRico Sonntag    public function averageLifespanFemale($show_years = false): string
13288add1155SRico Sonntag    {
13298add1155SRico Sonntag        return $this->averageLifespanQuery('F', $show_years);
13308add1155SRico Sonntag    }
13318add1155SRico Sonntag
13328add1155SRico Sonntag    /**
13338add1155SRico Sonntag     * Find the average male lifespan.
13348add1155SRico Sonntag     *
13358add1155SRico Sonntag     * @param bool $show_years
13368add1155SRico Sonntag     *
13378add1155SRico Sonntag     * @return string
13388add1155SRico Sonntag     */
13398add1155SRico Sonntag    public function averageLifespanMale($show_years = false): string
13408add1155SRico Sonntag    {
13418add1155SRico Sonntag        return $this->averageLifespanQuery('M', $show_years);
13428add1155SRico Sonntag    }
13438add1155SRico Sonntag
13448add1155SRico Sonntag    /**
13458add1155SRico Sonntag     * Convert totals into percentages.
13468add1155SRico Sonntag     *
13478add1155SRico Sonntag     * @param int $count
13488add1155SRico Sonntag     * @param int $total
13498add1155SRico Sonntag     *
13508add1155SRico Sonntag     * @return string
13518add1155SRico Sonntag     */
13528add1155SRico Sonntag    private function getPercentage(int $count, int $total): string
13538add1155SRico Sonntag    {
13548add1155SRico Sonntag        return I18N::percentage($count / $total, 1);
13558add1155SRico Sonntag    }
13568add1155SRico Sonntag
13578add1155SRico Sonntag    /**
13588add1155SRico Sonntag     * Returns how many individuals exist in the tree.
13598add1155SRico Sonntag     *
13608add1155SRico Sonntag     * @return int
13618add1155SRico Sonntag     */
13628add1155SRico Sonntag    private function totalIndividualsQuery(): int
13638add1155SRico Sonntag    {
13648add1155SRico Sonntag        return DB::table('individuals')
13658add1155SRico Sonntag            ->where('i_file', '=', $this->tree->id())
13668add1155SRico Sonntag            ->count();
13678add1155SRico Sonntag    }
13688add1155SRico Sonntag
13698add1155SRico Sonntag    /**
13708add1155SRico Sonntag     * Count the number of living individuals.
13718add1155SRico Sonntag     *
13728add1155SRico Sonntag     * The totalLiving/totalDeceased queries assume that every dead person will
13738add1155SRico Sonntag     * have a DEAT record. It will not include individuals who were born more
13748add1155SRico Sonntag     * than MAX_ALIVE_AGE years ago, and who have no DEAT record.
13758add1155SRico Sonntag     * A good reason to run the “Add missing DEAT records” batch-update!
13768add1155SRico Sonntag     *
13778add1155SRico Sonntag     * @return int
13788add1155SRico Sonntag     */
13798add1155SRico Sonntag    private function totalLivingQuery(): int
13808add1155SRico Sonntag    {
13813dc8167dSGreg Roach        $query = DB::table('individuals')
13823dc8167dSGreg Roach            ->where('i_file', '=', $this->tree->id());
13833dc8167dSGreg Roach
13843dc8167dSGreg Roach        foreach (Gedcom::DEATH_EVENTS as $death_event) {
13853dc8167dSGreg Roach            $query->where('i_gedcom', 'NOT LIKE', '%\n1 ' . $death_event);
13863dc8167dSGreg Roach        }
13873dc8167dSGreg Roach
13883dc8167dSGreg Roach        return $query->count();
13898add1155SRico Sonntag    }
13908add1155SRico Sonntag
13918add1155SRico Sonntag    /**
13928add1155SRico Sonntag     * Count the number of dead individuals.
13938add1155SRico Sonntag     *
13948add1155SRico Sonntag     * @return int
13958add1155SRico Sonntag     */
13968add1155SRico Sonntag    private function totalDeceasedQuery(): int
13978add1155SRico Sonntag    {
13988add1155SRico Sonntag        return DB::table('individuals')
13998add1155SRico Sonntag            ->where('i_file', '=', $this->tree->id())
14003dc8167dSGreg Roach            ->where(function (Builder $query): void {
14013dc8167dSGreg Roach                foreach (Gedcom::DEATH_EVENTS as $death_event) {
14023dc8167dSGreg Roach                    $query->orWhere('i_gedcom', 'NOT LIKE', '%\n1 ' . $death_event);
14033dc8167dSGreg Roach                }
14043dc8167dSGreg Roach            })
14058add1155SRico Sonntag            ->count();
14068add1155SRico Sonntag    }
14078add1155SRico Sonntag
14088add1155SRico Sonntag    /**
14098add1155SRico Sonntag     * Returns the total count of a specific sex.
14108add1155SRico Sonntag     *
14118add1155SRico Sonntag     * @param string $sex The sex to query
14128add1155SRico Sonntag     *
14138add1155SRico Sonntag     * @return int
14148add1155SRico Sonntag     */
14158add1155SRico Sonntag    private function getTotalSexQuery(string $sex): int
14168add1155SRico Sonntag    {
14178add1155SRico Sonntag        return DB::table('individuals')
14188add1155SRico Sonntag            ->where('i_file', '=', $this->tree->id())
14198add1155SRico Sonntag            ->where('i_sex', '=', $sex)
14208add1155SRico Sonntag            ->count();
14218add1155SRico Sonntag    }
14228add1155SRico Sonntag
14238add1155SRico Sonntag    /**
14248add1155SRico Sonntag     * Returns the total number of males.
14258add1155SRico Sonntag     *
14268add1155SRico Sonntag     * @return int
14278add1155SRico Sonntag     */
14288add1155SRico Sonntag    private function totalSexMalesQuery(): int
14298add1155SRico Sonntag    {
14308add1155SRico Sonntag        return $this->getTotalSexQuery('M');
14318add1155SRico Sonntag    }
14328add1155SRico Sonntag
14338add1155SRico Sonntag    /**
14348add1155SRico Sonntag     * Returns the total number of females.
14358add1155SRico Sonntag     *
14368add1155SRico Sonntag     * @return int
14378add1155SRico Sonntag     */
14388add1155SRico Sonntag    private function totalSexFemalesQuery(): int
14398add1155SRico Sonntag    {
14408add1155SRico Sonntag        return $this->getTotalSexQuery('F');
14418add1155SRico Sonntag    }
14428add1155SRico Sonntag
14438add1155SRico Sonntag    /**
14448add1155SRico Sonntag     * Returns the total number of individuals with unknown sex.
14458add1155SRico Sonntag     *
14468add1155SRico Sonntag     * @return int
14478add1155SRico Sonntag     */
14488add1155SRico Sonntag    private function totalSexUnknownQuery(): int
14498add1155SRico Sonntag    {
14508add1155SRico Sonntag        return $this->getTotalSexQuery('U');
14518add1155SRico Sonntag    }
14528add1155SRico Sonntag
14538add1155SRico Sonntag    /**
14548add1155SRico Sonntag     * Count the total families.
14558add1155SRico Sonntag     *
14568add1155SRico Sonntag     * @return int
14578add1155SRico Sonntag     */
14588add1155SRico Sonntag    private function totalFamiliesQuery(): int
14598add1155SRico Sonntag    {
14608add1155SRico Sonntag        return DB::table('families')
14618add1155SRico Sonntag            ->where('f_file', '=', $this->tree->id())
14628add1155SRico Sonntag            ->count();
14638add1155SRico Sonntag    }
14648add1155SRico Sonntag
14658add1155SRico Sonntag    /**
14668add1155SRico Sonntag     * How many individuals have one or more sources.
14678add1155SRico Sonntag     *
14688add1155SRico Sonntag     * @return int
14698add1155SRico Sonntag     */
14708add1155SRico Sonntag    private function totalIndisWithSourcesQuery(): int
14718add1155SRico Sonntag    {
14728add1155SRico Sonntag        return DB::table('individuals')
14738add1155SRico Sonntag            ->select(['i_id'])
14748add1155SRico Sonntag            ->distinct()
14758add1155SRico Sonntag            ->join('link', function (JoinClause $join) {
14768add1155SRico Sonntag                $join->on('i_id', '=', 'l_from')
14778add1155SRico Sonntag                    ->on('i_file', '=', 'l_file');
14788add1155SRico Sonntag            })
14798add1155SRico Sonntag            ->where('l_file', '=', $this->tree->id())
14808add1155SRico Sonntag            ->where('l_type', '=', 'SOUR')
14818add1155SRico Sonntag            ->count('i_id');
14828add1155SRico Sonntag    }
14838add1155SRico Sonntag
14848add1155SRico Sonntag    /**
14858add1155SRico Sonntag     * Count the families with source records.
14868add1155SRico Sonntag     *
14878add1155SRico Sonntag     * @return int
14888add1155SRico Sonntag     */
14898add1155SRico Sonntag    private function totalFamsWithSourcesQuery(): int
14908add1155SRico Sonntag    {
14918add1155SRico Sonntag        return DB::table('families')
14928add1155SRico Sonntag            ->select(['f_id'])
14938add1155SRico Sonntag            ->distinct()
14948add1155SRico Sonntag            ->join('link', function (JoinClause $join) {
14958add1155SRico Sonntag                $join->on('f_id', '=', 'l_from')
14968add1155SRico Sonntag                    ->on('f_file', '=', 'l_file');
14978add1155SRico Sonntag            })
14988add1155SRico Sonntag            ->where('l_file', '=', $this->tree->id())
14998add1155SRico Sonntag            ->where('l_type', '=', 'SOUR')
15008add1155SRico Sonntag            ->count('f_id');
15018add1155SRico Sonntag    }
15028add1155SRico Sonntag
15038add1155SRico Sonntag    /**
15048add1155SRico Sonntag     * Count the number of repositories.
15058add1155SRico Sonntag     *
15068add1155SRico Sonntag     * @return int
15078add1155SRico Sonntag     */
15088add1155SRico Sonntag    private function totalRepositoriesQuery(): int
15098add1155SRico Sonntag    {
15108add1155SRico Sonntag        return DB::table('other')
15118add1155SRico Sonntag            ->where('o_file', '=', $this->tree->id())
15128add1155SRico Sonntag            ->where('o_type', '=', 'REPO')
15138add1155SRico Sonntag            ->count();
15148add1155SRico Sonntag    }
15158add1155SRico Sonntag
15168add1155SRico Sonntag    /**
15178add1155SRico Sonntag     * Count the total number of sources.
15188add1155SRico Sonntag     *
15198add1155SRico Sonntag     * @return int
15208add1155SRico Sonntag     */
15218add1155SRico Sonntag    private function totalSourcesQuery(): int
15228add1155SRico Sonntag    {
15238add1155SRico Sonntag        return DB::table('sources')
15248add1155SRico Sonntag            ->where('s_file', '=', $this->tree->id())
15258add1155SRico Sonntag            ->count();
15268add1155SRico Sonntag    }
15278add1155SRico Sonntag
15288add1155SRico Sonntag    /**
15298add1155SRico Sonntag     * Count the number of notes.
15308add1155SRico Sonntag     *
15318add1155SRico Sonntag     * @return int
15328add1155SRico Sonntag     */
15338add1155SRico Sonntag    private function totalNotesQuery(): int
15348add1155SRico Sonntag    {
15358add1155SRico Sonntag        return DB::table('other')
15368add1155SRico Sonntag            ->where('o_file', '=', $this->tree->id())
15378add1155SRico Sonntag            ->where('o_type', '=', 'NOTE')
15388add1155SRico Sonntag            ->count();
15398add1155SRico Sonntag    }
15408add1155SRico Sonntag
15418add1155SRico Sonntag    /**
15428add1155SRico Sonntag     * Returns the total number of records.
15438add1155SRico Sonntag     *
15448add1155SRico Sonntag     * @return int
15458add1155SRico Sonntag     */
15468add1155SRico Sonntag    private function totalRecordsQuery(): int
15478add1155SRico Sonntag    {
15488add1155SRico Sonntag        return $this->totalIndividualsQuery()
15498add1155SRico Sonntag            + $this->totalFamiliesQuery()
15508add1155SRico Sonntag            + $this->totalNotesQuery()
15518add1155SRico Sonntag            + $this->totalRepositoriesQuery()
15528add1155SRico Sonntag            + $this->totalSourcesQuery();
15538add1155SRico Sonntag    }
15548add1155SRico Sonntag
15558add1155SRico Sonntag    /**
15568add1155SRico Sonntag     * @inheritDoc
15578add1155SRico Sonntag     */
15588add1155SRico Sonntag    public function totalRecords(): string
15598add1155SRico Sonntag    {
15608add1155SRico Sonntag        return I18N::number($this->totalRecordsQuery());
15618add1155SRico Sonntag    }
15628add1155SRico Sonntag
15638add1155SRico Sonntag    /**
15648add1155SRico Sonntag     * @inheritDoc
15658add1155SRico Sonntag     */
15668add1155SRico Sonntag    public function totalIndividuals(): string
15678add1155SRico Sonntag    {
15688add1155SRico Sonntag        return I18N::number($this->totalIndividualsQuery());
15698add1155SRico Sonntag    }
15708add1155SRico Sonntag
15718add1155SRico Sonntag    /**
15728add1155SRico Sonntag     * Count the number of living individuals.
15738add1155SRico Sonntag     *
15748add1155SRico Sonntag     * @return string
15758add1155SRico Sonntag     */
15768add1155SRico Sonntag    public function totalLiving(): string
15778add1155SRico Sonntag    {
15788add1155SRico Sonntag        return I18N::number($this->totalLivingQuery());
15798add1155SRico Sonntag    }
15808add1155SRico Sonntag
15818add1155SRico Sonntag    /**
15828add1155SRico Sonntag     * Count the number of dead individuals.
15838add1155SRico Sonntag     *
15848add1155SRico Sonntag     * @return string
15858add1155SRico Sonntag     */
15868add1155SRico Sonntag    public function totalDeceased(): string
15878add1155SRico Sonntag    {
15888add1155SRico Sonntag        return I18N::number($this->totalDeceasedQuery());
15898add1155SRico Sonntag    }
15908add1155SRico Sonntag
15918add1155SRico Sonntag    /**
15928add1155SRico Sonntag     * @inheritDoc
15938add1155SRico Sonntag     */
15948add1155SRico Sonntag    public function totalSexMales(): string
15958add1155SRico Sonntag    {
15968add1155SRico Sonntag        return I18N::number($this->totalSexMalesQuery());
15978add1155SRico Sonntag    }
15988add1155SRico Sonntag
15998add1155SRico Sonntag    /**
16008add1155SRico Sonntag     * @inheritDoc
16018add1155SRico Sonntag     */
16028add1155SRico Sonntag    public function totalSexFemales(): string
16038add1155SRico Sonntag    {
16048add1155SRico Sonntag        return I18N::number($this->totalSexFemalesQuery());
16058add1155SRico Sonntag    }
16068add1155SRico Sonntag
16078add1155SRico Sonntag    /**
16088add1155SRico Sonntag     * @inheritDoc
16098add1155SRico Sonntag     */
16108add1155SRico Sonntag    public function totalSexUnknown(): string
16118add1155SRico Sonntag    {
16128add1155SRico Sonntag        return I18N::number($this->totalSexUnknownQuery());
16138add1155SRico Sonntag    }
16148add1155SRico Sonntag
16158add1155SRico Sonntag    /**
16168add1155SRico Sonntag     * @inheritDoc
16178add1155SRico Sonntag     */
16188add1155SRico Sonntag    public function totalFamilies(): string
16198add1155SRico Sonntag    {
16208add1155SRico Sonntag        return I18N::number($this->totalFamiliesQuery());
16218add1155SRico Sonntag    }
16228add1155SRico Sonntag
16238add1155SRico Sonntag    /**
16248add1155SRico Sonntag     * How many individuals have one or more sources.
16258add1155SRico Sonntag     *
16268add1155SRico Sonntag     * @return string
16278add1155SRico Sonntag     */
16288add1155SRico Sonntag    public function totalIndisWithSources(): string
16298add1155SRico Sonntag    {
16308add1155SRico Sonntag        return I18N::number($this->totalIndisWithSourcesQuery());
16318add1155SRico Sonntag    }
16328add1155SRico Sonntag
16338add1155SRico Sonntag    /**
16348add1155SRico Sonntag     * Count the families with with source records.
16358add1155SRico Sonntag     *
16368add1155SRico Sonntag     * @return string
16378add1155SRico Sonntag     */
16388add1155SRico Sonntag    public function totalFamsWithSources(): string
16398add1155SRico Sonntag    {
16408add1155SRico Sonntag        return I18N::number($this->totalFamsWithSourcesQuery());
16418add1155SRico Sonntag    }
16428add1155SRico Sonntag
16438add1155SRico Sonntag    /**
16448add1155SRico Sonntag     * @inheritDoc
16458add1155SRico Sonntag     */
16468add1155SRico Sonntag    public function totalRepositories(): string
16478add1155SRico Sonntag    {
16488add1155SRico Sonntag        return I18N::number($this->totalRepositoriesQuery());
16498add1155SRico Sonntag    }
16508add1155SRico Sonntag
16518add1155SRico Sonntag    /**
16528add1155SRico Sonntag     * @inheritDoc
16538add1155SRico Sonntag     */
16548add1155SRico Sonntag    public function totalSources(): string
16558add1155SRico Sonntag    {
16568add1155SRico Sonntag        return I18N::number($this->totalSourcesQuery());
16578add1155SRico Sonntag    }
16588add1155SRico Sonntag
16598add1155SRico Sonntag    /**
16608add1155SRico Sonntag     * @inheritDoc
16618add1155SRico Sonntag     */
16628add1155SRico Sonntag    public function totalNotes(): string
16638add1155SRico Sonntag    {
16648add1155SRico Sonntag        return I18N::number($this->totalNotesQuery());
16658add1155SRico Sonntag    }
16668add1155SRico Sonntag
16678add1155SRico Sonntag    /**
16688add1155SRico Sonntag     * @inheritDoc
16698add1155SRico Sonntag     */
16708add1155SRico Sonntag    public function totalIndividualsPercentage(): string
16718add1155SRico Sonntag    {
16728add1155SRico Sonntag        return $this->getPercentage(
16738add1155SRico Sonntag            $this->totalIndividualsQuery(),
16748add1155SRico Sonntag            $this->totalRecordsQuery()
16758add1155SRico Sonntag        );
16768add1155SRico Sonntag    }
16778add1155SRico Sonntag
16788add1155SRico Sonntag    /**
16798add1155SRico Sonntag     * @inheritDoc
16808add1155SRico Sonntag     */
16818add1155SRico Sonntag    public function totalFamiliesPercentage(): string
16828add1155SRico Sonntag    {
16838add1155SRico Sonntag        return $this->getPercentage(
16848add1155SRico Sonntag            $this->totalFamiliesQuery(),
16858add1155SRico Sonntag            $this->totalRecordsQuery()
16868add1155SRico Sonntag        );
16878add1155SRico Sonntag    }
16888add1155SRico Sonntag
16898add1155SRico Sonntag    /**
16908add1155SRico Sonntag     * @inheritDoc
16918add1155SRico Sonntag     */
16928add1155SRico Sonntag    public function totalRepositoriesPercentage(): string
16938add1155SRico Sonntag    {
16948add1155SRico Sonntag        return $this->getPercentage(
16958add1155SRico Sonntag            $this->totalRepositoriesQuery(),
16968add1155SRico Sonntag            $this->totalRecordsQuery()
16978add1155SRico Sonntag        );
16988add1155SRico Sonntag    }
16998add1155SRico Sonntag
17008add1155SRico Sonntag    /**
17018add1155SRico Sonntag     * @inheritDoc
17028add1155SRico Sonntag     */
17038add1155SRico Sonntag    public function totalSourcesPercentage(): string
17048add1155SRico Sonntag    {
17058add1155SRico Sonntag        return $this->getPercentage(
17068add1155SRico Sonntag            $this->totalSourcesQuery(),
17078add1155SRico Sonntag            $this->totalRecordsQuery()
17088add1155SRico Sonntag        );
17098add1155SRico Sonntag    }
17108add1155SRico Sonntag
17118add1155SRico Sonntag    /**
17128add1155SRico Sonntag     * @inheritDoc
17138add1155SRico Sonntag     */
17148add1155SRico Sonntag    public function totalNotesPercentage(): string
17158add1155SRico Sonntag    {
17168add1155SRico Sonntag        return $this->getPercentage(
17178add1155SRico Sonntag            $this->totalNotesQuery(),
17188add1155SRico Sonntag            $this->totalRecordsQuery()
17198add1155SRico Sonntag        );
17208add1155SRico Sonntag    }
17218add1155SRico Sonntag
17228add1155SRico Sonntag    /**
17238add1155SRico Sonntag     * @inheritDoc
17248add1155SRico Sonntag     */
17258add1155SRico Sonntag    public function totalLivingPercentage(): string
17268add1155SRico Sonntag    {
17278add1155SRico Sonntag        return $this->getPercentage(
17288add1155SRico Sonntag            $this->totalLivingQuery(),
17298add1155SRico Sonntag            $this->totalIndividualsQuery()
17308add1155SRico Sonntag        );
17318add1155SRico Sonntag    }
17328add1155SRico Sonntag
17338add1155SRico Sonntag    /**
17348add1155SRico Sonntag     * @inheritDoc
17358add1155SRico Sonntag     */
17368add1155SRico Sonntag    public function totalDeceasedPercentage(): string
17378add1155SRico Sonntag    {
17388add1155SRico Sonntag        return $this->getPercentage(
17398add1155SRico Sonntag            $this->totalDeceasedQuery(),
17408add1155SRico Sonntag            $this->totalIndividualsQuery()
17418add1155SRico Sonntag        );
17428add1155SRico Sonntag    }
17438add1155SRico Sonntag
17448add1155SRico Sonntag    /**
17458add1155SRico Sonntag     * @inheritDoc
17468add1155SRico Sonntag     */
17478add1155SRico Sonntag    public function totalSexMalesPercentage(): string
17488add1155SRico Sonntag    {
17498add1155SRico Sonntag        return $this->getPercentage(
17508add1155SRico Sonntag            $this->totalSexMalesQuery(),
17518add1155SRico Sonntag            $this->totalIndividualsQuery()
17528add1155SRico Sonntag        );
17538add1155SRico Sonntag    }
17548add1155SRico Sonntag
17558add1155SRico Sonntag    /**
17568add1155SRico Sonntag     * @inheritDoc
17578add1155SRico Sonntag     */
17588add1155SRico Sonntag    public function totalSexFemalesPercentage(): string
17598add1155SRico Sonntag    {
17608add1155SRico Sonntag        return $this->getPercentage(
17618add1155SRico Sonntag            $this->totalSexFemalesQuery(),
17628add1155SRico Sonntag            $this->totalIndividualsQuery()
17638add1155SRico Sonntag        );
17648add1155SRico Sonntag    }
17658add1155SRico Sonntag
17668add1155SRico Sonntag    /**
17678add1155SRico Sonntag     * @inheritDoc
17688add1155SRico Sonntag     */
17698add1155SRico Sonntag    public function totalSexUnknownPercentage(): string
17708add1155SRico Sonntag    {
17718add1155SRico Sonntag        return $this->getPercentage(
17728add1155SRico Sonntag            $this->totalSexUnknownQuery(),
17738add1155SRico Sonntag            $this->totalIndividualsQuery()
17748add1155SRico Sonntag        );
17758add1155SRico Sonntag    }
17768add1155SRico Sonntag
17778add1155SRico Sonntag    /**
17788add1155SRico Sonntag     * Create a chart of common given names.
17798add1155SRico Sonntag     *
17808add1155SRico Sonntag     * @param string|null $size
17818add1155SRico Sonntag     * @param string|null $color_from
17828add1155SRico Sonntag     * @param string|null $color_to
17838add1155SRico Sonntag     * @param int         $maxtoshow
17848add1155SRico Sonntag     *
17858add1155SRico Sonntag     * @return string
17868add1155SRico Sonntag     */
17878add1155SRico Sonntag    public function chartCommonGiven(
17888add1155SRico Sonntag        string $size = null,
17898add1155SRico Sonntag        string $color_from = null,
17908add1155SRico Sonntag        string $color_to = null,
17918add1155SRico Sonntag        int $maxtoshow = 7
17928add1155SRico Sonntag    ): string {
17938add1155SRico Sonntag        $tot_indi = $this->totalIndividualsQuery();
17948add1155SRico Sonntag        $given    = $this->commonGivenQuery('B', 'chart', false, 1, $maxtoshow);
17958add1155SRico Sonntag
17968add1155SRico Sonntag        return (new ChartCommonGiven())
17978add1155SRico Sonntag            ->chartCommonGiven($tot_indi, $given, $size, $color_from, $color_to);
17988add1155SRico Sonntag    }
17998add1155SRico Sonntag
18008add1155SRico Sonntag    /**
18018add1155SRico Sonntag     * Create a chart of common surnames.
18028add1155SRico Sonntag     *
18038add1155SRico Sonntag     * @param string|null $size
18048add1155SRico Sonntag     * @param string|null $color_from
18058add1155SRico Sonntag     * @param string|null $color_to
18068add1155SRico Sonntag     * @param int         $number_of_surnames
18078add1155SRico Sonntag     *
18088add1155SRico Sonntag     * @return string
18098add1155SRico Sonntag     */
18108add1155SRico Sonntag    public function chartCommonSurnames(
18118add1155SRico Sonntag        string $size = null,
18128add1155SRico Sonntag        string $color_from = null,
18138add1155SRico Sonntag        string $color_to = null,
18148add1155SRico Sonntag        int $number_of_surnames = 10
18158add1155SRico Sonntag    ): string {
18168add1155SRico Sonntag        $tot_indi     = $this->totalIndividualsQuery();
18178add1155SRico Sonntag        $all_surnames = $this->topSurnames($number_of_surnames, 0);
18188add1155SRico Sonntag
18198add1155SRico Sonntag        return (new ChartCommonSurname($this->tree))
18208add1155SRico Sonntag            ->chartCommonSurnames($tot_indi, $all_surnames, $size, $color_from, $color_to);
18218add1155SRico Sonntag    }
18228add1155SRico Sonntag
18238add1155SRico Sonntag    /**
18248add1155SRico Sonntag     * Create a chart showing mortality.
18258add1155SRico Sonntag     *
18268add1155SRico Sonntag     * @param string|null $size
18278add1155SRico Sonntag     * @param string|null $color_living
18288add1155SRico Sonntag     * @param string|null $color_dead
18298add1155SRico Sonntag     *
18308add1155SRico Sonntag     * @return string
18318add1155SRico Sonntag     */
18328add1155SRico Sonntag    public function chartMortality(string $size = null, string $color_living = null, string $color_dead = null): string
18338add1155SRico Sonntag    {
18348add1155SRico Sonntag        $tot_l = $this->totalLivingQuery();
18358add1155SRico Sonntag        $tot_d = $this->totalDeceasedQuery();
18368add1155SRico Sonntag
18378add1155SRico Sonntag        return (new ChartMortality($this->tree))
18388add1155SRico Sonntag            ->chartMortality($tot_l, $tot_d, $size, $color_living, $color_dead);
18398add1155SRico Sonntag    }
18408add1155SRico Sonntag
18418add1155SRico Sonntag    /**
18428add1155SRico Sonntag     * Create a chart showing individuals with/without sources.
18438add1155SRico Sonntag     *
18448add1155SRico Sonntag     * @param string|null $size
18458add1155SRico Sonntag     * @param string|null $color_from
18468add1155SRico Sonntag     * @param string|null $color_to
18478add1155SRico Sonntag     *
18488add1155SRico Sonntag     * @return string
18498add1155SRico Sonntag     */
18508add1155SRico Sonntag    public function chartIndisWithSources(
18518add1155SRico Sonntag        string $size       = null,
18528add1155SRico Sonntag        string $color_from = null,
18538add1155SRico Sonntag        string $color_to   = null
18548add1155SRico Sonntag    ): string {
18558add1155SRico Sonntag        $tot_indi        = $this->totalIndividualsQuery();
18568add1155SRico Sonntag        $tot_indi_source = $this->totalIndisWithSourcesQuery();
18578add1155SRico Sonntag
18588add1155SRico Sonntag        return (new ChartIndividual())
18598add1155SRico Sonntag            ->chartIndisWithSources($tot_indi, $tot_indi_source, $size, $color_from, $color_to);
18608add1155SRico Sonntag    }
18618add1155SRico Sonntag
18628add1155SRico Sonntag    /**
18638add1155SRico Sonntag     * Create a chart of individuals with/without sources.
18648add1155SRico Sonntag     *
18658add1155SRico Sonntag     * @param string|null $size
18668add1155SRico Sonntag     * @param string|null $color_from
18678add1155SRico Sonntag     * @param string|null $color_to
18688add1155SRico Sonntag     *
18698add1155SRico Sonntag     * @return string
18708add1155SRico Sonntag     */
18718add1155SRico Sonntag    public function chartFamsWithSources(
18728add1155SRico Sonntag        string $size       = null,
18738add1155SRico Sonntag        string $color_from = null,
18748add1155SRico Sonntag        string $color_to   = null
18758add1155SRico Sonntag    ): string {
18768add1155SRico Sonntag        $tot_fam        = $this->totalFamiliesQuery();
18778add1155SRico Sonntag        $tot_fam_source = $this->totalFamsWithSourcesQuery();
18788add1155SRico Sonntag
18798add1155SRico Sonntag        return (new ChartFamilyWithSources())
18808add1155SRico Sonntag            ->chartFamsWithSources($tot_fam, $tot_fam_source, $size, $color_from, $color_to);
18818add1155SRico Sonntag    }
18828add1155SRico Sonntag
18838add1155SRico Sonntag    /**
18848add1155SRico Sonntag     * @inheritDoc
18858add1155SRico Sonntag     */
18868add1155SRico Sonntag    public function chartSex(
18878add1155SRico Sonntag        string $size          = null,
18888add1155SRico Sonntag        string $color_female  = null,
18898add1155SRico Sonntag        string $color_male    = null,
18908add1155SRico Sonntag        string $color_unknown = null
18918add1155SRico Sonntag    ): string {
18928add1155SRico Sonntag        $tot_m = $this->totalSexMalesQuery();
18938add1155SRico Sonntag        $tot_f = $this->totalSexFemalesQuery();
18948add1155SRico Sonntag        $tot_u = $this->totalSexUnknownQuery();
18958add1155SRico Sonntag
18968add1155SRico Sonntag        return (new ChartSex($this->tree))
18978add1155SRico Sonntag            ->chartSex($tot_m, $tot_f, $tot_u, $size, $color_female, $color_male, $color_unknown);
18988add1155SRico Sonntag    }
189944cdc21eSGreg Roach
190044cdc21eSGreg Roach    /**
190144cdc21eSGreg Roach     * Query individuals, with their births and deaths.
190244cdc21eSGreg Roach     *
190344cdc21eSGreg Roach     * @param string $sex
190444cdc21eSGreg Roach     *
190544cdc21eSGreg Roach     * @return Builder
190644cdc21eSGreg Roach     */
1907*e2cbf57aSGreg Roach    private function birthAndDeathQuery(string $sex): Builder
1908*e2cbf57aSGreg Roach    {
190944cdc21eSGreg Roach        $query = DB::table('individuals')
191044cdc21eSGreg Roach            ->where('i_file', '=', $this->tree->id())
191144cdc21eSGreg Roach            ->join('dates AS birth', function (JoinClause $join): void {
191244cdc21eSGreg Roach                $join
191344cdc21eSGreg Roach                    ->on('birth.d_file', '=', 'i_file')
191444cdc21eSGreg Roach                    ->on('birth.d_gid', '=', 'i_id');
191544cdc21eSGreg Roach            })
191644cdc21eSGreg Roach            ->join('dates AS death', function (JoinClause $join): void {
191744cdc21eSGreg Roach                $join
191844cdc21eSGreg Roach                    ->on('death.d_file', '=', 'i_file')
191944cdc21eSGreg Roach                    ->on('death.d_gid', '=', 'i_id');
192044cdc21eSGreg Roach            })
192144cdc21eSGreg Roach            ->where('birth.d_fact', '=', 'BIRT')
192244cdc21eSGreg Roach            ->where('death.d_fact', '=', 'DEAT')
192344cdc21eSGreg Roach            ->whereColumn('death.d_julianday1', '>=', 'birth.d_julianday2')
192444cdc21eSGreg Roach            ->where('birth.d_julianday2', '<>', 0);
192544cdc21eSGreg Roach
192644cdc21eSGreg Roach        if ($sex === 'M' || $sex === 'F') {
192744cdc21eSGreg Roach            $query->where('i_sex', '=', $sex);
192844cdc21eSGreg Roach        }
192944cdc21eSGreg Roach
193044cdc21eSGreg Roach        return $query;
193144cdc21eSGreg Roach    }
19328add1155SRico Sonntag}
1933