Clementine-audio-player-Mac.../src/devices/deviceview.cpp

431 lines
14 KiB
C++
Raw Normal View History

/* This file is part of Clementine.
Copyright 2010, David Sansome <me@davidsansome.com>
Clementine 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 3 of the License, or
(at your option) any later version.
Clementine 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 Clementine. If not, see <http://www.gnu.org/licenses/>.
*/
2010-07-04 01:00:07 +02:00
#include "connecteddevice.h"
#include "devicemanager.h"
2010-07-04 17:01:24 +02:00
#include "deviceproperties.h"
#include "deviceview.h"
#include "core/deletefiles.h"
2010-07-04 01:00:07 +02:00
#include "core/mergedproxymodel.h"
#include "core/mimedata.h"
#include "library/librarydirectorymodel.h"
2010-07-04 01:00:07 +02:00
#include "library/librarymodel.h"
#include "library/libraryview.h"
2010-07-04 01:00:07 +02:00
#include "ui/iconloader.h"
#include "ui/organisedialog.h"
#include "ui/organiseerrordialog.h"
#include <QApplication>
2010-07-04 01:00:07 +02:00
#include <QContextMenuEvent>
#include <QMenu>
#include <QMessageBox>
2010-07-04 13:34:25 +02:00
#include <QPainter>
#include <QPushButton>
2010-07-04 01:00:07 +02:00
#include <QSortFilterProxyModel>
#include <boost/shared_ptr.hpp>
2010-07-04 13:34:25 +02:00
const int DeviceItemDelegate::kIconPadding = 6;
DeviceItemDelegate::DeviceItemDelegate(QObject *parent)
: LibraryItemDelegate(parent)
{
}
void DeviceItemDelegate::paint(QPainter* p, const QStyleOptionViewItem& opt, const QModelIndex& index) const {
// Is it a device or a library item?
if (index.data(DeviceManager::Role_State).isNull()) {
LibraryItemDelegate::paint(p, opt, index);
return;
}
// Draw the background
const QStyleOptionViewItemV3* vopt = qstyleoption_cast<const QStyleOptionViewItemV3*>(&opt);
const QWidget* widget = vopt->widget;
QStyle* style = widget->style() ? widget->style() : QApplication::style();
style->drawPrimitive(QStyle::PE_PanelItemViewItem, &opt, p, widget);
2010-07-04 13:34:25 +02:00
p->save();
// Font for the status line
QFont status_font(opt.font);
#ifdef Q_OS_WIN32
2010-08-28 20:43:01 +02:00
status_font.setPointSize(status_font.pointSize() - 1);
#else
status_font.setPointSize(status_font.pointSize() - 2);
#endif
2010-07-04 13:34:25 +02:00
const int text_height = QFontMetrics(opt.font).height() +
QFontMetrics(status_font).height();
QRect line1(opt.rect);
QRect line2(opt.rect);
line1.setTop(line1.top() + (opt.rect.height() - text_height) / 2);
line2.setTop(line1.top() + QFontMetrics(opt.font).height());
line1.setLeft(line1.left() + DeviceManager::kDeviceIconSize + kIconPadding);
line2.setLeft(line2.left() + DeviceManager::kDeviceIconSize + kIconPadding);
// Change the color for selected items
if (opt.state & QStyle::State_Selected) {
p->setPen(opt.palette.color(QPalette::HighlightedText));
}
// Draw the icon
p->drawPixmap(opt.rect.topLeft(), index.data(Qt::DecorationRole).value<QPixmap>());
// Draw the first line (device name)
p->drawText(line1, Qt::AlignLeft | Qt::AlignTop, index.data().toString());
// Draw the second line (status)
DeviceManager::State state =
static_cast<DeviceManager::State>(index.data(DeviceManager::Role_State).toInt());
QVariant progress = index.data(DeviceManager::Role_UpdatingPercentage);
QString status_text;
if (progress.isValid()) {
status_text = tr("Updating %1%...").arg(progress.toInt());
} else {
switch (state) {
case DeviceManager::State_Remembered:
status_text = tr("Not connected");
break;
case DeviceManager::State_NotMounted:
status_text = tr("Not mounted - double click to mount");
break;
case DeviceManager::State_NotConnected:
status_text = tr("Double click to open");
break;
case DeviceManager::State_Connected: {
QVariant song_count = index.data(DeviceManager::Role_SongCount);
if (song_count.isValid()) {
int count = song_count.toInt();
if (count == 1) // TODO: Fix this properly
status_text = tr("%1 song").arg(count);
else
status_text = tr("%1 songs").arg(count);
} else {
status_text = index.data(DeviceManager::Role_MountPath).toString();
}
break;
}
}
2010-07-04 13:34:25 +02:00
}
2010-08-28 20:43:01 +02:00
if (opt.state & QStyle::State_Selected)
p->setPen(opt.palette.color(QPalette::HighlightedText));
else
p->setPen(opt.palette.color(QPalette::Dark));
p->setFont(status_font);
p->drawText(line2, Qt::AlignLeft | Qt::AlignTop, status_text);
2010-07-04 13:34:25 +02:00
p->restore();
}
DeviceView::DeviceView(QWidget* parent)
: AutoExpandingTreeView(parent),
2010-07-04 01:00:07 +02:00
manager_(NULL),
merged_model_(NULL),
sort_model_(NULL),
2010-07-04 17:01:24 +02:00
properties_dialog_(new DeviceProperties),
device_menu_(NULL),
library_menu_(NULL)
{
2010-07-04 13:34:25 +02:00
setItemDelegate(new DeviceItemDelegate(this));
SetExpandOnReset(false);
setAttribute(Qt::WA_MacShowFocusRect, false);
setHeaderHidden(true);
setAllColumnsShowFocus(true);
setDragEnabled(true);
setDragDropMode(QAbstractItemView::DragOnly);
setSelectionMode(QAbstractItemView::ExtendedSelection);
2010-07-04 01:00:07 +02:00
}
2010-07-04 17:01:24 +02:00
DeviceView::~DeviceView() {
}
2010-07-04 01:00:07 +02:00
void DeviceView::SetDeviceManager(DeviceManager *manager) {
Q_ASSERT(manager_ == NULL);
manager_ = manager;
connect(manager_, SIGNAL(DeviceConnected(int)), SLOT(DeviceConnected(int)));
2010-07-04 02:33:34 +02:00
connect(manager_, SIGNAL(DeviceDisconnected(int)), SLOT(DeviceDisconnected(int)));
2010-07-04 01:00:07 +02:00
2010-07-04 02:58:01 +02:00
sort_model_ = new QSortFilterProxyModel(this);
sort_model_->setSourceModel(manager_);
sort_model_->setDynamicSortFilter(true);
sort_model_->setSortCaseSensitivity(Qt::CaseInsensitive);
2010-07-04 02:58:01 +02:00
sort_model_->sort(0);
2010-07-04 01:00:07 +02:00
merged_model_ = new MergedProxyModel(this);
2010-07-04 02:58:01 +02:00
merged_model_->setSourceModel(sort_model_);
2010-07-04 01:00:07 +02:00
connect(merged_model_,
SIGNAL(SubModelReset(QModelIndex,QAbstractItemModel*)),
SLOT(RecursivelyExpand(QModelIndex)));
2010-07-04 02:58:01 +02:00
setModel(merged_model_);
2010-07-04 17:01:24 +02:00
properties_dialog_->SetDeviceManager(manager_);
2010-07-04 01:00:07 +02:00
}
void DeviceView::SetLibrary(LibraryModel* library) {
Q_ASSERT(manager_);
library_ = library;
organise_dialog_.reset(new OrganiseDialog(manager_->task_manager()));
organise_dialog_->SetDestinationModel(library_->directory_model());
}
2010-07-04 01:00:07 +02:00
void DeviceView::contextMenuEvent(QContextMenuEvent* e) {
if (!device_menu_) {
device_menu_ = new QMenu(this);
library_menu_ = new QMenu(this);
// Device menu
eject_action_ = device_menu_->addAction(
IconLoader::Load("media-eject"), tr("Safely remove device"), this, SLOT(Unmount()));
forget_action_ = device_menu_->addAction(
IconLoader::Load("list-remove"), tr("Forget device"), this, SLOT(Forget()));
device_menu_->addSeparator();
properties_action_ = device_menu_->addAction(
IconLoader::Load("configure"), tr("Device properties..."), this, SLOT(Properties()));
// Library menu
add_to_playlist_action_ = library_menu_->addAction(IconLoader::Load("media-playback-start"),
tr("Append to current playlist"), this, SLOT(AddToPlaylist()));
load_action_ = library_menu_->addAction(IconLoader::Load("media-playback-start"),
tr("Replace current playlist"), this, SLOT(Load()));
open_in_new_playlist_ = library_menu_->addAction(IconLoader::Load("document-new"),
tr("Open in new playlist"), this, SLOT(OpenInNewPlaylist()));
library_menu_->addSeparator();
organise_action_ = library_menu_->addAction(IconLoader::Load("edit-copy"),
tr("Copy to library..."), this, SLOT(Organise()));
delete_action_ = library_menu_->addAction(IconLoader::Load("edit-delete"),
tr("Delete from device..."), this, SLOT(Delete()));
}
2010-07-04 01:00:07 +02:00
menu_index_ = currentIndex();
const QModelIndex device_index = MapToDevice(menu_index_);
const QModelIndex library_index = MapToLibrary(menu_index_);
if (device_index.isValid()) {
const bool is_plugged_in = manager_->GetLister(device_index.row());
const bool is_remembered = manager_->GetDatabaseId(device_index.row()) != -1;
forget_action_->setEnabled(is_remembered);
eject_action_->setEnabled(is_plugged_in);
2010-07-04 02:33:34 +02:00
device_menu_->popup(e->globalPos());
} else if (library_index.isValid()) {
const QModelIndex parent_device_index = FindParentDevice(menu_index_);
bool is_filesystem_device = false;
if (parent_device_index.isValid()) {
boost::shared_ptr<ConnectedDevice> device = manager_->GetConnectedDevice(parent_device_index.row());
if (device && !device->LocalPath().isEmpty())
is_filesystem_device = true;
}
organise_action_->setEnabled(is_filesystem_device);
library_menu_->popup(e->globalPos());
}
2010-07-04 01:00:07 +02:00
}
2010-07-04 02:58:01 +02:00
QModelIndex DeviceView::MapToDevice(const QModelIndex& merged_model_index) const {
QModelIndex sort_model_index = merged_model_->mapToSource(merged_model_index);
2010-07-04 01:00:07 +02:00
if (sort_model_index.model() != sort_model_)
return QModelIndex();
2010-07-04 02:58:01 +02:00
return sort_model_->mapToSource(sort_model_index);
2010-07-04 01:00:07 +02:00
}
QModelIndex DeviceView::FindParentDevice(const QModelIndex& merged_model_index) const {
QModelIndex index = merged_model_->FindSourceParent(merged_model_index);
if (index.model() != sort_model_)
return QModelIndex();
return sort_model_->mapToSource(index);
}
QModelIndex DeviceView::MapToLibrary(const QModelIndex& merged_model_index) const {
QModelIndex sort_model_index = merged_model_->mapToSource(merged_model_index);
if (const QSortFilterProxyModel* sort_model =
qobject_cast<const QSortFilterProxyModel*>(sort_model_index.model())) {
return sort_model->mapToSource(sort_model_index);
}
return QModelIndex();
}
2010-07-04 01:00:07 +02:00
void DeviceView::Connect() {
QModelIndex device_idx = MapToDevice(menu_index_);
manager_->data(device_idx, MusicStorage::Role_StorageForceConnect);
}
void DeviceView::DeviceConnected(int row) {
boost::shared_ptr<ConnectedDevice> device = manager_->GetConnectedDevice(row);
2010-07-04 13:34:25 +02:00
if (!device)
return;
2010-07-04 02:58:01 +02:00
QModelIndex sort_idx = sort_model_->mapFromSource(manager_->index(row));
2010-07-04 02:58:01 +02:00
QSortFilterProxyModel* sort_model = new QSortFilterProxyModel(device->model());
sort_model->setSourceModel(device->model());
sort_model->setSortRole(LibraryModel::Role_SortText);
sort_model->setDynamicSortFilter(true);
sort_model->sort(0);
merged_model_->AddSubModel(sort_idx, sort_model);
expand(menu_index_);
}
2010-07-04 02:33:34 +02:00
void DeviceView::DeviceDisconnected(int row) {
2010-07-04 02:58:01 +02:00
merged_model_->RemoveSubModel(sort_model_->mapFromSource(manager_->index(row)));
2010-07-04 02:33:34 +02:00
}
void DeviceView::Forget() {
boost::scoped_ptr<QMessageBox> dialog(new QMessageBox(
QMessageBox::Question, tr("Forget device"),
tr("Forgetting a device will remove it from this list and Clementine will have to rescan all the songs again next time you connect it."),
QMessageBox::Cancel, this));
QPushButton* forget =
dialog->addButton(tr("Forget device"), QMessageBox::DestructiveRole);
dialog->exec();
if (dialog->clickedButton() != forget)
return;
QModelIndex device_idx = MapToDevice(menu_index_);
manager_->Forget(device_idx.row());
}
2010-07-04 17:01:24 +02:00
void DeviceView::Properties() {
properties_dialog_->ShowDevice(MapToDevice(menu_index_).row());
}
void DeviceView::mouseDoubleClickEvent(QMouseEvent *event) {
AutoExpandingTreeView::mouseDoubleClickEvent(event);
QModelIndex merged_index = indexAt(event->pos());
QModelIndex device_index = MapToDevice(merged_index);
if (device_index.isValid()) {
if (!manager_->GetConnectedDevice(device_index.row())) {
menu_index_ = merged_index;
Connect();
}
}
}
SongList DeviceView::GetSelectedSongs() const {
QModelIndexList selected_merged_indexes = selectionModel()->selectedRows();
SongList songs;
foreach (const QModelIndex& merged_index, selected_merged_indexes) {
QModelIndex library_index = MapToLibrary(merged_index);
if (!library_index.isValid())
continue;
const LibraryModel* library = qobject_cast<const LibraryModel*>(
library_index.model());
if (!library)
continue;
songs << library->GetChildSongs(library_index);
}
return songs;
}
void DeviceView::Load() {
QMimeData* data = model()->mimeData(selectedIndexes());
if (MimeData* mime_data = qobject_cast<MimeData*>(data)) {
mime_data->clear_first_ = true;
}
emit AddToPlaylistSignal(data);
}
void DeviceView::AddToPlaylist() {
emit AddToPlaylistSignal(model()->mimeData(selectedIndexes()));
}
void DeviceView::OpenInNewPlaylist() {
QMimeData* data = model()->mimeData(selectedIndexes());
if (MimeData* mime_data = qobject_cast<MimeData*>(data)) {
mime_data->open_in_new_playlist_ = true;
}
emit AddToPlaylistSignal(data);
}
void DeviceView::Delete() {
if (selectedIndexes().isEmpty())
return;
// Take the device of the first selected item
QModelIndex device_index = FindParentDevice(selectedIndexes()[0]);
if (!device_index.isValid())
return;
if (QMessageBox::question(this, tr("Delete files"),
tr("These files will be deleted from the device, are you sure you want to continue?"),
QMessageBox::Yes, QMessageBox::Cancel) != QMessageBox::Yes)
return;
boost::shared_ptr<MusicStorage> storage =
device_index.data(MusicStorage::Role_Storage).value<boost::shared_ptr<MusicStorage> >();
DeleteFiles* delete_files = new DeleteFiles(manager_->task_manager(), storage);
connect(delete_files, SIGNAL(Finished(SongList)), SLOT(DeleteFinished(SongList)));
delete_files->Start(GetSelectedSongs());
}
void DeviceView::Organise() {
SongList songs = GetSelectedSongs();
QStringList filenames;
foreach (const Song& song, songs) {
filenames << song.filename();
}
organise_dialog_->SetCopy(true);
organise_dialog_->SetFilenames(filenames);
organise_dialog_->show();
}
void DeviceView::Unmount() {
QModelIndex device_idx = MapToDevice(menu_index_);
manager_->Unmount(device_idx.row());
}
void DeviceView::DeleteFinished(const SongList& songs_with_errors) {
if (songs_with_errors.isEmpty())
return;
OrganiseErrorDialog* dialog = new OrganiseErrorDialog(this);
dialog->Show(OrganiseErrorDialog::Type_Delete, songs_with_errors);
// It deletes itself when the user closes it
}
bool DeviceView::CanRecursivelyExpand(const QModelIndex& index) const {
// Never expand devices
return index.parent().isValid();
}