367 lines
10 KiB
C++
367 lines
10 KiB
C++
/*
|
|
* Cantata
|
|
*
|
|
* Copyright (c) 2011-2013 Craig Drummond <craig.p.drummond@gmail.com>
|
|
*
|
|
* ----
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, 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 General Public License
|
|
* along with this program; see the file COPYING. If not, write to
|
|
* the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
|
|
* Boston, MA 02110-1301, USA.
|
|
*/
|
|
|
|
#include "treeview.h"
|
|
#include "itemview.h"
|
|
#include "icons.h"
|
|
#include "config.h"
|
|
#include "basicitemdelegate.h"
|
|
#include "utils.h"
|
|
#include <QMouseEvent>
|
|
#include <QPaintEvent>
|
|
#include <QPainter>
|
|
#include <QDrag>
|
|
#include <QMimeData>
|
|
#include <QMap>
|
|
#include <QStyle>
|
|
#include <QList>
|
|
|
|
#ifdef ENABLE_KDE_SUPPORT
|
|
#include <KDE/KGlobalSettings>
|
|
#endif
|
|
|
|
#ifdef ENABLE_KDE_SUPPORT
|
|
#define SINGLE_CLICK KGlobalSettings::singleClick()
|
|
#else
|
|
#define SINGLE_CLICK style()->styleHint(QStyle::SH_ItemView_ActivateItemOnSingleClick, 0, this)
|
|
#endif
|
|
|
|
|
|
QImage TreeView::setOpacity(const QImage &orig, double opacity)
|
|
{
|
|
QImage img=QImage::Format_ARGB32==orig.format() ? orig : orig.convertToFormat(QImage::Format_ARGB32);
|
|
uchar *bits = img.bits();
|
|
for (int i = 0; i < img.height()*img.bytesPerLine(); i+=4) {
|
|
if (0!=bits[i+3]) {
|
|
bits[i+3] = opacity*255;
|
|
}
|
|
}
|
|
return img;
|
|
}
|
|
|
|
QPixmap TreeView::createBgndPixmap(const QIcon &icon)
|
|
{
|
|
if (icon.isNull()) {
|
|
return QPixmap();
|
|
}
|
|
static int bgndSize=0;
|
|
if (0==bgndSize) {
|
|
bgndSize=QApplication::fontMetrics().height()*16;
|
|
}
|
|
|
|
QImage img=icon.pixmap(bgndSize, bgndSize).toImage();
|
|
if (img.width()!=bgndSize && img.height()!=bgndSize) {
|
|
img=img.scaled(bgndSize, bgndSize, Qt::KeepAspectRatio, Qt::SmoothTransformation);
|
|
}
|
|
return QPixmap::fromImage(setOpacity(img, 0.075));
|
|
}
|
|
|
|
static bool forceSingleClick=true;
|
|
|
|
void TreeView::setForceSingleClick(bool v)
|
|
{
|
|
forceSingleClick=v;
|
|
}
|
|
|
|
bool TreeView::getForceSingleClick()
|
|
{
|
|
return forceSingleClick;
|
|
}
|
|
|
|
TreeView::TreeView(QWidget *parent, bool menuAlwaysAllowed)
|
|
: QTreeView(parent)
|
|
, alwaysAllowMenu(menuAlwaysAllowed)
|
|
{
|
|
setDragEnabled(true);
|
|
setContextMenuPolicy(Qt::NoContextMenu);
|
|
// setRootIsDecorated(false);
|
|
setAllColumnsShowFocus(true);
|
|
setAlternatingRowColors(false);
|
|
setSelectionMode(QAbstractItemView::ExtendedSelection);
|
|
setSelectionBehavior(QAbstractItemView::SelectRows);
|
|
// Treeview does not seem to need WA_MouseTracking set, even with QGtkStyle items still
|
|
// highlight under mouse. And enabling WA_MouseTracking here seems to cause drag-n-drop
|
|
// errors if an item is dragged onto playqueue whilst playqueue has a selected item!
|
|
// BUG:145
|
|
//setAttribute(Qt::WA_MouseTracking);
|
|
|
|
if (SINGLE_CLICK) {
|
|
connect(this, SIGNAL(activated(const QModelIndex &)), this, SIGNAL(itemActivated(const QModelIndex &)));
|
|
}
|
|
}
|
|
|
|
TreeView::~TreeView()
|
|
{
|
|
}
|
|
|
|
void TreeView::setPageDefaults()
|
|
{
|
|
sortByColumn(0, Qt::AscendingOrder);
|
|
setHeaderHidden(true);
|
|
setDragDropMode(QAbstractItemView::DragOnly);
|
|
setSortingEnabled(true);
|
|
setAnimated(true);
|
|
}
|
|
|
|
void TreeView::setExpandOnClick()
|
|
{
|
|
connect(this, SIGNAL(clicked(const QModelIndex &)), this, SLOT(itemWasClicked(const QModelIndex &)));
|
|
}
|
|
|
|
void TreeView::selectionChanged(const QItemSelection &selected, const QItemSelection &deselected)
|
|
{
|
|
QTreeView::selectionChanged(selected, deselected);
|
|
bool haveSelection=haveSelectedItems();
|
|
|
|
if (!alwaysAllowMenu) {
|
|
setContextMenuPolicy(haveSelection ? Qt::ActionsContextMenu : Qt::NoContextMenu);
|
|
}
|
|
emit itemsSelected(haveSelection);
|
|
}
|
|
|
|
bool TreeView::haveSelectedItems() const
|
|
{
|
|
// Dont need the sorted type of 'selectedIndexes' here...
|
|
return selectionModel() && selectionModel()->selectedIndexes().count()>0;
|
|
}
|
|
|
|
bool TreeView::haveUnSelectedItems() const
|
|
{
|
|
// Dont need the sorted type of 'selectedIndexes' here...
|
|
return selectionModel() && selectionModel()->selectedIndexes().count()!=model()->rowCount();
|
|
}
|
|
|
|
void TreeView::startDrag(Qt::DropActions supportedActions)
|
|
{
|
|
QModelIndexList indexes = selectedIndexes();
|
|
if (indexes.count() > 0) {
|
|
QMimeData *data = model()->mimeData(indexes);
|
|
if (!data) {
|
|
return;
|
|
}
|
|
QDrag *drag = new QDrag(this);
|
|
drag->setMimeData(data);
|
|
QPixmap pix;
|
|
|
|
if (1==indexes.count()) {
|
|
QVariant var=model()->data(indexes.first(), ItemView::Role_Image);
|
|
QImage img=var.value<QImage>();
|
|
if (img.isNull()) {
|
|
pix=var.value<QPixmap>();
|
|
} else {
|
|
pix=QPixmap::fromImage(img);
|
|
}
|
|
}
|
|
int pixSize=Utils::isHighDpi() ? 64 : 32;
|
|
if (pix.isNull()) {
|
|
drag->setPixmap(Icons::self()->audioFileIcon.pixmap(pixSize, pixSize));
|
|
} else {
|
|
drag->setPixmap(pix.width()<pixSize ? pix : pix.scaled(QSize(pixSize, pixSize), Qt::KeepAspectRatio, Qt::SmoothTransformation));
|
|
}
|
|
drag->start(supportedActions);
|
|
}
|
|
}
|
|
|
|
void TreeView::mouseReleaseEvent(QMouseEvent *event)
|
|
{
|
|
if (Qt::NoModifier==event->modifiers() && Qt::LeftButton==event->button()) {
|
|
QTreeView::mouseReleaseEvent(event);
|
|
}
|
|
}
|
|
|
|
QModelIndexList TreeView::selectedIndexes(bool sorted) const
|
|
{
|
|
if (!selectionModel()) {
|
|
return QModelIndexList();
|
|
}
|
|
|
|
if (sorted) {
|
|
return sortIndexes(selectionModel()->selectedIndexes());
|
|
} else if (model() && model()->columnCount()>1) {
|
|
QModelIndexList list=selectionModel()->selectedIndexes();
|
|
QModelIndexList sel;
|
|
foreach (const QModelIndex &idx, list) {
|
|
if (0==idx.column()) {
|
|
sel.append(idx);
|
|
}
|
|
}
|
|
return sel;
|
|
}
|
|
return selectionModel()->selectedIndexes();
|
|
}
|
|
|
|
struct Index : public QModelIndex
|
|
{
|
|
Index(const QModelIndex &i)
|
|
: QModelIndex(i)
|
|
{
|
|
QModelIndex idx=i;
|
|
while (idx.isValid()) {
|
|
rows.prepend(idx.row());
|
|
idx=idx.parent();
|
|
}
|
|
count=rows.count();
|
|
}
|
|
|
|
bool operator<(const Index &rhs) const
|
|
{
|
|
int toCompare=qMax(count, rhs.count);
|
|
for (int i=0; i<toCompare; ++i) {
|
|
qint32 left=i<count ? rows.at(i) : -1;
|
|
qint32 right=i<rhs.count ? rhs.rows.at(i) : -1;
|
|
if (left<right) {
|
|
return true;
|
|
} else if (left>right) {
|
|
return false;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
QList<qint32> rows;
|
|
int count;
|
|
};
|
|
|
|
QModelIndexList TreeView::sortIndexes(const QModelIndexList &list)
|
|
{
|
|
if (list.isEmpty()) {
|
|
return list;
|
|
}
|
|
|
|
// QModelIndex::operator< sorts on row first - but this messes things up if rows
|
|
// have different parents. Therefore, we use the sort above - so that the hierarchy is preserved.
|
|
// First, create the list of 'Index' items to be sorted...
|
|
QList<Index> toSort;
|
|
foreach (const QModelIndex &i, list) {
|
|
if (0==i.column()) {
|
|
toSort.append(Index(i));
|
|
}
|
|
}
|
|
// Call qSort on these - this will use operator<
|
|
qSort(toSort);
|
|
|
|
// Now convert the QList<Index> into a QModelIndexList
|
|
QModelIndexList sorted;
|
|
foreach (const Index &i, toSort) {
|
|
sorted.append(i);
|
|
}
|
|
return sorted;
|
|
}
|
|
|
|
void TreeView::expandAll(const QModelIndex &idx)
|
|
{
|
|
quint32 count=model()->rowCount(idx);
|
|
for (quint32 i=0; i<count; ++i) {
|
|
expand(model()->index(i, 0, idx));
|
|
}
|
|
}
|
|
|
|
void TreeView::expand(const QModelIndex &idx, bool singleOnly)
|
|
{
|
|
if (idx.isValid()) {
|
|
setExpanded(idx, true);
|
|
if (!singleOnly) {
|
|
quint32 count=model()->rowCount(idx);
|
|
for (quint32 i=0; i<count; ++i) {
|
|
expand(idx.child(i, 0));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
bool TreeView::checkBoxClicked(const QModelIndex &idx) const
|
|
{
|
|
QRect rect = visualRect(idx);
|
|
rect.moveTo(viewport()->mapToGlobal(QPoint(rect.x(), rect.y())));
|
|
int itemIndentation = rect.x() - visualRect(rootIndex()).x();
|
|
rect = QRect(header()->sectionViewportPosition(0) + itemIndentation, rect.y(), style()->pixelMetric(QStyle::PM_IndicatorWidth), rect.height());
|
|
return rect.contains(QCursor::pos());
|
|
}
|
|
|
|
void TreeView::setUseSimpleDelegate()
|
|
{
|
|
setItemDelegate(new BasicItemDelegate(this));
|
|
}
|
|
|
|
void TreeView::setBackgroundImage(const QIcon &icon)
|
|
{
|
|
QPalette pal=parentWidget()->palette();
|
|
if (!icon.isNull()) {
|
|
pal.setColor(QPalette::Base, Qt::transparent);
|
|
}
|
|
setPalette(pal);
|
|
viewport()->setPalette(pal);
|
|
bgnd=createBgndPixmap(icon);
|
|
}
|
|
|
|
void TreeView::paintEvent(QPaintEvent *e)
|
|
{
|
|
if (!bgnd.isNull()) {
|
|
QPainter p(viewport());
|
|
QSize sz=size();
|
|
p.fillRect(0, 0, sz.width(), sz.height(), QApplication::palette().color(QPalette::Base));
|
|
p.drawPixmap((sz.width()-bgnd.width())/2, (sz.height()-bgnd.height())/2, bgnd);
|
|
}
|
|
QTreeView::paintEvent(e);
|
|
}
|
|
|
|
void TreeView::setModel(QAbstractItemModel *m)
|
|
{
|
|
QAbstractItemModel *old=model();
|
|
QTreeView::setModel(m);
|
|
|
|
if (old) {
|
|
disconnect(old, SIGNAL(layoutChanged()), this, SLOT(correctSelection()));
|
|
}
|
|
|
|
if (m && old!=m) {
|
|
connect(m, SIGNAL(layoutChanged()), this, SLOT(correctSelection()));
|
|
}
|
|
}
|
|
|
|
// Workaround for https://bugreports.qt-project.org/browse/QTBUG-18009
|
|
void TreeView::correctSelection()
|
|
{
|
|
if (!selectionModel()) {
|
|
return;
|
|
}
|
|
|
|
QItemSelection s = selectionModel()->selection();
|
|
setCurrentIndex(currentIndex());
|
|
selectionModel()->select(s, QItemSelectionModel::SelectCurrent);
|
|
}
|
|
|
|
void TreeView::itemWasActivated(const QModelIndex &index)
|
|
{
|
|
if (!forceSingleClick) {
|
|
setExpanded(index, !isExpanded(index));
|
|
}
|
|
}
|
|
|
|
void TreeView::itemWasClicked(const QModelIndex &index)
|
|
{
|
|
if (forceSingleClick) {
|
|
setExpanded(index, !isExpanded(index));
|
|
}
|
|
}
|