Как расширить виджеты Qt, которые по умолчанию не поддерживают модель

CustomMenu класс наследует от QMenu, Его обычай setData() метод принимает аргумент и устанавливает его в переменную класса model, Я сделал это, потому что QToolButton, QToolMenu а также QAction не поддерживает модель / вид рамки. Из того, что я знаю, кроме всех QList-QTable-QTree Views and Trees только QComboBox поддерживает model, Таким образом, вопрос: можно ли будет расширить функциональность других non-model widgets чтобы их можно было использовать как driven-by-model виджеты тоже?

введите описание изображения здесь

from PyQt4.QtCore import *
from PyQt4.QtGui import *
import sys, os

class CustomMenu(QMenu):
    def __init__(self):
        super(CustomMenu, self).__init__()
        self.model=None

    def setModel(self, model):
        self.model=model
        self.pupulate()
    def pupulate(self):
        for item in self.model.items:
            self.addAction(item)

class Model(QAbstractTableModel):
    def __init__(self, parent=None, *args):
        QAbstractTableModel.__init__(self, parent, *args)
        self.items = ['Row0_Column0','Row1_Column0','Row2_Column0']

    def flags(self, index):
        return Qt.ItemIsEnabled | Qt.ItemIsSelectable | Qt.ItemIsEditable
    def rowCount(self, QModelIndex):
        return len(self.items)
    def columnCount(self, QModelIndex):
        return 1

    def data(self, index, role):
        if not index.isValid(): return QVariant()
        if role == Qt.DisplayRole:
            return QVariant(self.items[index.row()])
        return QVariant()
    def setData(self, index, value, role=Qt.EditRole):
        if index.isValid():            
            if role == Qt.EditRole:                
                self.items[index.row()]=value  
                return True
        return False
class MyWindow(QWidget):
    def __init__(self, *args):
        QWidget.__init__(self, *args)
        layout=QVBoxLayout(self)
        self.setLayout(layout)

        tablemodel=Model(self)

        tableView=QTableView() 
        tableView.horizontalHeader().setStretchLastSection(True)
        tableView.setModel(tablemodel)         
        layout.addWidget(tableView)      

        combo=QComboBox()
        combo.setModel(tablemodel)
        layout.addWidget(combo)

        toolButton=QToolButton(self)
        toolButton.setText('Tool Button')
        toolButton.setPopupMode(QToolButton.InstantPopup)
        toolButton.setSizePolicy(QSizePolicy(QSizePolicy.Preferred, QSizePolicy.Fixed))

        menu=CustomMenu()
        menu.setModel(tablemodel)
        toolButton.setMenu(menu)
        layout.addWidget(toolButton)

if __name__ == "__main__":
    app = QApplication(sys.argv)
    w = MyWindow()
    w.show()
    sys.exit(app.exec_())

ИЗМЕНЕНО ПОЗЖЕ: попытка реализовать QDataWidgetMapper():

from PyQt4.QtCore import *
from PyQt4.QtGui import *
import sys, os

class CustomMenuButton(QWidget):
    def __init__(self, parent):
        super(CustomMenuButton, self).__init__(parent)
        self.dataMapper=QDataWidgetMapper()
        layout=QVBoxLayout()
        self.setLayout(layout)

        toolButton=QToolButton(self)
        toolButton.setText('Tool Button')
        toolButton.setPopupMode(QToolButton.InstantPopup)

        self.menu=QMenu(toolButton)
        toolButton.setMenu(self.menu)

    def setModel(self, model):
        self.dataMapper.setModel(model)

        for row in range(model.rowCount()):
            action=self.menu.addAction('Item')

            self.dataMapper.addMapping(action, row)

class Model(QAbstractTableModel):
    def __init__(self, parent=None, *args):
        QAbstractTableModel.__init__(self, parent, *args)
        self.items = ['Row0_Column0','Row1_Column0','Row2_Column0']

    def flags(self, index):
        return Qt.ItemIsEnabled | Qt.ItemIsSelectable | Qt.ItemIsEditable
    def rowCount(self, parent=QModelIndex()):
        return len(self.items)
    def columnCount(self, parent=QModelIndex()):
        return 1

    def data(self, index, role):
        if not index.isValid(): return QVariant()
        if role == Qt.DisplayRole:
            return QVariant(self.items[index.row()])
        return QVariant()
    def setData(self, index, value, role=Qt.EditRole):
        if index.isValid():            
            if role == Qt.EditRole:                
                self.items[index.row()]=value  
                self.dataChanged.emit(index, index)
                return True
        return False

