使用 qsettings 保存和加载修改

Posted

技术标签:

【中文标题】使用 qsettings 保存和加载修改【英文标题】:save and load modifications with qsettings 【发布时间】:2020-01-23 11:01:25 【问题描述】:

我有 2 个窗口(窗口 1)和(窗口 2):看图片。 - 在窗口 1 上有一个“编辑”按钮,应该打开窗口 2(并加载保存在窗口 2 上的修改) - 在窗口 2 上,我有 96 个按钮,最初是绿色的,如果我点击一个按钮,它会变成红色。 我拥有的“确定”按钮应该关闭窗口 2 并保存修改(红色按钮)。 如果我通过单击窗口 1 的“编辑”按钮重新打开窗口 2,则最后一步的红色按钮应保持红色并且无法选中。 我怎么能用 QSettings 做到这一点? 这是我的 savesettings 方法的代码:(我不知道它是否正确,但我认为是的!)

void Vessels::SaveSettings()

        QSettings setting("My_vessels","My_selected_vessels");
        setting.beginGroup("Vessels");
        if (ui->pushButton_4->isChecked())
        ui->pushButton_4->setCheckable(false);
        setting.setValue("selected",ui->pushButton_4->isCheckable());
        setting.endGroup();

我面临两个问题: 1)保存和加载按钮(在这种情况下分别为“确定”和“编辑”)不是来自同一个窗口。 2) 我不知道如何实现应该附加到“编辑”按钮的 loadSettings 方法。

【问题讨论】:

我是一个新的qt用户,我必须在几天内完成这个项目! 看起来你是手工创建的 96 个按钮,不是动态生成的,对吧? 【参考方案1】:

这是您的解决方案,正如我所提出的,请在您这边进行测试,它对我有用:

文件:QSettings2.pro

#-------------------------------------------------
#
# Project created by QtCreator 2020-01-23T19:21:17
#
#-------------------------------------------------

QT       += core gui

greaterThan(QT_MAJOR_VERSION, 4): QT += widgets

TARGET = QSettings2
TEMPLATE = app

# The following define makes your compiler emit warnings if you use
# any feature of Qt which has been marked as deprecated (the exact warnings
# depend on your compiler). Please consult the documentation of the
# deprecated API in order to know how to port your code away from it.
DEFINES += QT_DEPRECATED_WARNINGS

# You can also make your code fail to compile if you use deprecated APIs.
# In order to do so, uncomment the following line.
# You can also select to disable deprecated APIs only up to a certain version of Qt.
#DEFINES += QT_DISABLE_DEPRECATED_BEFORE=0x060000    # disables all the APIs deprecated before Qt 6.0.0

CONFIG += c++11

SOURCES += \
        dialog.cpp \
        main.cpp \
        mainwindow.cpp

HEADERS += \
        dialog.h \
        mainwindow.h

FORMS += \
        dialog.ui \
        mainwindow.ui

# Default rules for deployment.
qnx: target.path = /tmp/$$TARGET/bin
else: unix:!android: target.path = /opt/$$TARGET/bin
!isEmpty(target.path): INSTALLS += target

文件:dialog.h

#ifndef DIALOG_H
#define DIALOG_H

#include <QDialog>

class QLabel;
class QSettings;
namespace Ui 
class Dialog;


class Dialog : public QDialog

    Q_OBJECT

public:
    Dialog(QSettings *settings, QWidget *parent = nullptr);
    ~Dialog();

private slots:
    void on_pushButtonOK_clicked();
    void on_pushButtonCancel_clicked();

private:
    QLabel* CreateNewLabel(QString text, QWidget* parent);
    void saveSettings();
    void loadSettings();
    Ui::Dialog *ui;
    QSettings *m_settings;
    const int ROWS = 8, COLS = 12;
;

#endif // DIALOG_H

文件:mainwindow.h

#ifndef MAINWINDOW_H
#define MAINWINDOW_H

#include <QMainWindow>

class QSettings;
namespace Ui 
class MainWindow;


class MainWindow : public QMainWindow

    Q_OBJECT

public:
    explicit MainWindow(QWidget *parent = nullptr);
    ~MainWindow();

