pyqt5基础笔记(125-使用配置文件来保存和恢复应用的设置)
pyqt5基础笔记(125-使用配置文件来保存和恢复应用的设置)self.settings.setValue("mainwindow/size" self.win.size() self.settings.setValue("mainwindow/fullScreen" self.win.isFullScreen() self.settings.setValue("outputpanel/visible" self.panel.isVisible()如果要保存或者读取很多具有相同分组的设置,则可以使用beginGroup()指定分组前缀,并在最后调用endGroup()。使用分组机制,演示上面的示例代码如下:可以使用'/'字符作为分隔符来形成分层键,类似于Unix文件路径。例如:方式二:指定配置文件名和配置文件格式的方式来创建,如下面的语句将以ini文件的格式创建一个名为config
在通常情况下,用户使用一个软件的时候,期望保存一些数据状态,比如窗口的大小和位置,主题,选项,最近操作过的文件等等信息,在下一次启动软件的时候就自动加载这些信息,让软件恢复到上一次退出时的状态。在Windows系统中,这些信息通常存储在系统注册表中,在macOS和iOS则存储在属性列表中。在Unix系统上,在没有标准情况下,许多应用都使用ini文件来保存配置信息。 QSettings就是围绕不同的系统做了技术上的抽象,使开发者可以轻松实现在不同系统间可移植的方式来保存和恢复应用程序设置。同时QSetting还可以支持自定义存储格式来保存和恢复应用程序的设置。
QSettings简介QSettings类提供一种与平台无关的应用程序设置的保存和恢复技术。可以用下面的方式创建QSettings对象。
方式一: 传递给QSettings两个参数,第一个为公司或者组织的名称(如MySoft),第二个是应用程序的名称(如 MyProgram),创建一个QSettings对象:
self.settings = QSettings('MySoft' 'MyProgram')
说明,在Windows下同下,这种方式信息写在注册表中。
方式二:指定配置文件名和配置文件格式的方式来创建,如下面的语句将以ini文件的格式创建一个名为config.ini的配置信息文件:
self.settings = QSettings('config.ini' QSettings.IniFormat)
使用函数setValue()来存储数据,如果已经存在相同的键值,那么新设置的值将覆盖就的设置值。为了保证效率,所做的更改有可能不会马上存储到文件中,使用sync()函数可以保证设置同步存储到文件中。函数 value()返回存储的键值。
键(key)可以包含任何Unicode字符。Windows注册表和INI文件使用不区分大小写的键,而macOS和iOS上的CFPreferences API使用不区分大小写的键。为避免可移植性问题,请遵循以下简单规则:
- 始终使用相同的大小写引用相同的键。例如,如果在代码中的某个地方将键称为“text fonts”,则不要在其他地方将使用“Text Fonts”来引用。
- 除大小写外,请避免使用相同的键名。例如,如果您有一个名为“ MainWindow”的键,请不要尝试将另一个键另存为“ mainwindow”。
- 在节或键名中不要使用斜杠(“ /”和“ \”);反斜杠字符用于分隔子键。在Windows上,QSettings将“ \”转换为“ /”,从而使它们相同。
可以使用'/'字符作为分隔符来形成分层键,类似于Unix文件路径。例如:
self.settings.setValue("mainwindow/size" self.win.size()
self.settings.setValue("mainwindow/fullScreen" self.win.isFullScreen()
self.settings.setValue("outputpanel/visible" self.panel.isVisible()
如果要保存或者读取很多具有相同分组的设置,则可以使用beginGroup()指定分组前缀,并在最后调用endGroup()。使用分组机制,演示上面的示例代码如下:
self.settings.beginGroup("mainwindow");
self.settings.setValue("size" self.win->size())
self.settings.setValue("fullScreen" self.win.isFullScreen())
self.settings.endGroup();
self.settings.beginGroup("outputpanel");
self.settings.setValue("visible" self.panel.isVisible())
self.settings.endGroup()
演示
在前面notepay.py的基础上,添加了三个函数initRecentFile() updateRecentFileMneu() 和updateRecentFiles(),演示了使用QSettings为程序添加最近列表的功能。 完整代码如下:
importsys
fromPyQt5importQtCore QtGui QtWidgets
fromPyQt5.QtCoreimportQt QSettings
fromPyQt5.QtWidgetsimport(QApplication QMainWindow QMenuBar QMenu
QAction QPlainTextEdit QStyle QFileDialog
QMessageBox QToolBar QStatusBar QLabel)
classDemoNotepad(QMainWindow):
def__init__(self parent=None):
super(DemoNotepad self).__init__(parent)
#设置窗口标题
self.setWindowTitle('实战QtforPython:QSettings演示-记事本')
#设置窗口大小
self.resize(480 360)
self.path=None
self.initUi()
definitUi(self):
#设置一个文本编辑器作为中心小部件
self.txtEditor=QPlainTextEdit(self)
self.setCentralWidget(self.txtEditor)
self.initRecentFileList()
#创建菜单条和工具条使用的Action
self.initActions()
#初始化菜单条
self.initMenuBar()
#初始化化工具条
self.initToolBar()
#初始化状态条
self.initStatusBar()
definitActions(self):
style=QApplication.style()
#新建文件
self.aFileNew=QAction('新建(&N)' self)
#添加一个图标
self.aFileNew.setIcon(style.standardIcon(QStyle.SP_FileIcon))
#添加快捷键
self.aFileNew.setShortcut(Qt.CTRL Qt.Key_N)
self.aFileNew.setToolTip('新建一个文本文件')
self.aFileNew.triggered.connect(self.onFileNew)
#打开文件
self.aFileOpen=QAction('打开(&O)...' self)
self.aFileOpen.setIcon(style.standardIcon(QStyle.SP_DialogOpenButton))
self.aFileOpen.setShortcut(Qt.CTRL Qt.Key_O)
self.aFileOpen.setToolTip('打开一个文本文件')
self.aFileOpen.triggered.connect(self.onFileOpen)
#保存
self.aFileSave=QAction('保存(&S)' self)
self.aFileSave.setIcon(style.standardIcon(QStyle.SP_DialogSaveButton))
self.aFileSave.setShortcut(Qt.CTRL Qt.Key_S)
self.aFileSave.setToolTip('保存文本文件')
self.aFileSave.triggered.connect(self.onFileSave)
#另存为
self.aFileSaveAs=QAction('另存为(&A)...' self)
self.aFileSaveAs.triggered.connect(self.onFileSaveAs)
#退出
self.aFileExit=QAction('退出(&X)' self)
self.aFileExit.triggered.connect(self.close)
#撤销编辑
self.aEditUndo=QAction('撤销(&U)' self)
self.aEditUndo.setShortcut(Qt.CTRL Qt.Key_Z)
self.aEditUndo.triggered.connect(self.txtEditor.undo)
#恢复编辑
self.aEditRedo=QAction('恢复(&R)' self)
self.aEditRedo.setShortcut(Qt.CTRL Qt.Key_Y)
self.aEditUndo.triggered.connect(self.txtEditor.redo)
#剪切操作
self.aEditCut=QAction('剪切(&T)' self)
self.aEditCut.setShortcut(Qt.CTRL Qt.Key_X)
self.aEditCut.triggered.connect(self.txtEditor.cut)
#复制操作
self.aEditCopy=QAction('复制(&C)' self)
self.aEditCopy.setShortcut(Qt.CTRL Qt.Key_C)
self.aEditCopy.triggered.connect(self.txtEditor.copy)
#粘贴操作
self.aEditPaste=QAction('粘贴(&P)' self)
self.aEditPaste.setShortcut(Qt.CTRL Qt.Key_V)
self.aEditPaste.triggered.connect(self.txtEditor.paste)
#删除操作
self.aEditDel=QAction('删除(&L)' self)
self.aEditDel.setShortcut(Qt.Key_Delete)
self.aEditDel.triggered.connect(self.onEditDelete)
#全选操作
self.aEditSelectAll=QAction('全选(&A)' self)
self.aEditSelectAll.setShortcut(Qt.CTRL Qt.Key_A)
self.aEditSelectAll.triggered.connect(self.txtEditor.selectAll)
self.aFmtAutoLine=QAction('自动换行(&W)' self)
self.aFmtAutoLine.setCheckable(True)
self.aFmtAutoLine.setChecked(True)
self.aFmtAutoLine.triggered[bool].connect(self.onFormatAutoLine)
self.aHelpAbout=QAction('关于(&A)...' self)
self.aHelpAbout.triggered.connect(self.onHelpAbout)
definitMenuBar(self):
menuBar=self.menuBar()
self.fileMenu=menuBar.addMenu('文件(&F)')
editMenu=menuBar.addMenu('编辑(&E)')
formatMenu=menuBar.addMenu('格式(&O)')
helpMenu=menuBar.addMenu('帮助(&H)')
#====文件操作部分====#
self.fileMenu.addAction(self.aFileNew)
self.fileMenu.addAction(self.aFileOpen)
self.fileMenu.addAction(self.aFileSave)
self.fileMenu.addAction(self.aFileSaveAs)
self.fileMenu.addSeparator()
self.fileMenu.addAction(self.aFileExit)
self.updateRecentFileMenu()
#====编辑部分====#
editMenu.addAction(self.aEditUndo)
editMenu.addAction(self.aEditRedo)
editMenu.addSeparator()
editMenu.addAction(self.aEditCut)
editMenu.addAction(self.aEditCopy)
editMenu.addAction(self.aEditPaste)
editMenu.addAction(self.aEditDel)
editMenu.addSeparator()
editMenu.addAction(self.aEditSelectAll)
#当编辑栏的菜单对象被点击时 在状态栏显示信息
editMenu.triggered[QAction].connect(self.onProcessTrigger)
#====格式设置部分====#
formatMenu.addAction(self.aFmtAutoLine)
#====帮助部分====#
helpMenu.addAction(self.aHelpAbout)
definitToolBar(self):
toolBar=self.addToolBar('')
toolBar.addAction(self.aFileNew)
toolBar.addAction(self.aFileOpen)
toolBar.addAction(self.aFileSave)
definitStatusBar(self):
self.statusBar=QStatusBar(self)
#添加一个显示永久信息的标签控件
self.info=QLabel(self)
self.info.setText('实战QtforPython')
self.info.setAlignment(Qt.AlignRight)
self.statusBar.addPermanentWidget(self.info)
self.setStatusBar(self.statusBar)
defmsgCritical(self strInfo):
dlg=QMessageBox(self)
dlg.setIcon(QMessageBox.Critical)
dlg.setText(strInfo)
dlg.show()
defonFileNew(self):
self.txtEditor.clear()
defonFileOpen(self):
path _=QFileDialog.getOpenFileName(self '打开文件' '' '文本文件(*.txt)')
ifpath:
ifself.loadFile(path):
#添加到最近打开的菜单列表中
self.updateRecentFiles(path)
#加载文件
defloadFile(self path):
try:
withopen(path 'r')asf:
text=f.read()
exceptExceptionase:
self.msgCritical(str(e))
returnFalse
else:
self.path=path
self.txtEditor.setPlainText(text)
self.statusBar.showMessage('打开文件' path 5000)
returnTrue
defonFileSave(self):
ifself.pathisNone:
returnself.onFileSaveAs()
self._saveToPath(self.path)
defonFileSaveAs(self):
path _=QFileDialog.getSaveFileName(self '保存文件' '' '文本文件(*.txt)')
ifnotpath:
return
self._saveToPath(path)
def_saveToPath(self path):
text=self.txtEditor.toPlainText()
try:
withopen(path 'w')asf:
f.write(text)
exceptExceptionase:
self.msgCritical(str(e))
else:
self.path=path
defonEditDelete(self):
tc=self.txtEditor.textCursor()
#tc.select(QtGui.QTextCursor.BlockUnderCursor)这样删除一行
tc.removeSelectedText()
defonFormatAutoLine(self autoLine):
ifautoLine:
self.txtEditor.setLineWrapMode(QPlainTextEdit.WidgetWidth)
else:
self.txtEditor.setLineWrapMode(QPlainTextEdit.NoWrap)
defonHelpAbout(self):
QMessageBox.information(self '实战QtforPython' 'PyQt5实现的文本编辑器演示版')
defonProcessTrigger(self action):
self.statusBar.showMessage(action.text() '菜单项被点击了' 3000)
#初始化最近文件列表信息
definitRecentFileList(self):
#创建一个配置对象
#方式一:
#self.settings=QSettings('Qt-for-Python' 'MynotePad')
#方式二:
self.settings=QSettings('config.ini' QSettings.IniFormat)
#加载最近文件
self.recentFiles=self.settings.value('FileList/recentFiles')or[]
#添加最近文件菜单行为
self.maxNumRecentFiles=4#最多四个最近的文件
self.recentFileActions=[]
foriinrange(self.maxNumRecentFiles):
self.recentFileActions.append(QAction(self))
#更新最近文件列表菜单
defupdateRecentFileMenu(self):
#先移除添加的行为
foractioninself.recentFileActions:
self.fileMenu.removeAction(action)
#移除退出菜单项
self.fileMenu.removeAction(self.aFileExit)
#移除多余的
#最近文件
fori fnameinenumerate(self.recentFiles):
action=self.recentFileActions[i]
action.setText(fname)
action.setToolTip('最近打开的文件')
action.setData(fname)#保存数据
action.triggered.connect(self.loadRecentFile)
self.fileMenu.addAction(action)
#恢复退出菜单项
self.fileMenu.addSeparator()
self.fileMenu.addAction(self.aFileExit)
#更新最近文件
defupdateRecentFiles(self fname):
iffnamenotinself.recentFiles:
self.recentFiles.insert(0 fname)
iflen(self.recentFiles)>self.maxNumRecentFiles:
self.recentFiles.pop()
#更新菜单
self.updateRecentFileMenu()
#保存更新后最近菜单列表
self.settings.setValue('FileList/recentFiles' self.recentFiles)
defloadRecentFile(self):
sender=self.sender()
ifisinstance(sender QAction):
fname=sender.data()#取出保存的数据
self.loadFile(fname)
if__name__=='__main__':
app=QApplication(sys.argv)
window=DemoNotepad()
window.show()
sys.exit(app.exec())
演示结果如下图:
QSettings测试
本文知识点- 提供一种与平台无关的应用程序设置的保存和恢复技术。
- 在菜单条中插入和清除菜单项。
- 在QAction中保存数据。
- 使用QSettings实现最近打开文件列表。
前一篇:实战PyQt5: 124-在应用中访问系统的标准路径
请多多关注,评论,收藏,点赞,和转发。