qt — QML: GridView не обновляется после изменения своей модели в переполнении стека

Моя отправная точка — следующий исходный код QML, где GridView показывает ListModel с хорошей анимацией перестановки элементов:

import QtQuick 1.1
GridView {
id: mainGrid
width: 825; height: 400; cellWidth: 165; cellHeight: 95
model: myModel
delegate: myButton
ListModel {
id: myModel
function createModel() {
for (var i=1; i<=20; i++) {
append({"display":i})
}
}
function listItems() {
console.log("LIST OF ITEMS")
for (var i=0; i<=19; i++) {
console.log("model["+i+"]="+get(i).display)
}
}
function moveItem(inx1,inx2) {
move(inx1,inx2,1)
}
Component.onCompleted: {
createModel()
listItems()
}
}
Component {
id: myButton
Item {
id: item
width: 160; height: 90
Rectangle {
id: box
parent: mainGrid
x: item.x; y: item.y; width: item.width; height: item.height; border.width: 1
Behavior on x {NumberAnimation { duration: 1000;} }
Behavior on y {NumberAnimation { duration: 1000;} }
Text {
anchors.centerIn: parent
text: display
}
MouseArea {
anchors.fill: parent
onClicked: {
mainGrid.model.moveItem(index,0)
mainGrid.model.listItems()
}
}
}
}
}
}

Вместо QML ListModel я хочу использовать модель C ++, которая является производной от QStandardItemModel:

#include <QStandardItemModel>
class MyTable : public QStandardItemModel
{
private:
Q_OBJECT
int table[20];
public:
explicit MyTable(QObject *parent = 0) : QStandardItemModel(parent) {}
Q_INVOKABLE void createModel();
Q_INVOKABLE void moveItem(int inx1, int inx2);
};

Эта модель подвергается QML через qmlRegisterType<MyTable,1>("MyTable",1,0,"MyTable"),
Проблема в том, что я не получаю хороший эффект анимации, и я также не уверен, что после замены модель корректно обновляется.

Связанные (проблемные) функции C ++ находятся здесь:

void MyTable::createModel()
{
for (unsigned int i=0; i<20; i++) {
table[i]=i;
QStandardItem* item = new QStandardItem(QString("%1").arg(table[i]));
appendRow(item);
}
}

void MyTable::moveItem(int inx1, int inx2) {
//SOME MATH, IT WORKS
int tmp = table[inx1];
if (inx1<inx2) {
for (int i=inx1; i<inx2; i++) table[i] = table[i+1];
table[inx2] = tmp;
} else if (inx2<inx1) {
for (int i=inx1; i>inx2; i--) table[i] = table[i-1];
table[inx2] = tmp;
}
//UPDATE MODEL - IS SOMETHING MISSING HERE?
for (unsigned int i=0; i<20; i++) {
QStandardItem* item = new QStandardItem(QString("%1").arg(table[i]));
setItem(i,0,item);
}
}

3

Решение

Я нашел решение. Вот рабочий код (спасибо Borut123 за предложения).

class MyTable : public QStandardItemModel
{
private:
Q_OBJECT
public:
explicit MyTable(QObject *parent = 0) : QStandardItemModel(parent) {}
Q_INVOKABLE void createModel();
Q_INVOKABLE void moveItem(int inx1, int inx2);
Q_INVOKABLE void listItems();
};

void MyTable::createModel()
{
for (unsigned int i=0; i<20; i++) {
QStandardItem* item = new QStandardItem(QString("%1").arg(i));
appendRow(item);
}
}

void MyTable::moveItem(int inx1, int inx2) {
bool ok = false;
if (inx1 < inx2) ok = beginMoveRows(QModelIndex(),inx1,inx1,QModelIndex(),inx2+1);
if (inx1 > inx2) ok = beginMoveRows(QModelIndex(),inx1,inx1,QModelIndex(),inx2);
if (!ok) return;
QStandardItem* tmp1 = takeItem(inx1);
if (inx1<inx2) {
for (int i=inx1; i<inx2; i++) {
QStandardItem* tmp2 = takeItem(i+1);
setItem(i,tmp2);
}
} else if (inx2<inx1) {
for (int i=inx1; i>inx2; i--) {
QStandardItem* tmp2 = takeItem(i-1);
setItem(i,tmp2);
}
}
setItem(inx2,tmp1);
endMoveRows();
}

void MyTable::listItems() {
qDebug() << "LIST OF ITEMS";
for (unsigned int i=0; i<20; i++) {
qDebug() << QString("table[%1]=%2").arg(i).arg(item(i)->text()).toAscii().data();
}
}
2

Другие решения

Других решений пока нет …