我最近花了一些时间研究如何在PyQt中使用QDataStream与QTreeWidget配合。在我找不到这方面具体示例,并且关于QDataStream的pyqt文档似乎普遍缺乏的情况下,我想挂一个问题在这里,以便日后有需要的人可以参考。如果有人愿意尝试并提供帮助,我会稍等片刻,然后再发布我的进展。
问题是:在PyQt中,我怎样可以使用QDataStream将QTreeWidgetItems保存为本机QT对象并将文件读回来,以精确恢复树结构到之前保存的状态?
Eric
我最近花了一些时间研究如何在PyQt中使用QDataStream与QTreeWidget配合。在我找不到这方面具体示例,并且关于QDataStream的pyqt文档似乎普遍缺乏的情况下,我想挂一个问题在这里,以便日后有需要的人可以参考。如果有人愿意尝试并提供帮助,我会稍等片刻,然后再发布我的进展。
问题是:在PyQt中,我怎样可以使用QDataStream将QTreeWidgetItems保存为本机QT对象并将文件读回来,以精确恢复树结构到之前保存的状态?
Eric
import sip
sip.setapi('QString', 2)
from xml.etree import cElementTree as etree
from PyQt4 import QtGui, QtCore
class Window(QtGui.QWidget):
def __init__(self, xml):
QtGui.QWidget.__init__(self)
self.tree = QtGui.QTreeWidget(self)
self.tree.header().hide()
self.button = QtGui.QPushButton('Export', self)
self.button.clicked[()].connect(self.exportTree)
layout = QtGui.QVBoxLayout(self)
layout.addWidget(self.tree)
layout.addWidget(self.button)
self._array = QtCore.QByteArray()
self._buffer = QtCore.QBuffer(self._array, self)
self._buffer.open(QtCore.QIODevice.ReadWrite)
self._datastream = QtCore.QDataStream(self._buffer)
self.importTree(xml)
def importTree(self, xml):
def build(item, root):
for element in root.getchildren():
child = QtGui.QTreeWidgetItem(item)
data = element.attrib['data'].encode('ascii')
self._array.swap(self._array.fromBase64(data))
self._buffer.reset()
self._datastream >> child
build(child, element)
item.setExpanded(True)
root = etree.fromstring(xml)
build(self.tree.invisibleRootItem(), root)
def exportTree(self):
def build(item, root):
for row in range(item.childCount()):
child = item.child(row)
self._array.clear()
self._buffer.reset()
self._datastream << child
data = self._array.toBase64().data().decode('ascii')
element = etree.SubElement(root, 'node', data=data)
build(child, element)
root = etree.Element('root')
build(self.tree.invisibleRootItem(), root)
from xml.dom import minidom
print(minidom.parseString(etree.tostring(root)).toprettyxml())
if __name__ == '__main__':
import sys
app = QtGui.QApplication(sys.argv)
window = Window("""\
<?xml version="1.0" ?>
<root>
<node data="AAAAAQAAAAEAAAAJAAAAQwAB/////wAA
AAAAAAAAAAEAAAAKAAAAAAYAUgBlAGQ=">
<node data="AAAAAQAAAAEAAAAJAAAAQwAB//8AAP//
//8AAAAAAAEAAAAKAAAAAAgAQwB5AGEAbg==">
<node data="AAAAAQAAAAEAAAAJAAAAQwAB//8AAICA
AAAAAAAAAAEAAAAKAAAAAAoARwByAGUAZQBu"/>
<node data="AAAAAQAAAAEAAAAJAAAAQwAB//8AAAAA
//8AAAAAAAEAAAAKAAAAAAgAQgBsAHUAZQ=="/>
</node>
<node data="AAAAAQAAAAEAAAAJAAAAQwAB/////6Wl
AAAAAAAAAAEAAAAKAAAAAAwATwByAGEAbgBnAGU=">
<node data="AAAAAQAAAAEAAAAJAAAAQwAB//+AgAAA
gIAAAAAAAAEAAAAKAAAAAAwAUAB1AHIAcABsAGU="/>
<node data="AAAAAQAAAAEAAAAJAAAAQwAB//8AAICA
AAAAAAAAAAEAAAAKAAAAAAoARwByAGUAZQBu"/>
</node>
</node>
<node data="AAAAAQAAAAEAAAAJAAAAQwAB//8AAP//
//8AAAAAAAEAAAAKAAAAAAgAQwB5AGEAbg==">
<node data="AAAAAQAAAAEAAAAJAAAAQwAB/////6Wl
AAAAAAAAAAEAAAAKAAAAAAwATwByAGEAbgBnAGU=">
<node data="AAAAAQAAAAEAAAAJAAAAQwAB//8AAP//
//8AAAAAAAEAAAAKAAAAAAgAQwB5AGEAbg=="/>
<node data="AAAAAQAAAAEAAAAJAAAAQwAB//+AgAAA
gIAAAAAAAAEAAAAKAAAAAAwAUAB1AHIAcABsAGU="/>
</node>
<node data="AAAAAQAAAAEAAAAJAAAAQwAB//8AAICA
AAAAAAAAAAEAAAAKAAAAAAoARwByAGUAZQBu">
<node data="AAAAAQAAAAEAAAAJAAAAQwAB//8AAAAA
//8AAAAAAAEAAAAKAAAAAAgAQgBsAHUAZQ=="/>
<node data="AAAAAQAAAAEAAAAJAAAAQwAB/////wAA
AAAAAAAAAAEAAAAKAAAAAAYAUgBlAGQ="/>
</node>
</node>
</root>
""")
window.setGeometry(800, 300, 300, 300)
window.show()
sys.exit(app.exec_())
import sys,os.path
from PyQt4 import QtGui, QtCore
class TreeExperiment(QtGui.QWidget):
def __init__(self,parent=None):
QtGui.QWidget.__init__(self,parent)
self.tree=QtGui.QTreeWidget(self) #
self.tree.setObjectName("treeWidget") #
self.add_button=QtGui.QPushButton("Add", self) # Initialize a simple
self.save_button=QtGui.QPushButton("Save", self) # form containing a
gridlayout = QtGui.QGridLayout(self) # treeWidget, an
gridlayout.addWidget(self.tree,1,0,1,9) # 'Add' button, and a
gridlayout.addWidget(self.add_button,2,0,2,3) # 'Save' button
gridlayout.addWidget(self.save_button,2,3,2,3) #
self.tree.headerItem().setText(0,"Label") #
if os.path.isfile('native_tree_save.qfile'):
# First look for a previously saved tree. If found, define
# it as a QFile named 'file', open it, and define a datastream
# to read from it.
#
# Each tree node is saved to and read from the file in pairs:
# first, the QTreeWidgetItem itself, then the number of children
# the item has so that the tree structure can be re-created
#
# The first item is added directly as the root for simplicity,
# and is sent to the function which begins the tree reconstruction
file = QtCore.QFile('native_tree_save.qfile')
file.open(QtCore.QIODevice.ReadOnly)
datastream = QtCore.QDataStream(file)
child=QtGui.QTreeWidgetItem(self.tree.invisibleRootItem())
child.read(datastream)
num_childs=datastream.readUInt32()
self.restore_item(datastream,child,num_childs)
else: # Otherwise if this is the first use, create a root item
new_item=QtGui.QTreeWidgetItem(self.tree)
self.tree.setCurrentItem(self.tree.topLevelItem(0))
self.tree.currentItem().setText(0,'root')
self.tree.setItemSelected(self.tree.topLevelItem(0),1)
self.tree.setCurrentItem(self.tree.topLevelItem(0))
self.connect(self.add_button, QtCore.SIGNAL("clicked()"), self.add_item)
self.connect(self.save_button, QtCore.SIGNAL("clicked()"), self.save_tree)
self.added_item_count=0
def add_item(self): # Adds an item to whatever is selected
self.added_item_count+=1
label=str(self.added_item_count)
new_item=QtGui.QTreeWidgetItem(self.tree.currentItem())
new_item.setText(0,label)
self.tree.setCurrentItem(new_item)
def restore_item(self,datastream,item,num_childs):
for i in range(0, num_childs):
child=QtGui.QTreeWidgetItem(item)
child.read(datastream)
num_childs=datastream.readUInt32()
self.restore_item(datastream,child,num_childs)
def save_item(self,item,datastream):
num_childs=item.childCount()
for i in range(0,num_childs):
child = item.child(i)
child.write(datastream)
num_childs=child.childCount()
datastream.writeUInt32(num_childs)
self.save_item(child,datastream)
def save_tree(self):
file = QtCore.QFile('native_tree_save.qfile')
file.open(QtCore.QIODevice.WriteOnly)
datastream = QtCore.QDataStream(file)
self.save_item(self.tree.invisibleRootItem(),datastream)
if __name__=='__main__':
app = QtGui.QApplication(sys.argv)
window = TreeExperiment()
window.resize(200, 120)
window.show()
sys.exit(app.exec_())
QTreeWidget只是一个误导。你正在保存的是一个通用的QAbstractItemModel(treeWidget->model()
)——毕竟,QTreeWidget
是一个视图,并且具有内置模型。现在,那些模型项目只是QVariant
s,而那些只是Python类型,但也完全被QDataStream::operator<<
支持。你所需要的就是选择一种树遍历方式(深度优先,广度优先或其他方式),将树中的项目及其深度转储到流中。当你读取流时,这就足够重建整棵树。