Skip to content
Snippets Groups Projects
UserSettingsPage.cpp 48.4 KiB
Newer Older
  • Learn to ignore specific revisions
  • /*
     * nheko Copyright (C) 2017  Konstantinos Sideris <siderisk@auth.gr>
     *
     * This program is free software: you can redistribute it and/or modify
     * it under the terms of the GNU General Public License as published by
     * the Free Software Foundation, either version 3 of the License, or
     * (at your option) any later version.
     *
     * This program is distributed in the hope that it will be useful,
     * but WITHOUT ANY WARRANTY; without even the implied warranty of
     * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
     * GNU General Public License for more details.
     *
     * You should have received a copy of the GNU General Public License
     * along with this program.  If not, see <http://www.gnu.org/licenses/>.
     */
    
    
    #include <QApplication>
    
    #include <QComboBox>
    
    #include <QCoreApplication>
    
    #include <QFontComboBox>
    
    #include <QFormLayout>
    
    #include <QLabel>
    
    #include <QLineEdit>
    #include <QMessageBox>
    
    Nicolas Werner's avatar
    Nicolas Werner committed
    #include <QPainter>
    
    #include <QProcessEnvironment>
    
    #include <QPushButton>
    
    #include <QScrollArea>
    
    #include <QSettings>
    
    #include <QStandardPaths>
    
    #include <QTextStream>
    
    #include <QtQml>
    
    #include "Config.h"
    
    #include "MatrixClient.h"
    #include "Olm.h"
    
    #include "UserSettingsPage.h"
    
    #include "Utils.h"
    
    trilene's avatar
    trilene committed
    #include "WebRTCSession.h"
    
    #include "ui/FlatButton.h"
    #include "ui/ToggleButton.h"
    
    Konstantinos Sideris's avatar
    Konstantinos Sideris committed
    #include "config/nheko.h"
    
    Joe Donofry's avatar
    Joe Donofry committed
    UserSettings::UserSettings() { load(); }
    
    
    void
    UserSettings::load()
    {
            QSettings settings;
    
            tray_                    = settings.value("user/window/tray", false).toBool();
            hasDesktopNotifications_ = settings.value("user/desktop_notifications", true).toBool();
    
            hasAlertOnNotification_  = settings.value("user/alert_on_notification", false).toBool();
    
            startInTray_             = settings.value("user/window/start_in_tray", false).toBool();
            groupView_               = settings.value("user/group_view", true).toBool();
            buttonsInTimeline_       = settings.value("user/timeline/buttons", true).toBool();
            timelineMaxWidth_        = settings.value("user/timeline/max_width", 0).toInt();
            messageHoverHighlight_ =
    
              settings.value("user/timeline/message_hover_highlight", false).toBool();
    
            enlargeEmojiOnlyMessages_ =
    
              settings.value("user/timeline/enlarge_emoji_only_msg", false).toBool();
    
            markdown_            = settings.value("user/markdown_enabled", true).toBool();
            typingNotifications_ = settings.value("user/typing_notifications", true).toBool();
            sortByImportance_    = settings.value("user/sort_by_unread", true).toBool();
            readReceipts_        = settings.value("user/read_receipts", true).toBool();
            theme_               = settings.value("user/theme", defaultTheme_).toString();
            font_                = settings.value("user/font_family", "default").toString();
            avatarCircles_       = settings.value("user/avatar_circles", true).toBool();
            decryptSidebar_      = settings.value("user/decrypt_sidebar", true).toBool();
    
            shareKeysWithTrustedUsers_ =
              settings.value("user/share_keys_with_trusted_users", true).toBool();
    
            mobileMode_   = settings.value("user/mobile_mode", false).toBool();
    
            emojiFont_    = settings.value("user/emoji_font_family", "default").toString();
            baseFontSize_ = settings.value("user/font_size", QFont().pointSizeF()).toDouble();
    
            presence_ =
              settings.value("user/presence", QVariant::fromValue(Presence::AutomaticPresence))
                .value<Presence>();
    
            ringtone_         = settings.value("user/ringtone", "Default").toString();
    
    trilene's avatar
    trilene committed
            microphone_       = settings.value("user/microphone", QString()).toString();
            camera_           = settings.value("user/camera", QString()).toString();
            cameraResolution_ = settings.value("user/camera_resolution", QString()).toString();
            cameraFrameRate_  = settings.value("user/camera_frame_rate", QString()).toString();
            useStunServer_    = settings.value("user/use_stun_server", false).toBool();
    
    void
    UserSettings::setMessageHoverHighlight(bool state)
    {
    
            if (state == messageHoverHighlight_)
    
            messageHoverHighlight_ = state;
    
            emit messageHoverHighlightChanged(state);
            save();
    }
    void
    UserSettings::setEnlargeEmojiOnlyMessages(bool state)
    {
    
            if (state == enlargeEmojiOnlyMessages_)
    
            enlargeEmojiOnlyMessages_ = state;
    
            emit enlargeEmojiOnlyMessagesChanged(state);
            save();
    }
    void
    UserSettings::setTray(bool state)
    {
    
            if (state == tray_)
    
            tray_ = state;
    
            emit trayChanged(state);
            save();
    }
    
    void
    UserSettings::setStartInTray(bool state)
    {
    
            if (state == startInTray_)
    
            startInTray_ = state;
    
    void
    UserSettings::setMobileMode(bool state)
    {
            if (state == mobileMode_)
                    return;
            mobileMode_ = state;
            emit mobileModeChanged(state);
            save();
    }
    
    
            if (groupView_ != state)
    
            groupView_ = state;
    
    UserSettings::setMarkdown(bool state)
    
            if (state == markdown_)
    
            markdown_ = state;
    
            emit markdownChanged(state);
            save();
    }
    
    void
    UserSettings::setReadReceipts(bool state)
    {
    
            if (state == readReceipts_)
    
            readReceipts_ = state;
    
            emit readReceiptsChanged(state);
            save();
    }
    
    void
    UserSettings::setTypingNotifications(bool state)
    {
    
            if (state == typingNotifications_)
    
            typingNotifications_ = state;
    
            emit typingNotificationsChanged(state);
            save();
    }
    
    void
    UserSettings::setSortByImportance(bool state)
    {
            if (state == sortByImportance_)
                    return;
            sortByImportance_ = state;
            emit roomSortingChanged(state);
            save();
    }
    
    void
    UserSettings::setButtonsInTimeline(bool state)
    {
    
            if (state == buttonsInTimeline_)
    
            buttonsInTimeline_ = state;
    
            emit buttonInTimelineChanged(state);
            save();
    }
    
    void
    UserSettings::setTimelineMaxWidth(int state)
    {
            if (state == timelineMaxWidth_)
                    return;
            timelineMaxWidth_ = state;
            emit timelineMaxWidthChanged(state);
            save();
    }
    
    void
    UserSettings::setDesktopNotifications(bool state)
    {
            if (state == hasDesktopNotifications_)
                    return;
            hasDesktopNotifications_ = state;
            emit desktopNotificationsChanged(state);
            save();
    }
    
    
    void
    UserSettings::setAlertOnNotification(bool state)
    {
            if (state == hasAlertOnNotification_)
                    return;
            hasAlertOnNotification_ = state;
            emit alertOnNotificationChanged(state);
            save();
    }
    
    
    void
    UserSettings::setAvatarCircles(bool state)
    {
            if (state == avatarCircles_)
                    return;
            avatarCircles_ = state;
            emit avatarCirclesChanged(state);
            save();
    }
    
    void
    UserSettings::setDecryptSidebar(bool state)
    {
            if (state == decryptSidebar_)
                    return;
            decryptSidebar_ = state;
            emit decryptSidebarChanged(state);
            save();
    }
    
    void
    UserSettings::setFontSize(double size)
    {
    
    Joe Donofry's avatar
    Joe Donofry committed
    void
    UserSettings::setFontFamily(QString family)
    {
    
    Joe Donofry's avatar
    Joe Donofry committed
            font_ = family;
    
    Joe Donofry's avatar
    Joe Donofry committed
            save();
    }
    
    
    void
    UserSettings::setEmojiFontFamily(QString family)
    {
    
            emojiFont_ = family;
    
    void
    UserSettings::setPresence(Presence state)
    {
            if (state == presence_)
                    return;
            presence_ = state;
            emit presenceChanged(state);
            save();
    }
    
    
    void
    UserSettings::setTheme(QString theme)
    {
    
            if (theme == theme_)
    
            theme_ = theme;
            save();
            applyTheme();
    
    trilene's avatar
    trilene committed
    void
    UserSettings::setUseStunServer(bool useStunServer)
    {
            if (useStunServer == useStunServer_)
                    return;
            useStunServer_ = useStunServer;
            emit useStunServerChanged(useStunServer);
            save();
    }
    
    
    void
    UserSettings::setShareKeysWithTrustedUsers(bool shareKeys)
    {
            if (shareKeys == shareKeysWithTrustedUsers_)
                    return;
    
            shareKeysWithTrustedUsers_ = shareKeys;
            emit shareKeysWithTrustedUsersChanged(shareKeys);
            save();
    }
    
    
    void
    UserSettings::setRingtone(QString ringtone)
    {
            if (ringtone == ringtone_)
                    return;
            ringtone_ = ringtone;
            emit ringtoneChanged(ringtone);
            save();
    }
    
    
    void
    
    trilene's avatar
    trilene committed
    UserSettings::setMicrophone(QString microphone)
    
    trilene's avatar
    trilene committed
            if (microphone == microphone_)
    
                    return;
    
    trilene's avatar
    trilene committed
            microphone_ = microphone;
            emit microphoneChanged(microphone);
            save();
    }
    
    void
    UserSettings::setCamera(QString camera)
    {
            if (camera == camera_)
                    return;
            camera_ = camera;
            emit cameraChanged(camera);
            save();
    }
    
    void
    UserSettings::setCameraResolution(QString resolution)
    {
            if (resolution == cameraResolution_)
                    return;
            cameraResolution_ = resolution;
            emit cameraResolutionChanged(resolution);
            save();
    }
    
    void
    UserSettings::setCameraFrameRate(QString frameRate)
    {
            if (frameRate == cameraFrameRate_)
                    return;
            cameraFrameRate_ = frameRate;
            emit cameraFrameRateChanged(frameRate);
    
    void
    UserSettings::applyTheme()
    {
            QFile stylefile;
    
    
    Nicolas Werner's avatar
    Nicolas Werner committed
            static QPalette original;
            if (this->theme() == "light") {
    
                    stylefile.setFileName(":/styles/styles/nheko.qss");
    
                    QPalette lightActive(
                      /*windowText*/ QColor("#333"),
                      /*button*/ QColor("#333"),
                      /*light*/ QColor(0xef, 0xef, 0xef),
    
                      /*dark*/ QColor(110, 110, 110),
                      /*mid*/ QColor(220, 220, 220),
    
                      /*text*/ QColor("#333"),
                      /*bright_text*/ QColor("#333"),
    
                      /*base*/ QColor("#fff"),
    
                      /*window*/ QColor("white"));
    
                    lightActive.setColor(QPalette::AlternateBase, QColor("#eee"));
    
                    lightActive.setColor(QPalette::Highlight, QColor("#38a3d8"));
    
    Nicolas Werner's avatar
    Nicolas Werner committed
                    lightActive.setColor(QPalette::ToolTipBase, lightActive.base().color());
                    lightActive.setColor(QPalette::ToolTipText, lightActive.text().color());
                    lightActive.setColor(QPalette::Link, QColor("#0077b5"));
    
                    lightActive.setColor(QPalette::ButtonText, QColor("#495057"));
    
    Nicolas Werner's avatar
    Nicolas Werner committed
                    QApplication::setPalette(lightActive);
            } else if (this->theme() == "dark") {
    
                    stylefile.setFileName(":/styles/styles/nheko-dark.qss");
    
                    QPalette darkActive(
                      /*windowText*/ QColor("#caccd1"),
                      /*button*/ QColor(0xff, 0xff, 0xff),
                      /*light*/ QColor("#caccd1"),
    
                      /*dark*/ QColor(110, 110, 110),
                      /*mid*/ QColor("#202228"),
    
                      /*text*/ QColor("#caccd1"),
                      /*bright_text*/ QColor(0xff, 0xff, 0xff),
    
                      /*base*/ QColor("#202228"),
                      /*window*/ QColor("#2d3139"));
                    darkActive.setColor(QPalette::AlternateBase, QColor("#2d3139"));
    
    Nicolas Werner's avatar
    Nicolas Werner committed
                    darkActive.setColor(QPalette::Highlight, QColor("#38a3d8"));
                    darkActive.setColor(QPalette::ToolTipBase, darkActive.base().color());
                    darkActive.setColor(QPalette::ToolTipText, darkActive.text().color());
                    darkActive.setColor(QPalette::Link, QColor("#38a3d8"));
    
                    darkActive.setColor(QPalette::ButtonText, "#727274");
    
    Nicolas Werner's avatar
    Nicolas Werner committed
                    QApplication::setPalette(darkActive);
    
            } else {
                    stylefile.setFileName(":/styles/styles/system.qss");
    
    Nicolas Werner's avatar
    Nicolas Werner committed
                    QApplication::setPalette(original);
    
            }
    
            stylefile.open(QFile::ReadOnly);
            QString stylesheet = QString(stylefile.readAll());
    
            qobject_cast<QApplication *>(QApplication::instance())->setStyleSheet(stylesheet);
    
    }
    
    void
    UserSettings::save()
    {
            QSettings settings;
            settings.beginGroup("user");
    
    
            settings.beginGroup("window");
    
            settings.setValue("tray", tray_);
            settings.setValue("start_in_tray", startInTray_);
    
            settings.setValue("buttons", buttonsInTimeline_);
            settings.setValue("message_hover_highlight", messageHoverHighlight_);
            settings.setValue("enlarge_emoji_only_msg", enlargeEmojiOnlyMessages_);
    
            settings.setValue("max_width", timelineMaxWidth_);
    
            settings.setValue("avatar_circles", avatarCircles_);
    
            settings.setValue("decrypt_sidebar", decryptSidebar_);
    
            settings.setValue("share_keys_with_trusted_users", shareKeysWithTrustedUsers_);
    
            settings.setValue("mobile_mode", mobileMode_);
    
            settings.setValue("font_size", baseFontSize_);
    
            settings.setValue("typing_notifications", typingNotifications_);
    
            settings.setValue("minor_events", sortByImportance_);
    
            settings.setValue("read_receipts", readReceipts_);
            settings.setValue("group_view", groupView_);
            settings.setValue("markdown_enabled", markdown_);
    
            settings.setValue("desktop_notifications", hasDesktopNotifications_);
    
            settings.setValue("alert_on_notification", hasAlertOnNotification_);
    
            settings.setValue("theme", theme());
    
    Joe Donofry's avatar
    Joe Donofry committed
            settings.setValue("font_family", font_);
    
            settings.setValue("emoji_font_family", emojiFont_);
    
            settings.setValue("presence", QVariant::fromValue(presence_));
    
            settings.setValue("ringtone", ringtone_);
    
    trilene's avatar
    trilene committed
            settings.setValue("microphone", microphone_);
            settings.setValue("camera", camera_);
            settings.setValue("camera_resolution", cameraResolution_);
            settings.setValue("camera_frame_rate", cameraFrameRate_);
    
    trilene's avatar
    trilene committed
            settings.setValue("use_stun_server", useStunServer_);
    
            settings.endGroup();
    
    }
    
    HorizontalLine::HorizontalLine(QWidget *parent)
    
      : QFrame{parent}
    
    {
            setFrameShape(QFrame::HLine);
            setFrameShadow(QFrame::Sunken);
    }
    
    UserSettingsPage::UserSettingsPage(QSharedPointer<UserSettings> settings, QWidget *parent)
    
      : QWidget{parent}
      , settings_{settings}
    
            topLayout_ = new QVBoxLayout{this};
    
    
            QIcon icon;
            icon.addFile(":/icons/icons/ui/angle-pointing-to-left.png");
    
    
            auto backBtn_ = new FlatButton{this};
    
            backBtn_->setMinimumSize(QSize(24, 24));
            backBtn_->setIcon(icon);
            backBtn_->setIconSize(QSize(24, 24));
    
    
            QFont font;
            font.setPointSizeF(font.pointSizeF() * 1.1);
    
            auto versionInfo = new QLabel(QString("%1 | %2").arg(nheko::version).arg(nheko::build_os));
    
            if (QCoreApplication::applicationName() != "nheko")
    
                    versionInfo->setText(versionInfo->text() + " | " +
                                         tr("profile: %1").arg(QCoreApplication::applicationName()));
    
            versionInfo->setTextInteractionFlags(Qt::TextBrowserInteraction);
    
            topBarLayout_ = new QHBoxLayout;
            topBarLayout_->setSpacing(0);
            topBarLayout_->setMargin(0);
            topBarLayout_->addWidget(backBtn_, 1, Qt::AlignLeft | Qt::AlignVCenter);
            topBarLayout_->addStretch(1);
    
    
            formLayout_ = new QFormLayout;
    
            formLayout_->setLabelAlignment(Qt::AlignLeft);
            formLayout_->setFormAlignment(Qt::AlignRight);
            formLayout_->setFieldGrowthPolicy(QFormLayout::AllNonFixedFieldsGrow);
            formLayout_->setRowWrapPolicy(QFormLayout::WrapLongRows);
            formLayout_->setHorizontalSpacing(0);
    
            auto general_ = new QLabel{tr("GENERAL"), this};
    
            general_->setSizePolicy(QSizePolicy::Ignored, QSizePolicy::Fixed);
            general_->setFont(font);
    
            trayToggle_                = new Toggle{this};
            startInTrayToggle_         = new Toggle{this};
            avatarCircles_             = new Toggle{this};
            decryptSidebar_            = new Toggle(this);
            shareKeysWithTrustedUsers_ = new Toggle(this);
            groupViewToggle_           = new Toggle{this};
            timelineButtonsToggle_     = new Toggle{this};
            typingNotifications_       = new Toggle{this};
            messageHoverHighlight_     = new Toggle{this};
            enlargeEmojiOnlyMessages_  = new Toggle{this};
            sortByImportance_          = new Toggle{this};
            readReceipts_              = new Toggle{this};
            markdown_                  = new Toggle{this};
            desktopNotifications_      = new Toggle{this};
            alertOnNotification_       = new Toggle{this};
            useStunServer_             = new Toggle{this};
    
            mobileMode_                = new Toggle{this};
    
            scaleFactorCombo_          = new QComboBox{this};
            fontSizeCombo_             = new QComboBox{this};
    
            fontSelectionCombo_        = new QFontComboBox{this};
    
            emojiFontSelectionCombo_   = new QComboBox{this};
    
            ringtoneCombo_             = new QComboBox{this};
    
            microphoneCombo_           = new QComboBox{this};
            cameraCombo_               = new QComboBox{this};
            cameraResolutionCombo_     = new QComboBox{this};
            cameraFrameRateCombo_      = new QComboBox{this};
    
            timelineMaxWidthSpin_      = new QSpinBox{this};
    
            trayToggle_->setChecked(settings_->tray());
            startInTrayToggle_->setChecked(settings_->startInTray());
            avatarCircles_->setChecked(settings_->avatarCircles());
            decryptSidebar_->setChecked(settings_->decryptSidebar());
            shareKeysWithTrustedUsers_->setChecked(settings_->shareKeysWithTrustedUsers());
            groupViewToggle_->setChecked(settings_->groupView());
            timelineButtonsToggle_->setChecked(settings_->buttonsInTimeline());
            typingNotifications_->setChecked(settings_->typingNotifications());
            messageHoverHighlight_->setChecked(settings_->messageHoverHighlight());
            enlargeEmojiOnlyMessages_->setChecked(settings_->enlargeEmojiOnlyMessages());
            sortByImportance_->setChecked(settings_->sortByImportance());
            readReceipts_->setChecked(settings_->readReceipts());
            markdown_->setChecked(settings_->markdown());
            desktopNotifications_->setChecked(settings_->hasDesktopNotifications());
            alertOnNotification_->setChecked(settings_->hasAlertOnNotification());
            useStunServer_->setChecked(settings_->useStunServer());
            mobileMode_->setChecked(settings_->mobileMode());
    
    
            if (!settings_->tray()) {
                    startInTrayToggle_->setState(false);
    
                    startInTrayToggle_->setDisabled(true);
    
            avatarCircles_->setFixedSize(64, 48);
    
            auto uiLabel_ = new QLabel{tr("INTERFACE"), this};
    
    Adasauce's avatar
    Adasauce committed
            uiLabel_->setFixedHeight(uiLabel_->minimumHeight() + LayoutTopMargin);
            uiLabel_->setAlignment(Qt::AlignBottom);
    
            uiLabel_->setFont(font);
    
    
            for (double option = 1; option <= 3; option += 0.25)
                    scaleFactorCombo_->addItem(QString::number(option));
    
            for (double option = 10; option < 17; option += 0.5)
                    fontSizeCombo_->addItem(QString("%1 ").arg(QString::number(option)));
    
    
    Joe Donofry's avatar
    Joe Donofry committed
            QFontDatabase fontDb;
    
    
            // TODO: Is there a way to limit to just emojis, rather than
            // all emoji fonts?
            auto emojiFamilies = fontDb.families(QFontDatabase::Symbol);
    
            for (const auto &family : emojiFamilies) {
                    emojiFontSelectionCombo_->addItem(family);
            }
    
    
            fontSelectionCombo_->setCurrentIndex(fontSelectionCombo_->findText(settings_->font()));
    
            emojiFontSelectionCombo_->setCurrentIndex(
              emojiFontSelectionCombo_->findText(settings_->emojiFont()));
    
            themeCombo_ = new QComboBox{this};
    
    Konstantinos Sideris's avatar
    Konstantinos Sideris committed
            themeCombo_->addItem("Light");
            themeCombo_->addItem("Dark");
    
            themeCombo_->addItem("System");
    
    
            QString themeStr = settings_->theme();
            themeStr.replace(0, 1, themeStr[0].toUpper());
            int themeIndex = themeCombo_->findText(themeStr);
            themeCombo_->setCurrentIndex(themeIndex);
    
    
            timelineMaxWidthSpin_->setMinimum(0);
            timelineMaxWidthSpin_->setMaximum(100'000'000);
            timelineMaxWidthSpin_->setSingleStep(10);
    
    
    trilene's avatar
    trilene committed
            auto callsLabel = new QLabel{tr("CALLS"), this};
            callsLabel->setFixedHeight(callsLabel->minimumHeight() + LayoutTopMargin);
            callsLabel->setAlignment(Qt::AlignBottom);
            callsLabel->setFont(font);
    
    
            auto encryptionLabel_ = new QLabel{tr("ENCRYPTION"), this};
    
    Adasauce's avatar
    Adasauce committed
            encryptionLabel_->setFixedHeight(encryptionLabel_->minimumHeight() + LayoutTopMargin);
            encryptionLabel_->setAlignment(Qt::AlignBottom);
    
            encryptionLabel_->setFont(font);
    
            QFont monospaceFont;
            monospaceFont.setFamily("Monospace");
            monospaceFont.setStyleHint(QFont::Monospace);
    
            monospaceFont.setPointSizeF(monospaceFont.pointSizeF() * 0.9);
    
    
            deviceIdValue_ = new QLabel{this};
    
            deviceIdValue_->setTextInteractionFlags(Qt::TextSelectableByMouse);
            deviceIdValue_->setFont(monospaceFont);
    
    
            deviceFingerprintValue_ = new QLabel{this};
    
            deviceFingerprintValue_->setTextInteractionFlags(Qt::TextSelectableByMouse);
            deviceFingerprintValue_->setFont(monospaceFont);
    
    Adasauce's avatar
    Adasauce committed
            deviceFingerprintValue_->setText(utils::humanReadableFingerprint(QString(44, 'X')));
    
            backupSecretCached      = new QLabel{this};
            masterSecretCached      = new QLabel{this};
            selfSigningSecretCached = new QLabel{this};
            userSigningSecretCached = new QLabel{this};
            backupSecretCached->setFont(monospaceFont);
            masterSecretCached->setFont(monospaceFont);
            selfSigningSecretCached->setFont(monospaceFont);
            userSigningSecretCached->setFont(monospaceFont);
    
    
            auto sessionKeysLabel = new QLabel{tr("Session Keys"), this};
    
            sessionKeysLabel->setFont(font);
    
            sessionKeysLabel->setMargin(OptionMargin);
    
            auto sessionKeysImportBtn = new QPushButton{tr("IMPORT"), this};
            auto sessionKeysExportBtn = new QPushButton{tr("EXPORT"), this};
    
            auto sessionKeysLayout = new QHBoxLayout;
            sessionKeysLayout->addWidget(new QLabel{"", this}, 1, Qt::AlignRight);
    
            sessionKeysLayout->addWidget(sessionKeysExportBtn, 0, Qt::AlignRight);
            sessionKeysLayout->addWidget(sessionKeysImportBtn, 0, Qt::AlignRight);
    
    Nicolas Werner's avatar
    Nicolas Werner committed
            auto crossSigningKeysLabel = new QLabel{tr("Cross Signing Keys"), this};
            crossSigningKeysLabel->setFont(font);
            crossSigningKeysLabel->setMargin(OptionMargin);
    
            auto crossSigningRequestBtn  = new QPushButton{tr("REQUEST"), this};
            auto crossSigningDownloadBtn = new QPushButton{tr("DOWNLOAD"), this};
    
            auto crossSigningKeysLayout = new QHBoxLayout;
            crossSigningKeysLayout->addWidget(new QLabel{"", this}, 1, Qt::AlignRight);
            crossSigningKeysLayout->addWidget(crossSigningRequestBtn, 0, Qt::AlignRight);
            crossSigningKeysLayout->addWidget(crossSigningDownloadBtn, 0, Qt::AlignRight);
    
    
            auto boxWrap = [this, &font](QString labelText, QWidget *field, QString tooltipText = "") {
    
                    auto label = new QLabel{labelText, this};
    
                    label->setFont(font);
                    label->setMargin(OptionMargin);
    
                    if (!tooltipText.isEmpty()) {
                            label->setToolTip(tooltipText);
                    }
    
    
                    auto layout = new QHBoxLayout;
    
                    layout->addWidget(field, 0, Qt::AlignRight);
    
                    formLayout_->addRow(label, layout);
            };
    
            formLayout_->addRow(general_);
    
            formLayout_->addRow(new HorizontalLine{this});
    
            boxWrap(
              tr("Minimize to tray"),
              trayToggle_,
              tr("Keep the application running in the background after closing the client window."));
            boxWrap(tr("Start in tray"),
                    startInTrayToggle_,
                    tr("Start the application in the background without showing the client window."));
    
            formLayout_->addRow(new HorizontalLine{this});
    
            boxWrap(tr("Circular Avatars"),
                    avatarCircles_,
                    tr("Change the appearance of user avatars in chats.\nOFF - square, ON - Circle."));
            boxWrap(tr("Group's sidebar"),
                    groupViewToggle_,
                    tr("Show a column containing groups and tags next to the room list."));
            boxWrap(tr("Decrypt messages in sidebar"),
                    decryptSidebar_,
                    tr("Decrypt the messages shown in the sidebar.\nOnly affects messages in "
                       "encrypted chats."));
            boxWrap(tr("Show buttons in timeline"),
                    timelineButtonsToggle_,
                    tr("Show buttons to quickly reply, react or access additional options next to each "
                       "message."));
    
            boxWrap(tr("Limit width of timeline"),
                    timelineMaxWidthSpin_,
                    tr("Set the max width of messages in the timeline (in pixels). This can help "
                       "readability on wide screen, when Nheko is maximised"));
    
            boxWrap(tr("Typing notifications"),
                    typingNotifications_,
                    tr("Show who is typing in a room.\nThis will also enable or disable sending typing "
                       "notifications to others."));
            boxWrap(
              tr("Sort rooms by unreads"),
              sortByImportance_,
              tr(
                "Display rooms with new messages first.\nIf this is off, the list of rooms will only "
                "be sorted by the timestamp of the last message in a room.\nIf this is on, rooms which "
                "have active notifications (the small circle with a number in it) will be sorted on "
                "top. Rooms, that you have muted, will still be sorted by timestamp, since you don't "
                "seem to consider them as important as the other rooms."));
    
            formLayout_->addRow(new HorizontalLine{this});
    
            boxWrap(tr("Read receipts"),
                    readReceipts_,
                    tr("Show if your message was read.\nStatus is displayed next to timestamps."));
            boxWrap(
              tr("Send messages as Markdown"),
    
              tr("Allow using markdown in messages.\nWhen disabled, all messages are sent as a plain "
                 "text."));
            boxWrap(tr("Desktop notifications"),
                    desktopNotifications_,
                    tr("Notify about received message when the client is not currently focused."));
    
            boxWrap(tr("Alert on notification"),
                    alertOnNotification_,
                    tr("Show an alert when a message is received.\nThis usually causes the application "
                       "icon in the task bar to animate in some fashion."));
    
            boxWrap(tr("Highlight message on hover"),
                    messageHoverHighlight_,
                    tr("Change the background color of messages when you hover over them."));
            boxWrap(tr("Large Emoji in timeline"),
                    enlargeEmojiOnlyMessages_,
                    tr("Make font size larger if messages with only a few emojis are displayed."));
    
            formLayout_->addRow(uiLabel_);
    
            formLayout_->addRow(new HorizontalLine{this});
    
            boxWrap(tr("Touchscreen mode"),
    
                    tr("Will prevent text selection in the timeline to make touch scrolling easier."));
    
    #if !defined(Q_OS_MAC)
    
            boxWrap(tr("Scale factor"),
                    scaleFactorCombo_,
                    tr("Change the scale factor of the whole user interface."));
    
    #else
            scaleFactorCombo_->hide();
    #endif
    
            boxWrap(tr("Font size"), fontSizeCombo_);
            boxWrap(tr("Font Family"), fontSelectionCombo_);
    
    
    #if !defined(Q_OS_MAC)
    
            boxWrap(tr("Emoji Font Family"), emojiFontSelectionCombo_);
    
    #else
            emojiFontSelectionCombo_->hide();
    #endif
    
    
            boxWrap(tr("Theme"), themeCombo_);
    
    trilene's avatar
    trilene committed
    
            formLayout_->addRow(callsLabel);
            formLayout_->addRow(new HorizontalLine{this});
    
            boxWrap(tr("Ringtone"),
                    ringtoneCombo_,
                    tr("Set the notification sound to play when a call invite arrives"));
    
    trilene's avatar
    trilene committed
            boxWrap(tr("Microphone"), microphoneCombo_);
            boxWrap(tr("Camera"), cameraCombo_);
            boxWrap(tr("Camera resolution"), cameraResolutionCombo_);
            boxWrap(tr("Camera frame rate"), cameraFrameRateCombo_);
    
    
            ringtoneCombo_->setSizeAdjustPolicy(QComboBox::AdjustToContents);
            ringtoneCombo_->addItem("Mute");
            ringtoneCombo_->addItem("Default");
            ringtoneCombo_->addItem("Other...");
            const QString &ringtone = settings_->ringtone();
            if (!ringtone.isEmpty() && ringtone != "Mute" && ringtone != "Default")
                    ringtoneCombo_->addItem(ringtone);
    
    trilene's avatar
    trilene committed
            microphoneCombo_->setSizeAdjustPolicy(QComboBox::AdjustToContents);
            cameraCombo_->setSizeAdjustPolicy(QComboBox::AdjustToContents);
            cameraResolutionCombo_->setSizeAdjustPolicy(QComboBox::AdjustToContents);
            cameraFrameRateCombo_->setSizeAdjustPolicy(QComboBox::AdjustToContents);
    
            boxWrap(tr("Allow fallback call assist server"),
    
    trilene's avatar
    trilene committed
                    useStunServer_,
                    tr("Will use turn.matrix.org as assist when your home server does not offer one."));
    
    
            formLayout_->addRow(encryptionLabel_);
    
            formLayout_->addRow(new HorizontalLine{this});
    
            boxWrap(tr("Device ID"), deviceIdValue_);
            boxWrap(tr("Device Fingerprint"), deviceFingerprintValue_);
    
              tr("Share keys with verified users and devices"),
    
              shareKeysWithTrustedUsers_,
              tr("Automatically replies to key requests from other users, if they are verified."));
    
            formLayout_->addRow(new HorizontalLine{this});
    
            formLayout_->addRow(sessionKeysLabel, sessionKeysLayout);
    
    Nicolas Werner's avatar
    Nicolas Werner committed
            formLayout_->addRow(crossSigningKeysLabel, crossSigningKeysLayout);
    
            boxWrap(tr("Master signing key"),
                    masterSecretCached,
                    tr("Your most important key. You don't need to have it cached, since not caching "
                       "it makes it less likely it can be stolen and it is only needed to rotate your "
                       "other signing keys."));
            boxWrap(tr("User signing key"),
                    userSigningSecretCached,
                    tr("The key to verify other users. If it is cached, verifying a user will verify "
                       "all their devices."));
            boxWrap(
              tr("Self signing key"),
              selfSigningSecretCached,
              tr("The key to verify your own devices. If it is cached, verifying one of your devices "
                 "will mark it verified for all your other devices and for users, that have verified "
                 "you."));
            boxWrap(tr("Backup key"),
                    backupSecretCached,
                    tr("The key to decrypt online key backups. If it is cached, you can enable online "
                       "key backup to store encryption keys securely encrypted on the server."));
            updateSecretStatus();
    
    
            auto scrollArea_ = new QScrollArea{this};
    
            scrollArea_->setFrameShape(QFrame::NoFrame);
            scrollArea_->setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
            scrollArea_->setSizeAdjustPolicy(QAbstractScrollArea::AdjustToContents);
            scrollArea_->setWidgetResizable(true);
            scrollArea_->setAlignment(Qt::AlignTop | Qt::AlignVCenter);
    
    
            QScroller::grabGesture(scrollArea_, QScroller::TouchGesture);
    
    
            auto spacingAroundForm = new QHBoxLayout;
            spacingAroundForm->addStretch(1);
            spacingAroundForm->addLayout(formLayout_, 0);
            spacingAroundForm->addStretch(1);
    
    
            auto scrollAreaContents_ = new QWidget{this};
    
            scrollAreaContents_->setObjectName("UserSettingScrollWidget");
    
            scrollAreaContents_->setLayout(spacingAroundForm);
    
    
            scrollArea_->setWidget(scrollAreaContents_);
    
            topLayout_->addLayout(topBarLayout_);
    
            topLayout_->addWidget(scrollArea_, Qt::AlignTop);
            topLayout_->addStretch(1);
    
            topLayout_->addWidget(versionInfo);
    
    
            connect(themeCombo_,
    
    Nicolas Werner's avatar
    Nicolas Werner committed
                    static_cast<void (QComboBox::*)(const QString &)>(&QComboBox::currentTextChanged),
    
    Joe Donofry's avatar
    Joe Donofry committed
                    [this](const QString &text) {
                            settings_->setTheme(text.toLower());
                            emit themeChanged();
                    });
    
            connect(scaleFactorCombo_,
    
    Nicolas Werner's avatar
    Nicolas Werner committed
                    static_cast<void (QComboBox::*)(const QString &)>(&QComboBox::currentTextChanged),
    
                    [](const QString &factor) { utils::setScaleFactor(factor.toFloat()); });
    
            connect(fontSizeCombo_,
    
    Nicolas Werner's avatar
    Nicolas Werner committed
                    static_cast<void (QComboBox::*)(const QString &)>(&QComboBox::currentTextChanged),
    
                    [this](const QString &size) { settings_->setFontSize(size.trimmed().toDouble()); });
    
    Joe Donofry's avatar
    Joe Donofry committed
            connect(fontSelectionCombo_,
    
    Nicolas Werner's avatar
    Nicolas Werner committed
                    static_cast<void (QComboBox::*)(const QString &)>(&QComboBox::currentTextChanged),
    
    Joe Donofry's avatar
    Joe Donofry committed
                    [this](const QString &family) { settings_->setFontFamily(family.trimmed()); });
    
            connect(emojiFontSelectionCombo_,
    
    Nicolas Werner's avatar
    Nicolas Werner committed
                    static_cast<void (QComboBox::*)(const QString &)>(&QComboBox::currentTextChanged),
    
                    [this](const QString &family) { settings_->setEmojiFontFamily(family.trimmed()); });
    
    trilene's avatar
    trilene committed
    
    
            connect(ringtoneCombo_,
                    static_cast<void (QComboBox::*)(const QString &)>(&QComboBox::currentTextChanged),
                    [this](const QString &ringtone) {
                            if (ringtone == "Other...") {
                                    QString homeFolder =
                                      QStandardPaths::writableLocation(QStandardPaths::HomeLocation);
                                    auto filepath = QFileDialog::getOpenFileName(
                                      this, tr("Select a file"), homeFolder, tr("All Files (*)"));
                                    if (!filepath.isEmpty()) {
                                            const auto &oldSetting = settings_->ringtone();
                                            if (oldSetting != "Mute" && oldSetting != "Default")
                                                    ringtoneCombo_->removeItem(
                                                      ringtoneCombo_->findText(oldSetting));
                                            settings_->setRingtone(filepath);
                                            ringtoneCombo_->addItem(filepath);
                                            ringtoneCombo_->setCurrentText(filepath);
                                    } else {
                                            ringtoneCombo_->setCurrentText(settings_->ringtone());
                                    }
                            } else if (ringtone == "Mute" || ringtone == "Default") {
                                    const auto &oldSetting = settings_->ringtone();
                                    if (oldSetting != "Mute" && oldSetting != "Default")
                                            ringtoneCombo_->removeItem(
                                              ringtoneCombo_->findText(oldSetting));
                                    settings_->setRingtone(ringtone);
                            }
                    });
    
    
    trilene's avatar
    trilene committed
            connect(microphoneCombo_,
                    static_cast<void (QComboBox::*)(const QString &)>(&QComboBox::currentTextChanged),
                    [this](const QString &microphone) { settings_->setMicrophone(microphone); });
    
            connect(cameraCombo_,
                    static_cast<void (QComboBox::*)(const QString &)>(&QComboBox::currentTextChanged),
                    [this](const QString &camera) {
                            settings_->setCamera(camera);
                            std::vector<std::string> resolutions =
                              WebRTCSession::instance().getResolutions(camera.toStdString());
                            cameraResolutionCombo_->clear();
                            for (const auto &resolution : resolutions)
                                    cameraResolutionCombo_->addItem(QString::fromStdString(resolution));
                    });
    
            connect(cameraResolutionCombo_,
                    static_cast<void (QComboBox::*)(const QString &)>(&QComboBox::currentTextChanged),
                    [this](const QString &resolution) {
                            settings_->setCameraResolution(resolution);
                            std::vector<std::string> frameRates =
                              WebRTCSession::instance().getFrameRates(settings_->camera().toStdString(),
                                                                      resolution.toStdString());
                            cameraFrameRateCombo_->clear();
                            for (const auto &frameRate : frameRates)
                                    cameraFrameRateCombo_->addItem(QString::fromStdString(frameRate));
                    });
    
            connect(cameraFrameRateCombo_,
                    static_cast<void (QComboBox::*)(const QString &)>(&QComboBox::currentTextChanged),
                    [this](const QString &frameRate) { settings_->setCameraFrameRate(frameRate); });
    
    
    kirillpt's avatar
    kirillpt committed
            connect(trayToggle_, &Toggle::toggled, this, [this](bool enabled) {
                    settings_->setTray(enabled);
                    if (enabled) {
    
                            startInTrayToggle_->setChecked(false);
    
    Anton Karmanov's avatar
    Anton Karmanov committed
                            startInTrayToggle_->setEnabled(true);
    
                            startInTrayToggle_->setState(false);
                            settings_->setStartInTray(false);
    
    kirillpt's avatar
    kirillpt committed
                    } else {
    
                            startInTrayToggle_->setChecked(false);
    
                            startInTrayToggle_->setState(false);
    
    kirillpt's avatar
    kirillpt committed
                            startInTrayToggle_->setDisabled(true);
    
    kirillpt's avatar
    kirillpt committed
                    emit trayOptionChanged(enabled);
    
    kirillpt's avatar
    kirillpt committed
            connect(startInTrayToggle_, &Toggle::toggled, this, [this](bool enabled) {
                    settings_->setStartInTray(enabled);
    
            connect(mobileMode_, &Toggle::toggled, this, [this](bool enabled) {
                    settings_->setMobileMode(enabled);
    
            connect(groupViewToggle_, &Toggle::toggled, this, [this](bool enabled) {
                    settings_->setGroupView(enabled);
    
            connect(decryptSidebar_, &Toggle::toggled, this, [this](bool enabled) {
                    settings_->setDecryptSidebar(enabled);
    
                    emit decryptSidebarChanged();
    
            connect(shareKeysWithTrustedUsers_, &Toggle::toggled, this, [this](bool enabled) {
                    settings_->setShareKeysWithTrustedUsers(enabled);
    
            connect(avatarCircles_, &Toggle::toggled, this, [this](bool enabled) {
                    settings_->setAvatarCircles(enabled);
    
            connect(markdown_, &Toggle::toggled, this, [this](bool enabled) {
                    settings_->setMarkdown(enabled);
    
            connect(typingNotifications_, &Toggle::toggled, this, [this](bool enabled) {
                    settings_->setTypingNotifications(enabled);
    
            connect(sortByImportance_, &Toggle::toggled, this, [this](bool enabled) {
                    settings_->setSortByImportance(enabled);
    
            connect(timelineButtonsToggle_, &Toggle::toggled, this, [this](bool enabled) {
                    settings_->setButtonsInTimeline(enabled);
    
            connect(readReceipts_, &Toggle::toggled, this, [this](bool enabled) {
                    settings_->setReadReceipts(enabled);
    
            connect(desktopNotifications_, &Toggle::toggled, this, [this](bool enabled) {
                    settings_->setDesktopNotifications(enabled);