如何更改QTableView项目从垂直到水平的方向?

时间:2015-01-16 21:16:25

标签: python qt model pyqt qtableview

下面的代码创建了一个窗口,其中并排放置了两个QTableView:

enter image description here

左侧TableView链接到QAbstractTableModel。根据{{​​1}}中定义的逻辑,tableView的项目是垂直放置的。

右侧TableView链接到Model.data()。我想用它来将垂直TableView项目的位置更改为水平。

请发布您的建议,了解如何实现。如果改变项目方向的解决方案不需要ProxyModel ......只要它有效就行了,这没关系!

以后发布的源代码。

QSortFilterProxyModel

2 个答案:

答案 0 :(得分:2)

一项有趣的任务......

class Model2(QAbstractTableModel):
    def __init__(self, model, parent=None):
        self.model = model
        QAbstractTableModel.__init__(self, parent)

    def rowCount(self):
        return self.model.columnCount()

    def columnCount(self):
        return self.model.rowCount()

    def data(self, a, b):
        return self.model.data(b, a)

答案 1 :(得分:0)

以下是使用Proxy将项目从“垂直”重定向到“水平”的工作解决方案。

诀窍是通过在Proxy模型下声明它来覆盖soruce模型的.data()方法。在Proxy上声明时,它优先于源模型的.data()方法:

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

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

    def flags(self, index):
        return Qt.ItemIsEnabled | Qt.ItemIsSelectable | Qt.ItemIsEditable

    def rowCount(self, parent):
        return len(self.items)       
    def columnCount(self, parent):
        return len(self.items)  

    def data(self, index, role):
        if not index.isValid(): return QVariant()
        elif role != Qt.DisplayRole:
            return QVariant()

        row=index.row()
        if row<len(self.items):
            return QVariant(self.items[row])
        else:
            return QVariant()

    def setData(self, index, value, role=Qt.EditRole):
        if index.isValid():            
            if role == Qt.EditRole:                
                row = index.row()
                self.items[row]=value  
                return True
        return False

class Proxy(QSortFilterProxyModel):
    def __init__(self):
        super(Proxy, self).__init__()        

    def rowCount(self, parent):
        return 1 
    def columnCount(self, parent):
        sourceModel=self.sourceModel()
        return len(sourceModel.items) 

    def filterAcceptsRow(self, row, parent):
        sourceModel=self.sourceModel()
        sourceModelIndex=sourceModel.index(row, 0, QModelIndex())

        sourceModelIndexName=sourceModel.data(sourceModelIndex, Qt.DisplayRole).toString()
        return True

    def data(self, index, role):
        sourceModel=self.sourceModel()
        items=sourceModel.items

        if not index.isValid(): return QVariant()
        elif role != Qt.DisplayRole:
            return QVariant()

        row=index.row()
        column=index.column()

        if column<len(items):
            return QVariant(items[column])
        else:
            return QVariant()


class MyWindow(QWidget):
    def __init__(self, *args):
        QWidget.__init__(self, *args)

        tablemodel=Model(self)               

        proxy=Proxy()
        proxy.setSourceModel(tablemodel)

        tableviewA=QTableView() 
        tableviewA.setModel(tablemodel)

        tableviewB=QTableView() 
        tableviewB.setModel(proxy)

        layout = QHBoxLayout(self)
        layout.addWidget(tableviewA)
        layout.addWidget(tableviewB)
        self.setLayout(layout)

    def test(self, arg):
        print arg

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

这是受mdurant回答启发的第二个解决方案。我在这里起诉两个QAbstractTableModel。第二个模型查询第一个模型self.items数据变量。

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

class Horizontal(QAbstractTableModel):
    def __init__(self, parent=None, *args):
        QAbstractTableModel.__init__(self, parent, *args)
        self.items = ['Row0_Column0','Row0_Column1','Row0_Column2']
        self.childModel=None

    def flags(self, index):
        return Qt.ItemIsEnabled | Qt.ItemIsSelectable | Qt.ItemIsEditable

    def rowCount(self, parent):
        return len(self.items)       
    def columnCount(self, parent):
        return 1

    def data(self, index, role):
        if not index.isValid(): return QVariant()
        elif role != Qt.DisplayRole:
            return QVariant()

        row=index.row()
        if row<len(self.items):
            return QVariant(self.items[row])
        else:
            return QVariant()

    def setData(self, index, value, role=Qt.EditRole):
        if index.isValid():            
            if role == Qt.EditRole:                
                row = index.row()
                self.items[row]=value  
                if self.childModel: self.childModel.reset()
                return True
        return False

class Vertical(QAbstractTableModel):
    def __init__(self, items=None, parent=None, *args):
        QAbstractTableModel.__init__(self, parent, *args)
        self.items = items

    def rowCount(self, parent):
        return 1      
    def columnCount(self, parent):
        return len(self.items)    

    def data(self, index, role):
        if not index.isValid(): return QVariant()
        elif role != Qt.DisplayRole:
            return QVariant()

        column=index.column()
        if column<len(self.items):
            return QVariant(self.items[column])
        else:
            return QVariant()

class MyWindow(QWidget):
    def __init__(self, *args):
        QWidget.__init__(self, *args)

        modelV=Horizontal(self)         

        tableviewA=QTableView() 
        tableviewA.setModel(modelV)

        modelH=Vertical(modelV.items)
        modelV.childModel=modelH

        tableviewB=QTableView() 
        tableviewB.setModel(modelH)

        layout = QHBoxLayout(self)
        layout.addWidget(tableviewA)
        layout.addWidget(tableviewB)
        self.setLayout(layout)

    def test(self, arg):
        print arg

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