使用 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 - 无法找到/清除它
如何将所选项目从 QListWidget、QTableWidget 保存到 Qsettings