python QstandardItemModel 自然排序

Posted

技术标签:

【中文标题】python QstandardItemModel 自然排序【英文标题】:python QstandardItemModel Natural sort 【发布时间】:2017-10-05 10:00:45 【问题描述】:

我正在寻找 QstandardItemModel 自然排序方法。我正在阅读一些相关问题并尝试使用我的代码。但它没有工作

我的工具通过拖放接受一些文件列表。我想按自然排序对我的模型数据进行排序。并放到listview中。

所以这是我的代码:

class VideolistView (QtWidgets.QListView):
    def __init__(self, parent):
        super(VideolistView, self).__init__(parent)
        self.setAcceptDrops(True)
        self.setObjectName("VideolistView")
        self.setGeometry(QtCore.QRect(8, 30, 250, 301))
        self.Model = QtGui.QStandardItemModel(self)

    def dragEnterEvent(self, event):
        if event.mimeData().hasUrls:
            event.accept()
        else:
            event.ignore()

    def dragMoveEvent(self, event):
        if event.mimeData().hasUrls:
            event.accept()
        else:
            event.ignore()

    def dropEvent(self, event):
        if event.mimeData().hasUrls:
            event.setDropAction(QtCore.Qt.CopyAction)
            event.accept()

            for url in event.mimeData().urls():
                dropitem = str(url.toLocalFile())
                Fname = os.path.split(dropitem)
                if not self.Model.findItems(Fname[1]):
                    listitem = QtGui.QStandardItem(Fname[1])
                    self.Model.appendRow(listitem)
            self.Model.sort(0)
            self.setModel(self.Model)
        else:
            event.ignore()

我的一些尝试宣布。

import re

def _human_key(key):
    parts = re.split('(\d*\.\d+|\d+)', key)
    return tuple((e.swapcase() if i % 2 == 0 else float(e))
            for i, e in enumerate(parts))
   ......

    listitems=[]
         for url in event.mimeData().urls():
             dropitem = str(url.toLocalFile())
             Fname = os.path.split(dropitem)
             listitem = QtGui.QStandardItem(Fname[1])
             listitems.appent(listitem)
    listitems.sort(key=_human_key)    

Pycharm 调试错误 listitems.sort(key=_human_key) 我尝试重新实现 QstandardItemModel 类

import re

def _human_key(key):
    parts = re.split('(\d*\.\d+|\d+)', key)
    return tuple((e.swapcase() if i % 2 == 0 else float(e))
            for i, e in enumerate(parts))
   ......

Class NaturalSortModel (QtGui.QstandardItemModel)
    def __lt__(self, other):
        column = self.treeWidget().sortColumn()
        k1 = self.text(column)
        k2 = other.text(column)
        return _human_key(k1) < _human_key(k2)
               .......
Class VideolistView (QtWidgets.QListView)
    def __init__(self, parent):
       self.Model = self.NaturalSortModel(self)

               ........
            self.Model.appendRow(listitem)
        self.Model.sort(0)
        self.setModel(self.Model)
                ........

以上两个代码错误。如何使用 QstandardItemModel 进行自然排序?

链接:Python - Human sort of numbers with alpha numeric, but in pyQt and a __lt__ operator

使用@eyllanesc 回答,使用 QSortFilterProxyModel 的第一个解决方案运行良好。 它按自然顺序对数据进行排序。

但是@eyllanesc 第二个建议(Qstandarditem 继承)对我不起作用。它按字母顺序排序,就像原始来源一样。 @eyllanesc 非常感谢!!!

【问题讨论】:

什么意思:但是没用? Google 搜索向我介绍了列表自然排序。所以我用 listitem 列出清单。 Listitems.append(listitem) 和 Listitems.sort(key=nature_sort)。但发生了错误。 我不明白,那是python中列表的排序,它不仅适用于自然排序,而且适用于任何排序标准。 列表和 QStdarItemModel 是非常不同的东西,我试过虽然不是最好的但它确实有效,它以自然的方式对它们进行排序。 有效吗?我的 pycharm 调试和错误。我试过 def __lt__(self, other): return naturalkey(self.value, float) 【参考方案1】:

如果你想建立一个特定的顺序标准,你必须通过一个继承自 QSortFilterProxyModel 的模型来实现,并实现方法 lessThan,如下所示:

class NaturalSortFilterProxyModel(QtCore.QSortFilterProxyModel):
    @staticmethod
    def _human_key(key):
        parts = re.split('(\d*\.\d+|\d+)', key)
        return tuple((e.swapcase() if i % 2 == 0 else float(e)) for i, e in enumerate(parts))

    def lessThan(self, left, right):
        leftData = self.sourceModel().data(left)
        rightData = self.sourceModel().data(right)
        return self._human_key(leftData) < self._human_key(rightData)

class VideolistView (QtWidgets.QListView):
    def __init__(self, parent=None):
        super(VideolistView, self).__init__(parent)
        self.setAcceptDrops(True)
        self.setObjectName("VideolistView")
        self.setGeometry(QtCore.QRect(8, 30, 250, 301))
        model = QtGui.QStandardItemModel(self)
        proxyModel = NaturalSortFilterProxyModel()
        proxyModel.setSourceModel(model)
        proxyModel.sort(0, QtCore.Qt.AscendingOrder)
        self.setModel(proxyModel)

    def dragEnterEvent(self, event):
        if event.mimeData().hasUrls():
            event.accept()
        else:
            event.ignore()

    def dragMoveEvent(self, event):
        if event.mimeData().hasUrls():
            event.accept()
        else:
            event.ignore()

    def dropEvent(self, event):
        if event.mimeData().hasUrls():
            event.setDropAction(QtCore.Qt.CopyAction)
            event.accept()

            for url in event.mimeData().urls():
                dropitem = str(url.toLocalFile())
                Fname = os.path.split(dropitem)
                if not self.model().sourceModel().findItems(Fname[1]):
                    listitem = QtGui.QStandardItem(Fname[1])
                    self.model().sourceModel().appendRow(listitem)
        else:
            event.ignore()

正如@ekhumuro 所说,链接中显示的解决方案是指创建一个继承自QStandardItem 的类,如下所示

class NaturalStandardItem(QtGui.QStandardItem):
    @staticmethod
    def _human_key(key):
        parts = re.split('(\d*\.\d+|\d+)', key)
        return tuple((e.swapcase() if i % 2 == 0 else float(e)) for i, e in enumerate(parts))
    def __lt__(self, other):
        return self._human_key(self.text()) < self._human_key(other.text())


class VideolistView (QtWidgets.QListView):
    def __init__(self, parent=None):
        super(VideolistView, self).__init__(parent)
        self.setAcceptDrops(True)
        self.setObjectName("VideolistView")
        self.setGeometry(QtCore.QRect(8, 30, 250, 301))
        model = QtGui.QStandardItemModel(self)
        self.setModel(model)


    def dragEnterEvent(self, event):
        if event.mimeData().hasUrls():
            event.accept()
        else:
            event.ignore()

    def dragMoveEvent(self, event):
        if event.mimeData().hasUrls():
            event.accept()
        else:
            event.ignore()

    def dropEvent(self, event):
        if event.mimeData().hasUrls():
            event.setDropAction(QtCore.Qt.CopyAction)
            event.accept()

            for url in event.mimeData().urls():
                dropitem = str(url.toLocalFile())
                Fname = os.path.split(dropitem)
                if not self.model().findItems(Fname[1]):
                    listitem = NaturalStandardItem(Fname[1])
                    self.model().appendRow(listitem)
            self.model().sort(0)

        else:
            event.ignore()

【讨论】:

感谢您的帮助。我还没有测试这段代码。但我认为它运作良好。在我测试之后,我记下了我的问题。

以上是关于python QstandardItemModel 自然排序的主要内容,如果未能解决你的问题,请参考以下文章

如何从函数传递 QStandardItemModel?

Qtableview 中的 QStandardItemModel

如何调整 QTreeView 和 QStandardItemModel 中的行大小?

Qt入门教程数据模型篇 QStandardItemModel标准项目模型

Qt入门教程数据模型篇 QStandardItemModel标准项目模型

在 QML TableView 中显示 QStandardItemModel