mirror of
https://github.com/subsurface/subsurface.git
synced 2025-02-19 22:16:15 +00:00
Add possibility to filter by more than one criteria at a time
This new version of the TagFilterSortModel actually accepts *any* new MultiFilterInterface. So, how to use it to create a new filter: Implement a class that inherits from MultiFilterInterface Implement the filterRow method TagFilterSortModel::instance->add( myClass ); and you are done. [Dirk Hohndel: removed some debug code and did whitespace cleanup] Signed-off-by: Tomaz Canabrava <tomaz.canabrava@intel.com> Signed-off-by: Dirk Hohndel <dirk@hohndel.org>
This commit is contained in:
parent
28800622f0
commit
e0b60167f2
5 changed files with 86 additions and 19 deletions
|
@ -34,7 +34,7 @@ DiveListView::DiveListView(QWidget *parent) : QTreeView(parent), mouseClickSelec
|
||||||
setItemDelegate(new DiveListDelegate(this));
|
setItemDelegate(new DiveListDelegate(this));
|
||||||
setUniformRowHeights(true);
|
setUniformRowHeights(true);
|
||||||
setItemDelegateForColumn(DiveTripModel::RATING, new StarWidgetsDelegate(this));
|
setItemDelegateForColumn(DiveTripModel::RATING, new StarWidgetsDelegate(this));
|
||||||
TagFilterSortModel *model = new TagFilterSortModel(this);
|
TagFilterSortModel *model = TagFilterSortModel::instance();
|
||||||
model->setSortRole(DiveTripModel::SORT_ROLE);
|
model->setSortRole(DiveTripModel::SORT_ROLE);
|
||||||
model->setFilterKeyColumn(-1); // filter all columns
|
model->setFilterKeyColumn(-1); // filter all columns
|
||||||
model->setFilterCaseSensitivity(Qt::CaseInsensitive);
|
model->setFilterCaseSensitivity(Qt::CaseInsensitive);
|
||||||
|
|
|
@ -2193,25 +2193,20 @@ bool TagFilterModel::setData(const QModelIndex &index, const QVariant &value, in
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
TagFilterSortModel::TagFilterSortModel(QObject *parent) : QSortFilterProxyModel(parent)
|
bool TagFilterModel::filterRow(int source_row, const QModelIndex &source_parent, QAbstractItemModel *sourceModel) const
|
||||||
{
|
|
||||||
connect(TagFilterModel::instance(), SIGNAL(dataChanged(QModelIndex, QModelIndex)), this, SLOT(invalidate()));
|
|
||||||
}
|
|
||||||
|
|
||||||
bool TagFilterSortModel::filterAcceptsRow(int source_row, const QModelIndex &source_parent) const
|
|
||||||
{
|
{
|
||||||
// If there's nothing checked, this should show everythin.
|
// If there's nothing checked, this should show everythin.
|
||||||
if (!TagFilterModel::instance()->anyChecked) {
|
if (!anyChecked) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
QModelIndex index0 = sourceModel()->index(source_row, 0, source_parent);
|
QModelIndex index0 = sourceModel->index(source_row, 0, source_parent);
|
||||||
QVariant diveVariant = sourceModel()->data(index0, DiveTripModel::DIVE_ROLE);
|
QVariant diveVariant = sourceModel->data(index0, DiveTripModel::DIVE_ROLE);
|
||||||
struct dive *d = (struct dive *)diveVariant.value<void *>();
|
struct dive *d = (struct dive *)diveVariant.value<void *>();
|
||||||
|
|
||||||
if (!d) { // It's a trip, only show the ones that have dives to be shown.
|
if (!d) { // It's a trip, only show the ones that have dives to be shown.
|
||||||
for (int i = 0; i < sourceModel()->rowCount(index0); i++) {
|
for (int i = 0; i < sourceModel->rowCount(index0); i++) {
|
||||||
if (filterAcceptsRow(i, index0))
|
if (filterRow(i, index0, sourceModel))
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
|
@ -2220,23 +2215,69 @@ bool TagFilterSortModel::filterAcceptsRow(int source_row, const QModelIndex &sou
|
||||||
struct tag_entry *head = d->tag_list;
|
struct tag_entry *head = d->tag_list;
|
||||||
|
|
||||||
if (!head) { // last tag means "Show empty tags";
|
if (!head) { // last tag means "Show empty tags";
|
||||||
if (TagFilterModel::instance()->rowCount() > 0)
|
if (rowCount() > 0)
|
||||||
return TagFilterModel::instance()->checkState[TagFilterModel::instance()->rowCount() - 1];
|
return checkState[rowCount() - 1];
|
||||||
else
|
else
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
// have at least one tag.
|
// have at least one tag.
|
||||||
QStringList tagList = TagFilterModel::instance()->stringList();
|
QStringList tagList = stringList();
|
||||||
if (!tagList.isEmpty()) {
|
if (!tagList.isEmpty()) {
|
||||||
tagList.removeLast(); // remove the "Show Empty Tags";
|
tagList.removeLast(); // remove the "Show Empty Tags";
|
||||||
while (head) {
|
while (head) {
|
||||||
QString tagName(head->tag->name);
|
QString tagName(head->tag->name);
|
||||||
int index = tagList.indexOf(tagName);
|
int index = tagList.indexOf(tagName);
|
||||||
if (TagFilterModel::instance()->checkState[index])
|
if (checkState[index])
|
||||||
return true;
|
return true;
|
||||||
head = head->next;
|
head = head->next;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
TagFilterSortModel *TagFilterSortModel::instance()
|
||||||
|
{
|
||||||
|
static TagFilterSortModel *self = new TagFilterSortModel();
|
||||||
|
return self;
|
||||||
|
}
|
||||||
|
|
||||||
|
TagFilterSortModel::TagFilterSortModel(QObject *parent) : QSortFilterProxyModel(parent)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
bool TagFilterSortModel::filterAcceptsRow(int source_row, const QModelIndex &source_parent) const
|
||||||
|
{
|
||||||
|
if (models.isEmpty()) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
Q_FOREACH (MultiFilterInterface *model, models) {
|
||||||
|
if (model->filterRow(source_row, source_parent, sourceModel())) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
void TagFilterSortModel::myInvalidate()
|
||||||
|
{
|
||||||
|
invalidate();
|
||||||
|
}
|
||||||
|
|
||||||
|
void TagFilterSortModel::addFilterModel(MultiFilterInterface *model)
|
||||||
|
{
|
||||||
|
QAbstractItemModel *itemModel = dynamic_cast<QAbstractItemModel *>(model);
|
||||||
|
Q_ASSERT(itemModel);
|
||||||
|
models.append(model);
|
||||||
|
connect(itemModel, SIGNAL(dataChanged(QModelIndex, QModelIndex)), this, SLOT(myInvalidate()));
|
||||||
|
}
|
||||||
|
|
||||||
|
void TagFilterSortModel::removeFilterModel(MultiFilterInterface *model)
|
||||||
|
{
|
||||||
|
QAbstractItemModel *itemModel = dynamic_cast<QAbstractItemModel *>(model);
|
||||||
|
Q_ASSERT(itemModel);
|
||||||
|
models.removeAll(model);
|
||||||
|
disconnect(itemModel, SIGNAL(dataChanged(QModelIndex, QModelIndex)), this, SLOT(myInvalidate()));
|
||||||
|
}
|
||||||
|
|
|
@ -417,13 +417,19 @@ private:
|
||||||
QStringList languages;
|
QStringList languages;
|
||||||
};
|
};
|
||||||
|
|
||||||
class TagFilterModel : public QStringListModel {
|
class MultiFilterInterface {
|
||||||
|
public:
|
||||||
|
virtual bool filterRow(int source_row, const QModelIndex &source_parent, QAbstractItemModel *sourceModel) const = 0;
|
||||||
|
};
|
||||||
|
|
||||||
|
class TagFilterModel : public QStringListModel, public MultiFilterInterface{
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
public:
|
public:
|
||||||
static TagFilterModel *instance();
|
static TagFilterModel *instance();
|
||||||
virtual QVariant data(const QModelIndex &index, int role = Qt::DisplayRole) const;
|
virtual QVariant data(const QModelIndex &index, int role = Qt::DisplayRole) const;
|
||||||
virtual bool setData(const QModelIndex &index, const QVariant &value, int role = Qt::EditRole);
|
virtual bool setData(const QModelIndex &index, const QVariant &value, int role = Qt::EditRole);
|
||||||
virtual Qt::ItemFlags flags(const QModelIndex &index) const;
|
virtual Qt::ItemFlags flags(const QModelIndex &index) const;
|
||||||
|
virtual bool filterRow(int source_row, const QModelIndex &source_parent, QAbstractItemModel *sourceModel) const;
|
||||||
bool *checkState;
|
bool *checkState;
|
||||||
bool anyChecked;
|
bool anyChecked;
|
||||||
public
|
public
|
||||||
|
@ -437,7 +443,14 @@ private:
|
||||||
class TagFilterSortModel : public QSortFilterProxyModel {
|
class TagFilterSortModel : public QSortFilterProxyModel {
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
public:
|
public:
|
||||||
TagFilterSortModel(QObject *parent = 0);
|
static TagFilterSortModel *instance();
|
||||||
virtual bool filterAcceptsRow(int source_row, const QModelIndex &source_parent) const;
|
virtual bool filterAcceptsRow(int source_row, const QModelIndex &source_parent) const;
|
||||||
|
void addFilterModel(MultiFilterInterface *model);
|
||||||
|
void removeFilterModel(MultiFilterInterface *model);
|
||||||
|
public slots:
|
||||||
|
void myInvalidate();
|
||||||
|
private:
|
||||||
|
TagFilterSortModel(QObject *parent = 0);
|
||||||
|
QList<MultiFilterInterface*> models;
|
||||||
};
|
};
|
||||||
#endif // MODELS_H
|
#endif // MODELS_H
|
||||||
|
|
|
@ -464,3 +464,15 @@ TagFilter::TagFilter(QWidget *parent) : QWidget(parent)
|
||||||
connect(ui.filterTag, SIGNAL(textChanged(QString)), filter, SLOT(setFilterFixedString(QString)));
|
connect(ui.filterTag, SIGNAL(textChanged(QString)), filter, SLOT(setFilterFixedString(QString)));
|
||||||
ui.tagView->setModel(filter);
|
ui.tagView->setModel(filter);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void TagFilter::showEvent(QShowEvent *event)
|
||||||
|
{
|
||||||
|
TagFilterSortModel::instance()->addFilterModel(TagFilterModel::instance());
|
||||||
|
QWidget::showEvent(event);
|
||||||
|
}
|
||||||
|
|
||||||
|
void TagFilter::hideEvent(QHideEvent *event)
|
||||||
|
{
|
||||||
|
TagFilterSortModel::instance()->removeFilterModel(TagFilterModel::instance());
|
||||||
|
QWidget::hideEvent(event);
|
||||||
|
}
|
||||||
|
|
|
@ -134,7 +134,8 @@ class TagFilter : public QWidget {
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
public:
|
public:
|
||||||
TagFilter(QWidget *parent = 0);
|
TagFilter(QWidget *parent = 0);
|
||||||
|
virtual void showEvent(QShowEvent *);
|
||||||
|
virtual void hideEvent(QHideEvent *);
|
||||||
private:
|
private:
|
||||||
Ui::TagFilter ui;
|
Ui::TagFilter ui;
|
||||||
};
|
};
|
||||||
|
|
Loading…
Add table
Reference in a new issue