2019-11-28 14:39:58 +00:00
|
|
|
<?php
|
|
|
|
|
|
|
|
namespace Cite;
|
|
|
|
|
|
|
|
use Html;
|
|
|
|
use MediaWiki\MediaWikiServices;
|
|
|
|
use Parser;
|
|
|
|
|
2019-11-29 14:00:39 +00:00
|
|
|
/**
|
|
|
|
* @license GPL-2.0-or-later
|
|
|
|
*/
|
2019-11-28 14:39:58 +00:00
|
|
|
class FootnoteBodyFormatter {
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The backlinks, in order, to pass as $3 to
|
|
|
|
* 'cite_references_link_many_format', defined in
|
|
|
|
* 'cite_references_link_many_format_backlink_labels
|
|
|
|
*
|
|
|
|
* @var string[]
|
|
|
|
*/
|
|
|
|
private $backlinkLabels;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @var CiteErrorReporter
|
|
|
|
*/
|
|
|
|
private $errorReporter;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @var CiteKeyFormatter
|
|
|
|
*/
|
|
|
|
private $citeKeyFormatter;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @var Parser
|
|
|
|
*/
|
|
|
|
private $parser;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param Parser $parser
|
|
|
|
* @param CiteErrorReporter $errorReporter
|
|
|
|
* @param CiteKeyFormatter $citeKeyFormatter
|
|
|
|
*/
|
|
|
|
public function __construct(
|
|
|
|
Parser $parser,
|
|
|
|
CiteErrorReporter $errorReporter,
|
|
|
|
CiteKeyFormatter $citeKeyFormatter
|
|
|
|
) {
|
|
|
|
$this->errorReporter = $errorReporter;
|
|
|
|
$this->citeKeyFormatter = $citeKeyFormatter;
|
|
|
|
$this->parser = $parser;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param array $groupRefs
|
|
|
|
* @param bool $responsive
|
|
|
|
* @param bool $isSectionPreview
|
|
|
|
* @return string
|
|
|
|
*/
|
2019-11-28 10:15:19 +00:00
|
|
|
public function referencesFormat(
|
|
|
|
array $groupRefs,
|
|
|
|
bool $responsive,
|
|
|
|
bool $isSectionPreview
|
|
|
|
) : string {
|
2019-11-28 14:39:58 +00:00
|
|
|
if ( !$groupRefs ) {
|
|
|
|
return '';
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add new lines between the list items (ref entries) to avoid confusing tidy (T15073).
|
|
|
|
// Note: This builds a string of wikitext, not html.
|
|
|
|
$parserInput = "\n";
|
|
|
|
$indented = false;
|
2019-11-29 11:50:07 +00:00
|
|
|
// After sorting the list, we can assume that references are in the same order as their
|
|
|
|
// numbering. Subreferences will come immediately after their parent.
|
|
|
|
uasort(
|
|
|
|
$groupRefs,
|
|
|
|
function ( array $a, array $b ) : int {
|
|
|
|
return ( $a['number'] ?? '' ) <=> ( $b['number'] ?? '' ) ?:
|
|
|
|
( $a['extendsIndex'] ?? '0' ) <=> ( $b['extendsIndex'] ?? '0' );
|
|
|
|
}
|
|
|
|
);
|
2019-11-28 14:39:58 +00:00
|
|
|
foreach ( $groupRefs as $key => $value ) {
|
|
|
|
if ( !$indented && isset( $value['extends'] ) ) {
|
2019-11-29 16:27:16 +00:00
|
|
|
// The nested <ol> must be inside the parent's <li>
|
|
|
|
if ( preg_match( '#</li>\s*$#D', $parserInput, $matches, PREG_OFFSET_CAPTURE ) ) {
|
|
|
|
$parserInput = substr( $parserInput, 0, $matches[0][1] );
|
|
|
|
}
|
2019-11-28 14:39:58 +00:00
|
|
|
$parserInput .= Html::openElement( 'ol', [ 'class' => 'mw-extended-references' ] );
|
2019-11-29 16:27:16 +00:00
|
|
|
$indented = $matches[0][0] ?? true;
|
2019-11-28 14:39:58 +00:00
|
|
|
} elseif ( $indented && !isset( $value['extends'] ) ) {
|
2019-11-29 16:27:16 +00:00
|
|
|
$parserInput .= $this->closeIndention( $indented );
|
2019-11-28 14:39:58 +00:00
|
|
|
$indented = false;
|
|
|
|
}
|
2019-11-29 16:27:16 +00:00
|
|
|
$parserInput .= $this->referencesFormatEntry( $key, $value, $isSectionPreview ) . "\n";
|
2019-11-28 14:39:58 +00:00
|
|
|
}
|
2019-11-29 16:27:16 +00:00
|
|
|
$parserInput .= $this->closeIndention( $indented );
|
2019-11-28 14:39:58 +00:00
|
|
|
$parserInput = Html::rawElement( 'ol', [ 'class' => [ 'references' ] ], $parserInput );
|
|
|
|
// Live hack: parse() adds two newlines on WM, can't reproduce it locally -ævar
|
|
|
|
$ret = rtrim( $this->parser->recursiveTagParse( $parserInput ), "\n" );
|
|
|
|
|
|
|
|
if ( $responsive ) {
|
|
|
|
// Use a DIV wrap because column-count on a list directly is broken in Chrome.
|
|
|
|
// See https://bugs.chromium.org/p/chromium/issues/detail?id=498730.
|
|
|
|
$wrapClasses = [ 'mw-references-wrap' ];
|
|
|
|
if ( count( $groupRefs ) > 10 ) {
|
|
|
|
$wrapClasses[] = 'mw-references-columns';
|
|
|
|
}
|
|
|
|
return Html::rawElement( 'div', [ 'class' => $wrapClasses ], $ret );
|
|
|
|
} else {
|
|
|
|
return $ret;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-29 16:27:16 +00:00
|
|
|
/**
|
|
|
|
* @param string|bool $closingLi
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
private function closeIndention( $closingLi ) : string {
|
|
|
|
if ( !$closingLi ) {
|
|
|
|
return '';
|
|
|
|
}
|
|
|
|
|
|
|
|
return Html::closeElement( 'ol' ) . ( is_string( $closingLi ) ? $closingLi : '' );
|
|
|
|
}
|
|
|
|
|
2019-11-28 14:39:58 +00:00
|
|
|
/**
|
|
|
|
* Format a single entry for the referencesFormat() function
|
|
|
|
*
|
|
|
|
* @param string|int $key The key of the reference
|
|
|
|
* @param array $val A single reference as documented at {@see ReferenceStack::$refs}
|
|
|
|
* @param bool $isSectionPreview
|
|
|
|
* @return string Wikitext, wrapped in a single <li> element
|
|
|
|
*/
|
2019-11-28 10:15:19 +00:00
|
|
|
private function referencesFormatEntry( $key, array $val, bool $isSectionPreview ) : string {
|
2019-11-29 11:50:07 +00:00
|
|
|
$text = $this->referenceText( $key, ( $val['text'] ?? null ), $isSectionPreview );
|
2019-11-28 14:39:58 +00:00
|
|
|
$error = '';
|
|
|
|
$extraAttributes = '';
|
|
|
|
|
2019-11-29 11:50:07 +00:00
|
|
|
// TODO: Show an error if isset( $val['__placeholder__'] ), this is caused by extends
|
|
|
|
// with a missing parent.
|
|
|
|
|
2019-11-28 14:39:58 +00:00
|
|
|
if ( isset( $val['dir'] ) ) {
|
|
|
|
$dir = strtolower( $val['dir'] );
|
|
|
|
if ( in_array( $dir, [ 'ltr', 'rtl' ] ) ) {
|
|
|
|
$extraAttributes = Html::expandAttributes( [ 'class' => 'mw-cite-dir-' . $dir ] );
|
|
|
|
} else {
|
|
|
|
$error .= $this->errorReporter->plain( 'cite_error_ref_invalid_dir', $val['dir'] ) . "\n";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Fallback for a broken, and therefore unprocessed follow="…". Note this returns a <p>, not
|
|
|
|
// an <li> as expected!
|
|
|
|
if ( isset( $val['follow'] ) ) {
|
|
|
|
return wfMessage(
|
|
|
|
'cite_references_no_link',
|
2019-11-29 14:00:39 +00:00
|
|
|
$this->citeKeyFormatter->getReferencesKey( $val['follow'] ),
|
2019-11-28 14:39:58 +00:00
|
|
|
$text
|
|
|
|
)->inContentLanguage()->plain();
|
|
|
|
}
|
|
|
|
|
|
|
|
// This counts the number of reuses. 0 means the reference appears only 1 time.
|
|
|
|
if ( isset( $val['count'] ) && $val['count'] < 1 ) {
|
|
|
|
// Anonymous, auto-numbered references can't be reused and get marked with a -1.
|
|
|
|
if ( $val['count'] < 0 ) {
|
|
|
|
$id = $val['key'];
|
|
|
|
$backlinkId = $this->citeKeyFormatter->refKey( $val['key'] );
|
|
|
|
} else {
|
|
|
|
$id = $key . '-' . $val['key'];
|
|
|
|
$backlinkId = $this->citeKeyFormatter->refKey( $key, $val['key'] . '-' . $val['count'] );
|
|
|
|
}
|
|
|
|
return wfMessage(
|
|
|
|
'cite_references_link_one',
|
2019-11-29 14:00:39 +00:00
|
|
|
$this->citeKeyFormatter->getReferencesKey( $id ),
|
|
|
|
$backlinkId,
|
2019-11-28 14:39:58 +00:00
|
|
|
$text . $error,
|
|
|
|
$extraAttributes
|
|
|
|
)->inContentLanguage()->plain();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Named references with >1 occurrences
|
|
|
|
$backlinks = [];
|
|
|
|
// There is no count in case of a section preview
|
|
|
|
for ( $i = 0; $i <= ( $val['count'] ?? -1 ); $i++ ) {
|
|
|
|
$backlinks[] = wfMessage(
|
|
|
|
'cite_references_link_many_format',
|
2019-11-29 14:00:39 +00:00
|
|
|
$this->citeKeyFormatter->refKey( $key, $val['key'] . '-' . $i ),
|
2019-11-28 14:39:58 +00:00
|
|
|
$this->referencesFormatEntryNumericBacklinkLabel(
|
2019-11-29 23:10:11 +00:00
|
|
|
$val['number'] .
|
|
|
|
( isset( $val['extendsIndex'] ) ? '.' . $val['extendsIndex'] : '' ),
|
2019-11-28 14:39:58 +00:00
|
|
|
$i,
|
|
|
|
$val['count']
|
|
|
|
),
|
|
|
|
$this->referencesFormatEntryAlternateBacklinkLabel( $i )
|
|
|
|
)->inContentLanguage()->plain();
|
|
|
|
}
|
|
|
|
return wfMessage(
|
|
|
|
'cite_references_link_many',
|
2019-11-29 14:00:39 +00:00
|
|
|
$this->citeKeyFormatter->getReferencesKey( $key . '-' . ( $val['key'] ?? '' ) ),
|
2019-11-28 14:39:58 +00:00
|
|
|
$this->listToText( $backlinks ),
|
|
|
|
$text . $error,
|
|
|
|
$extraAttributes
|
|
|
|
)->inContentLanguage()->plain();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns formatted reference text
|
|
|
|
* @param string|int $key
|
|
|
|
* @param ?string $text
|
|
|
|
* @param bool $isSectionPreview
|
|
|
|
* @return string
|
|
|
|
*/
|
2019-11-28 10:15:19 +00:00
|
|
|
private function referenceText( $key, ?string $text, bool $isSectionPreview ) : string {
|
2019-11-26 09:31:41 +00:00
|
|
|
if ( $text === null ) {
|
2019-11-28 14:39:58 +00:00
|
|
|
if ( $isSectionPreview ) {
|
|
|
|
return $this->errorReporter->plain( 'cite_warning_sectionpreview_no_text', $key );
|
|
|
|
}
|
|
|
|
return $this->errorReporter->plain( 'cite_error_references_no_text', $key );
|
|
|
|
}
|
|
|
|
return '<span class="reference-text">' . rtrim( $text, "\n" ) . "</span>\n";
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Generate a numeric backlink given a base number and an
|
|
|
|
* offset, e.g. $base = 1, $offset = 2; = 1.2
|
|
|
|
* Since bug #5525, it correctly does 1.9 -> 1.10 as well as 1.099 -> 1.100
|
|
|
|
*
|
2019-11-28 10:15:19 +00:00
|
|
|
* @param int|string $base
|
2019-11-28 14:39:58 +00:00
|
|
|
* @param int $offset
|
|
|
|
* @param int $max Maximum value expected.
|
|
|
|
* @return string
|
|
|
|
*/
|
2019-11-28 10:15:19 +00:00
|
|
|
private function referencesFormatEntryNumericBacklinkLabel(
|
|
|
|
$base,
|
|
|
|
int $offset,
|
|
|
|
int $max
|
|
|
|
) : string {
|
2019-11-28 14:39:58 +00:00
|
|
|
$scope = strlen( $max );
|
|
|
|
$ret = MediaWikiServices::getInstance()->getContentLanguage()->formatNum(
|
2019-11-29 09:42:46 +00:00
|
|
|
$base . '.' . sprintf( "%0{$scope}s", $offset )
|
2019-11-28 14:39:58 +00:00
|
|
|
);
|
|
|
|
return $ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Generate a custom format backlink given an offset, e.g.
|
|
|
|
* $offset = 2; = c if $this->mBacklinkLabels = [ 'a',
|
|
|
|
* 'b', 'c', ...]. Return an error if the offset > the # of
|
|
|
|
* array items
|
|
|
|
*
|
|
|
|
* @param int $offset
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*/
|
2019-11-28 10:15:19 +00:00
|
|
|
private function referencesFormatEntryAlternateBacklinkLabel( int $offset ) : string {
|
2019-11-28 14:39:58 +00:00
|
|
|
if ( !isset( $this->backlinkLabels ) ) {
|
|
|
|
$this->genBacklinkLabels();
|
|
|
|
}
|
|
|
|
return $this->backlinkLabels[$offset]
|
|
|
|
?? $this->errorReporter->plain( 'cite_error_references_no_backlink_label', null );
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Generate the labels to pass to the
|
|
|
|
* 'cite_references_link_many_format' message, the format is an
|
|
|
|
* arbitrary number of tokens separated by whitespace.
|
|
|
|
*/
|
|
|
|
private function genBacklinkLabels() {
|
|
|
|
$text = wfMessage( 'cite_references_link_many_format_backlink_labels' )
|
|
|
|
->inContentLanguage()->plain();
|
|
|
|
$this->backlinkLabels = preg_split( '/\s+/', $text );
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* This does approximately the same thing as
|
|
|
|
* Language::listToText() but due to this being used for a
|
|
|
|
* slightly different purpose (people might not want , as the
|
|
|
|
* first separator and not 'and' as the second, and this has to
|
|
|
|
* use messages from the content language) I'm rolling my own.
|
|
|
|
*
|
|
|
|
* @param string[] $arr The array to format
|
|
|
|
* @return string
|
|
|
|
*/
|
2019-11-28 10:15:19 +00:00
|
|
|
private function listToText( array $arr ) : string {
|
2019-11-28 14:39:58 +00:00
|
|
|
$lastElement = array_pop( $arr );
|
|
|
|
|
|
|
|
if ( $arr === [] ) {
|
|
|
|
return (string)$lastElement;
|
|
|
|
}
|
|
|
|
|
|
|
|
$sep = wfMessage( 'cite_references_link_many_sep' )->inContentLanguage()->plain();
|
|
|
|
$and = wfMessage( 'cite_references_link_many_and' )->inContentLanguage()->plain();
|
|
|
|
return implode( $sep, $arr ) . $and . $lastElement;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|