在 QTableWidget 中对十六进制数进行排序的最佳方法

Best way of sorting hexadecimal numbers inside the QTableWidget

我想知道在 QTableWidget 中对十六进制数进行排序的最佳方法。

目前我所做的是转换所有十六进制值然后对其进行排序 (ascending/descending)。但我的问题是因为我确实将它转换为十进制。 table 中显示的值是十进制的。我想在不更改 table

中显示的值的情况下对十六进制进行排序

这是我当前的实现:

from PySide2.QtCore import Qt
from PySide2.QtWidgets import QTableWidget

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

class Window(QTableWidget):
    def __init__(self):
        super(Window, self).__init__(4, 2)
        for column, values in enumerate((
            ('ABCD', '1DCA', 'BD23', 'FFFFFFFF'),
            (1,2,3,4)
            )):
            for row, value in enumerate(values):
                if column == 0:
                    value = str(int(value, 16))
                else:
                    value = str(value)
                item = NumericData(value)
                item.setData(Qt.UserRole, value)
                self.setItem(row, column, item)
        self.setSortingEnabled(True)
        self.sortItems(0, Qt.AscendingOrder)

这是示例table

|--Values--|
|ABCD      |
|1BCD      |
|DEC1      |
|123       |
|2105      |
|AAAAAAAA  |

不必为了保存转换而使用新角色,因为如果用户更改值,您将不得不重新计算该值,您只能传入角色 Qt::DisplayRole。

考虑到这一点,我实现了以下示例:

from PySide2 import QtCore, QtGui, QtWidgets


class HexDelegate(QtWidgets.QStyledItemDelegate):
    def createEditor(self, parent, option, index):
        editor = QtWidgets.QSpinBox(parent)
        editor.setMaximum(2147483647)
        editor.setMinimum(-2147483647 - 1)
        editor.setDisplayIntegerBase(16)
        fnt = editor.font()
        fnt.setCapitalization(QtGui.QFont.AllUppercase)
        editor.setFont(fnt)
        return editor

    def setEditorData(self, editor, index):
        editor.setValue(int(index.data(QtCore.Qt.DisplayRole), 16))

    def setModelData(self, editor, model, index):
        value = ('{:x}'.format(editor.value())).upper()
        model.setData(index, value, QtCore.Qt.DisplayRole)


class NumericData(QtWidgets.QTableWidgetItem):
    def __lt__(self, other):
        return int(self.data(QtCore.Qt.DisplayRole), 16) < int(
            other.data(QtCore.Qt.DisplayRole), 16
        )


class Window(QtWidgets.QTableWidget):
    def __init__(self):
        super(Window, self).__init__(4, 2)
        delegate = HexDelegate(self)
        self.setItemDelegateForColumn(0, delegate)
        data = ("ABCD", "1DCA", "BD23", "FFFF"), (1, 2, 3, 4)
        for row, values in enumerate(zip(*data)):
            for col, value in enumerate(values):
                it = QtWidgets.QTableWidgetItem()
                it.setData(QtCore.Qt.DisplayRole, value)
                self.setItem(row, col, it)
        self.setSortingEnabled(True)
        self.sortItems(0, QtCore.Qt.AscendingOrder)


if __name__ == "__main__":
    import sys

    app = QtWidgets.QApplication(sys.argv)
    w = Window()
    w.show()
    sys.exit(app.exec_())