setHook( 'tabber', [ __CLASS__, 'renderTabber' ] ); $parser->setHook( 'tabbertransclude', [ __CLASS__, 'renderTabberTransclude' ] ); } /** * Renders the necessary HTML for a 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 renderTabber( $input, array $args, Parser $parser, PPFrame $frame ) { $parser->getOutput()->addModules( [ 'ext.tabberNeue' ] ); $arr = explode( "|-|", $input ); $htmlTabs = ''; foreach ( $arr as $tab ) { $htmlTabs .= self::buildTab( $tab, $parser, $frame ); } $html = '
' . '
' . $htmlTabs . "
"; return $html; } /** * Build individual tab. * * @param string $tab Tab information * @param Parser $parser Mediawiki Parser Object * @param PPFrame $frame Mediawiki PPFrame Object * * @return string HTML */ private static function buildTab( $tab, Parser $parser, PPFrame $frame ) { $tab = trim( $tab ); if ( empty( $tab ) ) { return $tab; } // Use array_pad to make sure at least 2 array values are always returned list( $tabName, $tabBody ) = array_pad( array_map( 'trim', explode( '=', $tab, 2 ) ), 2, '' ); // 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 ); $tab = '
' . $tabBody . '
'; return $tab; } /** * Renders the necessary HTML for a 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 = '
' . '
' . $htmlTabs . "
"; 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( '
Invalid title: %s
', $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( '
%s
', $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 = '
'; $selected = false; return $tab; } }