mirror of
https://gerrit.wikimedia.org/r/mediawiki/extensions/Echo
synced 2024-12-01 02:46:46 +00:00
983a01bdff
Disable some errors related to different members of an array having different taints, and phan-taint-check conflating them. Bug: T202383 Change-Id: Ic6c2c5bb7c6092d581e646358d836f55d5cf3222
269 lines
7.5 KiB
PHP
269 lines
7.5 KiB
PHP
<?php
|
|
|
|
class SpecialNotifications extends SpecialPage {
|
|
|
|
/**
|
|
* Number of notification records to display per page/load
|
|
*/
|
|
const DISPLAY_NUM = 20;
|
|
|
|
public function __construct() {
|
|
parent::__construct( 'Notifications' );
|
|
}
|
|
|
|
/**
|
|
* @param string $par
|
|
* @suppress SecurityCheck-DoubleEscaped Different members of $notifArray being conflated
|
|
*/
|
|
public function execute( $par ) {
|
|
$this->setHeaders();
|
|
|
|
$out = $this->getOutput();
|
|
$out->setPageTitle( $this->msg( 'echo-specialpage' )->text() );
|
|
|
|
$this->addHelpLink( 'Help:Notifications/Special:Notifications' );
|
|
|
|
$out->addJsConfigVars( 'wgNotificationsSpecialPageLinks', [
|
|
'help' => '//www.mediawiki.org/wiki/Special:MyLanguage/Help:Notifications/Special:Notifications',
|
|
'preferences' => SpecialPage::getTitleFor( 'Preferences' )->getLinkURL() . '#mw-prefsection-echo',
|
|
] );
|
|
|
|
$user = $this->getUser();
|
|
if ( $user->isAnon() ) {
|
|
// Redirect to login page and inform user of the need to login
|
|
$this->requireLogin( 'echo-notification-loginrequired' );
|
|
return;
|
|
}
|
|
|
|
$out->addSubtitle( $this->buildSubtitle() );
|
|
|
|
$out->enableOOUI();
|
|
|
|
$pager = new NotificationPager( $this->getContext() );
|
|
$pager->setOffset( $this->getRequest()->getVal( 'offset' ) );
|
|
$pager->setLimit( $this->getRequest()->getVal( 'limit', self::DISPLAY_NUM ) );
|
|
$notifications = $pager->getNotifications();
|
|
|
|
$noJSDiv = new OOUI\Tag();
|
|
$noJSDiv->addClasses( [ 'mw-echo-special-nojs' ] );
|
|
|
|
// If there are no notifications, display a message saying so
|
|
if ( !$notifications ) {
|
|
// Wrap this with nojs so it is still hidden if JS is loading
|
|
$noJSDiv->appendContent(
|
|
new OOUI\LabelWidget( [ 'label' => $this->msg( 'echo-none' )->text() ] )
|
|
);
|
|
$out->addHTML( $noJSDiv );
|
|
$out->addModules( [ 'ext.echo.special' ] );
|
|
return;
|
|
}
|
|
|
|
$notif = [];
|
|
foreach ( $notifications as $notification ) {
|
|
$output = EchoDataOutputFormatter::formatOutput( $notification, 'special', $user, $this->getLanguage() );
|
|
if ( $output ) {
|
|
$notif[] = $output;
|
|
}
|
|
}
|
|
|
|
// Add the notifications to the page (interspersed with date headers)
|
|
$dateHeader = '';
|
|
$unread = [];
|
|
$anyUnread = false;
|
|
$echoSeenTime = EchoSeenTime::newFromUser( $user );
|
|
$seenTime = $echoSeenTime->getTime();
|
|
$notifArray = [];
|
|
foreach ( $notif as $row ) {
|
|
if ( !$row['*'] ) {
|
|
continue;
|
|
}
|
|
|
|
$classes = [ 'mw-echo-notification' ];
|
|
|
|
if ( !isset( $row['read'] ) ) {
|
|
$classes[] = 'mw-echo-notification-unread';
|
|
if ( !$row['targetpages'] ) {
|
|
$unread[] = $row['id'];
|
|
}
|
|
}
|
|
|
|
if ( $seenTime !== null && $row['timestamp']['mw'] > $seenTime ) {
|
|
$classes[] = 'mw-echo-notification-unseen';
|
|
}
|
|
|
|
// Output the date header if it has not been displayed
|
|
if ( $dateHeader !== $row['timestamp']['date'] ) {
|
|
$dateHeader = $row['timestamp']['date'];
|
|
$notifArray[ $dateHeader ] = [
|
|
'unread' => [],
|
|
'notices' => []
|
|
];
|
|
}
|
|
|
|
// Collect unread IDs
|
|
if ( !isset( $row['read'] ) ) {
|
|
$anyUnread = true;
|
|
$notifArray[ $dateHeader ][ 'unread' ][] = $row['id'];
|
|
}
|
|
|
|
$li = new OOUI\Tag( 'li' );
|
|
$li
|
|
->addClasses( $classes )
|
|
->setAttributes( [
|
|
'data-notification-category' => $row['category'],
|
|
'data-notification-event' => $row['id'],
|
|
'data-notification-type' => $row['type']
|
|
] )
|
|
->appendContent( new OOUI\HtmlSnippet( $row['*'] ) );
|
|
|
|
// Store
|
|
$notifArray[ $dateHeader ][ 'notices' ][] = $li;
|
|
}
|
|
|
|
$markAllAsReadFormWrapper = '';
|
|
// Ensure there are some unread notifications
|
|
if ( $anyUnread ) {
|
|
$markReadSpecialPage = new SpecialNotificationsMarkRead();
|
|
|
|
$markAllAsReadText = $this->msg( 'echo-mark-all-as-read' )->text();
|
|
$markAllAsReadLabelIcon = new EchoOOUI\LabelIconWidget( [
|
|
'label' => $markAllAsReadText,
|
|
'icon' => 'checkAll',
|
|
] );
|
|
|
|
$markAllAsReadForm = $markReadSpecialPage->getMinimalForm(
|
|
[ 'ALL' ],
|
|
$markAllAsReadText,
|
|
true,
|
|
$markAllAsReadLabelIcon->toString()
|
|
);
|
|
|
|
$formHtml = $markAllAsReadForm->prepareForm()->getHTML( /* First submission attempt */ false );
|
|
|
|
$markAllAsReadFormWrapper = new OOUI\Tag();
|
|
$markAllAsReadFormWrapper
|
|
->addClasses( [ 'mw-echo-special-markAllReadButton' ] )
|
|
->appendContent( new OOUI\HtmlSnippet( $formHtml ) );
|
|
}
|
|
|
|
// Build the list
|
|
$notices = new OOUI\Tag( 'ul' );
|
|
$notices->addClasses( [ 'mw-echo-special-notifications' ] );
|
|
|
|
$markReadSpecialPage = new SpecialNotificationsMarkRead();
|
|
foreach ( $notifArray as $section => $data ) {
|
|
// Heading
|
|
$heading = ( new OOUI\Tag( 'li' ) )->addClasses( [ 'mw-echo-date-section' ] );
|
|
|
|
$dateTitle = new OOUI\LabelWidget( [
|
|
'classes' => [ 'mw-echo-date-section-text' ],
|
|
'label' => $section
|
|
] );
|
|
|
|
$heading->appendContent( $dateTitle );
|
|
|
|
// Mark all read button
|
|
if ( $data[ 'unread' ] !== [] ) {
|
|
// tell the UI to show 'unread' notifications only (instead of 'all')
|
|
$out->addJsConfigVars( 'wgEchoReadState', 'unread' );
|
|
|
|
$markReadSectionText = $this->msg( 'echo-specialpage-section-markread' )->text();
|
|
$markAsReadLabelIcon = new EchoOOUI\LabelIconWidget( [
|
|
'label' => $markReadSectionText,
|
|
'icon' => 'checkAll',
|
|
] );
|
|
|
|
// There are unread notices. Add the 'mark section as read' button
|
|
$markSectionAsReadForm = $markReadSpecialPage->getMinimalForm(
|
|
$data[ 'unread' ],
|
|
$markReadSectionText,
|
|
true,
|
|
$markAsReadLabelIcon->toString()
|
|
);
|
|
|
|
$formHtml = $markSectionAsReadForm->prepareForm()->getHTML( /* First submission attempt */ false );
|
|
|
|
$formWrapper = new OOUI\Tag();
|
|
$formWrapper
|
|
->addClasses( [ 'mw-echo-markAsReadSectionButton' ] )
|
|
->appendContent( new OOUI\HtmlSnippet( $formHtml ) );
|
|
|
|
$heading->appendContent( $formWrapper );
|
|
}
|
|
|
|
// These two must be separate, because $data[ 'notices' ]
|
|
// is an array
|
|
$notices
|
|
->appendContent( $heading )
|
|
->appendContent( $data[ 'notices' ] );
|
|
}
|
|
|
|
$navBar = $pager->getNavigationBar();
|
|
|
|
$navTop = new OOUI\Tag();
|
|
$navBottom = new OOUI\Tag();
|
|
$container = new OOUI\Tag();
|
|
|
|
$navTop
|
|
->addClasses( [ 'mw-echo-special-navbar-top' ] )
|
|
->appendContent( new OOUI\HtmlSnippet( $navBar ) );
|
|
$navBottom
|
|
->addClasses( [ 'mw-echo-special-navbar-bottom' ] )
|
|
->appendContent( new OOUI\HtmlSnippet( $navBar ) );
|
|
|
|
// Put it all together
|
|
$container
|
|
->addClasses( [ 'mw-echo-special-container' ] )
|
|
->appendContent(
|
|
$navTop,
|
|
$markAllAsReadFormWrapper,
|
|
$notices,
|
|
$navBottom
|
|
);
|
|
|
|
// Wrap with nojs div
|
|
$noJSDiv->appendContent( $container );
|
|
|
|
$out->addHTML( $noJSDiv );
|
|
|
|
$out->addModules( [ 'ext.echo.special' ] );
|
|
|
|
// For no-js support
|
|
$out->addModuleStyles( [
|
|
'ext.echo.styles.notifications',
|
|
'ext.echo.styles.special',
|
|
// We already load badgeicons in the BeforePageDisplay hook, but not for minerva
|
|
'ext.echo.badgeicons'
|
|
] );
|
|
|
|
// Log visit
|
|
MWEchoEventLogging::logSpecialPageVisit( $user, $out->getSkin()->getSkinName() );
|
|
}
|
|
|
|
/**
|
|
* Build the subtitle (more info and preference links)
|
|
* @return string HTML for the subtitle
|
|
*/
|
|
public function buildSubtitle() {
|
|
$lang = $this->getLanguage();
|
|
$subtitleLinks = [];
|
|
// Preferences link
|
|
$subtitleLinks[] = Html::element(
|
|
'a',
|
|
[
|
|
'href' => SpecialPage::getTitleFor( 'Preferences' )->getLinkURL() . '#mw-prefsection-echo',
|
|
'id' => 'mw-echo-pref-link',
|
|
'class' => 'mw-echo-special-header-link',
|
|
'title' => $this->msg( 'preferences' )->text()
|
|
],
|
|
$this->msg( 'preferences' )->text()
|
|
);
|
|
// using pipeList to make it easier to add some links in the future
|
|
return $lang->pipeList( $subtitleLinks );
|
|
}
|
|
|
|
protected function getGroupName() {
|
|
return 'users';
|
|
}
|
|
}
|