gpt4 book ai didi

qt - PyQt4 可检查目录 View

转载 作者:行者123 更新时间:2023-12-04 11:39:40 25 4
gpt4 key购买 nike

为了制作可检查的目录 View ,我编写了以下代码。但是在 CheckableDirModel 每次检查一个文件夹时,它必须遍历所有子文件夹来检查它们,这非常慢。我希望有人能帮我解决这个问题。

这就是这个人现在的样子。但它很慢,特别是如果单击一个大文件夹。

enter image description here

代码是可执行的...

from PyQt4 import QtGui, QtCore


class CheckableDirModel(QtGui.QDirModel):
def __init__(self, parent=None):
QtGui.QDirModel.__init__(self, None)
self.checks = {}

def data(self, index, role=QtCore.Qt.DisplayRole):
if role != QtCore.Qt.CheckStateRole:
return QtGui.QDirModel.data(self, index, role)
else:
if index.column() == 0:
return self.checkState(index)

def flags(self, index):
return QtGui.QDirModel.flags(self, index) | QtCore.Qt.ItemIsUserCheckable

def checkState(self, index):
if index in self.checks:
return self.checks[index]
else:
return QtCore.Qt.Unchecked

def setData(self, index, value, role):
if (role == QtCore.Qt.CheckStateRole and index.column() == 0):
self.checks[index] = value
for i in range(self.rowCount(index)):
self.setData(index.child(i,0),value,role)
return True

return QtGui.QDirModel.setData(self, index, value, role)

def exportChecked(self, acceptedSuffix=['jpg', 'png', 'bmp']):
selection=[]
for c in self.checks.keys():
if self.checks[c]==QtCore.Qt.Checked and self.fileInfo(c).completeSuffix().toLower() in acceptedSuffix:
try:

selection.append(self.filePath(c).toUtf8())
except:
pass
return selection

if __name__ == '__main__':
import sys

app = QtGui.QApplication(sys.argv)

model = QtGui.QDirModel()
tree = QtGui.QTreeView()
tree.setModel(CheckableDirModel())

tree.setAnimated(False)
tree.setIndentation(20)
tree.setSortingEnabled(True)

tree.setWindowTitle("Dir View")
tree.resize(640, 480)
tree.show()

sys.exit(app.exec_())

最佳答案

您不能单独存储每个文件的复选框状态。它们可能太多了。我建议您执行以下操作:

您保留用户实际单击的索引的复选框值列表。当用户单击某些内容时,您将一个条目添加到列表中(如果它已经存在,则更新它),然后删除列表中存在的子索引的所有条目。您需要发出有关父索引数据的信号,并且所有子索引都已更改。

当请求复选框值时(通过调用模型的 data()),您在列表中搜索请求的索引并返回其值。如果该索引不存在于列表中,则搜索最近的父索引并返回其值。

请注意,除了执行缓慢之外,您的代码还有另一个问题。当文件树的层数过多时,会发生“超出最大递归深度”异常。在实现我的建议时,不要以这种方式使用递归。文件树深度几乎是无限的。

这是实现:

from collections import deque

def are_parent_and_child(parent, child):
while child.isValid():
if child == parent:
return True
child = child.parent()
return False


class CheckableDirModel(QtGui.QDirModel):
def __init__(self, parent=None):
QtGui.QDirModel.__init__(self, None)
self.checks = {}

def data(self, index, role=QtCore.Qt.DisplayRole):
if role == QtCore.Qt.CheckStateRole and index.column() == 0:
return self.checkState(index)
return QtGui.QDirModel.data(self, index, role)

def flags(self, index):
return QtGui.QDirModel.flags(self, index) | QtCore.Qt.ItemIsUserCheckable

def checkState(self, index):
while index.isValid():
if index in self.checks:
return self.checks[index]
index = index.parent()
return QtCore.Qt.Unchecked

def setData(self, index, value, role):
if role == QtCore.Qt.CheckStateRole and index.column() == 0:
self.layoutAboutToBeChanged.emit()
for i, v in self.checks.items():
if are_parent_and_child(index, i):
self.checks.pop(i)
self.checks[index] = value
self.layoutChanged.emit()
return True

return QtGui.QDirModel.setData(self, index, value, role)

def exportChecked(self, acceptedSuffix=['jpg', 'png', 'bmp']):
selection=set()
for index in self.checks.keys():
if self.checks[index] == QtCore.Qt.Checked:
for path, dirs, files in os.walk(unicode(self.filePath(index))):
for filename in files:
if QtCore.QFileInfo(filename).completeSuffix().toLower() in acceptedSuffix:
if self.checkState(self.index(os.path.join(path, filename))) == QtCore.Qt.Checked:
try:
selection.add(os.path.join(path, filename))
except:
pass
return selection

我没有找到使用 dataChanged 的方法通知 View 所有子索引的数据已更改的信号。我们不知道当前显示了哪些索引,并且我们无法通知每个子索引,因为它可能很慢。所以我用了 layoutAboutToBeChangedlayoutChanged强制 View 更新所有数据。看来这个方法已经够快了。
exportChecked有点复杂。它没有优化,有时一个索引被处理很多次。我用过 set()过滤重复项。如果它工作得太慢,也许可以以某种方式对其进行优化。但是,如果用户检查了一些包含许多文件和子目录的海量目录,则该功能的任何实现都会很慢。所以优化没有意义,尽量不要经常调用这个函数。

关于qt - PyQt4 可检查目录 View ,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/16997793/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com