/* * Cantata * * Copyright (c) 2011-2014 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 "playqueueview.h" #include "playqueuemodel.h" #include "covers.h" #include "currentcover.h" #include "groupedview.h" #include "treeview.h" #include "settings.h" #include "mpdstatus.h" #include "localize.h" #include "spinner.h" #include "messageoverlay.h" #include "basicitemdelegate.h" #include #include #include #include // Exported by QtGui void qt_blurImage(QPainter *p, QImage &blurImage, qreal radius, bool quality, bool alphaOnly, int transposed = 0); class PlayQueueTreeViewItemDelegate : public BasicItemDelegate { public: PlayQueueTreeViewItemDelegate(QObject *p) : BasicItemDelegate(p) { } virtual ~PlayQueueTreeViewItemDelegate() { } void paint(QPainter *painter, const QStyleOptionViewItem &option, const QModelIndex &index) const { if (!index.isValid()) { return; } QStyleOptionViewItemV4 v4((QStyleOptionViewItemV4 &)option); if (QStyleOptionViewItemV4::Beginning==v4.viewItemPosition) { v4.icon=index.data(PlayQueueView::Role_Decoration).value(); if (!v4.icon.isNull()) { v4.features |= QStyleOptionViewItemV2::HasDecoration; v4.decorationSize=v4.icon.actualSize(option.decorationSize, QIcon::Normal, QIcon::Off); } } BasicItemDelegate::paint(painter, v4, index); } }; PlayQueueTreeView::PlayQueueTreeView(PlayQueueView *parent) : TableView(QLatin1String("playQueue"), parent, true) , view(parent) { setIndentation(0); setItemsExpandable(false); setExpandsOnDoubleClick(false); setRootIsDecorated(false); setItemDelegate(new PlayQueueTreeViewItemDelegate(this)); } void PlayQueueTreeView::paintEvent(QPaintEvent *e) { view->drawBackdrop(viewport(), size()); TreeView::paintEvent(e); } PlayQueueGroupedView::PlayQueueGroupedView(PlayQueueView *parent) : GroupedView(parent, true) , view(parent) { } PlayQueueGroupedView::~PlayQueueGroupedView() { } void PlayQueueGroupedView::paintEvent(QPaintEvent *e) { view->drawBackdrop(viewport(), size()); GroupedView::paintEvent(e); } PlayQueueView::PlayQueueView(QWidget *parent) : QStackedWidget(parent) , spinner(0) , msgOverlay(0) , backgroundImageType(BI_None) , backgroundOpacity(15) , backgroundBlur(0) { groupedView=new PlayQueueGroupedView(this); groupedView->setIndentation(0); groupedView->setItemsExpandable(false); groupedView->setExpandsOnDoubleClick(false); treeView=new PlayQueueTreeView(this); addWidget(groupedView); addWidget(treeView); setCurrentWidget(treeView); connect(groupedView, SIGNAL(itemsSelected(bool)), SIGNAL(itemsSelected(bool))); connect(treeView, SIGNAL(itemsSelected(bool)), SIGNAL(itemsSelected(bool))); connect(groupedView, SIGNAL(doubleClicked(const QModelIndex &)), SIGNAL(doubleClicked(const QModelIndex &))); connect(treeView, SIGNAL(doubleClicked(const QModelIndex &)), SIGNAL(doubleClicked(const QModelIndex &))); setContextMenuPolicy(Qt::ActionsContextMenu); animator.setPropertyName("fade"); animator.setTargetObject(this); connect(CurrentCover::self(), SIGNAL(coverImage(QImage)), this, SLOT(setImage(QImage))); } PlayQueueView::~PlayQueueView() { } bool PlayQueueView::readConfig() { int origOpacity=backgroundOpacity; int origBlur=backgroundBlur; QString origCustomBackgroundFile=customBackgroundFile; BackgroundImage origType=backgroundImageType; setAutoExpand(Settings::self()->playQueueAutoExpand()); setStartClosed(Settings::self()->playQueueStartClosed()); backgroundImageType=(BackgroundImage)Settings::self()->playQueueBackground(); backgroundOpacity=Settings::self()->playQueueBackgroundOpacity(); backgroundBlur=Settings::self()->playQueueBackgroundBlur(); customBackgroundFile=Settings::self()->playQueueBackgroundFile(); setGrouped(Settings::self()->playQueueGrouped()); switch (backgroundImageType) { case BI_None: if (origType!=backgroundImageType) { updatePalette(); previousBackground=QPixmap(); curentCover=QImage(); curentBackground=QPixmap(); view()->viewport()->update(); } break; case BI_Cover: if (BI_None==origType) { updatePalette(); } return origType!=backgroundImageType || backgroundOpacity!=origOpacity || backgroundBlur!=origBlur; case BI_Custom: if (BI_None==origType) { updatePalette(); } if (origType!=backgroundImageType || backgroundOpacity!=origOpacity || backgroundBlur!=origBlur || origCustomBackgroundFile!=customBackgroundFile) { setImage(QImage(customBackgroundFile)); } break; } return false; } void PlayQueueView::saveConfig() { if (treeView==currentWidget()) { treeView->saveHeader(); } } void PlayQueueView::setGrouped(bool g) { bool grouped=groupedView==currentWidget(); if (g!=grouped) { if (grouped) { treeView->setModel(groupedView->model()); treeView->initHeader(); groupedView->setModel(0); } else { treeView->saveHeader(); groupedView->setModel(treeView->model()); treeView->setModel(0); } grouped=g; setCurrentWidget(grouped ? static_cast(groupedView) : static_cast(treeView)); if (spinner) { spinner->setWidget(view()); if (spinner->isActive()) { spinner->start(); } } if (msgOverlay) { msgOverlay->setWidget(view()); } } } void PlayQueueView::setAutoExpand(bool ae) { groupedView->setAutoExpand(ae); } bool PlayQueueView::isAutoExpand() const { return groupedView->isAutoExpand(); } void PlayQueueView::setStartClosed(bool sc) { groupedView->setStartClosed(sc); } bool PlayQueueView::isStartClosed() const { return groupedView->isStartClosed(); } void PlayQueueView::setFilterActive(bool f) { if (isGrouped()) { groupedView->setFilterActive(f); } } void PlayQueueView::updateRows(qint32 row, quint16 curAlbum, bool scroll) { if (isGrouped()) { groupedView->updateRows(row, curAlbum, scroll); } } void PlayQueueView::scrollTo(const QModelIndex &index, QAbstractItemView::ScrollHint hint) { view()->scrollTo(index, hint); } QModelIndex PlayQueueView::indexAt(const QPoint &point) { return view()->indexAt(point); } void PlayQueueView::addAction(QAction *a) { groupedView->addAction(a); treeView->addAction(a); } void PlayQueueView::setFocus() { currentWidget()->setFocus(); } bool PlayQueueView::hasFocus() { return currentWidget()->hasFocus(); } void PlayQueueView::setContextMenuPolicy(Qt::ContextMenuPolicy policy) { groupedView->setContextMenuPolicy(policy); treeView->setContextMenuPolicy(policy); } bool PlayQueueView::haveSelectedItems() { return isGrouped() ? groupedView->haveSelectedItems() : treeView->haveSelectedItems(); } bool PlayQueueView::haveUnSelectedItems() { return isGrouped() ? groupedView->haveUnSelectedItems() : treeView->haveUnSelectedItems(); } void PlayQueueView::clearSelection() { if (groupedView->selectionModel()) { groupedView->selectionModel()->clear(); } if (treeView->selectionModel()) { treeView->selectionModel()->clear(); } } QAbstractItemView * PlayQueueView::tree() const { return treeView; } QAbstractItemView * PlayQueueView::list() const { return groupedView; } QAbstractItemView * PlayQueueView::view() const { return isGrouped() ? (QAbstractItemView *)groupedView : (QAbstractItemView *)treeView; } bool PlayQueueView::hasFocus() const { return isGrouped() ? groupedView->hasFocus() : treeView->hasFocus(); } QModelIndexList PlayQueueView::selectedIndexes(bool sorted) const { return isGrouped() ? groupedView->selectedIndexes(sorted) : treeView->selectedIndexes(sorted); } QList PlayQueueView::selectedSongs() const { const QModelIndexList selected = selectedIndexes(); QList songs; foreach (const QModelIndex &idx, selected) { Song song=idx.data(GroupedView::Role_Song).value(); if (!song.file.isEmpty() && !song.file.contains(":/") && !song.file.startsWith('/')) { songs.append(song); } } return songs; } void PlayQueueView::showSpinner() { if (!spinner) { spinner=new Spinner(this); } spinner->setWidget(view()); spinner->start(); } void PlayQueueView::hideSpinner() { if (spinner) { spinner->stop(); } } void PlayQueueView::setFade(float value) { if (fadeValue!=value) { fadeValue = value; if (qFuzzyCompare(fadeValue, qreal(1.0))) { previousBackground=QPixmap(); } view()->viewport()->update(); } } void PlayQueueView::updatePalette() { QPalette pal=palette(); if (BI_None!=backgroundImageType) { pal.setColor(QPalette::Base, Qt::transparent); } groupedView->setPalette(pal); groupedView->viewport()->setPalette(pal); treeView->setPalette(pal); treeView->viewport()->setPalette(pal); } void PlayQueueView::setImage(const QImage &img) { if (BI_None==backgroundImageType || (sender() && BI_Custom==backgroundImageType)) { return; } previousBackground=curentBackground; if (img.isNull() || QImage::Format_ARGB32==img.format()) { curentCover = img; } else { curentCover = img.convertToFormat(QImage::Format_ARGB32); } if (!curentCover.isNull()) { if (backgroundOpacity<100) { curentCover=TreeView::setOpacity(curentCover, (backgroundOpacity*1.0)/100.0); } if (backgroundBlur>0) { QImage blurred(curentCover.size(), QImage::Format_ARGB32_Premultiplied); blurred.fill(Qt::transparent); QPainter painter(&blurred); qt_blurImage(&painter, curentCover, backgroundBlur, true, false); painter.end(); curentCover = blurred; } } curentBackground=QPixmap(); animator.stop(); if (isVisible()) { fadeValue=0.0; animator.setDuration(250); animator.setEndValue(1.0); animator.start(); } } void PlayQueueView::streamFetchStatus(const QString &msg) { if (!msgOverlay) { msgOverlay=new MessageOverlay(this); msgOverlay->setWidget(view()); connect(msgOverlay, SIGNAL(cancel()), SIGNAL(cancelStreamFetch())); connect(msgOverlay, SIGNAL(cancel()), SLOT(hideSpinner())); } msgOverlay->setText(msg); } void PlayQueueView::drawBackdrop(QWidget *widget, const QSize &size) { if (BI_None==backgroundImageType) { return; } QPainter p(widget); p.fillRect(0, 0, size.width(), size.height(), QApplication::palette().color(QPalette::Base)); if (!curentCover.isNull() || !previousBackground.isNull()) { if (!curentCover.isNull() && (size!=lastBgndSize || curentBackground.isNull())) { curentBackground = QPixmap::fromImage(curentCover.scaled(size, Qt::KeepAspectRatioByExpanding, Qt::SmoothTransformation)); lastBgndSize=size; } if (!previousBackground.isNull()) { if (!qFuzzyCompare(fadeValue, qreal(0.0))) { p.setOpacity(1.0-fadeValue); } p.drawPixmap((size.width()-previousBackground.width())/2, (size.height()-previousBackground.height())/2, previousBackground); } if (!curentBackground.isNull()) { p.setOpacity(fadeValue); p.drawPixmap((size.width()-curentBackground.width())/2, (size.height()-curentBackground.height())/2, curentBackground); } } }