QmlObjectListModel.cc 8.43 KB
Newer Older
1 2
/****************************************************************************
 *
Gus Grubba's avatar
Gus Grubba committed
3
 * (c) 2009-2020 QGROUNDCONTROL PROJECT <http://www.qgroundcontrol.org>
4 5 6 7 8 9
 *
 * QGroundControl is licensed according to the terms in the file
 * COPYING.md in the root of the source code directory.
 *
 ****************************************************************************/

10 11 12 13 14 15 16

/// @file
///     @author Don Gagne <don@thegagnes.com>

#include "QmlObjectListModel.h"

#include <QDebug>
Don Gagne's avatar
Don Gagne committed
17
#include <QQmlEngine>
18 19

const int QmlObjectListModel::ObjectRole = Qt::UserRole;
20
const int QmlObjectListModel::TextRole = Qt::UserRole + 1;
21 22

QmlObjectListModel::QmlObjectListModel(QObject* parent)
23 24 25 26
    : QAbstractListModel        (parent)
    , _dirty                    (false)
    , _skipDirtyFirstItem       (false)
    , _externalBeginResetModel  (false)
27 28 29 30 31 32 33 34 35
{

}

QmlObjectListModel::~QmlObjectListModel()
{
    
}

Gus Grubba's avatar
Gus Grubba committed
36 37 38 39 40 41 42 43
QObject* QmlObjectListModel::get(int index)
{
    if (index < 0 || index >= _objectList.count()) {
        return nullptr;
    }
    return _objectList[index];
}

44 45 46 47 48 49 50 51 52 53 54 55 56
int QmlObjectListModel::rowCount(const QModelIndex& parent) const
{
    Q_UNUSED(parent);
    
    return _objectList.count();
}

QVariant QmlObjectListModel::data(const QModelIndex &index, int role) const
{
    if (!index.isValid()) {
        return QVariant();
    }
    
Gus Grubba's avatar
Gus Grubba committed
57
    if (index.row() < 0 || index.row() >= _objectList.count()) {
58 59 60 61 62
        return QVariant();
    }
    
    if (role == ObjectRole) {
        return QVariant::fromValue(_objectList[index.row()]);
63 64
    } else if (role == TextRole) {
        return QVariant::fromValue(_objectList[index.row()]->objectName());
65 66 67 68 69 70 71 72 73 74
    } else {
        return QVariant();
    }
}

QHash<int, QByteArray> QmlObjectListModel::roleNames(void) const
{
    QHash<int, QByteArray> hash;
    
    hash[ObjectRole] = "object";
75
    hash[TextRole] = "text";
76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94
    
    return hash;
}

bool QmlObjectListModel::setData(const QModelIndex& index, const QVariant& value, int role)
{
    if (index.isValid() && role == ObjectRole) {
        _objectList.replace(index.row(), value.value<QObject*>());
        emit dataChanged(index, index);
        return true;
    }
    
    return false;
}

bool QmlObjectListModel::insertRows(int position, int rows, const QModelIndex& parent)
{
    Q_UNUSED(parent);
    
95 96 97 98
    if (position < 0 || position > _objectList.count() + 1) {
        qWarning() << "Invalid position position:count" << position << _objectList.count();
    }
    
99 100 101 102 103 104 105 106 107 108 109 110
    beginInsertRows(QModelIndex(), position, position + rows - 1);
    endInsertRows();
    
    emit countChanged(count());
    
    return true;
}

bool QmlObjectListModel::removeRows(int position, int rows, const QModelIndex& parent)
{
    Q_UNUSED(parent);
    
111 112 113 114 115 116
    if (position < 0 || position >= _objectList.count()) {
        qWarning() << "Invalid position position:count" << position << _objectList.count();
    } else if (position + rows > _objectList.count()) {
        qWarning() << "Invalid rows position:rows:count" << position << rows << _objectList.count();
    }
    
117 118 119 120 121 122 123 124 125 126 127
    beginRemoveRows(QModelIndex(), position, position + rows - 1);
    for (int row=0; row<rows; row++) {
        _objectList.removeAt(position);
    }
    endRemoveRows();
    
    emit countChanged(count());
    
    return true;
}

