Python QFileSystemModel在文件更改时不更新
QFileSystemModel没有显示对文件的更改,我遇到了问题。第一次创建文件时,它会立即显示出来。但当文件本身发生更改时,大小和时间戳不会更新。我曾多次尝试尝试强制更新模型,但都没有真正成功。我所取得的最好成绩就是完全替换了模型。尽管这会导致此错误:Python QFileSystemModel在文件更改时不更新,python,qt,pyqt,updates,qfilesystemmodel,Python,Qt,Pyqt,Updates,Qfilesystemmodel,QFileSystemModel没有显示对文件的更改,我遇到了问题。第一次创建文件时,它会立即显示出来。但当文件本身发生更改时,大小和时间戳不会更新。我曾多次尝试尝试强制更新模型,但都没有真正成功。我所取得的最好成绩就是完全替换了模型。尽管这会导致此错误: QSortFilterProxyModel:将错误模型中的索引传递到mapToSource 下面的测试代码创建了一个空目录的表视图。单击左按钮可创建一个文件(foo.txt)。连续单击将数据附加到文件。据我所知,QFileSystemMod
QSortFilterProxyModel:将错误模型中的索引传递到mapToSource
下面的测试代码创建了一个空目录的表视图。单击左按钮可创建一个文件(foo.txt)。连续单击将数据附加到文件。据我所知,QFileSystemModel不需要刷新,但第二个按钮是我的尝试
任何关于我做错了什么的帮助都将不胜感激
# Testing with python3.6.3 and pip installed pyqt5 5.9.2 in virtualenv on Ubuntu
import os, sys, tempfile
from PyQt5 import QtCore, QtWidgets
class Widget(QtWidgets.QWidget):
def __init__(self, parent=None):
QtWidgets.QWidget.__init__(self, parent)
layout = QtWidgets.QVBoxLayout()
self.setLayout(layout)
self._view = QtWidgets.QTableView()
layout.addWidget(self._view)
self._modify_button = QtWidgets.QPushButton('Create')
layout.addWidget(self._modify_button)
self._refresh_button = QtWidgets.QPushButton('Refresh')
layout.addWidget(self._refresh_button)
self._modify_button.clicked.connect(self._modify)
self._refresh_button.clicked.connect(self._refresh)
self._model, self._proxy = None, None
self.temp_dir = tempfile.TemporaryDirectory(dir=os.path.dirname(os.path.abspath(__file__)))
self.init_model(self.temp_dir.name)
def init_model(self, path):
self._model = QtWidgets.QFileSystemModel()
self._model.setFilter(QtCore.QDir.AllDirs | QtCore.QDir.AllEntries)
self._proxy = QtCore.QSortFilterProxyModel(self)
self._proxy.setSourceModel(self._model)
self._view.setModel(self._proxy)
# self._view.setModel(self._model)
self._model.directoryLoaded.connect(self._loaded)
self._model.setRootPath(path)
def _loaded(self):
path = self._model.rootPath()
source_index = self._model.index(path)
index = self._proxy.mapFromSource(source_index)
self._view.setRootIndex(index)
# self._view.setRootIndex(source_index)
def _modify(self):
"""Create or modify foo.txt..model should see and update"""
self._modify_button.setText('Modify')
file_name = os.path.join(self.temp_dir.name, 'foo.txt')
with open(file_name, 'a') as txt_file:
print('foo', file=txt_file)
# def _refresh(self):
# # This only seems to work once..and its a flawed approach since it requires permission to write
# temp = tempfile.NamedTemporaryFile(dir=self.temp_dir.name)
# def _refresh(self):
# self._model.beginResetModel()
# self._model.endResetModel()
# def _refresh(self):
# self._proxy.setFilterRegExp('foo')
# self._proxy.setFilterRegExp(None)
# self._proxy.invalidate()
# self._proxy.invalidateFilter()
# self._proxy.reset()
#
# root_index = self._model.index(self._model.rootPath())
# rows = self._model.rowCount(root_index)
# proxy_root_index = self._proxy.mapFromSource(root_index)
# topLeft = self._proxy.index(0, 0, proxy_root_index)
# bottomRight = self._proxy.index(rows - 1, self._model.columnCount(proxy_root_index) - 1, proxy_root_index)
# # self._proxy.dataChanged.emit(topLeft, bottomRight)
# self._model.dataChanged.emit(topLeft, bottomRight)
# def _refresh(self):
# # This only seems to work once
# self._model.setRootPath('')
# self._model.setRootPath(self.temp_dir.name)
def _refresh(self):
# This seems heavy handed..but seems to work
# ..though generates "QSortFilterProxyModel: index from wrong model passed to mapToSource" spam in console
self.init_model(self.temp_dir.name)
if __name__ == "__main__":
app = QtWidgets.QApplication(sys.argv)
widget = Widget()
widget.show()
sys.exit(app.exec_())
更新: 从Qt-5.9.4开始,可以使用
Qt_FILESYSTEMMODEL_WATCH_FILES
environment变量打开每个文件的监视(请参阅)。在模型开始缓存有关文件的信息之前,需要将其设置为一个非空值。但请注意,这将为遇到的每个文件添加一个文件监视程序,因此这可能会使它成为一个新的文件
下面留下原始答案作为问题的解释
此问题是由长期存在的Qt错误引起的:。不幸的是,目前看来不太可能很快得到修复。正如bug报告评论中所指出的,问题的核心似乎是: 这是操作系统的限制。对文件的更改并不意味着目录 是修改过的 唯一真正的解决办法是将
QFileSystemWatcher
附加到每个文件,这显然是(在某些平台上)
除此之外,该类目前还没有提供强制刷新的API,并且,正如您所发现的,似乎没有任何可靠的解决方法。SO和其他地方提供的大多数“解决方案”都表明了这一点的一些变体:
root = fsmodel.rootPath()
fsmodel.setRootPath('')
fsmodel.setRootPath(root)
但正如您所知,这似乎只起作用一次——可能是由于当前实现文件信息缓存的方式存在一些怪癖
目前,强制更新的唯一方法似乎是替换整个模型。通过如下重构init_模型
方法,可以防止当前实现产生的错误消息:
def init_model(self, path):
if self._proxy is None:
self._proxy = QtCore.QSortFilterProxyModel(self)
else:
# remove the current source model
self._proxy.setSourceModel(None)
self._model = QtWidgets.QFileSystemModel()
self._model.setFilter(QtCore.QDir.AllDirs | QtCore.QDir.AllEntries)
self._proxy.setSourceModel(self._model)
self._view.setModel(self._proxy)
self._model.directoryLoaded.connect(self._loaded)
self._model.setRootPath(path)
这是一个非常不令人满意的情况,但目前似乎没有任何明显的解决办法。由于Qt v5.9.4可以设置环境变量
Qt\u FILESYSTEMMODEL\u WATCH\u FILES
,您可以在以下网站上阅读更多关于它的内容:
[QTBUG-46684]现在可以通过
设置环境变量QT_FILESYSTEMMODEL_WATCH_FILES,
允许跟踪例如文件大小的更改
两件事:
- 目前,您需要设置它,然后您可以将它设置到另一个文件夹,而不会出现任何问题
- 不过,请注意,这一功能是以潜在的重载为代价的