Commit c30c3d16 authored by Ronan Abhamon's avatar Ronan Abhamon

feat(ui/views/App/Settings/SettingsUi): supports languages changes

parent d21c6bfe
...@@ -859,6 +859,10 @@ Server url not configured.</translation> ...@@ -859,6 +859,10 @@ Server url not configured.</translation>
<source>languagesLabel</source> <source>languagesLabel</source>
<translation>Language</translation> <translation>Language</translation>
</message> </message>
<message>
<source>systemLocale</source>
<translation>System locale</translation>
</message>
</context> </context>
<context> <context>
<name>SettingsWindow</name> <name>SettingsWindow</name>
......
...@@ -869,6 +869,10 @@ Url du serveur non configurée.</translation> ...@@ -869,6 +869,10 @@ Url du serveur non configurée.</translation>
<source>languagesLabel</source> <source>languagesLabel</source>
<translation>Langue</translation> <translation>Langue</translation>
</message> </message>
<message>
<source>systemLocale</source>
<translation>Locale du système</translation>
</message>
</context> </context>
<context> <context>
<name>SettingsWindow</name> <name>SettingsWindow</name>
......
...@@ -29,18 +29,20 @@ ...@@ -29,18 +29,20 @@
#include "../components/settings/SettingsModel.hpp" #include "../components/settings/SettingsModel.hpp"
#include "../components/smart-search-bar/SmartSearchBarModel.hpp" #include "../components/smart-search-bar/SmartSearchBarModel.hpp"
#include "../components/timeline/TimelineModel.hpp" #include "../components/timeline/TimelineModel.hpp"
#include "../utils.hpp"
#include "App.hpp" #include "App.hpp"
#include "DefaultTranslator.hpp"
#include "Logger.hpp" #include "Logger.hpp"
#include <QDir>
#include <QFileSelector> #include <QFileSelector>
#include <QMenu> #include <QMenu>
#include <QQmlComponent>
#include <QQmlContext>
#include <QQuickView>
#include <QTimer> #include <QTimer>
#include <QtDebug> #include <QtDebug>
#define DEFAULT_LOCALE "en"
#define LANGUAGES_PATH ":/languages/" #define LANGUAGES_PATH ":/languages/"
#define WINDOW_ICON_PATH ":/assets/images/linphone_logo.svg" #define WINDOW_ICON_PATH ":/assets/images/linphone_logo.svg"
...@@ -53,45 +55,32 @@ ...@@ -53,45 +55,32 @@
App *App::m_instance = nullptr; App *App::m_instance = nullptr;
App::App (int &argc, char **argv) : QApplication(argc, argv) { inline bool installLocale (App &app, QTranslator &translator, const QLocale &locale) {
this->setApplicationVersion("4.0"); return translator.load(locale, LANGUAGES_PATH) && app.installTranslator(&translator);
if (m_english_translator.load(QLocale(QLocale::English), LANGUAGES_PATH))
installTranslator(&m_english_translator);
else
qWarning("Unable to install english translator.");
// Try to use default locale.
QLocale current_locale = QLocale::system();
if (m_default_translator.load(current_locale, LANGUAGES_PATH)) {
installTranslator(&m_default_translator);
m_locale = current_locale.name();
} else {
qWarning() << QStringLiteral("Unable to found translations for locale: %1.")
.arg(current_locale.name());
}
} }
// ----------------------------------------------------------------------------- App::App (int &argc, char **argv) : QApplication(argc, argv) {
setApplicationVersion("4.0");
QQuickWindow *App::getCallsWindow () const { // List available locales.
return m_calls_window; for (const auto &locale : QDir(LANGUAGES_PATH).entryList())
} m_available_locales << QLocale(locale);
QQuickWindow *App::getMainWindow () const { m_translator = new DefaultTranslator(this);
QQmlApplicationEngine &engine = const_cast<QQmlApplicationEngine &>(m_engine);
return qobject_cast<QQuickWindow *>(engine.rootObjects().at(0));
}
QQuickWindow *App::getSettingsWindow () const { // Try to use system locale.
return m_settings_window; QLocale sys_locale = QLocale::system();
} if (installLocale(*this, *m_translator, sys_locale)) {
m_locale = sys_locale.name();
// ----------------------------------------------------------------------------- qInfo() << QStringLiteral("Use system locale: %1").arg(m_locale);
return;
}
bool App::hasFocus () const { // Use english.
return getMainWindow()->isActive() || m_calls_window->isActive(); m_locale = DEFAULT_LOCALE;
if (!installLocale(*this, *m_translator, QLocale(m_locale)))
qFatal("Unable to install default translator.");
qInfo() << QStringLiteral("Use default locale: %1").arg(m_locale);
} }
// ----------------------------------------------------------------------------- // -----------------------------------------------------------------------------
...@@ -118,9 +107,30 @@ void App::initContentApp () { ...@@ -118,9 +107,30 @@ void App::initContentApp () {
// Init core. // Init core.
CoreManager::init(m_parser.value("config")); CoreManager::init(m_parser.value("config"));
qInfo() << "Core manager initialized."; qInfo() << "Core manager initialized.";
qInfo() << "Activated selectors:" << QQmlFileSelector::get(&m_engine)->selector()->allSelectors(); qInfo() << "Activated selectors:" << QQmlFileSelector::get(&m_engine)->selector()->allSelectors();
// Try to use preferred locale.
{
QString locale = getConfigLocale();
if (!locale.isEmpty()) {
DefaultTranslator *translator = new DefaultTranslator(this);
if (installLocale(*this, *translator, QLocale(locale))) {
// Use config.
m_translator->deleteLater();
m_translator = translator;
m_locale = locale;
qInfo() << QStringLiteral("Use preferred locale: %1").arg(locale);
} else {
// Reset config.
setConfigLocale("");
translator->deleteLater();
}
}
}
// Register types ans make sub windows. // Register types ans make sub windows.
registerTypes(); registerTypes();
createSubWindows(); createSubWindows();
...@@ -177,6 +187,27 @@ void App::parseArgs () { ...@@ -177,6 +187,27 @@ void App::parseArgs () {
// ----------------------------------------------------------------------------- // -----------------------------------------------------------------------------
QQuickWindow *App::getCallsWindow () const {
return m_calls_window;
}
QQuickWindow *App::getMainWindow () const {
QQmlApplicationEngine &engine = const_cast<QQmlApplicationEngine &>(m_engine);
return qobject_cast<QQuickWindow *>(engine.rootObjects().at(0));
}
QQuickWindow *App::getSettingsWindow () const {
return m_settings_window;
}
// -----------------------------------------------------------------------------
bool App::hasFocus () const {
return getMainWindow()->isActive() || m_calls_window->isActive();
}
// -----------------------------------------------------------------------------
void App::registerTypes () { void App::registerTypes () {
qInfo() << "Registering types..."; qInfo() << "Registering types...";
...@@ -323,6 +354,28 @@ void App::setTrayIcon () { ...@@ -323,6 +354,28 @@ void App::setTrayIcon () {
// ----------------------------------------------------------------------------- // -----------------------------------------------------------------------------
QString App::getConfigLocale () const {
return ::Utils::linphoneStringToQString(
CoreManager::getInstance()->getCore()->getConfig()->getString(
SettingsModel::UI_SECTION, "locale", ""
)
);
}
void App::setConfigLocale (const QString &locale) {
CoreManager::getInstance()->getCore()->getConfig()->setString(
SettingsModel::UI_SECTION, "locale", ::Utils::qStringToLinphoneString(locale)
);
emit configLocaleChanged(locale);
}
QString App::getLocale () const {
return m_locale;
}
// -----------------------------------------------------------------------------
void App::quit () { void App::quit () {
if (m_parser.isSet("selftest")) { if (m_parser.isSet("selftest")) {
cout << tr("selftestResult").toStdString() << endl; cout << tr("selftestResult").toStdString() << endl;
......
...@@ -25,7 +25,6 @@ ...@@ -25,7 +25,6 @@
#include "../components/notifier/Notifier.hpp" #include "../components/notifier/Notifier.hpp"
#include "AvatarProvider.hpp" #include "AvatarProvider.hpp"
#include "DefaultTranslator.hpp"
#include "ThumbnailProvider.hpp" #include "ThumbnailProvider.hpp"
#include <QApplication> #include <QApplication>
...@@ -37,10 +36,19 @@ ...@@ -37,10 +36,19 @@
// ============================================================================= // =============================================================================
class DefaultTranslator;
class App : public QApplication { class App : public QApplication {
Q_OBJECT; Q_OBJECT;
Q_PROPERTY(QString configLocale READ getConfigLocale WRITE setConfigLocale NOTIFY configLocaleChanged);
Q_PROPERTY(QString locale READ getLocale CONSTANT);
Q_PROPERTY(QVariantList availableLocales READ getAvailableLocales CONSTANT);
public: public:
void initContentApp ();
void parseArgs ();
QQmlEngine *getEngine () { QQmlEngine *getEngine () {
return &m_engine; return &m_engine;
} }
...@@ -54,16 +62,8 @@ public: ...@@ -54,16 +62,8 @@ public:
bool hasFocus () const; bool hasFocus () const;
void initContentApp ();
Q_INVOKABLE QQuickWindow *getSettingsWindow () const; Q_INVOKABLE QQuickWindow *getSettingsWindow () const;
Q_INVOKABLE QString locale () const {
return m_locale;
}
void parseArgs ();
static void create (int &argc, char **argv) { static void create (int &argc, char **argv) {
if (!m_instance) { if (!m_instance) {
// Instance must be exists before content. // Instance must be exists before content.
...@@ -78,6 +78,9 @@ public: ...@@ -78,6 +78,9 @@ public:
public slots: public slots:
void quit (); void quit ();
signals:
void configLocaleChanged (const QString &locale);
private: private:
App (int &argc, char **argv); App (int &argc, char **argv);
~App () = default; ~App () = default;
...@@ -86,6 +89,15 @@ private: ...@@ -86,6 +89,15 @@ private:
void createSubWindows (); void createSubWindows ();
void setTrayIcon (); void setTrayIcon ();
QString getConfigLocale () const;
void setConfigLocale (const QString &locale);
QString getLocale () const;
QVariantList getAvailableLocales () const {
return m_available_locales;
}
QCommandLineParser m_parser; QCommandLineParser m_parser;
QQmlApplicationEngine m_engine; QQmlApplicationEngine m_engine;
QQmlFileSelector *m_file_selector = nullptr; QQmlFileSelector *m_file_selector = nullptr;
...@@ -93,11 +105,13 @@ private: ...@@ -93,11 +105,13 @@ private:
AvatarProvider m_avatar_provider; AvatarProvider m_avatar_provider;
ThumbnailProvider m_thumbnail_provider; ThumbnailProvider m_thumbnail_provider;
DefaultTranslator m_default_translator;
QTranslator m_english_translator; DefaultTranslator *m_translator = nullptr;
Notifier *m_notifier = nullptr; Notifier *m_notifier = nullptr;
QString m_locale = "en";
QVariantList m_available_locales;
QString m_locale;
QQuickWindow *m_calls_window = nullptr; QQuickWindow *m_calls_window = nullptr;
QQuickWindow *m_settings_window = nullptr; QQuickWindow *m_settings_window = nullptr;
......
...@@ -27,7 +27,7 @@ ...@@ -27,7 +27,7 @@
// ============================================================================= // =============================================================================
DefaultTranslator::DefaultTranslator () { DefaultTranslator::DefaultTranslator (QObject *parent) : QTranslator(parent) {
QDirIterator it(":", QDirIterator::Subdirectories); QDirIterator it(":", QDirIterator::Subdirectories);
while (it.hasNext()) { while (it.hasNext()) {
QFileInfo info(it.next()); QFileInfo info(it.next());
......
...@@ -30,7 +30,7 @@ ...@@ -30,7 +30,7 @@
class DefaultTranslator : public QTranslator { class DefaultTranslator : public QTranslator {
public: public:
DefaultTranslator (); DefaultTranslator (QObject *parent = Q_NULLPTR);
~DefaultTranslator () = default; ~DefaultTranslator () = default;
QString translate ( QString translate (
......
...@@ -37,6 +37,8 @@ class SettingsModel : public QObject { ...@@ -37,6 +37,8 @@ class SettingsModel : public QObject {
public: public:
SettingsModel (QObject *parent = Q_NULLPTR); SettingsModel (QObject *parent = Q_NULLPTR);
static const std::string UI_SECTION;
signals: signals:
void autoAnswerStatusChanged (bool status); void autoAnswerStatusChanged (bool status);
void fileTransferUrlChanged (const QString &url); void fileTransferUrlChanged (const QString &url);
...@@ -49,8 +51,6 @@ private: ...@@ -49,8 +51,6 @@ private:
void setFileTransferUrl (const QString &url); void setFileTransferUrl (const QString &url);
std::shared_ptr<linphone::Config> m_config; std::shared_ptr<linphone::Config> m_config;
static const std::string UI_SECTION;
}; };
#endif // SETTINGS_MODEL_H_ #endif // SETTINGS_MODEL_H_
...@@ -43,9 +43,11 @@ int main (int argc, char *argv[]) { ...@@ -43,9 +43,11 @@ int main (int argc, char *argv[]) {
*/ */
App::create(argc, argv); App::create(argc, argv);
App::getInstance()->parseArgs();
App::getInstance()->initContentApp(); App *app = App::getInstance();
app->parseArgs();
app->initContentApp();
// Run! // Run!
return App::getInstance()->exec(); return app->exec();
} }
...@@ -150,7 +150,7 @@ Rectangle { ...@@ -150,7 +150,7 @@ Rectangle {
// Cast section to integer because Qt converts the // Cast section to integer because Qt converts the
// sectionDate in string!!! // sectionDate in string!!!
text: new Date(section).toLocaleDateString( text: new Date(section).toLocaleDateString(
Qt.locale(App.locale()) Qt.locale(App.locale)
) )
} }
} }
...@@ -237,7 +237,7 @@ Rectangle { ...@@ -237,7 +237,7 @@ Rectangle {
color: ChatStyle.entry.time.color color: ChatStyle.entry.time.color
font.pointSize: ChatStyle.entry.time.fontSize font.pointSize: ChatStyle.entry.time.fontSize
text: $chatEntry.timestamp.toLocaleString( text: $chatEntry.timestamp.toLocaleString(
Qt.locale(App.locale()), Qt.locale(App.locale),
'hh:mm' 'hh:mm'
) )
verticalAlignment: Text.AlignVCenter verticalAlignment: Text.AlignVCenter
......
...@@ -160,7 +160,7 @@ ColumnLayout { ...@@ -160,7 +160,7 @@ ColumnLayout {
anchors.fill: parent anchors.fill: parent
sourceComponent: TooltipArea { sourceComponent: TooltipArea {
text: $timelineEntry.timestamp.toLocaleString( text: $timelineEntry.timestamp.toLocaleString(
Qt.locale(App.locale()), Qt.locale(App.locale),
Locale.ShortFormat Locale.ShortFormat
) )
} }
......
...@@ -230,7 +230,7 @@ function _indexFinder (array, cb, context) { ...@@ -230,7 +230,7 @@ function _indexFinder (array, cb, context) {
var length = array.length var length = array.length
for (var i = 0; i < length; i++) { for (var i = 0; i < length; i++) {
if (cb(array[index], index, array)) { if (cb(array[i], i, array)) {
return i return i
} }
} }
...@@ -274,6 +274,12 @@ function basename (str) { ...@@ -274,6 +274,12 @@ function basename (str) {
// ----------------------------------------------------------------------------- // -----------------------------------------------------------------------------
function capitalizeFirstLetter (str) {
return str.charAt(0).toUpperCase() + str.slice(1)
}
// -----------------------------------------------------------------------------
function dirname (str) { function dirname (str) {
var str2 = str var str2 = str
var length = str2.length - 1 var length = str2.length - 1
...@@ -342,6 +348,15 @@ function find (obj, cb, context) { ...@@ -342,6 +348,15 @@ function find (obj, cb, context) {
// ----------------------------------------------------------------------------- // -----------------------------------------------------------------------------
function findIndex (array, cb, context) {
cb = _computeOptimizedCb(cb, context)
var key = _indexFinder(array, cb, context)
return key != null && key !== -1 ? key : null
}
// -----------------------------------------------------------------------------
function formatElapsedTime (seconds) { function formatElapsedTime (seconds) {
seconds = parseInt(seconds, 10) seconds = parseInt(seconds, 10)
......
import QtQuick 2.7 import QtQuick 2.7
import Common 1.0 import Common 1.0
import Linphone 1.0
import Utils 1.0
import App.Styles 1.0 import App.Styles 1.0
...@@ -24,10 +26,49 @@ TabContainer { ...@@ -24,10 +26,49 @@ TabContainer {
label: qsTr('languagesLabel') label: qsTr('languagesLabel')
ComboBox { ComboBox {
model: ListModel { function _getAvailableLocales () {
ListElement { key: 'English'; value: 0 } var locales = []
ListElement { key: 'Français'; value: 1 }
App.availableLocales.forEach(function (locale) {
locales.push({
key: Utils.capitalizeFirstLetter(locale.nativeLanguageName),
value: locale.name
})
})
return locales.sort(function (a, b) {
return a > b
})
} }
model: ListModel {}
Component.onCompleted: {
var locales = _getAvailableLocales()
model.append({
key: qsTr('systemLocale'),
value: ''
})
locales.forEach(function (locale) {
model.append(locale)
})
var locale = App.configLocale
if (!locale.length) {
currentIndex = 0
return
}
var value = Qt.locale(locale).name
var index = Utils.findIndex(locales, function (locale) {
return locale.value === value
})
currentIndex = index != null ? index + 1 : 0
}
onActivated: App.configLocale = model.get(index).value
} }
} }
} }
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment