Qt
Internal/Contributor docs for the Qt SDK. Note: These are NOT official API docs; those are found at https://doc.qt.io/
Loading...
Searching...
No Matches
numerus.cpp
Go to the documentation of this file.
1// Copyright (C) 2016 The Qt Company Ltd.
2// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0
3
4#include "translator.h"
5
6#include <QtCore/QByteArray>
7#include <QtCore/QDebug>
8#include <QtCore/QDir>
9#include <QtCore/QFile>
10#include <QtCore/QFileInfo>
11#include <QtCore/QMap>
12
13#include <private/qtranslator_p.h>
14
16
17using namespace Qt::Literals::StringLiterals;
18
19static const uchar englishStyleRules[] =
20 { Q_EQ, 1 };
21static const uchar frenchStyleRules[] =
22 { Q_LEQ, 1 };
23static const uchar latvianRules[] =
24 { Q_MOD_10 | Q_EQ, 1, Q_AND, Q_MOD_100 | Q_NEQ, 11, Q_NEWRULE,
25 Q_NEQ, 0 };
26static const uchar icelandicRules[] =
27 { Q_MOD_10 | Q_EQ, 1, Q_AND, Q_MOD_100 | Q_NEQ, 11 };
28static const uchar irishStyleRules[] =
29 { Q_EQ, 1, Q_NEWRULE,
30 Q_EQ, 2 };
31static const uchar gaelicStyleRules[] =
32 { Q_EQ, 1, Q_OR, Q_EQ, 11, Q_NEWRULE,
33 Q_EQ, 2, Q_OR, Q_EQ, 12, Q_NEWRULE,
34 Q_BETWEEN, 3, 19 };
35static const uchar slovakStyleRules[] =
36 { Q_EQ, 1, Q_NEWRULE,
37 Q_BETWEEN, 2, 4 };
38static const uchar macedonianRules[] =
39 { Q_MOD_10 | Q_EQ, 1, Q_NEWRULE,
40 Q_MOD_10 | Q_EQ, 2 };
41static const uchar lithuanianRules[] =
42 { Q_MOD_10 | Q_EQ, 1, Q_AND, Q_MOD_100 | Q_NEQ, 11, Q_NEWRULE,
43 Q_MOD_10 | Q_NEQ, 0, Q_AND, Q_MOD_100 | Q_NOT_BETWEEN, 10, 19 };
44static const uchar russianStyleRules[] =
45 { Q_MOD_10 | Q_EQ, 1, Q_AND, Q_MOD_100 | Q_NEQ, 11, Q_NEWRULE,
46 Q_MOD_10 | Q_BETWEEN, 2, 4, Q_AND, Q_MOD_100 | Q_NOT_BETWEEN, 10, 19 };
47static const uchar polishRules[] =
48 { Q_EQ, 1, Q_NEWRULE,
49 Q_MOD_10 | Q_BETWEEN, 2, 4, Q_AND, Q_MOD_100 | Q_NOT_BETWEEN, 10, 19 };
50static const uchar romanianRules[] =
51 { Q_EQ, 1, Q_NEWRULE,
52 Q_EQ, 0, Q_OR, Q_MOD_100 | Q_BETWEEN, 1, 19 };
53static const uchar slovenianRules[] =
54 { Q_MOD_100 | Q_EQ, 1, Q_NEWRULE,
55 Q_MOD_100 | Q_EQ, 2, Q_NEWRULE,
56 Q_MOD_100 | Q_BETWEEN, 3, 4 };
57static const uchar malteseRules[] =
58 { Q_EQ, 1, Q_NEWRULE,
59 Q_EQ, 0, Q_OR, Q_MOD_100 | Q_BETWEEN, 1, 10, Q_NEWRULE,
60 Q_MOD_100 | Q_BETWEEN, 11, 19 };
61static const uchar welshRules[] =
62 { Q_EQ, 0, Q_NEWRULE,
63 Q_EQ, 1, Q_NEWRULE,
64 Q_BETWEEN, 2, 5, Q_NEWRULE,
65 Q_EQ, 6 };
66static const uchar arabicRules[] =
67 { Q_EQ, 0, Q_NEWRULE,
68 Q_EQ, 1, Q_NEWRULE,
69 Q_EQ, 2, Q_NEWRULE,
70 Q_MOD_100 | Q_BETWEEN, 3, 10, Q_NEWRULE,
71 Q_MOD_100 | Q_GEQ, 11 };
72static const uchar tagalogRules[] =
73 { Q_LEQ, 1, Q_NEWRULE,
74 Q_MOD_10 | Q_EQ, 4, Q_OR, Q_MOD_10 | Q_EQ, 6, Q_OR, Q_MOD_10 | Q_EQ, 9 };
75
76static const char * const japaneseStyleForms[] = { "Universal Form", 0 };
77static const char * const englishStyleForms[] = { "Singular", "Plural", 0 };
78static const char * const frenchStyleForms[] = { "Singular", "Plural", 0 };
79static const char * const icelandicForms[] = { "Singular", "Plural", 0 };
80static const char * const latvianForms[] = { "Singular", "Plural", "Nullar", 0 };
81static const char * const irishStyleForms[] = { "Singular", "Dual", "Plural", 0 };
82// Gaelic uses the grammatical Singular for the Plural cardinality,
83// so using the Latin terms is expected to cause confusion.
84static const char * const gaelicStyleForms[] = { "1/11", "2/12", "Few", "Many", 0 };
85static const char * const slovakStyleForms[] = { "Singular", "Paucal", "Plural", 0 };
86static const char * const macedonianForms[] = { "Singular", "Dual", "Plural", 0 };
87static const char * const lithuanianForms[] = { "Singular", "Paucal", "Plural", 0 };
88static const char * const russianStyleForms[] = { "Singular", "Dual", "Plural", 0 };
89static const char * const polishForms[] = { "Singular", "Paucal", "Plural", 0 };
90static const char * const romanianForms[] = { "Singular", "Paucal", "Plural", 0 };
91static const char * const slovenianForms[] = { "Singular", "Dual", "Trial", "Plural", 0 };
92static const char * const malteseForms[] =
93 { "Singular", "Paucal", "Greater Paucal", "Plural", 0 };
94static const char * const welshForms[] =
95 { "Nullar", "Singular", "Dual", "Sexal", "Plural", 0 };
96static const char * const arabicForms[] =
97 { "Nullar", "Singular", "Dual", "Minority Plural", "Plural", "Plural (100-102, ...)", 0 };
98static const char * const tagalogForms[] =
99 { "Singular", "Plural (consonant-ended)", "Plural (vowel-ended)", 0 };
100
101#define EOL QLocale::C
102
104 QLocale::Bislama,
105 QLocale::Burmese,
106 QLocale::Chinese,
107 QLocale::Dzongkha,
108 QLocale::Fijian,
109 QLocale::Guarani,
110 QLocale::Hungarian,
111 QLocale::Indonesian,
112 QLocale::Japanese,
113 QLocale::Javanese,
114 QLocale::Korean,
115 QLocale::Malay,
116 QLocale::NauruLanguage,
117 QLocale::Oromo,
118 QLocale::Persian,
119 QLocale::Sundanese,
120 QLocale::Tatar,
121 QLocale::Thai,
122 QLocale::Tibetan,
123 QLocale::Turkish,
124 QLocale::Vietnamese,
125 QLocale::Yoruba,
126 QLocale::Zhuang,
127 EOL
128};
129
131 QLocale::Abkhazian,
132 QLocale::Afar,
133 QLocale::Afrikaans,
134 QLocale::Albanian,
135 QLocale::Amharic,
136 QLocale::Assamese,
137 QLocale::Aymara,
138 QLocale::Azerbaijani,
139 QLocale::Bashkir,
140 QLocale::Basque,
141 QLocale::Bengali,
142 QLocale::Bulgarian,
143 QLocale::Catalan,
144 QLocale::Cornish,
145 QLocale::Corsican,
146 QLocale::Danish,
147 QLocale::Dutch,
148 QLocale::English,
149 QLocale::Esperanto,
150 QLocale::Estonian,
151 QLocale::Faroese,
152 QLocale::Finnish,
153 QLocale::Friulian,
154 QLocale::WesternFrisian,
155 QLocale::Galician,
156 QLocale::Ganda,
157 QLocale::Georgian,
158 QLocale::German,
159 QLocale::Greek,
160 QLocale::Greenlandic,
161 QLocale::Gujarati,
162 QLocale::Hausa,
163 QLocale::Hebrew,
164 QLocale::Hindi,
165 QLocale::Interlingua,
166 QLocale::Interlingue,
167 QLocale::Italian,
168 QLocale::Kannada,
169 QLocale::Kashmiri,
170 QLocale::Kazakh,
171 QLocale::Khmer,
172 QLocale::Kinyarwanda,
173 QLocale::Kirghiz,
174 QLocale::Kurdish,
175 QLocale::Lao,
176 QLocale::Latin,
177 QLocale::Lingala,
178 QLocale::Luxembourgish,
179 QLocale::Malagasy,
180 QLocale::Malayalam,
181 QLocale::Marathi,
182 QLocale::Mongolian,
183 // Missing: Nahuatl,
184 QLocale::Nepali,
185 QLocale::NorthernSotho,
186 QLocale::NorwegianBokmal,
187 QLocale::NorwegianNynorsk,
188 QLocale::Occitan,
189 QLocale::Oriya,
190 QLocale::Pashto,
191 QLocale::Portuguese,
192 QLocale::Punjabi,
193 QLocale::Quechua,
194 QLocale::Romansh,
195 QLocale::Rundi,
196 QLocale::Shona,
197 QLocale::Sindhi,
198 QLocale::Sinhala,
199 QLocale::Somali,
200 QLocale::SouthernSotho,
201 QLocale::Spanish,
202 QLocale::Swahili,
203 QLocale::Swati,
204 QLocale::Swedish,
205 QLocale::Tajik,
206 QLocale::Tamil,
207 QLocale::Telugu,
208 QLocale::Tongan,
209 QLocale::Tsonga,
210 QLocale::Tswana,
211 QLocale::Turkmen,
212 QLocale::Uigur,
213 QLocale::Urdu,
214 QLocale::Uzbek,
215 QLocale::Volapuk,
216 QLocale::Wolof,
217 QLocale::Xhosa,
218 QLocale::Yiddish,
219 QLocale::Zulu,
220 EOL
221};
223 // keep synchronized with frenchStyleCountries
224 QLocale::Armenian,
225 QLocale::Breton,
226 QLocale::French,
227 QLocale::Portuguese,
228 QLocale::Filipino,
229 QLocale::Tigrinya,
230 QLocale::Walloon,
231 EOL
232};
233static const QLocale::Language latvianLanguage[] = { QLocale::Latvian, EOL };
234static const QLocale::Language icelandicLanguage[] = { QLocale::Icelandic, EOL };
236 QLocale::Divehi,
237 QLocale::Inuktitut,
238 QLocale::Inupiak,
239 QLocale::Irish,
240 QLocale::Manx,
241 QLocale::Maori,
242 QLocale::NorthernSami,
243 QLocale::Samoan,
244 QLocale::Sanskrit,
245 EOL
246};
247static const QLocale::Language gaelicStyleLanguages[] = { QLocale::Gaelic, EOL };
248static const QLocale::Language slovakStyleLanguages[] = { QLocale::Slovak, QLocale::Czech, EOL };
249static const QLocale::Language macedonianLanguage[] = { QLocale::Macedonian, EOL };
250static const QLocale::Language lithuanianLanguage[] = { QLocale::Lithuanian, EOL };
252 QLocale::Bosnian,
253 QLocale::Belarusian,
254 QLocale::Croatian,
255 QLocale::Russian,
256 QLocale::Serbian,
257 QLocale::Ukrainian,
258 EOL
259};
260static const QLocale::Language polishLanguage[] = { QLocale::Polish, EOL };
262 QLocale::Romanian,
263 EOL
264};
265static const QLocale::Language slovenianLanguage[] = { QLocale::Slovenian, EOL };
266static const QLocale::Language malteseLanguage[] = { QLocale::Maltese, EOL };
267static const QLocale::Language welshLanguage[] = { QLocale::Welsh, EOL };
268static const QLocale::Language arabicLanguage[] = { QLocale::Arabic, EOL };
269static const QLocale::Language tagalogLanguage[] = { QLocale::Filipino, EOL };
270
272 // keep synchronized with frenchStyleLanguages
273 QLocale::AnyTerritory,
274 QLocale::AnyTerritory,
275 QLocale::AnyTerritory,
276 QLocale::Brazil,
277 QLocale::AnyTerritory,
278 QLocale::AnyTerritory,
279 QLocale::AnyTerritory
280};
282 const uchar *rules;
284 const char * const *forms;
287 const char * const gettextRules;
288};
289
291 { 0, 0, japaneseStyleForms, japaneseStyleLanguages, 0, "nplurals=1; plural=0;" },
292 { englishStyleRules, sizeof(englishStyleRules), englishStyleForms, englishStyleLanguages, 0,
293 "nplurals=2; plural=(n != 1);" },
294 { frenchStyleRules, sizeof(frenchStyleRules), frenchStyleForms, frenchStyleLanguages,
295 frenchStyleCountries, "nplurals=2; plural=(n > 1);" },
296 { latvianRules, sizeof(latvianRules), latvianForms, latvianLanguage, 0,
297 "nplurals=3; plural=(n%10==1 && n%100!=11 ? 0 : n != 0 ? 1 : 2);" },
298 { icelandicRules, sizeof(icelandicRules), icelandicForms, icelandicLanguage, 0,
299 "nplurals=2; plural=(n%10==1 && n%100!=11 ? 0 : 1);" },
300 { irishStyleRules, sizeof(irishStyleRules), irishStyleForms, irishStyleLanguages, 0,
301 "nplurals=3; plural=(n==1 ? 0 : n==2 ? 1 : 2);" },
302 { gaelicStyleRules, sizeof(gaelicStyleRules), gaelicStyleForms, gaelicStyleLanguages, 0,
303 "nplurals=4; plural=(n==1 || n==11) ? 0 : (n==2 || n==12) ? 1 : (n > 2 && n < 20) ? 2 : 3;" },
304 { slovakStyleRules, sizeof(slovakStyleRules), slovakStyleForms, slovakStyleLanguages, 0,
305 "nplurals=3; plural=((n==1) ? 0 : (n>=2 && n<=4) ? 1 : 2);" },
306 { macedonianRules, sizeof(macedonianRules), macedonianForms, macedonianLanguage, 0,
307 "nplurals=3; plural=(n%100==1 ? 0 : n%100==2 ? 1 : 2);" },
308 { lithuanianRules, sizeof(lithuanianRules), lithuanianForms, lithuanianLanguage, 0,
309 "nplurals=3; plural=(n%10==1 && n%100!=11 ? 0 : n%10>=2 && (n%100<10 || n%100>=20) ? 1 : 2);" },
310 { russianStyleRules, sizeof(russianStyleRules), russianStyleForms, russianStyleLanguages, 0,
311 "nplurals=3; plural=(n%10==1 && n%100!=11 ? 0 : n%10>=2 && n%10<=4 && (n%100<10 || n%100>=20) ? 1 : 2);" },
312 { polishRules, sizeof(polishRules), polishForms, polishLanguage, 0,
313 "nplurals=3; plural=(n==1 ? 0 : n%10>=2 && n%10<=4 && (n%100<10 || n%100>=20) ? 1 : 2);" },
314 { romanianRules, sizeof(romanianRules), romanianForms, romanianLanguages, 0,
315 "nplurals=3; plural=(n==1 ? 0 : (n==0 || (n%100 > 0 && n%100 < 20)) ? 1 : 2);" },
316 { slovenianRules, sizeof(slovenianRules), slovenianForms, slovenianLanguage, 0,
317 "nplurals=4; plural=(n%100==1 ? 0 : n%100==2 ? 1 : n%100==3 || n%100==4 ? 2 : 3);" },
318 { malteseRules, sizeof(malteseRules), malteseForms, malteseLanguage, 0,
319 "nplurals=4; plural=(n==1 ? 0 : (n==0 || (n%100>=1 && n%100<=10)) ? 1 : (n%100>=11 && n%100<=19) ? 2 : 3);" },
320 { welshRules, sizeof(welshRules), welshForms, welshLanguage, 0,
321 "nplurals=5; plural=(n==0 ? 0 : n==1 ? 1 : (n>=2 && n<=5) ? 2 : n==6 ? 3 : 4);" },
322 { arabicRules, sizeof(arabicRules), arabicForms, arabicLanguage, 0,
323 "nplurals=6; plural=(n==0 ? 0 : n==1 ? 1 : n==2 ? 2 : (n%100>=3 && n%100<=10) ? 3 : n%100>=11 ? 4 : 5);" },
324 { tagalogRules, sizeof(tagalogRules), tagalogForms, tagalogLanguage, 0,
325 "nplurals=3; plural=(n==1 ? 0 : (n%10==4 || n%10==6 || n%10== 9) ? 1 : 2);" },
326};
327
328static const int NumerusTableSize = sizeof(numerusTable) / sizeof(numerusTable[0]);
329
330bool getNumerusInfo(QLocale::Language language, QLocale::Territory country,
331 QByteArray *rules, QStringList *forms, const char **gettextRules)
332{
333 while (true) {
334 for (int i = 0; i < NumerusTableSize; ++i) {
335 const NumerusTableEntry &entry = numerusTable[i];
336 for (int j = 0; entry.languages[j] != EOL; ++j) {
337 if (entry.languages[j] == language
338 && ((!entry.countries && country == QLocale::AnyTerritory)
339 || (entry.countries && entry.countries[j] == country))) {
340 if (rules) {
341 *rules = QByteArray::fromRawData(reinterpret_cast<const char *>(entry.rules),
342 entry.rulesSize);
343 }
344 if (gettextRules)
345 *gettextRules = entry.gettextRules;
346 if (forms) {
347 forms->clear();
348 for (int k = 0; entry.forms[k]; ++k)
349 forms->append(QLatin1String(entry.forms[k]));
350 }
351 return true;
352 }
353 }
354 }
355
356 if (country == QLocale::AnyTerritory)
357 break;
358 country = QLocale::AnyTerritory;
359 }
360 return false;
361}
362
364{
365 QStringList langs;
366
367 for (int i = 0; i < NumerusTableSize; ++i) {
368 const NumerusTableEntry &entry = numerusTable[i];
369 for (int j = 0; entry.languages[j] != EOL; ++j) {
370 QLocale loc(entry.languages[j], entry.countries ? entry.countries[j]
371 : QLocale::AnyTerritory);
372 QString lang = QLocale::languageToString(entry.languages[j]);
373 if (loc.language() == QLocale::C)
374 lang += " (!!!)"_L1;
375 else if (entry.countries && entry.countries[j] != QLocale::AnyTerritory)
376 lang += " (%1)"_L1.arg(QLocale::territoryToString(loc.territory()));
377 else
378 lang += " [%1]"_L1.arg(QLocale::territoryToString(loc.territory()));
379 langs << QString::fromLatin1("%1 %2 %3\n").arg(lang, -40).arg(loc.name(), -8)
380 .arg(QString::fromLatin1(entry.gettextRules));
381 }
382 }
383 langs.sort();
384 return langs.join(QString());
385}
386
387QT_END_NAMESPACE
Combined button and popup list for selecting options.
static const QLocale::Language englishStyleLanguages[]
Definition numerus.cpp:130
static const char *const russianStyleForms[]
Definition numerus.cpp:88
static const QLocale::Language romanianLanguages[]
Definition numerus.cpp:261
static const uchar welshRules[]
Definition numerus.cpp:61
static const uchar slovenianRules[]
Definition numerus.cpp:53
static const QLocale::Language japaneseStyleLanguages[]
Definition numerus.cpp:103
static const QLocale::Language icelandicLanguage[]
Definition numerus.cpp:234
static const QLocale::Language welshLanguage[]
Definition numerus.cpp:267
static const char *const romanianForms[]
Definition numerus.cpp:90
static const QLocale::Language tagalogLanguage[]
Definition numerus.cpp:269
static const char *const lithuanianForms[]
Definition numerus.cpp:87
static const uchar polishRules[]
Definition numerus.cpp:47
static const QLocale::Language arabicLanguage[]
Definition numerus.cpp:268
static const QLocale::Language polishLanguage[]
Definition numerus.cpp:260
static const QLocale::Territory frenchStyleCountries[]
Definition numerus.cpp:271
#define EOL
Definition numerus.cpp:101
static const QLocale::Language gaelicStyleLanguages[]
Definition numerus.cpp:247
static const QLocale::Language frenchStyleLanguages[]
Definition numerus.cpp:222
static const char *const japaneseStyleForms[]
Definition numerus.cpp:76
static const QLocale::Language macedonianLanguage[]
Definition numerus.cpp:249
static const uchar slovakStyleRules[]
Definition numerus.cpp:35
static const QLocale::Language malteseLanguage[]
Definition numerus.cpp:266
static const QLocale::Language slovenianLanguage[]
Definition numerus.cpp:265
static const char *const slovenianForms[]
Definition numerus.cpp:91
static const uchar tagalogRules[]
Definition numerus.cpp:72
static const uchar romanianRules[]
Definition numerus.cpp:50
static const uchar gaelicStyleRules[]
Definition numerus.cpp:31
static const char *const macedonianForms[]
Definition numerus.cpp:86
static const char *const icelandicForms[]
Definition numerus.cpp:79
static const uchar icelandicRules[]
Definition numerus.cpp:26
static const char *const welshForms[]
Definition numerus.cpp:94
static const char *const tagalogForms[]
Definition numerus.cpp:98
static const int NumerusTableSize
Definition numerus.cpp:328
static const NumerusTableEntry numerusTable[]
Definition numerus.cpp:290
static const char *const irishStyleForms[]
Definition numerus.cpp:81
static const QLocale::Language irishStyleLanguages[]
Definition numerus.cpp:235
static const char *const englishStyleForms[]
Definition numerus.cpp:77
static const char *const polishForms[]
Definition numerus.cpp:89
static const QLocale::Language lithuanianLanguage[]
Definition numerus.cpp:250
static const char *const frenchStyleForms[]
Definition numerus.cpp:78
static const uchar arabicRules[]
Definition numerus.cpp:66
static const uchar macedonianRules[]
Definition numerus.cpp:38
static const uchar irishStyleRules[]
Definition numerus.cpp:28
static const uchar frenchStyleRules[]
Definition numerus.cpp:21
static const uchar lithuanianRules[]
Definition numerus.cpp:41
static const char *const latvianForms[]
Definition numerus.cpp:80
static const uchar latvianRules[]
Definition numerus.cpp:23
static const char *const gaelicStyleForms[]
Definition numerus.cpp:84
static const char *const arabicForms[]
Definition numerus.cpp:96
static const uchar englishStyleRules[]
Definition numerus.cpp:19
static const QLocale::Language latvianLanguage[]
Definition numerus.cpp:233
static const QLocale::Language russianStyleLanguages[]
Definition numerus.cpp:251
static const char *const malteseForms[]
Definition numerus.cpp:92
static const QLocale::Language slovakStyleLanguages[]
Definition numerus.cpp:248
static const uchar malteseRules[]
Definition numerus.cpp:57
static const uchar russianStyleRules[]
Definition numerus.cpp:44
static const char *const slovakStyleForms[]
Definition numerus.cpp:85
const uchar * rules
Definition numerus.cpp:282
const char *const * forms
Definition numerus.cpp:284
const char *const gettextRules
Definition numerus.cpp:287
const QLocale::Language * languages
Definition numerus.cpp:285
const QLocale::Territory * countries
Definition numerus.cpp:286
QString getNumerusInfoString()
Definition numerus.cpp:363
bool getNumerusInfo(QLocale::Language language, QLocale::Territory territory, QByteArray *rules, QStringList *forms, const char **gettextRules)
Definition numerus.cpp:330