kde-extraapps/juk/playlistitem.cpp
2014-11-28 15:08:46 +00:00

485 lines
13 KiB
C++

/**
* Copyright (C) 2002-2004 Scott Wheeler <wheeler@kde.org>
*
* 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. If not, see <http://www.gnu.org/licenses/>.
*/
#include "playlistitem.h"
#include <config-juk.h>
#include <kdebug.h>
#include <kiconloader.h>
#include <QPixmap>
#include <QFileInfo>
#include "collectionlist.h"
#include "musicbrainzquery.h"
#include "tag.h"
#include "coverinfo.h"
#include "covermanager.h"
#include "tagtransactionmanager.h"
PlaylistItemList PlaylistItem::m_playingItems; // static
static void startMusicBrainzQuery(const FileHandle &file)
{
#if HAVE_TUNEPIMP
// This deletes itself when finished.
new MusicBrainzLookup(file);
#else
Q_UNUSED(file)
#endif
}
////////////////////////////////////////////////////////////////////////////////
// PlaylistItem public methods
////////////////////////////////////////////////////////////////////////////////
PlaylistItem::~PlaylistItem()
{
// Although this isn't the most efficient way to accomplish the task of
// stopping playback when deleting the item being played, it has the
// stark advantage of working reliably. I'll tell anyone who tries to
// optimize this, the timing issues can be *hard*. -- mpyne
m_collectionItem->removeChildItem(this);
if(m_playingItems.contains(this)) {
m_playingItems.removeAll(this);
if(m_playingItems.isEmpty())
playlist()->setPlaying(0);
}
playlist()->updateDeletedItem(this);
emit playlist()->signalAboutToRemove(this);
if(m_watched)
Pointer::clear(this);
}
void PlaylistItem::setFile(const FileHandle &file)
{
m_collectionItem->updateCollectionDict(d->fileHandle.absFilePath(), file.absFilePath());
d->fileHandle = file;
refresh();
}
void PlaylistItem::setFile(const QString &file)
{
QString oldPath = d->fileHandle.absFilePath();
d->fileHandle.setFile(file);
m_collectionItem->updateCollectionDict(oldPath, d->fileHandle.absFilePath());
refresh();
}
FileHandle PlaylistItem::file() const
{
return d->fileHandle;
}
K_GLOBAL_STATIC_WITH_ARGS(QPixmap, globalGenericImage, (SmallIcon("image-x-generic")))
K_GLOBAL_STATIC_WITH_ARGS(QPixmap, globalPlayingImage, (UserIcon("playing")))
const QPixmap *PlaylistItem::pixmap(int column) const
{
int offset = playlist()->columnOffset();
// Don't use hasCover here because that may dig into the track itself.
// Besides, we really just want to know if the cover manager has a cover
// for the track.
if((column - offset) == CoverColumn &&
d->fileHandle.coverInfo()->coverId() != CoverManager::NoMatch)
{
return globalGenericImage;
}
if(column == playlist()->leftColumn() &&
m_playingItems.contains(const_cast<PlaylistItem *>(this)))
{
return globalPlayingImage;
}
return K3ListViewItem::pixmap(column);
}
QString PlaylistItem::text(int column) const
{
if(!d->fileHandle.tag())
return QString();
int offset = playlist()->columnOffset();
switch(column - offset) {
case TrackColumn:
return d->fileHandle.tag()->title();
case ArtistColumn:
return d->fileHandle.tag()->artist();
case AlbumColumn:
return d->fileHandle.tag()->album();
case CoverColumn:
return QString();
case TrackNumberColumn:
return d->fileHandle.tag()->track() > 0
? QString::number(d->fileHandle.tag()->track())
: QString();
case GenreColumn:
return d->fileHandle.tag()->genre();
case YearColumn:
return d->fileHandle.tag()->year() > 0
? QString::number(d->fileHandle.tag()->year())
: QString();
case LengthColumn:
return d->fileHandle.tag()->lengthString();
case BitrateColumn:
return QString::number(d->fileHandle.tag()->bitrate());
case CommentColumn:
return d->fileHandle.tag()->comment();
case FileNameColumn:
return d->fileHandle.fileInfo().fileName();
case FullPathColumn:
return d->fileHandle.fileInfo().absoluteFilePath();
default:
return K3ListViewItem::text(column);
}
}
void PlaylistItem::setText(int column, const QString &text)
{
int offset = playlist()->columnOffset();
if(column - offset >= 0 && column + offset <= lastColumn()) {
K3ListViewItem::setText(column, QString());
return;
}
K3ListViewItem::setText(column, text);
playlist()->slotWeightDirty(column);
}
void PlaylistItem::setPlaying(bool playing, bool master)
{
m_playingItems.removeAll(this);
if(playing) {
if(master)
m_playingItems.prepend(this);
else
m_playingItems.append(this);
}
else {
// This is a tricky little recursion, but it
// in fact does clear the list.
if(!m_playingItems.isEmpty())
m_playingItems.front()->setPlaying(false);
}
listView()->triggerUpdate();
}
void PlaylistItem::setSelected(bool selected)
{
playlist()->markItemSelected(this, selected);
K3ListViewItem::setSelected(selected);
}
void PlaylistItem::guessTagInfo(TagGuesser::Type type)
{
switch(type) {
case TagGuesser::FileName:
{
TagGuesser guesser(d->fileHandle.absFilePath());
Tag *tag = TagTransactionManager::duplicateTag(d->fileHandle.tag());
if(!guesser.title().isNull())
tag->setTitle(guesser.title());
if(!guesser.artist().isNull())
tag->setArtist(guesser.artist());
if(!guesser.album().isNull())
tag->setAlbum(guesser.album());
if(!guesser.track().isNull())
tag->setTrack(guesser.track().toInt());
if(!guesser.comment().isNull())
tag->setComment(guesser.comment());
TagTransactionManager::instance()->changeTagOnItem(this, tag);
break;
}
case TagGuesser::MusicBrainz:
startMusicBrainzQuery(d->fileHandle);
break;
}
}
Playlist *PlaylistItem::playlist() const
{
return static_cast<Playlist *>(listView());
}
QVector<int> PlaylistItem::cachedWidths() const
{
return d->cachedWidths;
}
void PlaylistItem::refresh()
{
m_collectionItem->refresh();
}
void PlaylistItem::refreshFromDisk()
{
d->fileHandle.refresh();
refresh();
}
void PlaylistItem::clear()
{
playlist()->clearItem(this);
}
////////////////////////////////////////////////////////////////////////////////
// PlaylistItem protected methods
////////////////////////////////////////////////////////////////////////////////
PlaylistItem::PlaylistItem(CollectionListItem *item, Playlist *parent) :
K3ListViewItem(parent),
d(0),
m_watched(0)
{
setup(item);
}
PlaylistItem::PlaylistItem(CollectionListItem *item, Playlist *parent, Q3ListViewItem *after) :
K3ListViewItem(parent, after),
d(0),
m_watched(0)
{
setup(item);
}
// This constructor should only be used by the CollectionList subclass.
PlaylistItem::PlaylistItem(CollectionList *parent) :
K3ListViewItem(parent),
m_watched(0)
{
d = new Data;
m_collectionItem = static_cast<CollectionListItem *>(this);
setDragEnabled(true);
}
void PlaylistItem::paintCell(QPainter *p, const QColorGroup &cg, int column, int width, int align)
{
if(!m_playingItems.contains(this))
return K3ListViewItem::paintCell(p, cg, column, width, align);
QPalette colorGroup = cg;
QColor base = colorGroup.color( QPalette::Base );
QColor selection = colorGroup.color( QPalette::Highlight );
int r = (base.red() + selection.red()) / 2;
int b = (base.blue() + selection.blue()) / 2;
int g = (base.green() + selection.green()) / 2;
QColor c(r, g, b);
colorGroup.setColor(QPalette::Base, c);
Q3ListViewItem::paintCell(p, colorGroup, column, width, align);
}
int PlaylistItem::compare(Q3ListViewItem *item, int column, bool ascending) const
{
// reimplemented from QListViewItem
int offset = playlist()->columnOffset();
if(!item)
return 0;
PlaylistItem *playlistItem = static_cast<PlaylistItem *>(item);
// The following statments first check to see if you can sort based on the
// specified column. If the values for the two PlaylistItems are the same
// in that column it then tries to sort based on columns 1, 2, 3 and 0,
// (artist, album, track number, track name) in that order.
int c = compare(this, playlistItem, column, ascending);
if(c != 0)
return c;
else {
// Loop through the columns doing comparisons until something is differnt.
// If all else is the same, compare the track name.
int last = playlist()->isColumnVisible(AlbumColumn + offset) ? TrackNumberColumn : ArtistColumn;
for(int i = ArtistColumn; i <= last; i++) {
if(playlist()->isColumnVisible(i + offset)) {
c = compare(this, playlistItem, i, ascending);
if(c != 0)
return c;
}
}
return compare(this, playlistItem, TrackColumn + offset, ascending);
}
}
int PlaylistItem::compare(const PlaylistItem *firstItem, const PlaylistItem *secondItem, int column, bool) const
{
int offset = playlist()->columnOffset();
if(column < 0 || column > lastColumn() + offset || !firstItem->d || !secondItem->d)
return 0;
if(column < offset) {
QString first = firstItem->text(column).toLower();
QString second = secondItem->text(column).toLower();
return first.localeAwareCompare(second);
}
switch(column - offset) {
case TrackNumberColumn:
if(firstItem->d->fileHandle.tag()->track() > secondItem->d->fileHandle.tag()->track())
return 1;
else if(firstItem->d->fileHandle.tag()->track() < secondItem->d->fileHandle.tag()->track())
return -1;
else
return 0;
break;
case LengthColumn:
if(firstItem->d->fileHandle.tag()->seconds() > secondItem->d->fileHandle.tag()->seconds())
return 1;
else if(firstItem->d->fileHandle.tag()->seconds() < secondItem->d->fileHandle.tag()->seconds())
return -1;
else
return 0;
break;
case BitrateColumn:
if(firstItem->d->fileHandle.tag()->bitrate() > secondItem->d->fileHandle.tag()->bitrate())
return 1;
else if(firstItem->d->fileHandle.tag()->bitrate() < secondItem->d->fileHandle.tag()->bitrate())
return -1;
else
return 0;
break;
case CoverColumn:
if(firstItem->d->fileHandle.coverInfo()->coverId() == secondItem->d->fileHandle.coverInfo()->coverId())
return 0;
else if (firstItem->d->fileHandle.coverInfo()->coverId() != CoverManager::NoMatch)
return -1;
else
return 1;
break;
default:
return QString::localeAwareCompare(firstItem->d->metadata[column - offset],
secondItem->d->metadata[column - offset]);
}
}
bool PlaylistItem::isValid() const
{
return bool(d->fileHandle.tag());
}
void PlaylistItem::setTrackId(quint32 id)
{
m_trackId = id;
}
////////////////////////////////////////////////////////////////////////////////
// PlaylistItem private methods
////////////////////////////////////////////////////////////////////////////////
void PlaylistItem::setup(CollectionListItem *item)
{
m_collectionItem = item;
d = item->d;
item->addChildItem(this);
setDragEnabled(true);
}
////////////////////////////////////////////////////////////////////////////////
// PlaylistItem::Pointer implementation
////////////////////////////////////////////////////////////////////////////////
QMap<PlaylistItem *, QList<PlaylistItem::Pointer *> > PlaylistItem::Pointer::m_map; // static
PlaylistItem::Pointer::Pointer(PlaylistItem *item) :
m_item(item)
{
if(!m_item)
return;
m_item->m_watched = true;
m_map[m_item].append(this);
}
PlaylistItem::Pointer::Pointer(const Pointer &p) :
m_item(p.m_item)
{
m_map[m_item].append(this);
}
PlaylistItem::Pointer::~Pointer()
{
if(!m_item)
return;
m_map[m_item].removeAll(this);
if(m_map[m_item].isEmpty()) {
m_map.remove(m_item);
m_item->m_watched = false;
}
}
PlaylistItem::Pointer &PlaylistItem::Pointer::operator=(PlaylistItem *item)
{
if(item == m_item)
return *this;
if(m_item) {
m_map[m_item].removeAll(this);
if(m_map[m_item].isEmpty()) {
m_map.remove(m_item);
m_item->m_watched = false;
}
}
if(item) {
m_map[item].append(this);
item->m_watched = true;
}
m_item = item;
return *this;
}
void PlaylistItem::Pointer::clear(PlaylistItem *item) // static
{
if(!item)
return;
QList<Pointer *> l = m_map[item];
foreach(Pointer *pointer, l)
pointer->m_item = 0;
m_map.remove(item);
item->m_watched = false;
}
// vim: set et sw=4 tw=0 sta: