瀏覽代碼

Settings implementation bump 07.08.2019

Ivan Arkhipov 5 年之前
父節點
當前提交
c79b404883

+ 45 - 0
src/Legacy/models/patchlist.cpp

@@ -45,6 +45,9 @@ PatchList::PatchList(LotroDatManager *mgr, QObject *parent) : QObject(parent)
     graphics_patch_thread_->start();
     sounds_patch_thread_->start();
     videos_patch_thread_->start();
+
+    patch_update_timer.setInterval(5 * 60 * 1000); // Once in 5 minutes check for updates
+    connect(&patch_update_timer, &QTimer::timeout, this, &PatchList::update);
 }
 
 PatchList::~PatchList()
@@ -119,6 +122,12 @@ void PatchList::onLotroManagerOperationFinished(QString operation_name, QVector<
             qCritical() << "DatManager initialisation error!!!";
         }
     }
+    if (operation_name == "createBackup" || operation_name == "restoreFromBackup" || operation_name == "removeBackup") {
+        --active_operations_num_;
+        if (active_operations_num_ == 0) {
+            emit patchOperationsFinished();
+        }
+    }
 }
 
 void PatchList::startAutoUpdate()
@@ -133,6 +142,42 @@ void PatchList::initialize() {
     QMetaObject::invokeMethod(lotro_mgr_, &LotroDatManager::initializeManager, Qt::QueuedConnection);
 }
 
