6#ifndef LINGUIST_CONSOLE_APPLICATION
14using namespace Qt::Literals::StringLiterals;
18static QString leadingWhitespace(
const QString &str)
21 for (; i < str.size(); i++) {
22 if (!str[i].isSpace()) {
29static QString trailingWhitespace(
const QString &str)
33 if (!str[i].isSpace()) {
37 return str.mid(i + 1);
42 str = str.simplified();
46 switch (str.at(str.size() - 1).unicode()) {
48 if (str.endsWith(
"..."_L1))
72 return lang == QLocale::Greek ? Validator::End_Interrobang : Validator::End_None;
83static bool haveMnemonic(
const QString &str)
85 for (
const ushort *p = (ushort *)str.constData();;) {
96 if (c !=
'&' && c !=
' ' && c !=
'#' && QChar(c).isPrint()) {
98 for (; *p < 256 && isalpha(*p); p++)
100 if (pp == p || *p !=
';')
111static QHash<
int,
int> countPlaceMarkers(
const QString &str)
113 QHash<
int,
int> counts;
114 const QChar *c = str.unicode();
115 const QChar *cend = c + str.size();
117 if (c->unicode() ==
'%') {
118 const QChar *escape_start = ++c;
121 const QChar *escape_end = c;
124 QString::fromRawData(escape_start, escape_end - escape_start).toInt(&ok);
126 counts[markerIndex]++;
136 const QLocale::Language &locale,
137 const QHash<QString, QList<Phrase *>> &phrases)
142 v.m_haveMnemonic.emplace(haveMnemonic(source));
144 v.m_ending.emplace(ending(source, locale));
146 v.m_placeMarkerCounts.emplace(countPlaceMarkers(source));
148 v.m_leadingWhiteSpace.emplace(leadingWhitespace(source));
149 v.m_trailingWhiteSpace.emplace(trailingWhitespace(source));
151#ifndef LINGUIST_CONSOLE_APPLICATION
153 v.m_matchingPhraseTargets.emplace();
154 QString fsource = friendlyString(source);
155 QStringList lookupWords = fsource.split(QLatin1Char(
' '));
157 for (
const QString &s : std::as_const(lookupWords))
158 if (
auto wordPhrases = phrases.find(s); wordPhrases != phrases.constEnd())
159 for (
const Phrase *p : *wordPhrases)
160 if (fsource == friendlyString(p->source()))
161 v.m_matchingPhraseTargets.value()[s].append(friendlyString(p->target()));
169 const QLocale::Language &locale,
170 QList<
bool> countRefNeeds)
173 QMap<ErrorType, QString> errors;
174 for (QStringView translation : std::as_const(translations)) {
175 while (!translation.isEmpty()) {
176 auto sep = translation.indexOf(Translator::BinaryVariantSeparator);
178 sep = translation.size();
179 const QString trans = translation.first(sep).toString();
181 const bool needsRef = msg.isPlural() && countRefNeeds.at(i);
182 errors.insert(validateTranslation(trans, locale, needsRef));
183 translation.slice(std::min(sep + 1, translation.size()));
190QMap<Validator::ErrorType, QString>
Validator::validateTranslation(
const QString &translation,
191 const QLocale::Language &locale,
194 QMap<ErrorType, QString> errors;
195 if (m_haveMnemonic && *m_haveMnemonic != haveMnemonic(translation))
196 errors.insert(*m_haveMnemonic ? MissingAccelerator : SuperfluousAccelerator, translation);
197 if (m_placeMarkerCounts) {
198 if (*m_placeMarkerCounts != countPlaceMarkers(translation))
200 if (needsRef && !translation.contains(QLatin1String(
"%n"))
201 && !translation.contains(QLatin1String(
"%Ln")))
204 if (m_ending && *m_ending != ending(translation, locale))
207 if (m_leadingWhiteSpace
208 && (*m_leadingWhiteSpace != leadingWhitespace(translation)
209 || *m_trailingWhiteSpace != trailingWhitespace(translation)))
211#ifndef LINGUIST_CONSOLE_APPLICATION
212 if (m_matchingPhraseTargets) {
213 const QString ftranslation = friendlyString(translation);
214 for (
auto itr = m_matchingPhraseTargets->cbegin(); itr != m_matchingPhraseTargets->cend();
217 for (
const QString &target : itr.value()) {
218 if (ftranslation.indexOf(target) >= 0) {
bool surroundingWhiteSpace
QMap< ErrorType, QString > validate(QStringList translations, const TranslatorMessage &msg, const QLocale::Language &locale, QList< bool > countRefNeeds)
@ SurroundingWhitespaceDiffers