Table of contents
- Why this repository
- Class's features
- Code documentation
- How to use it
- Use example
- About dialog panels instant translation at running time
- About Cmake + Qt5LinguistTools
The Qt framework provide a fantastic module and tool chain to easily implement a multilingual user interface. Anyway when I first used them to implement a multilingual application, I felt that some aspects of my code where a bit messy :
- The language list is obtaind from qm file name analysis...
- No way to retrieve the QTranslator object from qApp once set. (If there is actually a way to do that, let me know)
So I decided to do a simple class in charge of everything about languages / translations in my apps. That is : deal with .qm files, language switch, language choice storing in user settings, system language...
Here is that class: LlmQtLocaleLanguageManager. The class is here contained in a simple Cmake project which is also a demonstration of use. The demo has a minimalist QMainWindow and two different language setting panels. The demo contains unit tests and a Doxygen doc generation, and is compilable with Windows and Linux. You can use / copy everything in here, especially if you face some issues with QTranslator, QLocale and qm files. Everything here is under MIT license.
I'm looking forward to hearing any comment and improvement of my code, so do not hesitate to do so: <jon.petrissans[at]protonmail.com>.
LlmQtLocaleLanguageManager is an utilitarian class that simplifies language choice management in a multilingual Qt application. Features are:
Qm files |
---|
- Qm files analyse, with clear error message in case of issue. Extraction of all the necessary informations. (The Qm file is Qt's app translation source file) - In case of incorrect language / country association, or incorrect language code, an error is raised (unlike QLocale behaviour which continues running without saying anything). |
System language |
---|
- System languages analyse, and confrontation to application's translations. |
App language choice and switch |
---|
- Provide the list of available languages. Languages are called in native form (English, Français, 简体中文...), with associated country if any. - App language switch: Translator installation in qApp and set of the default QLocale. - Instant translation at running time is possible. |
Settings saving and loading |
---|
- Save the chosen language in a QSettings (ini/conf file or registry) (If feature enabled) - At start up, load setting's language in qApp. (If feature enabled) |
Class's Doxygen documentation is here.
If you want the code documentation of all the demo project, just build the cmake project in your computer.
Prerequisites : C++11 (for a smart pointer), Qt5 framework, Qt5LinguistTools
- Copy the 4 files in /src/localeLanguageManager/ somewhere in your project. (LlmQtLocaleLanguageManager.h LlmQtLocaleLanguageManager.cpp LlmQtLanguage.h LlmQtLanguage.cpp)
- Include LlmQtLocaleLanguageManager.h where you want to use it.
- I suggest to create a single instance of the class in the main() function, and to send it to the sub-objects using it.
The LlmQtLocaleLanguageManager object has to be hold "alive" during all the application run time,
because it contains a pointer to the active QTranslator.
- At application start up, the LlmQtLocaleLanguageManager object needs some configurations:
- A QSettings object in which language choice is stored (optional)
- Application's source (primary) interface language.
- .qm files location.
- Then, the object can give many information:
- It indicates if it's well configured, if it has locale's translations (to enable or not the language settings menu).
- A list of available languages/translations for the application.
- System's language.
- Currently chosen language.
- The Object provides different methods to dynamically switch to a language at running time.
-
Translate to user's setting's language.
-
Translate to another language.
-
Translate to system language.
Optionally, the object is in charge of storing choices in user's settings.
LlmQtLocaleLanguageManager languageManager;
languageManager.setSettings(userSettings);
languageManager.setKeySettingsTranslateLanguage(SETTINGS_KEY_LOCALE_LANGUAGE);
languageManager.setSourceLanguage(QLocale::English);
// alternative to setSourceLanguage:
//languageManager.setSourceLanguageFromQAppLocalSettings();
sQmFilesPath = .....;
languageManager.setQmFilesDirectory(sQmFilesPath);
if (languageManager.analyseAvailableLanguages(&sErrMsg) == false)
{ ... }
if (languageManager.translateAppToUserSettingsLanguage(&sErrMsg) == false)
{ ... }
// Enable access to language menu if m_languageManager is usable.
// Hence the language setting panel is accessible only if we have qm files !!
menu->setEnabled(m_languageManager->isUsable());
QStringList languagesList = m_languageManager->languageNameNativeList(false);
...
if ( m_languageManager->translateAppToLanguage(sLanguageChosen, &sErrMsg) == false)
{ ... }
...
if (m_languageManager->translateAppToSystemLanguage(&sErrMsg) == false)
{ ... }
To achieve a straight translation when the .translateAppToLanguage() method is called, Each QDialog of the application has to override the .changeEvent() method, as in the following code. When a new translator is set up, a QEvent::LanguageChange is send to the dialogs.
// Override of QWidget::changeEvent, to implement the dynamic translation of the ui
// param [in] event UI's state change
void MainWindow::changeEvent(QEvent* event)
{
if (event->type() == QEvent::LanguageChange)
{
// retranslate qtDesigner form
ui->retranslateUi(this);
// retranslate other widgets which weren't added in designer, if needed
this->updateWidgets();
}
// call of base class implementation
QWidget::changeEvent(event);
}
That point was not easy to set up, so it may help some people.
You can find an example of integration of Qt's linguistic tools (lupdate and lrelease) into CMake's build. Thanks to Cmake's QT5_CREATE_TRANSLATION() command, each build updates the translation files (.ts) and generates the translation binaries (.qm). The translation files are updated according to added and deleted translations in the sources. The CMakeLists.txt file managing all the above is in /data/locale/.
In addition, you will notice that I found a way to add new .ts file through Cmake's cache.