feat: sort languages in selection menu (#1071)

## Summary

* **What is the goal of this PR?** (e.g., Implements the new feature for
file uploading.)

Currently we are displaying the languages in the order they were added
(as in the `Language` enum). However, as new languages are coming in,
this will quickly be confusing to the users.

But we can't just change the ordering of the enum if we want to respect
bakwards compatibility.

So my proposal is to add a mapping of the alphabetical order of the
languages. I've made it so that it's generated by the `gen_i18n.py`
script, which will be used when a new language is added.


* **What changes are included?**

Added the array from the python script and changed
`LanguageSelectActivity` to use the indices from there. Also commited
the generated `I18nKeys.h`

## Additional Context

* Add any other information that might be helpful for the reviewer
(e.g., performance implications, potential risks,
  specific areas to focus on).

I was wondering if there is a better way to sort it. Currently, it's by
unicode value and Czech and Russian are last, which I don't know it it's
the most intuitive.

The current order is:
`Català, Deutsch, English, Español, Français, Português (Brasil),
Română, Svenska, Čeština, Русский`

---

### AI Usage

While CrossPoint doesn't have restrictions on AI tools in contributing,
please be transparent about their usage as it
helps set the right context for reviewers.

Did you use AI tools to help write this code? _**< PARTIALLY >**_
This commit is contained in:
ariel-lindemann
2026-02-25 17:44:17 +01:00
committed by GitHub
parent b695a48af6
commit 7e214ea760
19 changed files with 68 additions and 34 deletions

View File

@@ -52,7 +52,7 @@ A file looks like this:
```yaml ```yaml
_language_name: "Español" _language_name: "Español"
_language_code: "SPANISH" _language_code: "ES"
_order: "1" _order: "1"
STR_CROSSPOINT: "CrossPoint" STR_CROSSPOINT: "CrossPoint"
@@ -62,7 +62,7 @@ STR_BROWSE_FILES: "Buscar archivos"
**Metadata keys** (prefixed with `_`): **Metadata keys** (prefixed with `_`):
- `_language_name` — Native display name shown to the user (e.g. "Français") - `_language_name` — Native display name shown to the user (e.g. "Français")
- `_language_code` — C++ enum name (e.g. "FRENCH"). Must be a valid C++ identifier. - `_language_code` — C++ enum name (e.g. "FR"). Please use the [ISO Code](https://en.wikipedia.org/wiki/List_of_ISO_639_language_codes) of the language. Must be a valid C++ identifier.
- `_order` — Controls the position in the Language enum (English is always 0) - `_order` — Controls the position in the Language enum (English is always 0)
**Rules:** **Rules:**
@@ -128,7 +128,7 @@ Create `lib/I18n/translations/italian.yaml`:
```yaml ```yaml
_language_name: "Italiano" _language_name: "Italiano"
_language_code: "ITALIAN" _language_code: "IT"
_order: "7" _order: "7"
STR_CROSSPOINT: "CrossPoint" STR_CROSSPOINT: "CrossPoint"
@@ -175,7 +175,7 @@ renderer.drawText(font, x, y, tr(STR_BROWSE_FILES));
Serial.printf("Status: %s\n", tr(STR_CONNECTED)); Serial.printf("Status: %s\n", tr(STR_CONNECTED));
// I18N - Shorthand for I18n::getInstance() // I18N - Shorthand for I18n::getInstance()
I18N.setLanguage(Language::SPANISH); I18N.setLanguage(Language::ES);
Language lang = I18N.getLanguage(); Language lang = I18N.getLanguage();
// === Full API === // === Full API ===
@@ -189,7 +189,7 @@ const char* text = I18N.get(StrId::STR_SETTINGS_TITLE); // Direct call
const char* text = I18N[StrId::STR_SETTINGS_TITLE]; // Operator overload const char* text = I18N[StrId::STR_SETTINGS_TITLE]; // Operator overload
// Set language // Set language
I18N.setLanguage(Language::SPANISH); I18N.setLanguage(Language::ES);
// Get current language // Get current language
Language lang = I18N.getLanguage(); Language lang = I18N.getLanguage();
@@ -201,7 +201,7 @@ I18N.saveSettings();
I18N.loadSettings(); I18N.loadSettings();
// Get character set for font subsetting (static method) // Get character set for font subsetting (static method)
const char* chars = I18n::getCharacterSet(Language::FRENCH); const char* chars = I18n::getCharacterSet(Language::FR);
``` ```
--- ---

View File

@@ -89,7 +89,7 @@ void I18n::loadSettings() {
const char* I18n::getCharacterSet(Language lang) { const char* I18n::getCharacterSet(Language lang) {
const auto langIndex = static_cast<size_t>(lang); const auto langIndex = static_cast<size_t>(lang);
if (langIndex >= static_cast<size_t>(Language::_COUNT)) { if (langIndex >= static_cast<size_t>(Language::_COUNT)) {
lang = Language::ENGLISH; // Fallback to first language lang = Language::EN; // Fallback to first language
} }
return CHARACTER_SETS[static_cast<size_t>(lang)]; return CHARACTER_SETS[static_cast<size_t>(lang)];

View File

@@ -32,7 +32,7 @@ class I18n {
static const char* getCharacterSet(Language lang); static const char* getCharacterSet(Language lang);
private: private:
I18n() : _language(Language::ENGLISH) {} I18n() : _language(Language::EN) {}
Language _language; Language _language;
}; };

View File

@@ -1,5 +1,5 @@
_language_name: "Беларуская" _language_name: "Беларуская"
_language_code: "BELARUSIAN" _language_code: "BE"
_order: "11" _order: "11"
STR_CROSSPOINT: "CrossPoint" STR_CROSSPOINT: "CrossPoint"

View File

@@ -1,5 +1,5 @@
_language_name: "Català" _language_name: "Català"
_language_code: "CATALAN" _language_code: "CA"
_order: "9" _order: "9"
STR_CROSSPOINT: "CrossPoint" STR_CROSSPOINT: "CrossPoint"

View File

@@ -1,5 +1,5 @@
_language_name: "Čeština" _language_name: "Čeština"
_language_code: "CZECH" _language_code: "CS"
_order: "4" _order: "4"
STR_CROSSPOINT: "CrossPoint" STR_CROSSPOINT: "CrossPoint"

View File

@@ -1,5 +1,5 @@
_language_name: "English" _language_name: "English"
_language_code: "ENGLISH" _language_code: "EN"
_order: "0" _order: "0"
STR_CROSSPOINT: "CrossPoint" STR_CROSSPOINT: "CrossPoint"

View File

@@ -1,5 +1,5 @@
_language_name: "Français" _language_name: "Français"
_language_code: "FRENCH" _language_code: "FR"
_order: "2" _order: "2"
STR_CROSSPOINT: "CrossPoint" STR_CROSSPOINT: "CrossPoint"

View File

@@ -1,5 +1,5 @@
_language_name: "Deutsch" _language_name: "Deutsch"
_language_code: "GERMAN" _language_code: "DE"
_order: "3" _order: "3"
STR_CROSSPOINT: "CrossPoint" STR_CROSSPOINT: "CrossPoint"

View File

@@ -1,5 +1,5 @@
_language_name: "Italiano" _language_name: "Italiano"
_language_code: "ITALIAN" _language_code: "IT"
_order: "12" _order: "12"
STR_CROSSPOINT: "CrossPoint" STR_CROSSPOINT: "CrossPoint"

View File

@@ -1,5 +1,5 @@
_language_name: "Português (Brasil)" _language_name: "Português (Brasil)"
_language_code: "PORTUGUESE" _language_code: "PT"
_order: "5" _order: "5"
STR_CROSSPOINT: "CrossPoint" STR_CROSSPOINT: "CrossPoint"

View File

@@ -1,5 +1,5 @@
_language_name: "Română" _language_name: "Română"
_language_code: "ROMANIAN" _language_code: "RO"
_order: "8" _order: "8"
STR_CROSSPOINT: "CrossPoint" STR_CROSSPOINT: "CrossPoint"

View File

@@ -1,5 +1,5 @@
_language_name: "Русский" _language_name: "Русский"
_language_code: "RUSSIAN" _language_code: "RU"
_order: "6" _order: "6"
STR_CROSSPOINT: "CrossPoint" STR_CROSSPOINT: "CrossPoint"

View File

@@ -1,5 +1,5 @@
_language_name: "Español" _language_name: "Español"
_language_code: "SPANISH" _language_code: "ES"
_order: "1" _order: "1"
STR_CROSSPOINT: "CrossPoint" STR_CROSSPOINT: "CrossPoint"

View File

@@ -1,5 +1,5 @@
_language_name: "Svenska" _language_name: "Svenska"
_language_code: "SWEDISH" _language_code: "SV"
_order: "7" _order: "7"
STR_CROSSPOINT: "Crosspoint" STR_CROSSPOINT: "Crosspoint"

View File

@@ -1,5 +1,5 @@
_language_name: "Українська" _language_name: "Українська"
_language_code: "UKRAINIAN" _language_code: "UK"
_order: "10" _order: "10"
STR_CROSSPOINT: "CrossPoint" STR_CROSSPOINT: "CrossPoint"

View File

@@ -9,7 +9,7 @@ Reads YAML files from a translations directory (one file per language) and gener
Each YAML file must contain: Each YAML file must contain:
_language_name: "Native Name" (e.g. "Español") _language_name: "Native Name" (e.g. "Español")
_language_code: "ENUM_NAME" (e.g. "SPANISH") _language_code: "ENUM_NAME" (e.g. "ES")
STR_KEY: "translation text" STR_KEY: "translation text"
The English file is the reference. Missing keys in other languages are The English file is the reference. Missing keys in other languages are
@@ -108,7 +108,7 @@ def load_translations(
) -> Tuple[List[str], List[str], List[str], Dict[str, List[str]]]: ) -> Tuple[List[str], List[str], List[str], Dict[str, List[str]]]:
""" """
Read every YAML file in *translations_dir* and return: Read every YAML file in *translations_dir* and return:
language_codes e.g. ["ENGLISH", "SPANISH", ...] language_codes e.g. ["EN", "ES", ...]
language_names e.g. ["English", "Español", ...] language_names e.g. ["English", "Español", ...]
string_keys ordered list of STR_* keys (from English) string_keys ordered list of STR_* keys (from English)
translations {key: [translation_per_language]} translations {key: [translation_per_language]}
@@ -131,12 +131,12 @@ def load_translations(
# Identify the English file (must exist) # Identify the English file (must exist)
english_file = None english_file = None
for name, data in parsed.items(): for name, data in parsed.items():
if data.get("_language_code", "").upper() == "ENGLISH": if data.get("_language_code", "").upper() == "EN":
english_file = name english_file = name
break break
if english_file is None: if english_file is None:
raise ValueError("No YAML file with _language_code: ENGLISH found") raise ValueError("No YAML file with _language_code: EN found")
# Order: English first, then by _order metadata (falls back to filename) # Order: English first, then by _order metadata (falls back to filename)
def sort_key(fname: str) -> Tuple[int, int, str]: def sort_key(fname: str) -> Tuple[int, int, str]:
@@ -220,7 +220,7 @@ LANG_ABBREVIATIONS = {
"العربية": "AR", "arabic": "AR", "العربية": "AR", "arabic": "AR",
"עברית": "HE", "hebrew": "HE", "עברית": "HE", "hebrew": "HE",
"فارسی": "FA", "persian": "FA", "فارسی": "FA", "persian": "FA",
"čeština": "CZ", "čeština": "CS",
} }
@@ -438,6 +438,31 @@ def generate_keys_header(
"constexpr uint8_t getLanguageCount() " "constexpr uint8_t getLanguageCount() "
"{ return static_cast<uint8_t>(Language::_COUNT); }" "{ return static_cast<uint8_t>(Language::_COUNT); }"
) )
lines.append("")
# Sorted language indices for display order
# (English first, then by language code alphabetically)
english_idx = languages.index("EN")
rest = sorted(
(i for i in range(len(languages)) if i != english_idx),
key=lambda i: languages[i],
)
sorted_indices = [english_idx] + rest
comment_names = ", ".join(language_names[i] for i in sorted_indices)
lines.append("// Sorted language indices by code (auto-generated by gen_i18n.py)")
lines.append(f"// Order: {comment_names}")
lines.append(
"constexpr uint8_t SORTED_LANGUAGE_INDICES[] = {"
f"{', '.join(str(i) for i in sorted_indices)}"
"};"
)
lines.append("")
lines.append(
"static_assert(sizeof(SORTED_LANGUAGE_INDICES) / sizeof(SORTED_LANGUAGE_INDICES[0]) == getLanguageCount(),"
)
lines.append(
' "SORTED_LANGUAGE_INDICES size mismatch");'
)
_write_file(output_path, lines) _write_file(output_path, lines)

View File

@@ -3,16 +3,22 @@
#include <GfxRenderer.h> #include <GfxRenderer.h>
#include <I18n.h> #include <I18n.h>
#include <algorithm>
#include <iterator>
#include "I18nKeys.h"
#include "MappedInputManager.h" #include "MappedInputManager.h"
#include "fontIds.h" #include "fontIds.h"
void LanguageSelectActivity::onEnter() { void LanguageSelectActivity::onEnter() {
Activity::onEnter(); Activity::onEnter();
totalItems = getLanguageCount();
// Set current selection based on current language // Set current selection based on current language
selectedIndex = static_cast<int>(I18N.getLanguage()); const auto currentLang = static_cast<uint8_t>(I18N.getLanguage());
const auto* begin = std::begin(SORTED_LANGUAGE_INDICES);
const auto* end = std::end(SORTED_LANGUAGE_INDICES);
const auto* it = std::find(begin, end, currentLang);
selectedIndex = (it != end) ? std::distance(begin, it) : 0;
requestUpdate(); requestUpdate();
} }
@@ -45,7 +51,7 @@ void LanguageSelectActivity::loop() {
void LanguageSelectActivity::handleSelection() { void LanguageSelectActivity::handleSelection() {
{ {
RenderLock lock(*this); RenderLock lock(*this);
I18N.setLanguage(static_cast<Language>(selectedIndex)); I18N.setLanguage(static_cast<Language>(SORTED_LANGUAGE_INDICES[selectedIndex]));
} }
// Return to previous page // Return to previous page
@@ -61,13 +67,16 @@ void LanguageSelectActivity::render(Activity::RenderLock&&) {
GUI.drawHeader(renderer, Rect{0, metrics.topPadding, pageWidth, metrics.headerHeight}, tr(STR_LANGUAGE)); GUI.drawHeader(renderer, Rect{0, metrics.topPadding, pageWidth, metrics.headerHeight}, tr(STR_LANGUAGE));
// Current language marker
const int contentTop = metrics.topPadding + metrics.headerHeight + metrics.verticalSpacing; const int contentTop = metrics.topPadding + metrics.headerHeight + metrics.verticalSpacing;
const int contentHeight = pageHeight - contentTop - metrics.buttonHintsHeight - metrics.verticalSpacing; const int contentHeight = pageHeight - contentTop - metrics.buttonHintsHeight - metrics.verticalSpacing;
const int currentLang = static_cast<int>(I18N.getLanguage()); const auto currentLang = static_cast<uint8_t>(I18N.getLanguage());
GUI.drawList( GUI.drawList(
renderer, Rect{0, contentTop, pageWidth, contentHeight}, totalItems, selectedIndex, renderer, Rect{0, contentTop, pageWidth, contentHeight}, totalItems, selectedIndex,
[this](int index) { return I18N.getLanguageName(static_cast<Language>(index)); }, nullptr, nullptr, [this](int index) { return I18N.getLanguageName(static_cast<Language>(SORTED_LANGUAGE_INDICES[index])); },
[this, currentLang](int index) { return index == currentLang ? tr(STR_SET) : ""; }, true); nullptr, nullptr,
[this, currentLang](int index) { return SORTED_LANGUAGE_INDICES[index] == currentLang ? tr(STR_SET) : ""; },
true);
// Button hints // Button hints
const auto labels = mappedInput.mapLabels(tr(STR_BACK), tr(STR_SELECT), tr(STR_DIR_UP), tr(STR_DIR_DOWN)); const auto labels = mappedInput.mapLabels(tr(STR_BACK), tr(STR_SELECT), tr(STR_DIR_UP), tr(STR_DIR_DOWN));

View File

@@ -31,5 +31,5 @@ class LanguageSelectActivity final : public Activity {
std::function<void()> onBack; std::function<void()> onBack;
ButtonNavigator buttonNavigator; ButtonNavigator buttonNavigator;
int selectedIndex = 0; int selectedIndex = 0;
int totalItems = 0; constexpr static uint8_t totalItems = getLanguageCount();
}; };