1<?php 2/** 3 * webtrees: online genealogy 4 * Copyright (C) 2019 webtrees development team 5 * This program is free software: you can redistribute it and/or modify 6 * it under the terms of the GNU General Public License as published by 7 * the Free Software Foundation, either version 3 of the License, or 8 * (at your option) any later version. 9 * This program is distributed in the hope that it will be useful, 10 * but WITHOUT ANY WARRANTY; without even the implied warranty of 11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 12 * GNU General Public License for more details. 13 * You should have received a copy of the GNU General Public License 14 * along with this program. If not, see <http://www.gnu.org/licenses/>. 15 */ 16declare(strict_types=1); 17 18namespace Fisharebest\Webtrees\Module; 19 20use Fisharebest\Webtrees\Auth; 21use Fisharebest\Webtrees\Date; 22use Fisharebest\Webtrees\Fact; 23use Fisharebest\Webtrees\Family; 24use Fisharebest\Webtrees\Functions\Functions; 25use Fisharebest\Webtrees\Gedcom; 26use Fisharebest\Webtrees\I18N; 27use Fisharebest\Webtrees\Individual; 28use Fisharebest\Webtrees\Module; 29use Fisharebest\Webtrees\Services\ModuleService; 30use Fisharebest\Webtrees\Site; 31use Illuminate\Support\Collection; 32 33/** 34 * Class IndividualFactsTabModule 35 */ 36class IndividualFactsTabModule extends AbstractModule implements ModuleTabInterface 37{ 38 use ModuleTabTrait; 39 40 /** 41 * @var ModuleService 42 */ 43 private $module_service; 44 45 /** 46 * UserWelcomeModule constructor. 47 * 48 * @param ModuleService $module_service 49 */ 50 public function __construct(ModuleService $module_service) { 51 $this->module_service = $module_service; 52 } 53 54 /** 55 * How should this module be labelled on tabs, menus, etc.? 56 * 57 * @return string 58 */ 59 public function title(): string 60 { 61 /* I18N: Name of a module/tab on the individual page. */ 62 return I18N::translate('Facts and events'); 63 } 64 65 /** 66 * A sentence describing what this module does. 67 * 68 * @return string 69 */ 70 public function description(): string 71 { 72 /* I18N: Description of the “Facts and events” module */ 73 return I18N::translate('A tab showing the facts and events of an individual.'); 74 } 75 76 /** 77 * The default position for this tab. It can be changed in the control panel. 78 * 79 * @return int 80 */ 81 public function defaultTabOrder(): int 82 { 83 return 2; 84 } 85 86 /** {@inheritdoc} */ 87 public function isGrayedOut(Individual $individual): bool 88 { 89 return false; 90 } 91 92 /** {@inheritdoc} */ 93 public function getTabContent(Individual $individual): string 94 { 95 // Only include events of close relatives that are between birth and death 96 $min_date = $individual->getEstimatedBirthDate(); 97 $max_date = $individual->getEstimatedDeathDate(); 98 99 $indifacts = []; 100 // The individual’s own facts 101 foreach ($individual->facts() as $fact) { 102 switch ($fact->getTag()) { 103 case 'SEX': 104 case 'NAME': 105 case 'SOUR': 106 case 'OBJE': 107 case 'NOTE': 108 case 'FAMC': 109 case 'FAMS': 110 break; 111 112 default: 113 $use_extra_info_module = $this->module_service->findByComponent('sidebar', $individual->tree(), Auth::user()) 114 ->filter(function (ModuleInterface $module): bool { 115 return $module instanceof ExtraInformationModule; 116 })->isNotEmpty(); 117 118 if (!$use_extra_info_module || !ExtraInformationModule::showFact($fact)) { 119 $indifacts[] = $fact; 120 } 121 break; 122 } 123 } 124 125 // Add spouse-family facts 126 foreach ($individual->getSpouseFamilies() as $family) { 127 foreach ($family->facts() as $fact) { 128 switch ($fact->getTag()) { 129 case 'SOUR': 130 case 'NOTE': 131 case 'OBJE': 132 case 'CHAN': 133 case '_UID': 134 case 'RIN': 135 case 'HUSB': 136 case 'WIFE': 137 case 'CHIL': 138 break; 139 default: 140 $indifacts[] = $fact; 141 break; 142 } 143 } 144 145 $spouse = $family->getSpouse($individual); 146 147 if ($spouse instanceof Individual) { 148 $spouse_facts = self::spouseFacts($individual, $spouse, $min_date, $max_date); 149 $indifacts = array_merge($indifacts, $spouse_facts); 150 } 151 152 $child_facts = self::childFacts($individual, $family, '_CHIL', '', $min_date, $max_date); 153 $indifacts = array_merge($indifacts, $child_facts); 154 } 155 156 $parent_facts = self::parentFacts($individual, 1, $min_date, $max_date); 157 $associate_facts = self::associateFacts($individual); 158 $historical_facts = self::historicalFacts($individual); 159 160 $indifacts = array_merge($indifacts, $parent_facts, $associate_facts, $historical_facts); 161 162 Functions::sortFacts($indifacts); 163 164 return view('modules/personal_facts/tab', [ 165 'can_edit' => $individual->canEdit(), 166 'has_historical_facts' => !empty($historical_facts), 167 'individual' => $individual, 168 'facts' => $indifacts, 169 ]); 170 } 171 172 /** 173 * Does a relative event occur within a date range (i.e. the individual's lifetime)? 174 * 175 * @param Fact $fact 176 * @param Date $min_date 177 * @param Date $max_date 178 * 179 * @return bool 180 */ 181 private static function includeFact(Fact $fact, Date $min_date, Date $max_date): bool 182 { 183 $fact_date = $fact->date(); 184 185 return $fact_date->isOK() && Date::compare($min_date, $fact_date) <= 0 && Date::compare($fact_date, $max_date) <= 0; 186 } 187 188 /** {@inheritdoc} */ 189 public function hasTabContent(Individual $individual): bool 190 { 191 return true; 192 } 193 194 /** {@inheritdoc} */ 195 public function canLoadAjax(): bool 196 { 197 return false; 198 } 199 200 /** 201 * Spouse facts that are shown on an individual’s page. 202 * 203 * @param Individual $individual Show events that occured during the lifetime of this individual 204 * @param Individual $spouse Show events of this individual 205 * @param Date $min_date 206 * @param Date $max_date 207 * 208 * @return Fact[] 209 */ 210 private static function spouseFacts(Individual $individual, Individual $spouse, Date $min_date, Date $max_date): array 211 { 212 $SHOW_RELATIVES_EVENTS = $individual->tree()->getPreference('SHOW_RELATIVES_EVENTS'); 213 214 $facts = []; 215 if (strstr($SHOW_RELATIVES_EVENTS, '_DEAT_SPOU')) { 216 foreach ($spouse->facts(Gedcom::DEATH_EVENTS) as $fact) { 217 if (self::includeFact($fact, $min_date, $max_date)) { 218 // Convert the event to a close relatives event. 219 $rela_fact = clone($fact); 220 $rela_fact->setTag('_' . $fact->getTag() . '_SPOU'); 221 $facts[] = $rela_fact; 222 } 223 } 224 } 225 226 return $facts; 227 } 228 229 /** 230 * Get the events of children and grandchildren. 231 * 232 * @param Individual $person 233 * @param Family $family 234 * @param string $option 235 * @param string $relation 236 * @param Date $min_date 237 * @param Date $max_date 238 * 239 * @return Fact[] 240 */ 241 private static function childFacts(Individual $person, Family $family, $option, $relation, Date $min_date, Date $max_date): array 242 { 243 $SHOW_RELATIVES_EVENTS = $person->tree()->getPreference('SHOW_RELATIVES_EVENTS'); 244 245 $facts = []; 246 247 // Deal with recursion. 248 switch ($option) { 249 case '_CHIL': 250 // Add grandchildren 251 foreach ($family->getChildren() as $child) { 252 foreach ($child->getSpouseFamilies() as $cfamily) { 253 switch ($child->getSex()) { 254 case 'M': 255 foreach (self::childFacts($person, $cfamily, '_GCHI', 'son', $min_date, $max_date) as $fact) { 256 $facts[] = $fact; 257 } 258 break; 259 case 'F': 260 foreach (self::childFacts($person, $cfamily, '_GCHI', 'dau', $min_date, $max_date) as $fact) { 261 $facts[] = $fact; 262 } 263 break; 264 default: 265 foreach (self::childFacts($person, $cfamily, '_GCHI', 'chi', $min_date, $max_date) as $fact) { 266 $facts[] = $fact; 267 } 268 break; 269 } 270 } 271 } 272 break; 273 } 274 275 // For each child in the family 276 foreach ($family->getChildren() as $child) { 277 if ($child->xref() == $person->xref()) { 278 // We are not our own sibling! 279 continue; 280 } 281 // add child’s birth 282 if (strpos($SHOW_RELATIVES_EVENTS, '_BIRT' . str_replace('_HSIB', '_SIBL', $option)) !== false) { 283 foreach ($child->facts(Gedcom::BIRTH_EVENTS) as $fact) { 284 // Always show _BIRT_CHIL, even if the dates are not known 285 if ($option == '_CHIL' || self::includeFact($fact, $min_date, $max_date)) { 286 if ($option == '_GCHI' && $relation == 'dau') { 287 // Convert the event to a close relatives event. 288 $rela_fact = clone($fact); 289 $rela_fact->setTag('_' . $fact->getTag() . '_GCH1'); 290 $facts[] = $rela_fact; 291 } elseif ($option == '_GCHI' && $relation == 'son') { 292 // Convert the event to a close relatives event. 293 $rela_fact = clone($fact); 294 $rela_fact->setTag('_' . $fact->getTag() . '_GCH2'); 295 $facts[] = $rela_fact; 296 } else { 297 // Convert the event to a close relatives event. 298 $rela_fact = clone($fact); 299 $rela_fact->setTag('_' . $fact->getTag() . $option); 300 $facts[] = $rela_fact; 301 } 302 } 303 } 304 } 305 // add child’s death 306 if (strpos($SHOW_RELATIVES_EVENTS, '_DEAT' . str_replace('_HSIB', '_SIBL', $option)) !== false) { 307 foreach ($child->facts(Gedcom::DEATH_EVENTS) as $fact) { 308 if (self::includeFact($fact, $min_date, $max_date)) { 309 if ($option == '_GCHI' && $relation == 'dau') { 310 // Convert the event to a close relatives event. 311 $rela_fact = clone($fact); 312 $rela_fact->setTag('_' . $fact->getTag() . '_GCH1'); 313 $facts[] = $rela_fact; 314 } elseif ($option == '_GCHI' && $relation == 'son') { 315 // Convert the event to a close relatives event. 316 $rela_fact = clone($fact); 317 $rela_fact->setTag('_' . $fact->getTag() . '_GCH2'); 318 $facts[] = $rela_fact; 319 } else { 320 // Convert the event to a close relatives event. 321 $rela_fact = clone($fact); 322 $rela_fact->setTag('_' . $fact->getTag() . $option); 323 $facts[] = $rela_fact; 324 } 325 } 326 } 327 } 328 // add child’s marriage 329 if (strstr($SHOW_RELATIVES_EVENTS, '_MARR' . str_replace('_HSIB', '_SIBL', $option))) { 330 foreach ($child->getSpouseFamilies() as $sfamily) { 331 foreach ($sfamily->facts(['MARR']) as $fact) { 332 if (self::includeFact($fact, $min_date, $max_date)) { 333 if ($option == '_GCHI' && $relation == 'dau') { 334 // Convert the event to a close relatives event. 335 $rela_fact = clone($fact); 336 $rela_fact->setTag('_' . $fact->getTag() . '_GCH1'); 337 $facts[] = $rela_fact; 338 } elseif ($option == '_GCHI' && $relation == 'son') { 339 // Convert the event to a close relatives event. 340 $rela_fact = clone($fact); 341 $rela_fact->setTag('_' . $fact->getTag() . '_GCH2'); 342 $facts[] = $rela_fact; 343 } else { 344 // Convert the event to a close relatives event. 345 $rela_fact = clone($fact); 346 $rela_fact->setTag('_' . $fact->getTag() . $option); 347 $facts[] = $rela_fact; 348 } 349 } 350 } 351 } 352 } 353 } 354 355 return $facts; 356 } 357 358 /** 359 * Get the events of parents and grandparents. 360 * 361 * @param Individual $person 362 * @param int $sosa 363 * @param Date $min_date 364 * @param Date $max_date 365 * 366 * @return Fact[] 367 */ 368 private static function parentFacts(Individual $person, $sosa, Date $min_date, Date $max_date): array 369 { 370 $SHOW_RELATIVES_EVENTS = $person->tree()->getPreference('SHOW_RELATIVES_EVENTS'); 371 372 $facts = []; 373 374 if ($sosa == 1) { 375 foreach ($person->getChildFamilies() as $family) { 376 // Add siblings 377 foreach (self::childFacts($person, $family, '_SIBL', '', $min_date, $max_date) as $fact) { 378 $facts[] = $fact; 379 } 380 foreach ($family->getSpouses() as $spouse) { 381 foreach ($spouse->getSpouseFamilies() as $sfamily) { 382 if ($family !== $sfamily) { 383 // Add half-siblings 384 foreach (self::childFacts($person, $sfamily, '_HSIB', '', $min_date, $max_date) as $fact) { 385 $facts[] = $fact; 386 } 387 } 388 } 389 // Add grandparents 390 foreach (self::parentFacts($spouse, $spouse->getSex() == 'F' ? 3 : 2, $min_date, $max_date) as $fact) { 391 $facts[] = $fact; 392 } 393 } 394 } 395 396 if (strstr($SHOW_RELATIVES_EVENTS, '_MARR_PARE')) { 397 // add father/mother marriages 398 foreach ($person->getChildFamilies() as $sfamily) { 399 foreach ($sfamily->facts(['MARR']) as $fact) { 400 if (self::includeFact($fact, $min_date, $max_date)) { 401 // marriage of parents (to each other) 402 $rela_fact = clone($fact); 403 $rela_fact->setTag('_' . $fact->getTag() . '_FAMC'); 404 $facts[] = $rela_fact; 405 } 406 } 407 } 408 foreach ($person->getChildStepFamilies() as $sfamily) { 409 foreach ($sfamily->facts(['MARR']) as $fact) { 410 if (self::includeFact($fact, $min_date, $max_date)) { 411 // marriage of a parent (to another spouse) 412 // Convert the event to a close relatives event 413 $rela_fact = clone($fact); 414 $rela_fact->setTag('_' . $fact->getTag() . '_PARE'); 415 $facts[] = $rela_fact; 416 } 417 } 418 } 419 } 420 } 421 422 foreach ($person->getChildFamilies() as $family) { 423 foreach ($family->getSpouses() as $parent) { 424 if (strstr($SHOW_RELATIVES_EVENTS, '_DEAT' . ($sosa == 1 ? '_PARE' : '_GPAR'))) { 425 foreach ($parent->facts(Gedcom::DEATH_EVENTS) as $fact) { 426 if (self::includeFact($fact, $min_date, $max_date)) { 427 switch ($sosa) { 428 case 1: 429 // Convert the event to a close relatives event. 430 $rela_fact = clone($fact); 431 $rela_fact->setTag('_' . $fact->getTag() . '_PARE'); 432 $facts[] = $rela_fact; 433 break; 434 case 2: 435 // Convert the event to a close relatives event 436 $rela_fact = clone($fact); 437 $rela_fact->setTag('_' . $fact->getTag() . '_GPA1'); 438 $facts[] = $rela_fact; 439 break; 440 case 3: 441 // Convert the event to a close relatives event 442 $rela_fact = clone($fact); 443 $rela_fact->setTag('_' . $fact->getTag() . '_GPA2'); 444 $facts[] = $rela_fact; 445 break; 446 } 447 } 448 } 449 } 450 } 451 } 452 453 return $facts; 454 } 455 456 /** 457 * Get any historical events. 458 * 459 * @param Individual $individual 460 * 461 * @return Fact[] 462 */ 463 private function historicalFacts(Individual $individual): array 464 { 465 return $this->module_service->findByInterface(ModuleHistoricEventsInterface::class) 466 ->map(function (ModuleHistoricEventsInterface $module) use ($individual): Collection { 467 return $module->historicEventsForIndividual($individual); 468 }) 469 ->flatten() 470 ->all(); 471 } 472 473 /** 474 * Get the events of associates. 475 * 476 * @param Individual $person 477 * 478 * @return Fact[] 479 */ 480 private static function associateFacts(Individual $person): array 481 { 482 $facts = []; 483 484 /** @var Individual[] $associates */ 485 $associates = array_merge( 486 $person->linkedIndividuals('ASSO'), 487 $person->linkedIndividuals('_ASSO'), 488 $person->linkedFamilies('ASSO'), 489 $person->linkedFamilies('_ASSO') 490 ); 491 foreach ($associates as $associate) { 492 foreach ($associate->facts() as $fact) { 493 $arec = $fact->attribute('_ASSO'); 494 if (!$arec) { 495 $arec = $fact->attribute('ASSO'); 496 } 497 if ($arec && trim($arec, '@') === $person->xref()) { 498 // Extract the important details from the fact 499 $factrec = '1 ' . $fact->getTag(); 500 if (preg_match('/\n2 DATE .*/', $fact->gedcom(), $match)) { 501 $factrec .= $match[0]; 502 } 503 if (preg_match('/\n2 PLAC .*/', $fact->gedcom(), $match)) { 504 $factrec .= $match[0]; 505 } 506 if ($associate instanceof Family) { 507 foreach ($associate->getSpouses() as $spouse) { 508 $factrec .= "\n2 _ASSO @" . $spouse->xref() . '@'; 509 } 510 } else { 511 $factrec .= "\n2 _ASSO @" . $associate->xref() . '@'; 512 } 513 $facts[] = new Fact($factrec, $associate, 'asso'); 514 } 515 } 516 } 517 518 return $facts; 519 } 520} 521