. */ 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 $_getEstimatedBirthDate; /** @var Date The estimated date of death */ private $_getEstimatedDeathDate; /** * 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($xref, Tree $tree, $gedcom = null) { $record = parent::getInstance($xref, $tree, $gedcom); if ($record instanceof Individual) { return $record; } else { 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($access_level = null) { 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($access_level) { // 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) { 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) { $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($xref, $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) { 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) { 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() { $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) { $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() { foreach ($this->getAllBirthDates() as $date) { if ($date->isOK()) { return $date; } } return new Date(''); } /** * Get the place of birth * * @return Place */ public function getBirthPlace() { 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() { return $this->getBirthDate()->minimumDate()->format('%Y'); } /** * Get the date of death * * @return Date */ public function getDeathDate() { foreach ($this->getAllDeathDates() as $date) { if ($date->isOK()) { return $date; } } return new Date(''); } /** * Get the place of death * * @return Place */ public function getDeathPlace() { 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() { 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() { // 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()); return /* I18N: A range of years, e.g. “1870–”, “1870–1920”, “–1920” */ I18N::translate( '%1$s–%2$s', '' . $this->getBirthYear() . '', '' . $this->getDeathYear() . '' ); } /** * Get all the birth dates - for the individual lists. * * @return Date[] */ public function getAllBirthDates() { 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() { 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() { 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() { 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() { if (is_null($this->_getEstimatedBirthDate)) { foreach ($this->getAllBirthDates() as $date) { if ($date->isOK()) { $this->_getEstimatedBirthDate = $date; break; } } if (is_null($this->_getEstimatedBirthDate)) { $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->_getEstimatedBirthDate = new Date('EST ' . $year); } else { $this->_getEstimatedBirthDate = new Date(''); // always return a date object } } } return $this->_getEstimatedBirthDate; } /** * Generate an estimated date of death. * * @return Date */ public function getEstimatedDeathDate() { if ($this->_getEstimatedDeathDate === null) { foreach ($this->getAllDeathDates() as $date) { if ($date->isOK()) { $this->_getEstimatedDeathDate = $date; break; } } if ($this->_getEstimatedDeathDate === null) { if ($this->getEstimatedBirthDate()->minimumJulianDay()) { $max_alive_age = (int) $this->tree->getPreference('MAX_ALIVE_AGE'); $this->_getEstimatedDeathDate = $this->getEstimatedBirthDate()->addYears($max_alive_age, 'BEF'); } else { $this->_getEstimatedDeathDate = new Date(''); // always return a date object } } } return $this->_getEstimatedDeathDate; } /** * 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]; } else { return 'U'; } } /** * Get the individual’s sex image * * @param string $size * * @return string */ public function getSexImage($size = 'small') { 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') { return ''; } /** * Generate the CSS class to be used for drawing this individual * * @return string */ public function getBoxStyle() { $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) { 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); } else { 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]; } else { $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) { 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 $famid => $fam) { 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 $famid => $fam) { 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 $famid => $fam) { 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() { $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() { $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]); } else { // 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) { 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)) { return /* I18N: A step-family. %s is an individual’s name */ I18N::translate('Father’s family with %s', $step_family->getSpouse($step_parent)->getFullName()); } else { return /* I18N: A step-family. */ I18N::translate('Father’s family with an unknown individual'); } } else { // Mother’s family with someone else if ($step_family->getSpouse($step_parent)) { return /* I18N: A step-family. %s is an individual’s name */ I18N::translate('Mother’s family with %s', $step_family->getSpouse($step_parent)->getFullName()); } else { return /* I18N: A step-family. */ 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) { return /* I18N: %s is the spouse name */ I18N::translate('Family with %s', $spouse->getFullName()); } else { 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 = '') { $fam = $this->getPrimaryChildFamily(); if (!$fam) { return ''; } $txt = 'getHusband(); if ($husb) { // Temporarily reset the 'prefered' display name, as we always // want the default name, not the one selected for display on the indilist. $primary = $husb->getPrimaryName(); $husb->setPrimaryName(null); $txt .= /* I18N: %s is the name of an individual’s father */ I18N::translate('Father: %s', $husb->getFullName()) . '
'; $husb->setPrimaryName($primary); } $wife = $fam->getWife(); if ($wife) { // Temporarily reset the 'prefered' display name, as we always // want the default name, not the one selected for display on the indilist. $primary = $wife->getPrimaryName(); $wife->setPrimaryName(null); $txt .= /* I18N: %s is the name of an individual’s mother */ I18N::translate('Mother: %s', $wife->getFullName()); $wife->setPrimaryName($primary); } $txt .= ''; return $txt; } /** {@inheritdoc} */ public function getFallBackName() { return '@P.N. /@N.N./'; } /** * Convert a name record into ‘full’ and ‘sort’ versions. * Use the NAME field to generate the ‘full’ version, as the * gedcom spec says that this is the individual’s name, as they would write it. * Use the SURN field to generate the sortable names. Note that this field * may also be used for the ‘true’ surname, perhaps spelt differently to that * recorded in the NAME field. e.g. * * 1 NAME Robert /de Gliderow/ * 2 GIVN Robert * 2 SPFX de * 2 SURN CLITHEROW * 2 NICK The Bald * * full=>'Robert de Gliderow 'The Bald'' * sort=>'CLITHEROW, ROBERT' * * Handle multiple surnames, either as; * * 1 NAME Carlos /Vasquez/ y /Sante/ * or * 1 NAME Carlos /Vasquez y Sante/ * 2 GIVN Carlos * 2 SURN Vasquez,Sante * * @param string $type * @param string $full * @param string $gedcom */ protected function addName($type, $full, $gedcom) { //////////////////////////////////////////////////////////////////////////// // Extract the structured name parts - use for "sortable" names and indexes //////////////////////////////////////////////////////////////////////////// $sublevel = 1 + (int) $gedcom[0]; $NPFX = preg_match("/\n{$sublevel} NPFX (.+)/", $gedcom, $match) ? $match[1] : ''; $GIVN = preg_match("/\n{$sublevel} GIVN (.+)/", $gedcom, $match) ? $match[1] : ''; $SURN = preg_match("/\n{$sublevel} SURN (.+)/", $gedcom, $match) ? $match[1] : ''; $NSFX = preg_match("/\n{$sublevel} NSFX (.+)/", $gedcom, $match) ? $match[1] : ''; $NICK = preg_match("/\n{$sublevel} NICK (.+)/", $gedcom, $match) ? $match[1] : ''; // SURN is an comma-separated list of surnames... if ($SURN) { $SURNS = preg_split('/ *, */', $SURN); } else { $SURNS = []; } // ...so is GIVN - but nobody uses it like that $GIVN = str_replace('/ *, */', ' ', $GIVN); //////////////////////////////////////////////////////////////////////////// // Extract the components from NAME - use for the "full" names //////////////////////////////////////////////////////////////////////////// // Fix bad slashes. e.g. 'John/Smith' => 'John/Smith/' if (substr_count($full, '/') % 2 == 1) { $full = $full . '/'; } // GEDCOM uses "//" to indicate an unknown surname $full = preg_replace('/\/\//', '/@N.N./', $full); // Extract the surname. // Note, there may be multiple surnames, e.g. Jean /Vasquez/ y /Cortes/ if (preg_match('/\/.*\//', $full, $match)) { $surname = str_replace('/', '', $match[0]); } else { $surname = ''; } // If we don’t have a SURN record, extract it from the NAME if (!$SURNS) { if (preg_match_all('/\/([^\/]*)\//', $full, $matches)) { // There can be many surnames, each wrapped with '/' $SURNS = $matches[1]; foreach ($SURNS as $n => $SURN) { // Remove surname prefixes, such as "van de ", "d'" and "'t " (lower case only) $SURNS[$n] = preg_replace('/^(?:[a-z]+ |[a-z]+\' ?|\'[a-z]+ )+/', '', $SURN); } } else { // It is valid not to have a surname at all $SURNS = ['']; } } // If we don’t have a GIVN record, extract it from the NAME if (!$GIVN) { $GIVN = preg_replace( [ '/ ?\/.*\/ ?/', // remove surname '/ ?".+"/', // remove nickname '/ {2,}/', // multiple spaces, caused by the above '/^ | $/', // leading/trailing spaces, caused by the above ], [ ' ', ' ', ' ', '', ], $full ); } // Add placeholder for unknown given name if (!$GIVN) { $GIVN = '@P.N.'; $pos = strpos($full, '/'); $full = substr($full, 0, $pos) . '@P.N. ' . substr($full, $pos); } // GEDCOM 5.5.1 nicknames should be specificied in a NICK field // GEDCOM 5.5 nicknames should be specified in the NAME field, surrounded by quotes if ($NICK && strpos($full, '"' . $NICK . '"') === false) { // A NICK field is present, but not included in the NAME. Show it at the end. $full .= ' "' . $NICK . '"'; } // Remove slashes - they don’t get displayed // $fullNN keeps the @N.N. placeholders, for the database // $full is for display on-screen $fullNN = str_replace('/', '', $full); // Insert placeholders for any missing/unknown names $full = str_replace('@N.N.', I18N::translateContext('Unknown surname', '…'), $full); $full = str_replace('@P.N.', I18N::translateContext('Unknown given name', '…'), $full); // Format for display $full = '' . preg_replace('/\/([^\/]*)\//', '$1', e($full)) . ''; // Localise quotation marks around the nickname $full = preg_replace_callback('/"([^&]*)"/', function ($matches) { return I18N::translate('“%s”', $matches[1]); }, $full); // A suffix of “*” indicates a preferred name $full = preg_replace('/([^ >]*)\*/', '\\1', $full); // Remove prefered-name indicater - they don’t go in the database $GIVN = str_replace('*', '', $GIVN); $fullNN = str_replace('*', '', $fullNN); foreach ($SURNS as $SURN) { // Scottish 'Mc and Mac ' prefixes both sort under 'Mac' if (strcasecmp(substr($SURN, 0, 2), 'Mc') == 0) { $SURN = substr_replace($SURN, 'Mac', 0, 2); } elseif (strcasecmp(substr($SURN, 0, 4), 'Mac ') == 0) { $SURN = substr_replace($SURN, 'Mac', 0, 4); } $this->_getAllNames[] = [ 'type' => $type, 'sort' => $SURN . ',' . $GIVN, 'full' => $full, // This is used for display 'fullNN' => $fullNN, // This goes into the database 'surname' => $surname, // This goes into the database 'givn' => $GIVN, // This goes into the database 'surn' => $SURN, // This goes into the database ]; } } /** * Extract names from the GEDCOM record. */ public function extractNames() { $this->extractNamesFromFacts(1, 'NAME', $this->getFacts('NAME', false, Auth::accessLevel($this->tree), $this->canShowName())); } /** * Extra info to display when displaying this record in a list of * selection items or favorites. * * @return string */ public function formatListDetails() { return $this->formatFirstMajorFact(WT_EVENTS_BIRT, 1) . $this->formatFirstMajorFact(WT_EVENTS_DEAT, 1); } /** * Create a short name for compact display on charts * * @return string */ public function getShortName() { global $bwidth; // Estimate number of characters that can fit in box. Calulates to 28 characters in webtrees theme, or 34 if no thumbnail used. if ($this->tree->getPreference('SHOW_HIGHLIGHT_IMAGES')) { $char = intval(($bwidth - 40) / 6.5); } else { $char = ($bwidth / 6.5); } if ($this->canShowName()) { $tmp = $this->getAllNames(); $givn = $tmp[$this->getPrimaryName()]['givn']; $surn = $tmp[$this->getPrimaryName()]['surname']; $new_givn = explode(' ', $givn); $count_givn = count($new_givn); $len_givn = mb_strlen($givn); $len_surn = mb_strlen($surn); $len = $len_givn + $len_surn; $i = 1; while ($len > $char && $i <= $count_givn) { $new_givn[$count_givn - $i] = mb_substr($new_givn[$count_givn - $i], 0, 1); $givn = implode(' ', $new_givn); $len_givn = mb_strlen($givn); $len = $len_givn + $len_surn; $i++; } $max_surn = $char - $i * 2; if ($len_surn > $max_surn) { $surn = substr($surn, 0, $max_surn) . '…'; } $shortname = str_replace( ['@P.N.', '@N.N.'], [I18N::translateContext('Unknown given name', '…'), I18N::translateContext('Unknown surname', '…')], $givn . ' ' . $surn ); return $shortname; } else { return I18N::translate('Private'); } } }