. */ namespace Fisharebest\Webtrees; use Fisharebest\ExtCalendar\GregorianCalendar; use Fisharebest\Webtrees\GedcomCode\GedcomCodePedi; /** * A GEDCOM individual (INDI) object. */ class Individual extends GedcomRecord { const RECORD_TYPE = 'INDI'; const ROUTE_NAME = 'individual'; /** @var int used in some lists to keep track of this individual’s generation in that list */ public $generation; /** @var Date The estimated date of birth */ private $estimated_birth_date; /** @var Date The estimated date of death */ private $estimated_death_date; /** * Get an instance of an individual object. For single records, * we just receive the XREF. For bulk records (such as lists * and search results) we can receive the GEDCOM data as well. * * @param string $xref * @param Tree $tree * @param string|null $gedcom * * @throws \Exception * * @return Individual|null */ public static function getInstance(string $xref, Tree $tree, string $gedcom = null) { $record = parent::getInstance($xref, $tree, $gedcom); if ($record instanceof Individual) { return $record; } return null; } /** * Sometimes, we'll know in advance that we need to load a set of records. * Typically when we load families and their members. * * @param Tree $tree * @param string[] $xrefs */ public static function load(Tree $tree, array $xrefs) { $args = [ 'tree_id' => $tree->getTreeId(), ]; $placeholders = []; foreach (array_unique($xrefs) as $n => $xref) { if (!isset(self::$gedcom_record_cache[$tree->getTreeId()][$xref])) { $placeholders[] = ':x' . $n; $args['x' . $n] = $xref; } } if (!empty($placeholders)) { $rows = Database::prepare( "SELECT i_id AS xref, i_gedcom AS gedcom" . " FROM `##individuals`" . " WHERE i_file = :tree_id AND i_id IN (" . implode(',', $placeholders) . ")" )->execute( $args )->fetchAll(); foreach ($rows as $row) { self::getInstance($row->xref, $tree, $row->gedcom); } } } /** * Can the name of this record be shown? * * @param int|null $access_level * * @return bool */ public function canShowName(int $access_level = null): bool { if ($access_level === null) { $access_level = Auth::accessLevel($this->tree); } return $this->tree->getPreference('SHOW_LIVING_NAMES') >= $access_level || $this->canShow($access_level); } /** * Can this individual be shown? * * @param int $access_level * * @return bool */ protected function canShowByType(int $access_level): bool { // Dead people... if ($this->tree->getPreference('SHOW_DEAD_PEOPLE') >= $access_level && $this->isDead()) { $keep_alive = false; $KEEP_ALIVE_YEARS_BIRTH = (int)$this->tree->getPreference('KEEP_ALIVE_YEARS_BIRTH'); if ($KEEP_ALIVE_YEARS_BIRTH) { preg_match_all('/\n1 (?:' . WT_EVENTS_BIRT . ').*(?:\n[2-9].*)*(?:\n2 DATE (.+))/', $this->gedcom, $matches, PREG_SET_ORDER); foreach ($matches as $match) { $date = new Date($match[1]); if ($date->isOK() && $date->gregorianYear() + $KEEP_ALIVE_YEARS_BIRTH > date('Y')) { $keep_alive = true; break; } } } $KEEP_ALIVE_YEARS_DEATH = (int)$this->tree->getPreference('KEEP_ALIVE_YEARS_DEATH'); if ($KEEP_ALIVE_YEARS_DEATH) { preg_match_all('/\n1 (?:' . WT_EVENTS_DEAT . ').*(?:\n[2-9].*)*(?:\n2 DATE (.+))/', $this->gedcom, $matches, PREG_SET_ORDER); foreach ($matches as $match) { $date = new Date($match[1]); if ($date->isOK() && $date->gregorianYear() + $KEEP_ALIVE_YEARS_DEATH > date('Y')) { $keep_alive = true; break; } } } if (!$keep_alive) { return true; } } // Consider relationship privacy (unless an admin is applying download restrictions) $user_path_length = (int)$this->tree->getUserPreference(Auth::user(), 'RELATIONSHIP_PATH_LENGTH'); $gedcomid = $this->tree->getUserPreference(Auth::user(), 'gedcomid'); if ($gedcomid !== '' && $user_path_length > 0) { return self::isRelated($this, $user_path_length); } // No restriction found - show living people to members only: return Auth::PRIV_USER >= $access_level; } /** * For relationship privacy calculations - is this individual a close relative? * * @param Individual $target * @param int $distance * * @return bool */ private static function isRelated(Individual $target, $distance): bool { static $cache = null; $user_individual = self::getInstance($target->tree->getUserPreference(Auth::user(), 'gedcomid'), $target->tree); if ($user_individual) { if (!$cache) { $cache = [ 0 => [$user_individual], 1 => [], ]; foreach ($user_individual->getFacts('FAM[CS]', false, Auth::PRIV_HIDE) as $fact) { $family = $fact->getTarget(); if ($family) { $cache[1][] = $family; } } } } else { // No individual linked to this account? Cannot use relationship privacy. return true; } // Double the distance, as we count the INDI-FAM and FAM-INDI links separately $distance *= 2; // Consider each path length in turn for ($n = 0; $n <= $distance; ++$n) { if (array_key_exists($n, $cache)) { // We have already calculated all records with this length if ($n % 2 == 0 && in_array($target, $cache[$n], true)) { return true; } } else { // Need to calculate these paths $cache[$n] = []; if ($n % 2 == 0) { // Add FAM->INDI links foreach ($cache[$n - 1] as $family) { foreach ($family->getFacts('HUSB|WIFE|CHIL', false, Auth::PRIV_HIDE) as $fact) { $individual = $fact->getTarget(); // Don’t backtrack if ($individual && !in_array($individual, $cache[$n - 2], true)) { $cache[$n][] = $individual; } } } if (in_array($target, $cache[$n], true)) { return true; } } else { // Add INDI->FAM links foreach ($cache[$n - 1] as $individual) { foreach ($individual->getFacts('FAM[CS]', false, Auth::PRIV_HIDE) as $fact) { $family = $fact->getTarget(); // Don’t backtrack if ($family && !in_array($family, $cache[$n - 2], true)) { $cache[$n][] = $family; } } } } } } return false; } /** * Generate a private version of this record * * @param int $access_level * * @return string */ protected function createPrivateGedcomRecord($access_level): string { $SHOW_PRIVATE_RELATIONSHIPS = $this->tree->getPreference('SHOW_PRIVATE_RELATIONSHIPS'); $rec = '0 @' . $this->xref . '@ INDI'; if ($this->tree->getPreference('SHOW_LIVING_NAMES') >= $access_level) { // Show all the NAME tags, including subtags foreach ($this->getFacts('NAME') as $fact) { $rec .= "\n" . $fact->getGedcom(); } } // Just show the 1 FAMC/FAMS tag, not any subtags, which may contain private data preg_match_all('/\n1 (?:FAMC|FAMS) @(' . WT_REGEX_XREF . ')@/', $this->gedcom, $matches, PREG_SET_ORDER); foreach ($matches as $match) { $rela = Family::getInstance($match[1], $this->tree); if ($rela && ($SHOW_PRIVATE_RELATIONSHIPS || $rela->canShow($access_level))) { $rec .= $match[0]; } } // Don’t privatize sex. if (preg_match('/\n1 SEX [MFU]/', $this->gedcom, $match)) { $rec .= $match[0]; } return $rec; } /** * Fetch data from the database * * @param string $xref * @param int $tree_id * * @return null|string */ protected static function fetchGedcomRecord(string $xref, int $tree_id) { return Database::prepare( "SELECT i_gedcom FROM `##individuals` WHERE i_id = :xref AND i_file = :tree_id" )->execute([ 'xref' => $xref, 'tree_id' => $tree_id, ])->fetchOne(); } /** * Static helper function to sort an array of people by birth date * * @param Individual $x * @param Individual $y * * @return int */ public static function compareBirthDate(Individual $x, Individual $y): int { return Date::compare($x->getEstimatedBirthDate(), $y->getEstimatedBirthDate()); } /** * Static helper function to sort an array of people by death date * * @param Individual $x * @param Individual $y * * @return int */ public static function compareDeathDate(Individual $x, Individual $y): int { return Date::compare($x->getEstimatedDeathDate(), $y->getEstimatedDeathDate()); } /** * Calculate whether this individual is living or dead. * If not known to be dead, then assume living. * * @return bool */ public function isDead(): bool { $MAX_ALIVE_AGE = (int)$this->tree->getPreference('MAX_ALIVE_AGE'); // "1 DEAT Y" or "1 DEAT/2 DATE" or "1 DEAT/2 PLAC" if (preg_match('/\n1 (?:' . WT_EVENTS_DEAT . ')(?: Y|(?:\n[2-9].+)*\n2 (DATE|PLAC) )/', $this->gedcom)) { return true; } // If any event occured more than $MAX_ALIVE_AGE years ago, then assume the individual is dead if (preg_match_all('/\n2 DATE (.+)/', $this->gedcom, $date_matches)) { foreach ($date_matches[1] as $date_match) { $date = new Date($date_match); if ($date->isOK() && $date->maximumJulianDay() <= WT_CLIENT_JD - 365 * $MAX_ALIVE_AGE) { return true; } } // The individual has one or more dated events. All are less than $MAX_ALIVE_AGE years ago. // If one of these is a birth, the individual must be alive. if (preg_match('/\n1 BIRT(?:\n[2-9].+)*\n2 DATE /', $this->gedcom)) { return false; } } // If we found no conclusive dates then check the dates of close relatives. // Check parents (birth and adopted) foreach ($this->getChildFamilies(Auth::PRIV_HIDE) as $family) { foreach ($family->getSpouses(Auth::PRIV_HIDE) as $parent) { // Assume parents are no more than 45 years older than their children preg_match_all('/\n2 DATE (.+)/', $parent->gedcom, $date_matches); foreach ($date_matches[1] as $date_match) { $date = new Date($date_match); if ($date->isOK() && $date->maximumJulianDay() <= WT_CLIENT_JD - 365 * ($MAX_ALIVE_AGE + 45)) { return true; } } } } // Check spouses foreach ($this->getSpouseFamilies(Auth::PRIV_HIDE) as $family) { preg_match_all('/\n2 DATE (.+)/', $family->gedcom, $date_matches); foreach ($date_matches[1] as $date_match) { $date = new Date($date_match); // Assume marriage occurs after age of 10 if ($date->isOK() && $date->maximumJulianDay() <= WT_CLIENT_JD - 365 * ($MAX_ALIVE_AGE - 10)) { return true; } } // Check spouse dates $spouse = $family->getSpouse($this, Auth::PRIV_HIDE); if ($spouse) { preg_match_all('/\n2 DATE (.+)/', $spouse->gedcom, $date_matches); foreach ($date_matches[1] as $date_match) { $date = new Date($date_match); // Assume max age difference between spouses of 40 years if ($date->isOK() && $date->maximumJulianDay() <= WT_CLIENT_JD - 365 * ($MAX_ALIVE_AGE + 40)) { return true; } } } // Check child dates foreach ($family->getChildren(Auth::PRIV_HIDE) as $child) { preg_match_all('/\n2 DATE (.+)/', $child->gedcom, $date_matches); // Assume children born after age of 15 foreach ($date_matches[1] as $date_match) { $date = new Date($date_match); if ($date->isOK() && $date->maximumJulianDay() <= WT_CLIENT_JD - 365 * ($MAX_ALIVE_AGE - 15)) { return true; } } // Check grandchildren foreach ($child->getSpouseFamilies(Auth::PRIV_HIDE) as $child_family) { foreach ($child_family->getChildren(Auth::PRIV_HIDE) as $grandchild) { preg_match_all('/\n2 DATE (.+)/', $grandchild->gedcom, $date_matches); // Assume grandchildren born after age of 30 foreach ($date_matches[1] as $date_match) { $date = new Date($date_match); if ($date->isOK() && $date->maximumJulianDay() <= WT_CLIENT_JD - 365 * ($MAX_ALIVE_AGE - 30)) { return true; } } } } } } return false; } /** * Find the highlighted media object for an individual * * @return null|MediaFile */ public function findHighlightedMediaFile() { foreach ($this->getFacts('OBJE') as $fact) { $media = $fact->getTarget(); if ($media instanceof Media) { foreach ($media->mediaFiles() as $media_file) { if ($media_file->isImage() && !$media_file->isExternal()) { return $media_file; } } } } return null; } /** * Display the prefered image for this individual. * Use an icon if no image is available. * * @param int $width Pixels * @param int $height Pixels * @param string $fit "crop" or "contain" * @param string[] $attributes Additional HTML attributes * * @return string */ public function displayImage($width, $height, $fit, $attributes): string { $media_file = $this->findHighlightedMediaFile(); if ($media_file !== null) { return $media_file->displayImage($width, $height, $fit, $attributes); } if ($this->tree->getPreference('USE_SILHOUETTE')) { return ''; } return ''; } /** * Get the date of birth * * @return Date */ public function getBirthDate(): Date { foreach ($this->getAllBirthDates() as $date) { if ($date->isOK()) { return $date; } } return new Date(''); } /** * Get the place of birth * * @return Place */ public function getBirthPlace(): Place { foreach ($this->getAllBirthPlaces() as $place) { if ($place) { return $place; } } return new Place('', $this->tree); } /** * Get the year of birth * * @return string the year of birth */ public function getBirthYear(): string { return $this->getBirthDate()->minimumDate()->format('%Y'); } /** * Get the date of death * * @return Date */ public function getDeathDate(): Date { foreach ($this->getAllDeathDates() as $date) { if ($date->isOK()) { return $date; } } return new Date(''); } /** * Get the place of death * * @return Place */ public function getDeathPlace(): Place { foreach ($this->getAllDeathPlaces() as $place) { if ($place) { return $place; } } return new Place('', $this->tree); } /** * get the death year * * @return string the year of death */ public function getDeathYear(): string { return $this->getDeathDate()->minimumDate()->format('%Y'); } /** * Get the range of years in which a individual lived. e.g. “1870–”, “1870–1920”, “–1920”. * Provide the place and full date using a tooltip. * For consistent layout in charts, etc., show just a “–” when no dates are known. * Note that this is a (non-breaking) en-dash, and not a hyphen. * * @return string */ public function getLifeSpan(): string { // Just the first part of the place name $birth_place = strip_tags($this->getBirthPlace()->getShortName()); $death_place = strip_tags($this->getDeathPlace()->getShortName()); // Remove markup from dates $birth_date = strip_tags($this->getBirthDate()->display()); $death_date = strip_tags($this->getDeathDate()->display()); /* I18N: A range of years, e.g. “1870–”, “1870–1920”, “–1920” */ return I18N::translate( '%1$s–%2$s', '' . $this->getBirthYear() . '', '' . $this->getDeathYear() . '' ); } /** * Get all the birth dates - for the individual lists. * * @return Date[] */ public function getAllBirthDates(): array { foreach (explode('|', WT_EVENTS_BIRT) as $event) { $tmp = $this->getAllEventDates($event); if ($tmp) { return $tmp; } } return []; } /** * Gat all the birth places - for the individual lists. * * @return Place[] */ public function getAllBirthPlaces(): array { foreach (explode('|', WT_EVENTS_BIRT) as $event) { $places = $this->getAllEventPlaces($event); if (!empty($places)) { return $places; } } return []; } /** * Get all the death dates - for the individual lists. * * @return Date[] */ public function getAllDeathDates(): array { foreach (explode('|', WT_EVENTS_DEAT) as $event) { $tmp = $this->getAllEventDates($event); if ($tmp) { return $tmp; } } return []; } /** * Get all the death places - for the individual lists. * * @return Place[] */ public function getAllDeathPlaces(): array { foreach (explode('|', WT_EVENTS_DEAT) as $event) { $places = $this->getAllEventPlaces($event); if (!empty($places)) { return $places; } } return []; } /** * Generate an estimate for the date of birth, based on dates of parents/children/spouses * * @return Date */ public function getEstimatedBirthDate(): Date { if ($this->estimated_birth_date === null) { foreach ($this->getAllBirthDates() as $date) { if ($date->isOK()) { $this->estimated_birth_date = $date; break; } } if ($this->estimated_birth_date === null) { $min = []; $max = []; $tmp = $this->getDeathDate(); if ($tmp->isOK()) { $min[] = $tmp->minimumJulianDay() - $this->tree->getPreference('MAX_ALIVE_AGE') * 365; $max[] = $tmp->maximumJulianDay(); } foreach ($this->getChildFamilies() as $family) { $tmp = $family->getMarriageDate(); if ($tmp->isOK()) { $min[] = $tmp->maximumJulianDay() - 365 * 1; $max[] = $tmp->minimumJulianDay() + 365 * 30; } if ($parent = $family->getHusband()) { $tmp = $parent->getBirthDate(); if ($tmp->isOK()) { $min[] = $tmp->maximumJulianDay() + 365 * 15; $max[] = $tmp->minimumJulianDay() + 365 * 65; } } if ($parent = $family->getWife()) { $tmp = $parent->getBirthDate(); if ($tmp->isOK()) { $min[] = $tmp->maximumJulianDay() + 365 * 15; $max[] = $tmp->minimumJulianDay() + 365 * 45; } } foreach ($family->getChildren() as $child) { $tmp = $child->getBirthDate(); if ($tmp->isOK()) { $min[] = $tmp->maximumJulianDay() - 365 * 30; $max[] = $tmp->minimumJulianDay() + 365 * 30; } } } foreach ($this->getSpouseFamilies() as $family) { $tmp = $family->getMarriageDate(); if ($tmp->isOK()) { $min[] = $tmp->maximumJulianDay() - 365 * 45; $max[] = $tmp->minimumJulianDay() - 365 * 15; } $spouse = $family->getSpouse($this); if ($spouse) { $tmp = $spouse->getBirthDate(); if ($tmp->isOK()) { $min[] = $tmp->maximumJulianDay() - 365 * 25; $max[] = $tmp->minimumJulianDay() + 365 * 25; } } foreach ($family->getChildren() as $child) { $tmp = $child->getBirthDate(); if ($tmp->isOK()) { $min[] = $tmp->maximumJulianDay() - 365 * ($this->getSex() == 'F' ? 45 : 65); $max[] = $tmp->minimumJulianDay() - 365 * 15; } } } if ($min && $max) { $gregorian_calendar = new GregorianCalendar(); list($year) = $gregorian_calendar->jdToYmd((int)((max($min) + min($max)) / 2)); $this->estimated_birth_date = new Date('EST ' . $year); } else { $this->estimated_birth_date = new Date(''); // always return a date object } } } return $this->estimated_birth_date; } /** * Generate an estimated date of death. * * @return Date */ public function getEstimatedDeathDate(): Date { if ($this->estimated_death_date === null) { foreach ($this->getAllDeathDates() as $date) { if ($date->isOK()) { $this->estimated_death_date = $date; break; } } if ($this->estimated_death_date === null) { if ($this->getEstimatedBirthDate()->minimumJulianDay()) { $max_alive_age = (int)$this->tree->getPreference('MAX_ALIVE_AGE'); $this->estimated_death_date = $this->getEstimatedBirthDate()->addYears($max_alive_age, 'BEF'); } else { $this->estimated_death_date = new Date(''); // always return a date object } } } return $this->estimated_death_date; } /** * Get the sex - M F or U * Use the un-privatised gedcom record. We call this function during * the privatize-gedcom function, and we are allowed to know this. * * @return string */ public function getSex() { if (preg_match('/\n1 SEX ([MF])/', $this->gedcom . $this->pending, $match)) { return $match[1]; } return 'U'; } /** * Get the individual’s sex image * * @param string $size * * @return string */ public function getSexImage($size = 'small'): string { return self::sexImage($this->getSex(), $size); } /** * Generate a sex icon/image * * @param string $sex * @param string $size * * @return string */ public static function sexImage($sex, $size = 'small'): string { return ''; } /** * Generate the CSS class to be used for drawing this individual * * @return string */ public function getBoxStyle(): string { $tmp = [ 'M' => '', 'F' => 'F', 'U' => 'NN', ]; return 'person_box' . $tmp[$this->getSex()]; } /** * Get a list of this individual’s spouse families * * @param int|null $access_level * * @return Family[] */ public function getSpouseFamilies($access_level = null): array { if ($access_level === null) { $access_level = Auth::accessLevel($this->tree); } $SHOW_PRIVATE_RELATIONSHIPS = $this->tree->getPreference('SHOW_PRIVATE_RELATIONSHIPS'); $families = []; foreach ($this->getFacts('FAMS', false, $access_level, $SHOW_PRIVATE_RELATIONSHIPS) as $fact) { $family = $fact->getTarget(); if ($family && ($SHOW_PRIVATE_RELATIONSHIPS || $family->canShow($access_level))) { $families[] = $family; } } return $families; } /** * Get the current spouse of this individual. * * Where an individual has multiple spouses, assume they are stored * in chronological order, and take the last one found. * * @return Individual|null */ public function getCurrentSpouse() { $tmp = $this->getSpouseFamilies(); $family = end($tmp); if ($family) { return $family->getSpouse($this); } return null; } /** * Count the children belonging to this individual. * * @return int */ public function getNumberOfChildren() { if (preg_match('/\n1 NCHI (\d+)(?:\n|$)/', $this->getGedcom(), $match)) { return $match[1]; } $children = []; foreach ($this->getSpouseFamilies() as $fam) { foreach ($fam->getChildren() as $child) { $children[$child->getXref()] = true; } } return count($children); } /** * Get a list of this individual’s child families (i.e. their parents). * * @param int|null $access_level * * @return Family[] */ public function getChildFamilies($access_level = null): array { if ($access_level === null) { $access_level = Auth::accessLevel($this->tree); } $SHOW_PRIVATE_RELATIONSHIPS = $this->tree->getPreference('SHOW_PRIVATE_RELATIONSHIPS'); $families = []; foreach ($this->getFacts('FAMC', false, $access_level, $SHOW_PRIVATE_RELATIONSHIPS) as $fact) { $family = $fact->getTarget(); if ($family && ($SHOW_PRIVATE_RELATIONSHIPS || $family->canShow($access_level))) { $families[] = $family; } } return $families; } /** * Get the preferred parents for this individual. * * An individual may multiple parents (e.g. birth, adopted, disputed). * The preferred family record is: * (a) the first one with an explicit tag "_PRIMARY Y" * (b) the first one with a pedigree of "birth" * (c) the first one with no pedigree (default is "birth") * (d) the first one found * * @return Family|null */ public function getPrimaryChildFamily() { $families = $this->getChildFamilies(); switch (count($families)) { case 0: return null; case 1: return $families[0]; default: // If there is more than one FAMC record, choose the preferred parents: // a) records with '2 _PRIMARY' foreach ($families as $fam) { $famid = $fam->getXref(); if (preg_match("/\n1 FAMC @{$famid}@\n(?:[2-9].*\n)*(?:2 _PRIMARY Y)/", $this->getGedcom())) { return $fam; } } // b) records with '2 PEDI birt' foreach ($families as $fam) { $famid = $fam->getXref(); if (preg_match("/\n1 FAMC @{$famid}@\n(?:[2-9].*\n)*(?:2 PEDI birth)/", $this->getGedcom())) { return $fam; } } // c) records with no '2 PEDI' foreach ($families as $fam) { $famid = $fam->getXref(); if (!preg_match("/\n1 FAMC @{$famid}@\n(?:[2-9].*\n)*(?:2 PEDI)/", $this->getGedcom())) { return $fam; } } // d) any record return $families[0]; } } /** * Get a list of step-parent families. * * @return Family[] */ public function getChildStepFamilies(): array { $step_families = []; $families = $this->getChildFamilies(); foreach ($families as $family) { $father = $family->getHusband(); if ($father) { foreach ($father->getSpouseFamilies() as $step_family) { if (!in_array($step_family, $families, true)) { $step_families[] = $step_family; } } } $mother = $family->getWife(); if ($mother) { foreach ($mother->getSpouseFamilies() as $step_family) { if (!in_array($step_family, $families, true)) { $step_families[] = $step_family; } } } } return $step_families; } /** * Get a list of step-parent families. * * @return Family[] */ public function getSpouseStepFamilies(): array { $step_families = []; $families = $this->getSpouseFamilies(); foreach ($families as $family) { $spouse = $family->getSpouse($this); if ($spouse) { foreach ($family->getSpouse($this)->getSpouseFamilies() as $step_family) { if (!in_array($step_family, $families, true)) { $step_families[] = $step_family; } } } } return $step_families; } /** * A label for a parental family group * * @param Family $family * * @return string */ public function getChildFamilyLabel(Family $family) { if (preg_match('/\n1 FAMC @' . $family->getXref() . '@(?:\n[2-9].*)*\n2 PEDI (.+)/', $this->getGedcom(), $match)) { // A specified pedigree return GedcomCodePedi::getChildFamilyLabel($match[1]); } // Default (birth) pedigree return GedcomCodePedi::getChildFamilyLabel(''); } /** * Create a label for a step family * * @param Family $step_family * * @return string */ public function getStepFamilyLabel(Family $step_family): string { foreach ($this->getChildFamilies() as $family) { if ($family !== $step_family) { // Must be a step-family foreach ($family->getSpouses() as $parent) { foreach ($step_family->getSpouses() as $step_parent) { if ($parent === $step_parent) { // One common parent - must be a step family if ($parent->getSex() == 'M') { // Father’s family with someone else if ($step_family->getSpouse($step_parent)) { /* I18N: A step-family. %s is an individual’s name */ return I18N::translate('Father’s family with %s', $step_family->getSpouse($step_parent)->getFullName()); } /* I18N: A step-family. */ return I18N::translate('Father’s family with an unknown individual'); } // Mother’s family with someone else if ($step_family->getSpouse($step_parent)) { /* I18N: A step-family. %s is an individual’s name */ return I18N::translate('Mother’s family with %s', $step_family->getSpouse($step_parent)->getFullName()); } /* I18N: A step-family. */ return I18N::translate('Mother’s family with an unknown individual'); } } } } } // Perahps same parents - but a different family record? return I18N::translate('Family with parents'); } /** * Get the description for the family. * * For example, "XXX's family with new wife". * * @param Family $family * * @return string */ public function getSpouseFamilyLabel(Family $family) { $spouse = $family->getSpouse($this); if ($spouse) { /* I18N: %s is the spouse name */ return I18N::translate('Family with %s', $spouse->getFullName()); } return $family->getFullName(); } /** * get primary parents names for this individual * * @param string $classname optional css class * @param string $display optional css style display * * @return string a div block with father & mother names */ public function getPrimaryParentsNames($classname = '', $display = ''): string { $fam = $this->getPrimaryChildFamily(); if (!$fam) { return ''; } $txt = '