данные qabstractitemmodel не изменяются в qml

#c #qt #qml #qt5 #qabstractitemmodel

#c #qt #qml #qt5 #qabstractitemmodel

Вопрос:

Я скопировал qabstractitemmodel из примера Qt animal и пытаюсь отобразить его в QML и изменить значения. Я добавил функцию в модель для этого

 Q_INVOKABLE void change()
{
     m_animals.first().m_size="newValue";
     // setData(this->index(0), "newValue", SizeRole); //always returns false, has no effect if uncommented
     qDebug() << this->data(this->index(0), SizeRole); //returns correctly new value as set in previous uncommented line

     emit dataChanged(this->index(0), this->index(this->rowCount()), {SizeRole}); // the value in QML is not updated at any point
}
  

Почему значение не обновляется в QML?

Я загрузил полный образец

https://ufile.io/jfflj

Спасибо.

Ответ №1:

Проблема вызвана тем, что index(rowCount() ) является недопустимым QModelIndex, вместо этого вы должны использовать index(rowCount()- 1) или лучше просто указать, что строка 0 обновляется с помощью index(0) :

 Q_INVOKABLE void change()
{
    m_animals.first().m_size="newValue";
    qDebug() << this->data(this->index(0), SizeRole);
    emit dataChanged(index(0), index(rowCount()-1), {SizeRole});
    // or better
    // emit dataChanged(index(0), index(0), {SizeRole});
}
  

С другой стороны, вы указываете в комментариях, что setData() всегда возвращает false , и это правильно, поскольку при использовании класса QAbstractListModel() в качестве базового вы должны реализовать этот метод:

 bool AnimalModel::setData(const QModelIndex amp;index, const QVariant amp;value, int role)
{
    if(!index.isValid()) return false;
    if (index.row() < 0 || index.row() >= rowCount()) return false;
    Animal amp; animal =  m_animals[index.row()];
    if(role == TypeRole)
        animal.m_type = value.toString();
    else if(role == SizeRole)
        m_animals[index.row()].m_size = value.toString();
    else
        return false;
    emit dataChanged(index, index, {role});
    return true;
}
  

И тогда вы можете использовать это:

 Q_INVOKABLE void change()
{
    setData(index(0), "newValue", SizeRole);
}