3 * Implements Special:DeletedContributions
5 * This program is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License as published by
7 * the Free Software Foundation; either version 2 of the License, or
8 * (at your option) any later version.
10 * This program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 * GNU General Public License for more details.
15 * You should have received a copy of the GNU General Public License along
16 * with this program; if not, write to the Free Software Foundation, Inc.,
17 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
18 * http://www.gnu.org/copyleft/gpl.html
21 * @ingroup SpecialPage
24 use MediaWiki\Block\DatabaseBlock
;
25 use MediaWiki\MediaWikiServices
;
28 * Implements Special:DeletedContributions to display archived revisions
29 * @ingroup SpecialPage
31 class DeletedContributionsPage
extends SpecialPage
{
32 /** @var FormOptions */
35 function __construct() {
36 parent
::__construct( 'DeletedContributions', 'deletedhistory' );
40 * Special page "deleted user contributions".
41 * Shows a list of the deleted contributions of a user.
43 * @param string $par (optional) user name of the user for which to show the contributions
45 function execute( $par ) {
47 $this->outputHeader();
48 $this->checkPermissions();
49 $this->addHelpLink( 'Help:User contributions' );
51 $out = $this->getOutput();
52 $out->setPageTitle( $this->msg( 'deletedcontributions-title' ) );
54 $opts = new FormOptions();
56 $opts->add( 'target', '' );
57 $opts->add( 'namespace', '' );
58 $opts->add( 'limit', 20 );
60 $opts->fetchValuesFromRequest( $this->getRequest() );
61 $opts->validateIntBounds( 'limit', 0, $this->getConfig()->get( 'QueryPageDefaultLimit' ) );
63 if ( $par !== null ) {
64 // Beautify the username
65 $par = User
::getCanonicalName( $par, false );
66 $opts->setValue( 'target', (string)$par );
69 $ns = $opts->getValue( 'namespace' );
70 if ( $ns !== null && $ns !== '' ) {
71 $opts->setValue( 'namespace', intval( $ns ) );
76 $target = trim( $opts->getValue( 'target' ) );
77 if ( !strlen( $target ) ) {
83 $userObj = User
::newFromName( $target, false );
89 $this->getSkin()->setRelevantUser( $userObj );
91 $target = $userObj->getName();
92 $out->addSubtitle( $this->getSubTitle( $userObj ) );
96 $pager = new DeletedContribsPager( $this->getContext(), $target, $opts->getValue( 'namespace' ) );
97 if ( !$pager->getNumRows() ) {
98 $out->addWikiMsg( 'nocontribs' );
103 # Show a message about replica DB lag, if applicable
104 $lag = $pager->getDatabase()->getSessionLagStatus()['lag'];
106 $out->showLagWarning( $lag );
110 '<p>' . $pager->getNavigationBar() . '</p>' .
112 '<p>' . $pager->getNavigationBar() . '</p>' );
114 # If there were contributions, and it was a valid user or IP, show
115 # the appropriate "footer" message - WHOIS tools, etc.
116 if ( $target != 'newbies' ) {
117 $message = IP
::isIPAddress( $target ) ?
118 'sp-contributions-footer-anon' :
119 'sp-contributions-footer';
121 if ( !$this->msg( $message )->isDisabled() ) {
123 "<div class='mw-contributions-footer'>\n$1\n</div>",
124 [ $message, $target ]
131 * Generates the subheading with links
132 * @param User $userObj User object for the target
133 * @return string Appropriately-escaped HTML to be output literally
135 function getSubTitle( $userObj ) {
136 $linkRenderer = $this->getLinkRenderer();
137 if ( $userObj->isAnon() ) {
138 $user = htmlspecialchars( $userObj->getName() );
140 $user = $linkRenderer->makeLink( $userObj->getUserPage(), $userObj->getName() );
143 $nt = $userObj->getUserPage();
144 $talk = $nt->getTalkPage();
146 $tools = SpecialContributions
::getUserLinks( $this, $userObj );
148 $contributionsLink = $linkRenderer->makeKnownLink(
149 SpecialPage
::getTitleFor( 'Contributions', $nt->getDBkey() ),
150 $this->msg( 'sp-deletedcontributions-contribs' )->text()
152 if ( isset( $tools['deletedcontribs'] ) ) {
153 // Swap out the deletedcontribs link for our contribs one
154 $tools = wfArrayInsertAfter(
155 $tools, [ 'contribs' => $contributionsLink ], 'deletedcontribs' );
156 unset( $tools['deletedcontribs'] );
158 $tools['contribs'] = $contributionsLink;
161 $links = $this->getLanguage()->pipeList( $tools );
163 // Show a note if the user is blocked and display the last block log entry.
164 $block = DatabaseBlock
::newFromTarget( $userObj, $userObj );
165 if ( !is_null( $block ) && $block->getType() != DatabaseBlock
::TYPE_AUTO
) {
166 if ( $block->getType() == DatabaseBlock
::TYPE_RANGE
) {
167 $nt = MediaWikiServices
::getInstance()->getNamespaceInfo()->
168 getCanonicalName( NS_USER
) . ':' . $block->getTarget();
171 // LogEventsList::showLogExtract() wants the first parameter by ref
172 $out = $this->getOutput();
173 LogEventsList
::showLogExtract(
180 'showIfEmpty' => false,
182 'sp-contributions-blocked-notice',
183 $userObj->getName() # Support GENDER in 'sp-contributions-blocked-notice'
185 'offset' => '' # don't use $this->getRequest() parameter offset
191 return $this->msg( 'contribsub2' )->rawParams( $user, $links )->params( $userObj->getName() );
195 * Generates the namespace selector form with hidden attributes.
198 $opts = $this->mOpts
;
204 'label-message' => 'sp-contributions-username',
205 'default' => $opts->getValue( 'target' ),
210 'type' => 'namespaceselect',
211 'name' => 'namespace',
212 'label-message' => 'namespace',
217 HTMLForm
::factory( 'ooui', $formDescriptor, $this->getContext() )
218 ->setWrapperLegendMsg( 'sp-contributions-search' )
219 ->setSubmitTextMsg( 'sp-contributions-submit' )
220 // prevent setting subpage and 'target' parameter at the same time
221 ->setAction( $this->getPageTitle()->getLocalURL() )
224 ->displayForm( false );
228 * Return an array of subpages beginning with $search that this special page will accept.
230 * @param string $search Prefix to search for
231 * @param int $limit Maximum number of results to return (usually 10)
232 * @param int $offset Number of results to skip (usually 0)
233 * @return string[] Matching subpages
235 public function prefixSearchSubpages( $search, $limit, $offset ) {
236 $user = User
::newFromName( $search );
238 // No prefix suggestion for invalid user
241 // Autocomplete subpage as user list - public to allow caching
242 return UserNamePrefixSearch
::search( 'public', $search, $limit, $offset );
245 protected function getGroupName() {