+void PatchList::createBackup()
+{
+    if (active_operations_num_ > 0) {
+        qWarning() << "Tried to start create backup operation, while others are still running!";
+        return;
+    }
+
+    ++active_operations_num_;
+    emit patchOperationsStarted();
+    QMetaObject::invokeMethod(lotro_mgr_, "createBackup", Qt::QueuedConnection);
+}
+
+void PatchList::restoreFromBackup()
+{
+    if (active_operations_num_ > 0) {
+        qWarning() << "Tried to start restore from backup operation, while others are still running!";
+        return;
+    }
+
+    ++active_operations_num_;
+    emit patchOperationsStarted();
+    QMetaObject::invokeMethod(lotro_mgr_, "restoreFromBackup", Qt::QueuedConnection);
+}
+
+void PatchList::removeBackup()
+{
+    if (active_operations_num_ > 0) {
+        qWarning() << "Tried to start remove backup operation, while others are still running!";
+        return;
+    }
+
+    ++active_operations_num_;
+    emit patchOperationsStarted();
+    QMetaObject::invokeMethod(lotro_mgr_, "removeBackup", Qt::QueuedConnection);
+}
+
 void PatchList::onPatchOperationStarted(Patch::Operation, Patch*)
 {
     if (active_operations_num_ == 0) {

+ 6 - 0
src/Legacy/models/patchlist.h

@@ -34,6 +34,12 @@ public slots:
 
     void initialize();
 
+    void createBackup();
+
+    void restoreFromBackup();
+
+    void removeBackup();
+
 private slots:
     void onPatchOperationStarted(Patch::Operation operation, Patch* patch);
     void onPatchOperationFinished(Patch::Operation operation, Patch* patch);

+ 89 - 0
src/Legacy/models/settings.cpp

@@ -5,6 +5,67 @@
 
 namespace Settings {
 
+QMap<QString, QVariant> defaults = {
+    // General info
+    {"General/UI_scale", 100},
+    {"General/CurrentInitStage", "0"},
+    {"General/MicroUpdates", false},
+    {"General/PatchDownloadDir", "data"},
+
+    // Lotro Manager
+    {"Lotro/game_path", "none"},
+    {"Lotro/original_locale", "English"},
+    {"Lotro/skip_raw_download", true},
+    {"Lotro/no_splash_screen", false},
+
+    // Backup
+    {"Backup/installed", false},
+    {"Backup/path", "/backup/"},
+    {"Backup/creation_time", "none"},
+
+    // Databases download settings
+    {"DatabaseDownload/text", false},          // TextsPatch
+    {"DatabaseDownload/font", false},          // TextsPatch
+    {"DatabaseDownload/image", false},         // GraphicsPatch
+    {"DatabaseDownload/loadscreen", false},    // GraphicsPatch
+    {"DatabaseDownload/texture", false},       // GraphicsPatch
+    {"DatabaseDownload/sound", false},         // SoundsPatch
+    {"DatabaseDownload/video", false},         // VideosPatch
+
+    // Flags, meaning that database is fresh and needs to be installed
+
+    {"DatabaseUpdated/text", false},           // TextsPatch
+    {"DatabaseUpdated/font", false},           // TextsPatch
+    {"DatabaseUpdated/image", false},          // GraphicsPatch
+    {"DatabaseUpdated/loadscreen", false},     // GraphicsPatch
+    {"DatabaseUpdated/texture", false},        // GraphicsPatch
+    {"DatabaseUpdated/sound", false},          // SoundsPatch
+    {"DatabaseUpdated/video", false},          // VideosPatch
+
+    // Localisation components
+    {"Components/texts_main", false},          // TextsPatch
+    {"Components/texts_items", false},         // TextsPatch
+    {"Components/texts_emotes", false},        // TextsPatch
+    {"Components/maps", false},                // ImagesPatch
+    {"Components/loadscreens", false},         // ImagesPatch
+    {"Components/textures", false},            // ImagesPatch
+    {"Components/sounds", false},              // SoundsPatch
+    {"Components/videos", false},              // VideosPatch
+    {"Components/micropatch", false},          // PatchList
+
+    // Network
+    {"Network/site_url", "http://translate.lotros.ru/"},
+    {"Network/forum_url", "http://lotros.ru/"},
+    {"Network/discord_url", "https://discord.gg/j25MdKR"},
+    {"Network/add_report_url", "http://translate.lotros.ru/bugs/add"},
+    {"Network/donate_url", "http://translate.lotros.ru/donate"},
+    {"Network/game_servers_status", "http://translate.lotros.ru/servers.txt"},
+    {"Network/game_servers_message", "http://translate.lotros.ru/profmessage.txt"},
+    {"Network/weekly_code_url", "http://translate.lotros.ru/coupon.txt"},
+    {"Network/news_list_url", "http://translate.lotros.ru/groupware/launcher_news/30/1"},
+    {"Network/patch_updates_url", "http://translate.lotros.ru/groupware/check_updates"}
+};
+
 void setDefaultSettings()
 {
     QSettings settings;
@@ -23,4 +84,32 @@ void setValue(QString key, QVariant value) {
     settings.setValue(key, value);
 }
 
+SettingsBackup createSettingsBackup() {
+    QSettings settings;
+    QMap<QString, QVariant> keysValuesPairs;
+    QStringList keys = settings.allKeys();
+    QStringListIterator it(keys);
+
+    while (it.hasNext()) {
+        QString currentKey = it.next();
+        keysValuesPairs.insert(currentKey, settings.value(currentKey));
+    }
+
+    return keysValuesPairs;
+}
+
+void restoreFromSettingsBackup(const SettingsBackup& backup) {
+    QSettings settings;
+    settings.clear();
+
+    for (auto i = backup.begin(); i != backup.end(); ++i) {
+        settings.setValue(i.key(), i.value());
+    }
 }
+
+void updatePatchComponentsDependencies() {
+
+}
+
+}  /* namespace Settings */
+

+ 8 - 60
src/Legacy/models/settings.h

@@ -5,73 +5,21 @@
 
 namespace Settings
 {
+    using SettingsBackup = QMap<QString, QVariant>;
 
-static QMap<QString, QVariant> defaults = {
-    // General info
-    {"General/UI_scale", 100},
-    {"General/CurrentInitStage", "0"},
-    {"General/MicroUpdates", false},
-    {"General/PatchDownloadDir", "data"},
-
-    // Lotro Manager
-    {"Lotro/game_path", "none"},
-    {"Lotro/original_locale", "English"},
-    {"Lotro/skip_raw_download", true},
-    {"Lotro/no_splash_screen", false},
-
-    // Backup
-    {"Backup/installed", false},
-    {"Backup/path", "/backup/"},
-    {"Backup/creation_time", "none"},
-
-    // Databases download settings
-    {"DatabaseDownload/text", false},          // TextsPatch
-    {"DatabaseDownload/font", false},          // TextsPatch
-    {"DatabaseDownload/image", false},         // GraphicsPatch
-    {"DatabaseDownload/loadscreen", false},    // GraphicsPatch
-    {"DatabaseDownload/texture", false},       // GraphicsPatch
-    {"DatabaseDownload/sound", false},         // SoundsPatch
-    {"DatabaseDownload/video", false},         // VideosPatch
-
-    // Flags, meaning that database is fresh and needs to be installed
-
-    {"DatabaseUpdated/text", false},           // TextsPatch
-    {"DatabaseUpdated/font", false},           // TextsPatch
-    {"DatabaseUpdated/image", false},          // GraphicsPatch
-    {"DatabaseUpdated/loadscreen", false},     // GraphicsPatch
-    {"DatabaseUpdated/texture", false},        // GraphicsPatch
-    {"DatabaseUpdated/sound", false},          // SoundsPatch
-    {"DatabaseUpdated/video", false},          // VideosPatch
-
-    // Localisation components
-    {"Components/texts_main", false},          // TextsPatch
-    {"Components/texts_items", false},         // TextsPatch
-    {"Components/texts_emotes", false},        // TextsPatch
-    {"Components/maps", false},                // ImagesPatch
-    {"Components/loadscreens", false},         // ImagesPatch
-    {"Components/textures", false},            // ImagesPatch
-    {"Components/sounds", false},              // SoundsPatch
-    {"Components/videos", false},              // VideosPatch
-    {"Components/micropatch", false},          // PatchList
-
-    // Network
-    {"Network/site_url", "http://translate.lotros.ru/"},
-    {"Network/forum_url", "http://lotros.ru/"},
-    {"Network/discord_url", "https://discord.gg/j25MdKR"},
-    {"Network/add_report_url", "http://translate.lotros.ru/bugs/add"},
-    {"Network/donate_url", "http://translate.lotros.ru/donate"},
-    {"Network/game_servers_status", "http://translate.lotros.ru/servers.txt"},
-    {"Network/game_servers_message", "http://translate.lotros.ru/profmessage.txt"},
-    {"Network/weekly_code_url", "http://translate.lotros.ru/coupon.txt"},
-    {"Network/news_list_url", "http://translate.lotros.ru/groupware/launcher_news/30/1"},
-    {"Network/patch_updates_url", "http://translate.lotros.ru/groupware/check_updates"}
-};
+    extern QMap<QString, QVariant> defaults;
 
     void setDefaultSettings();
 
     QVariant getValue(QString key);
 
     void setValue(QString key, QVariant value);
+
+    SettingsBackup createSettingsBackup();
+
+    void restoreFromSettingsBackup(const SettingsBackup& backup);
+
+    void updatePatchComponentsDependencies();
 }
 
 #endif // SETTINGS_H

+ 135 - 14
src/Legacy/widgets/settingswidget.cpp

@@ -12,6 +12,7 @@
 #include <QScrollBar>
 #include <QFileDialog>
 #include <QMetaObject>
+#include <QMessageBox>
 
 using namespace SettingsWidgetPrivate;
 
@@ -20,6 +21,8 @@ SettingsWidget::SettingsWidget(PatchList *legacy_patches, QWidget *parent)
     , ui(new Ui::SettingsWidget)
     , legacy_patches_(legacy_patches)
 {
+    settings_changed_ = false;
+
     ui->setupUi(this);
 
     combobox_scrolling_disabler = new ComboboxScrollingDisabler();
@@ -38,6 +41,9 @@ SettingsWidget::SettingsWidget(PatchList *legacy_patches, QWidget *parent)
     connect(legacy_patches, &PatchList::patchOperationsStarted, this, &SettingsWidget::onPatchTotalOperationsStarted);
     connect(legacy_patches, &PatchList::patchOperationsFinished, this, &SettingsWidget::onPatchTotalOperationsFinished);
 
+    ui->apply_changes_button->hide();
+    ui->apply_changes_label->hide();
+
     setActualParametersValues();
 }
 
@@ -188,20 +194,23 @@ void SettingsWidget::resizeEvent(QResizeEvent *)
     ui->apply_changes_button->setMinimumSize(QSize(160, 60) * coefficient);
 }
 
-void SettingsWidget::on_interface_scale_combobox_currentIndexChanged(const QString &arg1)
-{
-    MainWindow* window = qobject_cast<MainWindow*>(qApp->activeWindow());
-
-    if (!window) {
-        ui->interface_scale_combobox->setCurrentText(Settings::getValue("General/UI_scale").toString() + "%");
-        qDebug() << "CANNOT FIND MAIN WINDOW!!!";
-        return;
+void SettingsWidget::processParameterChange() {
+    if (!settings_changed_) {
+        settings_changed_ = true;
+        settings_backup_ = Settings::createSettingsBackup();
+        emit SettingsChanged();
+        ui->apply_changes_label->show();
+        ui->apply_changes_button->show();
     }
+}
 
-    int value = arg1.left(arg1.length() - 1).toInt();
-    window->resize(default_window_width * value / 100, default_window_height * value / 100);
-
-    Settings::setValue("General/UI_scale", value);
+void SettingsWidget::checkIfParametersWereReset() {
+    if (settings_changed_ && settings_backup_ == Settings::createSettingsBackup()) {
+        settings_changed_ = false;
+        emit SettingsReset();
+        ui->apply_changes_label->hide();
+        ui->apply_changes_button->hide();
+    }
 }
 
 void SettingsWidget::on_change_folder_button_clicked()
@@ -216,11 +225,14 @@ void SettingsWidget::on_change_folder_button_clicked()
 
     QString game_folder= str.replace("/LotroLauncher.exe", "").replace("\\", "/").replace("//", "/") + "/";
 
-    if (Settings::getValue("Lotro/game_path").toString() == game_folder)
+    if (Settings::getValue("Lotro/game_path").toString() == game_folder) {
         return;
+    }
 
+    processParameterChange();
     Settings::setValue("Lotro/game_path", game_folder);
     ui->game_folder_path->setText(game_folder);
+    checkIfParametersWereReset();
 }
 
 void SettingsWidget::on_lotro_base_language_combobox_currentIndexChanged(int index)
@@ -234,10 +246,116 @@ void SettingsWidget::on_lotro_base_language_combobox_currentIndexChanged(int ind
     if (index == 2)
         value = "FR";
 
-    if (Settings::getValue("Lotro/original_locale").toString() == value)
+    if (Settings::getValue("Lotro/original_locale").toString() == value) {
         return;
+    }
 
+    processParameterChange();
     Settings::setValue("Lotro/original_locale", value);
+    checkIfParametersWereReset();
+}
+
+void SettingsWidget::on_skiprawdownload_checkbox_stateChanged(int arg1) {
+    processParameterChange();
+    if (arg1 == Qt::Checked) {
+        Settings::setValue("Lotro/skip_raw_download", true);
+    }
+    if (arg1 == Qt::Unchecked) {
+        Settings::setValue("Lotro/skip_raw_download", false);
+    }
+    checkIfParametersWereReset();
+}
+
+void SettingsWidget::on_nosplashscreen_checkbox_stateChanged(int arg1)
+{
+    processParameterChange();
+    if (arg1 == Qt::Checked) {
+        Settings::setValue("Lotro/no_splash_screen", true);
+    }
+    if (arg1 == Qt::Unchecked) {
+        Settings::setValue("Lotro/no_splash_screen", false);
+    }
+    checkIfParametersWereReset();
+}
+
+void SettingsWidget::on_backup_create_button_clicked()
+{
+    if (settings_changed_) {
+        QMessageBox::warning(nullptr, "Невозможно выполнить действие!", "Некоторые настройки были изменены. Чтобы создать резервную копию, примените или отмените изменения!");
+        return;
+    }
+
+    if (Settings::getValue("Backup/installed").toBool()) {
+        int result = QMessageBox::question(nullptr, "Подтвердите действие", "Резервная копия уже существует. Вы хотите перезаписать существующую копию?", QMessageBox::Yes, QMessageBox::No, QMessageBox::NoButton);
+        if (result != QMessageBox::Yes) {
+            return;
+        }
+    }
+
+    QMetaObject::invokeMethod(legacy_patches_, "createBackup", Qt::QueuedConnection);
+}
+
+void SettingsWidget::on_backup_restore_button_clicked()
+{
+    if (!Settings::getValue("Backup/installed").toBool()) {
+        QMessageBox::warning(nullptr, "Невозможно выполнить действие.", "Резервная копия не найдена. Создайте резервную копию, чтобы иметь возможность восстановления данных.");
+        return;
+    }
+
+    if (settings_changed_) {
+        QMessageBox::warning(nullptr, "Невозможно выполнить действие.", "Некоторые настройки были изменены. Чтобы восстановить данные из резервной копии, примените или отмените изменения!");
+        return;
+    }
+
+    QMetaObject::invokeMethod(legacy_patches_, "restoreFromBackup", Qt::QueuedConnection);
+}
+
+void SettingsWidget::on_backup_remove_button_clicked()
+{
+    if (Settings::getValue("Backup/installed").toBool()) {
+        int result = QMessageBox::question(nullptr, "Подтвердите действие", "Вы уверены, что хотите удалить существующую резервную копию?", QMessageBox::Yes, QMessageBox::No, QMessageBox::NoButton);
+        if (result != QMessageBox::Yes) {
+            return;
+        }
+    } else {
+        QMessageBox::information(nullptr, "Невозможно выполнить действие.", "Удаление невозможно: резервной копии не существует.");
+        return;
+    }
+
+    QMetaObject::invokeMethod(legacy_patches_, "removeBackup", Qt::QueuedConnection);
+}
+
+void SettingsWidget::on_patch_texts_checkbox_stateChanged(int arg1)
+{
+    processParameterChange();
+    if (arg1 == Qt::Checked) {
+        Settings::setValue("Components/texts_main", true);
+        Settings::updatePatchComponentsDependencies();
+    } else {
+        Settings::setValue("Components/texts_main", false);
+        Settings::updatePatchComponentsDependencies();
+    }
+    checkIfParametersWereReset();
+}
+
+
+
+
+
+void SettingsWidget::on_interface_scale_combobox_currentIndexChanged(const QString &arg1)
+{
+    MainWindow* window = qobject_cast<MainWindow*>(qApp->activeWindow());
+
+    if (!window) {
+        ui->interface_scale_combobox->setCurrentText(Settings::getValue("General/UI_scale").toString() + "%");
+        qDebug() << "CANNOT FIND MAIN WINDOW!!!";
+        return;
+    }
+
+    int value = arg1.left(arg1.length() - 1).toInt();
+    window->resize(default_window_width * value / 100, default_window_height * value / 100);
+
+    Settings::setValue("General/UI_scale", value);
 }
 
 void SettingsWidget::onPatchTotalOperationsStarted()
@@ -287,3 +405,6 @@ void SettingsWidget::onPatchTotalOperationsFinished()
     ui->patch_force_apply_button->setEnabled(true);
     ui->micropatch_checkbox->setEnabled(true);
 }
+
+
+

+ 28 - 2
src/Legacy/widgets/settingswidget.h

@@ -12,6 +12,8 @@
 #include <QScrollBar>
 #include <QWheelEvent>
 
+#include "models/settings.h"
+
 namespace Ui {
 class SettingsWidget;
 }
@@ -104,17 +106,26 @@ public:
 
     ~SettingsWidget();
 
+signals:
+    void SettingsChanged(); // Settings were changed so we should not do any actions until SettingsApplied() signal
+
+    void SettingsApplied(); // Need to re-initialise all environment and start updates check
+
+    void SettingsReset(); // Need to drop updates blockage, but do not re-initialize all environment
+
 public slots:
     void setActualParametersValues();
 
     void updateFontsSizes();
 
-//    void checkIfPropertiesChanged();
-
 protected:
     void resizeEvent(QResizeEvent *event) override;
 
 private slots:
+    void processParameterChange();
+
+    void checkIfParametersWereReset();
+
     void on_interface_scale_combobox_currentIndexChanged(const QString &arg1);
 
     void on_change_folder_button_clicked();
@@ -125,6 +136,18 @@ private slots:
 
     void onPatchTotalOperationsFinished();
 
+    void on_skiprawdownload_checkbox_stateChanged(int arg1);
+
+    void on_nosplashscreen_checkbox_stateChanged(int arg1);
+
+    void on_backup_create_button_clicked();
+
+    void on_backup_restore_button_clicked();
+
+    void on_backup_remove_button_clicked();
+
+    void on_patch_texts_checkbox_stateChanged(int arg1);
+
 private:
     bool patch_operations_running_;
 
@@ -134,6 +157,9 @@ private:
 
     SettingsWidgetPrivate::ComboboxScrollingDisabler* combobox_scrolling_disabler;
     SettingsWidgetPrivate::SettingsTabsScroller* scroller;
+
+    Settings::SettingsBackup settings_backup_;
+    bool settings_changed_ = false;
 };
 
 #endif // SettingsWidget_H

+ 129 - 104
src/Legacy/widgets/settingswidget.ui

@@ -69,109 +69,7 @@
     <property name="verticalSpacing">
      <number>15</number>
     </property>
-    <item row="5" column="0">
-     <widget class="QLabel" name="patch_installing_label">
-      <property name="font">
-       <font>
-        <family>Trajan Pro 3</family>
-        <pointsize>10</pointsize>
-       </font>
-      </property>
-      <property name="styleSheet">
-       <string notr="true">color: red</string>
-      </property>
-      <property name="text">
-       <string>Выполняется установка русификации, изменение параметров недоступно</string>
-      </property>
-      <property name="alignment">
-       <set>Qt::AlignCenter</set>
-      </property>
-      <property name="wordWrap">
-       <bool>true</bool>
-      </property>
-      <property name="textInteractionFlags">
-       <set>Qt::LinksAccessibleByKeyboard|Qt::LinksAccessibleByMouse|Qt::TextBrowserInteraction|Qt::TextEditable|Qt::TextEditorInteraction|Qt::TextSelectableByKeyboard|Qt::TextSelectableByMouse</set>
-      </property>
-     </widget>
-    </item>
-    <item row="2" column="0">
-     <widget class="QLabel" name="group_label_3">
-      <property name="font">
-       <font>
-        <family>Trajan Pro 3</family>
-        <pointsize>10</pointsize>
-       </font>
-      </property>
-      <property name="text">
-       <string>III. Выбор компонентов русификации</string>
-      </property>
-      <property name="wordWrap">
-       <bool>true</bool>
-      </property>
-     </widget>
-    </item>
-    <item row="3" column="0">
-     <widget class="QLabel" name="group_label_4">
-      <property name="font">
-       <font>
-        <family>Trajan Pro 3</family>
-        <pointsize>10</pointsize>
-       </font>
-      </property>
-      <property name="text">
-       <string>IV. Параметры Наследия</string>
-      </property>
-      <property name="wordWrap">
-       <bool>true</bool>
-      </property>
-     </widget>
-    </item>
-    <item row="0" column="0">
-     <widget class="QLabel" name="group_label_1">
-      <property name="font">
-       <font>
-        <family>Trajan Pro 3</family>
-        <pointsize>10</pointsize>
-       </font>
-      </property>
-      <property name="text">
-       <string>I. Параметры клиента игры</string>
-      </property>
-      <property name="wordWrap">
-       <bool>true</bool>
-      </property>
-     </widget>
-    </item>
-    <item row="4" column="0">
-     <spacer name="verticalSpacer">
-      <property name="orientation">
-       <enum>Qt::Vertical</enum>
-      </property>
-      <property name="sizeHint" stdset="0">
-       <size>
-        <width>20</width>
-        <height>60</height>
-       </size>
-      </property>
-     </spacer>
-    </item>
-    <item row="1" column="0">
-     <widget class="QLabel" name="group_label_2">
-      <property name="font">
-       <font>
-        <family>Trajan Pro 3</family>
-        <pointsize>10</pointsize>
-       </font>
-      </property>
-      <property name="text">
-       <string>II. Параметры резервной копии</string>
-      </property>
-      <property name="wordWrap">
-       <bool>true</bool>
-      </property>
-     </widget>
-    </item>
-    <item row="6" column="0">
+    <item row="7" column="0">
      <widget class="QWidget" name="widget_3" native="true">
       <property name="sizePolicy">
        <sizepolicy hsizetype="Preferred" vsizetype="Fixed">
@@ -295,6 +193,133 @@ QPushButton#apply_changes_button:disabled {
       </layout>
      </widget>
     </item>
+    <item row="6" column="0">
+     <widget class="QLabel" name="patch_installing_label">
+      <property name="font">
+       <font>
+        <family>Trajan Pro 3</family>
+        <pointsize>10</pointsize>
+       </font>
+      </property>
+      <property name="styleSheet">
+       <string notr="true">color: red</string>
+      </property>
+      <property name="text">
+       <string>Выполняется установка русификации, изменение параметров недоступно</string>
+      </property>
+      <property name="alignment">
+       <set>Qt::AlignCenter</set>
+      </property>
+      <property name="wordWrap">
+       <bool>true</bool>
+      </property>
+      <property name="textInteractionFlags">
+       <set>Qt::LinksAccessibleByKeyboard|Qt::LinksAccessibleByMouse|Qt::TextBrowserInteraction|Qt::TextEditable|Qt::TextEditorInteraction|Qt::TextSelectableByKeyboard|Qt::TextSelectableByMouse</set>
+      </property>
+     </widget>
+    </item>
+    <item row="4" column="0">
+     <spacer name="verticalSpacer">
+      <property name="orientation">
+       <enum>Qt::Vertical</enum>
+      </property>
+      <property name="sizeHint" stdset="0">
+       <size>
+        <width>20</width>
+        <height>60</height>
+       </size>
+      </property>
+     </spacer>
+    </item>
+    <item row="2" column="0">
+     <widget class="QLabel" name="group_label_3">
+      <property name="font">
+       <font>
+        <family>Trajan Pro 3</family>
+        <pointsize>10</pointsize>
+       </font>
+      </property>
+      <property name="text">
+       <string>III. Выбор компонентов русификации</string>
+      </property>
+      <property name="wordWrap">
+       <bool>true</bool>
+      </property>
+     </widget>
+    </item>
+    <item row="3" column="0">
+     <widget class="QLabel" name="group_label_4">
+      <property name="font">
+       <font>
+        <family>Trajan Pro 3</family>
+        <pointsize>10</pointsize>
+       </font>
+      </property>
+      <property name="text">
+       <string>IV. Параметры Наследия</string>
+      </property>
+      <property name="wordWrap">
+       <bool>true</bool>
+      </property>
+     </widget>
+    </item>
+    <item row="0" column="0">
+     <widget class="QLabel" name="group_label_1">
+      <property name="font">
+       <font>
+        <family>Trajan Pro 3</family>
+        <pointsize>10</pointsize>
+       </font>
+      </property>
+      <property name="text">
+       <string>I. Параметры клиента игры</string>
+      </property>
+      <property name="wordWrap">
+       <bool>true</bool>
+      </property>
+     </widget>
+    </item>
+    <item row="1" column="0">
+     <widget class="QLabel" name="group_label_2">
+      <property name="font">
+       <font>
+        <family>Trajan Pro 3</family>
+        <pointsize>10</pointsize>
+       </font>
+      </property>
+      <property name="text">
+       <string>II. Параметры резервной копии</string>
+      </property>
+      <property name="wordWrap">
+       <bool>true</bool>
+      </property>
+     </widget>
+    </item>
+    <item row="5" column="0">
+     <widget class="QLabel" name="apply_changes_label">
+      <property name="font">
+       <font>
+        <family>Trajan Pro 3</family>
+        <pointsize>10</pointsize>
+       </font>
+      </property>
+      <property name="styleSheet">
+       <string notr="true">color: red</string>
+      </property>
+      <property name="text">
+       <string>Нажмите на кнопку &quot;Применить&quot;, чтобы изменения вступили в силу!</string>
+      </property>
+      <property name="alignment">
+       <set>Qt::AlignCenter</set>
+      </property>
+      <property name="wordWrap">
+       <bool>true</bool>
+      </property>
+      <property name="textInteractionFlags">
+       <set>Qt::LinksAccessibleByKeyboard|Qt::LinksAccessibleByMouse|Qt::TextBrowserInteraction|Qt::TextEditable|Qt::TextEditorInteraction|Qt::TextSelectableByKeyboard|Qt::TextSelectableByMouse</set>
+      </property>
+     </widget>
+    </item>
    </layout>
   </widget>
   <widget class="QScrollArea" name="content_scroll_area">
@@ -369,7 +394,7 @@ QScrollBar:vertical {
     <property name="geometry">
      <rect>
       <x>0</x>
-      <y>-409</y>
+      <y>0</y>
       <width>651</width>
       <height>967</height>
      </rect>