2015-12-10 05:06:13 +02:00
|
|
|
/****************************************************************************
|
|
|
|
**
|
|
|
|
** Copyright (C) 2015 The Qt Company Ltd.
|
2019-06-03 14:21:30 +00:00
|
|
|
** Copyright (C) 2016-2019 Ivailo Monev
|
2015-12-10 05:06:13 +02:00
|
|
|
**
|
2019-06-03 13:38:02 +00:00
|
|
|
** This file is part of the QtDeclarative module of the Katie Toolkit.
|
2015-12-10 05:06:13 +02:00
|
|
|
**
|
|
|
|
** $QT_BEGIN_LICENSE:LGPL$
|
|
|
|
** Commercial License Usage
|
|
|
|
** Licensees holding valid commercial Qt licenses may use this file in
|
|
|
|
** accordance with the commercial license agreement provided with the
|
|
|
|
** Software or, alternatively, in accordance with the terms contained in
|
|
|
|
** a written agreement between you and The Qt Company. For licensing terms
|
|
|
|
** and conditions see http://www.qt.io/terms-conditions. For further
|
|
|
|
** information use the contact form at http://www.qt.io/contact-us.
|
|
|
|
**
|
|
|
|
** GNU Lesser General Public License Usage
|
|
|
|
** Alternatively, this file may be used under the terms of the GNU Lesser
|
|
|
|
** General Public License version 2.1 or version 3 as published by the Free
|
|
|
|
** Software Foundation and appearing in the file LICENSE.LGPLv21 and
|
|
|
|
** LICENSE.LGPLv3 included in the packaging of this file. Please review the
|
|
|
|
** following information to ensure the GNU Lesser General Public License
|
|
|
|
** requirements will be met: https://www.gnu.org/licenses/lgpl.html and
|
|
|
|
** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
|
|
|
|
**
|
|
|
|
** As a special exception, The Qt Company gives you certain additional
|
|
|
|
** rights. These rights are described in The Qt Company LGPL Exception
|
|
|
|
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
|
|
|
|
**
|
|
|
|
** GNU General Public License Usage
|
|
|
|
** Alternatively, this file may be used under the terms of the GNU
|
|
|
|
** General Public License version 3.0 as published by the Free Software
|
|
|
|
** Foundation and appearing in the file LICENSE.GPL included in the
|
|
|
|
** packaging of this file. Please review the following information to
|
|
|
|
** ensure the GNU General Public License version 3.0 requirements will be
|
|
|
|
** met: http://www.gnu.org/copyleft/gpl.html.
|
|
|
|
**
|
|
|
|
** $QT_END_LICENSE$
|
|
|
|
**
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
#include "qdeclarativelistmodelworkeragent_p.h"
|
|
|
|
#include "qdeclarativelistmodel_p_p.h"
|
|
|
|
#include "qdeclarativedata_p.h"
|
|
|
|
#include "qdeclarativeengine_p.h"
|
|
|
|
#include "qdeclarativeinfo.h"
|
|
|
|
|
|
|
|
#include <QtCore/qcoreevent.h>
|
|
|
|
#include <QtCore/qcoreapplication.h>
|
|
|
|
#include <QtCore/qdebug.h>
|
|
|
|
|
|
|
|
|
|
|
|
QT_BEGIN_NAMESPACE
|
|
|
|
|
|
|
|
|
|
|
|
void QDeclarativeListModelWorkerAgent::Data::clearChange()
|
|
|
|
{
|
|
|
|
changes.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
void QDeclarativeListModelWorkerAgent::Data::insertChange(int index, int count)
|
|
|
|
{
|
|
|
|
Change c = { Change::Inserted, index, count, 0, QList<int>() };
|
|
|
|
changes << c;
|
|
|
|
}
|
|
|
|
|
|
|
|
void QDeclarativeListModelWorkerAgent::Data::removeChange(int index, int count)
|
|
|
|
{
|
|
|
|
Change c = { Change::Removed, index, count, 0, QList<int>() };
|
|
|
|
changes << c;
|
|
|
|
}
|
|
|
|
|
|
|
|
void QDeclarativeListModelWorkerAgent::Data::moveChange(int index, int count, int to)
|
|
|
|
{
|
|
|
|
Change c = { Change::Moved, index, count, to, QList<int>() };
|
|
|
|
changes << c;
|
|
|
|
}
|
|
|
|
|
|
|
|
void QDeclarativeListModelWorkerAgent::Data::changedChange(int index, int count, const QList<int> &roles)
|
|
|
|
{
|
|
|
|
Change c = { Change::Changed, index, count, 0, roles };
|
|
|
|
changes << c;
|
|
|
|
}
|
|
|
|
|
|
|
|
QDeclarativeListModelWorkerAgent::QDeclarativeListModelWorkerAgent(QDeclarativeListModel *model)
|
|
|
|
: m_engine(0),
|
|
|
|
m_ref(1),
|
|
|
|
m_orig(model),
|
|
|
|
m_copy(new QDeclarativeListModel(model, this))
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
QDeclarativeListModelWorkerAgent::~QDeclarativeListModelWorkerAgent()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void QDeclarativeListModelWorkerAgent::setScriptEngine(QScriptEngine *eng)
|
|
|
|
{
|
|
|
|
m_engine = eng;
|
|
|
|
if (m_copy->m_flat)
|
|
|
|
m_copy->m_flat->m_scriptEngine = eng;
|
|
|
|
}
|
|
|
|
|
|
|
|
QScriptEngine *QDeclarativeListModelWorkerAgent::scriptEngine() const
|
|
|
|
{
|
|
|
|
return m_engine;
|
|
|
|
}
|
|
|
|
|
|
|
|
void QDeclarativeListModelWorkerAgent::addref()
|
|
|
|
{
|
|
|
|
m_ref.ref();
|
|
|
|
}
|
|
|
|
|
|
|
|
void QDeclarativeListModelWorkerAgent::release()
|
|
|
|
{
|
|
|
|
bool del = !m_ref.deref();
|
|
|
|
|
|
|
|
if (del)
|
|
|
|
delete this;
|
|
|
|
}
|
|
|
|
|
|
|
|
int QDeclarativeListModelWorkerAgent::count() const
|
|
|
|
{
|
|
|
|
return m_copy->count();
|
|
|
|
}
|
|
|
|
|
|
|
|
void QDeclarativeListModelWorkerAgent::clear()
|
|
|
|
{
|
|
|
|
data.clearChange();
|
|
|
|
data.removeChange(0, m_copy->count());
|
|
|
|
m_copy->clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
void QDeclarativeListModelWorkerAgent::remove(int index)
|
|
|
|
{
|
|
|
|
int count = m_copy->count();
|
|
|
|
m_copy->remove(index);
|
|
|
|
|
|
|
|
if (m_copy->count() != count)
|
|
|
|
data.removeChange(index, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
void QDeclarativeListModelWorkerAgent::append(const QScriptValue &value)
|
|
|
|
{
|
|
|
|
int count = m_copy->count();
|
|
|
|
m_copy->append(value);
|
|
|
|
|
|
|
|
if (m_copy->count() != count)
|
|
|
|
data.insertChange(m_copy->count() - 1, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
void QDeclarativeListModelWorkerAgent::insert(int index, const QScriptValue &value)
|
|
|
|
{
|
|
|
|
int count = m_copy->count();
|
|
|
|
m_copy->insert(index, value);
|
|
|
|
|
|
|
|
if (m_copy->count() != count)
|
|
|
|
data.insertChange(index, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
QScriptValue QDeclarativeListModelWorkerAgent::get(int index) const
|
|
|
|
{
|
|
|
|
return m_copy->get(index);
|
|
|
|
}
|
|
|
|
|
|
|
|
void QDeclarativeListModelWorkerAgent::set(int index, const QScriptValue &value)
|
|
|
|
{
|
|
|
|
QList<int> roles;
|
|
|
|
m_copy->set(index, value, &roles);
|
|
|
|
if (!roles.isEmpty())
|
|
|
|
data.changedChange(index, 1, roles);
|
|
|
|
}
|
|
|
|
|
|
|
|
void QDeclarativeListModelWorkerAgent::setProperty(int index, const QString& property, const QVariant& value)
|
|
|
|
{
|
|
|
|
QList<int> roles;
|
|
|
|
m_copy->setProperty(index, property, value, &roles);
|
|
|
|
if (!roles.isEmpty())
|
|
|
|
data.changedChange(index, 1, roles);
|
|
|
|
}
|
|
|
|
|
|
|
|
void QDeclarativeListModelWorkerAgent::move(int from, int to, int count)
|
|
|
|
{
|
|
|
|
m_copy->move(from, to, count);
|
|
|
|
data.moveChange(from, to, count);
|
|
|
|
}
|
|
|
|
|
|
|
|
void QDeclarativeListModelWorkerAgent::sync()
|
|
|
|
{
|
|
|
|
Sync *s = new Sync;
|
|
|
|
s->data = data;
|
|
|
|
s->list = m_copy;
|
|
|
|
data.changes.clear();
|
|
|
|
|
|
|
|
mutex.lock();
|
|
|
|
QCoreApplication::postEvent(this, s);
|
|
|
|
syncDone.wait(&mutex);
|
|
|
|
mutex.unlock();
|
|
|
|
}
|
|
|
|
|
|
|
|
void QDeclarativeListModelWorkerAgent::changedData(int index, int count, const QList<int> &roles)
|
|
|
|
{
|
|
|
|
data.changedChange(index, count, roles);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool QDeclarativeListModelWorkerAgent::event(QEvent *e)
|
|
|
|
{
|
|
|
|
if (e->type() == QEvent::User) {
|
|
|
|
QMutexLocker locker(&mutex);
|
|
|
|
Sync *s = static_cast<Sync *>(e);
|
|
|
|
|
|
|
|
const QList<Change> &changes = s->data.changes;
|
|
|
|
|
|
|
|
if (m_copy) {
|
|
|
|
bool cc = m_orig->count() != s->list->count();
|
|
|
|
|
|
|
|
FlatListModel *orig = m_orig->m_flat;
|
|
|
|
FlatListModel *copy = s->list->m_flat;
|
|
|
|
if (!orig || !copy) {
|
|
|
|
syncDone.wakeAll();
|
|
|
|
return QObject::event(e);
|
|
|
|
}
|
|
|
|
|
|
|
|
orig->m_roles = copy->m_roles;
|
|
|
|
orig->m_strings = copy->m_strings;
|
|
|
|
orig->m_values = copy->m_values;
|
|
|
|
|
|
|
|
// update the orig->m_nodeData list
|
|
|
|
for (int ii = 0; ii < changes.count(); ++ii) {
|
|
|
|
const Change &change = changes.at(ii);
|
|
|
|
switch (change.type) {
|
|
|
|
case Change::Inserted:
|
|
|
|
orig->insertedNode(change.index);
|
|
|
|
break;
|
|
|
|
case Change::Removed:
|
|
|
|
orig->removedNode(change.index);
|
|
|
|
break;
|
|
|
|
case Change::Moved:
|
|
|
|
orig->moveNodes(change.index, change.to, change.count);
|
|
|
|
break;
|
|
|
|
case Change::Changed:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
syncDone.wakeAll();
|
|
|
|
locker.unlock();
|
|
|
|
|
|
|
|
for (int ii = 0; ii < changes.count(); ++ii) {
|
|
|
|
const Change &change = changes.at(ii);
|
|
|
|
switch (change.type) {
|
|
|
|
case Change::Inserted:
|
|
|
|
emit m_orig->itemsInserted(change.index, change.count);
|
|
|
|
break;
|
|
|
|
case Change::Removed:
|
|
|
|
emit m_orig->itemsRemoved(change.index, change.count);
|
|
|
|
break;
|
|
|
|
case Change::Moved:
|
|
|
|
emit m_orig->itemsMoved(change.index, change.to, change.count);
|
|
|
|
break;
|
|
|
|
case Change::Changed:
|
|
|
|
emit m_orig->itemsChanged(change.index, change.count, change.roles);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (cc)
|
|
|
|
emit m_orig->countChanged();
|
|
|
|
} else {
|
|
|
|
syncDone.wakeAll();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return QObject::event(e);
|
|
|
|
}
|
|
|
|
|
|
|
|
QT_END_NAMESPACE
|
|
|
|
|
|
|
|
#include <moc_qdeclarativelistmodelworkeragent_p.h>
|