Translate extension for MediaWiki
 
Loading...
Searching...
No Matches
TranslateSpecialPage.php
1<?php
2declare( strict_types = 1 );
3
4namespace MediaWiki\Extension\Translate\TranslatorInterface;
5
7use ErrorPageError;
8use Html;
9use Language;
10use LogicException;
14use MediaWiki\Languages\LanguageFactory;
15use MediaWiki\Languages\LanguageNameUtils;
16use MediaWiki\MediaWikiServices;
17use MessageGroup;
18use Skin;
19use SpecialPage;
20use Xml;
21
31class TranslateSpecialPage extends SpecialPage {
33 protected $group;
34 protected $defaults;
35 protected $nondefaults = [];
36 protected $options;
38 private $contentLanguage;
40 private $languageFactory;
42 private $languageNameUtils;
44 private $hookRunner;
45
46 public function __construct(
47 Language $contentLanguage,
48 LanguageFactory $languageFactory,
49 LanguageNameUtils $languageNameUtils,
50 HookRunner $hookRunner
51 ) {
52 parent::__construct( 'Translate' );
53 $this->contentLanguage = $contentLanguage;
54 $this->languageFactory = $languageFactory;
55 $this->languageNameUtils = $languageNameUtils;
56 $this->hookRunner = $hookRunner;
57 }
58
59 public function doesWrites() {
60 return true;
61 }
62
63 protected function getGroupName() {
64 return 'translation';
65 }
66
73 public function execute( $parameters ) {
74 $out = $this->getOutput();
75 $out->addModuleStyles( [
76 'ext.translate.special.translate.styles',
77 'jquery.uls.grid',
78 'mediawiki.ui.button'
79 ] );
80
81 $this->setHeaders();
82
83 $this->setup( $parameters );
84
85 // Redirect old export URLs to Special:ExportTranslations
86 if ( $this->getRequest()->getText( 'taction' ) === 'export' ) {
87 $exportPage = SpecialPage::getTitleFor( 'ExportTranslations' );
88 $out->redirect( $exportPage->getLocalURL( $this->nondefaults ) );
89 }
90
91 $out->addModules( 'ext.translate.special.translate' );
92 $out->addJsConfigVars(
93 'wgTranslateLanguages',
94 Utilities::getLanguageNames( LanguageNameUtils::AUTONYMS )
95 );
96
97 $out->addHTML( Html::openElement( 'div', [
98 'class' => 'grid ext-translate-container',
99 ] ) );
100
101 $out->addHTML( $this->tuxSettingsForm() );
102 $out->addHTML( $this->messageSelector() );
103
104 $table = new MessageTable( $this->getContext(), $this->group, $this->options['language'] );
105 $output = $table->fullTable();
106
107 $out->addHTML( $output );
108 $out->addHTML( Html::closeElement( 'div' ) );
109 }
110
111 protected function setup( ?string $parameters ): void {
112 $request = $this->getRequest();
113
114 $defaults = [
115 /* str */'language' => $this->getLanguage()->getCode(),
116 /* str */'group' => '!additions',
117 ];
118
119 // Dump everything here
120 $nondefaults = [];
121
122 $parameters = $parameters !== null ? array_map( 'trim', explode( ';', $parameters ) ) : [];
123 $pars = [];
124
125 foreach ( $parameters as $_ ) {
126 if ( $_ === '' ) {
127 continue;
128 }
129
130 if ( str_contains( $_, '=' ) ) {
131 [ $key, $value ] = array_map( 'trim', explode( '=', $_, 2 ) );
132 } else {
133 $key = 'group';
134 $value = $_;
135 }
136
137 $pars[$key] = $value;
138 }
139
140 foreach ( $defaults as $v => $t ) {
141 if ( is_bool( $t ) ) {
142 $r = isset( $pars[$v] ) ? (bool)$pars[$v] : $defaults[$v];
143 $r = $request->getBool( $v, $r );
144 } elseif ( is_int( $t ) ) {
145 $r = isset( $pars[$v] ) ? (int)$pars[$v] : $defaults[$v];
146 $r = $request->getInt( $v, $r );
147 } elseif ( is_string( $t ) ) {
148 $r = isset( $pars[$v] ) ? (string)$pars[$v] : $defaults[$v];
149 $r = $request->getText( $v, $r );
150 }
151
152 if ( !isset( $r ) ) {
153 throw new LogicException( '$r was not set' );
154 }
155
156 if ( $defaults[$v] !== $r ) {
157 $nondefaults[$v] = $r;
158 }
159 }
160
161 $this->defaults = $defaults;
162 $this->nondefaults = $nondefaults;
163 $this->hookRunner->onTranslateGetSpecialTranslateOptions( $defaults, $nondefaults );
164
165 $this->options = $nondefaults + $defaults;
166 $this->group = MessageGroups::getGroup( $this->options['group'] );
167 if ( $this->group ) {
168 $this->options['group'] = $this->group->getId();
169 } else {
170 $this->group = MessageGroups::getGroup( $this->defaults['group'] );
171 }
172
173 if ( !$this->languageNameUtils->isKnownLanguageTag( $this->options['language'] ) ) {
174 $this->options['language'] = $this->defaults['language'];
175 }
176
177 if ( MessageGroups::isDynamic( $this->group ) ) {
178 // @phan-suppress-next-line PhanUndeclaredMethod
179 $this->group->setLanguage( $this->options['language'] );
180 }
181 }
182
183 protected function tuxSettingsForm(): string {
184 $nojs = Html::errorBox(
185 $this->msg( 'tux-nojs' )->plain(),
186 '',
187 'tux-nojs'
188 );
189
190 $attrs = [ 'class' => 'row tux-editor-header' ];
191 $selectors = $this->tuxGroupSelector() .
192 $this->tuxLanguageSelector() .
193 $this->tuxGroupDescription() .
194 $this->tuxWorkflowSelector() .
195 $this->tuxGroupWarning();
196
197 return Html::rawElement( 'div', $attrs, $selectors ) . $nojs;
198 }
199
200 protected function messageSelector(): string {
201 $output = Html::openElement( 'div', [ 'class' => 'row tux-messagetable-header hide' ] );
202 $output .= Html::openElement( 'div', [ 'class' => 'nine columns' ] );
203 $output .= Html::openElement( 'ul', [ 'class' => 'row tux-message-selector' ] );
204 $userId = $this->getUser()->getId();
205 $tabs = [
206 'all' => '',
207 'untranslated' => '!translated',
208 'outdated' => 'fuzzy',
209 'translated' => 'translated',
210 'unproofread' => "translated|!reviewer:$userId|!last-translator:$userId",
211 ];
212
213 $params = $this->nondefaults;
214
215 foreach ( $tabs as $tab => $filter ) {
216 // Possible classes and messages, for grepping:
217 // tux-tab-all
218 // tux-tab-untranslated
219 // tux-tab-outdated
220 // tux-tab-translated
221 // tux-tab-unproofread
222 $tabClass = "tux-tab-$tab";
223 $taskParams = [ 'filter' => $filter ] + $params;
224 ksort( $taskParams );
225 $href = $this->getPageTitle()->getLocalURL( $taskParams );
226 $link = Html::element( 'a', [ 'href' => $href ], $this->msg( $tabClass )->text() );
227 $output .= Html::rawElement( 'li', [
228 'class' => 'column ' . $tabClass,
229 'data-filter' => $filter,
230 'data-title' => $tab,
231 ], $link );
232 }
233
234 // Check boxes for the "more" tab.
235 // The array keys are used as the name attribute of the checkbox.
236 // in the id attribute as tux-option-KEY,
237 // and also for the data-filter attribute.
238 // The message is shown as the checkbox's label.
239 $options = [
240 'optional' => $this->msg( 'tux-message-filter-optional-messages-label' )->text(),
241 ];
242
243 $container = Html::openElement( 'ul', [ 'class' => 'column tux-message-selector' ] );
244 foreach ( $options as $optFilter => $optLabel ) {
245 $container .= Html::rawElement( 'li',
246 [ 'class' => 'column' ],
247 Xml::checkLabel(
248 $optLabel,
249 $optFilter,
250 "tux-option-$optFilter",
251 isset( $this->nondefaults[$optFilter] ),
252 [ 'data-filter' => $optFilter ]
253 )
254 );
255 }
256
257 $container .= Html::closeElement( 'ul' );
258
259 // @todo FIXME: Hard coded "ellipsis".
260 $output .= Html::openElement( 'li', [ 'class' => 'column more' ] ) .
261 '...' .
262 $container .
263 Html::closeElement( 'li' );
264
265 $output .= Html::closeElement( 'ul' );
266 $output .= Html::closeElement( 'div' ); // close nine columns
267 $output .= Html::openElement( 'div', [ 'class' => 'three columns' ] );
268 $output .= Html::rawElement(
269 'div',
270 [ 'class' => 'tux-message-filter-wrapper' ],
271 Html::element( 'input', [
272 'class' => 'tux-message-filter-box',
273 'type' => 'search',
274 'placeholder' => $this->msg( 'tux-message-filter-placeholder' )->text()
275 ] )
276 );
277
278 // close three columns and the row
279 $output .= Html::closeElement( 'div' ) . Html::closeElement( 'div' );
280
281 return $output;
282 }
283
284 protected function tuxGroupSelector(): string {
285 $groupClass = [ 'grouptitle', 'grouplink' ];
286 if ( $this->group instanceof AggregateMessageGroup ) {
287 $groupClass[] = 'tux-breadcrumb__item--aggregate';
288 }
289
290 // @todo FIXME The selector should have expanded parent-child lists
291 $output = Html::openElement( 'div', [
292 'class' => 'eight columns tux-breadcrumb',
293 'data-language' => $this->options['language'],
294 ] ) .
295 Html::element( 'span',
296 [ 'class' => 'grouptitle' ],
297 $this->msg( 'translate-msggroupselector-projects' )->text()
298 ) .
299 Html::element( 'span',
300 [ 'class' => 'grouptitle grouplink tux-breadcrumb__item--aggregate' ],
301 $this->msg( 'translate-msggroupselector-search-all' )->text()
302 ) .
303 Html::element( 'span',
304 [
305 'class' => $groupClass,
306 'data-msggroupid' => $this->group->getId(),
307 ],
308 $this->group->getLabel( $this->getContext() )
309 ) .
310 Html::closeElement( 'div' );
311
312 return $output;
313 }
314
315 protected function tuxLanguageSelector(): string {
316 global $wgTranslateDocumentationLanguageCode;
317
318 if ( $this->options['language'] === $wgTranslateDocumentationLanguageCode ) {
319 $targetLangName = $this->msg( 'translate-documentation-language' )->text();
320 $targetLanguage = $this->contentLanguage;
321 } else {
322 $targetLangName = $this->languageNameUtils->getLanguageName( $this->options['language'] );
323 $targetLanguage = $this->languageFactory->getLanguage( $this->options['language'] );
324 }
325
326 $label = Html::element( 'span', [], $this->msg( 'tux-languageselector' )->text() );
327
328 $languageIcon = Html::element(
329 'span',
330 [ 'class' => 'ext-translate-language-icon' ]
331 );
332
333 $targetLanguageName = Html::element(
334 'span',
335 [
336 'class' => 'ext-translate-target-language',
337 'dir' => $targetLanguage->getDir(),
338 'lang' => $targetLanguage->getHtmlCode()
339 ],
340 $targetLangName
341 );
342
343 $expandIcon = Html::element(
344 'span',
345 [ 'class' => 'ext-translate-language-selector-expand' ]
346 );
347
348 $value = Html::rawElement(
349 'span',
350 [
351 'class' => 'uls mw-ui-button',
352 'tabindex' => 0,
353 'title' => $this->msg( 'tux-select-target-language' )->text()
354 ],
355 $languageIcon . $targetLanguageName . $expandIcon
356 );
357
358 return Html::rawElement(
359 'div',
360 [ 'class' => 'four columns ext-translate-language-selector' ],
361 "$label $value"
362 );
363 }
364
365 protected function tuxGroupDescription(): string {
366 // Initialize an empty warning box to be filled client-side.
367 return Html::rawElement(
368 'div',
369 [ 'class' => 'twelve columns description' ],
370 $this->getGroupDescription( $this->group )
371 );
372 }
373
374 protected function getGroupDescription( MessageGroup $group ): string {
375 $description = $group->getDescription( $this->getContext() );
376 return $description === null ?
377 '' : $this->getOutput()->parseAsInterface( $description );
378 }
379
380 protected function tuxGroupWarning(): string {
381 if ( $this->options['group'] === '' ) {
382 return Html::warningBox(
383 $this->msg( 'tux-translate-page-no-such-group' )->parse(),
384 'tux-group-warning twelve column'
385 );
386 }
387
388 return '';
389 }
390
391 protected function tuxWorkflowSelector(): string {
392 return Html::element( 'div', [ 'class' => 'tux-workflow twelve columns' ] );
393 }
394
399 public static function tabify( Skin $skin, array &$tabs ): bool {
400 $title = $skin->getTitle();
401 if ( !$title->isSpecialPage() ) {
402 return true;
403 }
404 [ $alias, $sub ] = MediaWikiServices::getInstance()
405 ->getSpecialPageFactory()->resolveAlias( $title->getText() );
406
407 $pagesInGroup = [ 'Translate', 'LanguageStats', 'MessageGroupStats', 'ExportTranslations' ];
408 if ( !in_array( $alias, $pagesInGroup, true ) ) {
409 return true;
410 }
411
412 // Extract subpage syntax, otherwise the values are not passed forward
413 $params = [];
414 if ( $sub !== null && trim( $sub ) !== '' ) {
415 if ( $alias === 'Translate' || $alias === 'MessageGroupStats' ) {
416 $params['group'] = $sub;
417 } elseif ( $alias === 'LanguageStats' ) {
418 // Breaks if additional parameters besides language are code provided
419 $params['language'] = $sub;
420 }
421 }
422
423 $request = $skin->getRequest();
424 // However, query string params take precedence
425 $params['language'] = $request->getRawVal( 'language' ) ?? '';
426 $params['group'] = $request->getRawVal( 'group' ) ?? '';
427
428 // Remove empty values from params
429 $params = array_filter( $params, static function ( string $param ) {
430 return $param !== '';
431 } );
432
433 $translate = SpecialPage::getTitleFor( 'Translate' );
434 $languagestats = SpecialPage::getTitleFor( 'LanguageStats' );
435 $messagegroupstats = SpecialPage::getTitleFor( 'MessageGroupStats' );
436
437 // Clear the special page tab that might be there already
438 $tabs['namespaces'] = [];
439
440 $tabs['namespaces']['translate'] = [
441 'text' => wfMessage( 'translate-taction-translate' )->text(),
442 'href' => $translate->getLocalURL( $params ),
443 'class' => 'tux-tab',
444 ];
445
446 if ( $alias === 'Translate' ) {
447 $tabs['namespaces']['translate']['class'] .= ' selected';
448 }
449
450 $tabs['views']['lstats'] = [
451 'text' => wfMessage( 'translate-taction-lstats' )->text(),
452 'href' => $languagestats->getLocalURL( $params ),
453 'class' => 'tux-tab',
454 ];
455 if ( $alias === 'LanguageStats' ) {
456 $tabs['views']['lstats']['class'] .= ' selected';
457 }
458
459 $tabs['views']['mstats'] = [
460 'text' => wfMessage( 'translate-taction-mstats' )->text(),
461 'href' => $messagegroupstats->getLocalURL( $params ),
462 'class' => 'tux-tab',
463 ];
464
465 if ( $alias === 'MessageGroupStats' ) {
466 $tabs['views']['mstats']['class'] .= ' selected';
467 }
468
469 $tabs['views']['export'] = [
470 'text' => wfMessage( 'translate-taction-export' )->text(),
471 'href' => SpecialPage::getTitleFor( 'ExportTranslations' )->getLocalURL( $params ),
472 'class' => 'tux-tab',
473 ];
474
475 return true;
476 }
477}
return[ 'Translate:ConfigHelper'=> static function():ConfigHelper { return new ConfigHelper();}, 'Translate:CsvTranslationImporter'=> static function(MediaWikiServices $services):CsvTranslationImporter { return new CsvTranslationImporter( $services->getWikiPageFactory());}, 'Translate:EntitySearch'=> static function(MediaWikiServices $services):EntitySearch { return new EntitySearch($services->getMainWANObjectCache(), $services->getCollationFactory() ->makeCollation( 'uca-default-u-kn'), MessageGroups::singleton(), $services->getNamespaceInfo(), $services->get( 'Translate:MessageIndex'), $services->getTitleParser(), $services->getTitleFormatter());}, 'Translate:ExternalMessageSourceStateImporter'=> static function(MediaWikiServices $services):ExternalMessageSourceStateImporter { return new ExternalMessageSourceStateImporter($services->getMainConfig(), $services->get( 'Translate:GroupSynchronizationCache'), $services->getJobQueueGroup(), LoggerFactory::getInstance( 'Translate.GroupSynchronization'), $services->get( 'Translate:MessageIndex'));}, 'Translate:FileFormatFactory'=> static function(MediaWikiServices $services):FileFormatFactory { return new FileFormatFactory( $services->getObjectFactory());}, 'Translate:GroupSynchronizationCache'=> static function(MediaWikiServices $services):GroupSynchronizationCache { return new GroupSynchronizationCache( $services->get( 'Translate:PersistentCache'));}, 'Translate:HookRunner'=> static function(MediaWikiServices $services):HookRunner { return new HookRunner( $services->getHookContainer());}, 'Translate:MessageBundleStore'=> static function(MediaWikiServices $services):MessageBundleStore { return new MessageBundleStore($services->get( 'Translate:RevTagStore'), $services->getJobQueueGroup(), $services->getLanguageNameUtils(), $services->get( 'Translate:MessageIndex'));}, 'Translate:MessageGroupReviewStore'=> static function(MediaWikiServices $services):MessageGroupReviewStore { return new MessageGroupReviewStore($services->getDBLoadBalancer(), $services->get( 'Translate:HookRunner'));}, 'Translate:MessageGroupStatsTableFactory'=> static function(MediaWikiServices $services):MessageGroupStatsTableFactory { return new MessageGroupStatsTableFactory($services->get( 'Translate:ProgressStatsTableFactory'), $services->getDBLoadBalancer(), $services->getLinkRenderer(), $services->get( 'Translate:MessageGroupReviewStore'), $services->getMainConfig() ->get( 'TranslateWorkflowStates') !==false);}, 'Translate:MessageIndex'=> static function(MediaWikiServices $services):MessageIndex { $params=$services->getMainConfig() ->get( 'TranslateMessageIndex');if(is_string( $params)) { $params=(array) $params;} $class=array_shift( $params);return new $class( $params);}, 'Translate:MessagePrefixStats'=> static function(MediaWikiServices $services):MessagePrefixStats { return new MessagePrefixStats( $services->getTitleParser());}, 'Translate:ParsingPlaceholderFactory'=> static function():ParsingPlaceholderFactory { return new ParsingPlaceholderFactory();}, 'Translate:PersistentCache'=> static function(MediaWikiServices $services):PersistentCache { return new PersistentDatabaseCache($services->getDBLoadBalancer(), $services->getJsonCodec());}, 'Translate:ProgressStatsTableFactory'=> static function(MediaWikiServices $services):ProgressStatsTableFactory { return new ProgressStatsTableFactory($services->getLinkRenderer(), $services->get( 'Translate:ConfigHelper'));}, 'Translate:RevTagStore'=> static function(MediaWikiServices $services):RevTagStore { return new RevTagStore($services->getDBLoadBalancerFactory());}, 'Translate:SubpageListBuilder'=> static function(MediaWikiServices $services):SubpageListBuilder { return new SubpageListBuilder($services->get( 'Translate:TranslatableBundleFactory'), $services->getLinkBatchFactory());}, 'Translate:TranslatableBundleExporter'=> static function(MediaWikiServices $services):TranslatableBundleExporter { return new TranslatableBundleExporter($services->get( 'Translate:SubpageListBuilder'), $services->getWikiExporterFactory(), $services->getDBLoadBalancer());}, 'Translate:TranslatableBundleFactory'=> static function(MediaWikiServices $services):TranslatableBundleFactory { return new TranslatableBundleFactory($services->get( 'Translate:TranslatablePageStore'), $services->get( 'Translate:MessageBundleStore'));}, 'Translate:TranslatableBundleImporter'=> static function(MediaWikiServices $services):TranslatableBundleImporter { return new TranslatableBundleImporter($services->getWikiImporterFactory(), $services->get( 'Translate:TranslatablePageParser'), $services->getRevisionLookup());}, 'Translate:TranslatableBundleMover'=> static function(MediaWikiServices $services):TranslatableBundleMover { return new TranslatableBundleMover($services->getMovePageFactory(), $services->getJobQueueGroup(), $services->getLinkBatchFactory(), $services->get( 'Translate:TranslatableBundleFactory'), $services->get( 'Translate:SubpageListBuilder'), $services->getMainConfig() ->get( 'TranslatePageMoveLimit'));}, 'Translate:TranslatableBundleStatusStore'=> static function(MediaWikiServices $services):TranslatableBundleStatusStore { return new TranslatableBundleStatusStore($services->getDBLoadBalancer() ->getConnection(DB_PRIMARY), $services->getCollationFactory() ->makeCollation( 'uca-default-u-kn'), $services->getDBLoadBalancer() ->getMaintenanceConnectionRef(DB_PRIMARY));}, 'Translate:TranslatablePageParser'=> static function(MediaWikiServices $services):TranslatablePageParser { return new TranslatablePageParser($services->get( 'Translate:ParsingPlaceholderFactory'));}, 'Translate:TranslatablePageStore'=> static function(MediaWikiServices $services):TranslatablePageStore { return new TranslatablePageStore($services->get( 'Translate:MessageIndex'), $services->getJobQueueGroup(), $services->get( 'Translate:RevTagStore'), $services->getDBLoadBalancer(), $services->get( 'Translate:TranslatableBundleStatusStore'), $services->get( 'Translate:TranslatablePageParser'),);}, 'Translate:TranslationStashReader'=> static function(MediaWikiServices $services):TranslationStashReader { $db=$services->getDBLoadBalancer() ->getConnection(DB_REPLICA);return new TranslationStashStorage( $db);}, 'Translate:TranslationStatsDataProvider'=> static function(MediaWikiServices $services):TranslationStatsDataProvider { return new TranslationStatsDataProvider(new ServiceOptions(TranslationStatsDataProvider::CONSTRUCTOR_OPTIONS, $services->getMainConfig()), $services->getObjectFactory(), $services->getDBLoadBalancer());}, 'Translate:TranslationUnitStoreFactory'=> static function(MediaWikiServices $services):TranslationUnitStoreFactory { return new TranslationUnitStoreFactory( $services->getDBLoadBalancer());}, 'Translate:TranslatorActivity'=> static function(MediaWikiServices $services):TranslatorActivity { $query=new TranslatorActivityQuery($services->getMainConfig(), $services->getDBLoadBalancer());return new TranslatorActivity($services->getMainObjectStash(), $query, $services->getJobQueueGroup());}, 'Translate:TtmServerFactory'=> static function(MediaWikiServices $services):TtmServerFactory { $config=$services->getMainConfig();$default=$config->get( 'TranslateTranslationDefaultService');if( $default===false) { $default=null;} return new TtmServerFactory( $config->get( 'TranslateTranslationServices'), $default);}]
@phpcs-require-sorted-array
Groups multiple message groups together as one group.
Hook runner for the Translate extension.
Factory class for accessing message groups individually by id or all of them as a list.
Implements the core of Translate extension - a special page which shows a list of messages in a forma...
static tabify(Skin $skin, array &$tabs)
Adds the task-based tabs on Special:Translate and few other special pages.
Essentially random collection of helper functions, similar to GlobalFunctions.php.
Definition Utilities.php:31
Interface for message groups.