Ok, Twój komentarz przypomniał mi, że rzeczywiście musisz ponownie zaimplementować data()
ze względu na model QML. Czemu? Ponieważ model QML wywołuje data()
z rolami nadanymi przez roleName(). Nie wywołuje funkcji data()
z Qt::DisplayRole
jak w świecie QWidget. Ponadto musisz zdefiniować TableViewColumn
z nazwami ról, w przeciwnym razie model nie wywoła funkcji data()
. Oto przykład ponownego zaimplementowania metody data()
:
import sys
from PyQt5.QtCore import QUrl, Qt, QVariant
from PyQt5.QtCore import QObject, pyqtSlot
from PyQt5.QtGui import QGuiApplication
from PyQt5.QtQuick import QQuickView
from PyQt5.QtSql import QSqlDatabase, QSqlQuery, QSqlQueryModel
class QtTabModel(QSqlQueryModel):
def __init__(self):
super(QtTabModel, self).__init__()
def roleNames(self):
roles = {
Qt.UserRole + 1 : 'id',
Qt.UserRole + 2 : 'name'
}
return roles
def data(self, index, role):
if role < Qt.UserRole:
# caller requests non-UserRole data, just pass to papa
return super(QtTabModel, self).data(index, role)
# caller requests UserRole data, convert role to column (role - Qt.UserRole -1) to return correct data
return super(QtTabModel, self).data(self.index(index.row(), role - Qt.UserRole -1), Qt.DisplayRole)
@pyqtSlot(result=QVariant) # don't know how to return a python array/list, so just use QVariant
def roleNameArray(self):
# This method is used to return a list that QML understands
list = []
# list = self.roleNames().items()
for key, value in self.roleNames().items():
list.append(value)
return QVariant(list)
Dodaj TableViewColumn
do TableView
. Pamiętaj, że w rolach rozróżniana jest wielkość liter. Muszą dokładnie odpowiadać temu, co zwraca funkcja roleNames():
import QtQuick 2.2
import QtQuick.Controls 1.1
TableView {
width: 200
height: 300
model: tabmodel
TableViewColumn {
role: "id" // case-sensitive, must match a role returned by roleNames()
}
TableViewColumn {
role: "name"
}
}
Oto sposób automatycznego generowania TableViewColumn. Wywołuje slot roleNameArray zdefiniowany w powyższym kodzie Pythona, aby uzyskać listę nazw ról. Nie wywołujemy tutaj funkcji roleNames(), ponieważ nie wiem, jak sprawić, by QML rozumiał zwracany wynik :), więc musimy przekonwertować go na listę. Na koniec przeglądamy listę i wywołujemy TableView.addColumn, aby utworzyć kolumny:
TableView {
width: 200
height: 300
model: tabmodel
Component.onCompleted: {
var roles = model.roleNameArray()
for (var i=0; i<roles.length; i++) {
var column = addColumn( Qt.createQmlObject(
"import QtQuick.Controls 1.1; TableViewColumn {}",
this) )
column.role = roles[i]
column.title = roles[i]
}
}
}