123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187 |
- <?php
- namespace MediaWiki\Widget\Search;
- use MediaWiki\Interwiki\InterwikiLookup;
- use MediaWiki\Linker\LinkRenderer;
- use ISearchResultSet;
- use SpecialSearch;
- use Title;
- use Html;
- use OOUI;
- /**
- * Renders one or more ISearchResultSets into a sidebar grouped by
- * interwiki prefix. Includes a per-wiki header indicating where
- * the results are from.
- */
- class InterwikiSearchResultSetWidget implements SearchResultSetWidget {
- /** @var SpecialSearch $specialSearch */
- protected $specialSearch;
- /** @var SearchResultWidget $resultWidget */
- protected $resultWidget;
- /** @var string[]|null $customCaptions */
- protected $customCaptions;
- /** @var LinkRenderer $linkRenderer */
- protected $linkRenderer;
- /** @var InterwikiLookup $iwLookup */
- protected $iwLookup;
- /** @var \OutputPage $output */
- protected $output;
- /** @var bool $showMultimedia */
- protected $showMultimedia;
- public function __construct(
- SpecialSearch $specialSearch,
- SearchResultWidget $resultWidget,
- LinkRenderer $linkRenderer,
- InterwikiLookup $iwLookup,
- $showMultimedia = false
- ) {
- $this->specialSearch = $specialSearch;
- $this->resultWidget = $resultWidget;
- $this->linkRenderer = $linkRenderer;
- $this->iwLookup = $iwLookup;
- $this->output = $specialSearch->getOutput();
- $this->showMultimedia = $showMultimedia;
- }
- /**
- * @param string $term User provided search term
- * @param ISearchResultSet|ISearchResultSet[] $resultSets List of interwiki
- * results to render.
- * @return string HTML
- */
- public function render( $term, $resultSets ) {
- if ( !is_array( $resultSets ) ) {
- $resultSets = [ $resultSets ];
- }
- $this->loadCustomCaptions();
- if ( $this->showMultimedia ) {
- $this->output->addModules( 'mediawiki.special.search.commonsInterwikiWidget' );
- }
- $this->output->addModuleStyles( 'mediawiki.special.search.interwikiwidget.styles' );
- $iwResults = [];
- foreach ( $resultSets as $resultSet ) {
- foreach ( $resultSet as $result ) {
- if ( !$result->isBrokenTitle() ) {
- $iwResults[$result->getTitle()->getInterwiki()][] = $result;
- }
- }
- }
- $iwResultSetPos = 1;
- $iwResultListOutput = '';
- foreach ( $iwResults as $iwPrefix => $results ) {
- // TODO: Assumes interwiki results are never paginated
- $position = 0;
- $iwResultItemOutput = '';
- foreach ( $results as $result ) {
- $iwResultItemOutput .= $this->resultWidget->render( $result, $position++ );
- }
- $footerHtml = $this->footerHtml( $term, $iwPrefix );
- $iwResultListOutput .= Html::rawElement( 'li',
- [
- 'class' => 'iw-resultset',
- 'data-iw-resultset-pos' => $iwResultSetPos,
- 'data-iw-resultset-source' => $iwPrefix
- ],
- $iwResultItemOutput .
- $footerHtml
- );
- $iwResultSetPos++;
- }
- return Html::rawElement(
- 'div',
- [ 'id' => 'mw-interwiki-results' ],
- Html::rawElement(
- 'p',
- [ 'class' => 'iw-headline' ],
- $this->specialSearch->msg( 'search-interwiki-caption' )->parse()
- ) .
- Html::rawElement(
- 'ul', [ 'class' => 'iw-results', ], $iwResultListOutput
- )
- );
- }
- /**
- * Generates an HTML footer for the given interwiki prefix
- *
- * @param string $term User provided search term
- * @param string $iwPrefix Interwiki prefix of wiki to show footer for
- * @return string HTML
- */
- protected function footerHtml( $term, $iwPrefix ) {
- $href = Title::makeTitle( NS_SPECIAL, 'Search', null, $iwPrefix )->getLocalURL(
- [ 'search' => $term, 'fulltext' => 1 ]
- );
- $interwiki = $this->iwLookup->fetch( $iwPrefix );
- $parsed = wfParseUrl( wfExpandUrl( $interwiki ? $interwiki->getURL() : '/' ) );
- $caption = $this->customCaptions[$iwPrefix] ??
- $this->specialSearch->msg( 'search-interwiki-default', $parsed['host'] )->escaped();
- $searchLink = Html::rawElement( 'em', null,
- Html::rawElement( 'a', [ 'href' => $href, 'target' => '_blank' ], $caption )
- );
- return Html::rawElement( 'div',
- [ 'class' => 'iw-result__footer' ],
- $this->iwIcon( $iwPrefix ) . $searchLink );
- }
- protected function loadCustomCaptions() {
- if ( $this->customCaptions !== null ) {
- return;
- }
- $this->customCaptions = [];
- $customLines = explode( "\n", $this->specialSearch->msg( 'search-interwiki-custom' )->escaped() );
- foreach ( $customLines as $line ) {
- $parts = explode( ':', $line, 2 );
- if ( count( $parts ) === 2 ) {
- $this->customCaptions[$parts[0]] = $parts[1];
- }
- }
- }
- /**
- * Generates a custom OOUI icon element with a favicon as the image.
- * The favicon image URL is generated by parsing the interwiki URL
- * and returning the default location of the favicon for that domain,
- * which is assumed to be '/favicon.ico'.
- *
- * @param string $iwPrefix Interwiki prefix
- * @return OOUI\IconWidget
- */
- protected function iwIcon( $iwPrefix ) {
- $interwiki = $this->iwLookup->fetch( $iwPrefix );
- $parsed = wfParseUrl( wfExpandUrl( $interwiki ? $interwiki->getURL() : '/' ) );
- $iwIconUrl = $parsed['scheme'] .
- $parsed['delimiter'] .
- $parsed['host'] .
- ( isset( $parsed['port'] ) ? ':' . $parsed['port'] : '' ) .
- '/favicon.ico';
- $iwIcon = new OOUI\IconWidget( [
- 'icon' => 'favicon'
- ] );
- $iwIcon->setAttributes( [ 'style' => "background-image:url($iwIconUrl);" ] );
- return $iwIcon;
- }
- }
|