We always implicitly converted a string argument to an array anyway; just ask the caller to do this instead so that we can have a simpler and more straight-forward method signature which matches the plural form of the method name. Part of the ParserOutput API cleanup / Parsoid unification discussed in T287216. In a number of places we also rename $out to $parserOutput, to make it easier for codesearch (and human readers) to distinguish between ParserOutput and OutputPage methods. Code search: https://codesearch.wmcloud.org/deployed/?q=p%28arser%29%3F%28Out%7Cout%29%28put%29%3F-%3EaddModule%28Style%29%3Fs%5C%28&i=nope&files=&excludeFiles=&repos= https://codesearch.wmcloud.org/deployed/?q=arser-%3EgetOutput%5C%28%5C%29-%3EaddModule%28Style%29%3Fs%5C%28&i=nope&files=&excludeFiles=&repos= Bug: T296123 Depends-On: Iedea960bd450474966eb60ff8dfbf31c127025b6 Depends-On: I7900c5746a9ea75ce4918ffd97d45128038ab3f0 Depends-On: If29dc1d696b3a4c249fa9b150cedf2a502796ea1 Depends-On: I8f1bc7233a00382123a9b1b0bb549bd4dbc4a095 Depends-On: I52dda72aee6c7784a8961488c437863e31affc17 Depends-On: Ia1dcc86cb64f6aa39c68403d37bd76f970e55b97 Depends-On: Ib89ef9c900514d50173e13ab49d17c312b729900 Depends-On: If54244a0278d532c8553029c487c916068e1300f Depends-On: I8d9b34f5d1ed5b1534bb29f5cd6edcdc086b71ca Depends-On: I068f9f8e85e88a5c457d40e6a92f09b7eddd6b81 Depends-On: Iced2fc7b4f3cda5296532f22d233875bbc2f5d1b Depends-On: If14866f76703aa62d33e197bb18a5eacde7a55c0 Depends-On: I9b7fe5acee73c3a378153c0820b46816164ebf21 Depends-On: I95858c08bce0d90709ac7771a910f73d78cc8be4 Depends-On: If9a70e8f8545d4f9ee3b605ad849dbd7de742fc1 Depends-On: I982c81e1ad73b58a90649648e19501cf9172d493 Depends-On: I53a8fd22b22c93bba703233b62377c49ba9f5562 Depends-On: Ic532bca4348b17882716fcb2ca8656a04766c095 Depends-On: If34330acf97d2c4e357b693b086264a718738fb1 Change-Id: Ie4d6bbe258cc483d5693f7a27dbccb60d8f37e2c
315 lines
8.8 KiB
PHP
315 lines
8.8 KiB
PHP
<?php
|
|
/**
|
|
* Content handler for wiki text pages.
|
|
*
|
|
* 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
|
|
*
|
|
* @since 1.21
|
|
*
|
|
* @file
|
|
* @ingroup Content
|
|
*/
|
|
|
|
use MediaWiki\Content\Renderer\ContentParseParams;
|
|
use MediaWiki\Content\Transform\PreloadTransformParams;
|
|
use MediaWiki\Content\Transform\PreSaveTransformParams;
|
|
use MediaWiki\Languages\LanguageNameUtils;
|
|
use MediaWiki\MediaWikiServices;
|
|
use MediaWiki\Parser\ParserOutputFlags;
|
|
|
|
/**
|
|
* Content handler for wiki text pages.
|
|
*
|
|
* @ingroup Content
|
|
*/
|
|
class WikitextContentHandler extends TextContentHandler {
|
|
|
|
public function __construct( $modelId = CONTENT_MODEL_WIKITEXT ) {
|
|
parent::__construct( $modelId, [ CONTENT_FORMAT_WIKITEXT ] );
|
|
}
|
|
|
|
protected function getContentClass() {
|
|
return WikitextContent::class;
|
|
}
|
|
|
|
/**
|
|
* Returns a WikitextContent object representing a redirect to the given destination page.
|
|
*
|
|
* @param Title $destination The page to redirect to.
|
|
* @param string $text Text to include in the redirect, if possible.
|
|
*
|
|
* @return Content
|
|
*
|
|
* @see ContentHandler::makeRedirectContent
|
|
*/
|
|
public function makeRedirectContent( Title $destination, $text = '' ) {
|
|
$optionalColon = '';
|
|
|
|
$services = MediaWikiServices::getInstance();
|
|
if ( $destination->getNamespace() === NS_CATEGORY ) {
|
|
$optionalColon = ':';
|
|
} else {
|
|
$iw = $destination->getInterwiki();
|
|
if ( $iw && $services
|
|
->getLanguageNameUtils()
|
|
->getLanguageName( $iw,
|
|
LanguageNameUtils::AUTONYMS,
|
|
LanguageNameUtils::DEFINED )
|
|
) {
|
|
$optionalColon = ':';
|
|
}
|
|
}
|
|
|
|
$mwRedir = $services->getMagicWordFactory()->get( 'redirect' );
|
|
$redirectText = $mwRedir->getSynonym( 0 ) .
|
|
' [[' . $optionalColon . $destination->getFullText() . ']]';
|
|
|
|
if ( $text != '' ) {
|
|
$redirectText .= "\n" . $text;
|
|
}
|
|
|
|
$class = $this->getContentClass();
|
|
return new $class( $redirectText );
|
|
}
|
|
|
|
/**
|
|
* Returns true because wikitext supports redirects.
|
|
*
|
|
* @return bool Always true.
|
|
*
|
|
* @see ContentHandler::supportsRedirects
|
|
*/
|
|
public function supportsRedirects() {
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* Returns true because wikitext supports sections.
|
|
*
|
|
* @return bool Always true.
|
|
*
|
|
* @see ContentHandler::supportsSections
|
|
*/
|
|
public function supportsSections() {
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* Returns true, because wikitext supports caching using the
|
|
* ParserCache mechanism.
|
|
*
|
|
* @since 1.21
|
|
*
|
|
* @return bool Always true.
|
|
*
|
|
* @see ContentHandler::isParserCacheSupported
|
|
*/
|
|
public function isParserCacheSupported() {
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* @return FileContentHandler
|
|
*/
|
|
protected function getFileHandler() {
|
|
return new FileContentHandler();
|
|
}
|
|
|
|
public function getFieldsForSearchIndex( SearchEngine $engine ) {
|
|
$fields = parent::getFieldsForSearchIndex( $engine );
|
|
|
|
$fields['heading'] =
|
|
$engine->makeSearchFieldMapping( 'heading', SearchIndexField::INDEX_TYPE_TEXT );
|
|
$fields['heading']->setFlag( SearchIndexField::FLAG_SCORING );
|
|
|
|
$fields['auxiliary_text'] =
|
|
$engine->makeSearchFieldMapping( 'auxiliary_text', SearchIndexField::INDEX_TYPE_TEXT );
|
|
|
|
$fields['opening_text'] =
|
|
$engine->makeSearchFieldMapping( 'opening_text', SearchIndexField::INDEX_TYPE_TEXT );
|
|
$fields['opening_text']->setFlag(
|
|
SearchIndexField::FLAG_SCORING | SearchIndexField::FLAG_NO_HIGHLIGHT
|
|
);
|
|
// Until we have full first-class content handler for files, we invoke it explicitly here
|
|
$fields = array_merge( $fields, $this->getFileHandler()->getFieldsForSearchIndex( $engine ) );
|
|
|
|
return $fields;
|
|
}
|
|
|
|
public function getDataForSearchIndex(
|
|
WikiPage $page,
|
|
ParserOutput $parserOutput,
|
|
SearchEngine $engine
|
|
) {
|
|
$fields = parent::getDataForSearchIndex( $page, $parserOutput, $engine );
|
|
|
|
$structure = new WikiTextStructure( $parserOutput );
|
|
$fields['heading'] = $structure->headings();
|
|
// text fields
|
|
$fields['opening_text'] = $structure->getOpeningText();
|
|
$fields['text'] = $structure->getMainText(); // overwrites one from ContentHandler
|
|
$fields['auxiliary_text'] = $structure->getAuxiliaryText();
|
|
$fields['defaultsort'] = $structure->getDefaultSort();
|
|
|
|
// Until we have full first-class content handler for files, we invoke it explicitly here
|
|
if ( $page->getTitle()->getNamespace() === NS_FILE ) {
|
|
$fields = array_merge( $fields,
|
|
$this->getFileHandler()->getDataForSearchIndex( $page, $parserOutput, $engine ) );
|
|
}
|
|
return $fields;
|
|
}
|
|
|
|
/**
|
|
* Returns the content's text as-is.
|
|
*
|
|
* @param Content $content
|
|
* @param string|null $format The serialization format to check
|
|
*
|
|
* @return mixed
|
|
*/
|
|
public function serializeContent( Content $content, $format = null ) {
|
|
$this->checkFormat( $format );
|
|
|
|
// NOTE: MessageContent also uses CONTENT_MODEL_WIKITEXT, but it's not a TextContent!
|
|
// Perhaps MessageContent should use a separate ContentHandler instead.
|
|
if ( $content instanceof MessageContent ) {
|
|
return $content->getMessage()->plain();
|
|
}
|
|
|
|
return parent::serializeContent( $content, $format );
|
|
}
|
|
|
|
public function preSaveTransform(
|
|
Content $content,
|
|
PreSaveTransformParams $pstParams
|
|
): Content {
|
|
$shouldCallDeprecatedMethod = $this->shouldCallDeprecatedContentTransformMethod(
|
|
$content,
|
|
$pstParams
|
|
);
|
|
|
|
if ( $shouldCallDeprecatedMethod ) {
|
|
return $this->callDeprecatedContentPST(
|
|
$content,
|
|
$pstParams
|
|
);
|
|
}
|
|
|
|
'@phan-var WikitextContent $content';
|
|
|
|
$text = $content->getText();
|
|
|
|
$parser = MediaWikiServices::getInstance()->getParser();
|
|
$pst = $parser->preSaveTransform(
|
|
$text,
|
|
$pstParams->getPage(),
|
|
$pstParams->getUser(),
|
|
$pstParams->getParserOptions()
|
|
);
|
|
|
|
if ( $text === $pst ) {
|
|
return $content;
|
|
}
|
|
|
|
$contentClass = $this->getContentClass();
|
|
$ret = new $contentClass( $pst );
|
|
$ret->setPreSaveTransformFlags( $parser->getOutput()->getAllFlags() );
|
|
return $ret;
|
|
}
|
|
|
|
/**
|
|
* Returns a Content object with preload transformations applied (or this
|
|
* object if no transformations apply).
|
|
*
|
|
* @param Content $content
|
|
* @param PreloadTransformParams $pltParams
|
|
*
|
|
* @return Content
|
|
*/
|
|
public function preloadTransform(
|
|
Content $content,
|
|
PreloadTransformParams $pltParams
|
|
): Content {
|
|
$shouldCallDeprecatedMethod = $this->shouldCallDeprecatedContentTransformMethod(
|
|
$content,
|
|
$pltParams
|
|
);
|
|
|
|
if ( $shouldCallDeprecatedMethod ) {
|
|
return $this->callDeprecatedContentPLT(
|
|
$content,
|
|
$pltParams
|
|
);
|
|
}
|
|
|
|
'@phan-var WikitextContent $content';
|
|
|
|
$text = $content->getText();
|
|
$plt = MediaWikiServices::getInstance()->getParser()->getPreloadText(
|
|
$text,
|
|
$pltParams->getPage(),
|
|
$pltParams->getParserOptions(),
|
|
$pltParams->getParams()
|
|
);
|
|
|
|
$contentClass = $this->getContentClass();
|
|
return new $contentClass( $plt );
|
|
}
|
|
|
|
/**
|
|
* Returns a ParserOutput object resulting from parsing the content's text
|
|
* using the global Parser service.
|
|
*
|
|
* @since 1.38
|
|
* @param Content $content
|
|
* @param ContentParseParams $cpoParams
|
|
* @param ParserOutput &$parserOutput The output object to fill (reference).
|
|
*/
|
|
protected function fillParserOutput(
|
|
Content $content,
|
|
ContentParseParams $cpoParams,
|
|
ParserOutput &$parserOutput
|
|
) {
|
|
'@phan-var WikitextContent $content';
|
|
$services = MediaWikiServices::getInstance();
|
|
$title = $services->getTitleFactory()->castFromPageReference( $cpoParams->getPage() );
|
|
$parserOptions = $cpoParams->getParserOptions();
|
|
$revId = $cpoParams->getRevId();
|
|
|
|
list( $redir, $text ) = $content->getRedirectTargetAndText();
|
|
$parserOutput = $services->getParser()->getFreshParser()
|
|
->parse( $text, $title, $parserOptions, true, true, $revId );
|
|
|
|
// Add redirect indicator at the top
|
|
if ( $redir ) {
|
|
// Make sure to include the redirect link in pagelinks
|
|
$parserOutput->addLink( $redir );
|
|
if ( $cpoParams->getGenerateHtml() ) {
|
|
$redirTarget = $content->getRedirectTarget();
|
|
$parserOutput->setText(
|
|
Article::getRedirectHeaderHtml( $title->getPageLanguage(), $redirTarget, false ) .
|
|
$parserOutput->getRawText()
|
|
);
|
|
$parserOutput->addModuleStyles( [ 'mediawiki.action.view.redirectPage' ] );
|
|
}
|
|
}
|
|
|
|
// Pass along user-signature flag
|
|
if ( in_array( 'user-signature', $content->getPreSaveTransformFlags() ) ) {
|
|
$parserOutput->setOutputFlag( ParserOutputFlags::USER_SIGNATURE );
|
|
}
|
|
}
|
|
}
|