55 public function __construct( Title $title =
null, $group =
null, $code =
'en' ) {
56 $this->setTitle( $title );
58 $this->setCode( $code );
66 public function setTitle( Title $title ): void {
67 $this->title = $title;
70 public function getUser(): User {
71 return $this->user ?: RequestContext::getMain()->getUser();
74 public function setUser( User $user ): void {
85 $this->group = $group;
87 $this->group = MessageGroups::getGroup( $group );
91 public function getCode(): string {
95 public function setCode(
string $code =
'en' ): void {
99 protected function getAction(): string {
100 return $this->getTitle()->getLocalURL();
103 protected function doHeader(): string {
106 'action' => $this->getAction(),
107 'class' =>
'mw-translate-manage'
110 $csrfTokenSet = RequestContext::getMain()->getCsrfTokenSet();
111 return Xml::openElement(
'form', $formParams ) .
112 Html::hidden(
'title', $this->getTitle()->getPrefixedText() ) .
113 Html::hidden(
'token', $csrfTokenSet->getToken() ) .
114 Html::hidden(
'process', 1 );
117 protected function doFooter(): string {
121 protected function allowProcess(): bool {
122 $context = RequestContext::getMain();
123 $request = $context->getRequest();
124 $csrfTokenSet = $context->getCsrfTokenSet();
126 return $request->wasPosted()
127 && $request->getBool(
'process',
false )
128 && $csrfTokenSet->matchTokenField(
'token' );
131 protected function getActions(): array {
132 if ( $this->code ===
'en' ) {
133 return [
'import',
'fuzzy',
'ignore' ];
136 return [
'import',
'conflict',
'ignore' ];
139 protected function getDefaultAction(
bool $fuzzy, ?
string $action ): string {
144 return $fuzzy ?
'conflict' :
'import';
147 public function execute( array $messages ): bool {
148 $context = RequestContext::getMain();
149 $this->out = $context->getOutput();
152 $diff =
new DifferenceEngine();
153 $diff->showDiffStyle();
154 $diff->setReducedLineNumbers();
157 $process = $this->allowProcess();
160 $group = $this->getGroup();
161 $code = $this->getCode();
162 $collection = $group->initCollection( $code );
163 $collection->loadTranslations();
165 $this->out->addHTML( $this->doHeader() );
174 foreach ( $messages as $key => $value ) {
178 if ( isset( $collection[$key] ) ) {
180 $old = $collection[$key]->translation();
184 if ( (
string)$old === (
string)$value ) {
188 if ( $old ===
null ) {
204 $para =
'<code class="mw-tmi-new">' . htmlspecialchars( $key ) .
'</code>';
205 $name = $context->msg(
'translate-manage-import-new' )->rawParams( $para )
207 $text = Utilities::convertWhiteSpaceToHTML( $value );
208 $changed[] = self::makeSectionElement( $name,
'new', $text );
210 $oldContent = ContentHandler::makeContent( $old, $diff->getTitle() );
211 $newContent = ContentHandler::makeContent( $value, $diff->getTitle() );
212 $diff->setContent( $oldContent, $newContent );
213 $text = $diff->getDiff(
'',
'' );
220 $action = $context->getRequest()
221 ->getVal( self::escapeNameForPHP(
"action-$type-$key" ) );
224 if ( $changed === [] ) {
229 if ( $action ===
null ) {
232 $message = $context->msg(
233 'translate-manage-inconsistent',
234 wfEscapeWikiText(
"action-$type-$key" )
236 $changed[] =
"<li>$message</li></ul>";
242 if ( !isset( $this->time ) ) {
243 $this->time = wfTimestamp();
248 $messageKeyAndParams = self::doAction(
259 $msgKey = array_shift( $messageKeyAndParams );
260 $params = $messageKeyAndParams;
261 $message = $context->msg( $msgKey, $params )->parse();
262 $changed[] =
"<li>$message</li>";
266 if ( $this->checkProcessTime() ) {
268 $message = $context->msg(
'translate-manage-toolong' )
269 ->numParams( $this->processingTime )->parse();
270 $changed[] =
"<li>$message</li></ul>";
282 $actions = $this->getActions();
283 $defaction = $this->getDefaultAction( $fuzzy, $action );
290 foreach ( $actions as $action ) {
291 $label = $context->msg(
"translate-manage-action-$action" )->text();
292 $name = self::escapeNameForPHP(
"action-$type-$key" );
293 $id = Sanitizer::escapeIdForAttribute(
"action-$key-$action" );
294 $act[] = Xml::radioLabel( $label, $name, $action, $id, $action === $defaction );
297 $param =
'<code class="mw-tmi-diff">' . htmlspecialchars( $key ) .
'</code>';
298 $name = $context->msg(
'translate-manage-import-diff' )
299 ->rawParams( $param, implode(
' ', $act ) )
302 $changed[] = self::makeSectionElement( $name, $type, $text );
307 $collection->filter(
'hastranslation',
false );
308 $keys = $collection->getMessageKeys();
310 $diff = array_diff( $keys, array_keys( $messages ) );
312 foreach ( $diff as $s ) {
313 $para =
'<code class="mw-tmi-deleted">' . htmlspecialchars( $s ) .
'</code>';
314 $name = $context->msg(
'translate-manage-import-deleted' )->rawParams( $para )->escaped();
315 $text = Utilities::convertWhiteSpaceToHTML( $collection[$s]->translation() );
316 $changed[] = self::makeSectionElement( $name,
'deleted', $text );
320 if ( $process || ( $changed === [] && $code !==
'en' ) ) {
321 if ( $changed === [] ) {
322 $this->out->addWikiMsg(
'translate-manage-nochanges-other' );
325 if ( $changed === [] || strpos( end( $changed ),
'<li>' ) !== 0 ) {
329 $message = $context->msg(
'translate-manage-import-done' )->parse();
330 $changed[] =
"<li>$message</li></ul>";
331 $this->out->addHTML( implode(
"\n", $changed ) );
334 if ( $changed !== [] ) {
335 if ( $code ===
'en' ) {
336 $this->out->addWikiMsg(
'translate-manage-intro-en' );
338 $lang = Utilities::getLanguageName(
340 $context->getLanguage()->getCode()
342 $this->out->addWikiMsg(
'translate-manage-intro-other', $lang );
344 $this->out->addHTML( Html::hidden(
'language', $code ) );
345 $this->out->addHTML( implode(
"\n", $changed ) );
346 $this->out->addHTML( Xml::submitButton( $context->msg(
'translate-manage-submit' )->text() ) );
348 $this->out->addWikiMsg(
'translate-manage-nochanges' );
352 $this->out->addHTML( $this->doFooter() );
378 string $comment =
'',
382 global $wgTranslateDocumentationLanguageCode;
384 $title = self::makeTranslationTitle( $group, $key, $code );
386 if ( $action ===
'import' || $action ===
'conflict' ) {
387 if ( $action ===
'import' ) {
388 $comment = wfMessage(
'translate-manage-import-summary' )->inContentLanguage()->plain();
390 $comment = wfMessage(
'translate-manage-conflict-summary' )->inContentLanguage()->plain();
391 $message = self::makeTextFuzzy( $message );
394 return self::doImport( $title, $message, $comment, $user, $editFlags );
395 } elseif ( $action ===
'ignore' ) {
396 return [
'translate-manage-import-ignore', $key ];
397 } elseif ( $action ===
'fuzzy' && $code !==
'en' &&
398 $code !== $wgTranslateDocumentationLanguageCode
400 $message = self::makeTextFuzzy( $message );
402 return self::doImport( $title, $message, $comment, $user, $editFlags );
403 } elseif ( $action ===
'fuzzy' && $code ===
'en' ) {
404 return self::doFuzzy( $title, $message, $comment, $user, $editFlags );
406 throw new MWException(
"Unhandled action $action" );
410 protected function checkProcessTime() {
411 return wfTimestamp() - $this->time >= $this->processingTime;
425 $wikiPage = MediaWikiServices::getInstance()->getWikiPageFactory()->newFromTitle( $title );
426 $content = ContentHandler::makeContent( $message, $title );
427 $status = $wikiPage->doUserEditContent(
433 $success = $status->isOK();
436 return [
'translate-manage-import-ok',
437 wfEscapeWikiText( $title->getPrefixedText() )
441 $text =
"Failed to import new version of page {$title->getPrefixedText()}\n";
442 $text .= $status->getWikiText();
443 throw new MWException( $text );
454 $context = RequestContext::getMain();
455 $services = MediaWikiServices::getInstance();
457 if ( !$context->getUser()->isAllowed(
'translate-manage' ) ) {
458 return [
'badaccess-group0' ];
463 $user = FuzzyBot::getUser();
468 $titleText = $handle->getKey();
470 $revStore = $services->getRevisionStore();
471 $queryInfo = $revStore->getQueryInfo( [
'page' ] );
472 $dbw = $services->getDBLoadBalancer()->getConnectionRef( DB_PRIMARY );
473 $rows = $dbw->select(
474 $queryInfo[
'tables'],
475 $queryInfo[
'fields'],
477 'page_namespace' => $title->getNamespace(),
478 'page_latest=rev_id',
479 'page_title' . $dbw->buildLike(
"$titleText/", $dbw->anyString() ),
487 $slots = $revStore->getContentBlobsForBatch( $rows, [ SlotRecord::MAIN ] )->getValue();
489 foreach ( $rows as $row ) {
490 global $wgTranslateDocumentationLanguageCode;
492 $ttitle = Title::makeTitle( (
int)$row->page_namespace, $row->page_title );
495 if ( $ttitle->getSubpageText() ===
'en' ||
496 $ttitle->getSubpageText() === $wgTranslateDocumentationLanguageCode
500 } elseif ( isset( $slots[$row->rev_id] ) ) {
501 $slot = $slots[$row->rev_id][SlotRecord::MAIN];
502 $text = self::makeTextFuzzy( $slot->blob_data );
504 $text = self::makeTextFuzzy(
505 Utilities::getTextFromTextContent(
506 $revStore->newRevisionFromRow( $row )->getContent( SlotRecord::MAIN )
512 $changed[] = self::doImport(
523 foreach ( $changed as $c ) {
524 $key = array_shift( $c );
525 $text .=
'* ' . $context->msg( $key, $c )->plain() .
"\n";
528 return [
'translate-manage-import-fuzzy',
"\n" . $text ];
541 $ns = $group->getNamespace();
543 return Title::makeTitleSafe( $ns,
"$key/$code" );
559 Language $lang =
null
561 $containerParams = [
'class' =>
"mw-tpt-sp-section mw-tpt-sp-section-type-{$type}" ];
562 $legendParams = [
'class' =>
'mw-tpt-sp-legend' ];
563 $contentParams = [
'class' =>
'mw-tpt-sp-content' ];
565 $contentParams[
'dir'] = $lang->getDir();
566 $contentParams[
'lang'] = $lang->getCode();
569 $output = Html::rawElement(
'div', $containerParams,
570 Html::rawElement(
'div', $legendParams, $legend ) .
571 Html::rawElement(
'div', $contentParams, $content )
584 $message = str_replace( TRANSLATE_FUZZY,
'', $message );
586 return TRANSLATE_FUZZY . $message;
606 return strtr( $name, $replacements );
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:GroupSynchronizationCache'=> static function(MediaWikiServices $services):GroupSynchronizationCache { return new GroupSynchronizationCache( $services->get( 'Translate:PersistentCache'));}, 'Translate:MessageBundleStore'=> static function(MediaWikiServices $services):MessageBundleStore { return new MessageBundleStore(new RevTagStore(), $services->getJobQueueGroup(), $services->getLanguageNameUtils(), $services->get( 'Translate:MessageIndex'));}, 'Translate:MessageGroupReview'=> static function(MediaWikiServices $services):MessageGroupReview { return new MessageGroupReview($services->getDBLoadBalancer(), $services->getHookContainer());}, 'Translate:MessageGroupStatsTableFactory'=> static function(MediaWikiServices $services):MessageGroupStatsTableFactory { return new MessageGroupStatsTableFactory($services->get( 'Translate:ProgressStatsTableFactory'), $services->getDBLoadBalancer(), $services->getLinkRenderer(), $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:SubpageListBuilder'=> static function(MediaWikiServices $services):SubpageListBuilder { return new SubpageListBuilder($services->get( 'Translate:TranslatableBundleFactory'), $services->getLinkBatchFactory());}, 'Translate:TranslatableBundleFactory'=> static function(MediaWikiServices $services):TranslatableBundleFactory { return new TranslatableBundleFactory($services->get( 'Translate:TranslatablePageStore'), $services->get( 'Translate:MessageBundleStore'));}, '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(), new RevTagStore(), $services->getDBLoadBalancer(), $services->get( 'Translate:TranslatableBundleStatusStore'));}, 'Translate:TranslationStashReader'=> static function(MediaWikiServices $services):TranslationStashReader { $db=$services->getDBLoadBalancer() ->getConnectionRef(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());}, '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