. */ namespace Fisharebest\Webtrees\Module; use Fisharebest\Webtrees\Auth; use Fisharebest\Webtrees\Bootstrap4; use Fisharebest\Webtrees\Database; use Fisharebest\Webtrees\Filter; use Fisharebest\Webtrees\FontAwesome; use Fisharebest\Webtrees\Functions\FunctionsEdit; use Fisharebest\Webtrees\GedcomRecord; use Fisharebest\Webtrees\GedcomTag; use Fisharebest\Webtrees\Html; use Fisharebest\Webtrees\I18N; use Fisharebest\Webtrees\Individual; use Fisharebest\Webtrees\Theme; use Fisharebest\Webtrees\Tree; use Ramsey\Uuid\Uuid; /** * Class RecentChangesModule */ class RecentChangesModule extends AbstractModule implements ModuleBlockInterface { const DEFAULT_BLOCK = '1'; const DEFAULT_DAYS = 7; const DEFAULT_HIDE_EMPTY = '0'; const DEFAULT_SHOW_USER = '1'; const DEFAULT_SORT_STYLE = 'date_desc'; const DEFAULT_INFO_STYLE = 'table'; const MAX_DAYS = 90; /** {@inheritdoc} */ public function getTitle() { return /* I18N: Name of a module */ I18N::translate('Recent changes'); } /** {@inheritdoc} */ public function getDescription() { return /* I18N: Description of the “Recent changes” module */ I18N::translate('A list of records that have been updated recently.'); } /** {@inheritdoc} */ public function getBlock($block_id, $template = true, $cfg = []) { global $ctype, $WT_TREE; $days = $this->getBlockSetting($block_id, 'days', self::DEFAULT_DAYS); $infoStyle = $this->getBlockSetting($block_id, 'infoStyle', self::DEFAULT_INFO_STYLE); $sortStyle = $this->getBlockSetting($block_id, 'sortStyle', self::DEFAULT_SORT_STYLE); $show_user = $this->getBlockSetting($block_id, 'show_user', self::DEFAULT_SHOW_USER); foreach (['days', 'infoStyle', 'sortStyle', 'show_user'] as $name) { if (array_key_exists($name, $cfg)) { $$name = $cfg[$name]; } } $records = $this->getRecentChanges($WT_TREE, $days); // Print block header $id = $this->getName() . $block_id; $class = $this->getName() . '_block'; if ($ctype === 'gedcom' && Auth::isManager($WT_TREE) || $ctype === 'user' && Auth::check()) { $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]) . ' '; } else { $title = ''; } $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)); $content = ''; // Print block content if (count($records) == 0) { $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)); } else { switch ($infoStyle) { case 'list': $content .= $this->changesList($records, $sortStyle, $show_user); break; case 'table': $content .= $this->changesTable($records, $sortStyle, $show_user); break; } } if ($template) { return Theme::theme()->formatBlock($id, $title, $class, $content); } else { return $content; } } /** {@inheritdoc} */ public function loadAjax() { return true; } /** {@inheritdoc} */ public function isUserBlock() { return true; } /** {@inheritdoc} */ public function isGedcomBlock() { return true; } /** {@inheritdoc} */ public function configureBlock($block_id) { if (Filter::postBool('save') && Filter::checkCsrf()) { $this->setBlockSetting($block_id, 'days', Filter::postInteger('days', 1, self::MAX_DAYS)); $this->setBlockSetting($block_id, 'infoStyle', Filter::post('infoStyle', 'list|table')); $this->setBlockSetting($block_id, 'sortStyle', Filter::post('sortStyle', 'name|date_asc|date_desc')); $this->setBlockSetting($block_id, 'show_user', Filter::postBool('show_user')); } $days = $this->getBlockSetting($block_id, 'days', self::DEFAULT_DAYS); $infoStyle = $this->getBlockSetting($block_id, 'infoStyle', self::DEFAULT_INFO_STYLE); $sortStyle = $this->getBlockSetting($block_id, 'sortStyle', self::DEFAULT_SORT_STYLE); $show_user = $this->getBlockSetting($block_id, 'show_user', self::DEFAULT_SHOW_USER); echo '
'; echo ''; echo ' ' . I18N::plural('maximum %s day', 'maximum %s days', I18N::number(self::MAX_DAYS), I18N::number(self::MAX_DAYS)); echo '
'; echo '
'; echo Bootstrap4::select(['list' => I18N::translate('list'), 'table' => I18N::translate('table')], $infoStyle, ['id' => 'infoStyle', 'name' => 'infoStyle']); echo '
'; echo '
'; echo Bootstrap4::select([ 'name' => /* I18N: An option in a list-box */ I18N::translate('sort by name'), 'date_asc' => /* I18N: An option in a list-box */ I18N::translate('sort by date, oldest first'), 'date_desc' => /* I18N: An option in a list-box */ I18N::translate('sort by date, newest first'), ], $sortStyle, ['id' => 'sortStyle', 'name' => 'sortStyle']); echo '
'; echo '
'; echo Bootstrap4::radioButtons('show_user', FunctionsEdit::optionsNoYes(), $show_user, true); echo '
'; } /** * Find records that have changed since a given julian day * * @param Tree $tree Changes for which tree * @param int $days Number of days * * @return GedcomRecord[] List of records with changes */ private function getRecentChanges(Tree $tree, $days) { $sql = "SELECT xref FROM `##change`" . " WHERE new_gedcom != '' AND change_time > DATE_SUB(NOW(), INTERVAL :days DAY) AND gedcom_id = :tree_id" . " GROUP BY xref" . " ORDER BY MAX(change_id) DESC"; $vars = [ 'days' => $days, 'tree_id' => $tree->getTreeId(), ]; $xrefs = Database::prepare($sql)->execute($vars)->fetchOneColumn(); $records = []; foreach ($xrefs as $xref) { $record = GedcomRecord::getInstance($xref, $tree); if ($record && $record->canShow()) { $records[] = $record; } } return $records; } /** * Format a table of events * * @param GedcomRecord[] $records * @param string $sort * @param bool $show_user * * @return string */ private function changesList(array $records, $sort, $show_user) { switch ($sort) { case 'name': uasort($records, ['self', 'sortByNameAndChangeDate']); break; case 'date_asc': uasort($records, ['self', 'sortByChangeDateAndName']); $records = array_reverse($records); break; case 'date_desc': uasort($records, ['self', 'sortByChangeDateAndName']); } $html = ''; foreach ($records as $record) { $html .= '' . $record->getFullName() . ''; $html .= '
'; if ($record instanceof Individual) { if ($record->getAddName()) { $html .= '' . $record->getAddName() . ''; } } // The timestamp may be missing or private. $timestamp = $record->lastChangeTimestamp(); if ($timestamp !== '') { if ($show_user) { $html .= /* I18N: [a record was] Changed on by */ I18N::translate('Changed on %1$s by %2$s', $timestamp, Html::escape($record->lastChangeUser())); } else { $html .= /* I18N: [a record was] Changed on */ I18N::translate('Changed on %1$s', $timestamp); } } $html .= '
'; } return $html; } /** * Format a table of events * * @param GedcomRecord[] $records * @param string $sort * @param bool $show_user * * @return string */ private function changesTable($records, $sort, $show_user) { global $controller; $table_id = 'table-chan-' . Uuid::uuid4(); // lists requires a unique ID in case there are multiple lists per page switch ($sort) { case 'name': default: $aaSorting = "[1,'asc'], [2,'desc']"; break; case 'date_asc': $aaSorting = "[2,'asc'], [1,'asc']"; break; case 'date_desc': $aaSorting = "[2,'desc'], [1,'asc']"; break; } $html = ''; $controller ->addInlineJavascript(' $("#' . $table_id . '").dataTable({ dom: \'t\', paging: false, autoWidth:false, lengthChange: false, filter: false, ' . I18N::datatablesI18N() . ', sorting: [' . $aaSorting . '], columns: [ { sortable: false, class: "center" }, null, null, { visible: ' . ($show_user ? 'true' : 'false') . ' } ] }); '); $html .= ''; $html .= ''; $html .= ''; $html .= ''; $html .= ''; $html .= ''; $html .= ''; foreach ($records as $record) { $html .= ''; $html .= ''; $html .= ''; $html .= ''; $html .= ''; } $html .= '
' . I18N::translate('Record') . '' . I18N::translate('Last change') . '' . GedcomTag::getLabel('_WT_USER') . '
'; switch ($record::RECORD_TYPE) { case 'INDI': $html .= FontAwesome::semanticIcon('individual', I18N::translate('Individual')); break; case 'FAM': $html .= FontAwesome::semanticicon('family', I18N::translate('Family')); break; case 'OBJE': $html .= FontAwesome::semanticIcon('media', I18N::translate('Media')); break; case 'NOTE': $html .= FontAwesome::semanticIcon('note', I18N::translate('Note')); break; case 'SOUR': $html .= FontAwesome::semanticIcon('source', I18N::translate('Source')); break; case 'SUBM': $html .= FontAwesome::semanticIcon('submitter', I18N::translate('Submitter')); break; case 'REPO': $html .= FontAwesome::semanticIcon('repository', I18N::translate('Repository')); break; } $html .= ''; $html .= '' . $record->getFullName() . ''; $addname = $record->getAddName(); if ($addname) { $html .= ''; } $html .= '' . $record->lastChangeTimestamp() . '' . Html::escape($record->lastChangeUser()) . '
'; return $html; } /** * Sort the records by (1) last change date and (2) name * * @param GedcomRecord $a * @param GedcomRecord $b * * @return int */ private static function sortByChangeDateAndName(GedcomRecord $a, GedcomRecord $b) { return $b->lastChangeTimestamp(true) - $a->lastChangeTimestamp(true) ?: GedcomRecord::compare($a, $b); } /** * Sort the records by (1) name and (2) last change date * * @param GedcomRecord $a * @param GedcomRecord $b * * @return int */ private static function sortByNameAndChangeDate(GedcomRecord $a, GedcomRecord $b) { return GedcomRecord::compare($a, $b) ?: $b->lastChangeTimestamp(true) - $a->lastChangeTimestamp(true); } }