Translate extension for MediaWiki
 
Loading...
Searching...
No Matches
StatsTable.php
1<?php
2declare( strict_types=1 );
3
4namespace MediaWiki\Extension\Translate\Statistics;
5
6use HtmlArmor;
7use Language;
10use MediaWiki\Html\Html;
11use MediaWiki\Linker\LinkRenderer;
12use MediaWiki\SpecialPage\SpecialPage;
13use Message;
14use MessageGroup;
15use MessageLocalizer;
16use TitleValue;
17use Xml;
18
26 protected TitleValue $translate;
27 private LinkRenderer $linkRenderer;
28 private ConfigHelper $configHelper;
29 private MessageLocalizer $messageLocalizer;
30 protected Language $language;
31 protected string $mainColumnHeader;
33 protected array $extraColumns = [];
34 private MessageGroupMetadata $messageGroupMetadata;
35
36 public function __construct(
37 LinkRenderer $linkRenderer,
38 ConfigHelper $configHelper,
39 MessageLocalizer $messageLocalizer,
40 Language $language,
41 MessageGroupMetadata $messageGroupMetadata
42 ) {
43 $this->translate = SpecialPage::getTitleValueFor( 'Translate' );
44 $this->linkRenderer = $linkRenderer;
45 $this->configHelper = $configHelper;
46 $this->messageLocalizer = $messageLocalizer;
47 $this->language = $language;
48 $this->messageGroupMetadata = $messageGroupMetadata;
49 }
50
58 public function element( string $in, string $bgcolor = '', string $sort = '' ): string {
59 $attributes = [];
60
61 if ( $sort ) {
62 $attributes['data-sort-value'] = $sort;
63 }
64
65 if ( $bgcolor ) {
66 $attributes['style'] = 'background-color: #' . $bgcolor;
67 }
68
69 $element = Html::element( 'td', $attributes, $in );
70
71 return $element;
72 }
73
74 public function getBackgroundColor( float $percentage, bool $fuzzy = false ): string {
75 if ( $fuzzy ) {
76 // Steeper scale for fuzzy
77 // (0), [0-2), [2-4), ... [12-100)
78 $index = min( 7, ceil( 50 * $percentage ) );
79 $colors = [
80 '', 'fedbd7', 'fecec8', 'fec1b9',
81 'fcb5ab', 'fba89d', 'f89b8f', 'f68d81'
82 ];
83 return $colors[ $index ];
84 }
85
86 // https://gka.github.io/palettes/#colors=#36c,#eaf3ff|steps=20|bez=1|coL=1
87 // Color groups for (0-10], (10-20], ... (90-100], (100)
88 $index = (int)floor( $percentage * 10 );
89 $colors = [
90 'eaf3ff', 'e2ebfc', 'dae3fa', 'd2dbf7', 'c9d4f5',
91 'c1ccf2', 'b8c4ef', 'b1bced', 'a8b4ea', '9fade8',
92 '96a6e5'
93 ];
94
95 return $colors[ $index ];
96 }
97
98 private function getMainColumnHeader(): string {
99 return $this->mainColumnHeader;
100 }
101
102 public function setMainColumnHeader( Message $msg ): void {
103 $this->mainColumnHeader = $this->createColumnHeader( $msg );
104 }
105
107 private function createColumnHeader( Message $msg ): string {
108 return Html::element( 'th', [], $msg->text() );
109 }
110
111 public function addExtraColumn( Message $column ): void {
112 $this->extraColumns[] = $column;
113 }
114
116 private function getOtherColumnHeaders(): array {
117 return array_merge( [
118 $this->messageLocalizer->msg( 'translate-total' ),
119 $this->messageLocalizer->msg( 'translate-untranslated' ),
120 $this->messageLocalizer->msg( 'translate-percentage-complete' ),
121 $this->messageLocalizer->msg( 'translate-percentage-proofread' ),
122 $this->messageLocalizer->msg( 'translate-percentage-fuzzy' ),
123 ], $this->extraColumns );
124 }
125
127 public function createHeader(): string {
128 // Create table header
129 $out = Html::openElement(
130 'table',
131 [ 'class' => 'statstable' ]
132 );
133
134 $out .= "\n\t" . Html::openElement( 'thead' );
135 $out .= "\n\t" . Html::openElement( 'tr' );
136
137 $out .= "\n\t\t" . $this->getMainColumnHeader();
138 foreach ( $this->getOtherColumnHeaders() as $label ) {
139 $out .= "\n\t\t" . $this->createColumnHeader( $label );
140 }
141 $out .= "\n\t" . Html::closeElement( 'tr' );
142 $out .= "\n\t" . Html::closeElement( 'thead' );
143 $out .= "\n\t" . Html::openElement( 'tbody' );
144
145 return $out;
146 }
147
154 public function makeTotalRow( Message $message, array $stats ): string {
155 $out = "\t" . Html::openElement( 'tr' );
156 $out .= "\n\t\t" . Html::element( 'td', [], $message->text() );
157 $out .= $this->makeNumberColumns( $stats );
158 $out .= "\n\t" . Xml::closeElement( 'tr' ) . "\n";
159
160 return $out;
161 }
162
167 public function makeNumberColumns( array $stats ): string {
168 $total = $stats[MessageGroupStats::TOTAL];
169 $translated = $stats[MessageGroupStats::TRANSLATED];
170 $fuzzy = $stats[MessageGroupStats::FUZZY];
171 $proofread = $stats[MessageGroupStats::PROOFREAD];
172
173 if ( $total === null ) {
174 $na = "\n\t\t" . Html::element( 'td', [ 'data-sort-value' => -1 ], '...' );
175 $nap = "\n\t\t" . $this->element( '...', 'AFAFAF', '-1' );
176 $out = $na . $na . $nap . $nap;
177
178 return $out;
179 }
180
181 $out = "\n\t\t" . Html::element( 'td',
182 [ 'data-sort-value' => $total ],
183 $this->language->formatNum( $total ) );
184
185 $out .= "\n\t\t" . Html::element( 'td',
186 [ 'data-sort-value' => $total - $translated ],
187 $this->language->formatNum( $total - $translated ) );
188
189 if ( $total === 0 ) {
190 $transRatio = 0;
191 $fuzzyRatio = 0;
192 $proofRatio = 0;
193 } else {
194 $transRatio = $translated / $total;
195 $fuzzyRatio = $fuzzy / $total;
196 $proofRatio = $translated ? $proofread / $translated : 0;
197 }
198
199 $out .= "\n\t\t" . $this->element( $this->formatPercentage( $transRatio, 'floor' ),
200 $this->getBackgroundColor( $transRatio ),
201 sprintf( '%1.5f', $transRatio ) );
202
203 $out .= "\n\t\t" . $this->element( $this->formatPercentage( $proofRatio, 'floor' ),
204 $this->getBackgroundColor( $proofRatio ),
205 sprintf( '%1.5f', $proofRatio ) );
206
207 $out .= "\n\t\t" . $this->element( $this->formatPercentage( $fuzzyRatio, 'ceil' ),
208 $this->getBackgroundColor( $fuzzyRatio, true ),
209 sprintf( '%1.5f', $fuzzyRatio ) );
210
211 return $out;
212 }
213
214 public function makeWorkflowStateCell( ?string $state, MessageGroup $group, string $language ): string {
215 if ( $state === null || $group->getSourceLanguage() === $language ) {
216 return "\n\t\t" . $this->element( '', '', '-1' );
217 }
218
219 $stateConfig = $group->getMessageGroupStates()->getStates();
220 $sortValue = '-1';
221 $stateColor = '';
222 if ( isset( $stateConfig[$state] ) ) {
223 $sortIndex = array_flip( array_keys( $stateConfig ) );
224 $sortValue = $sortIndex[$state] + 1;
225
226 if ( is_string( $stateConfig[$state] ) ) {
227 // BC for old configuration format
228 $stateColor = $stateConfig[$state];
229 } elseif ( isset( $stateConfig[$state]['color'] ) ) {
230 $stateColor = $stateConfig[$state]['color'];
231 }
232 }
233
234 $stateMessage = $this->messageLocalizer->msg( "translate-workflow-state-$state" );
235 $stateText = $stateMessage->isBlank() ? $state : $stateMessage->text();
236
237 return "\n\t\t" . $this->element(
238 $stateText,
239 $stateColor,
240 (string)$sortValue
241 );
242 }
243
250 public function formatPercentage( $num, string $to = 'floor' ): string {
251 $num = $to === 'floor' ? floor( 100 * $num ) : ceil( 100 * $num );
252 $fmt = $this->language->formatNum( $num );
253
254 return $this->messageLocalizer->msg( 'percent', $fmt )->text();
255 }
256
261 private function getGroupLabel( MessageGroup $group ): string {
262 $groupLabel = htmlspecialchars( $group->getLabel() );
263
264 // Bold for meta groups.
265 if ( $group->isMeta() ) {
266 $groupLabel = Html::rawElement( 'b', [], $groupLabel );
267 }
268
269 return $groupLabel;
270 }
271
279 public function makeGroupLink( MessageGroup $group, string $code, array $params ): string {
280 $queryParameters = $params + [
281 'group' => $group->getId(),
282 'language' => $code
283 ];
284
285 return $this->linkRenderer->makeLink(
286 $this->translate,
287 new HtmlArmor( $this->getGroupLabel( $group ) ),
288 [],
289 $queryParameters
290 );
291 }
292
299 public function isExcluded( MessageGroup $group, string $code ): bool {
300 $excluded = null;
301 $groupId = $group->getId();
302
303 $checks = [
304 $groupId,
305 strtok( $groupId, '-' ),
306 '*'
307 ];
308
309 $disabledLanguages = $this->configHelper->getDisabledTargetLanguages();
310
311 foreach ( $checks as $check ) {
312 if ( isset( $disabledLanguages[$check] ) && isset( $disabledLanguages[$check][$code] ) ) {
313 $excluded = $disabledLanguages[$check][$code];
314 }
315
316 if ( $excluded !== null ) {
317 break;
318 }
319 }
320
321 $languages = $group->getTranslatableLanguages();
322 if ( $languages !== null && !isset( $languages[$code] ) ) {
323 $excluded = true;
324 }
325
326 if ( $this->messageGroupMetadata->isExcluded( $groupId, $code ) ) {
327 $excluded = true;
328 }
329
330 return (bool)$excluded;
331 }
332}
return[ 'Translate:AggregateGroupManager'=> static function(MediaWikiServices $services):AggregateGroupManager { return new AggregateGroupManager( $services->getTitleFactory());}, 'Translate:AggregateGroupMessageGroupFactory'=> static function(MediaWikiServices $services):AggregateGroupMessageGroupFactory { return new AggregateGroupMessageGroupFactory($services->get( 'Translate:MessageGroupMetadata'));}, '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:ExternalMessageSourceStateComparator'=> static function(MediaWikiServices $services):ExternalMessageSourceStateComparator { return new ExternalMessageSourceStateComparator(new SimpleStringComparator(), $services->getRevisionLookup(), $services->getPageStore());}, 'Translate:ExternalMessageSourceStateImporter'=> static function(MediaWikiServices $services):ExternalMessageSourceStateImporter { return new ExternalMessageSourceStateImporter($services->get( 'Translate:GroupSynchronizationCache'), $services->getJobQueueGroup(), LoggerFactory::getInstance( 'Translate.GroupSynchronization'), $services->get( 'Translate:MessageIndex'), $services->getTitleFactory(), new ServiceOptions(ExternalMessageSourceStateImporter::CONSTRUCTOR_OPTIONS, $services->getMainConfig()));}, 'Translate:FileBasedMessageGroupFactory'=> static function(MediaWikiServices $services):FileBasedMessageGroupFactory { return new FileBasedMessageGroupFactory(new MessageGroupConfigurationParser(), new ServiceOptions(FileBasedMessageGroupFactory::SERVICE_OPTIONS, $services->getMainConfig()),);}, '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:HookDefinedMessageGroupFactory'=> static function(MediaWikiServices $services):HookDefinedMessageGroupFactory { return new HookDefinedMessageGroupFactory( $services->get( 'Translate:HookRunner'));}, 'Translate:HookRunner'=> static function(MediaWikiServices $services):HookRunner { return new HookRunner( $services->getHookContainer());}, 'Translate:MessageBundleMessageGroupFactory'=> static function(MediaWikiServices $services):MessageBundleMessageGroupFactory { return new MessageBundleMessageGroupFactory($services->get( 'Translate:MessageGroupMetadata'), new ServiceOptions(MessageBundleMessageGroupFactory::SERVICE_OPTIONS, $services->getMainConfig()),);}, 'Translate:MessageBundleStore'=> static function(MediaWikiServices $services):MessageBundleStore { return new MessageBundleStore($services->get( 'Translate:RevTagStore'), $services->getJobQueueGroup(), $services->getLanguageNameUtils(), $services->get( 'Translate:MessageIndex'), $services->get( 'Translate:MessageGroupMetadata'));}, 'Translate:MessageBundleTranslationLoader'=> static function(MediaWikiServices $services):MessageBundleTranslationLoader { return new MessageBundleTranslationLoader( $services->getLanguageFallback());}, 'Translate:MessageGroupMetadata'=> static function(MediaWikiServices $services):MessageGroupMetadata { return new MessageGroupMetadata( $services->getDBLoadBalancer());}, '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->get( 'Translate:MessageGroupMetadata'), $services->getMainConfig() ->get( 'TranslateWorkflowStates') !==false);}, 'Translate:MessageGroupSubscription'=> static function(MediaWikiServices $services):MessageGroupSubscription { return new MessageGroupSubscription($services->get( 'Translate:MessageGroupSubscriptionStore'), $services->getJobQueueGroup(), $services->getUserIdentityLookup(), LoggerFactory::getInstance( 'Translate.MessageGroupSubscription'), new ServiceOptions(MessageGroupSubscription::CONSTRUCTOR_OPTIONS, $services->getMainConfig()));}, 'Translate:MessageGroupSubscriptionHookHandler'=> static function(MediaWikiServices $services):MessageGroupSubscriptionHookHandler { return new MessageGroupSubscriptionHookHandler($services->get( 'Translate:MessageGroupSubscription'), $services->getUserFactory());}, 'Translate:MessageGroupSubscriptionStore'=> static function(MediaWikiServices $services):MessageGroupSubscriptionStore { return new MessageGroupSubscriptionStore( $services->getDBLoadBalancerFactory());}, 'Translate:MessageIndex'=> static function(MediaWikiServices $services):MessageIndex { $params=(array) $services->getMainConfig() ->get( 'TranslateMessageIndex');$class=array_shift( $params);$implementationMap=['HashMessageIndex'=> HashMessageIndex::class, 'CDBMessageIndex'=> CDBMessageIndex::class, 'DatabaseMessageIndex'=> DatabaseMessageIndex::class, 'hash'=> HashMessageIndex::class, 'cdb'=> CDBMessageIndex::class, 'database'=> DatabaseMessageIndex::class,];$messageIndexStoreClass=$implementationMap[$class] ?? $implementationMap['database'];return new MessageIndex(new $messageIndexStoreClass, $services->getMainWANObjectCache(), $services->getJobQueueGroup(), $services->get( 'Translate:HookRunner'), LoggerFactory::getInstance( 'Translate'), $services->getMainObjectStash(), $services->getDBLoadBalancerFactory(), $services->get( 'Translate:MessageGroupSubscription'), new ServiceOptions(MessageIndex::SERVICE_OPTIONS, $services->getMainConfig()),);}, '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'), $services->get( 'Translate:MessageGroupMetadata'));}, 'Translate:RevTagStore'=> static function(MediaWikiServices $services):RevTagStore { return new RevTagStore( $services->getDBLoadBalancer());}, 'Translate:SubpageListBuilder'=> static function(MediaWikiServices $services):SubpageListBuilder { return new SubpageListBuilder($services->get( 'Translate:TranslatableBundleFactory'), $services->getLinkBatchFactory());}, 'Translate:TranslatableBundleDeleter'=> static function(MediaWikiServices $services):TranslatableBundleDeleter { return new TranslatableBundleDeleter($services->getMainObjectStash(), $services->getJobQueueGroup(), $services->get( 'Translate:SubpageListBuilder'), $services->get( 'Translate:TranslatableBundleFactory'));}, '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(), $services->getNamespaceInfo(), $services->getTitleFactory());}, '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->getDBLoadBalancerFactory(), $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:TranslatablePageMarker'=> static function(MediaWikiServices $services):TranslatablePageMarker { return new TranslatablePageMarker($services->getDBLoadBalancer(), $services->getJobQueueGroup(), $services->getLinkRenderer(), MessageGroups::singleton(), $services->get( 'Translate:MessageIndex'), $services->getTitleFormatter(), $services->getTitleParser(), $services->get( 'Translate:TranslatablePageParser'), $services->get( 'Translate:TranslatablePageStore'), $services->get( 'Translate:TranslatablePageStateStore'), $services->get( 'Translate:TranslationUnitStoreFactory'), $services->get( 'Translate:MessageGroupMetadata'), $services->getWikiPageFactory(), $services->get( 'Translate:TranslatablePageView'));}, 'Translate:TranslatablePageMessageGroupFactory'=> static function(MediaWikiServices $services):TranslatablePageMessageGroupFactory { return new TranslatablePageMessageGroupFactory(new ServiceOptions(TranslatablePageMessageGroupFactory::SERVICE_OPTIONS, $services->getMainConfig()),);}, 'Translate:TranslatablePageParser'=> static function(MediaWikiServices $services):TranslatablePageParser { return new TranslatablePageParser($services->get( 'Translate:ParsingPlaceholderFactory'));}, 'Translate:TranslatablePageStateStore'=> static function(MediaWikiServices $services):TranslatablePageStateStore { return new TranslatablePageStateStore($services->get( 'Translate:PersistentCache'), $services->getPageStore());}, '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'), $services->get( 'Translate:MessageGroupMetadata'));}, 'Translate:TranslatablePageView'=> static function(MediaWikiServices $services):TranslatablePageView { return new TranslatablePageView($services->getDBLoadBalancerFactory(), $services->get( 'Translate:TranslatablePageStateStore'), new ServiceOptions(TranslatablePageView::SERVICE_OPTIONS, $services->getMainConfig()));}, 'Translate:TranslateSandbox'=> static function(MediaWikiServices $services):TranslateSandbox { return new TranslateSandbox($services->getUserFactory(), $services->getDBLoadBalancer(), $services->getPermissionManager(), $services->getAuthManager(), $services->getUserGroupManager(), $services->getActorStore(), $services->getUserOptionsManager(), $services->getJobQueueGroup(), $services->get( 'Translate:HookRunner'), new ServiceOptions(TranslateSandbox::CONSTRUCTOR_OPTIONS, $services->getMainConfig()));}, '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
Offers functionality for reading and updating Translate group related metadata.
This class aims to provide efficient mechanism for fetching translation completion stats.
Implements generation of HTML stats table.
makeTotalRow(Message $message, array $stats)
Makes a row with aggregate numbers.
isExcluded(MessageGroup $group, string $code)
Check whether translations in given group in given language has been disabled.
element(string $in, string $bgcolor='', string $sort='')
Statistics table element (heading or regular cell)
formatPercentage( $num, string $to='floor')
Makes a nice print from plain float.
makeGroupLink(MessageGroup $group, string $code, array $params)
Gets the name of group linked to translation tool.
makeNumberColumns(array $stats)
Makes partial row from completion numbers.
A helper class added to work with configuration values of the Translate Extension.
Interface for message groups.
getTranslatableLanguages()
Get all the translatable languages for a group, considering the inclusion and exclusion list.
getId()
Returns the unique identifier for this group.