summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
Diffstat (limited to 'MLEB/Translate/specials/SpecialSearchTranslations.php')
-rw-r--r--MLEB/Translate/specials/SpecialSearchTranslations.php395
1 files changed, 310 insertions, 85 deletions
diff --git a/MLEB/Translate/specials/SpecialSearchTranslations.php b/MLEB/Translate/specials/SpecialSearchTranslations.php
index 945d060b..8b2f476b 100644
--- a/MLEB/Translate/specials/SpecialSearchTranslations.php
+++ b/MLEB/Translate/specials/SpecialSearchTranslations.php
@@ -4,8 +4,7 @@
*
* @file
* @author Niklas Laxström
- * @copyright Copyright © 2013 Niklas Laxström
- * @license GPL-2.0+
+ * @license GPL-2.0-or-later
*/
/**
@@ -13,7 +12,7 @@
*
* @ingroup SpecialPage TranslateSpecialPage
*/
-class SpecialSearchTranslations extends TranslateSpecialPage {
+class SpecialSearchTranslations extends SpecialPage {
/** @var FormOptions */
protected $opts;
@@ -24,7 +23,7 @@ class SpecialSearchTranslations extends TranslateSpecialPage {
* don't contain any chars that are escaped in html.
* @var array
*/
- protected $hl = array();
+ protected $hl = [];
/**
* How many search results to display per page
@@ -34,10 +33,10 @@ class SpecialSearchTranslations extends TranslateSpecialPage {
public function __construct() {
parent::__construct( 'SearchTranslations' );
- $this->hl = array(
+ $this->hl = [
TranslateUtils::getPlaceholder(),
TranslateUtils::getPlaceholder(),
- );
+ ];
}
public function setHeaders() {
@@ -52,6 +51,7 @@ class SpecialSearchTranslations extends TranslateSpecialPage {
}
public function execute( $par ) {
+ global $wgLanguageCode;
$this->setHeaders();
$this->checkPermissions();
@@ -61,13 +61,24 @@ class SpecialSearchTranslations extends TranslateSpecialPage {
}
$out = $this->getOutput();
+ $out->addModuleStyles( 'jquery.uls.grid' );
+ $out->addModuleStyles( 'ext.translate.special.searchtranslations.styles' );
+ $out->addModuleStyles( 'ext.translate.special.translate.styles' );
+ $out->addModuleStyles( [ 'mediawiki.ui.button', 'mediawiki.ui.input', 'mediawiki.ui.checkbox' ] );
$out->addModules( 'ext.translate.special.searchtranslations' );
+ $out->addModules( 'ext.translate.special.searchtranslations.operatorsuggest' );
+ $out->addHelpLink( 'Help:Extension:Translate#searching' );
+ $out->addJsConfigVars( 'wgTranslateLanguages', TranslateUtils::getLanguageNames( null ) );
$this->opts = $opts = new FormOptions();
$opts->add( 'query', '' );
+ $opts->add( 'sourcelanguage', $wgLanguageCode );
$opts->add( 'language', '' );
$opts->add( 'group', '' );
$opts->add( 'grouppath', '' );
+ $opts->add( 'filter', '' );
+ $opts->add( 'match', '' );
+ $opts->add( 'case', '' );
$opts->add( 'limit', $this->limit );
$opts->add( 'offset', 0 );
@@ -81,41 +92,88 @@ class SpecialSearchTranslations extends TranslateSpecialPage {
return;
}
+ $options = $params = $opts->getAllValues();
+ $filter = $opts->getValue( 'filter' );
try {
- $resultset = $server->search( $queryString, $opts, $this->hl );
+ if ( $opts->getValue( 'language' ) === '' ) {
+ $options['language'] = $this->getLanguage()->getCode();
+ }
+ $translationSearch = new CrossLanguageTranslationSearchQuery( $options, $server );
+ if ( in_array( $filter, $translationSearch->getAvailableFilters() ) ) {
+ $opts->setValue( 'language', $options['language'] );
+ $documents = $translationSearch->getDocuments();
+ $total = $translationSearch->getTotalHits();
+ $resultset = $translationSearch->getResultSet();
+ } else {
+ $resultset = $server->search( $queryString, $params, $this->hl );
+ $documents = $server->getDocuments( $resultset );
+ $total = $server->getTotalHits( $resultset );
+ }
} catch ( TTMServerException $e ) {
- error_log( 'Translation search server unavailable:' . $e->getMessage() );
+ error_log( 'Translation search server unavailable: ' . $e->getMessage() );
throw new ErrorPageError( 'tux-sst-solr-offline-title', 'tux-sst-solr-offline-body' );
}
// Part 1: facets
$facets = $server->getFacets( $resultset );
+ $facetHtml = '';
- $facetHtml = Html::element( 'div',
- array( 'class' => 'row facet languages',
- 'data-facets' => FormatJson::encode( $this->getLanguages( $facets['language'] ) ),
- 'data-language' => $opts->getValue( 'language' ),
- ),
- $this->msg( 'tux-sst-facet-language' )
- );
+ if ( $facets['language'] !== [] ) {
+ if ( $filter !== '' ) {
+ $facets['language'] = array_merge(
+ $facets['language'],
+ [ $opts->getValue( 'language' ) => $total ]
+ );
+ }
+ $facetHtml = Html::element( 'div',
+ [ 'class' => 'row facet languages',
+ 'data-facets' => FormatJson::encode( $this->getLanguages( $facets['language'] ) ),
+ 'data-language' => $opts->getValue( 'language' ),
+ ],
+ $this->msg( 'tux-sst-facet-language' )->text()
+ );
+ }
- $facetHtml .= Html::element( 'div',
- array( 'class' => 'row facet groups',
- 'data-facets' => FormatJson::encode( $this->getGroups( $facets['group'] ) ),
- 'data-group' => $opts->getValue( 'group' ) ),
- $this->msg( 'tux-sst-facet-group' )
- );
+ if ( $facets['group'] !== [] ) {
+ $facetHtml .= Html::element( 'div',
+ [ 'class' => 'row facet groups',
+ 'data-facets' => FormatJson::encode( $this->getGroups( $facets['group'] ) ),
+ 'data-group' => $opts->getValue( 'group' ) ],
+ $this->msg( 'tux-sst-facet-group' )->text()
+ );
+ }
// Part 2: results
$resultsHtml = '';
- $documents = $server->getDocuments( $resultset );
+
+ $title = Title::newFromText( $queryString );
+ if ( $title && !in_array( $filter, $translationSearch->getAvailableFilters() ) ) {
+ $handle = new MessageHandle( $title );
+ $code = $handle->getCode();
+ $language = $opts->getValue( 'language' );
+ if ( $code !== '' && $code !== $language && $handle->isValid() ) {
+ $dataProvider = new TranslationAidDataProvider( $handle );
+ $aid = new CurrentTranslationAid(
+ $handle->getGroup(),
+ $handle,
+ $this->getContext(),
+ $dataProvider
+ );
+ $document['wiki'] = wfWikiID();
+ $document['localid'] = $handle->getTitleForBase()->getPrefixedText();
+ $document['content'] = $aid->getData()['value'];
+ $document['language'] = $handle->getCode();
+ array_unshift( $documents, $document );
+ $total++;
+ }
+ }
foreach ( $documents as $document ) {
$text = $document['content'];
$text = TranslateUtils::convertWhiteSpaceToHTML( $text );
list( $pre, $post ) = $this->hl;
- $text = str_replace( $pre, '<strong class="tux-highlight">', $text );
+ $text = str_replace( $pre, '<strong class="tux-search-highlight">', $text );
$text = str_replace( $post, '</strong>', $text );
$title = Title::newFromText( $document['localid'] . '/' . $document['language'] );
@@ -124,107 +182,118 @@ class SpecialSearchTranslations extends TranslateSpecialPage {
continue;
}
- $resultAttribs = array(
+ $resultAttribs = [
'class' => 'row tux-message',
'data-title' => $title->getPrefixedText(),
'data-language' => $document['language'],
- );
+ ];
$handle = new MessageHandle( $title );
- $edit = '';
if ( $handle->isValid() ) {
- $groupId = $handle->getGroup()->getId();
- $helpers = new TranslationHelpers( $title, $groupId );
- $resultAttribs['data-definition'] = $helpers->getDefinition();
- $resultAttribs['data-translation'] = $helpers->getTranslation();
- $resultAttribs['data-group'] = $groupId;
-
- $uri = wfAppendQuery( $document['uri'], array( 'action' => 'edit' ) );
- $link = Html::element( 'a', array(
- 'href' => $uri,
- ), $this->msg( 'tux-sst-edit' )->text() );
- $edit = Html::rawElement(
- 'div',
- array( 'class' => 'row tux-edit tux-message-item' ),
- $link
+ $uri = TranslateUtils::getEditorUrl( $handle );
+ $link = Html::element(
+ 'a',
+ [ 'href' => $uri ],
+ $this->msg( 'tux-sst-edit' )->text()
+ );
+ } else {
+ $url = wfParseUrl( $document['uri'] );
+ $domain = $url['host'];
+ $link = Html::element(
+ 'a',
+ [ 'href' => $document['uri'] ],
+ $this->msg( 'tux-sst-view-foreign', $domain )->text()
);
}
+ $access = Html::rawElement(
+ 'div',
+ [ 'class' => 'row tux-edit tux-message-item' ],
+ $link
+ );
+
$titleText = $title->getPrefixedText();
- $titleAttribs = array(
+ $titleAttribs = [
'class' => 'row tux-title',
'dir' => 'ltr',
- );
+ ];
- $textAttribs = array(
+ $language = Language::factory( $document['language'] );
+ $textAttribs = [
'class' => 'row tux-text',
- 'lang' => wfBCP47( $document['language'] ),
- 'dir' => Language::factory( $document['language'] )->getDir(),
- );
+ 'lang' => $language->getHtmlCode(),
+ 'dir' => $language->getDir(),
+ ];
$resultsHtml = $resultsHtml
. Html::openElement( 'div', $resultAttribs )
. Html::rawElement( 'div', $textAttribs, $text )
. Html::element( 'div', $titleAttribs, $titleText )
- . $edit
+ . $access
. Html::closeElement( 'div' );
}
- $resultsHtml .= Html::rawElement( 'hr', array( 'class' => 'tux-pagination-line' ) );
+ $resultsHtml .= Html::rawElement( 'hr', [ 'class' => 'tux-pagination-line' ] );
$prev = $next = '';
- $total = $server->getTotalHits( $resultset );
$offset = $this->opts->getValue( 'offset' );
$params = $this->opts->getChangedValues();
if ( $total - $offset > $this->limit ) {
- $newParams = array( 'offset' => $offset + $this->limit ) + $params;
- $attribs = array(
+ $newParams = [ 'offset' => $offset + $this->limit ] + $params;
+ $attribs = [
'class' => 'mw-ui-button pager-next',
- 'href' => $this->getTitle()->getLocalUrl( $newParams ),
- );
+ 'href' => $this->getPageTitle()->getLocalURL( $newParams ),
+ ];
$next = Html::element( 'a', $attribs, $this->msg( 'tux-sst-next' )->text() );
}
if ( $offset ) {
- $newParams = array( 'offset' => max( 0, $offset - $this->limit ) ) + $params;
- $attribs = array(
+ $newParams = [ 'offset' => max( 0, $offset - $this->limit ) ] + $params;
+ $attribs = [
'class' => 'mw-ui-button pager-prev',
- 'href' => $this->getTitle()->getLocalUrl( $newParams ),
- );
+ 'href' => $this->getPageTitle()->getLocalURL( $newParams ),
+ ];
$prev = Html::element( 'a', $attribs, $this->msg( 'tux-sst-prev' )->text() );
}
- $resultsHtml .= Html::rawElement( 'div', array( 'class' => 'tux-pagination-links' ),
+ $resultsHtml .= Html::rawElement( 'div', [ 'class' => 'tux-pagination-links' ],
"$prev $next"
);
$search = $this->getSearchInput( $queryString );
$count = $this->msg( 'tux-sst-count' )->numParams( $total );
- $this->showSearch( $search, $count, $facetHtml, $resultsHtml );
+ $this->showSearch( $search, $count, $facetHtml, $resultsHtml, $total );
}
protected function getLanguages( array $facet ) {
- $output = array();
+ $output = [];
$nondefaults = $this->opts->getChangedValues();
$selected = $this->opts->getValue( 'language' );
+ $filter = $this->opts->getValue( 'filter' );
foreach ( $facet as $key => $value ) {
- if ( $key === $selected ) {
+ if ( $filter !== '' && $key === $selected ) {
+ unset( $nondefaults['language'] );
+ unset( $nondefaults['filter'] );
+ } elseif ( $filter !== '' ) {
+ $nondefaults['language'] = $key;
+ $nondefaults['filter'] = $filter;
+ } elseif ( $key === $selected ) {
unset( $nondefaults['language'] );
} else {
$nondefaults['language'] = $key;
}
- $url = $this->getTitle()->getLocalUrl( $nondefaults );
+ $url = $this->getPageTitle()->getLocalURL( $nondefaults );
$value = $this->getLanguage()->formatNum( $value );
- $output[$key] = array(
+ $output[$key] = [
'count' => $value,
'url' => $url
- );
+ ];
}
return $output;
@@ -236,7 +305,7 @@ class SpecialSearchTranslations extends TranslateSpecialPage {
}
protected function makeGroupFacetRows( array $groups, $counts, $level = 0, $pathString = '' ) {
- $output = array();
+ $output = [];
$nondefaults = $this->opts->getChangedValues();
$selected = $this->opts->getValue( 'group' );
@@ -248,7 +317,7 @@ class SpecialSearchTranslations extends TranslateSpecialPage {
if ( is_array( $mixed ) ) {
$group = array_shift( $subgroups );
} else {
- $subgroups = array();
+ $subgroups = [];
}
$id = $group->getId();
@@ -266,13 +335,12 @@ class SpecialSearchTranslations extends TranslateSpecialPage {
}
$value = isset( $counts[$id] ) ? $counts[$id] : 0;
- $count = $this->getLanguage()->formatNum( $value );
- $output[$id] = array(
+ $output[$id] = [
'id' => $id,
- 'count' => $count,
+ 'count' => $value,
'label' => $group->getLabel(),
- );
+ ];
if ( isset( $path[$level] ) && $path[$level] === $id ) {
$output[$id]['groups'] = $this->makeGroupFacetRows(
@@ -287,15 +355,40 @@ class SpecialSearchTranslations extends TranslateSpecialPage {
return $output;
}
- protected function showSearch( $search, $count, $facets, $results ) {
- $this->getOutput()->addHtml( <<<HTML
+ protected function showSearch( $search, $count, $facets, $results, $total ) {
+ $messageSelector = $this->messageSelector();
+ $this->getOutput()->addHTML( <<<HTML
<div class="grid tux-searchpage">
- <div class="row searchinput">
- <div class="nine columns offset-by-three">$search</div>
- </div>
- <div class="row count">
- <div class="nine columns offset-by-three">$count</div>
+ <div class="row tux-searchboxform">
+ <div class="tux-search-tabs offset-by-three">$messageSelector</div>
+ <div class="row tux-search-options">
+ <div class="offset-by-three nine columns tux-search-inputs">
+ <div class="row searchinput">$search</div>
+ <div class="row count">$count</div>
+ </div>
+ </div>
</div>
+HTML
+ );
+
+ $query = trim( $this->opts->getValue( 'query' ) );
+ $hasSpace = preg_match( '/\s/', $query );
+ $match = $this->opts->getValue( 'match' );
+ $size = 100;
+ if ( $total > $size && $match !== 'all' && $hasSpace ) {
+ $params = $this->opts->getChangedValues();
+ $params = [ 'match' => 'all' ] + $params;
+ $linkText = $this->msg( 'tux-sst-link-all-match' )->text();
+ $link = $this->getPageTitle()->getFullURL( $params );
+ $link = "<span class='plainlinks'>[$link $linkText]</span>";
+
+ $this->getOutput()->wrapWikiMsg(
+ '<div class="successbox">$1</div>',
+ [ 'tux-sst-match-message', $link ]
+ );
+ }
+
+ $this->getOutput()->addHTML( <<<HTML
<div class="row searchcontent">
<div class="three columns facets">$facets</div>
<div class="nine columns results">$results</div>
@@ -307,7 +400,7 @@ HTML
protected function showEmptySearch() {
$search = $this->getSearchInput( '' );
- $this->getOutput()->addHtml( <<<HTML
+ $this->getOutput()->addHTML( <<<HTML
<div class="grid tux-searchpage">
<div class="row searchinput">
<div class="nine columns offset-by-three">$search</div>
@@ -317,21 +410,153 @@ HTML
);
}
+ /**
+ * Build ellipsis to select options
+ * @param string $key
+ * @param string $value
+ * @return string
+ */
+ protected function ellipsisSelector( $key, $value ) {
+ $nondefaults = $this->opts->getChangedValues();
+ $taskParams = [ 'filter' => $value ] + $nondefaults;
+ ksort( $taskParams );
+ $href = $this->getPageTitle()->getLocalURL( $taskParams );
+ $link = Html::element( 'a',
+ [ 'href' => $href ],
+ // Messages for grepping:
+ // tux-sst-ellipsis-untranslated
+ // tux-sst-ellipsis-outdated
+ $this->msg( 'tux-sst-ellipsis-' . $key )->text()
+ );
+
+ $container = Html::rawElement( 'li', [
+ 'class' => 'column',
+ 'data-filter' => $value,
+ 'data-title' => $key,
+ ], $link );
+
+ return $container;
+ }
+
+ /**
+ * Design the tabs
+ * @return string
+ */
+ protected function messageSelector() {
+ $nondefaults = $this->opts->getChangedValues();
+ $output = Html::openElement( 'div', [ 'class' => 'row tux-messagetable-header' ] );
+ $output .= Html::openElement( 'div', [ 'class' => 'nine columns' ] );
+ $output .= Html::openElement( 'ul', [ 'class' => 'row tux-message-selector' ] );
+ $tabs = [
+ 'default' => '',
+ 'translated' => 'translated',
+ 'untranslated' => 'untranslated'
+ ];
+
+ $ellipsisOptions = [
+ 'outdated' => 'fuzzy'
+ ];
+
+ $selected = $this->opts->getValue( 'filter' );
+ $keys = array_keys( $tabs );
+ if ( in_array( $selected, array_values( $ellipsisOptions ) ) ) {
+ $key = $keys[count( $keys ) - 1];
+ $ellipsisOptions = [ $key => $tabs[$key] ];
+
+ // Remove the last tab
+ unset( $tabs[$key] );
+ $tabs = array_merge( $tabs, [ 'outdated' => $selected ] );
+ } elseif ( !in_array( $selected, array_values( $tabs ) ) ) {
+ $selected = '';
+ }
+
+ $container = Html::openElement( 'ul', [ 'class' => 'column tux-message-selector' ] );
+ foreach ( $ellipsisOptions as $optKey => $optValue ) {
+ $container .= $this->ellipsisSelector( $optKey, $optValue );
+ }
+
+ $sourcelanguage = $this->opts->getValue( 'sourcelanguage' );
+ $sourcelanguage = TranslateUtils::getLanguageName( $sourcelanguage );
+ foreach ( $tabs as $tab => $filter ) {
+ // Messages for grepping:
+ // tux-sst-default
+ // tux-sst-translated
+ // tux-sst-untranslated
+ // tux-sst-outdated
+ $tabClass = "tux-sst-$tab";
+ $taskParams = [ 'filter' => $filter ] + $nondefaults;
+ ksort( $taskParams );
+ $href = $this->getPageTitle()->getLocalURL( $taskParams );
+ if ( $tab === 'default' ) {
+ $link = Html::element(
+ 'a',
+ [ 'href' => $href ],
+ $this->msg( $tabClass )->text()
+ );
+ } else {
+ $link = Html::element(
+ 'a',
+ [ 'href' => $href ],
+ $this->msg( $tabClass, $sourcelanguage )->text()
+ );
+ }
+
+ if ( $selected === $filter ) {
+ $tabClass = $tabClass . ' selected';
+ }
+ $output .= Html::rawElement( 'li', [
+ 'class' => [ 'column', $tabClass ],
+ 'data-filter' => $filter,
+ 'data-title' => $tab,
+ ], $link );
+ }
+
+ // More column
+ $output .= Html::openElement( 'li', [ 'class' => 'column more' ] ) .
+ '...' .
+ $container .
+ Html::closeElement( 'li' );
+
+ $output .= Html::closeElement( 'ul' );
+ $output .= Html::closeElement( 'div' );
+ $output .= Html::closeElement( 'div' );
+
+ return $output;
+ }
+
protected function getSearchInput( $query ) {
- $attribs = array(
+ $attribs = [
'placeholder' => $this->msg( 'tux-sst-search-ph' ),
- 'class' => 'searchinputbox',
+ 'class' => 'searchinputbox mw-ui-input',
'dir' => $this->getLanguage()->getDir(),
- );
+ ];
- $title = Html::hidden( 'title', $this->getTitle()->getPrefixedText() );
+ $title = Html::hidden( 'title', $this->getPageTitle()->getPrefixedText() );
$input = Xml::input( 'query', false, $query, $attribs );
- $submit = Xml::submitButton( $this->msg( 'tux-sst-search' ), array( 'class' => 'button' ) );
+ $submit = Xml::submitButton(
+ $this->msg( 'tux-sst-search' ),
+ [ 'class' => 'mw-ui-button' ]
+ );
+
+ $nondefaults = $this->opts->getChangedValues();
+ $checkLabel = Xml::checkLabel(
+ $this->msg( 'tux-sst-case-sensitive' )->text(),
+ 'case',
+ 'tux-case-sensitive',
+ isset( $nondefaults['case'] )
+ );
+ $checkLabel = Html::openElement(
+ 'div',
+ [ 'class' => 'tux-search-operators mw-ui-checkbox' ]
+ ) .
+ $checkLabel .
+ Html::closeElement( 'div' );
+
$lang = $this->getRequest()->getVal( 'language' );
$language = is_null( $lang ) ? '' : Html::hidden( 'language', $lang );
- $form = Html::rawElement( 'form', array( 'action' => wfScript() ),
- $title . $input . $submit . $language
+ $form = Html::rawElement( 'form', [ 'action' => wfScript(), 'name' => 'searchform' ],
+ $title . $input . $submit . $checkLabel . $language
);
return $form;