Python QFileSystemModel在文件更改时不更新

Python QFileSystemModel在文件更改时不更新,python,qt,pyqt,updates,qfilesystemmodel,Python,Qt,Pyqt,Updates,Qfilesystemmodel,QFileSystemModel没有显示对文件的更改,我遇到了问题。第一次创建文件时,它会立即显示出来。但当文件本身发生更改时,大小和时间戳不会更新。我曾多次尝试尝试强制更新模型,但都没有真正成功。我所取得的最好成绩就是完全替换了模型。尽管这会导致此错误: QSortFilterProxyModel:将错误模型中的索引传递到mapToSource 下面的测试代码创建了一个空目录的表视图。单击左按钮可创建一个文件(foo.txt)。连续单击将数据附加到文件。据我所知,QFileSystemMod

QFileSystemModel没有显示对文件的更改,我遇到了问题。第一次创建文件时,它会立即显示出来。但当文件本身发生更改时,大小和时间戳不会更新。我曾多次尝试尝试强制更新模型,但都没有真正成功。我所取得的最好成绩就是完全替换了模型。尽管这会导致此错误:

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, 允许跟踪例如文件大小的更改

两件事:

  • 目前,您需要设置它,然后您可以将它设置到另一个文件夹,而不会出现任何问题
  • 不过,请注意,这一功能是以潜在的重载为代价的

您可以尝试直接在文件中使用。但是如果你看了很多文件,可能会降低性能。非常感谢你的解释,链接到问题,并修复了我在尝试重新初始化模型时收到的警告!