如何在 QTableWidget 中自定义排序行为

How to customize sorting behaviour in QTableWidget

如何更改列排序行为,而不是像这样按字母顺序排序:

1, 11.1, 2, 22.2, 3, etc.

它将按数字排序,如下所示:

1, 2, 3, 11.1, 22.2, etc

我只需要改变特定列的排序行为,而不是整个 table,因为我想保持名称的正常字母顺序。

我还有一个包含文件大小的列。它们以 MB 和 GB 表示,我想对其进行排序,以便 800 MB 排在前面,1.4 GB 排在后面。它也有字母排序的问题。有什么方法可以做到这一点吗?

可能最简单和最灵活的方法是将原始值(用于排序)与格式化值(用于显示)一起存储。然后,您可以子类化 QTableWidgetItem 并重新实现它的 less than operator,以便在排序时只比较原始值。这个子类可以选择性地用于需要特殊排序的列。

为了使它能正确处理文件大小,您需要将原始值存储为字节数,以便可以以相同的方式比较所有格式。

这是一个简单的演示,展示了如何实现它:

import sys
from PyQt5 import QtCore, QtWidgets

class NumericItem(QtWidgets.QTableWidgetItem):
    def __lt__(self, other):
        return (self.data(QtCore.Qt.UserRole) <
                other.data(QtCore.Qt.UserRole))

class Window(QtWidgets.QTableWidget):
    def __init__(self):
        super(Window, self).__init__(6, 3)
        for column, values in enumerate((
            ('Red', 'Green', 'Yellow', 'Blue', 'White', 'Black'),
            (1, 11.1, 2, 22.2, 3, 17),
            (37885792, 755, 25504, 4805, 3751225472, 14529792),
            )):
            for row, value in enumerate(values):
                if column == 0:
                    item = QtWidgets.QTableWidgetItem(value)
                else:
                    if column == 1:
                        text = str(value)
                    else:
                        text = self.formatSize(value)
                    item = NumericItem(text)
                    item.setData(QtCore.Qt.UserRole, value)
                self.setItem(row, column, item)
        self.setSortingEnabled(True)
        self.sortItems(0, QtCore.Qt.AscendingOrder)

    def formatSize(self, size, precision=2):
        if size < 0:
            return ''
        if size < 1024:
            return '%.0f B' % size
        size /= 1024.0
        if size < 1024:
            return '%.*f KiB' % (precision, size)
        size /= 1024
        if size < 1024:
            return '%.*f MiB' % (precision, size)
        return '%.*f GiB' % (precision, size / 1024)

if __name__ == '__main__':

    app = QtWidgets.QApplication(sys.argv)
    window = Window()
    window.setGeometry(600, 100, 350, 250)
    window.show()
    sys.exit(app.exec_())