The context source is not set at this point and the global request is used and logged: SpecialPage::getContext called and $mContext is null. Using RequestContext::getMain(); for sanity Change-Id: Ia1ce0a6f379dfeaa76d831d169ed97c059c48533
861 lines
24 KiB
PHP
861 lines
24 KiB
PHP
<?php
|
|
/**
|
|
* @defgroup Watchlist Users watchlist handling
|
|
*/
|
|
|
|
/**
|
|
* Implements Special:EditWatchlist
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License along
|
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
* http://www.gnu.org/copyleft/gpl.html
|
|
*
|
|
* @file
|
|
* @ingroup SpecialPage
|
|
* @ingroup Watchlist
|
|
*/
|
|
|
|
use MediaWiki\Cache\LinkBatchFactory;
|
|
use MediaWiki\Linker\LinkRenderer;
|
|
use MediaWiki\Linker\LinkTarget;
|
|
use MediaWiki\MediaWikiServices;
|
|
use MediaWiki\Page\WikiPageFactory;
|
|
use MediaWiki\Watchlist\WatchlistManager;
|
|
|
|
/**
|
|
* Provides the UI through which users can perform editing
|
|
* operations on their watchlist
|
|
*
|
|
* @ingroup SpecialPage
|
|
* @ingroup Watchlist
|
|
* @author Rob Church <robchur@gmail.com>
|
|
*/
|
|
class SpecialEditWatchlist extends UnlistedSpecialPage {
|
|
/**
|
|
* Editing modes. EDIT_CLEAR is no longer used; the "Clear" link scared people
|
|
* too much. Now it's passed on to the raw editor, from which it's very easy to clear.
|
|
*/
|
|
public const EDIT_CLEAR = 1;
|
|
public const EDIT_RAW = 2;
|
|
public const EDIT_NORMAL = 3;
|
|
|
|
protected $successMessage;
|
|
|
|
protected $toc;
|
|
|
|
private $badItems = [];
|
|
|
|
/** @var TitleParser */
|
|
private $titleParser;
|
|
|
|
/** @var WatchedItemStoreInterface */
|
|
private $watchedItemStore;
|
|
|
|
/** @var GenderCache */
|
|
private $genderCache;
|
|
|
|
/** @var LinkBatchFactory */
|
|
private $linkBatchFactory;
|
|
|
|
/** @var NamespaceInfo */
|
|
private $nsInfo;
|
|
|
|
/** @var WikiPageFactory */
|
|
private $wikiPageFactory;
|
|
|
|
/** @var WatchlistManager */
|
|
private $watchlistManager;
|
|
|
|
/**
|
|
* @param WatchedItemStoreInterface|null $watchedItemStore
|
|
* @param TitleParser|null $titleParser
|
|
* @param GenderCache|null $genderCache
|
|
* @param LinkBatchFactory|null $linkBatchFactory
|
|
* @param NamespaceInfo|null $nsInfo
|
|
* @param WikiPageFactory|null $wikiPageFactory
|
|
* @param WatchlistManager|null $watchlistManager
|
|
*/
|
|
public function __construct(
|
|
WatchedItemStoreInterface $watchedItemStore = null,
|
|
TitleParser $titleParser = null,
|
|
GenderCache $genderCache = null,
|
|
LinkBatchFactory $linkBatchFactory = null,
|
|
NamespaceInfo $nsInfo = null,
|
|
WikiPageFactory $wikiPageFactory = null,
|
|
WatchlistManager $watchlistManager = null
|
|
) {
|
|
parent::__construct( 'EditWatchlist', 'editmywatchlist' );
|
|
// This class is extended and therefor fallback to global state - T266065
|
|
$services = MediaWikiServices::getInstance();
|
|
$this->watchedItemStore = $watchedItemStore ?? $services->getWatchedItemStore();
|
|
$this->titleParser = $titleParser ?? $services->getTitleParser();
|
|
$this->genderCache = $genderCache ?? $services->getGenderCache();
|
|
$this->linkBatchFactory = $linkBatchFactory ?? $services->getLinkBatchFactory();
|
|
$this->nsInfo = $nsInfo ?? $services->getNamespaceInfo();
|
|
$this->wikiPageFactory = $wikiPageFactory ?? $services->getWikiPageFactory();
|
|
$this->watchlistManager = $watchlistManager ?? $services->getWatchlistManager();
|
|
}
|
|
|
|
public function doesWrites() {
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* Main execution point
|
|
*
|
|
* @param string|null $mode
|
|
*/
|
|
public function execute( $mode ) {
|
|
$this->setHeaders();
|
|
|
|
# Anons don't get a watchlist
|
|
$this->requireLogin( 'watchlistanontext' );
|
|
|
|
$out = $this->getOutput();
|
|
|
|
$this->checkPermissions();
|
|
$this->checkReadOnly();
|
|
|
|
$this->outputHeader();
|
|
$this->outputSubtitle();
|
|
$out->addModuleStyles( 'mediawiki.special' );
|
|
|
|
$mode = self::getMode( $this->getRequest(), $mode );
|
|
|
|
switch ( $mode ) {
|
|
case self::EDIT_RAW:
|
|
$out->setPageTitle( $this->msg( 'watchlistedit-raw-title' ) );
|
|
$form = $this->getRawForm();
|
|
if ( $form->show() ) {
|
|
$out->addHTML( $this->successMessage );
|
|
$out->addReturnTo( SpecialPage::getTitleFor( 'Watchlist' ) );
|
|
}
|
|
break;
|
|
case self::EDIT_CLEAR:
|
|
$out->setPageTitle( $this->msg( 'watchlistedit-clear-title' ) );
|
|
$form = $this->getClearForm();
|
|
if ( $form->show() ) {
|
|
$out->addHTML( $this->successMessage );
|
|
$out->addReturnTo( SpecialPage::getTitleFor( 'Watchlist' ) );
|
|
}
|
|
break;
|
|
|
|
case self::EDIT_NORMAL:
|
|
default:
|
|
$this->executeViewEditWatchlist();
|
|
break;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Renders a subheader on the watchlist page.
|
|
*/
|
|
protected function outputSubtitle() {
|
|
$out = $this->getOutput();
|
|
$out->addSubtitle( $this->msg( 'watchlistfor2', $this->getUser()->getName() )
|
|
->rawParams(
|
|
self::buildTools(
|
|
$this->getLanguage(),
|
|
$this->getLinkRenderer()
|
|
)
|
|
)
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Executes an edit mode for the watchlist view, from which you can manage your watchlist
|
|
*/
|
|
protected function executeViewEditWatchlist() {
|
|
$out = $this->getOutput();
|
|
$out->setPageTitle( $this->msg( 'watchlistedit-normal-title' ) );
|
|
$form = $this->getNormalForm();
|
|
if ( $form->show() ) {
|
|
$out->addHTML( $this->successMessage );
|
|
$out->addReturnTo( SpecialPage::getTitleFor( 'Watchlist' ) );
|
|
} elseif ( $this->toc !== false ) {
|
|
$out->prependHTML( $this->toc );
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Return an array of subpages that this special page will accept.
|
|
*
|
|
* @see also SpecialWatchlist::getSubpagesForPrefixSearch
|
|
* @return string[] subpages
|
|
*/
|
|
public function getSubpagesForPrefixSearch() {
|
|
// SpecialWatchlist uses SpecialEditWatchlist::getMode, so new types should be added
|
|
// here and there - no 'edit' here, because that the default for this page
|
|
return [
|
|
'clear',
|
|
'raw',
|
|
];
|
|
}
|
|
|
|
/**
|
|
* Extract a list of titles from a blob of text, returning
|
|
* (prefixed) strings; unwatchable titles are ignored
|
|
*
|
|
* @param string $list
|
|
* @return array
|
|
*/
|
|
private function extractTitles( $list ) {
|
|
$list = explode( "\n", trim( $list ) );
|
|
if ( !is_array( $list ) ) {
|
|
return [];
|
|
}
|
|
|
|
$titles = [];
|
|
|
|
foreach ( $list as $text ) {
|
|
$text = trim( $text );
|
|
if ( strlen( $text ) > 0 ) {
|
|
$title = Title::newFromText( $text );
|
|
if ( $title instanceof Title && $this->watchlistManager->isWatchable( $title ) ) {
|
|
$titles[] = $title;
|
|
}
|
|
}
|
|
}
|
|
|
|
$this->genderCache->doTitlesArray( $titles );
|
|
|
|
$list = [];
|
|
/** @var Title $title */
|
|
foreach ( $titles as $title ) {
|
|
$list[] = $title->getPrefixedText();
|
|
}
|
|
|
|
return array_unique( $list );
|
|
}
|
|
|
|
public function submitRaw( $data ) {
|
|
$wanted = $this->extractTitles( $data['Titles'] );
|
|
$current = $this->getWatchlist();
|
|
|
|
if ( count( $wanted ) > 0 ) {
|
|
$toWatch = array_diff( $wanted, $current );
|
|
$toUnwatch = array_diff( $current, $wanted );
|
|
$this->watchTitles( $toWatch );
|
|
$this->unwatchTitles( $toUnwatch );
|
|
$this->getUser()->invalidateCache();
|
|
|
|
if ( count( $toWatch ) > 0 || count( $toUnwatch ) > 0 ) {
|
|
$this->successMessage = $this->msg( 'watchlistedit-raw-done' )->parse();
|
|
} else {
|
|
return false;
|
|
}
|
|
|
|
if ( count( $toWatch ) > 0 ) {
|
|
$this->successMessage .= ' ' . $this->msg( 'watchlistedit-raw-added' )
|
|
->numParams( count( $toWatch ) )->parse();
|
|
$this->showTitles( $toWatch, $this->successMessage );
|
|
}
|
|
|
|
if ( count( $toUnwatch ) > 0 ) {
|
|
$this->successMessage .= ' ' . $this->msg( 'watchlistedit-raw-removed' )
|
|
->numParams( count( $toUnwatch ) )->parse();
|
|
$this->showTitles( $toUnwatch, $this->successMessage );
|
|
}
|
|
} else {
|
|
|
|
if ( count( $current ) === 0 ) {
|
|
return false;
|
|
}
|
|
|
|
$this->clearUserWatchedItems( 'raw' );
|
|
$this->showTitles( $current, $this->successMessage );
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* Handler for the clear form submission
|
|
*
|
|
* @param array $data
|
|
* @return bool
|
|
*/
|
|
public function submitClear( $data ): bool {
|
|
$this->clearUserWatchedItems( 'clear' );
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* Makes a decision about using the JobQueue or not for clearing a users watchlist.
|
|
* Also displays the appropriate messages to the user based on that decision.
|
|
*
|
|
* @param string $messageFor 'raw' or 'clear'. Only used when JobQueue is not used.
|
|
*/
|
|
private function clearUserWatchedItems( string $messageFor ): void {
|
|
if ( $this->watchedItemStore->mustClearWatchedItemsUsingJobQueue( $this->getUser() ) ) {
|
|
$this->clearUserWatchedItemsUsingJobQueue();
|
|
} else {
|
|
$this->clearUserWatchedItemsNow( $messageFor );
|
|
}
|
|
}
|
|
|
|
/**
|
|
* You should call clearUserWatchedItems() instead to decide if this should use the JobQueue
|
|
*
|
|
* @param string $messageFor 'raw' or 'clear'
|
|
*/
|
|
private function clearUserWatchedItemsNow( string $messageFor ): void {
|
|
$current = $this->getWatchlist();
|
|
if ( !$this->watchedItemStore->clearUserWatchedItems( $this->getUser() ) ) {
|
|
throw new LogicException(
|
|
__METHOD__ . ' should only be called when able to clear synchronously'
|
|
);
|
|
}
|
|
$this->successMessage = $this->msg( 'watchlistedit-' . $messageFor . '-done' )->parse();
|
|
$this->successMessage .= ' ' . $this->msg( 'watchlistedit-' . $messageFor . '-removed' )
|
|
->numParams( count( $current ) )->parse();
|
|
$this->getUser()->invalidateCache();
|
|
$this->showTitles( $current, $this->successMessage );
|
|
}
|
|
|
|
/**
|
|
* You should call clearUserWatchedItems() instead to decide if this should use the JobQueue
|
|
*/
|
|
private function clearUserWatchedItemsUsingJobQueue(): void {
|
|
$this->watchedItemStore->clearUserWatchedItemsUsingJobQueue( $this->getUser() );
|
|
$this->successMessage = $this->msg( 'watchlistedit-clear-jobqueue' )->parse();
|
|
}
|
|
|
|
/**
|
|
* Print out a list of linked titles
|
|
*
|
|
* $titles can be an array of strings or Title objects; the former
|
|
* is preferred, since Titles are very memory-heavy
|
|
*
|
|
* @param array $titles Array of strings, or Title objects
|
|
* @param string &$output
|
|
*/
|
|
private function showTitles( $titles, &$output ) {
|
|
$talk = $this->msg( 'talkpagelinktext' )->text();
|
|
// Do a batch existence check
|
|
$batch = $this->linkBatchFactory->newLinkBatch();
|
|
if ( count( $titles ) >= 100 ) {
|
|
$output = $this->msg( 'watchlistedit-too-many' )->parse();
|
|
return;
|
|
}
|
|
foreach ( $titles as $title ) {
|
|
if ( !$title instanceof Title ) {
|
|
$title = Title::newFromText( $title );
|
|
}
|
|
|
|
if ( $title instanceof Title ) {
|
|
$batch->addObj( $title );
|
|
$batch->addObj( $title->getTalkPage() );
|
|
}
|
|
}
|
|
|
|
$batch->execute();
|
|
|
|
// Print out the list
|
|
$output .= "<ul>\n";
|
|
|
|
$linkRenderer = $this->getLinkRenderer();
|
|
foreach ( $titles as $title ) {
|
|
if ( !$title instanceof Title ) {
|
|
$title = Title::newFromText( $title );
|
|
}
|
|
|
|
if ( $title instanceof Title ) {
|
|
$output .= '<li>' .
|
|
$linkRenderer->makeLink( $title ) . ' ' .
|
|
$this->msg( 'parentheses' )->rawParams(
|
|
$linkRenderer->makeLink( $title->getTalkPage(), $talk )
|
|
)->escaped() .
|
|
"</li>\n";
|
|
}
|
|
}
|
|
|
|
$output .= "</ul>\n";
|
|
}
|
|
|
|
/**
|
|
* Prepare a list of titles on a user's watchlist (excluding talk pages)
|
|
* and return an array of (prefixed) strings
|
|
*
|
|
* @return array
|
|
*/
|
|
private function getWatchlist() {
|
|
$list = [];
|
|
|
|
$watchedItems = $this->watchedItemStore->getWatchedItemsForUser(
|
|
$this->getUser(),
|
|
[ 'forWrite' => $this->getRequest()->wasPosted() ]
|
|
);
|
|
|
|
if ( $watchedItems ) {
|
|
/** @var Title[] $titles */
|
|
$titles = [];
|
|
foreach ( $watchedItems as $watchedItem ) {
|
|
$namespace = $watchedItem->getTarget()->getNamespace();
|
|
$dbKey = $watchedItem->getTarget()->getDBkey();
|
|
$title = Title::makeTitleSafe( $namespace, $dbKey );
|
|
|
|
if ( $this->checkTitle( $title, $namespace, $dbKey )
|
|
&& !$title->isTalkPage()
|
|
) {
|
|
$titles[] = $title;
|
|
}
|
|
}
|
|
|
|
$this->genderCache->doTitlesArray( $titles );
|
|
|
|
foreach ( $titles as $title ) {
|
|
$list[] = $title->getPrefixedText();
|
|
}
|
|
}
|
|
|
|
$this->cleanupWatchlist();
|
|
|
|
return $list;
|
|
}
|
|
|
|
/**
|
|
* Get a list of titles on a user's watchlist, excluding talk pages,
|
|
* and return as a two-dimensional array with namespace and title.
|
|
*
|
|
* @return array
|
|
*/
|
|
protected function getWatchlistInfo() {
|
|
$titles = [];
|
|
$options = [ 'sort' => WatchedItemStore::SORT_ASC ];
|
|
|
|
if ( $this->getConfig()->get( 'WatchlistExpiry' ) ) {
|
|
$options[ 'sortByExpiry'] = true;
|
|
}
|
|
|
|
$watchedItems = $this->watchedItemStore->getWatchedItemsForUser(
|
|
$this->getUser(), $options
|
|
);
|
|
|
|
$lb = $this->linkBatchFactory->newLinkBatch();
|
|
$context = $this->getContext();
|
|
|
|
foreach ( $watchedItems as $watchedItem ) {
|
|
$namespace = $watchedItem->getTarget()->getNamespace();
|
|
$dbKey = $watchedItem->getTarget()->getDBkey();
|
|
$lb->add( $namespace, $dbKey );
|
|
if ( !$this->nsInfo->isTalk( $namespace ) ) {
|
|
$titles[$namespace][$dbKey] = $watchedItem->getExpiryInDaysText( $context );
|
|
}
|
|
}
|
|
|
|
$lb->execute();
|
|
|
|
return $titles;
|
|
}
|
|
|
|
/**
|
|
* Validates watchlist entry
|
|
*
|
|
* @param Title $title
|
|
* @param int $namespace
|
|
* @param string $dbKey
|
|
* @return bool Whether this item is valid
|
|
*/
|
|
private function checkTitle( $title, $namespace, $dbKey ) {
|
|
if ( $title
|
|
&& ( $title->isExternal()
|
|
|| $title->getNamespace() < 0
|
|
)
|
|
) {
|
|
$title = false; // unrecoverable
|
|
}
|
|
|
|
if ( !$title
|
|
|| $title->getNamespace() != $namespace
|
|
|| $title->getDBkey() != $dbKey
|
|
) {
|
|
$this->badItems[] = [ $title, $namespace, $dbKey ];
|
|
}
|
|
|
|
return (bool)$title;
|
|
}
|
|
|
|
/**
|
|
* Attempts to clean up broken items
|
|
*/
|
|
private function cleanupWatchlist() {
|
|
if ( $this->badItems === [] ) {
|
|
return; // nothing to do
|
|
}
|
|
|
|
$user = $this->getUser();
|
|
$badItems = $this->badItems;
|
|
DeferredUpdates::addCallableUpdate( function () use ( $user, $badItems ) {
|
|
foreach ( $badItems as [ $title, $namespace, $dbKey ] ) {
|
|
$action = $title ? 'cleaning up' : 'deleting';
|
|
wfDebug( "User {$user->getName()} has broken watchlist item " .
|
|
"ns($namespace):$dbKey, $action." );
|
|
|
|
// NOTE: We *know* that the title is invalid. TitleValue may refuse instantiation.
|
|
// XXX: We may need an InvalidTitleValue class that allows instantiation of
|
|
// known bad title values.
|
|
$this->watchedItemStore->removeWatch( $user, Title::makeTitle( (int)$namespace, $dbKey ) );
|
|
// Can't just do an UPDATE instead of DELETE/INSERT due to unique index
|
|
if ( $title ) {
|
|
$this->watchlistManager->addWatch( $user, $title );
|
|
}
|
|
}
|
|
} );
|
|
}
|
|
|
|
/**
|
|
* Add a list of targets to a user's watchlist
|
|
*
|
|
* @param string[]|LinkTarget[] $targets
|
|
* @return bool
|
|
* @throws FatalError
|
|
* @throws MWException
|
|
*/
|
|
private function watchTitles( array $targets ) {
|
|
return $this->watchedItemStore->addWatchBatchForUser(
|
|
$this->getUser(), $this->getExpandedTargets( $targets )
|
|
) && $this->runWatchUnwatchCompleteHook( 'Watch', $targets );
|
|
}
|
|
|
|
/**
|
|
* Remove a list of titles from a user's watchlist
|
|
*
|
|
* $titles can be an array of strings or Title objects; the former
|
|
* is preferred, since Titles are very memory-heavy
|
|
*
|
|
* @param string[]|LinkTarget[] $targets
|
|
*
|
|
* @return bool
|
|
* @throws FatalError
|
|
* @throws MWException
|
|
*/
|
|
private function unwatchTitles( array $targets ) {
|
|
return $this->watchedItemStore->removeWatchBatchForUser(
|
|
$this->getUser(), $this->getExpandedTargets( $targets )
|
|
) && $this->runWatchUnwatchCompleteHook( 'Unwatch', $targets );
|
|
}
|
|
|
|
/**
|
|
* @param string $action
|
|
* Can be "Watch" or "Unwatch"
|
|
* @param string[]|LinkTarget[] $targets
|
|
* @return bool
|
|
* @throws FatalError
|
|
* @throws MWException
|
|
*/
|
|
private function runWatchUnwatchCompleteHook( $action, $targets ) {
|
|
foreach ( $targets as $target ) {
|
|
$title = $target instanceof LinkTarget ?
|
|
Title::newFromLinkTarget( $target ) :
|
|
Title::newFromText( $target );
|
|
$page = $this->wikiPageFactory->newFromTitle( $title );
|
|
$user = $this->getUser();
|
|
if ( $action === 'Watch' ) {
|
|
$this->getHookRunner()->onWatchArticleComplete( $user, $page );
|
|
} else {
|
|
$this->getHookRunner()->onUnwatchArticleComplete( $user, $page );
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* @param string[]|LinkTarget[] $targets
|
|
* @return TitleValue[]
|
|
*/
|
|
private function getExpandedTargets( array $targets ) {
|
|
$expandedTargets = [];
|
|
foreach ( $targets as $target ) {
|
|
if ( !$target instanceof LinkTarget ) {
|
|
try {
|
|
$target = $this->titleParser->parseTitle( $target, NS_MAIN );
|
|
}
|
|
catch ( MalformedTitleException $e ) {
|
|
continue;
|
|
}
|
|
}
|
|
|
|
$ns = $target->getNamespace();
|
|
$dbKey = $target->getDBkey();
|
|
$expandedTargets[] =
|
|
new TitleValue( $this->nsInfo->getSubject( $ns ), $dbKey );
|
|
$expandedTargets[] =
|
|
new TitleValue( $this->nsInfo->getTalk( $ns ), $dbKey );
|
|
}
|
|
return $expandedTargets;
|
|
}
|
|
|
|
public function submitNormal( $data ) {
|
|
$removed = [];
|
|
|
|
foreach ( $data as $titles ) {
|
|
$this->unwatchTitles( $titles );
|
|
$removed = array_merge( $removed, $titles );
|
|
}
|
|
|
|
if ( count( $removed ) > 0 ) {
|
|
$this->successMessage = $this->msg( 'watchlistedit-normal-done'
|
|
)->numParams( count( $removed ) )->parse();
|
|
$this->showTitles( $removed, $this->successMessage );
|
|
|
|
return true;
|
|
} else {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Get the standard watchlist editing form
|
|
*
|
|
* @return HTMLForm
|
|
*/
|
|
protected function getNormalForm() {
|
|
$fields = [];
|
|
$count = 0;
|
|
|
|
// Allow subscribers to manipulate the list of watched pages (or use it
|
|
// to preload lots of details at once)
|
|
$watchlistInfo = $this->getWatchlistInfo();
|
|
$this->getHookRunner()->onWatchlistEditorBeforeFormRender( $watchlistInfo );
|
|
|
|
foreach ( $watchlistInfo as $namespace => $pages ) {
|
|
$options = [];
|
|
foreach ( $pages as $dbkey => $expiryDaysText ) {
|
|
$title = Title::makeTitleSafe( $namespace, $dbkey );
|
|
|
|
if ( $this->checkTitle( $title, $namespace, $dbkey ) ) {
|
|
$text = $this->buildRemoveLine( $title, $expiryDaysText );
|
|
$options[$text] = $title->getPrefixedText();
|
|
$count++;
|
|
}
|
|
}
|
|
|
|
// checkTitle can filter some options out, avoid empty sections
|
|
if ( count( $options ) > 0 ) {
|
|
$fields['TitlesNs' . $namespace] = [
|
|
'class' => EditWatchlistCheckboxSeriesField::class,
|
|
'options' => $options,
|
|
'section' => "ns$namespace",
|
|
];
|
|
}
|
|
}
|
|
$this->cleanupWatchlist();
|
|
|
|
if ( count( $fields ) > 1 && $count > 30 ) {
|
|
$this->toc = Linker::tocIndent();
|
|
$tocLength = 0;
|
|
$contLang = $this->getContentLanguage();
|
|
|
|
foreach ( $fields as $data ) {
|
|
# strip out the 'ns' prefix from the section name:
|
|
$ns = substr( $data['section'], 2 );
|
|
|
|
$nsText = ( $ns == NS_MAIN )
|
|
? $this->msg( 'blanknamespace' )->escaped()
|
|
: htmlspecialchars( $contLang->getFormattedNsText( $ns ) );
|
|
$this->toc .= Linker::tocLine( "editwatchlist-{$data['section']}", $nsText,
|
|
$this->getLanguage()->formatNum( ++$tocLength ), 1 ) . Linker::tocLineEnd();
|
|
}
|
|
|
|
$this->toc = Linker::tocList( $this->toc );
|
|
} else {
|
|
$this->toc = false;
|
|
}
|
|
|
|
$context = new DerivativeContext( $this->getContext() );
|
|
$context->setTitle( $this->getPageTitle() ); // Remove subpage
|
|
$form = new EditWatchlistNormalHTMLForm( $fields, $context );
|
|
$form->setSubmitTextMsg( 'watchlistedit-normal-submit' );
|
|
$form->setSubmitDestructive();
|
|
# Used message keys:
|
|
# 'accesskey-watchlistedit-normal-submit', 'tooltip-watchlistedit-normal-submit'
|
|
$form->setSubmitTooltip( 'watchlistedit-normal-submit' );
|
|
$form->setWrapperLegendMsg( 'watchlistedit-normal-legend' );
|
|
$form->addHeaderText( $this->msg( 'watchlistedit-normal-explain' )->parse() );
|
|
$form->setSubmitCallback( [ $this, 'submitNormal' ] );
|
|
|
|
return $form;
|
|
}
|
|
|
|
/**
|
|
* Build the label for a checkbox, with a link to the title, and various additional bits
|
|
*
|
|
* @param Title $title
|
|
* @param string $expiryDaysText message shows the number of days a title has remaining in a user's watchlist.
|
|
* If this param is not empty then include a message that states the time remaining in a watchlist.
|
|
* @return string
|
|
*/
|
|
private function buildRemoveLine( $title, string $expiryDaysText = '' ): string {
|
|
$linkRenderer = $this->getLinkRenderer();
|
|
$link = $linkRenderer->makeLink( $title );
|
|
|
|
$tools = [];
|
|
$tools['talk'] = $linkRenderer->makeLink(
|
|
$title->getTalkPage(),
|
|
$this->msg( 'talkpagelinktext' )->text()
|
|
);
|
|
|
|
if ( $title->exists() ) {
|
|
$tools['history'] = $linkRenderer->makeKnownLink(
|
|
$title,
|
|
$this->msg( 'history_small' )->text(),
|
|
[],
|
|
[ 'action' => 'history' ]
|
|
);
|
|
}
|
|
|
|
if ( $title->getNamespace() === NS_USER && !$title->isSubpage() ) {
|
|
$tools['contributions'] = $linkRenderer->makeKnownLink(
|
|
SpecialPage::getTitleFor( 'Contributions', $title->getText() ),
|
|
$this->msg( 'contribslink' )->text()
|
|
);
|
|
}
|
|
|
|
$this->getHookRunner()->onWatchlistEditorBuildRemoveLine(
|
|
$tools, $title, $title->isRedirect(), $this->getSkin(), $link );
|
|
|
|
if ( $title->isRedirect() ) {
|
|
// Linker already makes class mw-redirect, so this is redundant
|
|
$link = '<span class="watchlistredir">' . $link . '</span>';
|
|
}
|
|
|
|
$watchlistExpiringMessage = '';
|
|
if ( $this->getConfig()->get( 'WatchlistExpiry' ) && $expiryDaysText ) {
|
|
$watchlistExpiringMessage = Html::element(
|
|
'span',
|
|
[ 'class' => 'mw-watchlistexpiry-msg' ],
|
|
$expiryDaysText
|
|
);
|
|
}
|
|
|
|
return $link . ' ' .
|
|
$this->msg( 'parentheses' )->rawParams( $this->getLanguage()->pipeList( $tools ) )->escaped() .
|
|
$watchlistExpiringMessage;
|
|
}
|
|
|
|
/**
|
|
* Get a form for editing the watchlist in "raw" mode
|
|
*
|
|
* @return HTMLForm
|
|
*/
|
|
protected function getRawForm() {
|
|
$titles = implode( "\n", $this->getWatchlist() );
|
|
$fields = [
|
|
'Titles' => [
|
|
'type' => 'textarea',
|
|
'label-message' => 'watchlistedit-raw-titles',
|
|
'default' => $titles,
|
|
],
|
|
];
|
|
$context = new DerivativeContext( $this->getContext() );
|
|
$context->setTitle( $this->getPageTitle( 'raw' ) ); // Reset subpage
|
|
$form = new OOUIHTMLForm( $fields, $context );
|
|
$form->setSubmitTextMsg( 'watchlistedit-raw-submit' );
|
|
# Used message keys: 'accesskey-watchlistedit-raw-submit', 'tooltip-watchlistedit-raw-submit'
|
|
$form->setSubmitTooltip( 'watchlistedit-raw-submit' );
|
|
$form->setWrapperLegendMsg( 'watchlistedit-raw-legend' );
|
|
$form->addHeaderText( $this->msg( 'watchlistedit-raw-explain' )->parse() );
|
|
$form->setSubmitCallback( [ $this, 'submitRaw' ] );
|
|
|
|
return $form;
|
|
}
|
|
|
|
/**
|
|
* Get a form for clearing the watchlist
|
|
*
|
|
* @return HTMLForm
|
|
*/
|
|
protected function getClearForm() {
|
|
$context = new DerivativeContext( $this->getContext() );
|
|
$context->setTitle( $this->getPageTitle( 'clear' ) ); // Reset subpage
|
|
$form = new OOUIHTMLForm( [], $context );
|
|
$form->setSubmitTextMsg( 'watchlistedit-clear-submit' );
|
|
# Used message keys: 'accesskey-watchlistedit-clear-submit', 'tooltip-watchlistedit-clear-submit'
|
|
$form->setSubmitTooltip( 'watchlistedit-clear-submit' );
|
|
$form->setWrapperLegendMsg( 'watchlistedit-clear-legend' );
|
|
$form->addHeaderText( $this->msg( 'watchlistedit-clear-explain' )->parse() );
|
|
$form->setSubmitCallback( [ $this, 'submitClear' ] );
|
|
$form->setSubmitDestructive();
|
|
|
|
return $form;
|
|
}
|
|
|
|
/**
|
|
* Determine whether we are editing the watchlist, and if so, what
|
|
* kind of editing operation
|
|
*
|
|
* @param WebRequest $request
|
|
* @param string|null $par
|
|
* @return int|false
|
|
*/
|
|
public static function getMode( $request, $par ) {
|
|
$mode = strtolower( $request->getRawVal( 'action', $par ) );
|
|
|
|
switch ( $mode ) {
|
|
case 'clear':
|
|
case self::EDIT_CLEAR:
|
|
return self::EDIT_CLEAR;
|
|
case 'raw':
|
|
case self::EDIT_RAW:
|
|
return self::EDIT_RAW;
|
|
case 'edit':
|
|
case self::EDIT_NORMAL:
|
|
return self::EDIT_NORMAL;
|
|
default:
|
|
return false;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Build a set of links for convenient navigation
|
|
* between watchlist viewing and editing modes
|
|
*
|
|
* @param Language $lang
|
|
* @param LinkRenderer|null $linkRenderer
|
|
* @return string
|
|
*/
|
|
public static function buildTools( $lang, LinkRenderer $linkRenderer = null ) {
|
|
if ( !$lang instanceof Language ) {
|
|
// back-compat where the first parameter was $unused
|
|
global $wgLang;
|
|
$lang = $wgLang;
|
|
}
|
|
if ( !$linkRenderer ) {
|
|
$linkRenderer = MediaWikiServices::getInstance()->getLinkRenderer();
|
|
}
|
|
|
|
$tools = [];
|
|
$modes = [
|
|
'view' => [ 'Watchlist', false ],
|
|
'edit' => [ 'EditWatchlist', false ],
|
|
'raw' => [ 'EditWatchlist', 'raw' ],
|
|
'clear' => [ 'EditWatchlist', 'clear' ],
|
|
];
|
|
|
|
foreach ( $modes as $mode => $arr ) {
|
|
// can use messages 'watchlisttools-view', 'watchlisttools-edit', 'watchlisttools-raw'
|
|
$tools[] = $linkRenderer->makeKnownLink(
|
|
SpecialPage::getTitleFor( $arr[0], $arr[1] ),
|
|
wfMessage( "watchlisttools-{$mode}" )->text()
|
|
);
|
|
}
|
|
|
|
return Html::rawElement(
|
|
'span',
|
|
[ 'class' => 'mw-watchlist-toollinks' ],
|
|
wfMessage( 'parentheses' )->rawParams( $lang->pipeList( $tools ) )->escaped()
|
|
);
|
|
}
|
|
}
|