/* * Cantata * * Copyright (c) 2011-2022 Craig Drummond * * ---- * * 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 "models/roles.h" #include "icons.h" #include "config.h" #include "basicitemdelegate.h" #include "support/utils.h" #include "mpd-interface/song.h" #include #include #include #include #include #include #include #include #include #include #include #define SINGLE_CLICK style()->styleHint(QStyle::SH_ItemView_ActivateItemOnSingleClick, 0, this) 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; } } 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) , eventFilter(nullptr) , forceSingleColumn(false) , alwaysAllowMenu(menuAlwaysAllowed) { setDragEnabled(true); setContextMenuPolicy(Qt::NoContextMenu); // setRootIsDecorated(false); setAllColumnsShowFocus(true); setAlternatingRowColors(false); setSelectionMode(QAbstractItemView::ExtendedSelection); setSelectionBehavior(QAbstractItemView::SelectRows); setVerticalScrollMode(QAbstractItemView::ScrollPerPixel); // 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); forceSingleColumn=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() && model() && selectionModel()->selectedIndexes().count()!=model()->rowCount(); } void TreeView::drag(Qt::DropActions supportedActions, QAbstractItemView *view, const QModelIndexList &items) { if (items.count() > 0) { QMimeData *data = view->model()->mimeData(items); if (!data) { return; } QDrag *drag = new QDrag(view); drag->setMimeData(data); int pixSize=Icon::stdSize(Utils::scaleForDpi(32)); drag->setPixmap(Icons::self()->audioListIcon.pixmap(pixSize, pixSize)); drag->exec(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; for (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; iright) { return false; } } return false; } QList 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 toSort; for (const QModelIndex &i: list) { if (0==i.column()) { toSort.append(Index(i)); } } // Call std::sort on these - this will use operator< std::sort(toSort.begin(), toSort.end()); // Now convert the QList into a QModelIndexList QModelIndexList sorted; for (const Index &i: toSort) { sorted.append(i); } return sorted; } void TreeView::expandAll(const QModelIndex &idx, bool singleLevelOnly) { quint32 count=model()->rowCount(idx); for (quint32 i=0; iindex(i, 0, idx), singleLevelOnly); } } void TreeView::collapseToLevel(int level, const QModelIndex &idx) { quint32 count=model()->rowCount(idx); if (level) { for (quint32 i=0; iindex(i, 0, idx)); } } else { for (quint32 i=0; iindex(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; iindex(i, 0, idx)); } } } } void TreeView::collapse(const QModelIndex &idx, bool singleOnly) { if (idx.isValid()) { setExpanded(idx, false); if (!singleOnly) { quint32 count=model()->rowCount(idx); for (quint32 i=0; iindex(i, 0, idx)); } } } } 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); // } #ifndef Q_OS_MAC setPalette(pal); #endif 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); } if (!info.isEmpty() && model() && 0==model()->rowCount()) { QPainter p(viewport()); QColor col(palette().text().color()); col.setAlphaF(0.5); QFont f(font()); f.setItalic(true); p.setPen(col); p.setFont(f); p.drawText(rect().adjusted(8, 8, -16, -16), Qt::AlignCenter|Qt::TextWordWrap, info); } QTreeView::paintEvent(e); } void TreeView::setModel(QAbstractItemModel *m) { QAbstractItemModel *old=model(); QTreeView::setModel(m); if (forceSingleColumn && m) { int columnCount=m->columnCount(); if (columnCount>1) { QHeaderView *hdr=header(); for (int i=1; isetSectionHidden(i, true); } } } 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)); } } #include "moc_treeview.cpp"