Files
cantata/devices/synccollectionwidget.cpp
2020-02-16 09:51:37 +00:00

239 lines
7.6 KiB
C++

/*
* Cantata
*
* Copyright (c) 2011-2014 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 "synccollectionwidget.h"
#include "widgets/treeview.h"
#include "widgets/toolbutton.h"
#include "widgets/icons.h"
#include "models/musiclibraryitemartist.h"
#include "models/musiclibraryitemalbum.h"
#include "models/musiclibraryitemsong.h"
#include "support/icon.h"
#include "support/actioncollection.h"
#include <QTimer>
#include <QAction>
#include <algorithm>
SyncCollectionWidget::SyncCollectionWidget(QWidget *parent, const QString &title)
: QWidget(parent)
, performedSearch(false)
, searchTimer(nullptr)
{
setupUi(this);
titleLabel->setText(title);
cfgButton->setIcon(Icons::self()->configureIcon);
connect(cfgButton, SIGNAL(clicked(bool)), SIGNAL(configure()));
proxy.setSourceModel(&model);
tree->setModel(&proxy);
tree->setPageDefaults();
tree->setUseSimpleDelegate();
search->setText(QString());
search->setPlaceholderText(tr("Search"));
connect(&proxy, SIGNAL(dataChanged(QModelIndex,QModelIndex)), this, SLOT(dataChanged(QModelIndex,QModelIndex)));
connect(search, SIGNAL(returnPressed()), this, SLOT(delaySearchItems()));
connect(search, SIGNAL(textChanged(const QString)), this, SLOT(delaySearchItems()));
checkAction=new Action(tr("Check Items"), this);
connect(checkAction, SIGNAL(triggered()), SLOT(checkItems()));
unCheckAction=new Action(tr("Uncheck Items"), this);
connect(unCheckAction, SIGNAL(triggered()), SLOT(unCheckItems()));
tree->addAction(checkAction);
tree->addAction(unCheckAction);
tree->setContextMenuPolicy(Qt::ActionsContextMenu);
QAction *expand=ActionCollection::get()->action("expandall");
QAction *collapse=ActionCollection::get()->action("collapseall");
if (expand && collapse) {
tree->addAction(expand);
tree->addAction(collapse);
addAction(expand);
addAction(collapse);
connect(expand, SIGNAL(triggered()), this, SLOT(expandAll()));
connect(collapse, SIGNAL(triggered()), this, SLOT(collapseAll()));
}
connect(tree, SIGNAL(itemsSelected(bool)), checkAction, SLOT(setEnabled(bool)));
connect(tree, SIGNAL(itemsSelected(bool)), unCheckAction, SLOT(setEnabled(bool)));
connect(tree, SIGNAL(itemActivated(const QModelIndex &)), this, SLOT(itemActivated(const QModelIndex &)));
connect(tree, SIGNAL(clicked(const QModelIndex &)), this, SLOT(itemClicked(const QModelIndex &)));
}
SyncCollectionWidget::~SyncCollectionWidget()
{
}
void SyncCollectionWidget::update(const QSet<Song> &songs)
{
model.setSongs(songs);
}
QList<Song> SyncCollectionWidget::checkedSongs() const
{
QList<Song> songs;
for (const Song *s: checked) {
songs.append(*s);
}
std::sort(songs.begin(), songs.end());
return songs;
}
void SyncCollectionWidget::dataChanged(const QModelIndex &tl, const QModelIndex &br)
{
bool haveChecked=numCheckedSongs()>0;
QModelIndex firstIndex = proxy.mapToSource(tl);
QModelIndex lastIndex = proxy.mapToSource(br);
const MusicLibraryItem *item=static_cast<const MusicLibraryItem *>(firstIndex.internalPointer());
switch (item->itemType()) {
case MusicLibraryItem::Type_Artist:
for (int i=firstIndex.row(); i<=lastIndex.row(); ++i) {
QModelIndex index=model.index(i, 0, firstIndex.parent());
const MusicLibraryItemArtist *artist=static_cast<const MusicLibraryItemArtist *>(index.internalPointer());
for (const MusicLibraryItem *alItem: artist->childItems()) {
for (const MusicLibraryItem *sItem: static_cast<const MusicLibraryItemContainer *>(alItem)->childItems()) {
songToggled(static_cast<const MusicLibraryItemSong *>(sItem));
}
}
}
break;
case MusicLibraryItem::Type_Album:
for (int i=firstIndex.row(); i<=lastIndex.row(); ++i) {
QModelIndex index=model.index(i, 0, firstIndex.parent());
const MusicLibraryItemAlbum *album=static_cast<const MusicLibraryItemAlbum *>(index.internalPointer());
for (const MusicLibraryItem *sItem: album->childItems()) {
songToggled(static_cast<const MusicLibraryItemSong *>(sItem));
}
}
break;
case MusicLibraryItem::Type_Song:
for (int i=firstIndex.row(); i<=lastIndex.row(); ++i) {
QModelIndex index=model.index(i, 0, firstIndex.parent());
songToggled(static_cast<MusicLibraryItemSong *>(index.internalPointer()));
}
default:
break;
}
if (haveChecked!=(numCheckedSongs()>0)) {
emit selectionChanged();
}
}
void SyncCollectionWidget::songToggled(const MusicLibraryItemSong *song)
{
const Song &s=song->song();
if (Qt::Checked==song->checkState()) {
checked.insert(&s);
spaceRequired+=s.size;
} else {
checked.remove(&s);
spaceRequired-=s.size;
}
}
void SyncCollectionWidget::checkItems()
{
checkItems(true);
}
void SyncCollectionWidget::unCheckItems()
{
checkItems(false);
}
void SyncCollectionWidget::checkItems(bool c)
{
const QModelIndexList selected = tree->selectedIndexes();
if (0==selected.size()) {
return;
}
for (const QModelIndex &idx: selected) {
model.setData(proxy.mapToSource(idx), c, Qt::CheckStateRole);
}
}
void SyncCollectionWidget::delaySearchItems()
{
if (search->text().trimmed().isEmpty()) {
if (searchTimer) {
searchTimer->stop();
}
if (performedSearch) {
tree->collapseToLevel(0);
}
searchItems();
performedSearch=false;
} else {
if (!searchTimer) {
searchTimer=new QTimer(this);
searchTimer->setSingleShot(true);
connect(searchTimer, SIGNAL(timeout()), SLOT(searchItems()));
}
searchTimer->start(500);
}
}
void SyncCollectionWidget::searchItems()
{
QString text=search->text().trimmed();
proxy.update(text);
if (proxy.enabled() && !text.isEmpty()) {
tree->expandAll();
}
performedSearch=true;
}
void SyncCollectionWidget::expandAll()
{
QWidget *f=QApplication::focusWidget();
if (f && qobject_cast<QTreeView *>(f)) {
static_cast<QTreeView *>(f)->expandAll();
}
}
void SyncCollectionWidget::collapseAll()
{
QWidget *f=QApplication::focusWidget();
if (f && qobject_cast<QTreeView *>(f)) {
static_cast<QTreeView *>(f)->collapseAll();
}
}
void SyncCollectionWidget::itemClicked(const QModelIndex &index)
{
if (TreeView::getForceSingleClick() && !tree->checkBoxClicked(index)) {
tree->setExpanded(index, !tree->isExpanded(index));
}
}
void SyncCollectionWidget::itemActivated(const QModelIndex &index)
{
if (!TreeView::getForceSingleClick()) {
tree->setExpanded(index, !tree->isExpanded(index));
}
}
#include "moc_synccollectionwidget.cpp"