2010-10-26 12:01:59 +00:00
|
|
|
/*
|
|
|
|
* Copyright 2010 by Marco MArtin <mart@kde.org>
|
|
|
|
|
|
|
|
* This program is free software; you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU Library General Public License as
|
|
|
|
* published by the Free Software Foundation; either version 2, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Library General Public
|
|
|
|
* License along with this program; if not, write to the
|
|
|
|
* Free Software Foundation, Inc.,
|
|
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef DATAMODEL_H
|
|
|
|
#define DATAMODEL_H
|
|
|
|
|
|
|
|
#include <QAbstractItemModel>
|
2015-02-27 18:54:07 +01:00
|
|
|
#include <QJSValue>
|
2010-10-26 21:19:04 +00:00
|
|
|
#include <QSortFilterProxyModel>
|
2010-10-26 12:01:59 +00:00
|
|
|
#include <QVector>
|
|
|
|
|
2010-10-26 13:18:19 +00:00
|
|
|
#include <Plasma/DataEngine>
|
|
|
|
|
2011-04-07 22:37:34 +02:00
|
|
|
class QTimer;
|
2010-10-28 13:06:40 +00:00
|
|
|
|
2010-10-26 12:01:59 +00:00
|
|
|
namespace Plasma
|
|
|
|
{
|
|
|
|
|
2010-10-26 13:18:19 +00:00
|
|
|
class DataSource;
|
2010-10-28 13:06:40 +00:00
|
|
|
class DataModel;
|
|
|
|
|
2014-08-12 23:12:15 +02:00
|
|
|
/**
|
|
|
|
* @class SortFilterModel
|
|
|
|
* @short Filter and sort an existing QAbstractItemModel
|
|
|
|
*/
|
2010-10-28 13:10:10 +00:00
|
|
|
class SortFilterModel : public QSortFilterProxyModel
|
2010-10-26 12:01:59 +00:00
|
|
|
{
|
|
|
|
Q_OBJECT
|
2011-12-23 12:17:48 +01:00
|
|
|
/**
|
|
|
|
* The source model of this sorting proxy model. It has to inherit QAbstractItemModel (ListModel is not supported)
|
|
|
|
*/
|
2012-08-29 17:35:56 +02:00
|
|
|
Q_PROPERTY(QAbstractItemModel *sourceModel READ sourceModel WRITE setModel NOTIFY sourceModelChanged)
|
2010-10-26 13:18:19 +00:00
|
|
|
|
2011-12-23 12:17:48 +01:00
|
|
|
/**
|
|
|
|
* The regular expression for the filter, only items with their filterRole matching filterRegExp will be displayed
|
|
|
|
*/
|
2012-08-28 11:42:56 +02:00
|
|
|
Q_PROPERTY(QString filterRegExp READ filterRegExp WRITE setFilterRegExp NOTIFY filterRegExpChanged)
|
2011-12-23 12:17:48 +01:00
|
|
|
|
2015-02-23 20:02:41 +01:00
|
|
|
/**
|
|
|
|
* The string for the filter, only items with their filterRole matching filterString will be displayed
|
|
|
|
*/
|
2015-02-24 13:06:12 +01:00
|
|
|
Q_PROPERTY(QString filterString READ filterString WRITE setFilterString NOTIFY filterStringChanged REVISION 1)
|
2015-02-23 20:02:41 +01:00
|
|
|
|
2015-02-27 18:54:07 +01:00
|
|
|
/**
|
|
|
|
* A JavaScript callable that is passed the source model row index as first argument and the value
|
|
|
|
* of filterRole as second argument. The callable's return value is evaluated as boolean to determine
|
|
|
|
* whether the row is accepted (true) or filtered out (false). It overrides the default implementation
|
|
|
|
* that uses filterRegExp or filterString; while filterCallable is set those two properties are
|
|
|
|
* ignored. Attempts to write a non-callable to this property are silently ignored, but you can set
|
|
|
|
* it to null.
|
|
|
|
*/
|
|
|
|
Q_PROPERTY(QJSValue filterCallback READ filterCallback WRITE setFilterCallback NOTIFY filterCallbackChanged REVISION 1)
|
|
|
|
|
2011-12-23 12:17:48 +01:00
|
|
|
/**
|
|
|
|
* The role of the sourceModel on which filterRegExp must be applied.
|
|
|
|
*/
|
2010-10-26 21:19:04 +00:00
|
|
|
Q_PROPERTY(QString filterRole READ filterRole WRITE setFilterRole)
|
2011-12-23 12:17:48 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* The role of the sourceModel that will be used for sorting. if empty the order will be left unaltered
|
|
|
|
*/
|
2010-10-26 21:19:04 +00:00
|
|
|
Q_PROPERTY(QString sortRole READ sortRole WRITE setSortRole)
|
2011-12-23 12:17:48 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* One of Qt.Ascending or Qt.Descending
|
|
|
|
*/
|
2010-10-27 21:19:03 +00:00
|
|
|
Q_PROPERTY(Qt::SortOrder sortOrder READ sortOrder WRITE setSortOrder)
|
2011-12-23 12:17:48 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* How many items are in this model
|
|
|
|
*/
|
2010-12-05 22:23:13 +00:00
|
|
|
Q_PROPERTY(int count READ count NOTIFY countChanged)
|
2010-10-26 21:19:04 +00:00
|
|
|
|
2010-10-28 13:06:40 +00:00
|
|
|
friend class DataModel;
|
2010-10-26 21:19:04 +00:00
|
|
|
|
2010-10-26 12:01:59 +00:00
|
|
|
public:
|
2014-04-26 01:45:47 +02:00
|
|
|
SortFilterModel(QObject *parent = 0);
|
2010-10-28 13:10:10 +00:00
|
|
|
~SortFilterModel();
|
2010-10-26 12:01:59 +00:00
|
|
|
|
2012-08-29 17:35:56 +02:00
|
|
|
void setModel(QAbstractItemModel *source);
|
2010-10-26 13:18:19 +00:00
|
|
|
|
2010-10-26 21:19:04 +00:00
|
|
|
void setFilterRegExp(const QString &exp);
|
|
|
|
QString filterRegExp() const;
|
|
|
|
|
2015-02-23 20:02:41 +01:00
|
|
|
void setFilterString(const QString &filterString);
|
|
|
|
QString filterString() const;
|
|
|
|
|
2015-02-27 18:54:07 +01:00
|
|
|
void setFilterCallback(const QJSValue &callback);
|
|
|
|
QJSValue filterCallback() const;
|
|
|
|
|
2010-10-26 21:19:04 +00:00
|
|
|
void setFilterRole(const QString &role);
|
|
|
|
QString filterRole() const;
|
|
|
|
|
|
|
|
void setSortRole(const QString &role);
|
|
|
|
QString sortRole() const;
|
|
|
|
|
2010-10-27 21:19:03 +00:00
|
|
|
void setSortOrder(const Qt::SortOrder order);
|
|
|
|
|
2014-04-26 01:45:47 +02:00
|
|
|
int count() const
|
|
|
|
{
|
|
|
|
return QSortFilterProxyModel::rowCount();
|
|
|
|
}
|
2010-12-05 22:23:13 +00:00
|
|
|
|
2011-12-23 12:17:48 +01:00
|
|
|
/**
|
|
|
|
* Returns the item at index in the list model.
|
|
|
|
* This allows the item data to be accessed (but not modified) from JavaScript.
|
|
|
|
* It returns an Object with a property for each role.
|
|
|
|
*
|
|
|
|
* @arg int i the row we want
|
|
|
|
*/
|
2013-07-03 00:39:54 +02:00
|
|
|
Q_INVOKABLE QVariantMap get(int i) const;
|
2011-11-25 18:47:34 +01:00
|
|
|
|
2012-05-25 14:33:56 +02:00
|
|
|
Q_INVOKABLE int mapRowToSource(int i) const;
|
|
|
|
|
|
|
|
Q_INVOKABLE int mapRowFromSource(int i) const;
|
|
|
|
|
2010-12-05 22:23:13 +00:00
|
|
|
Q_SIGNALS:
|
|
|
|
void countChanged();
|
2012-08-28 11:42:56 +02:00
|
|
|
void sourceModelChanged(QObject *);
|
|
|
|
void filterRegExpChanged(const QString &);
|
2015-02-24 13:06:12 +01:00
|
|
|
Q_REVISION(1) void filterStringChanged(const QString &);
|
2015-02-27 18:54:07 +01:00
|
|
|
Q_REVISION(1) void filterCallbackChanged(const QJSValue &);
|
2010-12-05 22:23:13 +00:00
|
|
|
|
2010-10-28 13:06:40 +00:00
|
|
|
protected:
|
2015-02-27 19:05:06 +01:00
|
|
|
int roleNameToId(const QString &name); //FIXME TODO KF6: This should have been const.
|
2015-02-27 18:54:07 +01:00
|
|
|
virtual bool filterAcceptsRow(int source_row, const QModelIndex &source_parent) const;
|
2010-10-28 13:06:40 +00:00
|
|
|
|
|
|
|
protected Q_SLOTS:
|
|
|
|
void syncRoleNames();
|
|
|
|
|
2010-10-26 21:19:04 +00:00
|
|
|
private:
|
|
|
|
QString m_filterRole;
|
|
|
|
QString m_sortRole;
|
2015-02-23 20:02:41 +01:00
|
|
|
QString m_filterString;
|
2015-02-27 18:54:07 +01:00
|
|
|
QJSValue m_filterCallback;
|
2010-10-28 13:06:40 +00:00
|
|
|
QHash<QString, int> m_roleIds;
|
2010-10-26 21:19:04 +00:00
|
|
|
};
|
|
|
|
|
2014-08-12 23:12:15 +02:00
|
|
|
/**
|
|
|
|
* @class DataModel
|
|
|
|
* @short DataSource data as a model
|
|
|
|
*/
|
2010-10-28 13:06:40 +00:00
|
|
|
class DataModel : public QAbstractItemModel
|
2010-10-26 21:19:04 +00:00
|
|
|
{
|
|
|
|
Q_OBJECT
|
2011-12-23 12:17:48 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* The instance of DataSource to construct this model on
|
|
|
|
*/
|
2010-10-26 21:19:04 +00:00
|
|
|
Q_PROPERTY(QObject *dataSource READ dataSource WRITE setDataSource)
|
2011-12-23 12:17:48 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* It's a regular expression. Only data with keys that match this filter expression will be inserted in the model
|
|
|
|
*/
|
2010-11-08 10:27:36 +00:00
|
|
|
Q_PROPERTY(QString keyRoleFilter READ keyRoleFilter WRITE setKeyRoleFilter)
|
2011-12-23 12:17:48 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* it's a regular expression. If the DataSource is connected to more than one source, only inserts data from sources matching this filter expression in the model.
|
|
|
|
* If we want to have a source watch all sources beginning with say "name:", the required regexp would be sourceFilter: "name:.*"
|
|
|
|
*/
|
2011-04-09 20:14:22 +02:00
|
|
|
Q_PROPERTY(QString sourceFilter READ sourceFilter WRITE setSourceFilter)
|
2011-12-23 12:17:48 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* How many items are in this model
|
|
|
|
*/
|
2010-12-05 22:23:13 +00:00
|
|
|
Q_PROPERTY(int count READ count NOTIFY countChanged)
|
2010-10-26 21:19:04 +00:00
|
|
|
|
|
|
|
public:
|
2014-04-26 01:45:47 +02:00
|
|
|
DataModel(QObject *parent = 0);
|
2010-10-28 13:06:40 +00:00
|
|
|
~DataModel();
|
2010-10-26 21:19:04 +00:00
|
|
|
|
|
|
|
void setDataSource(QObject *source);
|
|
|
|
QObject *dataSource() const;
|
|
|
|
|
2011-04-09 20:14:22 +02:00
|
|
|
/**
|
|
|
|
* Include only items with a key that matches this regexp in the model
|
|
|
|
*/
|
2014-04-26 01:45:47 +02:00
|
|
|
void setKeyRoleFilter(const QString &key);
|
2010-11-08 10:27:36 +00:00
|
|
|
QString keyRoleFilter() const;
|
2010-10-26 12:01:59 +00:00
|
|
|
|
2011-04-09 20:14:22 +02:00
|
|
|
/**
|
2011-09-22 22:43:32 +02:00
|
|
|
* Include only sources that matches this regexp in the model
|
2011-04-09 20:14:22 +02:00
|
|
|
*/
|
2014-04-26 01:45:47 +02:00
|
|
|
void setSourceFilter(const QString &key);
|
2011-04-09 20:14:22 +02:00
|
|
|
QString sourceFilter() const;
|
|
|
|
|
2015-02-27 19:05:06 +01:00
|
|
|
int roleNameToId(const QString &name); //FIXME TODO KF6: This should have been const.
|
2010-10-26 21:19:04 +00:00
|
|
|
|
2010-10-26 12:01:59 +00:00
|
|
|
//Reimplemented
|
2015-02-13 13:31:30 +01:00
|
|
|
QVariant data(const QModelIndex &index, int role) const Q_DECL_OVERRIDE;
|
2010-10-26 12:01:59 +00:00
|
|
|
QVariant headerData(int section, Qt::Orientation orientation,
|
2015-02-13 13:31:30 +01:00
|
|
|
int role = Qt::DisplayRole) const Q_DECL_OVERRIDE;
|
2010-10-26 12:01:59 +00:00
|
|
|
QModelIndex index(int row, int column,
|
2015-02-13 13:31:30 +01:00
|
|
|
const QModelIndex &parent = QModelIndex()) const Q_DECL_OVERRIDE;
|
|
|
|
QModelIndex parent(const QModelIndex &child) const Q_DECL_OVERRIDE;
|
|
|
|
int rowCount(const QModelIndex &parent = QModelIndex()) const Q_DECL_OVERRIDE;
|
|
|
|
int columnCount(const QModelIndex &parent = QModelIndex()) const Q_DECL_OVERRIDE;
|
2010-10-26 12:01:59 +00:00
|
|
|
|
2014-04-26 01:45:47 +02:00
|
|
|
int count() const
|
|
|
|
{
|
|
|
|
return countItems();
|
|
|
|
}
|
2010-12-05 22:23:13 +00:00
|
|
|
|
2011-12-23 12:17:48 +01:00
|
|
|
/**
|
|
|
|
* Returns the item at index in the list model.
|
|
|
|
* This allows the item data to be accessed (but not modified) from JavaScript.
|
|
|
|
* It returns an Object with a property for each role.
|
|
|
|
*
|
|
|
|
* @arg int i the row we want
|
|
|
|
*/
|
2013-07-03 00:39:54 +02:00
|
|
|
Q_INVOKABLE QVariantMap get(int i) const;
|
2011-11-25 18:47:34 +01:00
|
|
|
|
2010-11-08 10:27:36 +00:00
|
|
|
protected:
|
|
|
|
void setItems(const QString &sourceName, const QVariantList &list);
|
|
|
|
inline int countItems() const;
|
|
|
|
|
2010-10-26 12:01:59 +00:00
|
|
|
Q_SIGNALS:
|
2010-12-05 22:23:13 +00:00
|
|
|
void countChanged();
|
2012-08-28 11:42:56 +02:00
|
|
|
void sourceModelChanged(QObject *);
|
|
|
|
void filterRegExpChanged(const QString &);
|
2010-10-26 12:01:59 +00:00
|
|
|
|
2010-10-26 13:18:19 +00:00
|
|
|
private Q_SLOTS:
|
2013-08-28 22:11:55 +02:00
|
|
|
void dataUpdated(const QString &sourceName, const QVariantMap &data);
|
2010-11-08 23:07:25 +00:00
|
|
|
void removeSource(const QString &sourceName);
|
2010-10-26 13:18:19 +00:00
|
|
|
|
2010-10-26 12:01:59 +00:00
|
|
|
private:
|
2010-10-26 13:18:19 +00:00
|
|
|
DataSource *m_dataSource;
|
2010-11-08 10:27:36 +00:00
|
|
|
QString m_keyRoleFilter;
|
2011-09-22 22:43:32 +02:00
|
|
|
QRegExp m_keyRoleFilterRE;
|
2011-04-09 20:14:22 +02:00
|
|
|
QString m_sourceFilter;
|
2011-09-22 22:43:32 +02:00
|
|
|
QRegExp m_sourceFilterRE;
|
2010-11-08 10:27:36 +00:00
|
|
|
QMap<QString, QVector<QVariant> > m_items;
|
2010-10-26 12:01:59 +00:00
|
|
|
QHash<int, QByteArray> m_roleNames;
|
2010-10-26 21:19:04 +00:00
|
|
|
QHash<QString, int> m_roleIds;
|
2011-01-20 22:03:52 +00:00
|
|
|
int m_maxRoleId;
|
2010-10-26 12:01:59 +00:00
|
|
|
};
|
|
|
|
|
2010-12-05 22:23:13 +00:00
|
|
|
int DataModel::countItems() const
|
|
|
|
{
|
|
|
|
int count = 0;
|
|
|
|
foreach (const QVector<QVariant> &v, m_items) {
|
|
|
|
count += v.count();
|
|
|
|
}
|
|
|
|
return count;
|
|
|
|
}
|
|
|
|
|
2010-10-26 12:01:59 +00:00
|
|
|
}
|
|
|
|
#endif
|