簡體   English   中英

python QstandardItemModel自然排序

[英]python QstandardItemModel Natural sort

我在尋找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-帶有字母數字的人類數字,但使用pyQt和__lt__運算符

使用@eyllanesc Answer,使用QSortFilterProxyModel的第一個解決方案效果很好。 它按自然順序對數據進行排序。

但是@eyllanesc第二個建議(Qstandarditem繼承)對我不起作用。 它按字母順序排序,就像原始來源一樣。 @eyllanesc非常感謝!!!

如果要建立特定的訂單標准,則必須通過從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()

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM