286 lines
8.0 KiB
C++
286 lines
8.0 KiB
C++
#include "covers.h"
|
|
#include "musiclibrarymodel.h"
|
|
#include "lib/song.h"
|
|
#include "lib/mpdparseutils.h"
|
|
#include "maiaXmlRpcClient.h"
|
|
#include "networkaccessmanager.h"
|
|
#include <QtCore/QFile>
|
|
#include <QtCore/QCryptographicHash>
|
|
#include <QtNetwork/QNetworkReply>
|
|
#include <QtGui/QImage>
|
|
|
|
#ifdef ENABLE_KDE_SUPPORT
|
|
#include <KDE/KStandardDirs>
|
|
#include <KDE/KGlobal>
|
|
K_GLOBAL_STATIC(Covers, instance)
|
|
#else
|
|
#endif
|
|
|
|
static const QLatin1String constApiKey("11172d35eb8cc2fd33250a9e45a2d486");
|
|
static const QLatin1String constCoverDir("covers/");
|
|
static const QLatin1String constFileName("cover");
|
|
static const QLatin1String constExtension(".jpg");
|
|
static const QLatin1String constAltExtension(".png");
|
|
|
|
static const QString typeFromRaw(const QByteArray &raw)
|
|
{
|
|
if (raw.size()>9 && /*raw[0]==0xFF && raw[1]==0xD8 && raw[2]==0xFF*/ raw[6]=='J' && raw[7]=='F' && raw[8]=='I' && raw[9]=='F') {
|
|
return constExtension;
|
|
} else if (raw.size()>4 && /*raw[0]==0x89 &&*/ raw[1]=='P' && raw[2]=='N' && raw[3]=='G') {
|
|
return constAltExtension;
|
|
}
|
|
return QString();
|
|
}
|
|
|
|
static bool save(const QString &mimeType, const QString &extension, const QString &filePrefix, const QImage &img, const QByteArray &raw)
|
|
{
|
|
if (!mimeType.isEmpty()) {
|
|
if (QFile::exists(filePrefix+mimeType)) {
|
|
return true;
|
|
}
|
|
|
|
QFile f(filePrefix+mimeType);
|
|
if (f.open(QIODevice::WriteOnly) && raw.size()==f.write(raw)) {
|
|
return true;
|
|
}
|
|
}
|
|
|
|
if (extension!=mimeType) {
|
|
if (QFile::exists(filePrefix+extension)) {
|
|
return true;
|
|
}
|
|
|
|
if (img.save(filePrefix+extension)) {
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
static QString encodeName(QString name)
|
|
{
|
|
name.replace("/", "_");
|
|
return name;
|
|
}
|
|
|
|
static QImage amarokCover(const Covers::Job &job)
|
|
{
|
|
#ifdef ENABLE_KDE_SUPPORT
|
|
QString kdeDir=KGlobal::dirs()->localkdedir();
|
|
#else
|
|
QString kdeDir=QDir::homePath()+"/.kde";
|
|
#endif
|
|
QString amarokName=kdeDir+"/share/apps/amarok/albumcovers/large/"+
|
|
QCryptographicHash::hash(job.artist.toLower().toLocal8Bit()+job.album.toLower().toLocal8Bit(),
|
|
QCryptographicHash::Md5).toHex();
|
|
|
|
QImage img(amarokName);
|
|
|
|
if (!img.isNull() && !job.dir.isEmpty()) {
|
|
QFile f(amarokName);
|
|
if (f.open(QIODevice::ReadOnly)) {
|
|
QByteArray raw=f.readAll();
|
|
if (!raw.isEmpty()) {
|
|
QString mimeType=typeFromRaw(raw);
|
|
save(mimeType, mimeType.isEmpty() ? constExtension : mimeType, job.dir+constFileName, img, raw);
|
|
}
|
|
}
|
|
}
|
|
return img;
|
|
}
|
|
|
|
Covers * Covers::self()
|
|
{
|
|
#ifdef ENABLE_KDE_SUPPORT
|
|
return instance;
|
|
#else
|
|
static Covers *instance=0;;
|
|
if(!instance) {
|
|
instance=new Covers;
|
|
}
|
|
return instance;
|
|
#endif
|
|
}
|
|
|
|
Covers::Covers()
|
|
: rpc(0)
|
|
, manager(0)
|
|
{
|
|
}
|
|
|
|
void Covers::get(const Song &song)
|
|
{
|
|
QString dirName;
|
|
QStringList extensions;
|
|
extensions << constAltExtension << constExtension;
|
|
|
|
if (!mpdDir.isEmpty()) {
|
|
dirName=song.file.endsWith("/") ? mpdDir+song.file : MPDParseUtils::getDir(mpdDir+song.file);
|
|
foreach (const QString &ext, extensions) {
|
|
if (QFile::exists(dirName+constFileName+ext)) {
|
|
QImage img(dirName+constFileName+ext);
|
|
|
|
if (!img.isNull()) {
|
|
emit cover(song.albumArtist(), song.album, img);
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
QString artist=encodeName(song.albumArtist());
|
|
QString album=encodeName(song.album);
|
|
// Check if cover is already cached
|
|
QString dir(MusicLibraryModel::cacheDir(constCoverDir+artist, false));
|
|
foreach (const QString &ext, extensions) {
|
|
if (QFile::exists(dir+album+ext)) {
|
|
QImage img(dir+album+ext);
|
|
if (!img.isNull()) {
|
|
emit cover(artist, album, img);
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (jobs.end()!=findJob(song)) {
|
|
return;
|
|
}
|
|
|
|
Job job(song.albumArtist(), song.album, dirName);
|
|
|
|
// See if amarok has it...
|
|
QImage img=amarokCover(job);
|
|
if (!img.isNull()) {
|
|
emit cover(artist, album, img);
|
|
return;
|
|
}
|
|
|
|
// Query lastfm...
|
|
if (!rpc) {
|
|
rpc = new MaiaXmlRpcClient(QUrl("http://ws.audioscrobbler.com/2.0/"));
|
|
}
|
|
|
|
if (!manager) {
|
|
manager=new NetworkAccessManager(this);
|
|
connect(manager, SIGNAL(finished(QNetworkReply *)), this, SLOT(jobFinished(QNetworkReply *)));
|
|
}
|
|
|
|
QMap<QString, QVariant> args;
|
|
|
|
args["artist"] = song.albumArtist();
|
|
args["album"] = song.album;
|
|
args["api_key"] = constApiKey;
|
|
QNetworkReply *reply=rpc->call("album.getInfo", QVariantList() << args,
|
|
this, SLOT(albumInfo(QVariant &, QNetworkReply *)),
|
|
this, SLOT(albumFailure(int, const QString &, QNetworkReply *)));
|
|
jobs.insert(reply, job);
|
|
}
|
|
|
|
void Covers::albumInfo(QVariant &value, QNetworkReply *reply)
|
|
{
|
|
QMap<QNetworkReply *, Job>::Iterator it(jobs.find(reply));
|
|
QMap<QNetworkReply *, Job>::Iterator end(jobs.end());
|
|
|
|
if (it!=end) {
|
|
QString xmldoc = value.toString();
|
|
xmldoc.replace("\\\"", "\"");
|
|
|
|
QXmlStreamReader doc(xmldoc);
|
|
QString url;
|
|
Job job=it.value();
|
|
|
|
while (!doc.atEnd() && url.isEmpty()) {
|
|
doc.readNext();
|
|
if (url.isEmpty() && doc.isStartElement() && doc.name() == "image" && doc.attributes().value("size").toString() == "extralarge") {
|
|
url = doc.readElementText();
|
|
}
|
|
}
|
|
|
|
if (!url.isEmpty()) {
|
|
QUrl u;
|
|
u.setEncodedUrl(url.toLatin1());
|
|
jobs.remove(it.key());
|
|
jobs.insert(manager->get(QNetworkRequest(u)), job);
|
|
} else {
|
|
emit cover(job.artist, job.album, QImage());
|
|
}
|
|
}
|
|
reply->deleteLater();
|
|
}
|
|
|
|
void Covers::albumFailure(int, const QString &, QNetworkReply *reply)
|
|
{
|
|
QMap<QNetworkReply *, Job>::Iterator it(jobs.find(reply));
|
|
QMap<QNetworkReply *, Job>::Iterator end(jobs.end());
|
|
|
|
if (it!=end) {
|
|
Job job=it.value();
|
|
emit cover(job.artist, job.album, amarokCover(job));
|
|
jobs.remove(it.key());
|
|
}
|
|
|
|
reply->deleteLater();
|
|
}
|
|
|
|
void Covers::jobFinished(QNetworkReply *reply)
|
|
{
|
|
QMap<QNetworkReply *, Job>::Iterator it(jobs.find(reply));
|
|
QMap<QNetworkReply *, Job>::Iterator end(jobs.end());
|
|
|
|
if (it!=end) {
|
|
QByteArray data=QNetworkReply::NoError==reply->error() ? reply->readAll() : QByteArray();
|
|
QImage img = data.isEmpty() ? QImage() : QImage::fromData(data);
|
|
Job job=it.value();
|
|
|
|
if (!img.isNull() && img.size().width()<32) {
|
|
img = QImage();
|
|
}
|
|
|
|
if (!img.isNull()) {
|
|
saveImg(job, img, data);
|
|
}
|
|
|
|
jobs.remove(it.key());
|
|
|
|
if (img.isNull()) {
|
|
emit cover(job.artist, job.album, amarokCover(job));
|
|
} else {
|
|
emit cover(job.artist, job.album, img);
|
|
}
|
|
}
|
|
|
|
reply->deleteLater();
|
|
}
|
|
|
|
void Covers::saveImg(const Job &job, const QImage &img, const QByteArray &raw)
|
|
{
|
|
QString mimeType=typeFromRaw(raw);
|
|
QString extension=mimeType.isEmpty() ? constExtension : mimeType;
|
|
|
|
// Try to save as cover.jpg in album dir...
|
|
if (!job.dir.isEmpty() && save(mimeType, extension, job.dir+constFileName, img, raw)) {
|
|
return;
|
|
}
|
|
|
|
// Could not save with album, save in cache dir...
|
|
QString dir = MusicLibraryModel::cacheDir(constCoverDir+encodeName(job.artist));
|
|
if (!dir.isEmpty()) {
|
|
save(mimeType, extension, dir+encodeName(job.album), img, raw);
|
|
}
|
|
}
|
|
|
|
QMap<QNetworkReply *, Covers::Job>::Iterator Covers::findJob(const Song &song)
|
|
{
|
|
QMap<QNetworkReply *, Job>::Iterator it(jobs.begin());
|
|
QMap<QNetworkReply *, Job>::Iterator end(jobs.end());
|
|
|
|
for (; it!=end; ++it) {
|
|
if ((*it).artist==song.albumArtist() && (*it).album==song.album) {
|
|
return it;
|
|
}
|
|
}
|
|
|
|
return end;
|
|
}
|