-
-
Notifications
You must be signed in to change notification settings - Fork 99
Try adding translations to the screen #373
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
base: master
Are you sure you want to change the base?
Changes from 2 commits
35d5f80
fc7ec6f
c49eb97
8012c4f
5516d9f
63ee27c
d0cb1b4
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
| Original file line number | Diff line number | Diff line change |
|---|---|---|
| @@ -0,0 +1,140 @@ | ||
| #include "Translation.h" | ||
| #include <stdarg.h> | ||
|
|
||
| Language Translation::currentLanguage = Language::ENGLISH; | ||
|
|
||
| void Translation::setLanguage(Language lang) { | ||
| currentLanguage = lang; | ||
| } | ||
|
|
||
| Language Translation::getLanguage() { | ||
| return currentLanguage; | ||
| } | ||
|
|
||
| const char* Translation::get(TranslationKey key) { | ||
| const char *text = nullptr; | ||
| switch (currentLanguage) { | ||
| case Language::GERMAN: | ||
| text = getGerman(key); | ||
| break; | ||
| case Language::FRENCH: | ||
| text = getFrench(key); | ||
| break; | ||
| case Language::SPANISH: | ||
| text = getSpanish(key); | ||
| break; | ||
| case Language::ENGLISH: | ||
| default: | ||
| text = getEnglish(key); | ||
| break; | ||
| } | ||
| if (text == nullptr || text[0] == '\0') { | ||
| return getEnglish(key); | ||
| } | ||
| return text; | ||
| } | ||
|
|
||
| String Translation::format(const char* format, ...) { | ||
| char buffer[256]; | ||
| va_list args; | ||
| va_start(args, format); | ||
| vsnprintf(buffer, sizeof(buffer), format, args); | ||
| va_end(args); | ||
| return String(buffer); | ||
| } | ||
|
|
||
| const char* Translation::getEnglish(TranslationKey key) { | ||
| switch (key) { | ||
| case TranslationKey::BREW: return "Brew"; | ||
| case TranslationKey::STEAM: return "Steam"; | ||
| case TranslationKey::WATER: return "Water"; | ||
| case TranslationKey::GRIND: return "Grind"; | ||
| case TranslationKey::SELECT_PROFILE: return "Select profile"; | ||
| case TranslationKey::STARTING: return "Starting..."; | ||
| case TranslationKey::UPDATING: return "Updating..."; | ||
| case TranslationKey::TEMPERATURE_ERROR: return "Temperature error, please restart"; | ||
| case TranslationKey::AUTOTUNING: return "Autotuning..."; | ||
| case TranslationKey::FINISHED: return "Finished"; | ||
| case TranslationKey::INFUSION: return "INFUSION"; | ||
| case TranslationKey::BREW_PHASE: return "BREW"; | ||
| case TranslationKey::STEPS: return "Steps"; | ||
| case TranslationKey::PHASES: return "Phases"; | ||
| case TranslationKey::STEP: return "step"; | ||
| case TranslationKey::PHASE: return "phase"; | ||
| case TranslationKey::SELECTED_PROFILE: return "Selected profile"; | ||
| case TranslationKey::RESTART_REQUIRED: return "Restart required"; | ||
| default: return ""; | ||
| } | ||
| } | ||
|
|
||
| const char* Translation::getGerman(TranslationKey key) { | ||
| switch (key) { | ||
| case TranslationKey::BREW: return "Kaffee"; | ||
| case TranslationKey::STEAM: return "Dampf"; | ||
| case TranslationKey::WATER: return "Wasser"; | ||
| case TranslationKey::GRIND: return "Mahlen"; | ||
| case TranslationKey::SELECT_PROFILE: return "Profil auswählen"; | ||
| case TranslationKey::STARTING: return "Starten..."; | ||
| case TranslationKey::UPDATING: return "Aktualisieren..."; | ||
| case TranslationKey::TEMPERATURE_ERROR: return "Temperaturfehler, bitte neu starten"; | ||
| case TranslationKey::AUTOTUNING: return "Autotune..."; | ||
| case TranslationKey::FINISHED: return "Fertig"; | ||
| case TranslationKey::INFUSION: return "INFUSION"; | ||
| case TranslationKey::BREW_PHASE: return "BEZUG"; | ||
| case TranslationKey::STEPS: return "Schritte"; | ||
| case TranslationKey::PHASES: return "Phasen"; | ||
| case TranslationKey::STEP: return "Schritt"; | ||
| case TranslationKey::PHASE: return "Phase"; | ||
| case TranslationKey::SELECTED_PROFILE: return "Gewähltes Profil"; | ||
| case TranslationKey::RESTART_REQUIRED: return "Neustart benötigt"; | ||
| default: return ""; | ||
| } | ||
| } | ||
|
|
||
| const char* Translation::getFrench(TranslationKey key) { | ||
| switch (key) { | ||
| case TranslationKey::BREW: return "Brew"; | ||
| case TranslationKey::STEAM: return "Steam"; | ||
| case TranslationKey::WATER: return "Water"; | ||
| case TranslationKey::GRIND: return "Grind"; | ||
| case TranslationKey::SELECT_PROFILE: return "Select profile"; | ||
| case TranslationKey::STARTING: return "Starting..."; | ||
| case TranslationKey::UPDATING: return "Updating..."; | ||
| case TranslationKey::TEMPERATURE_ERROR: return "Temperature error, please restart"; | ||
| case TranslationKey::AUTOTUNING: return "Autotuning..."; | ||
| case TranslationKey::FINISHED: return "Finished"; | ||
| case TranslationKey::INFUSION: return "INFUSION"; | ||
| case TranslationKey::BREW_PHASE: return "BREW"; | ||
| case TranslationKey::STEPS: return "Steps"; | ||
| case TranslationKey::PHASES: return "Phases"; | ||
| case TranslationKey::STEP: return "step"; | ||
| case TranslationKey::PHASE: return "phase"; | ||
| case TranslationKey::SELECTED_PROFILE: return "Selected profile"; | ||
| case TranslationKey::RESTART_REQUIRED: return "Restart required"; | ||
| default: return ""; | ||
| } | ||
| } | ||
|
|
||
| const char* Translation::getSpanish(TranslationKey key) { | ||
| switch (key) { | ||
| case TranslationKey::BREW: return "Brew"; | ||
| case TranslationKey::STEAM: return "Steam"; | ||
| case TranslationKey::WATER: return "Water"; | ||
| case TranslationKey::GRIND: return "Grind"; | ||
| case TranslationKey::SELECT_PROFILE: return "Select profile"; | ||
| case TranslationKey::STARTING: return "Starting..."; | ||
| case TranslationKey::UPDATING: return "Updating..."; | ||
| case TranslationKey::TEMPERATURE_ERROR: return "Temperature error, please restart"; | ||
| case TranslationKey::AUTOTUNING: return "Autotuning..."; | ||
| case TranslationKey::FINISHED: return "Finished"; | ||
| case TranslationKey::INFUSION: return "INFUSION"; | ||
| case TranslationKey::BREW_PHASE: return "BREW"; | ||
| case TranslationKey::STEPS: return "Steps"; | ||
| case TranslationKey::PHASES: return "Phases"; | ||
| case TranslationKey::STEP: return "step"; | ||
| case TranslationKey::PHASE: return "phase"; | ||
| case TranslationKey::SELECTED_PROFILE: return "Selected profile"; | ||
| case TranslationKey::RESTART_REQUIRED: return "Restart required"; | ||
| default: return ""; | ||
| } | ||
| } | ||
| Original file line number | Diff line number | Diff line change |
|---|---|---|
| @@ -0,0 +1,51 @@ | ||
| #ifndef TRANSLATION_H | ||
| #define TRANSLATION_H | ||
|
|
||
| #include <Arduino.h> | ||
|
|
||
| enum class Language { | ||
| ENGLISH, | ||
| GERMAN, | ||
| FRENCH, | ||
| SPANISH | ||
| }; | ||
|
|
||
| enum class TranslationKey { | ||
| BREW, | ||
| STEAM, | ||
| WATER, | ||
| GRIND, | ||
| SELECT_PROFILE, | ||
| STARTING, | ||
| UPDATING, | ||
| TEMPERATURE_ERROR, | ||
| AUTOTUNING, | ||
| FINISHED, | ||
| INFUSION, | ||
| BREW_PHASE, | ||
| STEPS, | ||
| PHASES, | ||
| STEP, | ||
| PHASE, | ||
| SELECTED_PROFILE, | ||
| RESTART_REQUIRED | ||
| }; | ||
|
|
||
| class Translation { | ||
| public: | ||
| static void setLanguage(Language lang); | ||
| static Language getLanguage(); | ||
| static const char* get(TranslationKey key); | ||
| static String format(const char* format, ...); | ||
|
|
||
| private: | ||
| static Language currentLanguage; | ||
| static const char* getEnglish(TranslationKey key); | ||
| static const char* getGerman(TranslationKey key); | ||
| static const char* getFrench(TranslationKey key); | ||
| static const char* getSpanish(TranslationKey key); | ||
| }; | ||
|
|
||
| #define TR(key) Translation::get(TranslationKey::key) | ||
|
|
||
| #endif // TRANSLATION_H |
| Original file line number | Diff line number | Diff line change | ||||||||||||||||||||||||||||||||||||||||||||||||||||
|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
| @@ -0,0 +1,51 @@ | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
| #include "LanguagePlugin.h" | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
| #include <display/core/Translation.h> | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||
| LanguagePlugin::LanguagePlugin(Controller *controller) : Plugin(controller) { | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
| controller->getPluginManager()->registerPlugin(this); | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
| } | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||
| void LanguagePlugin::init() { | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
| controller->getPluginManager()->on("language:change", [this](Event const &event) { | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
| int language = event.getInt("language"); | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
| setLanguage(language); | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
| }); | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
| } | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||
| void LanguagePlugin::setLanguage(int language) { | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
| Translation::setLanguage(static_cast<Language>(language)); | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
| controller->getSettings().setLanguage(language); | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
| controller->getPluginManager()->emit("ui:refresh", {}); | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
| } | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
Comment on lines
+15
to
+19
Contributor
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. 🛠️ Refactor suggestion Validate and short-circuit language changes to prevent invalid state and unnecessary writes Currently, any integer is cast to Language and persisted. Add range checking and skip if unchanged. void LanguagePlugin::setLanguage(int language) {
- Translation::setLanguage(static_cast<Language>(language));
- controller->getSettings().setLanguage(language);
- controller->getPluginManager()->emit("ui:refresh", {});
+ auto newLang = static_cast<Language>(language);
+ // Clamp to known enum values
+ if (newLang != Language::ENGLISH && newLang != Language::GERMAN &&
+ newLang != Language::FRENCH && newLang != Language::SPANISH) {
+ // Ignore invalid values
+ return;
+ }
+ if (Translation::getLanguage() == newLang) {
+ return; // no-op if same language
+ }
+ Translation::setLanguage(newLang);
+ controller->getSettings().setLanguage(static_cast<int>(newLang));
+ controller->getPluginManager()->emit("ui:refresh", {});
}📝 Committable suggestion
Suggested change
🤖 Prompt for AI Agents |
||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||
| const char* LanguagePlugin::getName() const { | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
| return "Language"; | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
| } | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||
| const char* LanguagePlugin::getDescription() const { | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
| return "Language settings for the display"; | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
| } | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||
| bool LanguagePlugin::isEnabled() const { | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
| return true; | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
| } | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||
| void LanguagePlugin::setEnabled(bool enabled) { | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
| } | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||
| String LanguagePlugin::getConfig() const { | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
| return "{\"language\":" + String(controller->getSettings().getLanguage()) + "}"; | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
| } | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||
| void LanguagePlugin::setConfig(const String &config) { | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
| if (config.indexOf("\"language\":") != -1) { | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
| int start = config.indexOf("\"language\":") + 11; | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
| int end = config.indexOf(",", start); | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
| if (end == -1) end = config.indexOf("}", start); | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
| if (end != -1) { | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
| String langStr = config.substring(start, end); | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
| int language = langStr.toInt(); | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
| setLanguage(language); | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
| } | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
| } | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
| } | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
Comment on lines
+40
to
+51
Contributor
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. 🛠️ Refactor suggestion Make config parsing tolerant to whitespace and validate bounds Parsing by fixed offset fails with spaces and will silently accept invalid values. Trim and bound-check before applying. void LanguagePlugin::setConfig(const String &config) {
- if (config.indexOf("\"language\":") != -1) {
- int start = config.indexOf("\"language\":") + 11;
- int end = config.indexOf(",", start);
- if (end == -1) end = config.indexOf("}", start);
- if (end != -1) {
- String langStr = config.substring(start, end);
- int language = langStr.toInt();
- setLanguage(language);
- }
- }
+ int keyPos = config.indexOf("\"language\"");
+ if (keyPos == -1) return;
+ int colon = config.indexOf(":", keyPos);
+ if (colon == -1) return;
+ int end = config.indexOf(",", colon);
+ if (end == -1) end = config.indexOf("}", colon);
+ if (end == -1) return;
+ String langStr = config.substring(colon + 1, end);
+ langStr.trim();
+ if (langStr.length() == 0) return;
+ int language = langStr.toInt(); // tolerates leading '+'/'-' and spaces
+ setLanguage(language);
}📝 Committable suggestion
Suggested change
🤖 Prompt for AI Agents |
||||||||||||||||||||||||||||||||||||||||||||||||||||||
| Original file line number | Diff line number | Diff line change |
|---|---|---|
| @@ -0,0 +1,22 @@ | ||
| #ifndef LANGUAGE_PLUGIN_H | ||
| #define LANGUAGE_PLUGIN_H | ||
|
|
||
| #include <display/core/Plugin.h> | ||
|
|
||
| class LanguagePlugin : public Plugin { | ||
| public: | ||
| explicit LanguagePlugin(Controller *controller); | ||
|
|
||
| void init() override; | ||
| const char* getName() const override; | ||
| const char* getDescription() const override; | ||
| bool isEnabled() const override; | ||
| void setEnabled(bool enabled) override; | ||
| String getConfig() const override; | ||
| void setConfig(const String &config) override; | ||
|
|
||
| private: | ||
| void setLanguage(int language); | ||
| }; | ||
coderabbitai[bot] marked this conversation as resolved.
Show resolved
Hide resolved
|
||
|
|
||
| #endif // LANGUAGE_PLUGIN_H | ||
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Include cstdio for vsnprintf declaration
vsnprintf is declared in /<stdio.h>. Without it, some toolchains will error.
#include <stdarg.h> +#include <cstdio>📝 Committable suggestion
🤖 Prompt for AI Agents