private slots:
    void on_pushButton_clicked();

private:
    Ui::MainWindow *ui;
    QSettings* m_settings;
;

#endif // MAINWINDOW_H

文件:dialog.cpp

#include "dialog.h"
#include "ui_dialog.h"
#include <QGridLayout>
#include <QDebug>
#include <QSettings>

Dialog::Dialog(QSettings *settings, QWidget *parent) :
    QDialog(parent),
    ui(new Ui::Dialog),
    m_settings(settings)

    ui->setupUi(this);
    ui->frame->setStyleSheet(QString());//we did draw border just for design time view
    ui->pushButton_Circular->hide();

    QGridLayout *gridLayout = new QGridLayout();

    //gridLayout->setColumnStretch(0, 1);
    //gridLayout->setRowStretch(0, 1);

    for(int i = 1; i <= COLS; i++)
    
        //gridLayout->setColumnStretch(i, 1);
        QLabel *label = CreateNewLabel(QString::number(i), this);
        gridLayout->addWidget(label, 0, i);
    

    for(int i = 1; i <= ROWS; i++)
    
        //gridLayout->setRowStretch(i, 1);
        QLabel *label = CreateNewLabel(QString("%1").arg(static_cast<char>('A' + i -1)), this);
        gridLayout->addWidget(label, i, 0);
    

    QString buttonIdText;
    for(int i = 1; i <= ROWS; i++)
    
        for(int j = 1; j<= COLS; j++)
        
            QPushButton *button = new QPushButton(this);
            button->setMaximumSize(40, 40);
            button->setMinimumSize(40, 40);
            button->setStyleSheet(ui->pushButton_Circular->styleSheet());
            button->setCheckable(true);

            buttonIdText = QString::number((i-1)*COLS + j);
            button->setText(buttonIdText);
            button->setObjectName("GridButton_" + buttonIdText);

            gridLayout->addWidget(button, i, j);
        
    

    ui->frame->setLayout(gridLayout);
    loadSettings();


Dialog::~Dialog()

    delete ui;


QLabel *Dialog::CreateNewLabel(QString text, QWidget *parent)

    QLabel *label = new QLabel(text, parent);
    label->setFont(QFont(label->font().family(), 10, 500));
    label->setAlignment(Qt::AlignCenter);
    label->setStyleSheet("background-color: rgb(170, 0, 0);");
    label->setMaximumSize(40, 40);
    label->setMinimumSize(40, 40);
    return label;


void Dialog::saveSettings()

    QString key;
    QVariant value;
    m_settings->beginGroup("GridButtonsStatus");
    foreach(QObject *childObject, ui->frame->children())
    
        if(QPushButton *button = qobject_cast<QPushButton*>(childObject))
        
            key = button->objectName();
            value = button->isChecked();
            m_settings->setValue(key, value);
        
    
    m_settings->endGroup();
    m_settings->sync();


void Dialog::loadSettings()

    QString key;
    m_settings->beginGroup("GridButtonsStatus");
    for(int i = 1; i <= ROWS * COLS; i++)
    
        key = QString("GridButton_%1").arg(i);
        qDebug() << "--- key = " << key <<", value = " <<  m_settings->value(key) << endl;
        QPushButton *button = ui->frame->findChild<QPushButton*>(key); //key was push button object name
        if(button)
        
            bool is_Checked = m_settings->value(key, false).toBool();
            button->setChecked(is_Checked);
            button->setEnabled(!is_Checked);
        
    
    m_settings->endGroup();


void Dialog::on_pushButtonOK_clicked()

    saveSettings();
    close();


void Dialog::on_pushButtonCancel_clicked()

    close();


文件:main.cpp

#include "mainwindow.h"
#include <QApplication>

int main(int argc, char *argv[])

    QApplication a(argc, argv);
    MainWindow w;
    w.show();

    return a.exec();


文件:mainwindow.cpp

#include "mainwindow.h"
#include "ui_mainwindow.h"
#include "dialog.h"

#include <QSettings>

MainWindow::MainWindow(QWidget *parent) :
    QMainWindow(parent),
    ui(new Ui::MainWindow), m_settings(new QSettings(qAppName() + ".ini", QSettings::IniFormat))

    ui->setupUi(this);


