consider language scripts when determining the locale to use (#10910)

This commit is contained in:
Kyℓe Hensel
2025-03-25 00:27:18 +11:00
committed by GitHub
parent 255db2f170
commit fd49a10f1c
3 changed files with 47 additions and 10 deletions
+2
View File
@@ -57,6 +57,7 @@ _Breaking developer changes, which may affect downstream projects or sites that
* Make features clickable when "Full Fill" rendering style is selected
* Fix calculation of access field placeholders for multi selections ([#9333])
#### :earth_asia: Localization
* Consider language scripts when determining the locale to use ([#10910], thanks [@k-yle])
#### :hourglass: Performance
#### :mortar_board: Walkthrough / Help
#### :rocket: Presets
@@ -74,6 +75,7 @@ _Breaking developer changes, which may affect downstream projects or sites that
[#10843]: https://github.com/openstreetmap/iD/pull/10843
[#10852]: https://github.com/openstreetmap/iD/issues/10852
[#10885]: https://github.com/openstreetmap/iD/issues/10885
[#10910]: https://github.com/openstreetmap/iD/pull/10910
[id-tagging-schema#609]: https://github.com/openstreetmap/id-tagging-schema/issues/609
[@0xatulpatil]: https://github.com/0xatulpatil
[@MohamedAli00949]: https://github.com/MohamedAli00949
+20 -10
View File
@@ -108,11 +108,8 @@ export function coreLocalizer() {
_dataLocales = results[1];
let indexes = results.slice(2);
let requestedLocales = (_preferredLocaleCodes || [])
.concat(utilDetect().browserLocales) // List of locales preferred by the browser in priority order.
.concat(['en']); // fallback to English since it's the only guaranteed complete language
_localeCodes = localesToUseFrom(requestedLocales);
_localeCodes = localizer.localesToUseFrom(_dataLocales);
_localeCode = _localeCodes[0]; // Run iD in the highest-priority locale; the rest are fallbacks
let loadStringsPromises = [];
@@ -139,23 +136,36 @@ export function coreLocalizer() {
};
// Returns the locales from `requestedLocales` supported by iD that we should use
function localesToUseFrom(requestedLocales) {
let supportedLocales = _dataLocales;
/** @param {{ [locale: string]: unknown }} supportedLocales */
localizer.localesToUseFrom = (supportedLocales) => {
const requestedLocales = [
...(_preferredLocaleCodes || []),
...utilDetect().browserLocales, // List of locales preferred by the browser in priority order.
'en', // fallback to English since it's the only guaranteed complete language
];
/** @type {string[]} */
let toUse = [];
for (let i in requestedLocales) {
let locale = requestedLocales[i];
for (const locale of requestedLocales) {
if (supportedLocales[locale]) toUse.push(locale);
if (locale.includes('-')) {
if ('Intl' in window && 'Locale' in window.Intl) {
// Full locale ('es-ES'), add fallback to the base ('es')
const localeObj = new Intl.Locale(locale);
const withoutScript = `${localeObj.language}-${localeObj.region}`;
const base = localeObj.language;
if (supportedLocales[withoutScript]) toUse.push(withoutScript);
if (supportedLocales[base]) toUse.push(base);
} else if (locale.includes('-')) {
// legacy logic: if Intl.Locale is not available
let langPart = locale.split('-')[0];
if (supportedLocales[langPart]) toUse.push(langPart);
}
}
// remove duplicates
return utilArrayUniq(toUse);
}
};
function updateForCurrentLocale() {
if (!_localeCode) return;
+25
View File
@@ -90,4 +90,29 @@ describe('iD.coreLocalizer', function() {
expect(countDecimalPlaces('10')).to.eql(0);
});
});
describe('localesToUseFrom', () => {
const SUPPORTED_LANGS = {
en: true,
'en-AU': true,
fr: true,
zh: true,
'zh-CN': true,
};
it.each([
/* [requested, matching] */
[[], ['en']],
[['en'], ['en']],
[['en-AU'], ['en-AU', 'en']],
[['zh'], ['zh', 'en']],
[['zh-CN'], ['zh-CN', 'zh', 'en']],
[['zh-Hans-CN'], ['zh-CN', 'zh', 'en']],
[['zh-Hans'], ['zh', 'en']],
[['fr-Latn'], ['fr', 'en']],
])('resolves %s to %s', (requested, matching) => {
const localiser = iD.coreLocalizer();
localiser.preferredLocaleCodes(requested);
expect(localiser.localesToUseFrom(SUPPORTED_LANGS)).toStrictEqual(matching);
});
});
});