128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143
void QmlObjectListModel::move(int from, int to)
{
    if(0 <= from && from < count() && 0 <= to && to < count() && from != to) {
        // Workaround to allow move item to the bottom. Done according to
        // beginMoveRows() documentation and implementation specificity:
        // https://doc.qt.io/qt-5/qabstractitemmodel.html#beginMoveRows
        // (see 3rd picture explanation there)
        if(from == to - 1) {
            to = from++;
        }
        beginMoveRows(QModelIndex(), from, from, QModelIndex(), to);
        _objectList.move(from, to);
        endMoveRows();
    }
}

Don Gagne's avatar
Don Gagne committed
144 145
QObject* QmlObjectListModel::operator[](int index)
{
Gus Grubba's avatar
Gus Grubba committed
146
    if (index < 0 || index >= _objectList.count()) {
147
        return nullptr;
Gus Grubba's avatar
Gus Grubba committed
148
    }
Don Gagne's avatar
Don Gagne committed
149 150 151 152
    return _objectList[index];
}

const QObject* QmlObjectListModel::operator[](int index) const
153
{
Gus Grubba's avatar
Gus Grubba committed
154
    if (index < 0 || index >= _objectList.count()) {
155
        return nullptr;
Gus Grubba's avatar
Gus Grubba committed
156
    }
157 158 159
    return _objectList[index];
}

Gus Grubba's avatar
Gus Grubba committed
160
void QmlObjectListModel::clear()
161
{
162 163 164 165 166 167 168
    if (!_externalBeginResetModel) {
        beginResetModel();
    }
    _objectList.clear();
    if (!_externalBeginResetModel) {
        endResetModel();
        emit countChanged(count());
169 170 171
    }
}

Don Gagne's avatar
Don Gagne committed
172
QObject* QmlObjectListModel::removeAt(int i)
173
{
Don Gagne's avatar
Don Gagne committed
174
    QObject* removedObject = _objectList[i];
175 176 177 178 179 180
    if(removedObject) {
        // Look for a dirtyChanged signal on the object
        if (_objectList[i]->metaObject()->indexOfSignal(QMetaObject::normalizedSignature("dirtyChanged(bool)")) != -1) {
            if (!_skipDirtyFirstItem || i != 0) {
                QObject::disconnect(_objectList[i], SIGNAL(dirtyChanged(bool)), this, SLOT(_childDirtyChanged(bool)));
            }
Don Gagne's avatar
Don Gagne committed
181
        }
Don Gagne's avatar
Don Gagne committed
182
    }
183
    removeRows(i, 1);
184
    setDirty(true);
Don Gagne's avatar
Don Gagne committed
185
    return removedObject;
186 187
}

188 189 190 191 192
void QmlObjectListModel::insert(int i, QObject* object)
{
    if (i < 0 || i > _objectList.count()) {
        qWarning() << "Invalid index index:count" << i << _objectList.count();
    }
Gus Grubba's avatar
Gus Grubba committed
193 194 195 196 197 198 199
    if(object) {
        QQmlEngine::setObjectOwnership(object, QQmlEngine::CppOwnership);
        // Look for a dirtyChanged signal on the object
        if (object->metaObject()->indexOfSignal(QMetaObject::normalizedSignature("dirtyChanged(bool)")) != -1) {
            if (!_skipDirtyFirstItem || i != 0) {
                QObject::connect(object, SIGNAL(dirtyChanged(bool)), this, SLOT(_childDirtyChanged(bool)));
            }
Don Gagne's avatar
Don Gagne committed
200
        }
Don Gagne's avatar
Don Gagne committed
201
    }
202 203
    _objectList.insert(i, object);
    insertRows(i, 1);
204
    setDirty(true);
205 206
}

