You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
289 lines
10 KiB
289 lines
10 KiB
/* |
|
Copyright (C) 2010 Klarälvdalens Datakonsult AB, |
|
a KDAB Group company, info@kdab.net, |
|
author Stephen Kelly <stephen@kdab.com> |
|
|
|
This library 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 of the License, or (at your |
|
option) any later version. |
|
|
|
This library 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 Library General Public |
|
License for more details. |
|
|
|
You should have received a copy of the GNU Library General Public License |
|
along with this library; see the file COPYING.LIB. If not, write to the |
|
Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA |
|
02110-1301, USA. |
|
*/ |
|
|
|
#include "breadcrumbnavigationwidget.h" |
|
|
|
#include "dynamictreemodel.h" |
|
#include "dynamictreewidget.h" |
|
|
|
#include <QSplitter> |
|
#include <QListView> |
|
#include <QTreeView> |
|
#include <QHBoxLayout> |
|
|
|
#include "kselectionproxymodel.h" |
|
#include "kbreadcrumbselectionmodel.h" |
|
|
|
#define SON(object) object->setObjectName(QStringLiteral(#object)) |
|
|
|
CurrentItemLabel::CurrentItemLabel(QAbstractItemModel *model, QWidget *parent, Qt::WindowFlags f) |
|
: QLabel(parent, f), m_model(model) |
|
{ |
|
connect(model, SIGNAL(dataChanged(QModelIndex,QModelIndex)), SLOT(dataChanged(QModelIndex,QModelIndex))); |
|
connect(model, SIGNAL(rowsInserted(QModelIndex,int,int)), SLOT(rowsInserted(QModelIndex,int,int))); |
|
connect(model, SIGNAL(rowsRemoved(QModelIndex,int,int)), SLOT(rowsRemoved(QModelIndex,int,int))); |
|
connect(model, SIGNAL(modelReset()), SLOT(modelReset())); |
|
|
|
if (!m_model->hasChildren()) { |
|
setText(QStringLiteral("No selection")); |
|
} |
|
} |
|
|
|
void CurrentItemLabel::dataChanged(const QModelIndex &, const QModelIndex &) |
|
{ |
|
setText(m_model->index(0, 0).data().toString()); |
|
} |
|
|
|
void CurrentItemLabel::rowsInserted(const QModelIndex &, int, int) |
|
{ |
|
setText(m_model->index(0, 0).data().toString()); |
|
} |
|
|
|
void CurrentItemLabel::rowsRemoved(const QModelIndex &, int, int) |
|
{ |
|
if (!m_model->hasChildren()) { |
|
setText(QStringLiteral("No selection")); |
|
return; |
|
} |
|
setText(m_model->index(0, 0).data().toString()); |
|
} |
|
|
|
void CurrentItemLabel::modelReset() |
|
{ |
|
if (!m_model->hasChildren()) { |
|
setText(QStringLiteral("No selection")); |
|
} |
|
setText(m_model->index(0, 0).data().toString()); |
|
} |
|
|
|
KBreadcrumbNavigationProxyModel::KBreadcrumbNavigationProxyModel(QItemSelectionModel *selectionModel, QObject *parent) |
|
: KSelectionProxyModel(selectionModel, parent) |
|
{ |
|
|
|
} |
|
|
|
QVariant KBreadcrumbNavigationProxyModel::data(const QModelIndex &index, int role) const |
|
{ |
|
if (rowCount() > 2 && index.row() == 0 && role == Qt::DisplayRole) { |
|
QModelIndex sourceIndex = mapToSource(index); |
|
QStringList dataList; |
|
while (sourceIndex.isValid()) { |
|
dataList.prepend(sourceIndex.data().toString()); |
|
sourceIndex = sourceIndex.parent(); |
|
} |
|
return dataList.join(QStringLiteral(" > ")); |
|
} |
|
return KSelectionProxyModel::data(index, role); |
|
} |
|
|
|
void KBreadcrumbNavigationProxyModel::setShowHiddenAscendantData(bool showHiddenAscendantData) |
|
{ |
|
m_showHiddenAscendantData = showHiddenAscendantData; |
|
} |
|
|
|
bool KBreadcrumbNavigationProxyModel::showHiddenAscendantData() const |
|
{ |
|
return m_showHiddenAscendantData; |
|
} |
|
|
|
KNavigatingProxyModel::KNavigatingProxyModel(QItemSelectionModel *selectionModel, QObject *parent) |
|
: KSelectionProxyModel(selectionModel, parent), m_selectionModel(selectionModel) |
|
{ |
|
|
|
} |
|
|
|
void KNavigatingProxyModel::setSourceModel(QAbstractItemModel *sourceModel) |
|
{ |
|
connect(m_selectionModel, SIGNAL(selectionChanged(QItemSelection,QItemSelection)), |
|
SLOT(navigationSelectionChanged(QItemSelection,QItemSelection))); |
|
|
|
KSelectionProxyModel::setSourceModel(sourceModel); |
|
updateNavigation(); |
|
} |
|
|
|
void KNavigatingProxyModel::navigationSelectionChanged(const QItemSelection &, const QItemSelection &) |
|
{ |
|
updateNavigation(); |
|
} |
|
|
|
void KNavigatingProxyModel::updateNavigation() |
|
{ |
|
if (!sourceModel()) { |
|
return; |
|
} |
|
|
|
if (m_selectionModel->selection().isEmpty()) { |
|
setFilterBehavior(KSelectionProxyModel::ExactSelection); |
|
QModelIndex top = sourceModel()->index(0, 0); |
|
QModelIndex bottom = sourceModel()->index(sourceModel()->rowCount() - 1, 0); |
|
|
|
disconnect(m_selectionModel, SIGNAL(selectionChanged(QItemSelection,QItemSelection)), |
|
this, SLOT(navigationSelectionChanged(QItemSelection,QItemSelection))); |
|
m_selectionModel->select(QItemSelection(top, bottom), QItemSelectionModel::Select); |
|
connect(m_selectionModel, SIGNAL(selectionChanged(QItemSelection,QItemSelection)), |
|
SLOT(navigationSelectionChanged(QItemSelection,QItemSelection))); |
|
} else if (filterBehavior() != KSelectionProxyModel::ChildrenOfExactSelection) { |
|
setFilterBehavior(KSelectionProxyModel::ChildrenOfExactSelection); |
|
} |
|
} |
|
|
|
void KNavigatingProxyModel::modelReset() |
|
{ |
|
updateNavigation(); |
|
} |
|
|
|
QVariant KNavigatingProxyModel::data(const QModelIndex &index, int role) const |
|
{ |
|
if (role == Qt::DisplayRole && sourceModel()->hasChildren(mapToSource(index))) { |
|
return QString("+ " + KSelectionProxyModel::data(index, role).toString()); |
|
} |
|
return KSelectionProxyModel::data(index, role); |
|
} |
|
|
|
KForwardingItemSelectionModel::KForwardingItemSelectionModel(QAbstractItemModel *model, QItemSelectionModel *selectionModel, QObject *parent) |
|
: QItemSelectionModel(model, parent), m_selectionModel(selectionModel), m_direction(Forward) |
|
{ |
|
Q_ASSERT(model == selectionModel->model()); |
|
connect(selectionModel, SIGNAL(selectionChanged(QItemSelection,QItemSelection)), |
|
SLOT(navigationSelectionChanged(QItemSelection,QItemSelection))); |
|
} |
|
|
|
KForwardingItemSelectionModel::KForwardingItemSelectionModel(QAbstractItemModel *model, QItemSelectionModel *selectionModel, Direction direction, QObject *parent) |
|
: QItemSelectionModel(model, parent), m_selectionModel(selectionModel), m_direction(direction) |
|
{ |
|
Q_ASSERT(model == selectionModel->model()); |
|
if (m_direction == Forward) |
|
connect(selectionModel, SIGNAL(selectionChanged(QItemSelection,QItemSelection)), |
|
SLOT(navigationSelectionChanged(QItemSelection,QItemSelection))); |
|
} |
|
|
|
void KForwardingItemSelectionModel::select(const QModelIndex &index, QItemSelectionModel::SelectionFlags command) |
|
{ |
|
if (m_direction == Reverse) { |
|
m_selectionModel->select(index, command); |
|
} else { |
|
QItemSelectionModel::select(index, command); |
|
} |
|
} |
|
|
|
void KForwardingItemSelectionModel::select(const QItemSelection &selection, QItemSelectionModel::SelectionFlags command) |
|
{ |
|
if (m_direction == Reverse) { |
|
m_selectionModel->select(selection, command); |
|
} else { |
|
QItemSelectionModel::select(selection, command); |
|
} |
|
} |
|
|
|
void KForwardingItemSelectionModel::navigationSelectionChanged(const QItemSelection &selected, const QItemSelection &) |
|
{ |
|
select(selected, ClearAndSelect); |
|
} |
|
|
|
BreadcrumbNavigationWidget::BreadcrumbNavigationWidget(QWidget *parent, Qt::WindowFlags f) |
|
: QWidget(parent, f) |
|
{ |
|
DynamicTreeModel *rootModel = new DynamicTreeModel(this); |
|
QSplitter *splitter = new QSplitter(this); |
|
QHBoxLayout *layout = new QHBoxLayout(this); |
|
layout->addWidget(splitter); |
|
|
|
DynamicTreeWidget *dynamicTree = new DynamicTreeWidget(rootModel, splitter); |
|
dynamicTree->treeView()->setSelectionMode(QAbstractItemView::SingleSelection); |
|
dynamicTree->setInitialTree( |
|
QLatin1String("- 1" |
|
"- - 2" |
|
"- - 2" |
|
"- - - 3" |
|
"- - - - 4" |
|
"- - - - - 5" |
|
"- - 2" |
|
"- 6" |
|
"- 6" |
|
"- 6" |
|
"- - 7" |
|
"- - - 8" |
|
"- - - 8" |
|
"- - - - 9" |
|
"- - - - - 10" |
|
"- - - 8" |
|
"- - - 8" |
|
"- - 8" |
|
"- 16" |
|
"- - 17" |
|
"- - - 18" |
|
"- - - - 19" |
|
"- - - - - 20")); |
|
|
|
QList<QItemSelectionModel *> selectionModelList; |
|
|
|
QSplitter *vSplitter = new QSplitter(Qt::Vertical, splitter); |
|
|
|
QItemSelectionModel *rootSelectionModel = new QItemSelectionModel(rootModel, this); |
|
SON(rootSelectionModel); |
|
|
|
dynamicTree->treeView()->setSelectionModel(rootSelectionModel); |
|
|
|
KBreadcrumbSelectionModel *breadcrumbOnlyProxySelector2 = new KBreadcrumbSelectionModel(rootSelectionModel, KBreadcrumbSelectionModel::MakeBreadcrumbSelectionInOther, this); |
|
SON(breadcrumbOnlyProxySelector2); |
|
breadcrumbOnlyProxySelector2->setActualSelectionIncluded(false); |
|
|
|
KBreadcrumbNavigationProxyModel *breadcrumbNavigationModel = new KBreadcrumbNavigationProxyModel(breadcrumbOnlyProxySelector2, this); |
|
SON(breadcrumbNavigationModel); |
|
breadcrumbNavigationModel->setSourceModel(rootModel); |
|
breadcrumbNavigationModel->setFilterBehavior(KSelectionProxyModel::ExactSelection); |
|
|
|
QListView *breadcrumbView = new QListView(vSplitter); |
|
// SON(breadcrumbNavigationModel); |
|
breadcrumbView->setModel(breadcrumbNavigationModel); |
|
|
|
// This shouldn't operate on rootSelectionModel. It should operate on oneway instead? |
|
KLinkItemSelectionModel *breadcrumbViewSelectionModel = new KLinkItemSelectionModel(breadcrumbNavigationModel, rootSelectionModel, this); |
|
SON(breadcrumbViewSelectionModel); |
|
|
|
KForwardingItemSelectionModel *oneway2 = new KForwardingItemSelectionModel(breadcrumbNavigationModel, breadcrumbViewSelectionModel, KForwardingItemSelectionModel::Reverse); |
|
SON(oneway2); |
|
|
|
breadcrumbView->setSelectionModel(oneway2); |
|
|
|
KSelectionProxyModel *currentItemSelectionModel = new KSelectionProxyModel(rootSelectionModel, this); |
|
currentItemSelectionModel->setFilterBehavior(KSelectionProxyModel::ExactSelection); |
|
currentItemSelectionModel->setSourceModel(rootModel); |
|
SON(currentItemSelectionModel); |
|
|
|
new CurrentItemLabel(currentItemSelectionModel, vSplitter); |
|
|
|
QListView *selectionView = new QListView(vSplitter); |
|
|
|
// Need a one-way connection from rootSelectionModel to rootSelectionModel2 |
|
|
|
KForwardingItemSelectionModel *oneway = new KForwardingItemSelectionModel(rootModel, rootSelectionModel); |
|
|
|
KNavigatingProxyModel *navigatingProxyModel = new KNavigatingProxyModel(oneway, this); |
|
SON(navigatingProxyModel); |
|
navigatingProxyModel->setSourceModel(rootModel); |
|
selectionView->setModel(navigatingProxyModel); |
|
|
|
KLinkItemSelectionModel *selectedChildrenSelectionModel = new KLinkItemSelectionModel(navigatingProxyModel, rootSelectionModel, this); |
|
|
|
selectionView->setSelectionModel(selectedChildrenSelectionModel); |
|
} |
|
|
|
|