如何在没有元类冲突的情况下对PyQt子类使用泛型类型?

问题描述 投票:7回答:1

我曾尝试使用sip包装类型的abc.ABCMeta,当使用abc.ABC的子类时,它运行良好。

class QABCMeta(wrappertype, ABCMeta):
    pass

class WidgetBase(QWidget, metaclass=QABCMeta):
    ...

class InterfaceWidget(WidgetBase, ABC):
    ...

class MainWidget(InterfaceWidget):
    ...

但它不适用于typing.Generic

class QGenericMeta(wrappertype, GenericMeta):
    pass

class WidgetBase(QWidget, Generic[T], metaclass=QGenericMeta):
    ...

class GenericWidget(WidgetBase[float]):
    ...

它提出:

line 980, in __new__
    self if not origin else origin._gorg)
TypeError: can't apply this __setattr__ to sip.wrappertype object

我希望它像往常一样使用泛型子类:

class TableBase(QTableWidget, Generic[T]):
    @abstractmethod
    def raw_item(self, row: int) -> T:
        ...
    def data(self) -> Iterator[T]:
        yield from (self.raw_item(row) for row in range(self.rowCount()))

class MainTable(TableBase[float]):
    def raw_item(self, row: int) -> float:
        return float(self.item(row, 1).text())  # implementation

table = MainTable()
for data in table.data():
    data: float

但是,如果没有继承dataAny仍然是Generic[T]

可以通过PEP 560解决类型检查吗?

python pyqt5 metaclass typing pyside2
1个回答
3
投票

好吧,我找到了答案。

由于typing.Generic的元类是abc.ABC,它也应该基于abc.ABCMeta。但这只适用于Python 3.7或更高版本。

然后,只需使用type(QObject)而不是sip.wrappertype

# -*- coding: utf-8 -*-

from abc import abstractmethod, ABC, ABCMeta
from typing import TypeVar, Generic, Iterator
from PyQt5.QtCore import QObject
from PyQt5.QtWidgets import QTableWidget

QObjectType = type(QObject)
T = TypeVar('T')


class QABCMeta(QObjectType, ABCMeta):
    pass


class BaseWidget(QTableWidget, Generic[T], metaclass=QABCMeta):

    @abstractmethod
    def raw_item(self, row: int) -> T:
        ...

    def data(self) -> Iterator[T]:
        yield from (self.raw_item(row) for row in range(self.rowCount()))


class TestWidget(BaseWidget[float], ABC):  # optional inherit ABC.

    def raw_item(self, row: int) -> float:
        return float(self.item(row, 1).text())


if __name__ == '__main__':
    w = TestWidget()
    for f in w.data():
        pass

此代码适用于PyCharm IDE,变量f的注释是float

当将PyQt5改为PySide2时,它也有效!

© www.soinside.com 2019 - 2024. All rights reserved.