Files
cantata/models/dirviewmodel.cpp
craig.p.drummond 8894263b1f More android work...
2012-06-19 15:48:18 +00:00

346 lines
9.2 KiB
C++

/*
* Cantata
*
* Copyright (c) 2011-2012 Craig Drummond <craig.p.drummond@gmail.com>
*
*/
/*
* Copyright (c) 2008 Sander Knopper (sander AT knopper DOT tk) and
* Roeland Douma (roeland AT rullzer DOT com)
*
* This file is part of QtMPC.
*
* QtMPC 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.
*
* QtMPC 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 QtMPC. If not, see <http://www.gnu.org/licenses/>.
*/
#include <QtCore/QModelIndex>
#include <QtCore/QString>
#include <QtCore/QVariant>
#include <QtCore/QMimeData>
#include "localize.h"
#include "dirviewmodel.h"
#include "dirviewitem.h"
#include "playqueuemodel.h"
#include "itemview.h"
#include "settings.h"
#include "mpdconnection.h"
#include "debugtimer.h"
#include "icon.h"
#ifdef ENABLE_KDE_SUPPORT
K_GLOBAL_STATIC(DirViewModel, instance)
#endif
DirViewModel * DirViewModel::self()
{
#ifdef ENABLE_KDE_SUPPORT
return instance;
#else
static DirViewModel *instance=0;
if(!instance) {
instance=new DirViewModel;
}
return instance;
#endif
}
DirViewModel::DirViewModel(QObject *parent)
: QAbstractItemModel(parent)
, rootItem(new DirViewItemRoot)
, enabled(false)
{
}
DirViewModel::~DirViewModel()
{
delete rootItem;
}
void DirViewModel::setEnabled(bool e)
{
if (e==enabled) {
return;
}
enabled=e;
if (enabled) {
connect(MPDConnection::self(), SIGNAL(dirViewUpdated(DirViewItemRoot *)), this, SLOT(updateDirView(DirViewItemRoot *)));
} else {
disconnect(MPDConnection::self(), SIGNAL(dirViewUpdated(DirViewItemRoot *)), this, SLOT(updateDirView(DirViewItemRoot *)));
}
}
QModelIndex DirViewModel::index(int row, int column, const QModelIndex &parent) const
{
if (!hasIndex(row, column, parent)) {
return QModelIndex();
}
const DirViewItem * parentItem;
if (!parent.isValid()) {
parentItem = rootItem;
} else {
parentItem = static_cast<DirViewItem *>(parent.internalPointer());
}
DirViewItem * const childItem = parentItem->child(row);
if (childItem) {
return createIndex(row, column, childItem);
}
return QModelIndex();
}
QModelIndex DirViewModel::parent(const QModelIndex &index) const
{
if (!index.isValid()) {
return QModelIndex();
}
const DirViewItem * const childItem = static_cast<DirViewItem *>(index.internalPointer());
DirViewItem * const parentItem = childItem->parent();
if (parentItem == rootItem) {
return QModelIndex();
}
return createIndex(parentItem->row(), 0, parentItem);
}
QVariant DirViewModel::headerData(int /*section*/, Qt::Orientation /*orientation*/, int /*role*/) const
{
return QVariant();
}
int DirViewModel::rowCount(const QModelIndex &parent) const
{
if (parent.column() > 0) {
return 0;
}
const DirViewItem *parentItem;
if (!parent.isValid()) {
parentItem = rootItem;
} else {
parentItem = static_cast<DirViewItem *>(parent.internalPointer());
}
return parentItem->childCount();
}
int DirViewModel::columnCount(const QModelIndex &parent) const
{
if (parent.isValid()) {
return static_cast<DirViewItem *>(parent.internalPointer())->columnCount();
} else {
return rootItem->columnCount();
}
}
QVariant DirViewModel::data(const QModelIndex &index, int role) const
{
if (!index.isValid()) {
return QVariant();
}
DirViewItem *item = static_cast<DirViewItem *>(index.internalPointer());
switch (role) {
case Qt::DecorationRole: {
if (item->type() == DirViewItem::Type_Dir) {
return Icon("inode-directory");
} else if (item->type() == DirViewItem::Type_File) {
return Icon("audio-x-generic");
}
break;
}
case Qt::DisplayRole:
return item->data();
case Qt::ToolTipRole:
return 0==item->childCount()
? item->data()
:
#ifdef ENABLE_KDE_SUPPORT
i18np("%1\n1 Entry", "%1\n%2 Entries", item->data(), item->childCount());
#else
(item->childCount()>1
? tr("%1\n%2 Entries").arg(item->data()).arg(item->childCount())
: tr("%1\n1 Entry").arg(item->data()));
#endif
case ItemView::Role_SubText:
switch (item->type()) {
case DirViewItem::Type_Dir:
#ifdef ENABLE_KDE_SUPPORT
return i18np("1 Entry", "%1 Entries", item->childCount());
#else
return 1==item->childCount() ? tr("1 Entry") : tr("%1 Entries").arg(item->childCount());
#endif
break;
case DirViewItem::Type_File:
default: return QVariant();
}
default:
break;
}
return QVariant();
}
void DirViewModel::clear()
{
updateDirView(new DirViewItemRoot());
}
void DirViewModel::updateDirView(DirViewItemRoot *newroot)
{
TF_DEBUG
bool incremental=enabled && rootItem->childCount() && newroot->childCount();
if (incremental) {
QSet<QString> currentFiles=rootItem->allFiles();
QSet<QString> updateFiles=newroot->allFiles();
QSet<QString> removed=currentFiles-updateFiles;
QSet<QString> added=updateFiles-currentFiles;
foreach (const QString &s, added) {
addFileToList(s);
}
foreach (const QString &s, removed) {
removeFileFromList(s);
}
} else {
const DirViewItemRoot *oldRoot = rootItem;
beginResetModel();
rootItem = newroot;
delete oldRoot;
endResetModel();
}
}
void DirViewModel::addFileToList(const QString &file)
{
if (!enabled) {
return;
}
addFileToList(file.split('/'), QModelIndex(), rootItem);
}
void DirViewModel::removeFileFromList(const QString &file)
{
if (!enabled) {
return;
}
removeFileFromList(file.split('/'), QModelIndex(), rootItem);
}
void DirViewModel::addFileToList(const QStringList &parts, const QModelIndex &parent, DirViewItemDir *dir)
{
if (0==parts.count()) {
return;
}
QString p=parts[0];
DirViewItem *child=dir->child(p);
if (child) {
if (DirViewItem::Type_Dir==child->type()) {
addFileToList(parts.mid(1), index(dir->indexOf(child), 0, parent), static_cast<DirViewItemDir *>(child));
}
} else {
beginInsertRows(parent, dir->childCount(), dir->childCount());
dir->insertFile(parts);
endInsertRows();
}
}
void DirViewModel::removeFileFromList(const QStringList &parts, const QModelIndex &parent, DirViewItemDir *dir)
{
if (0==parts.count()) {
return;
}
QString p=parts[0];
DirViewItem *child=dir->child(p);
if (child) {
if (DirViewItem::Type_Dir==child->type()) {
removeFileFromList(parts.mid(1), index(dir->indexOf(child), 0, parent), static_cast<DirViewItemDir *>(child));
} else if (DirViewItem::Type_File==child->type()) {
int index=dir->indexOf(child);
beginRemoveRows(parent, index, index);
dir->remove(child);
endRemoveRows();
}
}
if (0==dir->childCount() && dir->parent()) {
DirViewItemDir *pd=static_cast<DirViewItemDir *>(dir->parent());
int index=pd->indexOf(dir);
beginRemoveRows(parent.parent(), index, index);
pd->remove(dir);
delete dir;
endRemoveRows();
}
}
Qt::ItemFlags DirViewModel::flags(const QModelIndex &index) const
{
if (index.isValid()) {
return Qt::ItemIsSelectable | Qt::ItemIsDragEnabled | Qt::ItemIsEnabled;
} else {
return Qt::ItemIsDropEnabled;
}
}
QStringList DirViewModel::filenames(const QModelIndexList &indexes) const
{
QStringList fnames;
foreach(QModelIndex index, indexes) {
DirViewItem *item = static_cast<DirViewItem *>(index.internalPointer());
recurseDirItems(*item, fnames);
}
return fnames;
}
QMimeData *DirViewModel::mimeData(const QModelIndexList &indexes) const
{
QMimeData *mimeData = new QMimeData();
QStringList files=filenames(indexes);
PlayQueueModel::encode(*mimeData, PlayQueueModel::constFileNameMimeType, files);
if (!MPDConnection::self()->getDetails().dir.isEmpty()) {
QStringList paths;
foreach (const QString &f, files) {
paths << MPDConnection::self()->getDetails().dir+f;
}
PlayQueueModel::encode(*mimeData, PlayQueueModel::constUriMimeType, paths);
}
return mimeData;
}
void DirViewModel::recurseDirItems(DirViewItem &parent, QStringList &filenames) const
{
if (parent.childCount() == 0) {
if (!filenames.contains(parent.fullName())) {
filenames << parent.fullName();
}
} else {
for (int i = 0; i < parent.childCount(); i++) {
recurseDirItems(*parent.child(i), filenames);
}
}
}