class MyWindow(QWidget):
    def __init__(self, *args):
        QWidget.__init__(self, *args)
        layout=QVBoxLayout(self)
        self.setLayout(layout)

        tablemodel=Model(self)

        tableView=QTableView() 
        tableView.horizontalHeader().setStretchLastSection(True)
        tableView.setModel(tablemodel)         
        layout.addWidget(tableView)      

        combo=QComboBox()
        combo.setModel(tablemodel)
        layout.addWidget(combo)

        menuButton=CustomMenuButton(self)
        layout.addWidget(menuButton)

        menuButton.setModel(tablemodel)

if __name__ == "__main__":
    app = QApplication(sys.argv)
    w = MyWindow()
    w.show()
    sys.exit(app.exec_())

РЕДАКТИРОВАТЬ 2: Определил класс CustomAction(QAction) с dataChanged = pyqtSignal(QModelIndex,QModelIndex)...

from PyQt4.QtCore import *
from PyQt4.QtGui import *
import sys, os

class CustomAction(QAction):
    dataChanged = pyqtSignal(QModelIndex,QModelIndex)
    def __init__(self, name, parent):
        super(CustomAction, self).__init__(name, parent)

class CustomMenuButton(QWidget):
    def __init__(self, parent):
        super(CustomMenuButton, self).__init__(parent)
        self.dataMapper=QDataWidgetMapper()
        layout=QVBoxLayout()
        self.setLayout(layout)

        toolButton=QToolButton(self)
        toolButton.setText('Tool Button')
        toolButton.setPopupMode(QToolButton.InstantPopup)

        self.menu=QMenu(toolButton)
        toolButton.setMenu(self.menu)

    def setModel(self, model):
        self.dataMapper.setModel(model)
        for row in range(model.rowCount()):
            index=model.index(row,0)
            itemName=model.data(index, Qt.DisplayRole).toPyObject()

            actn=CustomAction(itemName, self.menu)
            self.menu.addAction(actn)

            self.dataMapper.addMapping(actn, row)

class Model(QAbstractTableModel):
    def __init__(self, parent=None, *args):
        QAbstractTableModel.__init__(self, parent, *args)
        self.items = ['Row0_Column0','Row1_Column0','Row2_Column0']

    def flags(self, index):
        return Qt.ItemIsEnabled | Qt.ItemIsSelectable | Qt.ItemIsEditable
    def rowCount(self, parent=QModelIndex()):
        return len(self.items)
    def columnCount(self, parent=QModelIndex()):
        return 1

    def data(self, index, role):
        if not index.isValid(): return QVariant()
        if role == Qt.DisplayRole:
            return QVariant(self.items[index.row()])
        return QVariant()
    def setData(self, index, value, role=Qt.EditRole):
        if index.isValid():            
            if role == Qt.EditRole:                
                self.items[index.row()]=value  
                self.dataChanged.emit(index, index)
                return True
        return False

class MyWindow(QWidget):
    def __init__(self, *args):
        QWidget.__init__(self, *args)
        layout=QVBoxLayout(self)
        self.setLayout(layout)

        tablemodel=Model(self)

        tableView=QTableView() 
        tableView.horizontalHeader().setStretchLastSection(True)
        tableView.setModel(tablemodel)         
        layout.addWidget(tableView)      

        combo=QComboBox()
        combo.setModel(tablemodel)
        layout.addWidget(combo)

        menuButton=CustomMenuButton(self)
        layout.addWidget(menuButton)

        menuButton.setModel(tablemodel)

if __name__ == "__main__":
    app = QApplication(sys.argv)
    w = MyWindow()
    w.show()
    sys.exit(app.exec_())

1 ответ

Решение

Это то, для чего был разработан класс QDataWidgetMapper. В следующем блоге есть отличное учебное пособие, рассказывающее, как именно его реализовать: http://www.yasinuludag.com/blog/?p=98

РЕДАКТИРОВАТЬ: я должен добавить, что QDataWidgetMapper работает с QWidget и его подклассы, так QAction не будет применяться. Для классов, которые не QWidget или его подклассы, вам, возможно, придется реализовать собственную привязку вида модели.

Другие вопросы по тегам