2019-05-14 18:59:33 +00:00
|
|
|
<?php
|
|
|
|
/**
|
2021-06-21 23:48:07 +00:00
|
|
|
* TabberNeue
|
|
|
|
* TabberNeue Hooks Class
|
2019-05-14 18:59:33 +00:00
|
|
|
*
|
2021-06-21 23:48:07 +00:00
|
|
|
* @package TabberNeue
|
2022-04-19 01:45:33 +00:00
|
|
|
* @author alistair3149, Eric Fortin, Alexia E. Smith, Ciencia Al Poder
|
2021-06-21 23:48:07 +00:00
|
|
|
* @license GPL-3.0-or-later
|
|
|
|
* @link https://www.mediawiki.org/wiki/Extension:TabberNeue
|
2021-06-21 23:56:28 +00:00
|
|
|
*/
|
2019-05-14 18:59:33 +00:00
|
|
|
|
2021-10-01 16:20:00 +00:00
|
|
|
declare( strict_types=1 );
|
|
|
|
|
2021-06-21 17:49:47 +00:00
|
|
|
namespace TabberNeue;
|
2019-05-14 18:59:33 +00:00
|
|
|
|
2022-04-19 01:45:33 +00:00
|
|
|
use Hooks;
|
|
|
|
use MediaWiki\MediaWikiServices;
|
2019-05-14 18:59:33 +00:00
|
|
|
use Parser;
|
|
|
|
use PPFrame;
|
2022-04-19 01:45:33 +00:00
|
|
|
use Title;
|
2019-05-14 18:59:33 +00:00
|
|
|
|
2021-06-21 17:49:47 +00:00
|
|
|
class TabberNeueHooks {
|
2019-05-14 18:59:33 +00:00
|
|
|
/**
|
|
|
|
* Sets up this extension's parser functions.
|
|
|
|
*
|
2021-06-22 19:23:39 +00:00
|
|
|
* @param Parser $parser Parser object passed as a reference.
|
2019-05-14 18:59:33 +00:00
|
|
|
*/
|
2021-06-22 19:23:39 +00:00
|
|
|
public static function onParserFirstCallInit( Parser $parser ) {
|
|
|
|
$parser->setHook( 'tabber', [ __CLASS__, 'renderTabber' ] );
|
2022-04-19 01:45:33 +00:00
|
|
|
$parser->setHook( 'tabbertransclude', [ __CLASS__, 'renderTabberTransclude' ] );
|
2019-05-14 18:59:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Renders the necessary HTML for a <tabber> tag.
|
|
|
|
*
|
2021-06-21 23:56:28 +00:00
|
|
|
* @param string $input The input URL between the beginning and ending tags.
|
|
|
|
* @param array $args Array of attribute arguments on that beginning tag.
|
2021-06-22 00:01:05 +00:00
|
|
|
* @param Parser $parser Mediawiki Parser Object
|
|
|
|
* @param PPFrame $frame Mediawiki PPFrame Object
|
2019-05-14 18:59:33 +00:00
|
|
|
*
|
2021-06-21 23:56:28 +00:00
|
|
|
* @return string HTML
|
2019-05-14 18:59:33 +00:00
|
|
|
*/
|
2021-06-21 23:56:28 +00:00
|
|
|
public static function renderTabber( $input, array $args, Parser $parser, PPFrame $frame ) {
|
2022-04-19 00:24:21 +00:00
|
|
|
$parser->getOutput()->addModules( [ 'ext.tabberNeue' ] );
|
2021-06-21 23:56:28 +00:00
|
|
|
$arr = explode( "|-|", $input );
|
2019-05-14 18:59:33 +00:00
|
|
|
$htmlTabs = '';
|
2021-06-21 23:56:28 +00:00
|
|
|
foreach ( $arr as $tab ) {
|
|
|
|
$htmlTabs .= self::buildTab( $tab, $parser, $frame );
|
2019-05-14 18:59:33 +00:00
|
|
|
}
|
|
|
|
|
2021-10-23 04:47:49 +00:00
|
|
|
$html = '<div class="tabber">' .
|
2021-06-21 17:49:47 +00:00
|
|
|
'<section class="tabber__section">' . $htmlTabs . "</section></div>";
|
2019-05-14 18:59:33 +00:00
|
|
|
|
2021-06-21 17:49:47 +00:00
|
|
|
return $html;
|
2019-05-14 18:59:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Build individual tab.
|
|
|
|
*
|
2021-06-21 23:56:28 +00:00
|
|
|
* @param string $tab Tab information
|
2021-06-22 00:01:05 +00:00
|
|
|
* @param Parser $parser Mediawiki Parser Object
|
|
|
|
* @param PPFrame $frame Mediawiki PPFrame Object
|
2019-05-14 18:59:33 +00:00
|
|
|
*
|
2021-06-21 23:56:28 +00:00
|
|
|
* @return string HTML
|
2019-05-14 18:59:33 +00:00
|
|
|
*/
|
2021-06-21 23:56:28 +00:00
|
|
|
private static function buildTab( $tab, Parser $parser, PPFrame $frame ) {
|
|
|
|
$tab = trim( $tab );
|
|
|
|
if ( empty( $tab ) ) {
|
2019-05-14 18:59:33 +00:00
|
|
|
return $tab;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Use array_pad to make sure at least 2 array values are always returned
|
2022-04-18 23:43:08 +00:00
|
|
|
list( $tabName, $tabBody ) = array_pad( array_map( 'trim', explode( '=', $tab, 2 ) ), 2, '' );
|
2019-05-14 18:59:33 +00:00
|
|
|
|
2022-02-02 04:04:24 +00:00
|
|
|
// Append __NOEDITSECTION__ to prevent section edit links from being added by the parser
|
|
|
|
// since section edit links do not work inside a tabber body
|
|
|
|
$tabBody = $parser->recursiveTagParseFully( '__NOEDITSECTION__' . $tabBody, $frame );
|
2019-05-14 18:59:33 +00:00
|
|
|
|
2021-06-21 23:56:28 +00:00
|
|
|
$tab = '<article class="tabber__panel" title="' . htmlspecialchars( $tabName ) .
|
2021-10-01 04:11:04 +00:00
|
|
|
'">' . $tabBody . '</article>';
|
2021-06-21 17:49:47 +00:00
|
|
|
|
2019-05-14 18:59:33 +00:00
|
|
|
return $tab;
|
|
|
|
}
|
2022-04-19 01:45:33 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Renders the necessary HTML for a <tabbertransclude> tag.
|
|
|
|
*
|
|
|
|
* @param string $input The input URL between the beginning and ending tags.
|
|
|
|
* @param array $args Array of attribute arguments on that beginning tag.
|
|
|
|
* @param Parser $parser Mediawiki Parser Object
|
|
|
|
* @param PPFrame $frame Mediawiki PPFrame Object
|
|
|
|
*
|
|
|
|
* @return string HTML
|
|
|
|
*/
|
|
|
|
public static function renderTabberTransclude( $input, array $args, Parser $parser, PPFrame $frame ) {
|
|
|
|
$parser->getOutput()->addModules( [ 'ext.tabberNeue' ] );
|
|
|
|
$selected = true;
|
|
|
|
|
|
|
|
$arr = explode( "\n", $input );
|
|
|
|
$htmlTabs = '';
|
|
|
|
foreach ( $arr as $tab ) {
|
|
|
|
$htmlTabs .= self::buildTabTransclude( $tab, $parser, $frame, $selected );
|
|
|
|
}
|
|
|
|
|
|
|
|
$html = '<div class="tabber">' .
|
|
|
|
'<section class="tabber__section">' . $htmlTabs . "</section></div>";
|
|
|
|
|
|
|
|
return $html;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Build individual tab.
|
|
|
|
*
|
|
|
|
* @param string $tab Tab information
|
|
|
|
* @param Parser $parser Mediawiki Parser Object
|
|
|
|
* @param PPFrame $frame Mediawiki PPFrame Object
|
|
|
|
* @param bool $selected The tab is the selected one
|
|
|
|
*
|
|
|
|
* @return string HTML
|
|
|
|
*/
|
|
|
|
private static function buildTabTransclude( $tab, Parser $parser, PPFrame $frame, &$selected ) {
|
|
|
|
$tab = trim( $tab );
|
|
|
|
if ( empty( $tab ) ) {
|
|
|
|
return '';
|
|
|
|
}
|
|
|
|
|
|
|
|
$tabBody = '';
|
|
|
|
$dataProps = [];
|
|
|
|
// Use array_pad to make sure at least 2 array values are always returned
|
|
|
|
list( $pageName, $tabName ) = array_pad( explode( '|', $tab, 2 ), 2, '' );
|
|
|
|
$title = Title::newFromText( trim( $pageName ) );
|
|
|
|
if ( !$title ) {
|
|
|
|
if ( empty( $tabName ) ) {
|
|
|
|
$tabName = $pageName;
|
|
|
|
}
|
|
|
|
$tabBody = sprintf( '<div class="error">Invalid title: %s</div>', $pageName );
|
|
|
|
$pageName = '';
|
|
|
|
} else {
|
|
|
|
$pageName = $title->getPrefixedText();
|
|
|
|
if ( empty( $tabName ) ) {
|
|
|
|
$tabName = $pageName;
|
|
|
|
}
|
|
|
|
$dataProps['page-title'] = $pageName;
|
|
|
|
if ( $selected ) {
|
|
|
|
$tabBody = $parser->recursiveTagParseFully(
|
|
|
|
sprintf( '{{:%s}}', $pageName ),
|
|
|
|
$frame
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
// Add a link placeholder, as a fallback if JavaScript doesn't execute
|
|
|
|
$linkRenderer = MediaWikiServices::getInstance()->getLinkRenderer();
|
|
|
|
$tabBody = sprintf(
|
|
|
|
'<div class="tabber__ajaxplaceholder">%s</div>',
|
|
|
|
$linkRenderer->makeLink( $title, null, [ 'rel' => 'nofollow' ] )
|
|
|
|
);
|
|
|
|
$dataProps['pending-load'] = '1';
|
|
|
|
// 1.37: $currentTitle = $parser->getPage();
|
|
|
|
$currentTitle = $parser->getTitle();
|
|
|
|
$query = sprintf(
|
|
|
|
'?action=parse&format=json&formatversion=2&title=%s&text={{:%s}}&redirects=1&prop=text&disablelimitreport=1&disabletoc=1&wrapoutputclass=',
|
|
|
|
urlencode( $currentTitle->getPrefixedText() ),
|
|
|
|
urlencode( $pageName )
|
|
|
|
);
|
|
|
|
$dataProps['load-url'] = wfExpandUrl( wfScript( 'api' ) . $query, PROTO_CANONICAL );
|
|
|
|
$oldTabBody = $tabBody;
|
|
|
|
// Allow extensions to update the lazy loaded tab
|
|
|
|
Hooks::run( 'TabberTranscludeRenderLazyLoadedTab', [ &$tabBody, &$dataProps, $parser, $frame ] );
|
|
|
|
if ( $oldTabBody != $tabBody ) {
|
|
|
|
$parser->getOutput()->recordOption( 'tabbertranscludelazyupdated' );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Register as a template
|
|
|
|
$revRecord = $parser->fetchCurrentRevisionRecordOfTitle( $title );
|
|
|
|
$parser->getOutput()->addTemplate(
|
|
|
|
$title,
|
|
|
|
$title->getArticleId(),
|
|
|
|
$revRecord ? $revRecord->getId() : null
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
$tab = '<article class="tabber__panel" title="' . htmlspecialchars( $tabName ) . '"';
|
|
|
|
$tab .= implode( array_map( static function ( $prop, $value ) {
|
|
|
|
return sprintf( ' data-tabber-%s="%s"', $prop, htmlspecialchars( $value ) );
|
|
|
|
}, array_keys( $dataProps ), $dataProps ) );
|
|
|
|
$tab .= '>' . $tabBody . '</article>';
|
|
|
|
$selected = false;
|
|
|
|
|
|
|
|
return $tab;
|
|
|
|
}
|
2019-05-14 18:59:33 +00:00
|
|
|
}
|