从 PyQt5 迁移到 PySide2 时的 QVariant 替代方案

San*_*era 5 python pyqt5 pyside2

由于以下代码,我从 PyQt5 切换到 PySide2 时遇到一些问题:

class EnumModel(QtCore.QAbstractListModel):

    def __init__(self, list_of_enums):
        """
        Enumeration model
        :param list_of_enums: list of enumeration values to show
        """
        QtCore.QAbstractListModel.__init__(self)
        self.items = list_of_enums

    def rowCount(self, parent=QtCore.QModelIndex()):
        return len(self.items)

    def data(self, index, role=QtCore.Qt.DisplayRole):
        if index.isValid() is True:
            if role == QtCore.Qt.DisplayRole:
                return QtCore.QVariant(self.items[index.row()].value[0])
            elif role == QtCore.Qt.ItemDataRole:
                return QtCore.QVariant(self.items[index.row()].value[0])
        return QtCore.QVariant()
Run Code Online (Sandbox Code Playgroud)

代码在 PyQt5 下运行良好。

在我的迁移过程中我发现官方网站说:

PySide 仅支持 PyQt 的 API 2(请参阅 PSEP 101)了解详细信息。因此 Qt 类(例如 QStrings、QStringLists 和 QVariants)在 PySide 上不可用。相反,您应该简单地使用本机 Python 数据类型。

所以解决方案是(我猜)简单地更改QVariantstr. 当我这样做时,该类不会抛出任何错误,但它也不显示模型。

事实上,该函数data正在接收role=13而不是role=QtCore.Qt.DisplayRole

我不知道这是否是 PySide2 的一个错误(在 Linux 下有点错误)或者是因为其他原因。

一个最小可行的例子是这样的:

from PySide2.QtWidgets import *
from PySide2 import QtCore
from enum import Enum


class SomeEnum(Enum):
    A = 'A'
    B = 'B'
    C = 'C'


class EnumModel(QtCore.QAbstractListModel):

    def __init__(self, list_of_enums):
        """
        Enumeration model
        :param list_of_enums: list of enumeration values to show
        """
        QtCore.QAbstractListModel.__init__(self)
        self.items = list_of_enums

    def rowCount(self, parent=QtCore.QModelIndex()):
        return len(self.items)

    def data(self, index, role=QtCore.Qt.DisplayRole):
        if index.isValid() is True:
            if role == QtCore.Qt.DisplayRole:
                return self.items[index.row()].value[0]
            elif role == QtCore.Qt.ItemDataRole:
                return self.items[index.row()].value[0]
            else:
                print('not recognised')
        return ""


if __name__ == '__main__':
    import sys

    model = EnumModel([SomeEnum.A, SomeEnum.A, SomeEnum.B, SomeEnum.C])

    app = QApplication(sys.argv)
    lst = QListView()
    lst.setModel(model)
    lst.show()
    sys.exit(app.exec_())
Run Code Online (Sandbox Code Playgroud)

eyl*_*esc 3

问题是,当视图需要与角色Qt::SizeHintRole(13) 相关的信息时,您将向其传递一个空字符串,而必须返回 None 或干脆不返回任何内容,因为它会干扰其他角色:

def data(self, index, role=QtCore.Qt.DisplayRole):
    if index.isValid():
        print(role)
        if role == QtCore.Qt.DisplayRole:
            return self.items[index.row()].value[0]
        elif role == QtCore.Qt.ItemDataRole:
            return self.items[index.row()].value[0]
        else:
            print('not recognised')
Run Code Online (Sandbox Code Playgroud)