18c2e8227SGreg Roach<?php 28c2e8227SGreg Roach/** 38c2e8227SGreg Roach * webtrees: online genealogy 46bdf7674SGreg Roach * Copyright (C) 2017 webtrees development team 58c2e8227SGreg Roach * This program is free software: you can redistribute it and/or modify 68c2e8227SGreg Roach * it under the terms of the GNU General Public License as published by 78c2e8227SGreg Roach * the Free Software Foundation, either version 3 of the License, or 88c2e8227SGreg Roach * (at your option) any later version. 98c2e8227SGreg Roach * This program is distributed in the hope that it will be useful, 108c2e8227SGreg Roach * but WITHOUT ANY WARRANTY; without even the implied warranty of 118c2e8227SGreg Roach * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 128c2e8227SGreg Roach * GNU General Public License for more details. 138c2e8227SGreg Roach * You should have received a copy of the GNU General Public License 148c2e8227SGreg Roach * along with this program. If not, see <http://www.gnu.org/licenses/>. 158c2e8227SGreg Roach */ 1676692c8bSGreg Roachnamespace Fisharebest\Webtrees\Module; 1776692c8bSGreg Roach 180e62c4b8SGreg Roachuse Fisharebest\Webtrees\Auth; 1915d603e7SGreg Roachuse Fisharebest\Webtrees\Bootstrap4; 206e45321fSGreg Roachuse Fisharebest\Webtrees\Database; 210e62c4b8SGreg Roachuse Fisharebest\Webtrees\Filter; 2215d603e7SGreg Roachuse Fisharebest\Webtrees\FontAwesome; 233d7a8a4cSGreg Roachuse Fisharebest\Webtrees\Functions\FunctionsEdit; 246e45321fSGreg Roachuse Fisharebest\Webtrees\GedcomRecord; 256e45321fSGreg Roachuse Fisharebest\Webtrees\GedcomTag; 26047f239bSGreg Roachuse Fisharebest\Webtrees\Html; 270e62c4b8SGreg Roachuse Fisharebest\Webtrees\I18N; 286e45321fSGreg Roachuse Fisharebest\Webtrees\Individual; 290e62c4b8SGreg Roachuse Fisharebest\Webtrees\Theme; 306e45321fSGreg Roachuse Fisharebest\Webtrees\Tree; 31*577e6b29SGreg Roachuse Ramsey\Uuid\Uuid; 328c2e8227SGreg Roach 338c2e8227SGreg Roach/** 348c2e8227SGreg Roach * Class RecentChangesModule 358c2e8227SGreg Roach */ 36e2a378d3SGreg Roachclass RecentChangesModule extends AbstractModule implements ModuleBlockInterface { 376e45321fSGreg Roach const DEFAULT_BLOCK = '1'; 388c2e8227SGreg Roach const DEFAULT_DAYS = 7; 396e45321fSGreg Roach const DEFAULT_HIDE_EMPTY = '0'; 406e45321fSGreg Roach const DEFAULT_SHOW_USER = '1'; 416e45321fSGreg Roach const DEFAULT_SORT_STYLE = 'date_desc'; 426e45321fSGreg Roach const DEFAULT_INFO_STYLE = 'table'; 438c2e8227SGreg Roach const MAX_DAYS = 90; 448c2e8227SGreg Roach 458c2e8227SGreg Roach /** {@inheritdoc} */ 468c2e8227SGreg Roach public function getTitle() { 478c2e8227SGreg Roach return /* I18N: Name of a module */ I18N::translate('Recent changes'); 488c2e8227SGreg Roach } 498c2e8227SGreg Roach 508c2e8227SGreg Roach /** {@inheritdoc} */ 518c2e8227SGreg Roach public function getDescription() { 528c2e8227SGreg Roach return /* I18N: Description of the “Recent changes” module */ I18N::translate('A list of records that have been updated recently.'); 538c2e8227SGreg Roach } 548c2e8227SGreg Roach 556e45321fSGreg Roach /** {@inheritdoc} */ 5613abd6f3SGreg Roach public function getBlock($block_id, $template = true, $cfg = []) { 574b9ff166SGreg Roach global $ctype, $WT_TREE; 588c2e8227SGreg Roach 59e2a378d3SGreg Roach $days = $this->getBlockSetting($block_id, 'days', self::DEFAULT_DAYS); 606e45321fSGreg Roach $infoStyle = $this->getBlockSetting($block_id, 'infoStyle', self::DEFAULT_INFO_STYLE); 616e45321fSGreg Roach $sortStyle = $this->getBlockSetting($block_id, 'sortStyle', self::DEFAULT_SORT_STYLE); 626e45321fSGreg Roach $show_user = $this->getBlockSetting($block_id, 'show_user', self::DEFAULT_SHOW_USER); 638c2e8227SGreg Roach 6415d603e7SGreg Roach foreach (['days', 'infoStyle', 'sortStyle', 'show_user'] as $name) { 658c2e8227SGreg Roach if (array_key_exists($name, $cfg)) { 668c2e8227SGreg Roach $$name = $cfg[$name]; 678c2e8227SGreg Roach } 688c2e8227SGreg Roach } 698c2e8227SGreg Roach 7024319d9dSGreg Roach $records = $this->getRecentChanges($WT_TREE, $days); 718c2e8227SGreg Roach 728c2e8227SGreg Roach // Print block header 738c2e8227SGreg Roach $id = $this->getName() . $block_id; 748c2e8227SGreg Roach $class = $this->getName() . '_block'; 756e45321fSGreg Roach 764b9ff166SGreg Roach if ($ctype === 'gedcom' && Auth::isManager($WT_TREE) || $ctype === 'user' && Auth::check()) { 7715d603e7SGreg Roach $title = FontAwesome::linkIcon('preferences', I18N::translate('Preferences'), ['class' => 'btn btn-link', 'href' => 'block_edit.php?block_id=' . $block_id . '&ged=' . $WT_TREE->getNameHtml() . '&ctype=' . $ctype]) . ' '; 788c2e8227SGreg Roach } else { 798c2e8227SGreg Roach $title = ''; 808c2e8227SGreg Roach } 818c2e8227SGreg Roach $title .= /* I18N: title for list of recent changes */ I18N::plural('Changes in the last %s day', 'Changes in the last %s days', $days, I18N::number($days)); 828c2e8227SGreg Roach 838c2e8227SGreg Roach $content = ''; 848c2e8227SGreg Roach // Print block content 856e45321fSGreg Roach if (count($records) == 0) { 868c2e8227SGreg Roach $content .= I18N::plural('There have been no changes within the last %s day.', 'There have been no changes within the last %s days.', $days, I18N::number($days)); 878c2e8227SGreg Roach } else { 888c2e8227SGreg Roach switch ($infoStyle) { 898c2e8227SGreg Roach case 'list': 906e45321fSGreg Roach $content .= $this->changesList($records, $sortStyle, $show_user); 918c2e8227SGreg Roach break; 928c2e8227SGreg Roach case 'table': 936e45321fSGreg Roach $content .= $this->changesTable($records, $sortStyle, $show_user); 948c2e8227SGreg Roach break; 958c2e8227SGreg Roach } 968c2e8227SGreg Roach } 978c2e8227SGreg Roach 988c2e8227SGreg Roach if ($template) { 998c2e8227SGreg Roach return Theme::theme()->formatBlock($id, $title, $class, $content); 1008c2e8227SGreg Roach } else { 1018c2e8227SGreg Roach return $content; 1028c2e8227SGreg Roach } 1038c2e8227SGreg Roach } 1048c2e8227SGreg Roach 1058c2e8227SGreg Roach /** {@inheritdoc} */ 1068c2e8227SGreg Roach public function loadAjax() { 1078c2e8227SGreg Roach return true; 1088c2e8227SGreg Roach } 1098c2e8227SGreg Roach 1108c2e8227SGreg Roach /** {@inheritdoc} */ 1118c2e8227SGreg Roach public function isUserBlock() { 1128c2e8227SGreg Roach return true; 1138c2e8227SGreg Roach } 1148c2e8227SGreg Roach 1158c2e8227SGreg Roach /** {@inheritdoc} */ 1168c2e8227SGreg Roach public function isGedcomBlock() { 1178c2e8227SGreg Roach return true; 1188c2e8227SGreg Roach } 1198c2e8227SGreg Roach 1206e45321fSGreg Roach /** {@inheritdoc} */ 1218c2e8227SGreg Roach public function configureBlock($block_id) { 1228c2e8227SGreg Roach if (Filter::postBool('save') && Filter::checkCsrf()) { 1236e45321fSGreg Roach $this->setBlockSetting($block_id, 'days', Filter::postInteger('days', 1, self::MAX_DAYS)); 1246e45321fSGreg Roach $this->setBlockSetting($block_id, 'infoStyle', Filter::post('infoStyle', 'list|table')); 1256e45321fSGreg Roach $this->setBlockSetting($block_id, 'sortStyle', Filter::post('sortStyle', 'name|date_asc|date_desc')); 126782f6af0SGreg Roach $this->setBlockSetting($block_id, 'show_user', Filter::postBool('show_user')); 1278c2e8227SGreg Roach } 1288c2e8227SGreg Roach 129e2a378d3SGreg Roach $days = $this->getBlockSetting($block_id, 'days', self::DEFAULT_DAYS); 1306e45321fSGreg Roach $infoStyle = $this->getBlockSetting($block_id, 'infoStyle', self::DEFAULT_INFO_STYLE); 1316e45321fSGreg Roach $sortStyle = $this->getBlockSetting($block_id, 'sortStyle', self::DEFAULT_SORT_STYLE); 1326e45321fSGreg Roach $show_user = $this->getBlockSetting($block_id, 'show_user', self::DEFAULT_SHOW_USER); 1338c2e8227SGreg Roach 13415d603e7SGreg Roach echo '<div class="form-group row"><label class="col-sm-3 col-form-label" for="days">'; 1358c2e8227SGreg Roach echo I18N::translate('Number of days to show'); 13615d603e7SGreg Roach echo '</div><div class="col-sm-9">'; 1378c2e8227SGreg Roach echo '<input type="text" name="days" size="2" value="', $days, '">'; 13815d603e7SGreg Roach echo ' ' . I18N::plural('maximum %s day', 'maximum %s days', I18N::number(self::MAX_DAYS), I18N::number(self::MAX_DAYS)); 13915d603e7SGreg Roach echo '</div></div>'; 1408c2e8227SGreg Roach 14115d603e7SGreg Roach echo '<div class="form-group row"><label class="col-sm-3 col-form-label" for="infoStyle">'; 1428c2e8227SGreg Roach echo I18N::translate('Presentation style'); 14315d603e7SGreg Roach echo '</div><div class="col-sm-9">'; 14415d603e7SGreg Roach echo Bootstrap4::select(['list' => I18N::translate('list'), 'table' => I18N::translate('table')], $infoStyle, ['id' => 'infoStyle', 'name' => 'infoStyle']); 14515d603e7SGreg Roach echo '</div></div>'; 1468c2e8227SGreg Roach 14715d603e7SGreg Roach echo '<div class="form-group row"><label class="col-sm-3 col-form-label" for="sortStyle">'; 1488c2e8227SGreg Roach echo I18N::translate('Sort order'); 14915d603e7SGreg Roach echo '</div><div class="col-sm-9">'; 15015d603e7SGreg Roach echo Bootstrap4::select([ 1518c2e8227SGreg Roach 'name' => /* I18N: An option in a list-box */ I18N::translate('sort by name'), 1528c2e8227SGreg Roach 'date_asc' => /* I18N: An option in a list-box */ I18N::translate('sort by date, oldest first'), 153cbc1590aSGreg Roach 'date_desc' => /* I18N: An option in a list-box */ I18N::translate('sort by date, newest first'), 15415d603e7SGreg Roach ], $sortStyle, ['id' => 'sortStyle', 'name' => 'sortStyle']); 15515d603e7SGreg Roach echo '</div></div>'; 1568c2e8227SGreg Roach 15715d603e7SGreg Roach echo '<div class="form-group row"><label class="col-sm-3 col-form-label" for="show_usere">'; 158782f6af0SGreg Roach echo /* I18N: label for a yes/no option */ I18N::translate('Show the user who made the change'); 15915d603e7SGreg Roach echo '</div><div class="col-sm-9">'; 16015d603e7SGreg Roach echo Bootstrap4::radioButtons('show_user', FunctionsEdit::optionsNoYes(), $show_user, true); 16115d603e7SGreg Roach echo '</div></div>'; 1628c2e8227SGreg Roach } 1638c2e8227SGreg Roach 1646e45321fSGreg Roach /** 1656e45321fSGreg Roach * Find records that have changed since a given julian day 1666e45321fSGreg Roach * 1676e45321fSGreg Roach * @param Tree $tree Changes for which tree 16824319d9dSGreg Roach * @param int $days Number of days 1696e45321fSGreg Roach * 1706e45321fSGreg Roach * @return GedcomRecord[] List of records with changes 1716e45321fSGreg Roach */ 17224319d9dSGreg Roach private function getRecentChanges(Tree $tree, $days) { 1736e45321fSGreg Roach $sql = 17424319d9dSGreg Roach "SELECT xref FROM `##change`" . 175b03b6f21SGreg Roach " WHERE new_gedcom != '' AND change_time > DATE_SUB(NOW(), INTERVAL :days DAY) AND gedcom_id = :tree_id" . 17624319d9dSGreg Roach " GROUP BY xref" . 17724319d9dSGreg Roach " ORDER BY MAX(change_id) DESC"; 1786e45321fSGreg Roach 17913abd6f3SGreg Roach $vars = [ 18024319d9dSGreg Roach 'days' => $days, 1816e45321fSGreg Roach 'tree_id' => $tree->getTreeId(), 18213abd6f3SGreg Roach ]; 1836e45321fSGreg Roach 1846e45321fSGreg Roach $xrefs = Database::prepare($sql)->execute($vars)->fetchOneColumn(); 1856e45321fSGreg Roach 18613abd6f3SGreg Roach $records = []; 1876e45321fSGreg Roach foreach ($xrefs as $xref) { 1886e45321fSGreg Roach $record = GedcomRecord::getInstance($xref, $tree); 1898e8029e5SRico Sonntag if ($record && $record->canShow()) { 1906e45321fSGreg Roach $records[] = $record; 1916e45321fSGreg Roach } 1926e45321fSGreg Roach } 1936e45321fSGreg Roach 1946e45321fSGreg Roach return $records; 1956e45321fSGreg Roach } 1966e45321fSGreg Roach 1976e45321fSGreg Roach /** 1986e45321fSGreg Roach * Format a table of events 1996e45321fSGreg Roach * 2006e45321fSGreg Roach * @param GedcomRecord[] $records 2016e45321fSGreg Roach * @param string $sort 2026e45321fSGreg Roach * @param bool $show_user 2036e45321fSGreg Roach * 2046e45321fSGreg Roach * @return string 2056e45321fSGreg Roach */ 2066e45321fSGreg Roach private function changesList(array $records, $sort, $show_user) { 2076e45321fSGreg Roach switch ($sort) { 2086e45321fSGreg Roach case 'name': 20913abd6f3SGreg Roach uasort($records, ['self', 'sortByNameAndChangeDate']); 2106e45321fSGreg Roach break; 2116e45321fSGreg Roach case 'date_asc': 21213abd6f3SGreg Roach uasort($records, ['self', 'sortByChangeDateAndName']); 2136e45321fSGreg Roach $records = array_reverse($records); 2146e45321fSGreg Roach break; 2156e45321fSGreg Roach case 'date_desc': 21613abd6f3SGreg Roach uasort($records, ['self', 'sortByChangeDateAndName']); 2176e45321fSGreg Roach } 2186e45321fSGreg Roach 2196e45321fSGreg Roach $html = ''; 2206e45321fSGreg Roach foreach ($records as $record) { 2216e45321fSGreg Roach $html .= '<a href="' . $record->getHtmlUrl() . '" class="list_item name2">' . $record->getFullName() . '</a>'; 2226e45321fSGreg Roach $html .= '<div class="indent" style="margin-bottom: 5px;">'; 2236e45321fSGreg Roach if ($record instanceof Individual) { 2246e45321fSGreg Roach if ($record->getAddName()) { 2256e45321fSGreg Roach $html .= '<a href="' . $record->getHtmlUrl() . '" class="list_item">' . $record->getAddName() . '</a>'; 2266e45321fSGreg Roach } 2276e45321fSGreg Roach } 22856834ce1SGreg Roach 22956834ce1SGreg Roach // The timestamp may be missing or private. 23056834ce1SGreg Roach $timestamp = $record->lastChangeTimestamp(); 23156834ce1SGreg Roach if ($timestamp !== '') { 2326e45321fSGreg Roach if ($show_user) { 2336e45321fSGreg Roach $html .= /* I18N: [a record was] Changed on <date/time> by <user> */ 234cc5ab399SGreg Roach I18N::translate('Changed on %1$s by %2$s', $timestamp, Html::escape($record->lastChangeUser())); 2356e45321fSGreg Roach } else { 2366e45321fSGreg Roach $html .= /* I18N: [a record was] Changed on <date/time> */ 23756834ce1SGreg Roach I18N::translate('Changed on %1$s', $timestamp); 23856834ce1SGreg Roach } 2396e45321fSGreg Roach } 2406e45321fSGreg Roach $html .= '</div>'; 2416e45321fSGreg Roach } 2426e45321fSGreg Roach 2436e45321fSGreg Roach return $html; 2446e45321fSGreg Roach } 2456e45321fSGreg Roach 2466e45321fSGreg Roach /** 2476e45321fSGreg Roach * Format a table of events 2486e45321fSGreg Roach * 2496e45321fSGreg Roach * @param GedcomRecord[] $records 2506e45321fSGreg Roach * @param string $sort 2516e45321fSGreg Roach * @param bool $show_user 2526e45321fSGreg Roach * 2536e45321fSGreg Roach * @return string 2546e45321fSGreg Roach */ 2556e45321fSGreg Roach private function changesTable($records, $sort, $show_user) { 2566e45321fSGreg Roach global $controller; 2576e45321fSGreg Roach 2586e45321fSGreg Roach $table_id = 'table-chan-' . Uuid::uuid4(); // lists requires a unique ID in case there are multiple lists per page 2596e45321fSGreg Roach 2606e45321fSGreg Roach switch ($sort) { 2616e45321fSGreg Roach case 'name': 2626e45321fSGreg Roach default: 2630c597aa6SGreg Roach $aaSorting = "[1,'asc'], [2,'desc']"; 2646e45321fSGreg Roach break; 2656e45321fSGreg Roach case 'date_asc': 2660c597aa6SGreg Roach $aaSorting = "[2,'asc'], [1,'asc']"; 2676e45321fSGreg Roach break; 2686e45321fSGreg Roach case 'date_desc': 2690c597aa6SGreg Roach $aaSorting = "[2,'desc'], [1,'asc']"; 2706e45321fSGreg Roach break; 2716e45321fSGreg Roach } 2726e45321fSGreg Roach 2736e45321fSGreg Roach $html = ''; 2746e45321fSGreg Roach $controller 2756e45321fSGreg Roach ->addInlineJavascript(' 27615d603e7SGreg Roach $("#' . $table_id . '").dataTable({ 2776e45321fSGreg Roach dom: \'t\', 2786e45321fSGreg Roach paging: false, 2796e45321fSGreg Roach autoWidth:false, 2806e45321fSGreg Roach lengthChange: false, 2816e45321fSGreg Roach filter: false, 2826e45321fSGreg Roach ' . I18N::datatablesI18N() . ', 2836e45321fSGreg Roach sorting: [' . $aaSorting . '], 2846e45321fSGreg Roach columns: [ 2850c597aa6SGreg Roach { sortable: false, class: "center" }, 2860c597aa6SGreg Roach null, 2870c597aa6SGreg Roach null, 2880c597aa6SGreg Roach { visible: ' . ($show_user ? 'true' : 'false') . ' } 2896e45321fSGreg Roach ] 2906e45321fSGreg Roach }); 2916e45321fSGreg Roach '); 2926e45321fSGreg Roach 2936e45321fSGreg Roach $html .= '<table id="' . $table_id . '" class="width100">'; 2946e45321fSGreg Roach $html .= '<thead><tr>'; 2956e45321fSGreg Roach $html .= '<th></th>'; 2966e45321fSGreg Roach $html .= '<th>' . I18N::translate('Record') . '</th>'; 29715d603e7SGreg Roach $html .= '<th>' . I18N::translate('Last change') . '</th>'; 2986e45321fSGreg Roach $html .= '<th>' . GedcomTag::getLabel('_WT_USER') . '</th>'; 2996e45321fSGreg Roach $html .= '</tr></thead><tbody>'; 3006e45321fSGreg Roach 3016e45321fSGreg Roach foreach ($records as $record) { 3026e45321fSGreg Roach $html .= '<tr><td>'; 3036e45321fSGreg Roach switch ($record::RECORD_TYPE) { 3046e45321fSGreg Roach case 'INDI': 30515d603e7SGreg Roach $html .= FontAwesome::semanticIcon('individual', I18N::translate('Individual')); 3066e45321fSGreg Roach break; 3076e45321fSGreg Roach case 'FAM': 30815d603e7SGreg Roach $html .= FontAwesome::semanticicon('family', I18N::translate('Family')); 3096e45321fSGreg Roach break; 3106e45321fSGreg Roach case 'OBJE': 31115d603e7SGreg Roach $html .= FontAwesome::semanticIcon('media', I18N::translate('Media')); 3126e45321fSGreg Roach break; 3136e45321fSGreg Roach case 'NOTE': 31415d603e7SGreg Roach $html .= FontAwesome::semanticIcon('note', I18N::translate('Note')); 3156e45321fSGreg Roach break; 3166e45321fSGreg Roach case 'SOUR': 31715d603e7SGreg Roach $html .= FontAwesome::semanticIcon('source', I18N::translate('Source')); 31815d603e7SGreg Roach break; 31915d603e7SGreg Roach case 'SUBM': 32015d603e7SGreg Roach $html .= FontAwesome::semanticIcon('submitter', I18N::translate('Submitter')); 3216e45321fSGreg Roach break; 3226e45321fSGreg Roach case 'REPO': 32315d603e7SGreg Roach $html .= FontAwesome::semanticIcon('repository', I18N::translate('Repository')); 3246e45321fSGreg Roach break; 3256e45321fSGreg Roach } 3266e45321fSGreg Roach $html .= '</td>'; 327cc5ab399SGreg Roach $html .= '<td data-sort="' . Html::escape($record->getSortName()) . '">'; 3280c597aa6SGreg Roach $html .= '<a href="' . $record->getHtmlUrl() . '">' . $record->getFullName() . '</a>'; 3296e45321fSGreg Roach $addname = $record->getAddName(); 3306e45321fSGreg Roach if ($addname) { 3316e45321fSGreg Roach $html .= '<div class="indent"><a href="' . $record->getHtmlUrl() . '">' . $addname . '</a></div>'; 3326e45321fSGreg Roach } 3336e45321fSGreg Roach $html .= '</td>'; 3340c597aa6SGreg Roach $html .= '<td data-sort="' . $record->lastChangeTimestamp(true) . '">' . $record->lastChangeTimestamp() . '</td>'; 335cc5ab399SGreg Roach $html .= '<td>' . Html::escape($record->lastChangeUser()) . '</td>'; 3366e45321fSGreg Roach $html .= '</tr>'; 3376e45321fSGreg Roach } 3386e45321fSGreg Roach 3396e45321fSGreg Roach $html .= '</tbody></table>'; 3406e45321fSGreg Roach 3416e45321fSGreg Roach return $html; 3426e45321fSGreg Roach } 3436e45321fSGreg Roach 3446e45321fSGreg Roach /** 3456e45321fSGreg Roach * Sort the records by (1) last change date and (2) name 3466e45321fSGreg Roach * 3476e45321fSGreg Roach * @param GedcomRecord $a 3486e45321fSGreg Roach * @param GedcomRecord $b 3496e45321fSGreg Roach * 3506e45321fSGreg Roach * @return int 3516e45321fSGreg Roach */ 3526e45321fSGreg Roach private static function sortByChangeDateAndName(GedcomRecord $a, GedcomRecord $b) { 3536e45321fSGreg Roach return $b->lastChangeTimestamp(true) - $a->lastChangeTimestamp(true) ?: GedcomRecord::compare($a, $b); 3546e45321fSGreg Roach } 3556e45321fSGreg Roach 3566e45321fSGreg Roach /** 3576e45321fSGreg Roach * Sort the records by (1) name and (2) last change date 3586e45321fSGreg Roach * 3596e45321fSGreg Roach * @param GedcomRecord $a 3606e45321fSGreg Roach * @param GedcomRecord $b 3616e45321fSGreg Roach * 3626e45321fSGreg Roach * @return int 3636e45321fSGreg Roach */ 3646e45321fSGreg Roach private static function sortByNameAndChangeDate(GedcomRecord $a, GedcomRecord $b) { 3656e45321fSGreg Roach return GedcomRecord::compare($a, $b) ?: $b->lastChangeTimestamp(true) - $a->lastChangeTimestamp(true); 3666e45321fSGreg Roach } 3678c2e8227SGreg Roach} 368