24use MediaWiki\HookContainer\ProtectedHookAccessorTrait;
33use Wikimedia\NonSerializable\NonSerializableTrait;
46 use ProtectedHookAccessorTrait;
47 use NonSerializableTrait;
69 public $mRedirectUrl =
false;
89 protected $viewIsRenderAction =
false;
117 $this->mOldId = $oldId;
120 $services = MediaWikiServices::getInstance();
121 $this->linkRenderer = $services->getLinkRenderer();
122 $this->revisionStore = $services->getRevisionStore();
139 $t = Title::newFromID( $id );
140 return $t ==
null ? null :
new static(
$t );
157 Hooks::runner()->onArticleFromTitle(
$title, $page, $context );
159 switch (
$title->getNamespace() ) {
170 $page->setContext( $context );
183 $article = self::newFromTitle( $page->
getTitle(), $context );
184 $article->mPage = $page;
194 return $this->mRedirectedFrom;
203 $this->mRedirectedFrom = $from;
212 return $this->mPage->getTitle();
226 $this->mRedirectedFrom =
null; #
Title object if set
227 $this->mRedirectUrl =
false;
228 $this->mRevisionRecord =
null;
229 $this->fetchResult =
null;
233 $this->mPage->clear();
256 if ( $this->mPage->getId() === 0 ) {
263 RevisionRecord::FOR_THIS_USER,
277 # If this is a MediaWiki:x message, then load the messages
278 # and return the message value for x.
280 $text = $this->
getTitle()->getDefaultMessageText();
281 if ( $text ===
false ) {
287 $message = $this->
getContext()->getUser()->isRegistered() ?
'noarticletext' :
'noarticletextanon';
302 if ( $this->mOldId ===
null ) {
306 return $this->mOldId;
315 $this->mRedirectUrl =
false;
318 $oldid = $request->getIntOrNull(
'oldid' );
320 if ( $oldid ===
null ) {
324 if ( $oldid !== 0 ) {
325 # Load the given revision and check whether the page is another one.
326 # In that case, update this instance to reflect the change.
327 if ( $oldid === $this->mPage->getLatest() ) {
328 $this->mRevisionRecord = $this->mPage->getRevisionRecord();
330 $this->mRevisionRecord = $this->revisionStore->getRevisionById( $oldid );
331 if ( $this->mRevisionRecord !==
null ) {
332 $revPageId = $this->mRevisionRecord->getPageId();
334 if ( $this->mPage->getId() != $revPageId ) {
335 $function = get_class( $this->mPage ) .
'::newFromID';
336 $this->mPage = $function( $revPageId );
342 $oldRev = $this->mRevisionRecord;
343 if ( $request->getVal(
'direction' ) ==
'next' ) {
346 $nextRev = $this->revisionStore->getNextRevision( $oldRev );
348 $nextid = $nextRev->getId();
353 $this->mRevisionRecord =
null;
355 $this->mRedirectUrl = $this->
getTitle()->getFullURL(
'redirect=no' );
357 } elseif ( $request->getVal(
'direction' ) ==
'prev' ) {
360 $prevRev = $this->revisionStore->getPreviousRevision( $oldRev );
362 $previd = $prevRev->getId();
367 $this->mRevisionRecord =
null;
384 if ( $this->fetchResult ) {
385 return $this->mRevisionRecord;
391 if ( !$this->mRevisionRecord ) {
393 $this->mRevisionRecord = $this->mPage->getRevisionRecord();
395 if ( !$this->mRevisionRecord ) {
396 wfDebug( __METHOD__ .
" failed to find page data for title " .
397 $this->
getTitle()->getPrefixedText() );
400 $this->fetchResult = Status::newFatal(
'noarticletext' );
404 $this->mRevisionRecord = $this->revisionStore->getRevisionById( $oldid );
406 if ( !$this->mRevisionRecord ) {
407 wfDebug( __METHOD__ .
" failed to load revision, rev_id $oldid" );
409 $this->fetchResult = Status::newFatal(
'missing-revision', $oldid );
415 if ( !$this->mRevisionRecord->userCan( RevisionRecord::DELETED_TEXT, $this->getContext()->getAuthority() ) ) {
416 wfDebug( __METHOD__ .
" failed to retrieve content of revision " .
417 $this->mRevisionRecord->getId() );
421 $this->fetchResult = Status::newFatal(
422 'rev-deleted-text-permission', $this->
getTitle()->getPrefixedDBkey() );
426 $this->fetchResult = Status::newGood( $this->mRevisionRecord );
427 return $this->mRevisionRecord;
436 # If no oldid, this is the current version.
441 return $this->mPage->exists() &&
442 $this->mRevisionRecord &&
443 $this->mRevisionRecord->isCurrent();
459 return $revRecord ?
new Revision( $revRecord ) :
null;
471 if ( $this->fetchResult && $this->fetchResult->isOK() ) {
472 return $this->fetchResult->value->getId();
474 return $this->mPage->getLatest();
485 # Get variables from query string
486 # As side effect this will load the revision and update the title
487 # in a revision ID is passed in the request, so this should remain
488 # the first call of this method even if $oldid is used way below.
492 # Another whitelist check in case getOldID() is altering the title
493 $permissionStatus = PermissionStatus::newEmpty();
495 ->authorizeRead(
'read', $this->
getTitle(), $permissionStatus )
497 wfDebug( __METHOD__ .
": denied on secondary read check" );
502 # getOldID() may as well want us to redirect somewhere else
503 if ( $this->mRedirectUrl ) {
504 $outputPage->
redirect( $this->mRedirectUrl );
505 wfDebug( __METHOD__ .
": redirecting due to oldid" );
510 # If we got diff in the query, we want to see a diff page instead of the article.
511 if ( $this->
getContext()->getRequest()->getCheck(
'diff' ) ) {
512 wfDebug( __METHOD__ .
": showing diff page" );
518 # Set page title (may be overridden by DISPLAYTITLE)
522 # Allow frames by default
527 # Allow extensions to vary parser options used for article rendering
528 Hooks::runner()->onArticleParserOptions( $this, $parserOptions );
529 # Render printable version, use printable version cache
532 $poOptions[
'enableSectionEditLinks'] =
false;
535 $outputPage->
msg(
'printableversion-deprecated-warning' )->escaped()
538 } elseif ( $this->viewIsRenderAction || !$this->
isCurrent() ||
541 $poOptions[
'enableSectionEditLinks'] =
false;
544 # Try client and file cache
545 if ( $oldid === 0 && $this->mPage->checkTouched() ) {
546 # Try to stream the output from file cache
548 wfDebug( __METHOD__ .
": done file cache" );
549 # tell wgOut that output is taken care of
551 $this->mPage->doViewUpdates( $user, $oldid );
567 # For the main page, overwrite the <title> element with the con-
568 # tents of 'pagetitle-view-mainpage' instead of the default (if
570 # This message always exists because it is in the i18n files
571 if ( $this->
getTitle()->isMainPage() ) {
572 $msg =
wfMessage(
'pagetitle-view-mainpage' )->inContentLanguage();
573 if ( !$msg->isDisabled() ) {
574 $outputPage->
setHTMLTitle( $msg->title( $this->getTitle() )->text() );
578 # Use adaptive TTLs for CDN so delayed/failed purges are noticed less often.
579 # This could use getTouched(), but that could be scary for major template edits.
580 $outputPage->
adaptCdnTTL( $this->mPage->getTimestamp(), IExpiringStore::TTL_DAY );
583 $this->mPage->doViewUpdates( $user, $oldid );
585 # Load the postEdit module if the user just saved this revision
586 # See also EditPage::setPostEditCookie
589 $postEdit = $request->getCookie( $cookieKey );
591 # Clear the cookie. This also prevents caching of the response.
592 $request->response()->clearCookie( $cookieKey );
594 $outputPage->
addModules(
'mediawiki.action.view.postEdit' );
617 # Should the parser cache be used?
618 $useParserCache =
true;
620 $parserOutputAccess = MediaWikiServices::getInstance()->getParserOutputAccess();
623 $this->getHookRunner()->onArticleViewHeader( $this, $outputDone, $useParserCache );
626 $pOutput = $outputDone;
636 if ( !$this->mPage->exists() ) {
637 wfDebug( __METHOD__ .
": showing missing article" );
638 $this->showMissingArticle();
639 $this->mPage->doViewUpdates( $user );
645 if ( $useParserCache && !$oldid ) {
646 $pOutput = $parserOutputAccess->getCachedParserOutput(
650 ParserOutputAccess::OPT_NO_AUDIENCE_CHECK
661 if ( !$this->fetchResult->isOK() ) {
663 false,
false, $this->getContext()->getLanguage()
668 # Are we looking at an old revision
673 wfDebug( __METHOD__ .
": cannot view deleted revision" );
680 if ( $useParserCache ) {
681 $pOutput = $parserOutputAccess->getCachedParserOutput(
685 ParserOutputAccess::OPT_NO_AUDIENCE_CHECK
696 # Ensure that UI elements requiring revision ID have
697 # the correct version information.
699 # Preload timestamp to avoid a DB hit
702 # Pages containing custom CSS or JavaScript get special treatment
703 if ( $this->
getTitle()->isSiteConfigPage() || $this->
getTitle()->isUserConfigPage() ) {
704 $dir = $this->
getContext()->getLanguage()->getDir();
708 "<div id='mw-clearyourcache' lang='$lang' dir='$dir' class='mw-content-$dir'>\n$1\n</div>",
712 } elseif ( !$this->getHookRunner()->onArticleRevisionViewCustom(
724 # Run the parse, protected by a pool counter
725 wfDebug( __METHOD__ .
": doing uncached parse" );
735 $opt |= ParserOutputAccess::OPT_NO_CHECK_CACHE;
738 $opt |= ParserOutputAccess::OPT_NO_AUDIENCE_CHECK;
740 if ( !$rev->getId() || !$useParserCache ) {
742 $opt |= ParserOutputAccess::OPT_NO_CACHE;
745 $renderStatus = $parserOutputAccess->getParserOutput(
759 if ( !$renderStatus->isOK() ) {
763 $pOutput = $renderStatus->getValue();
773 # Adjust title for main page & pages with displaytitle
775 $this->adjustDisplayTitle( $pOutput );
778 # Check for any __NOINDEX__ tags on the page using $pOutput
779 $policy = $this->getRobotPolicy(
'view', $pOutput ?: null );
783 $this->mParserOutput = $pOutput;
797 # Ensure that UI elements requiring revision ID have
798 # the correct version information.
800 # Preload timestamp to avoid a DB hit
802 if ( $cachedTimestamp !==
null ) {
804 $this->mPage->setTimestamp( $cachedTimestamp );
821 $ok = $renderStatus->
isOK();
823 $pOutput = $ok ? $renderStatus->
getValue() :
null;
826 if ( $ok && $renderStatus->
hasMessage(
'view-pool-dirty-output' ) ) {
829 $staleReason = $renderStatus->
hasMessage(
'view-pool-contention' )
830 ? $this->
getContext()->msg(
'view-pool-contention' )
831 : $this->
getContext()->msg(
'view-pool-timeout' );
832 $outputPage->
addHTML(
"<!-- parser cache is expired, " .
833 "sending anyway due to $staleReason-->\n" );
836 if ( !$renderStatus->
isOK() ) {
838 false,
'view-pool-error', $this->getContext()->getLanguage()
847 if ( $this->getRevisionRedirectTarget( $rev ) ) {
848 $outputPage->
addSubtitle(
"<span id=\"redirectsub\">" .
849 $this->
getContext()->msg(
'redirectpagesub' )->parse() .
"</span>" );
872 # Adjust the title if it was set by displaytitle, -{T|}- or language conversion
874 if ( strval( $titleText ) !==
'' ) {
875 $out->setPageTitle( $titleText );
876 $out->setDisplayTitle( $titleText );
887 $diff = $request->getVal(
'diff' );
888 $rcid = $request->getVal(
'rcid' );
889 $diffOnly = $request->getBool(
'diffonly', $user->
getOption(
'diffonly' ) );
890 $purge = $request->getVal(
'action' ) ==
'purge';
891 $unhide = $request->getInt(
'unhide' ) == 1;
892 $oldid = $this->getOldID();
894 $rev = $this->fetchRevisionRecord();
899 $rev = $this->revisionStore->getRevisionById( $oldid );
906 $msg = $this->
getContext()->msg(
'difference-missing-revision' )
910 $this->
getContext()->getOutput()->addHTML( $msg );
915 $contentHandler = MediaWikiServices::getInstance()
916 ->getContentHandlerFactory()
918 $rev->getSlot( SlotRecord::MAIN, RevisionRecord::RAW )->getModel()
920 $de = $contentHandler->createDifferenceEngine(
928 $de->setSlotDiffOptions( [
929 'diff-type' => $request->getVal(
'diff-type' )
931 $de->showDiffPage( $diffOnly );
935 list( $old, $new ) = $de->mapDiffPrevNext( $oldid, $diff );
937 $this->mPage->doViewUpdates( $user, (
int)$new );
950 $ns = $this->
getTitle()->getNamespace();
952 # Don't index user and user talk pages for blocked users (T13443)
954 $specificTarget =
null;
956 $titleText = $this->
getTitle()->getText();
957 if ( IPUtils::isValid( $titleText ) ) {
958 $vagueTarget = $titleText;
960 $specificTarget = $titleText;
962 if ( DatabaseBlock::newFromTarget( $specificTarget, $vagueTarget ) instanceof
DatabaseBlock ) {
964 'index' =>
'noindex',
965 'follow' =>
'nofollow'
970 if ( $this->mPage->getId() === 0 || $this->getOldID() ) {
971 # Non-articles (special pages etc), and old revisions
973 'index' =>
'noindex',
974 'follow' =>
'nofollow'
976 } elseif ( $this->
getContext()->getOutput()->isPrintable() ) {
977 # Discourage indexing of printable versions, but encourage following
979 'index' =>
'noindex',
982 } elseif ( $this->
getContext()->getRequest()->getInt(
'curid' ) ) {
983 # For ?curid=x urls, disallow indexing
985 'index' =>
'noindex',
990 # Otherwise, construct the policy based on the various config variables.
994 # Honour customised robot policies for this namespace
995 $policy = array_merge(
1000 if ( $this->
getTitle()->canUseNoindex() && is_object( $pOutput ) && $pOutput->getIndexPolicy() ) {
1001 # __INDEX__ and __NOINDEX__ magic words, if allowed. Incorporates
1002 # a final sanity check that we have really got the parser output.
1003 $policy = array_merge(
1005 [
'index' => $pOutput->getIndexPolicy() ]
1010 # (T16900) site config can override user-defined __INDEX__ or __NOINDEX__
1011 $policy = array_merge(
1028 if ( is_array( $policy ) ) {
1030 } elseif ( !$policy ) {
1034 $policy = explode(
',', $policy );
1035 $policy = array_map(
'trim', $policy );
1038 foreach ( $policy as $var ) {
1039 if ( in_array( $var, [
'index',
'noindex' ] ) ) {
1040 $arr[
'index'] = $var;
1041 } elseif ( in_array( $var, [
'follow',
'nofollow' ] ) ) {
1042 $arr[
'follow'] = $var;
1061 $request = $context->getRequest();
1062 $rdfrom = $request->getVal(
'rdfrom' );
1065 $query = $request->getValues();
1066 unset( $query[
'rdfrom'] );
1067 unset( $query[
'title'] );
1068 if ( $this->
getTitle()->isRedirect() ) {
1070 $query[
'redirect'] =
'no';
1072 $redirectTargetUrl = $this->
getTitle()->getLinkURL( $query );
1074 if ( isset( $this->mRedirectedFrom ) ) {
1077 if ( $this->getHookRunner()->onArticleViewRedirect( $this ) ) {
1078 $redir = $this->linkRenderer->makeKnownLink(
1079 $this->mRedirectedFrom,
1082 [
'redirect' =>
'no' ]
1085 $outputPage->
addSubtitle(
"<span class=\"mw-redirectedfrom\">" .
1086 $context->msg(
'redirectedfrom' )->rawParams( $redir )->parse()
1092 'wgInternalRedirectTargetUrl' => $redirectTargetUrl,
1094 $outputPage->
addModules(
'mediawiki.action.view.redirect' );
1104 } elseif ( $rdfrom ) {
1109 $outputPage->
addSubtitle(
"<span class=\"mw-redirectedfrom\">" .
1110 $context->msg(
'redirectedfrom' )->rawParams( $redir )->parse()
1115 'wgInternalRedirectTargetUrl' => $redirectTargetUrl,
1117 $outputPage->
addModules(
'mediawiki.action.view.redirect' );
1131 if ( $this->
getTitle()->isTalkPage() && !
wfMessage(
'talkpageheader' )->isDisabled() ) {
1132 $this->
getContext()->getOutput()->wrapWikiMsg(
1133 "<div class=\"mw-talkpageheader\">\n$1\n</div>",
1134 [
'talkpageheader' ]
1143 # check if we're displaying a [[User talk:x.x.x.x]] anonymous talk page
1145 && IPUtils::isValid( $this->
getTitle()->getText() )
1147 $this->
getContext()->getOutput()->addWikiMsg(
'anontalkpagetext' );
1151 $patrolFooterShown = $this->showPatrolFooter();
1153 $this->getHookRunner()->onArticleViewFooter( $this, $patrolFooterShown );
1170 if ( !$this->getHookRunner()->onArticleShowPatrolFooter( $this ) ) {
1187 if ( $this->mRevisionRecord
1188 && !RecentChange::isInRCLifespan( $this->mRevisionRecord->getTimestamp(), 21600 )
1196 $cache = MediaWikiServices::getInstance()->getMainWANObjectCache();
1197 $key =
$cache->makeKey(
'unpatrollable-page',
$title->getArticleID() );
1198 if (
$cache->get( $key ) ) {
1203 $oldestRevisionTimestamp =
$dbr->selectField(
1205 'MIN( rev_timestamp )',
1206 [
'rev_page' =>
$title->getArticleID() ],
1215 $recentPageCreation =
false;
1216 if ( $oldestRevisionTimestamp
1217 && RecentChange::isInRCLifespan( $oldestRevisionTimestamp, 21600 )
1220 $recentPageCreation =
true;
1221 $rc = RecentChange::newFromConds(
1224 'rc_timestamp' => $oldestRevisionTimestamp,
1225 'rc_namespace' =>
$title->getNamespace(),
1226 'rc_cur_id' =>
$title->getArticleID()
1232 $markPatrolledMsg =
wfMessage(
'markaspatrolledtext' );
1240 $recentFileUpload =
false;
1244 $newestUploadTimestamp =
$dbr->selectField(
1246 'MAX( img_timestamp )',
1247 [
'img_name' =>
$title->getDBkey() ],
1250 if ( $newestUploadTimestamp
1251 && RecentChange::isInRCLifespan( $newestUploadTimestamp, 21600 )
1254 $recentFileUpload =
true;
1255 $rc = RecentChange::newFromConds(
1258 'rc_log_type' =>
'upload',
1259 'rc_timestamp' => $newestUploadTimestamp,
1261 'rc_cur_id' =>
$title->getArticleID()
1267 $markPatrolledMsg =
wfMessage(
'markaspatrolledtext-file' );
1272 if ( !$recentPageCreation && !$recentFileUpload ) {
1277 $cache->set( $key,
'1' );
1289 if ( $rc->getAttribute(
'rc_patrolled' ) ) {
1294 $cache->set( $key,
'1' );
1299 if ( $rc->getPerformer()->equals( $user ) ) {
1307 $outputPage->
addModules(
'mediawiki.misc-authed-curate' );
1310 $link = $this->linkRenderer->makeKnownLink(
1312 $markPatrolledMsg->text(),
1315 'action' =>
'markpatrolled',
1316 'rcid' => $rc->getAttribute(
'rc_id' ),
1321 "<div class='patrollink' data-mw='interface'>" .
1322 wfMessage(
'markaspatrolledlink' )->rawParams( $link )->escaped() .
1336 $cache = MediaWikiServices::getInstance()->getMainWANObjectCache();
1337 $cache->delete(
$cache->makeKey(
'unpatrollable-page', $articleID ) );
1349 $validUserPage =
false;
1353 $services = MediaWikiServices::getInstance();
1355 $contextUser = $this->
getContext()->getUser();
1357 # Show info in user (talk) namespace. Does the user exist? Is he blocked?
1361 $rootPart = explode(
'/',
$title->getText() )[0];
1364 $block = DatabaseBlock::newFromTarget( $user, $user );
1367 !$this->getContext()->getAuthority()->isAllowed(
'hideuser' )
1374 $outputPage->
wrapWikiMsg(
"<div class=\"mw-userpage-userdoesnotexist error\">\n\$1\n</div>",
1378 $block->getType() != DatabaseBlock::TYPE_AUTO &&
1383 LogEventsList::showLogExtract(
1386 $services->getNamespaceInfo()->getCanonicalName(
NS_USER ) .
':' .
1391 'showIfEmpty' =>
false,
1393 'blocked-notice-logextract',
1394 $user->
getName() # Support GENDER in notice
1398 $validUserPage = !
$title->isSubpage();
1400 $validUserPage = !
$title->isSubpage();
1404 $this->getHookRunner()->onShowMissingArticle( $this );
1406 # Show delete and move logs if there were any such events.
1407 # The logging query can DOS the site when bots/crawlers cause 404 floods,
1408 # so be careful showing this. 404 pages must be cheap as they are hard to cache.
1409 $dbCache = ObjectCache::getInstance(
'db-replicated' );
1410 $key = $dbCache->makeKey(
'page-recent-delete', md5(
$title->getPrefixedText() ) );
1411 $isRegistered = $contextUser->isRegistered();
1412 $sessionExists = $this->
getContext()->getRequest()->getSession()->isPersistent();
1414 if ( $isRegistered || $dbCache->get( $key ) || $sessionExists ) {
1415 $logTypes = [
'delete',
'move',
'protect' ];
1419 $conds = [
'log_action != ' .
$dbr->addQuotes(
'revision' ) ];
1421 $this->getHookRunner()->onArticle__MissingArticleConditions( $conds, $logTypes );
1422 LogEventsList::showLogExtract(
1430 'showIfEmpty' =>
false,
1431 'msgKey' => [ $isRegistered || $sessionExists
1432 ?
'moveddeleted-notice'
1433 :
'moveddeleted-notice-recent'
1439 if ( !$this->mPage->hasViewableContent() &&
$wgSend404Code && !$validUserPage ) {
1442 $this->
getContext()->getRequest()->response()->statusHeader( 404 );
1446 $policy = $this->getRobotPolicy(
'view' );
1450 $hookResult = $this->getHookRunner()->onBeforeDisplayNoArticleText( $this );
1452 if ( !$hookResult ) {
1456 # Show error message
1457 $oldid = $this->getOldID();
1459 $text = $this->
getTitle()->getDefaultMessageText() ??
'';
1466 $revRecord = $pa->getArchivedRevisionRecord( $oldid );
1467 if ( $revRecord && $revRecord->userCan(
1468 RevisionRecord::DELETED_TEXT,
1469 $this->getContext()->getAuthority()
1472 'missing-revision-permission', $oldid,
1473 $revRecord->getTimestamp(),
1474 $title->getPrefixedDBkey()
1477 $text =
wfMessage(
'missing-revision', $oldid )->plain();
1483 $message = $isRegistered ?
'noarticletext' :
'noarticletextanon';
1486 $text =
wfMessage(
'noarticletext-nopermission' )->plain();
1489 $dir = $this->
getContext()->getLanguage()->getDir();
1492 'class' =>
"noarticletext mw-content-$dir",
1495 ] ) .
"\n$text\n</div>" );
1519 if ( !$this->mRevisionRecord->isDeleted( RevisionRecord::DELETED_TEXT ) ) {
1526 $titleText = $this->
getTitle()->getPrefixedDBkey();
1528 if ( !$this->mRevisionRecord->userCan(
1529 RevisionRecord::DELETED_TEXT,
1530 $this->getContext()->getAuthority()
1532 $outputPage->addHtml(
1534 $outputPage->
msg(
'rev-deleted-text-permission', $titleText )->parse(),
1541 } elseif ( $this->
getContext()->getRequest()->getInt(
'unhide' ) != 1 ) {
1542 # Give explanation and add a link to view the revision...
1543 $oldid = intval( $this->getOldID() );
1544 $link = $this->
getTitle()->getFullURL(
"oldid={$oldid}&unhide=1" );
1545 $msg = $this->mRevisionRecord->isDeleted( RevisionRecord::DELETED_RESTRICTED ) ?
1546 'rev-suppressed-text-unhide' :
'rev-deleted-text-unhide';
1547 $outputPage->addHtml(
1549 $outputPage->
msg( $msg, $link )->parse(),
1557 $msg = $this->mRevisionRecord->isDeleted( RevisionRecord::DELETED_RESTRICTED )
1558 ? [
'rev-suppressed-text-view', $titleText ]
1559 : [
'rev-deleted-text-view', $titleText ];
1560 $outputPage->addHtml(
1562 $outputPage->
msg( $msg[0], $msg[1] )->parse(),
1580 if ( !$this->getHookRunner()->onDisplayOldSubtitle( $this, $oldid ) ) {
1585 $unhide = $context->getRequest()->getInt(
'unhide' ) == 1;
1587 # Cascade unhide param in links for easy deletion browsing
1590 $extraParams[
'unhide'] = 1;
1593 if ( $this->mRevisionRecord && $this->mRevisionRecord->getId() === $oldid ) {
1594 $revisionRecord = $this->mRevisionRecord;
1596 $revisionRecord = $this->revisionStore->getRevisionById( $oldid );
1599 $timestamp = $revisionRecord->getTimestamp();
1601 $current = ( $oldid == $this->mPage->getLatest() );
1602 $language = $context->getLanguage();
1605 $td = $language->userTimeAndDate( $timestamp, $user );
1606 $tddate = $language->userDate( $timestamp, $user );
1607 $tdtime = $language->userTime( $timestamp, $user );
1609 # Show user links if allowed to see them. If hidden, then show them only if requested...
1612 $infomsg = $current && !$context->msg(
'revision-info-current' )->isDisabled()
1613 ?
'revision-info-current'
1618 'mediawiki.action.styles',
1619 'mediawiki.interface.helpers.styles'
1622 $revisionUser = $revisionRecord->getUser();
1623 $revisionInfo =
"<div id=\"mw-{$infomsg}\">" .
1624 $context->msg( $infomsg, $td )
1625 ->rawParams( $userlinks )
1627 $revisionRecord->getId(),
1630 $revisionUser ? $revisionUser->getName() :
''
1641 ? $context->msg(
'currentrevisionlink' )->escaped()
1642 : $this->linkRenderer->makeKnownLink(
1644 $context->msg(
'currentrevisionlink' )->text(),
1649 ? $context->msg(
'diff' )->escaped()
1650 : $this->linkRenderer->makeKnownLink(
1652 $context->msg(
'diff' )->text(),
1659 $prevExist = (bool)$this->revisionStore->getPreviousRevision( $revisionRecord );
1660 $prevlink = $prevExist
1661 ? $this->linkRenderer->makeKnownLink(
1663 $context->msg(
'previousrevision' )->text(),
1666 'direction' =>
'prev',
1670 : $context->msg(
'previousrevision' )->escaped();
1671 $prevdiff = $prevExist
1672 ? $this->linkRenderer->makeKnownLink(
1674 $context->msg(
'diff' )->text(),
1681 : $context->msg(
'diff' )->escaped();
1682 $nextlink = $current
1683 ? $context->msg(
'nextrevision' )->escaped()
1684 : $this->linkRenderer->makeKnownLink(
1686 $context->msg(
'nextrevision' )->text(),
1689 'direction' =>
'next',
1693 $nextdiff = $current
1694 ? $context->msg(
'diff' )->escaped()
1695 : $this->linkRenderer->makeKnownLink(
1697 $context->msg(
'diff' )->text(),
1710 if ( $cdel !==
'' ) {
1715 $outputPage->
addSubtitle(
"<div class=\"mw-revision warningbox\">" . $revisionInfo .
1716 "<div id=\"mw-revision-nav\">" . $cdel .
1717 $context->msg(
'revision-nav' )->rawParams(
1718 $prevdiff, $prevlink, $lnk, $curdiff, $nextlink, $nextdiff
1719 )->escaped() .
"</div></div>" );
1735 public function viewRedirect( $target, $appendSubtitle =
true, $forceKnown =
false ) {
1738 if ( $appendSubtitle ) {
1739 $out->addSubtitle(
wfMessage(
'redirectpagesub' ) );
1741 $out->addModuleStyles(
'mediawiki.action.view.redirectPage' );
1742 return static::getRedirectHeaderHtml(
$lang, $target, $forceKnown );
1758 if ( !is_array( $target ) ) {
1759 $target = [ $target ];
1762 $linkRenderer = MediaWikiServices::getInstance()->getLinkRenderer();
1764 $html =
'<ul class="redirectText">';
1766 foreach ( $target as
$title ) {
1767 if ( $forceKnown ) {
1773 $title->isRedirect() ? [
'redirect' =>
'no' ] : []
1781 $title->isRedirect() ? [
'redirect' =>
'no' ] : []
1784 $html .=
'<li>' . $link .
'</li>';
1788 $redirectToText =
wfMessage(
'redirectto' )->inLanguage(
$lang )->escaped();
1790 return '<div class="redirectMsg">' .
1791 '<p>' . $redirectToText .
'</p>' .
1806 $msg = $out->msg(
'namespace-' . $this->
getTitle()->getNamespace() .
'-helppage' );
1808 if ( !$msg->isDisabled() ) {
1809 $title = Title::newFromText( $msg->plain() );
1811 $out->addHelpLink(
$title->getLocalURL(),
true );
1814 $out->addHelpLink( $to, $overrideBaseUrl );
1822 $this->
getContext()->getRequest()->response()->header(
'X-Robots-Tag: noindex' );
1823 $this->
getContext()->getOutput()->setArticleBodyOnly(
true );
1825 $this->viewIsRenderAction =
true;
1847 public function delete() {
1848 # This code desperately needs to be totally rewritten
1853 $request = $context->getRequest();
1856 $permissionStatus = PermissionStatus::newEmpty();
1857 if ( !$context->getAuthority()->authorizeWrite(
'delete',
$title, $permissionStatus ) ) {
1861 # Read-only check...
1866 # Better double-check that it hasn't been deleted yet!
1867 $this->mPage->loadPageData(
1868 $request->wasPosted() ? WikiPage::READ_LATEST : WikiPage::READ_NORMAL
1870 if ( !$this->mPage->exists() ) {
1871 $deleteLogPage =
new LogPage(
'delete' );
1873 $outputPage->
setPageTitle( $context->msg(
'cannotdelete-title',
$title->getPrefixedText() ) );
1874 $outputPage->
wrapWikiMsg(
"<div class=\"error mw-error-cannotdelete\">\n$1\n</div>",
1878 Xml::element(
'h2',
null, $deleteLogPage->getName()->text() )
1880 LogEventsList::showLogExtract(
1889 $deleteReasonList = $request->getText(
'wpDeleteReasonList',
'other' );
1890 $deleteReason = $request->getText(
'wpReason' );
1892 if ( $deleteReasonList ==
'other' ) {
1893 $reason = $deleteReason;
1894 } elseif ( $deleteReason !=
'' ) {
1896 $colonseparator =
wfMessage(
'colon-separator' )->inContentLanguage()->text();
1897 $reason = $deleteReasonList . $colonseparator . $deleteReason;
1899 $reason = $deleteReasonList;
1902 if ( $request->wasPosted() && $user->
matchEditToken( $request->getVal(
'wpEditToken' ),
1903 [
'delete', $this->getTitle()->getPrefixedText() ] )
1905 # Flag to hide all contents of the archived revisions
1907 $suppress = $request->getCheck(
'wpSuppress' ) &&
1908 $context->getAuthority()->isAllowed(
'suppressrevision' );
1910 $this->doDelete( $reason, $suppress );
1918 $hasHistory =
false;
1921 $reason = $this->getPage()
1922 ->getAutoDeleteReason( $hasHistory );
1923 }
catch ( Exception $e ) {
1924 # if a page is horribly broken, we still want to be able to
1925 # delete it. So be lenient about errors here.
1926 wfDebug(
"Error while building auto delete summary: $e" );
1932 if ( $hasHistory ) {
1941 $revisions = $edits = (int)
$dbr->selectField(
1944 [
'rev_page' =>
$title->getArticleID() ],
1949 $context->getOutput()->addHTML(
1950 '<strong class="mw-delete-warning-revisions">' .
1951 $context->msg(
'historywarning' )->numParams( $revisions )->parse() .
1952 $context->msg(
'word-separator' )->escaped() . $this->linkRenderer->makeKnownLink(
1954 $context->msg(
'history' )->text(),
1956 [
'action' =>
'history' ] ) .
1960 if (
$title->isBigDeletion() ) {
1962 $context->getOutput()->wrapWikiMsg(
"<div class='error'>\n$1\n</div>\n",
1964 'delete-warning-toobig',
1971 $this->confirmDelete( $reason );
1980 wfDebug(
"Article::confirmDelete" );
1988 $outputPage->
addModules(
'mediawiki.action.delete' );
1991 $backlinkCache =
$title->getBacklinkCache();
1992 if ( $backlinkCache->hasLinks(
'pagelinks' ) || $backlinkCache->hasLinks(
'templatelinks' ) ) {
1993 $outputPage->addHtml(
1995 $outputPage->
msg(
'deleting-backlinks-warning' )->parse(),
2001 $subpageQueryLimit = 51;
2002 $subpages =
$title->getSubpages( $subpageQueryLimit );
2003 $subpageCount = count( $subpages );
2004 if ( $subpageCount > 0 ) {
2005 $outputPage->addHtml(
2012 $outputPage->
addWikiMsg(
'confirmdeletetext' );
2014 $this->getHookRunner()->onArticleConfirmDelete( $this, $outputPage, $reason );
2017 $checkWatch = MediaWikiServices::getInstance()->getUserOptionsLookup()
2018 ->getBoolOption( $user,
'watchdeletion' ) || $user->
isWatched(
$title );
2024 $suppressAllowed = $this->
getContext()->getAuthority()->isAllowed(
'suppressrevision' );
2025 $dropDownReason = $ctx->msg(
'deletereason-dropdown' )->inContentLanguage()->text();
2027 if ( $suppressAllowed ) {
2028 $dropDownReason .=
"\n" . $ctx->msg(
'deletereason-dropdown-suppress' )
2029 ->inContentLanguage()->text();
2032 $options = Xml::listDropDownOptions(
2034 [
'other' => $ctx->msg(
'deletereasonotherlist' )->inContentLanguage()->text() ]
2036 $options = Xml::listDropDownOptionsOoui( $options );
2038 $fields[] =
new OOUI\FieldLayout(
2039 new OOUI\DropdownInputWidget( [
2040 'name' =>
'wpDeleteReasonList',
2041 'inputId' =>
'wpDeleteReasonList',
2043 'infusable' =>
true,
2045 'options' => $options
2048 'label' => $ctx->msg(
'deletecomment' )->text(),
2056 $fields[] =
new OOUI\FieldLayout(
2057 new OOUI\TextInputWidget( [
2058 'name' =>
'wpReason',
2059 'inputId' =>
'wpReason',
2061 'maxLength' => CommentStore::COMMENT_CHARACTER_LIMIT,
2062 'infusable' =>
true,
2064 'autofocus' =>
true,
2067 'label' => $ctx->msg(
'deleteotherreason' )->text(),
2073 $fields[] =
new OOUI\FieldLayout(
2074 new OOUI\CheckboxInputWidget( [
2075 'name' =>
'wpWatch',
2076 'inputId' =>
'wpWatch',
2078 'selected' => $checkWatch,
2081 'label' => $ctx->msg(
'watchthis' )->text(),
2082 'align' =>
'inline',
2083 'infusable' =>
true,
2087 if ( $suppressAllowed ) {
2088 $fields[] =
new OOUI\FieldLayout(
2089 new OOUI\CheckboxInputWidget( [
2090 'name' =>
'wpSuppress',
2091 'inputId' =>
'wpSuppress',
2095 'label' => $ctx->msg(
'revdelete-suppress' )->text(),
2096 'align' =>
'inline',
2097 'infusable' =>
true,
2102 $fields[] =
new OOUI\FieldLayout(
2103 new OOUI\ButtonInputWidget( [
2104 'name' =>
'wpConfirmB',
2105 'inputId' =>
'wpConfirmB',
2107 'value' => $ctx->msg(
'deletepage' )->text(),
2108 'label' => $ctx->msg(
'deletepage' )->text(),
2109 'flags' => [
'primary',
'destructive' ],
2117 $fieldset =
new OOUI\FieldsetLayout( [
2118 'label' => $ctx->msg(
'delete-legend' )->text(),
2119 'id' =>
'mw-delete-table',
2123 $form =
new OOUI\FormLayout( [
2125 'action' =>
$title->getLocalURL(
'action=delete' ),
2126 'id' =>
'deleteconfirm',
2128 $form->appendContent(
2130 new OOUI\HtmlSnippet(
2131 Html::hidden(
'wpEditToken', $user->
getEditToken( [
'delete',
$title->getPrefixedText() ] ) )
2136 new OOUI\PanelLayout( [
2137 'classes' => [
'deletepage-wrapper' ],
2138 'expanded' =>
false,
2147 if ( $suppressAllowed ) {
2148 $link .= $this->linkRenderer->makeKnownLink(
2149 $ctx->msg(
'deletereason-dropdown-suppress' )->inContentLanguage()->getTitle(),
2150 $ctx->msg(
'delete-edit-reasonlist-suppress' )->text(),
2152 [
'action' =>
'edit' ]
2154 $link .= $ctx->msg(
'pipe-separator' )->escaped();
2156 $link .= $this->linkRenderer->makeKnownLink(
2157 $ctx->msg(
'deletereason-dropdown' )->inContentLanguage()->getTitle(),
2158 $ctx->msg(
'delete-edit-reasonlist' )->text(),
2160 [
'action' =>
'edit' ]
2162 $outputPage->
addHTML(
'<p class="mw-delete-editreasons">' . $link .
'</p>' );
2165 $deleteLogPage =
new LogPage(
'delete' );
2166 $outputPage->
addHTML( Xml::element(
'h2',
null, $deleteLogPage->getName()->text() ) );
2167 LogEventsList::showLogExtract( $outputPage,
'delete',
$title );
2178 public function doDelete( $reason, $suppress =
false, $immediate =
false ) {
2183 $status = $this->mPage->doDeleteArticleReal(
2184 $reason, $user, $suppress,
null, $error,
2185 null, [],
'delete', $immediate
2188 if ( $status->isOK() ) {
2189 $deleted = $this->
getTitle()->getPrefixedText();
2194 if ( $status->isGood() ) {
2195 $loglink =
'[[Special:Log/delete|' .
wfMessage(
'deletionlog' )->text() .
']]';
2197 $this->getHookRunner()->onArticleDeleteAfterSuccess( $this->
getTitle(), $outputPage );
2206 $this->
getTitle()->getPrefixedText() )
2209 if ( $error ==
'' ) {
2211 'error mw-error-cannotdelete',
2212 $status->getWikiText(
false,
false, $context->getLanguage() )
2214 $deleteLogPage =
new LogPage(
'delete' );
2215 $outputPage->
addHTML( Xml::element(
'h2',
null, $deleteLogPage->getName()->text() ) );
2217 LogEventsList::showLogExtract(
2223 $outputPage->
addHTML( $error );
2238 static $called =
false;
2241 wfDebug(
"Article::tryFileCache(): called twice!?" );
2246 if ( $this->isFileCacheable() ) {
2248 if (
$cache->isCacheGood( $this->mPage->getTouched() ) ) {
2249 wfDebug(
"Article::tryFileCache(): about to load file" );
2253 wfDebug(
"Article::tryFileCache(): starting buffer" );
2254 ob_start( [ &
$cache,
'saveToFileCache' ] );
2257 wfDebug(
"Article::tryFileCache(): not cacheable" );
2272 $cacheable = $this->mPage->getId()
2273 && !$this->mRedirectedFrom && !$this->
getTitle()->isRedirect();
2276 $cacheable = $this->getHookRunner()->onIsFileCacheable( $this );
2297 if ( $user ===
null ) {
2298 $parserOptions = $this->getParserOptions();
2300 $parserOptions = $this->mPage->makeParserOptions( $user );
2303 return $this->mPage->getParserOutput( $parserOptions, $oldid );
2311 return $this->mPage->makeParserOptions( $this->
getContext() );
2321 $this->mContext = $context;
2332 return $this->mContext;
2334 wfDebug( __METHOD__ .
" called and \$mContext is null. " .
2335 "Return RequestContext::getMain(); for sanity" );
2336 return RequestContext::getMain();
2350 wfDeprecatedMsg(
"Accessing Article::\$$fname is deprecated since MediaWiki 1.35",
2353 if ( $fname ===
'mRevision' ) {
2354 $record = $this->fetchRevisionRecord();
2355 return $record ?
new Revision( $record ) :
null;
2358 if ( property_exists( $this->mPage, $fname ) ) {
2359 return $this->mPage->$fname;
2361 trigger_error(
'Inaccessible property via __get(): ' . $fname, E_USER_NOTICE );
2373 public function __set( $fname, $fvalue ) {
2374 wfDeprecatedMsg(
"Setting Article::\$$fname is deprecated since MediaWiki 1.35",
2377 if ( $fname ===
'mRevision' ) {
2378 $this->mRevisionRecord = $fvalue ?
2379 $fvalue->getRevisionRecord() :
2384 if ( property_exists( $this->mPage, $fname ) ) {
2385 $this->mPage->$fname = $fvalue;
2387 } elseif ( !in_array( $fname, [
'mContext',
'mPage' ] ) ) {
2388 $this->mPage->$fname = $fvalue;
2390 trigger_error(
'Inaccessible property via __set(): ' . $fname, E_USER_NOTICE );
2400 return $this->mPage->getActionOverrides();
2410 return $this->mPage->getTimestamp();
$wgArticleRobotPolicies
Robot policies per article.
$wgCdnMaxageStale
Cache timeout when delivering a stale ParserCache response due to PoolCounter contention.
$wgDefaultRobotPolicy
Default robot policy.
$wgSend404Code
Some web hosts attempt to rewrite all responses with a 404 (not found) status code,...
$wgRedirectSources
If local interwikis are set up which allow redirects, set this regexp to restrict URLs which will be ...
$wgUseFilePatrol
Use file patrolling to check new files on Special:Newfiles.
$wgUseRCPatrol
Use RC Patrolling to check for vandalism (from recent changes and watchlists) New pages and new files...
$wgNamespaceRobotPolicies
Robot policies per namespaces.
$wgUseNPPatrol
Use new page patrolling to check new pages on Special:Newpages.
$wgDeleteRevisionsLimit
Optional to restrict deletion of pages with higher revision counts to users with the 'bigdelete' perm...
$wgUseFileCache
This will cache static pages for non-logged-in users to reduce database traffic on public sites.
wfDebug( $text, $dest='all', array $context=[])
Sends a line to the debug log if enabled or, optionally, to a comment in output.
wfReadOnly()
Check whether the wiki is in read-only mode.
wfGetDB( $db, $groups=[], $wiki=false)
Get a Database object.
wfDeprecatedMsg( $msg, $version=false, $component=false, $callerOffset=2)
Log a deprecation warning with arbitrary message text.
wfMessage( $key,... $params)
This is the function for getting translated interface messages.
wfEscapeWikiText( $text)
Escapes the given text so that it may be output using addWikiText() without any linking,...
wfDeprecated( $function, $version=false, $component=false, $callerOffset=2)
Logs a warning that $function is deprecated.
Class for viewing MediaWiki article and history.
getRevisionFetched()
Get the fetched Revision object depending on request parameters or null on failure.
getSubstituteContent()
Returns Content object to use when the page does not exist.
static newFromWikiPage(WikiPage $page, IContextSource $context)
Create an Article object of the appropriate class for the given page.
generateContentOutput(User $user, ParserOptions $parserOptions, int $oldid, OutputPage $outputPage, array $textOptions)
Determines the desired ParserOutput and passes it to $outputPage.
getContext()
Gets the context this Article is executed in.
getOldIDFromRequest()
Sets $this->mRedirectUrl to a correct URL if the query parameters are incorrect.
getParserOutput( $oldid=null, User $user=null)
#-
getRedirectedFrom()
Get the page this view was redirected from.
Title null $mRedirectedFrom
Title from which we were redirected here, if any.
bool $viewIsRenderAction
Whether render() was called.
RevisionRecord null $mRevisionRecord
Revision to be shown.
RevisionStore $revisionStore
view()
This is the default action of the index.php entry point: just view the page of the given title.
__construct(Title $title, $oldId=null)
getRobotPolicy( $action, ParserOutput $pOutput=null)
Get the robot policy to be used for the current view.
static purgePatrolFooterCache( $articleID)
Purge the cache used to check if it is worth showing the patrol footer For example,...
doDelete( $reason, $suppress=false, $immediate=false)
Perform a deletion and output success or failure messages.
ParserOutput null false $mParserOutput
The ParserOutput generated for viewing the page, initialized by view().
LinkRenderer $linkRenderer
getTitle()
Get the title object of the article.
getActionOverrides()
Call to WikiPage function for backwards compatibility.
adjustDisplayTitle(ParserOutput $pOutput)
Adjust title for pages with displaytitle, -{T|}- or language conversion.
showDeletedRevisionHeader()
If the revision requested for view is deleted, check permissions.
getParserOptions()
Get parser options suitable for rendering the primary article wikitext.
IContextSource null $mContext
The context this Article is executed in.
showViewError(string $errortext)
Show error text for errors generated in Article::view().
static getRedirectHeaderHtml(Language $lang, $target, $forceKnown=false)
Return the HTML for the top of a redirect page.
protect()
action=protect handler
isCurrent()
Returns true if the currently-referenced revision is the current edit to this page (and it exists).
showMissingArticle()
Show the error text for a missing article.
unprotect()
action=unprotect handler (alias)
confirmDelete( $reason)
Output deletion confirmation dialog.
getPage()
Get the WikiPage object of this instance.
addHelpLink( $to, $overrideBaseUrl=false)
Adds help link with an icon via page indicators.
string bool $mRedirectUrl
URL to redirect to or false if none.
static newFromID( $id)
Constructor from a page id.
int null $mOldId
The oldid of the article that was requested to be shown, 0 for the current revision.
static formatRobotPolicy( $policy)
Converts a String robot policy into an associative array, to allow merging of several policies using ...
fetchRevisionRecord()
Fetches the revision to work on.
viewRedirect( $target, $appendSubtitle=true, $forceKnown=false)
Return the HTML for the top of a redirect page.
showPatrolFooter()
If patrol is possible, output a patrol UI box.
setOldSubtitle( $oldid=0)
Generate the navigation links when browsing through an article revisions It shows the information as:...
showViewFooter()
Show the footer section of an ordinary page view.
Status null $fetchResult
represents the outcome of fetchRevisionRecord().
WikiPage $mPage
The WikiPage object of this instance.
setRedirectedFrom(Title $from)
Tell the page view functions that this view was redirected from another page on the wiki.
isFileCacheable( $mode=HTMLFileCache::MODE_NORMAL)
Check if the page can be cached.
doOutputMetaData(?ParserOutput $pOutput, OutputPage $outputPage)
tryFileCache()
checkLastModified returns true if it has taken care of all output to the client that is necessary for...
getRevIdFetched()
Use this to fetch the rev ID used on page views.
showNamespaceHeader()
Show a header specific to the namespace currently being viewed, like [[MediaWiki:Talkpagetext]].
getRevisionRedirectTarget(RevisionRecord $revision)
static newFromTitle( $title, IContextSource $context)
Create an Article object of the appropriate class for the given page.
doOutputFromRenderStatus(?RevisionRecord $rev, Status $renderStatus, OutputPage $outputPage, array $textOptions)
getContentObject()
Returns a Content object representing the pages effective display content, not necessarily the revisi...
showDiffPage()
Show a diff page according to current request variables.
render()
Handle action=render.
showRedirectedFromHeader()
If this request is a redirect view, send "redirected from" subtitle to the output.
doOutputFromParserCache(ParserOutput $pOutput, OutputPage $outputPage, array $textOptions)
setContext( $context)
Sets the context this Article is executed in.
Special handling for category description pages, showing pages, subcategories and file that belong to...
msg( $key,... $params)
Get a Message object with context set Parameters are the same as wfMessage()
const POST_EDIT_COOKIE_KEY_PREFIX
Prefix of key for cookie used to pass post-edit state.
Page view caching in the file system.
static useFileCache(IContextSource $context, $mode=self::MODE_NORMAL)
Check if pages can be cached for this request/user.
Class for viewing MediaWiki file description pages.
Internationalisation code See https://www.mediawiki.org/wiki/Special:MyLanguage/Localisation for more...
static getRevDeleteLink(Authority $performer, $rev, LinkTarget $title)
Get a revision-deletion link, or disabled link, or nothing, depending on user permissions & the setti...
static revComment( $rev, $local=false, $isPublic=false, $useParentheses=true)
Wrap and format the given revision's comment block, if the current user is allowed to view it.
static revUserTools( $rev, $isPublic=false, $useParentheses=true)
Generate a user tool link cluster if the current user is allowed to view it.
static makeExternalLink( $url, $text, $escape=true, $linktype='', $attribs=[], $title=null)
Make an external link.
Class to simplify the use of log pages.
Service for getting rendered output of a given page.
Wrapper allowing us to handle a system message as a Content object.
This is one of the Core classes and should be read at least once by any new developers.
disable()
Disable output completely, i.e.
addWikiMsg(... $args)
Add a wikitext-formatted message to the output.
allowClickjacking()
Turn off frame-breaking.
wrapWikiTextAsInterface( $wrapperClass, $text)
Convert wikitext in the user interface language to HTML and add it to the buffer with a <div class="$...
setArticleFlag( $newVal)
Set whether the displayed content is related to the source of the corresponding article on the wiki S...
setRobotPolicy( $policy)
Set the robot policy for the page: http://www.robotstxt.org/meta.html
setIndexPolicy( $policy)
Set the index policy for the page, but leave the follow policy un- touched.
setPageTitle( $name)
"Page title" means the contents of <h1>.
redirect( $url, $responsecode='302')
Redirect to $url rather than displaying the normal page.
setLastModified( $timestamp)
Override the last modified timestamp.
adaptCdnTTL( $mtime, $minTTL=0, $maxTTL=0)
Get TTL in [$minTTL,$maxTTL] and pass it to lowerCdnMaxage()
wrapWikiMsg( $wrap,... $msgSpecs)
This function takes a number of message/argument specifications, wraps them in some overall structure...
setCdnMaxage( $maxage)
Set the value of the "s-maxage" part of the "Cache-control" HTTP header.
addParserOutput(ParserOutput $parserOutput, $poOptions=[])
Add everything from a ParserOutput object.
preventClickjacking( $enable=true)
Set a flag which will cause an X-Frame-Options header appropriate for edit pages to be sent.
setFollowPolicy( $policy)
Set the follow policy for the page, but leave the index policy un- touched.
isPrintable()
Return whether the page is "printable".
setRedirectedFrom( $t)
Set $mRedirectedFrom, the Title of the page which redirected us to the current page.
addModuleStyles( $modules)
Load the styles of one or more style-only ResourceLoader modules on this page.
setHTMLTitle( $name)
"HTML title" means the contents of "<title>".
returnToMain( $unused=null, $returnto=null, $returntoquery=null)
Add a "return to" link pointing to a specified title, or the title indicated in the request,...
addSubtitle( $str)
Add $str to the subtitle.
setRevisionId( $revid)
Set the revision ID which will be seen by the wiki text parser for things such as embedded {{REVISION...
enableOOUI()
Add ResourceLoader module styles for OOUI and set up the PHP implementation of it for use with MediaW...
clearHTML()
Clear the body HTML.
enableClientCache( $state)
Use enableClientCache(false) to force it to send nocache headers.
addHTML( $text)
Append $text to the body HTML.
addJsConfigVars( $keys, $value=null)
Add one or more variables to be set in mw.config in JavaScript.
setCanonicalUrl( $url)
Set the URL to be used for the <link rel=canonical>>.
setRevisionTimestamp( $timestamp)
Set the timestamp of the revision which will be displayed.
addWikiTextAsContent( $text, $linestart=true, Title $title=null)
Convert wikitext in the page content language to HTML and add it to the buffer.
addBacklinkSubtitle(Title $title, $query=[])
Add a subtitle containing a backlink to a page.
prependHTML( $text)
Prepend $text to the body HTML.
addModules( $modules)
Load one or more ResourceLoader modules on this page.
addWikiTextAsInterface( $text, $linestart=true, Title $title=null)
Convert wikitext in the user interface language to HTML and add it to the buffer.
Used to show archived pages and eventually restore them.
Set options of the Parser.
setIsPrintable( $x)
Parsing the printable version of the page?
Show an error when a user tries to do something they do not have the necessary permissions for.
Show an error when the wiki is locked/read-only and the user tries to do something that requires writ...
hasMessage( $message)
Returns true if the specified message is present as a warning or error.
isOK()
Returns whether the operation completed.
Generic operation result class Has warning/error list, boolean status and arbitrary value.
getWikiText( $shortContext=false, $longContext=false, $lang=null)
Get the error list as a wikitext formatted list.
Represents a title within MediaWiki.
The User object encapsulates all of the user-specific settings (user_id, name, rights,...
getName()
Get the user name, or the IP of an anonymous user.
static newFromName( $name, $validate='valid')
getEditToken( $salt='', $request=null)
Initialize (if necessary) and return a session token value which can be used in edit forms to show th...
isRegistered()
Get whether the user is registered.
getOption( $oname, $defaultOverride=null, $ignoreHidden=false)
Get the user's current setting for a given option.
isHidden()
Check if user account is hidden.
isWatched( $title, $checkRights=self::CHECK_USER_RIGHTS)
Check the watched status of an article.
static isIP( $name)
Does the string match an anonymous IP address?
matchEditToken( $val, $salt='', $request=null, $maxage=null)
Check given value against the token value stored in the session.
isBlockedFrom( $title, $fromReplica=false)
Check if user is blocked from editing a particular article.
static doWatchOrUnwatch( $watch, Title $title, Authority $performer, string $expiry=null)
Watch or unwatch a page.
Class representing a MediaWiki article and history.
getTitle()
Get the title object of the article.
Interface for objects which can provide a MediaWiki context on request.
Interface for type hinting (accepts WikiPage, Article, ImagePage, CategoryPage)
if(!isset( $args[0])) $lang