MainWindow::~MainWindow()

    delete ui;

#include <QDebug>
void MainWindow::on_pushButton_clicked()

    Dialog dialog(m_settings);
    dialog.exec();


文件:dialog.ui

<?xml version="1.0" encoding="UTF-8"?>
<ui version="4.0">
 <class>Dialog</class>
 <widget class="QDialog" name="Dialog">
  <property name="geometry">
   <rect>
    <x>0</x>
    <y>0</y>
    <width>730</width>
    <height>501</height>
   </rect>
  </property>
  <property name="windowTitle">
   <string>Dialog</string>
  </property>
  <widget class="QLabel" name="label">
   <property name="geometry">
    <rect>
     <x>50</x>
     <y>10</y>
     <width>331</width>
     <height>31</height>
    </rect>
   </property>
   <property name="font">
    <font>
     <pointsize>14</pointsize>
     <weight>75</weight>
     <bold>true</bold>
    </font>
   </property>
   <property name="text">
    <string>Choose your vessels:</string>
   </property>
  </widget>
  <widget class="QPushButton" name="pushButtonOK">
   <property name="geometry">
    <rect>
     <x>560</x>
     <y>440</y>
     <width>160</width>
     <height>41</height>
    </rect>
   </property>
   <property name="text">
    <string>OK</string>
   </property>
  </widget>
  <widget class="QPushButton" name="pushButton_Circular">
   <property name="geometry">
    <rect>
     <x>20</x>
     <y>440</y>
     <width>40</width>
     <height>40</height>
    </rect>
   </property>
   <property name="styleSheet">
    <string notr="true">QPushButton 
    border-radius: 20px;
    border-style: outset;
    background: qradialgradient(
        cx: 0.3, cy: -0.4, fx: 0.3, fy: -0.4,
        radius: 1.35, stop: 0 green, stop: 1 #009900
        );
    padding: 5px;
    

QPushButton:hover 
    background: qradialgradient(
        cx: 0.3, cy: -0.4, fx: 0.3, fy: -0.4,
         radius: 1.35, stop: 0 red, stop: 1 #004400
        );
    

QPushButton:checked 
    border-style: inset;
    background: qradialgradient(
        cx: 0.4, cy: -0.1, fx: 0.4, fy: -0.1,
        radius: 1.35, stop: 0 red, stop: 1 #990000
        );
    
</string>
   </property>
   <property name="text">
    <string/>
   </property>
   <property name="checkable">
    <bool>true</bool>
   </property>
   <property name="checked">
    <bool>false</bool>
   </property>
  </widget>
  <widget class="QFrame" name="frame">
   <property name="geometry">
    <rect>
     <x>20</x>
     <y>50</y>
     <width>520</width>
     <height>360</height>
    </rect>
   </property>
   <property name="sizePolicy">
    <sizepolicy hsizetype="Preferred" vsizetype="Preferred">
     <horstretch>0</horstretch>
     <verstretch>0</verstretch>
    </sizepolicy>
   </property>
   <property name="minimumSize">
    <size>
     <width>520</width>
     <height>360</height>
    </size>
   </property>
   <property name="maximumSize">
    <size>
     <width>520</width>
     <height>360</height>
    </size>
   </property>
   <property name="styleSheet">
    <string notr="true">border:1px solid red</string>
   </property>
   <property name="frameShape">
    <enum>QFrame::StyledPanel</enum>
   </property>
   <property name="frameShadow">
    <enum>QFrame::Raised</enum>
   </property>
  </widget>
  <widget class="QPushButton" name="pushButtonCancel">
   <property name="geometry">
    <rect>
     <x>370</x>
     <y>440</y>
     <width>160</width>
     <height>41</height>
    </rect>
   </property>
   <property name="text">
    <string>CANCEL</string>
   </property>
  </widget>
  <zorder>frame</zorder>
  <zorder>label</zorder>
  <zorder>pushButtonOK</zorder>
  <zorder>pushButton_Circular</zorder>
  <zorder>pushButtonCancel</zorder>
 </widget>
 <resources/>
 <connections/>
</ui>

文件:mainwindow.ui

