. */ declare(strict_types=1); namespace Fisharebest\Webtrees\Factories; use Closure; use Fisharebest\Webtrees\Contracts\FamilyFactoryInterface; use Fisharebest\Webtrees\Factory; use Fisharebest\Webtrees\Family; use Fisharebest\Webtrees\Gedcom; use Fisharebest\Webtrees\Tree; use Illuminate\Database\Capsule\Manager as DB; use stdClass; use function assert; use function preg_match; /** * Make a Family object. */ class FamilyFactory extends AbstractGedcomRecordFactory implements FamilyFactoryInterface { private const TYPE_CHECK_REGEX = '/^0 @[^@]+@ ' . Family::RECORD_TYPE . '/'; /** * Create a family. * * @param string $xref * @param Tree $tree * @param string|null $gedcom * * @return Family|null */ public function make(string $xref, Tree $tree, string $gedcom = null): ?Family { return $this->cache->remember(__CLASS__ . $xref . '@' . $tree->id(), function () use ($xref, $tree, $gedcom) { $gedcom = $gedcom ?? $this->gedcom($xref, $tree); $pending = $this->pendingChanges($tree)->get($xref); if ($gedcom === null && ($pending === null || !preg_match(self::TYPE_CHECK_REGEX, $pending))) { return null; } $xref = $this->extractXref($gedcom ?? $pending, $xref); // Preload all the family members using a single database query. preg_match_all('/\n1 (?:HUSB|WIFE|CHIL) @(' . Gedcom::REGEX_XREF . ')@/', $gedcom . "\n" . $pending, $match); DB::table('individuals') ->where('i_file', '=', $tree->id()) ->whereIn('i_id', $match[1]) ->get() ->map(Factory::individual()->mapper($tree)); return new Family($xref, $gedcom ?? '', $pending, $tree); }); } /** * Create a Family object from a row in the database. * * @param Tree $tree * * @return Closure */ public function mapper(Tree $tree): Closure { return function (stdClass $row) use ($tree): Family { $family = $this->make($row->f_id, $tree, $row->f_gedcom); assert($family instanceof Family); return $family; }; } /** * Create a Family object from raw GEDCOM data. * * @param string $xref * @param string $gedcom an empty string for new/pending records * @param string|null $pending null for a record with no pending edits, * empty string for records with pending deletions * @param Tree $tree * * @return Family */ public function new(string $xref, string $gedcom, ?string $pending, Tree $tree): Family { return new Family($xref, $gedcom, $pending, $tree); } /** * Fetch GEDCOM data from the database. * * @param string $xref * @param Tree $tree * * @return string|null */ protected function gedcom(string $xref, Tree $tree): ?string { return DB::table('families') ->where('f_id', '=', $xref) ->where('f_file', '=', $tree->id()) ->value('f_gedcom'); } }