207 208 209 210 211 212 213
void QmlObjectListModel::insert(int i, QList<QObject*> objects)
{
    if (i < 0 || i > _objectList.count()) {
        qWarning() << "Invalid index index:count" << i << _objectList.count();
    }

    int j = i;
214
    for (QObject* object: objects) {
215 216 217 218 219 220 221 222 223 224
        QQmlEngine::setObjectOwnership(object, QQmlEngine::CppOwnership);

        // Look for a dirtyChanged signal on the object
        if (object->metaObject()->indexOfSignal(QMetaObject::normalizedSignature("dirtyChanged(bool)")) != -1) {
            if (!_skipDirtyFirstItem || j != 0) {
                QObject::connect(object, SIGNAL(dirtyChanged(bool)), this, SLOT(_childDirtyChanged(bool)));
            }
        }
        j++;

225
        _objectList.insert(j, object);
226 227 228 229 230 231 232
    }

    insertRows(i, objects.count());

    setDirty(true);
}

233 234
void QmlObjectListModel::append(QObject* object)
{
235
    insert(_objectList.count(), object);
236 237
}

238 239 240 241 242
void QmlObjectListModel::append(QList<QObject*> objects)
{
    insert(_objectList.count(), objects);
}

243
QObjectList QmlObjectListModel::swapObjectList(const QObjectList& newlist)
244 245
{
    QObjectList oldlist(_objectList);
246 247 248
    if (!_externalBeginResetModel) {
        beginResetModel();
    }
249
    _objectList = newlist;
250 251 252 253
    if (!_externalBeginResetModel) {
        endResetModel();
        emit countChanged(count());
    }
254 255 256
    return oldlist;
}

Gus Grubba's avatar
Gus Grubba committed
257
int QmlObjectListModel::count() const
258 259 260 261
{
    return rowCount();
}

Don Gagne's avatar
Don Gagne committed
262 263
void QmlObjectListModel::setDirty(bool dirty)
{
264 265 266 267
    if (_dirty != dirty) {
        _dirty = dirty;
        if (!dirty) {
            // Need to clear dirty from all children
268
            for(QObject* object: _objectList) {
269 270 271
                if (object->property("dirty").isValid()) {
                    object->setProperty("dirty", false);
                }
Don Gagne's avatar
Don Gagne committed
272 273
            }
        }
274
        emit dirtyChanged(_dirty);
Don Gagne's avatar
Don Gagne committed
275 276 277 278 279 280
    }
}

void QmlObjectListModel::_childDirtyChanged(bool dirty)
{
    _dirty |= dirty;
281 282
    // We want to emit dirtyChanged even if the actual value of _dirty didn't change. It can be a useful
    // signal to know when a child has changed dirty state
Don Gagne's avatar
Don Gagne committed
283 284
    emit dirtyChanged(_dirty);
}
Don Gagne's avatar
Don Gagne committed
285

Gus Grubba's avatar
Gus Grubba committed
286
void QmlObjectListModel::deleteListAndContents()
Don Gagne's avatar
Don Gagne committed
287 288 289 290 291 292
{
    for (int i=0; i<_objectList.count(); i++) {
        _objectList[i]->deleteLater();
    }
    deleteLater();
}
Don Gagne's avatar
Don Gagne committed
293

Gus Grubba's avatar
Gus Grubba committed
294
void QmlObjectListModel::clearAndDeleteContents()
Don Gagne's avatar
Don Gagne committed
295
{
296
    beginResetModel();
Don Gagne's avatar
Don Gagne committed
297 298 299 300
    for (int i=0; i<_objectList.count(); i++) {
        _objectList[i]->deleteLater();
    }
    clear();
301
    endResetModel();
Don Gagne's avatar
Don Gagne committed
302
}
303

304
void QmlObjectListModel::beginReset()
305 306 307 308 309 310 311 312
{
    if (_externalBeginResetModel) {
        qWarning() << "QmlObjectListModel::beginReset already set";
    }
    _externalBeginResetModel = true;
    beginResetModel();
}

313
void QmlObjectListModel::endReset()
314 315 316 317 318 319 320
{
    if (!_externalBeginResetModel) {
        qWarning() << "QmlObjectListModel::endReset begin not set";
    }
    _externalBeginResetModel = false;
    endResetModel();
}