<?xml version="1.0" encoding="UTF-8"?>
<ui version="4.0">
 <class>MainWindow</class>
 <widget class="QMainWindow" name="MainWindow">
  <property name="geometry">
   <rect>
    <x>0</x>
    <y>0</y>
    <width>573</width>
    <height>374</height>
   </rect>
  </property>
  <property name="windowTitle">
   <string>MainWindow</string>
  </property>
  <widget class="QWidget" name="centralWidget">
   <widget class="QLabel" name="label">
    <property name="geometry">
     <rect>
      <x>40</x>
      <y>20</y>
      <width>271</width>
      <height>81</height>
     </rect>
    </property>
    <property name="font">
     <font>
      <pointsize>14</pointsize>
      <weight>75</weight>
      <bold>true</bold>
     </font>
    </property>
    <property name="text">
     <string>Prepare YourSolutions</string>
    </property>
   </widget>
   <widget class="QLabel" name="label_2">
    <property name="geometry">
     <rect>
      <x>50</x>
      <y>150</y>
      <width>91</width>
      <height>31</height>
     </rect>
    </property>
    <property name="font">
     <font>
      <pointsize>9</pointsize>
     </font>
    </property>
    <property name="text">
     <string>Solution 1</string>
    </property>
   </widget>
   <widget class="QPushButton" name="pushButton">
    <property name="geometry">
     <rect>
      <x>154</x>
      <y>152</y>
      <width>101</width>
      <height>31</height>
     </rect>
    </property>
    <property name="text">
     <string>Edit</string>
    </property>
   </widget>
  </widget>
 </widget>
 <layoutdefault spacing="6" margin="11"/>
 <resources/>
 <connections/>
</ui>

【讨论】:

感谢您的代码。当我停止运行程序时,我想初始化设置(所有按钮均为绿色且可检查)。我试图创建一个全局变量 bool init(true)。然后在 dialog.cpp 中: if(!init) loadSettings;在 saveSettings 我将添加 init=false;问题是我不能这个变量,我总是得到以下错误: undefined reference to 'init' 您在哪个文件中声明了该初始化变量?一般来说,如果没有其他方法,让全局变量来做这件事是个坏主意。您应该将变量从一个函数传递到另一个函数。此外,如果您必须创建全局变量,然后在 .cpp 文件中而不是在 .h 中定义它,那么如果 .h 包含在多个文件中,这将使您避免出现多重定义错误。但是为了让 access-or 知道这样的变量是在某些地方定义的,您应该在 access-or 文件中声明与 extern 相同的变量。 我在 .h 文件中声明它为 extern bool init 。我在 .cpp 文件(init=true)中对其进行了初始化;问题是当我将一个值( 0 或 1 )附加到 init 时,我得到了错误。在这种情况下我能做什么?如果没有全局变量,是否还有另一种可能性? 追加 ??或分配?如果可以的话,你应该展示你的代码。您可以将任何变量传递给公共函数(bool isInit);对话框类甚至在构造函数中,因为我已经传递了设置变量。 Dialog::Dialog(QSettings *settings, QWidget *parent) 可以在设置后多加一个 bool isInit 参数 能否请您修改您编写的代码,这样如果我第一次运行程序(首先点击编辑),所有按钮都是绿色的?【参考方案2】:

首先,尝试检查您的注册表以检查您是否正确保存了值。

关于加载函数,你可以使用类似的东西:

QSettings settings("My_vessels","My_selected_vessels");
settings.beginGroup("Vessels");
checked = settings.value("selected").toBool();
settings.endGroup();

对不起,我不知道你所有的变量名称。但是,这与我从 QSettings 加载内容的函数格式相同。

希望对你有帮助。

【讨论】:

以上是关于使用 qsettings 保存和加载修改的主要内容,如果未能解决你的问题,请参考以下文章

OS X 10.9 上的 QSettings - 无法找到/清除它

为啥 QSettings 不存储任何内容?

如何将所选项目从 QListWidget、QTableWidget 保存到 Qsettings

QT QSettings读写配置文件

以人类可读的形式在 QSettings 中保存自定义 QMap 模板实例化

QSettings保存程序设置