1<?php 2 3/** 4 * webtrees: online genealogy 5 * Copyright (C) 2023 webtrees development team 6 * This program is free software: you can redistribute it and/or modify 7 * it under the terms of the GNU General Public License as published by 8 * the Free Software Foundation, either version 3 of the License, or 9 * (at your option) any later version. 10 * This program is distributed in the hope that it will be useful, 11 * but WITHOUT ANY WARRANTY; without even the implied warranty of 12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 13 * GNU General Public License for more details. 14 * You should have received a copy of the GNU General Public License 15 * along with this program. If not, see <https://www.gnu.org/licenses/>. 16 */ 17 18declare(strict_types=1); 19 20namespace Fisharebest\Webtrees; 21 22use Fisharebest\Webtrees\Module\ModuleBlockInterface; 23use Fisharebest\Webtrees\Module\ModuleInterface; 24use Fisharebest\Webtrees\Services\ModuleService; 25use Fisharebest\Webtrees\Services\UserService; 26use Fisharebest\Webtrees\Statistics\Repository\BrowserRepository; 27use Fisharebest\Webtrees\Statistics\Repository\ContactRepository; 28use Fisharebest\Webtrees\Statistics\Repository\EventRepository; 29use Fisharebest\Webtrees\Statistics\Repository\FamilyDatesRepository; 30use Fisharebest\Webtrees\Statistics\Repository\FamilyRepository; 31use Fisharebest\Webtrees\Statistics\Repository\FavoritesRepository; 32use Fisharebest\Webtrees\Statistics\Repository\GedcomRepository; 33use Fisharebest\Webtrees\Statistics\Repository\HitCountRepository; 34use Fisharebest\Webtrees\Statistics\Repository\IndividualRepository; 35use Fisharebest\Webtrees\Statistics\Repository\Interfaces\BrowserRepositoryInterface; 36use Fisharebest\Webtrees\Statistics\Repository\Interfaces\ContactRepositoryInterface; 37use Fisharebest\Webtrees\Statistics\Repository\Interfaces\EventRepositoryInterface; 38use Fisharebest\Webtrees\Statistics\Repository\Interfaces\FamilyDatesRepositoryInterface; 39use Fisharebest\Webtrees\Statistics\Repository\Interfaces\FavoritesRepositoryInterface; 40use Fisharebest\Webtrees\Statistics\Repository\Interfaces\GedcomRepositoryInterface; 41use Fisharebest\Webtrees\Statistics\Repository\Interfaces\HitCountRepositoryInterface; 42use Fisharebest\Webtrees\Statistics\Repository\Interfaces\IndividualRepositoryInterface; 43use Fisharebest\Webtrees\Statistics\Repository\Interfaces\LatestUserRepositoryInterface; 44use Fisharebest\Webtrees\Statistics\Repository\Interfaces\MediaRepositoryInterface; 45use Fisharebest\Webtrees\Statistics\Repository\Interfaces\MessageRepositoryInterface; 46use Fisharebest\Webtrees\Statistics\Repository\Interfaces\NewsRepositoryInterface; 47use Fisharebest\Webtrees\Statistics\Repository\Interfaces\PlaceRepositoryInterface; 48use Fisharebest\Webtrees\Statistics\Repository\Interfaces\ServerRepositoryInterface; 49use Fisharebest\Webtrees\Statistics\Repository\Interfaces\UserRepositoryInterface; 50use Fisharebest\Webtrees\Statistics\Repository\LatestUserRepository; 51use Fisharebest\Webtrees\Statistics\Repository\MediaRepository; 52use Fisharebest\Webtrees\Statistics\Repository\MessageRepository; 53use Fisharebest\Webtrees\Statistics\Repository\NewsRepository; 54use Fisharebest\Webtrees\Statistics\Repository\PlaceRepository; 55use Fisharebest\Webtrees\Statistics\Repository\ServerRepository; 56use Fisharebest\Webtrees\Statistics\Repository\UserRepository; 57use Fisharebest\Webtrees\Statistics\Service\CenturyService; 58use Fisharebest\Webtrees\Statistics\Service\ColorService; 59use Fisharebest\Webtrees\Statistics\Service\CountryService; 60use Illuminate\Database\Query\Builder; 61use Illuminate\Support\Collection; 62use ReflectionClass; 63use ReflectionException; 64use ReflectionMethod; 65use ReflectionNamedType; 66use stdClass; 67 68use function count; 69use function in_array; 70use function str_contains; 71 72/** 73 * A selection of pre-formatted statistical queries. 74 * These are primarily used for embedded keywords on HTML blocks, but 75 * are also used elsewhere in the code. 76 */ 77class Statistics implements 78 GedcomRepositoryInterface, 79 IndividualRepositoryInterface, 80 EventRepositoryInterface, 81 MediaRepositoryInterface, 82 UserRepositoryInterface, 83 ServerRepositoryInterface, 84 BrowserRepositoryInterface, 85 HitCountRepositoryInterface, 86 LatestUserRepositoryInterface, 87 FavoritesRepositoryInterface, 88 NewsRepositoryInterface, 89 MessageRepositoryInterface, 90 ContactRepositoryInterface, 91 FamilyDatesRepositoryInterface, 92 PlaceRepositoryInterface 93{ 94 private Tree $tree; 95 96 private GedcomRepository $gedcom_repository; 97 98 private IndividualRepository $individual_repository; 99 100 private FamilyRepository $family_repository; 101 102 private MediaRepository $media_repository; 103 104 private EventRepository $event_repository; 105 106 private UserRepository $user_repository; 107 108 private ServerRepository $server_repository; 109 110 private BrowserRepository $browser_repository; 111 112 private HitCountRepository $hit_count_repository; 113 114 private LatestUserRepository $latest_user_repository; 115 116 private FavoritesRepository $favorites_repository; 117 118 private NewsRepository $news_repository; 119 120 private MessageRepository $message_repository; 121 122 private ContactRepository $contact_repository; 123 124 private FamilyDatesRepository $family_dates_repository; 125 126 private PlaceRepository $place_repository; 127 128 private ModuleService $module_service; 129 130 /** 131 * Create the statistics for a tree. 132 * 133 * @param CenturyService $century_service 134 * @param ColorService $color_service 135 * @param CountryService $country_service 136 * @param ModuleService $module_service 137 * @param Tree $tree Generate statistics for this tree 138 * @param UserService $user_service 139 */ 140 public function __construct( 141 CenturyService $century_service, 142 ColorService $color_service, 143 CountryService $country_service, 144 ModuleService $module_service, 145 Tree $tree, 146 UserService $user_service 147 ) { 148 $this->tree = $tree; 149 $this->gedcom_repository = new GedcomRepository($tree); 150 $this->individual_repository = new IndividualRepository($century_service, $color_service, $tree); 151 $this->family_repository = new FamilyRepository($century_service, $color_service, $tree); 152 $this->family_dates_repository = new FamilyDatesRepository($tree); 153 $this->media_repository = new MediaRepository($color_service, $tree); 154 $this->event_repository = new EventRepository($tree); 155 $this->user_repository = new UserRepository($tree, $user_service); 156 $this->server_repository = new ServerRepository(); 157 $this->browser_repository = new BrowserRepository(); 158 $this->hit_count_repository = new HitCountRepository($tree, $user_service); 159 $this->latest_user_repository = new LatestUserRepository($user_service); 160 $this->favorites_repository = new FavoritesRepository($tree, $module_service); 161 $this->news_repository = new NewsRepository($tree); 162 $this->message_repository = new MessageRepository(); 163 $this->contact_repository = new ContactRepository($tree, $user_service); 164 $this->place_repository = new PlaceRepository($tree, $country_service, $this->individual_repository); 165 $this->module_service = $module_service; 166 } 167 168 /** 169 * Return a string of all supported tags and an example of its output in table row form. 170 * 171 * @return string 172 */ 173 public function getAllTagsTable(): string 174 { 175 try { 176 $class = new ReflectionClass($this); 177 178 $public_methods = $class->getMethods(ReflectionMethod::IS_PUBLIC); 179 180 $examples = Collection::make($public_methods) 181 ->filter(static fn(ReflectionMethod $method): bool => !in_array($method->getName(), ['embedTags', 'getAllTagsTable'], true)) 182 ->filter(static function (ReflectionMethod $method): bool { 183 $type = $method->getReturnType(); 184 185 return $type instanceof ReflectionNamedType && $type->getName() === 'string'; 186 }) 187 ->sort(static fn(ReflectionMethod $x, ReflectionMethod $y): int => $x->getName() <=> $y->getName()) 188 ->map(function (ReflectionMethod $method): string { 189 $tag = $method->getName(); 190 191 return '<dt>#' . $tag . '#</dt><dd>' . $this->$tag() . '</dd>'; 192 }); 193 194 return '<dl>' . $examples->implode('') . '</dl>'; 195 } catch (ReflectionException $ex) { 196 return $ex->getMessage(); 197 } 198 } 199 200 /** 201 * Embed tags in text 202 * 203 * @param string $text 204 * 205 * @return string 206 */ 207 public function embedTags(string $text): string 208 { 209 if (str_contains($text, '#')) { 210 $text = strtr($text, $this->getTags($text)); 211 } 212 213 return $text; 214 } 215 216 /** 217 * @return string 218 */ 219 public function gedcomFilename(): string 220 { 221 return $this->gedcom_repository->gedcomFilename(); 222 } 223 224 /** 225 * @return int 226 */ 227 public function gedcomId(): int 228 { 229 return $this->gedcom_repository->gedcomId(); 230 } 231 232 /** 233 * @return string 234 */ 235 public function gedcomTitle(): string 236 { 237 return $this->gedcom_repository->gedcomTitle(); 238 } 239 240 /** 241 * @return string 242 */ 243 public function gedcomCreatedSoftware(): string 244 { 245 return $this->gedcom_repository->gedcomCreatedSoftware(); 246 } 247 248 /** 249 * @return string 250 */ 251 public function gedcomCreatedVersion(): string 252 { 253 return $this->gedcom_repository->gedcomCreatedVersion(); 254 } 255 256 /** 257 * @return string 258 */ 259 public function gedcomDate(): string 260 { 261 return $this->gedcom_repository->gedcomDate(); 262 } 263 264 /** 265 * @return string 266 */ 267 public function gedcomUpdated(): string 268 { 269 return $this->gedcom_repository->gedcomUpdated(); 270 } 271 272 /** 273 * @return string 274 */ 275 public function gedcomRootId(): string 276 { 277 return $this->gedcom_repository->gedcomRootId(); 278 } 279 280 /** 281 * @return string 282 */ 283 public function totalRecords(): string 284 { 285 return $this->individual_repository->totalRecords(); 286 } 287 288 /** 289 * @return string 290 */ 291 public function totalIndividuals(): string 292 { 293 return $this->individual_repository->totalIndividuals(); 294 } 295 296 /** 297 * @return string 298 */ 299 public function totalIndisWithSources(): string 300 { 301 return $this->individual_repository->totalIndisWithSources(); 302 } 303 304 /** 305 * @return string 306 */ 307 public function totalIndisWithSourcesPercentage(): string 308 { 309 return $this->individual_repository->totalIndisWithSourcesPercentage(); 310 } 311 312 /** 313 * @param string|null $color_from 314 * @param string|null $color_to 315 * 316 * @return string 317 */ 318 public function chartIndisWithSources( 319 string|null $color_from = null, 320 string $color_to = null 321 ): string { 322 return $this->individual_repository->chartIndisWithSources($color_from, $color_to); 323 } 324 325 /** 326 * @return string 327 */ 328 public function totalIndividualsPercentage(): string 329 { 330 return $this->individual_repository->totalIndividualsPercentage(); 331 } 332 333 /** 334 * @return string 335 */ 336 public function totalFamilies(): string 337 { 338 return $this->individual_repository->totalFamilies(); 339 } 340 341 /** 342 * @return string 343 */ 344 public function totalFamiliesPercentage(): string 345 { 346 return $this->individual_repository->totalFamiliesPercentage(); 347 } 348 349 /** 350 * @return string 351 */ 352 public function totalFamsWithSources(): string 353 { 354 return $this->individual_repository->totalFamsWithSources(); 355 } 356 357 /** 358 * @return string 359 */ 360 public function totalFamsWithSourcesPercentage(): string 361 { 362 return $this->individual_repository->totalFamsWithSourcesPercentage(); 363 } 364 365 /** 366 * @param string|null $color_from 367 * @param string|null $color_to 368 * 369 * @return string 370 */ 371 public function chartFamsWithSources( 372 string|null $color_from = null, 373 string $color_to = null 374 ): string { 375 return $this->individual_repository->chartFamsWithSources($color_from, $color_to); 376 } 377 378 /** 379 * @return string 380 */ 381 public function totalSources(): string 382 { 383 return $this->individual_repository->totalSources(); 384 } 385 386 /** 387 * @return string 388 */ 389 public function totalSourcesPercentage(): string 390 { 391 return $this->individual_repository->totalSourcesPercentage(); 392 } 393 394 /** 395 * @return string 396 */ 397 public function totalNotes(): string 398 { 399 return $this->individual_repository->totalNotes(); 400 } 401 402 /** 403 * @return string 404 */ 405 public function totalNotesPercentage(): string 406 { 407 return $this->individual_repository->totalNotesPercentage(); 408 } 409 410 /** 411 * @return string 412 */ 413 public function totalRepositories(): string 414 { 415 return $this->individual_repository->totalRepositories(); 416 } 417 418 /** 419 * @return string 420 */ 421 public function totalRepositoriesPercentage(): string 422 { 423 return $this->individual_repository->totalRepositoriesPercentage(); 424 } 425 426 /** 427 * @param array<string> ...$params 428 * 429 * @return string 430 */ 431 public function totalSurnames(...$params): string 432 { 433 return $this->individual_repository->totalSurnames(...$params); 434 } 435 436 /** 437 * @param array<string> ...$params 438 * 439 * @return string 440 */ 441 public function totalGivennames(...$params): string 442 { 443 return $this->individual_repository->totalGivennames(...$params); 444 } 445 446 /** 447 * @param array<string> $events 448 * 449 * @return string 450 */ 451 public function totalEvents(array $events = []): string 452 { 453 return $this->event_repository->totalEvents($events); 454 } 455 456 /** 457 * @return string 458 */ 459 public function totalEventsBirth(): string 460 { 461 return $this->event_repository->totalEventsBirth(); 462 } 463 464 /** 465 * @return string 466 */ 467 public function totalBirths(): string 468 { 469 return $this->event_repository->totalBirths(); 470 } 471 472 /** 473 * @return string 474 */ 475 public function totalEventsDeath(): string 476 { 477 return $this->event_repository->totalEventsDeath(); 478 } 479 480 /** 481 * @return string 482 */ 483 public function totalDeaths(): string 484 { 485 return $this->event_repository->totalDeaths(); 486 } 487 488 /** 489 * @return string 490 */ 491 public function totalEventsMarriage(): string 492 { 493 return $this->event_repository->totalEventsMarriage(); 494 } 495 496 /** 497 * @return string 498 */ 499 public function totalMarriages(): string 500 { 501 return $this->event_repository->totalMarriages(); 502 } 503 504 /** 505 * @return string 506 */ 507 public function totalEventsDivorce(): string 508 { 509 return $this->event_repository->totalEventsDivorce(); 510 } 511 512 /** 513 * @return string 514 */ 515 public function totalDivorces(): string 516 { 517 return $this->event_repository->totalDivorces(); 518 } 519 520 /** 521 * @return string 522 */ 523 public function totalEventsOther(): string 524 { 525 return $this->event_repository->totalEventsOther(); 526 } 527 528 /** 529 * @return string 530 */ 531 public function totalSexMales(): string 532 { 533 return $this->individual_repository->totalSexMales(); 534 } 535 536 /** 537 * @return string 538 */ 539 public function totalSexMalesPercentage(): string 540 { 541 return $this->individual_repository->totalSexMalesPercentage(); 542 } 543 544 /** 545 * @return string 546 */ 547 public function totalSexFemales(): string 548 { 549 return $this->individual_repository->totalSexFemales(); 550 } 551 552 /** 553 * @return string 554 */ 555 public function totalSexFemalesPercentage(): string 556 { 557 return $this->individual_repository->totalSexFemalesPercentage(); 558 } 559 560 /** 561 * @return string 562 */ 563 public function totalSexUnknown(): string 564 { 565 return $this->individual_repository->totalSexUnknown(); 566 } 567 568 /** 569 * @return string 570 */ 571 public function totalSexUnknownPercentage(): string 572 { 573 return $this->individual_repository->totalSexUnknownPercentage(); 574 } 575 576 /** 577 * @param string|null $color_female 578 * @param string|null $color_male 579 * @param string|null $color_unknown 580 * 581 * @return string 582 */ 583 public function chartSex( 584 string|null $color_female = null, 585 string|null $color_male = null, 586 string $color_unknown = null 587 ): string { 588 return $this->individual_repository->chartSex($color_female, $color_male, $color_unknown); 589 } 590 591 /** 592 * @return string 593 */ 594 public function totalLiving(): string 595 { 596 return $this->individual_repository->totalLiving(); 597 } 598 599 /** 600 * @return string 601 */ 602 public function totalLivingPercentage(): string 603 { 604 return $this->individual_repository->totalLivingPercentage(); 605 } 606 607 /** 608 * @return string 609 */ 610 public function totalDeceased(): string 611 { 612 return $this->individual_repository->totalDeceased(); 613 } 614 615 /** 616 * @return string 617 */ 618 public function totalDeceasedPercentage(): string 619 { 620 return $this->individual_repository->totalDeceasedPercentage(); 621 } 622 623 /** 624 * @param string|null $color_living 625 * @param string|null $color_dead 626 * 627 * @return string 628 */ 629 public function chartMortality(string|null $color_living = null, string|null $color_dead = null): string 630 { 631 return $this->individual_repository->chartMortality($color_living, $color_dead); 632 } 633 634 /** 635 * @return string 636 */ 637 public function totalMedia(): string 638 { 639 return $this->media_repository->totalMedia(); 640 } 641 642 /** 643 * @return string 644 */ 645 public function totalMediaAudio(): string 646 { 647 return $this->media_repository->totalMediaAudio(); 648 } 649 650 /** 651 * @return string 652 */ 653 public function totalMediaBook(): string 654 { 655 return $this->media_repository->totalMediaBook(); 656 } 657 658 /** 659 * @return string 660 */ 661 public function totalMediaCard(): string 662 { 663 return $this->media_repository->totalMediaCard(); 664 } 665 666 /** 667 * @return string 668 */ 669 public function totalMediaCertificate(): string 670 { 671 return $this->media_repository->totalMediaCertificate(); 672 } 673 674 /** 675 * @return string 676 */ 677 public function totalMediaCoatOfArms(): string 678 { 679 return $this->media_repository->totalMediaCoatOfArms(); 680 } 681 682 /** 683 * @return string 684 */ 685 public function totalMediaDocument(): string 686 { 687 return $this->media_repository->totalMediaDocument(); 688 } 689 690 /** 691 * @return string 692 */ 693 public function totalMediaElectronic(): string 694 { 695 return $this->media_repository->totalMediaElectronic(); 696 } 697 698 /** 699 * @return string 700 */ 701 public function totalMediaMagazine(): string 702 { 703 return $this->media_repository->totalMediaMagazine(); 704 } 705 706 /** 707 * @return string 708 */ 709 public function totalMediaManuscript(): string 710 { 711 return $this->media_repository->totalMediaManuscript(); 712 } 713 714 /** 715 * @return string 716 */ 717 public function totalMediaMap(): string 718 { 719 return $this->media_repository->totalMediaMap(); 720 } 721 722 /** 723 * @return string 724 */ 725 public function totalMediaFiche(): string 726 { 727 return $this->media_repository->totalMediaFiche(); 728 } 729 730 /** 731 * @return string 732 */ 733 public function totalMediaFilm(): string 734 { 735 return $this->media_repository->totalMediaFilm(); 736 } 737 738 /** 739 * @return string 740 */ 741 public function totalMediaNewspaper(): string 742 { 743 return $this->media_repository->totalMediaNewspaper(); 744 } 745 746 /** 747 * @return string 748 */ 749 public function totalMediaPainting(): string 750 { 751 return $this->media_repository->totalMediaPainting(); 752 } 753 754 /** 755 * @return string 756 */ 757 public function totalMediaPhoto(): string 758 { 759 return $this->media_repository->totalMediaPhoto(); 760 } 761 762 /** 763 * @return string 764 */ 765 public function totalMediaTombstone(): string 766 { 767 return $this->media_repository->totalMediaTombstone(); 768 } 769 770 /** 771 * @return string 772 */ 773 public function totalMediaVideo(): string 774 { 775 return $this->media_repository->totalMediaVideo(); 776 } 777 778 /** 779 * @return string 780 */ 781 public function totalMediaOther(): string 782 { 783 return $this->media_repository->totalMediaOther(); 784 } 785 786 /** 787 * @return string 788 */ 789 public function totalMediaUnknown(): string 790 { 791 return $this->media_repository->totalMediaUnknown(); 792 } 793 794 /** 795 * @param string|null $color_from 796 * @param string|null $color_to 797 * 798 * @return string 799 */ 800 public function chartMedia(string|null $color_from = null, string|null $color_to = null): string 801 { 802 return $this->media_repository->chartMedia($color_from, $color_to); 803 } 804 805 /** 806 * @return string 807 */ 808 public function totalPlaces(): string 809 { 810 return $this->place_repository->totalPlaces(); 811 } 812 813 /** 814 * @param string $chart_shows 815 * @param string $chart_type 816 * @param string $surname 817 * 818 * @return string 819 */ 820 public function chartDistribution( 821 string $chart_shows = 'world', 822 string $chart_type = '', 823 string $surname = '' 824 ): string { 825 return $this->place_repository->chartDistribution($chart_shows, $chart_type, $surname); 826 } 827 828 /** 829 * @return string 830 */ 831 public function commonCountriesList(): string 832 { 833 return $this->place_repository->commonCountriesList(); 834 } 835 836 /** 837 * @return string 838 */ 839 public function commonBirthPlacesList(): string 840 { 841 return $this->place_repository->commonBirthPlacesList(); 842 } 843 844 /** 845 * @return string 846 */ 847 public function commonDeathPlacesList(): string 848 { 849 return $this->place_repository->commonDeathPlacesList(); 850 } 851 852 /** 853 * @return string 854 */ 855 public function commonMarriagePlacesList(): string 856 { 857 return $this->place_repository->commonMarriagePlacesList(); 858 } 859 860 /** 861 * @return string 862 */ 863 public function firstBirth(): string 864 { 865 return $this->family_dates_repository->firstBirth(); 866 } 867 868 /** 869 * @return string 870 */ 871 public function firstBirthYear(): string 872 { 873 return $this->family_dates_repository->firstBirthYear(); 874 } 875 876 /** 877 * @return string 878 */ 879 public function firstBirthName(): string 880 { 881 return $this->family_dates_repository->firstBirthName(); 882 } 883 884 /** 885 * @return string 886 */ 887 public function firstBirthPlace(): string 888 { 889 return $this->family_dates_repository->firstBirthPlace(); 890 } 891 892 /** 893 * @return string 894 */ 895 public function lastBirth(): string 896 { 897 return $this->family_dates_repository->lastBirth(); 898 } 899 900 /** 901 * @return string 902 */ 903 public function lastBirthYear(): string 904 { 905 return $this->family_dates_repository->lastBirthYear(); 906 } 907 908 /** 909 * @return string 910 */ 911 public function lastBirthName(): string 912 { 913 return $this->family_dates_repository->lastBirthName(); 914 } 915 916 /** 917 * @return string 918 */ 919 public function lastBirthPlace(): string 920 { 921 return $this->family_dates_repository->lastBirthPlace(); 922 } 923 924 /** 925 * @param int $year1 926 * @param int $year2 927 * 928 * @return Builder 929 */ 930 public function statsBirthQuery(int $year1 = -1, int $year2 = -1): Builder 931 { 932 return $this->individual_repository->statsBirthQuery($year1, $year2); 933 } 934 935 /** 936 * @param int $year1 937 * @param int $year2 938 * 939 * @return Builder 940 */ 941 public function statsBirthBySexQuery(int $year1 = -1, int $year2 = -1): Builder 942 { 943 return $this->individual_repository->statsBirthBySexQuery($year1, $year2); 944 } 945 946 /** 947 * @param string|null $color_from 948 * @param string|null $color_to 949 * 950 * @return string 951 */ 952 public function statsBirth(string|null $color_from = null, string|null $color_to = null): string 953 { 954 return $this->individual_repository->statsBirth($color_from, $color_to); 955 } 956 957 /** 958 * @return string 959 */ 960 public function firstDeath(): string 961 { 962 return $this->family_dates_repository->firstDeath(); 963 } 964 965 /** 966 * @return string 967 */ 968 public function firstDeathYear(): string 969 { 970 return $this->family_dates_repository->firstDeathYear(); 971 } 972 973 /** 974 * @return string 975 */ 976 public function firstDeathName(): string 977 { 978 return $this->family_dates_repository->firstDeathName(); 979 } 980 981 /** 982 * @return string 983 */ 984 public function firstDeathPlace(): string 985 { 986 return $this->family_dates_repository->firstDeathPlace(); 987 } 988 989 /** 990 * @return string 991 */ 992 public function lastDeath(): string 993 { 994 return $this->family_dates_repository->lastDeath(); 995 } 996 997 /** 998 * @return string 999 */ 1000 public function lastDeathYear(): string 1001 { 1002 return $this->family_dates_repository->lastDeathYear(); 1003 } 1004 1005 /** 1006 * @return string 1007 */ 1008 public function lastDeathName(): string 1009 { 1010 return $this->family_dates_repository->lastDeathName(); 1011 } 1012 1013 /** 1014 * @return string 1015 */ 1016 public function lastDeathPlace(): string 1017 { 1018 return $this->family_dates_repository->lastDeathPlace(); 1019 } 1020 1021 /** 1022 * @param int $year1 1023 * @param int $year2 1024 * 1025 * @return Builder 1026 */ 1027 public function statsDeathQuery(int $year1 = -1, int $year2 = -1): Builder 1028 { 1029 return $this->individual_repository->statsDeathQuery($year1, $year2); 1030 } 1031 1032 /** 1033 * @param int $year1 1034 * @param int $year2 1035 * 1036 * @return Builder 1037 */ 1038 public function statsDeathBySexQuery(int $year1 = -1, int $year2 = -1): Builder 1039 { 1040 return $this->individual_repository->statsDeathBySexQuery($year1, $year2); 1041 } 1042 1043 /** 1044 * @param string|null $color_from 1045 * @param string|null $color_to 1046 * 1047 * @return string 1048 */ 1049 public function statsDeath(string|null $color_from = null, string|null $color_to = null): string 1050 { 1051 return $this->individual_repository->statsDeath($color_from, $color_to); 1052 } 1053 1054 /** 1055 * General query on ages. 1056 * 1057 * @param string $related 1058 * @param string $sex 1059 * @param int $year1 1060 * @param int $year2 1061 * 1062 * @return array<array<stdClass>> 1063 */ 1064 public function statsAgeQuery(string $related = 'BIRT', string $sex = 'BOTH', int $year1 = -1, int $year2 = -1): array 1065 { 1066 return $this->individual_repository->statsAgeQuery($related, $sex, $year1, $year2); 1067 } 1068 1069 /** 1070 * @return string 1071 */ 1072 public function statsAge(): string 1073 { 1074 return $this->individual_repository->statsAge(); 1075 } 1076 1077 /** 1078 * @return string 1079 */ 1080 public function longestLife(): string 1081 { 1082 return $this->individual_repository->longestLife(); 1083 } 1084 1085 /** 1086 * @return string 1087 */ 1088 public function longestLifeAge(): string 1089 { 1090 return $this->individual_repository->longestLifeAge(); 1091 } 1092 1093 /** 1094 * @return string 1095 */ 1096 public function longestLifeName(): string 1097 { 1098 return $this->individual_repository->longestLifeName(); 1099 } 1100 1101 /** 1102 * @return string 1103 */ 1104 public function longestLifeFemale(): string 1105 { 1106 return $this->individual_repository->longestLifeFemale(); 1107 } 1108 1109 /** 1110 * @return string 1111 */ 1112 public function longestLifeFemaleAge(): string 1113 { 1114 return $this->individual_repository->longestLifeFemaleAge(); 1115 } 1116 1117 /** 1118 * @return string 1119 */ 1120 public function longestLifeFemaleName(): string 1121 { 1122 return $this->individual_repository->longestLifeFemaleName(); 1123 } 1124 1125 /** 1126 * @return string 1127 */ 1128 public function longestLifeMale(): string 1129 { 1130 return $this->individual_repository->longestLifeMale(); 1131 } 1132 1133 /** 1134 * @return string 1135 */ 1136 public function longestLifeMaleAge(): string 1137 { 1138 return $this->individual_repository->longestLifeMaleAge(); 1139 } 1140 1141 /** 1142 * @return string 1143 */ 1144 public function longestLifeMaleName(): string 1145 { 1146 return $this->individual_repository->longestLifeMaleName(); 1147 } 1148 1149 /** 1150 * @param string $total 1151 * 1152 * @return string 1153 */ 1154 public function topTenOldest(string $total = '10'): string 1155 { 1156 return $this->individual_repository->topTenOldest((int) $total); 1157 } 1158 1159 /** 1160 * @param string $total 1161 * 1162 * @return string 1163 */ 1164 public function topTenOldestList(string $total = '10'): string 1165 { 1166 return $this->individual_repository->topTenOldestList((int) $total); 1167 } 1168 1169 /** 1170 * @param string $total 1171 * 1172 * @return string 1173 */ 1174 public function topTenOldestFemale(string $total = '10'): string 1175 { 1176 return $this->individual_repository->topTenOldestFemale((int) $total); 1177 } 1178 1179 /** 1180 * @param string $total 1181 * 1182 * @return string 1183 */ 1184 public function topTenOldestFemaleList(string $total = '10'): string 1185 { 1186 return $this->individual_repository->topTenOldestFemaleList((int) $total); 1187 } 1188 1189 /** 1190 * @param string $total 1191 * 1192 * @return string 1193 */ 1194 public function topTenOldestMale(string $total = '10'): string 1195 { 1196 return $this->individual_repository->topTenOldestMale((int) $total); 1197 } 1198 1199 /** 1200 * @param string $total 1201 * 1202 * @return string 1203 */ 1204 public function topTenOldestMaleList(string $total = '10'): string 1205 { 1206 return $this->individual_repository->topTenOldestMaleList((int) $total); 1207 } 1208 1209 /** 1210 * @param string $total 1211 * 1212 * @return string 1213 */ 1214 public function topTenOldestAlive(string $total = '10'): string 1215 { 1216 return $this->individual_repository->topTenOldestAlive((int) $total); 1217 } 1218 1219 /** 1220 * @param string $total 1221 * 1222 * @return string 1223 */ 1224 public function topTenOldestListAlive(string $total = '10'): string 1225 { 1226 return $this->individual_repository->topTenOldestListAlive((int) $total); 1227 } 1228 1229 /** 1230 * @param string $total 1231 * 1232 * @return string 1233 */ 1234 public function topTenOldestFemaleAlive(string $total = '10'): string 1235 { 1236 return $this->individual_repository->topTenOldestFemaleAlive((int) $total); 1237 } 1238 1239 /** 1240 * @param string $total 1241 * 1242 * @return string 1243 */ 1244 public function topTenOldestFemaleListAlive(string $total = '10'): string 1245 { 1246 return $this->individual_repository->topTenOldestFemaleListAlive((int) $total); 1247 } 1248 1249 /** 1250 * @param string $total 1251 * 1252 * @return string 1253 */ 1254 public function topTenOldestMaleAlive(string $total = '10'): string 1255 { 1256 return $this->individual_repository->topTenOldestMaleAlive((int) $total); 1257 } 1258 1259 /** 1260 * @param string $total 1261 * 1262 * @return string 1263 */ 1264 public function topTenOldestMaleListAlive(string $total = '10'): string 1265 { 1266 return $this->individual_repository->topTenOldestMaleListAlive((int) $total); 1267 } 1268 1269 /** 1270 * @param string $show_years 1271 * 1272 * @return string 1273 */ 1274 public function averageLifespan(string $show_years = ''): string 1275 { 1276 return $this->individual_repository->averageLifespan((bool) $show_years); 1277 } 1278 1279 /** 1280 * @param string $show_years 1281 * 1282 * @return string 1283 */ 1284 public function averageLifespanFemale(string $show_years = ''): string 1285 { 1286 return $this->individual_repository->averageLifespanFemale((bool) $show_years); 1287 } 1288 1289 /** 1290 * @param string $show_years 1291 * 1292 * @return string 1293 */ 1294 public function averageLifespanMale(string $show_years = ''): string 1295 { 1296 return $this->individual_repository->averageLifespanMale((bool) $show_years); 1297 } 1298 1299 /** 1300 * @return string 1301 */ 1302 public function firstEvent(): string 1303 { 1304 return $this->event_repository->firstEvent(); 1305 } 1306 1307 /** 1308 * @return string 1309 */ 1310 public function firstEventYear(): string 1311 { 1312 return $this->event_repository->firstEventYear(); 1313 } 1314 1315 /** 1316 * @return string 1317 */ 1318 public function firstEventType(): string 1319 { 1320 return $this->event_repository->firstEventType(); 1321 } 1322 1323 /** 1324 * @return string 1325 */ 1326 public function firstEventName(): string 1327 { 1328 return $this->event_repository->firstEventName(); 1329 } 1330 1331 /** 1332 * @return string 1333 */ 1334 public function firstEventPlace(): string 1335 { 1336 return $this->event_repository->firstEventPlace(); 1337 } 1338 1339 /** 1340 * @return string 1341 */ 1342 public function lastEvent(): string 1343 { 1344 return $this->event_repository->lastEvent(); 1345 } 1346 1347 /** 1348 * @return string 1349 */ 1350 public function lastEventYear(): string 1351 { 1352 return $this->event_repository->lastEventYear(); 1353 } 1354 1355 /** 1356 * @return string 1357 */ 1358 public function lastEventType(): string 1359 { 1360 return $this->event_repository->lastEventType(); 1361 } 1362 1363 /** 1364 * @return string 1365 */ 1366 public function lastEventName(): string 1367 { 1368 return $this->event_repository->lastEventName(); 1369 } 1370 1371 /** 1372 * @return string 1373 */ 1374 public function lastEventPlace(): string 1375 { 1376 return $this->event_repository->lastEventPlace(); 1377 } 1378 1379 /** 1380 * @return string 1381 */ 1382 public function firstMarriage(): string 1383 { 1384 return $this->family_dates_repository->firstMarriage(); 1385 } 1386 1387 /** 1388 * @return string 1389 */ 1390 public function firstMarriageYear(): string 1391 { 1392 return $this->family_dates_repository->firstMarriageYear(); 1393 } 1394 1395 /** 1396 * @return string 1397 */ 1398 public function firstMarriageName(): string 1399 { 1400 return $this->family_dates_repository->firstMarriageName(); 1401 } 1402 1403 /** 1404 * @return string 1405 */ 1406 public function firstMarriagePlace(): string 1407 { 1408 return $this->family_dates_repository->firstMarriagePlace(); 1409 } 1410 1411 /** 1412 * @return string 1413 */ 1414 public function lastMarriage(): string 1415 { 1416 return $this->family_dates_repository->lastMarriage(); 1417 } 1418 1419 /** 1420 * @return string 1421 */ 1422 public function lastMarriageYear(): string 1423 { 1424 return $this->family_dates_repository->lastMarriageYear(); 1425 } 1426 1427 /** 1428 * @return string 1429 */ 1430 public function lastMarriageName(): string 1431 { 1432 return $this->family_dates_repository->lastMarriageName(); 1433 } 1434 1435 /** 1436 * @return string 1437 */ 1438 public function lastMarriagePlace(): string 1439 { 1440 return $this->family_dates_repository->lastMarriagePlace(); 1441 } 1442 1443 /** 1444 * @param int $year1 1445 * @param int $year2 1446 * 1447 * @return Builder 1448 */ 1449 public function statsMarriageQuery(int $year1 = -1, int $year2 = -1): Builder 1450 { 1451 return $this->family_repository->statsMarriageQuery($year1, $year2); 1452 } 1453 1454 /** 1455 * @param int $year1 1456 * @param int $year2 1457 * 1458 * @return Builder 1459 */ 1460 public function statsFirstMarriageQuery(int $year1 = -1, int $year2 = -1): Builder 1461 { 1462 return $this->family_repository->statsFirstMarriageQuery($year1, $year2); 1463 } 1464 1465 /** 1466 * @param string|null $color_from 1467 * @param string|null $color_to 1468 * 1469 * @return string 1470 */ 1471 public function statsMarr(string|null $color_from = null, string|null $color_to = null): string 1472 { 1473 return $this->family_repository->statsMarr($color_from, $color_to); 1474 } 1475 1476 /** 1477 * @return string 1478 */ 1479 public function firstDivorce(): string 1480 { 1481 return $this->family_dates_repository->firstDivorce(); 1482 } 1483 1484 /** 1485 * @return string 1486 */ 1487 public function firstDivorceYear(): string 1488 { 1489 return $this->family_dates_repository->firstDivorceYear(); 1490 } 1491 1492 /** 1493 * @return string 1494 */ 1495 public function firstDivorceName(): string 1496 { 1497 return $this->family_dates_repository->firstDivorceName(); 1498 } 1499 1500 /** 1501 * @return string 1502 */ 1503 public function firstDivorcePlace(): string 1504 { 1505 return $this->family_dates_repository->firstDivorcePlace(); 1506 } 1507 1508 /** 1509 * @return string 1510 */ 1511 public function lastDivorce(): string 1512 { 1513 return $this->family_dates_repository->lastDivorce(); 1514 } 1515 1516 /** 1517 * @return string 1518 */ 1519 public function lastDivorceYear(): string 1520 { 1521 return $this->family_dates_repository->lastDivorceYear(); 1522 } 1523 1524 /** 1525 * @return string 1526 */ 1527 public function lastDivorceName(): string 1528 { 1529 return $this->family_dates_repository->lastDivorceName(); 1530 } 1531 1532 /** 1533 * @return string 1534 */ 1535 public function lastDivorcePlace(): string 1536 { 1537 return $this->family_dates_repository->lastDivorcePlace(); 1538 } 1539 1540 /** 1541 * @param string|null $color_from 1542 * @param string|null $color_to 1543 * 1544 * @return string 1545 */ 1546 public function statsDiv(string|null $color_from = null, string|null $color_to = null): string 1547 { 1548 return $this->family_repository->statsDiv($color_from, $color_to); 1549 } 1550 1551 /** 1552 * @return string 1553 */ 1554 public function youngestMarriageFemale(): string 1555 { 1556 return $this->family_repository->youngestMarriageFemale(); 1557 } 1558 1559 /** 1560 * @return string 1561 */ 1562 public function youngestMarriageFemaleName(): string 1563 { 1564 return $this->family_repository->youngestMarriageFemaleName(); 1565 } 1566 1567 /** 1568 * @param string $show_years 1569 * 1570 * @return string 1571 */ 1572 public function youngestMarriageFemaleAge(string $show_years = ''): string 1573 { 1574 return $this->family_repository->youngestMarriageFemaleAge($show_years); 1575 } 1576 1577 /** 1578 * @return string 1579 */ 1580 public function oldestMarriageFemale(): string 1581 { 1582 return $this->family_repository->oldestMarriageFemale(); 1583 } 1584 1585 /** 1586 * @return string 1587 */ 1588 public function oldestMarriageFemaleName(): string 1589 { 1590 return $this->family_repository->oldestMarriageFemaleName(); 1591 } 1592 1593 /** 1594 * @param string $show_years 1595 * 1596 * @return string 1597 */ 1598 public function oldestMarriageFemaleAge(string $show_years = ''): string 1599 { 1600 return $this->family_repository->oldestMarriageFemaleAge($show_years); 1601 } 1602 1603 /** 1604 * @return string 1605 */ 1606 public function youngestMarriageMale(): string 1607 { 1608 return $this->family_repository->youngestMarriageMale(); 1609 } 1610 1611 /** 1612 * @return string 1613 */ 1614 public function youngestMarriageMaleName(): string 1615 { 1616 return $this->family_repository->youngestMarriageMaleName(); 1617 } 1618 1619 /** 1620 * @param string $show_years 1621 * 1622 * @return string 1623 */ 1624 public function youngestMarriageMaleAge(string $show_years = ''): string 1625 { 1626 return $this->family_repository->youngestMarriageMaleAge($show_years); 1627 } 1628 1629 /** 1630 * @return string 1631 */ 1632 public function oldestMarriageMale(): string 1633 { 1634 return $this->family_repository->oldestMarriageMale(); 1635 } 1636 1637 /** 1638 * @return string 1639 */ 1640 public function oldestMarriageMaleName(): string 1641 { 1642 return $this->family_repository->oldestMarriageMaleName(); 1643 } 1644 1645 /** 1646 * @param string $show_years 1647 * 1648 * @return string 1649 */ 1650 public function oldestMarriageMaleAge(string $show_years = ''): string 1651 { 1652 return $this->family_repository->oldestMarriageMaleAge($show_years); 1653 } 1654 1655 /** 1656 * @param string $sex 1657 * @param int $year1 1658 * @param int $year2 1659 * 1660 * @return array<stdClass> 1661 */ 1662 public function statsMarrAgeQuery(string $sex, int $year1 = -1, int $year2 = -1): array 1663 { 1664 return $this->family_repository->statsMarrAgeQuery($sex, $year1, $year2); 1665 } 1666 1667 /** 1668 * @return string 1669 */ 1670 public function statsMarrAge(): string 1671 { 1672 return $this->family_repository->statsMarrAge(); 1673 } 1674 1675 /** 1676 * @param string $total 1677 * 1678 * @return string 1679 */ 1680 public function ageBetweenSpousesMF(string $total = '10'): string 1681 { 1682 return $this->family_repository->ageBetweenSpousesMF((int) $total); 1683 } 1684 1685 /** 1686 * @param string $total 1687 * 1688 * @return string 1689 */ 1690 public function ageBetweenSpousesMFList(string $total = '10'): string 1691 { 1692 return $this->family_repository->ageBetweenSpousesMFList((int) $total); 1693 } 1694 1695 /** 1696 * @param string $total 1697 * 1698 * @return string 1699 */ 1700 public function ageBetweenSpousesFM(string $total = '10'): string 1701 { 1702 return $this->family_repository->ageBetweenSpousesFM((int) $total); 1703 } 1704 1705 /** 1706 * @param string $total 1707 * 1708 * @return string 1709 */ 1710 public function ageBetweenSpousesFMList(string $total = '10'): string 1711 { 1712 return $this->family_repository->ageBetweenSpousesFMList((int) $total); 1713 } 1714 1715 /** 1716 * @return string 1717 */ 1718 public function topAgeOfMarriageFamily(): string 1719 { 1720 return $this->family_repository->topAgeOfMarriageFamily(); 1721 } 1722 1723 /** 1724 * @return string 1725 */ 1726 public function topAgeOfMarriage(): string 1727 { 1728 return $this->family_repository->topAgeOfMarriage(); 1729 } 1730 1731 /** 1732 * @param string $total 1733 * 1734 * @return string 1735 */ 1736 public function topAgeOfMarriageFamilies(string $total = '10'): string 1737 { 1738 return $this->family_repository->topAgeOfMarriageFamilies((int) $total); 1739 } 1740 1741 /** 1742 * @param string $total 1743 * 1744 * @return string 1745 */ 1746 public function topAgeOfMarriageFamiliesList(string $total = '10'): string 1747 { 1748 return $this->family_repository->topAgeOfMarriageFamiliesList((int) $total); 1749 } 1750 1751 /** 1752 * @return string 1753 */ 1754 public function minAgeOfMarriageFamily(): string 1755 { 1756 return $this->family_repository->minAgeOfMarriageFamily(); 1757 } 1758 1759 /** 1760 * @return string 1761 */ 1762 public function minAgeOfMarriage(): string 1763 { 1764 return $this->family_repository->minAgeOfMarriage(); 1765 } 1766 1767 /** 1768 * @param string $total 1769 * 1770 * @return string 1771 */ 1772 public function minAgeOfMarriageFamilies(string $total = '10'): string 1773 { 1774 return $this->family_repository->minAgeOfMarriageFamilies((int) $total); 1775 } 1776 1777 /** 1778 * @param string $total 1779 * 1780 * @return string 1781 */ 1782 public function minAgeOfMarriageFamiliesList(string $total = '10'): string 1783 { 1784 return $this->family_repository->minAgeOfMarriageFamiliesList((int) $total); 1785 } 1786 1787 /** 1788 * @return string 1789 */ 1790 public function youngestMother(): string 1791 { 1792 return $this->family_repository->youngestMother(); 1793 } 1794 1795 /** 1796 * @return string 1797 */ 1798 public function youngestMotherName(): string 1799 { 1800 return $this->family_repository->youngestMotherName(); 1801 } 1802 1803 /** 1804 * @param string $show_years 1805 * 1806 * @return string 1807 */ 1808 public function youngestMotherAge(string $show_years = ''): string 1809 { 1810 return $this->family_repository->youngestMotherAge($show_years); 1811 } 1812 1813 /** 1814 * @return string 1815 */ 1816 public function oldestMother(): string 1817 { 1818 return $this->family_repository->oldestMother(); 1819 } 1820 1821 /** 1822 * @return string 1823 */ 1824 public function oldestMotherName(): string 1825 { 1826 return $this->family_repository->oldestMotherName(); 1827 } 1828 1829 /** 1830 * @param string $show_years 1831 * 1832 * @return string 1833 */ 1834 public function oldestMotherAge(string $show_years = ''): string 1835 { 1836 return $this->family_repository->oldestMotherAge($show_years); 1837 } 1838 1839 /** 1840 * @return string 1841 */ 1842 public function youngestFather(): string 1843 { 1844 return $this->family_repository->youngestFather(); 1845 } 1846 1847 /** 1848 * @return string 1849 */ 1850 public function youngestFatherName(): string 1851 { 1852 return $this->family_repository->youngestFatherName(); 1853 } 1854 1855 /** 1856 * @param string $show_years 1857 * 1858 * @return string 1859 */ 1860 public function youngestFatherAge(string $show_years = ''): string 1861 { 1862 return $this->family_repository->youngestFatherAge($show_years); 1863 } 1864 1865 /** 1866 * @return string 1867 */ 1868 public function oldestFather(): string 1869 { 1870 return $this->family_repository->oldestFather(); 1871 } 1872 1873 /** 1874 * @return string 1875 */ 1876 public function oldestFatherName(): string 1877 { 1878 return $this->family_repository->oldestFatherName(); 1879 } 1880 1881 /** 1882 * @param string $show_years 1883 * 1884 * @return string 1885 */ 1886 public function oldestFatherAge(string $show_years = ''): string 1887 { 1888 return $this->family_repository->oldestFatherAge($show_years); 1889 } 1890 1891 /** 1892 * @return string 1893 */ 1894 public function totalMarriedMales(): string 1895 { 1896 return $this->family_repository->totalMarriedMales(); 1897 } 1898 1899 /** 1900 * @return string 1901 */ 1902 public function totalMarriedFemales(): string 1903 { 1904 return $this->family_repository->totalMarriedFemales(); 1905 } 1906 1907 /** 1908 * @param int $year1 1909 * @param int $year2 1910 * 1911 * @return Builder 1912 */ 1913 public function monthFirstChildQuery(int $year1 = -1, int $year2 = -1): Builder 1914 { 1915 return $this->family_repository->monthFirstChildQuery($year1, $year2); 1916 } 1917 1918 /** 1919 * @param int $year1 1920 * @param int $year2 1921 * 1922 * @return Builder 1923 */ 1924 public function monthFirstChildBySexQuery(int $year1 = -1, int $year2 = -1): Builder 1925 { 1926 return $this->family_repository->monthFirstChildBySexQuery($year1, $year2); 1927 } 1928 1929 /** 1930 * @return string 1931 */ 1932 public function largestFamily(): string 1933 { 1934 return $this->family_repository->largestFamily(); 1935 } 1936 1937 /** 1938 * @return string 1939 */ 1940 public function largestFamilySize(): string 1941 { 1942 return $this->family_repository->largestFamilySize(); 1943 } 1944 1945 /** 1946 * @return string 1947 */ 1948 public function largestFamilyName(): string 1949 { 1950 return $this->family_repository->largestFamilyName(); 1951 } 1952 1953 /** 1954 * @param string $total 1955 * 1956 * @return string 1957 */ 1958 public function topTenLargestFamily(string $total = '10'): string 1959 { 1960 return $this->family_repository->topTenLargestFamily((int) $total); 1961 } 1962 1963 /** 1964 * @param string $total 1965 * 1966 * @return string 1967 */ 1968 public function topTenLargestFamilyList(string $total = '10'): string 1969 { 1970 return $this->family_repository->topTenLargestFamilyList((int) $total); 1971 } 1972 1973 /** 1974 * @param string|null $color_from 1975 * @param string|null $color_to 1976 * @param string $total 1977 * 1978 * @return string 1979 */ 1980 public function chartLargestFamilies( 1981 string|null $color_from = null, 1982 string|null $color_to = null, 1983 string $total = '10' 1984 ): string { 1985 return $this->family_repository->chartLargestFamilies($color_from, $color_to, (int) $total); 1986 } 1987 1988 /** 1989 * @return string 1990 */ 1991 public function totalChildren(): string 1992 { 1993 return $this->family_repository->totalChildren(); 1994 } 1995 1996 /** 1997 * @return string 1998 */ 1999 public function averageChildren(): string 2000 { 2001 return $this->family_repository->averageChildren(); 2002 } 2003 2004 /** 2005 * @param int $year1 2006 * @param int $year2 2007 * 2008 * @return array<stdClass> 2009 */ 2010 public function statsChildrenQuery(int $year1 = -1, int $year2 = -1): array 2011 { 2012 return $this->family_repository->statsChildrenQuery($year1, $year2); 2013 } 2014 2015 /** 2016 * @return string 2017 */ 2018 public function statsChildren(): string 2019 { 2020 return $this->family_repository->statsChildren(); 2021 } 2022 2023 /** 2024 * @param string $total 2025 * 2026 * @return string 2027 */ 2028 public function topAgeBetweenSiblingsName(string $total = '10'): string 2029 { 2030 return $this->family_repository->topAgeBetweenSiblingsName((int) $total); 2031 } 2032 2033 /** 2034 * @param string $total 2035 * 2036 * @return string 2037 */ 2038 public function topAgeBetweenSiblings(string $total = '10'): string 2039 { 2040 return $this->family_repository->topAgeBetweenSiblings((int) $total); 2041 } 2042 2043 /** 2044 * @param string $total 2045 * 2046 * @return string 2047 */ 2048 public function topAgeBetweenSiblingsFullName(string $total = '10'): string 2049 { 2050 return $this->family_repository->topAgeBetweenSiblingsFullName((int) $total); 2051 } 2052 2053 /** 2054 * @param string $total 2055 * @param string $one 2056 * 2057 * @return string 2058 */ 2059 public function topAgeBetweenSiblingsList(string $total = '10', string $one = ''): string 2060 { 2061 return $this->family_repository->topAgeBetweenSiblingsList((int) $total, $one); 2062 } 2063 2064 /** 2065 * @return string 2066 */ 2067 public function noChildrenFamilies(): string 2068 { 2069 return $this->family_repository->noChildrenFamilies(); 2070 } 2071 2072 /** 2073 * @param string $type 2074 * 2075 * @return string 2076 */ 2077 public function noChildrenFamiliesList(string $type = 'list'): string 2078 { 2079 return $this->family_repository->noChildrenFamiliesList($type); 2080 } 2081 2082 /** 2083 * @param string $year1 2084 * @param string $year2 2085 * 2086 * @return string 2087 */ 2088 public function chartNoChildrenFamilies( 2089 string $year1 = '-1', 2090 string $year2 = '-1' 2091 ): string { 2092 return $this->family_repository->chartNoChildrenFamilies((int) $year1, (int) $year2); 2093 } 2094 2095 /** 2096 * @param string $total 2097 * 2098 * @return string 2099 */ 2100 public function topTenLargestGrandFamily(string $total = '10'): string 2101 { 2102 return $this->family_repository->topTenLargestGrandFamily((int) $total); 2103 } 2104 2105 /** 2106 * @param string $total 2107 * 2108 * @return string 2109 */ 2110 public function topTenLargestGrandFamilyList(string $total = '10'): string 2111 { 2112 return $this->family_repository->topTenLargestGrandFamilyList((int) $total); 2113 } 2114 2115 /** 2116 * @return string 2117 */ 2118 public function getCommonSurname(): string 2119 { 2120 return $this->individual_repository->getCommonSurname(); 2121 } 2122 2123 /** 2124 * @param string $threshold 2125 * @param string $number_of_surnames 2126 * @param string $sorting 2127 * 2128 * @return string 2129 */ 2130 public function commonSurnames( 2131 string $threshold = '1', 2132 string $number_of_surnames = '10', 2133 string $sorting = 'alpha' 2134 ): string { 2135 return $this->individual_repository->commonSurnames((int) $threshold, (int) $number_of_surnames, $sorting); 2136 } 2137 2138 /** 2139 * @param string $threshold 2140 * @param string $number_of_surnames 2141 * @param string $sorting 2142 * 2143 * @return string 2144 */ 2145 public function commonSurnamesTotals( 2146 string $threshold = '1', 2147 string $number_of_surnames = '10', 2148 string $sorting = 'count' 2149 ): string { 2150 return $this->individual_repository->commonSurnamesTotals((int) $threshold, (int) $number_of_surnames, $sorting); 2151 } 2152 2153 /** 2154 * @param string $threshold 2155 * @param string $number_of_surnames 2156 * @param string $sorting 2157 * 2158 * @return string 2159 */ 2160 public function commonSurnamesList( 2161 string $threshold = '1', 2162 string $number_of_surnames = '10', 2163 string $sorting = 'alpha' 2164 ): string { 2165 return $this->individual_repository->commonSurnamesList((int) $threshold, (int) $number_of_surnames, $sorting); 2166 } 2167 2168 /** 2169 * @param string $threshold 2170 * @param string $number_of_surnames 2171 * @param string $sorting 2172 * 2173 * @return string 2174 */ 2175 public function commonSurnamesListTotals( 2176 string $threshold = '1', 2177 string $number_of_surnames = '10', 2178 string $sorting = 'count' 2179 ): string { 2180 return $this->individual_repository 2181 ->commonSurnamesListTotals((int) $threshold, (int) $number_of_surnames, $sorting); 2182 } 2183 2184 /** 2185 * @param string|null $color_from 2186 * @param string|null $color_to 2187 * @param string $number_of_surnames 2188 * 2189 * @return string 2190 */ 2191 public function chartCommonSurnames( 2192 string|null $color_from = null, 2193 string|null $color_to = null, 2194 string $number_of_surnames = '10' 2195 ): string { 2196 return $this->individual_repository 2197 ->chartCommonSurnames($color_from, $color_to, (int) $number_of_surnames); 2198 } 2199 2200 /** 2201 * @param string $threshold 2202 * @param string $maxtoshow 2203 * 2204 * @return string 2205 */ 2206 public function commonGiven(string $threshold = '1', string $maxtoshow = '10'): string 2207 { 2208 return $this->individual_repository->commonGiven((int) $threshold, (int) $maxtoshow); 2209 } 2210 2211 /** 2212 * @param string $threshold 2213 * @param string $maxtoshow 2214 * 2215 * @return string 2216 */ 2217 public function commonGivenTotals(string $threshold = '1', string $maxtoshow = '10'): string 2218 { 2219 return $this->individual_repository->commonGivenTotals((int) $threshold, (int) $maxtoshow); 2220 } 2221 2222 /** 2223 * @param string $threshold 2224 * @param string $maxtoshow 2225 * 2226 * @return string 2227 */ 2228 public function commonGivenList(string $threshold = '1', string $maxtoshow = '10'): string 2229 { 2230 return $this->individual_repository->commonGivenList((int) $threshold, (int) $maxtoshow); 2231 } 2232 2233 /** 2234 * @param string $threshold 2235 * @param string $maxtoshow 2236 * 2237 * @return string 2238 */ 2239 public function commonGivenListTotals(string $threshold = '1', string $maxtoshow = '10'): string 2240 { 2241 return $this->individual_repository->commonGivenListTotals((int) $threshold, (int) $maxtoshow); 2242 } 2243 2244 /** 2245 * @param string $threshold 2246 * @param string $maxtoshow 2247 * 2248 * @return string 2249 */ 2250 public function commonGivenTable(string $threshold = '1', string $maxtoshow = '10'): string 2251 { 2252 return $this->individual_repository->commonGivenTable((int) $threshold, (int) $maxtoshow); 2253 } 2254 2255 /** 2256 * @param string $threshold 2257 * @param string $maxtoshow 2258 * 2259 * @return string 2260 */ 2261 public function commonGivenFemale(string $threshold = '1', string $maxtoshow = '10'): string 2262 { 2263 return $this->individual_repository->commonGivenFemale((int) $threshold, (int) $maxtoshow); 2264 } 2265 2266 /** 2267 * @param string $threshold 2268 * @param string $maxtoshow 2269 * 2270 * @return string 2271 */ 2272 public function commonGivenFemaleTotals(string $threshold = '1', string $maxtoshow = '10'): string 2273 { 2274 return $this->individual_repository->commonGivenFemaleTotals((int) $threshold, (int) $maxtoshow); 2275 } 2276 2277 /** 2278 * @param string $threshold 2279 * @param string $maxtoshow 2280 * 2281 * @return string 2282 */ 2283 public function commonGivenFemaleList(string $threshold = '1', string $maxtoshow = '10'): string 2284 { 2285 return $this->individual_repository->commonGivenFemaleList((int) $threshold, (int) $maxtoshow); 2286 } 2287 2288 /** 2289 * @param string $threshold 2290 * @param string $maxtoshow 2291 * 2292 * @return string 2293 */ 2294 public function commonGivenFemaleListTotals(string $threshold = '1', string $maxtoshow = '10'): string 2295 { 2296 return $this->individual_repository->commonGivenFemaleListTotals((int) $threshold, (int) $maxtoshow); 2297 } 2298 2299 /** 2300 * @param string $threshold 2301 * @param string $maxtoshow 2302 * 2303 * @return string 2304 */ 2305 public function commonGivenFemaleTable(string $threshold = '1', string $maxtoshow = '10'): string 2306 { 2307 return $this->individual_repository->commonGivenFemaleTable((int) $threshold, (int) $maxtoshow); 2308 } 2309 2310 /** 2311 * @param string $threshold 2312 * @param string $maxtoshow 2313 * 2314 * @return string 2315 */ 2316 public function commonGivenMale(string $threshold = '1', string $maxtoshow = '10'): string 2317 { 2318 return $this->individual_repository->commonGivenMale((int) $threshold, (int) $maxtoshow); 2319 } 2320 2321 /** 2322 * @param string $threshold 2323 * @param string $maxtoshow 2324 * 2325 * @return string 2326 */ 2327 public function commonGivenMaleTotals(string $threshold = '1', string $maxtoshow = '10'): string 2328 { 2329 return $this->individual_repository->commonGivenMaleTotals((int) $threshold, (int) $maxtoshow); 2330 } 2331 2332 /** 2333 * @param string $threshold 2334 * @param string $maxtoshow 2335 * 2336 * @return string 2337 */ 2338 public function commonGivenMaleList(string $threshold = '1', string $maxtoshow = '10'): string 2339 { 2340 return $this->individual_repository->commonGivenMaleList((int) $threshold, (int) $maxtoshow); 2341 } 2342 2343 /** 2344 * @param string $threshold 2345 * @param string $maxtoshow 2346 * 2347 * @return string 2348 */ 2349 public function commonGivenMaleListTotals(string $threshold = '1', string $maxtoshow = '10'): string 2350 { 2351 return $this->individual_repository->commonGivenMaleListTotals((int) $threshold, (int) $maxtoshow); 2352 } 2353 2354 /** 2355 * @param string $threshold 2356 * @param string $maxtoshow 2357 * 2358 * @return string 2359 */ 2360 public function commonGivenMaleTable(string $threshold = '1', string $maxtoshow = '10'): string 2361 { 2362 return $this->individual_repository->commonGivenMaleTable((int) $threshold, (int) $maxtoshow); 2363 } 2364 2365 /** 2366 * @param string $threshold 2367 * @param string $maxtoshow 2368 * 2369 * @return string 2370 */ 2371 public function commonGivenUnknown(string $threshold = '1', string $maxtoshow = '10'): string 2372 { 2373 return $this->individual_repository->commonGivenUnknown((int) $threshold, (int) $maxtoshow); 2374 } 2375 2376 /** 2377 * @param string $threshold 2378 * @param string $maxtoshow 2379 * 2380 * @return string 2381 */ 2382 public function commonGivenUnknownTotals(string $threshold = '1', string $maxtoshow = '10'): string 2383 { 2384 return $this->individual_repository->commonGivenUnknownTotals((int) $threshold, (int) $maxtoshow); 2385 } 2386 2387 /** 2388 * @param string $threshold 2389 * @param string $maxtoshow 2390 * 2391 * @return string 2392 */ 2393 public function commonGivenUnknownList(string $threshold = '1', string $maxtoshow = '10'): string 2394 { 2395 return $this->individual_repository->commonGivenUnknownList((int) $threshold, (int) $maxtoshow); 2396 } 2397 2398 /** 2399 * @param string $threshold 2400 * @param string $maxtoshow 2401 * 2402 * @return string 2403 */ 2404 public function commonGivenUnknownListTotals(string $threshold = '1', string $maxtoshow = '10'): string 2405 { 2406 return $this->individual_repository->commonGivenUnknownListTotals((int) $threshold, (int) $maxtoshow); 2407 } 2408 2409 /** 2410 * @param string $threshold 2411 * @param string $maxtoshow 2412 * 2413 * @return string 2414 */ 2415 public function commonGivenUnknownTable(string $threshold = '1', string $maxtoshow = '10'): string 2416 { 2417 return $this->individual_repository->commonGivenUnknownTable((int) $threshold, (int) $maxtoshow); 2418 } 2419 2420 /** 2421 * @param string|null $color_from 2422 * @param string|null $color_to 2423 * @param string $maxtoshow 2424 * 2425 * @return string 2426 */ 2427 public function chartCommonGiven( 2428 string|null $color_from = null, 2429 string|null $color_to = null, 2430 string $maxtoshow = '7' 2431 ): string { 2432 return $this->individual_repository->chartCommonGiven($color_from, $color_to, (int) $maxtoshow); 2433 } 2434 2435 /** 2436 * @return string 2437 */ 2438 public function usersLoggedIn(): string 2439 { 2440 return $this->user_repository->usersLoggedIn(); 2441 } 2442 2443 /** 2444 * @return string 2445 */ 2446 public function usersLoggedInList(): string 2447 { 2448 return $this->user_repository->usersLoggedInList(); 2449 } 2450 2451 /** 2452 * @return int 2453 */ 2454 public function usersLoggedInTotal(): int 2455 { 2456 return $this->user_repository->usersLoggedInTotal(); 2457 } 2458 2459 /** 2460 * @return int 2461 */ 2462 public function usersLoggedInTotalAnon(): int 2463 { 2464 return $this->user_repository->usersLoggedInTotalAnon(); 2465 } 2466 2467 /** 2468 * @return int 2469 */ 2470 public function usersLoggedInTotalVisible(): int 2471 { 2472 return $this->user_repository->usersLoggedInTotalVisible(); 2473 } 2474 2475 /** 2476 * @return string 2477 */ 2478 public function userId(): string 2479 { 2480 return $this->user_repository->userId(); 2481 } 2482 2483 /** 2484 * @param string $visitor_text 2485 * 2486 * @return string 2487 */ 2488 public function userName(string $visitor_text = ''): string 2489 { 2490 return $this->user_repository->userName($visitor_text); 2491 } 2492 2493 /** 2494 * @return string 2495 */ 2496 public function userFullName(): string 2497 { 2498 return $this->user_repository->userFullName(); 2499 } 2500 2501 /** 2502 * @return string 2503 */ 2504 public function totalUsers(): string 2505 { 2506 return $this->user_repository->totalUsers(); 2507 } 2508 2509 /** 2510 * @return string 2511 */ 2512 public function totalAdmins(): string 2513 { 2514 return $this->user_repository->totalAdmins(); 2515 } 2516 2517 /** 2518 * @return string 2519 */ 2520 public function totalNonAdmins(): string 2521 { 2522 return $this->user_repository->totalNonAdmins(); 2523 } 2524 2525 /** 2526 * @return string 2527 */ 2528 public function latestUserId(): string 2529 { 2530 return $this->latest_user_repository->latestUserId(); 2531 } 2532 2533 /** 2534 * @return string 2535 */ 2536 public function latestUserName(): string 2537 { 2538 return $this->latest_user_repository->latestUserName(); 2539 } 2540 2541 /** 2542 * @return string 2543 */ 2544 public function latestUserFullName(): string 2545 { 2546 return $this->latest_user_repository->latestUserFullName(); 2547 } 2548 2549 /** 2550 * @param string|null $format 2551 * 2552 * @return string 2553 */ 2554 public function latestUserRegDate(string|null $format = null): string 2555 { 2556 return $this->latest_user_repository->latestUserRegDate($format); 2557 } 2558 2559 /** 2560 * @param string|null $format 2561 * 2562 * @return string 2563 */ 2564 public function latestUserRegTime(string|null $format = null): string 2565 { 2566 return $this->latest_user_repository->latestUserRegTime($format); 2567 } 2568 2569 /** 2570 * @param string|null $yes 2571 * @param string|null $no 2572 * 2573 * @return string 2574 */ 2575 public function latestUserLoggedin(string|null $yes = null, string|null $no = null): string 2576 { 2577 return $this->latest_user_repository->latestUserLoggedin($yes, $no); 2578 } 2579 2580 /** 2581 * @return string 2582 */ 2583 public function contactWebmaster(): string 2584 { 2585 return $this->contact_repository->contactWebmaster(); 2586 } 2587 2588 /** 2589 * @return string 2590 */ 2591 public function contactGedcom(): string 2592 { 2593 return $this->contact_repository->contactGedcom(); 2594 } 2595 2596 /** 2597 * @return string 2598 */ 2599 public function serverDate(): string 2600 { 2601 return $this->server_repository->serverDate(); 2602 } 2603 2604 /** 2605 * @return string 2606 */ 2607 public function serverTime(): string 2608 { 2609 return $this->server_repository->serverTime(); 2610 } 2611 2612 /** 2613 * @return string 2614 */ 2615 public function serverTime24(): string 2616 { 2617 return $this->server_repository->serverTime24(); 2618 } 2619 2620 /** 2621 * What is the timezone of the server. 2622 * 2623 * @return string 2624 */ 2625 public function serverTimezone(): string 2626 { 2627 return $this->server_repository->serverTimezone(); 2628 } 2629 2630 /** 2631 * @return string 2632 */ 2633 public function browserDate(): string 2634 { 2635 return $this->browser_repository->browserDate(); 2636 } 2637 2638 /** 2639 * @return string 2640 */ 2641 public function browserTime(): string 2642 { 2643 return $this->browser_repository->browserTime(); 2644 } 2645 2646 /** 2647 * @return string 2648 */ 2649 public function browserTimezone(): string 2650 { 2651 return $this->browser_repository->browserTimezone(); 2652 } 2653 2654 /** 2655 * @param string $page_parameter 2656 * 2657 * @return string 2658 */ 2659 public function hitCount(string $page_parameter = ''): string 2660 { 2661 return $this->hit_count_repository->hitCount($page_parameter); 2662 } 2663 2664 /** 2665 * @param string $page_parameter 2666 * 2667 * @return string 2668 */ 2669 public function hitCountUser(string $page_parameter = ''): string 2670 { 2671 return $this->hit_count_repository->hitCountUser($page_parameter); 2672 } 2673 2674 /** 2675 * @param string $page_parameter 2676 * 2677 * @return string 2678 */ 2679 public function hitCountIndi(string $page_parameter = ''): string 2680 { 2681 return $this->hit_count_repository->hitCountIndi($page_parameter); 2682 } 2683 2684 /** 2685 * @param string $page_parameter 2686 * 2687 * @return string 2688 */ 2689 public function hitCountFam(string $page_parameter = ''): string 2690 { 2691 return $this->hit_count_repository->hitCountFam($page_parameter); 2692 } 2693 2694 /** 2695 * @param string $page_parameter 2696 * 2697 * @return string 2698 */ 2699 public function hitCountSour(string $page_parameter = ''): string 2700 { 2701 return $this->hit_count_repository->hitCountSour($page_parameter); 2702 } 2703 2704 /** 2705 * @param string $page_parameter 2706 * 2707 * @return string 2708 */ 2709 public function hitCountRepo(string $page_parameter = ''): string 2710 { 2711 return $this->hit_count_repository->hitCountRepo($page_parameter); 2712 } 2713 2714 /** 2715 * @param string $page_parameter 2716 * 2717 * @return string 2718 */ 2719 public function hitCountNote(string $page_parameter = ''): string 2720 { 2721 return $this->hit_count_repository->hitCountNote($page_parameter); 2722 } 2723 2724 /** 2725 * @param string $page_parameter 2726 * 2727 * @return string 2728 */ 2729 public function hitCountObje(string $page_parameter = ''): string 2730 { 2731 return $this->hit_count_repository->hitCountObje($page_parameter); 2732 } 2733 2734 /** 2735 * @return string 2736 */ 2737 public function gedcomFavorites(): string 2738 { 2739 return $this->favorites_repository->gedcomFavorites(); 2740 } 2741 2742 /** 2743 * @return string 2744 */ 2745 public function userFavorites(): string 2746 { 2747 return $this->favorites_repository->userFavorites(); 2748 } 2749 2750 /** 2751 * @return string 2752 */ 2753 public function totalGedcomFavorites(): string 2754 { 2755 return $this->favorites_repository->totalGedcomFavorites(); 2756 } 2757 2758 /** 2759 * @return string 2760 */ 2761 public function totalUserFavorites(): string 2762 { 2763 return $this->favorites_repository->totalUserFavorites(); 2764 } 2765 2766 /** 2767 * @return string 2768 */ 2769 public function totalUserMessages(): string 2770 { 2771 return $this->message_repository->totalUserMessages(); 2772 } 2773 2774 /** 2775 * @return string 2776 */ 2777 public function totalUserJournal(): string 2778 { 2779 return $this->news_repository->totalUserJournal(); 2780 } 2781 2782 /** 2783 * @return string 2784 */ 2785 public function totalGedcomNews(): string 2786 { 2787 return $this->news_repository->totalGedcomNews(); 2788 } 2789 2790 /** 2791 * Create any of the other blocks. 2792 * Use as #callBlock:block_name# 2793 * 2794 * @param string $block 2795 * @param string ...$params 2796 * 2797 * @return string|null 2798 */ 2799 public function callBlock(string $block = '', ...$params): ?string 2800 { 2801 /** @var ModuleBlockInterface|null $module */ 2802 $module = $this->module_service 2803 ->findByComponent(ModuleBlockInterface::class, $this->tree, Auth::user()) 2804 ->first(static fn(ModuleInterface $module): bool => $module->name() === $block && $module->name() !== 'html'); 2805 2806 if ($module === null) { 2807 return ''; 2808 } 2809 2810 // Build the config array 2811 $cfg = []; 2812 foreach ($params as $config) { 2813 $bits = explode('=', $config); 2814 2815 if (count($bits) < 2) { 2816 continue; 2817 } 2818 2819 $v = array_shift($bits); 2820 $cfg[$v] = implode('=', $bits); 2821 } 2822 2823 return $module->getBlock($this->tree, 0, ModuleBlockInterface::CONTEXT_EMBED, $cfg); 2824 } 2825 2826 /** 2827 * What is the current version of webtrees. 2828 * 2829 * @return string 2830 */ 2831 public function webtreesVersion(): string 2832 { 2833 return Webtrees::VERSION; 2834 } 2835 2836 /** 2837 * Get tags and their parsed results. 2838 * 2839 * @param string $text 2840 * 2841 * @return array<string> 2842 */ 2843 private function getTags(string $text): array 2844 { 2845 $tags = []; 2846 $matches = []; 2847 2848 preg_match_all('/#([^#\n]+)(?=#)/', $text, $matches, PREG_SET_ORDER); 2849 2850 foreach ($matches as $match) { 2851 $params = explode(':', $match[1]); 2852 $method = array_shift($params); 2853 2854 if (method_exists($this, $method)) { 2855 $tags[$match[0] . '#'] = $this->$method(...$params); 2856 } 2857 } 2858 2859 return $tags; 2860 } 2861} 2862