2012-05-30 01:32:34 +02:00
|
|
|
/* This file is part of Clementine.
|
|
|
|
Copyright 2012, 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/>.
|
|
|
|
*/
|
|
|
|
|
2012-05-30 01:31:27 +02:00
|
|
|
#include "songkickconcerts.h"
|
|
|
|
|
2012-06-04 14:40:08 +02:00
|
|
|
#include <QImage>
|
2012-06-04 14:06:43 +02:00
|
|
|
#include <QXmlStreamWriter>
|
|
|
|
|
2012-05-30 01:31:27 +02:00
|
|
|
#include <echonest/Artist.h>
|
2012-06-04 14:06:43 +02:00
|
|
|
|
|
|
|
#include <qjson/parser.h>
|
|
|
|
|
2012-05-30 01:31:27 +02:00
|
|
|
#include "core/closure.h"
|
2012-06-04 14:06:43 +02:00
|
|
|
#include "songinfotextview.h"
|
2012-05-30 01:31:27 +02:00
|
|
|
|
|
|
|
const char* SongkickConcerts::kSongkickArtistBucket = "id:songkick";
|
|
|
|
const char* SongkickConcerts::kSongkickArtistCalendarUrl =
|
2012-05-30 22:30:02 +02:00
|
|
|
"http://api.songkick.com/api/3.0/artists/%1/calendar.json?"
|
|
|
|
"per_page=5&"
|
|
|
|
"apikey=8rgKfy1WU6IlJFfN";
|
2012-05-30 01:31:27 +02:00
|
|
|
|
|
|
|
SongkickConcerts::SongkickConcerts() {
|
2012-06-20 14:12:37 +02:00
|
|
|
Geolocator* geolocator = new Geolocator;
|
|
|
|
geolocator->Geolocate();
|
|
|
|
connect(geolocator, SIGNAL(Finished(Geolocator::LatLng)), SLOT(GeolocateFinished(Geolocator::LatLng)));
|
|
|
|
NewClosure(geolocator, SIGNAL(Finished(Geolocator::LatLng)), geolocator, SLOT(deleteLater()));
|
2012-05-30 01:31:27 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void SongkickConcerts::FetchInfo(int id, const Song& metadata) {
|
|
|
|
Echonest::Artist::SearchParams params;
|
|
|
|
params.push_back(qMakePair(Echonest::Artist::Name, QVariant(metadata.artist())));
|
|
|
|
params.push_back(qMakePair(Echonest::Artist::IdSpace, QVariant(kSongkickArtistBucket)));
|
|
|
|
qLog(Debug) << "Params:" << params;
|
|
|
|
QNetworkReply* reply = Echonest::Artist::search(params);
|
|
|
|
qLog(Debug) << reply->request().url();
|
|
|
|
NewClosure(reply, SIGNAL(finished()), this, SLOT(ArtistSearchFinished(QNetworkReply*, int)), reply, id);
|
|
|
|
}
|
|
|
|
|
|
|
|
void SongkickConcerts::ArtistSearchFinished(QNetworkReply* reply, int id) {
|
|
|
|
reply->deleteLater();
|
|
|
|
try {
|
|
|
|
Echonest::Artists artists = Echonest::Artist::parseSearch(reply);
|
|
|
|
if (artists.isEmpty()) {
|
|
|
|
qLog(Debug) << "Failed to find artist in echonest";
|
2012-08-27 11:49:54 +02:00
|
|
|
emit Finished(id);
|
2012-05-30 01:31:27 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const Echonest::Artist& artist = artists[0];
|
|
|
|
const Echonest::ForeignIds& foreign_ids = artist.foreignIds();
|
|
|
|
QString songkick_id;
|
|
|
|
foreach (const Echonest::ForeignId& id, foreign_ids) {
|
|
|
|
if (id.catalog == "songkick") {
|
|
|
|
songkick_id = id.foreign_id;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (songkick_id.isEmpty()) {
|
|
|
|
qLog(Debug) << "Failed to fetch songkick foreign id for artist";
|
2012-08-27 11:49:54 +02:00
|
|
|
emit Finished(id);
|
2012-05-30 01:31:27 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2012-05-30 22:30:02 +02:00
|
|
|
QStringList split = songkick_id.split(':');
|
|
|
|
if (split.count() != 3) {
|
|
|
|
qLog(Error) << "Weird songkick id";
|
2012-08-27 11:49:54 +02:00
|
|
|
emit Finished(id);
|
2012-05-30 22:30:02 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
FetchSongkickCalendar(split[2], id);
|
2012-05-30 01:31:27 +02:00
|
|
|
} catch (Echonest::ParseError& e) {
|
|
|
|
qLog(Error) << "Error parsing echonest reply:" << e.errorType() << e.what();
|
2012-08-27 11:49:54 +02:00
|
|
|
emit Finished(id);
|
2012-05-30 01:31:27 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void SongkickConcerts::FetchSongkickCalendar(const QString& artist_id, int id) {
|
|
|
|
QUrl url(QString(kSongkickArtistCalendarUrl).arg(artist_id));
|
2012-05-30 22:30:02 +02:00
|
|
|
qLog(Debug) << url;
|
|
|
|
QNetworkReply* reply = network_.get(QNetworkRequest(url));
|
|
|
|
NewClosure(reply, SIGNAL(finished()), this, SLOT(CalendarRequestFinished(QNetworkReply*, int)), reply, id);
|
2012-05-30 01:31:27 +02:00
|
|
|
}
|
|
|
|
|
2012-05-30 22:30:02 +02:00
|
|
|
void SongkickConcerts::CalendarRequestFinished(QNetworkReply* reply, int id) {
|
2012-06-04 14:06:43 +02:00
|
|
|
static const char* kStaticMapUrl =
|
|
|
|
"http://maps.googleapis.com/maps/api/staticmap"
|
|
|
|
"?key=AIzaSyDDJqmLOeE1mY_EBONhnQmdXbKtasgCtqg"
|
|
|
|
"&sensor=false"
|
|
|
|
"&size=100x100"
|
|
|
|
"&zoom=12"
|
|
|
|
"¢er=%1,%2"
|
|
|
|
"&markers=%1,%2";
|
|
|
|
|
|
|
|
QJson::Parser parser;
|
|
|
|
bool ok = false;
|
|
|
|
QVariant result = parser.parse(reply, &ok);
|
|
|
|
|
|
|
|
if (!ok) {
|
|
|
|
qLog(Error) << "Error parsing Songkick reply";
|
2012-08-27 11:49:54 +02:00
|
|
|
emit Finished(id);
|
2012-06-04 14:06:43 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
QString html;
|
|
|
|
QXmlStreamWriter writer(&html);
|
2012-06-04 14:40:08 +02:00
|
|
|
SongInfoTextView* text_view = new SongInfoTextView;
|
2012-06-04 14:06:43 +02:00
|
|
|
|
|
|
|
QVariantMap root = result.toMap();
|
|
|
|
QVariantMap results_page = root["resultsPage"].toMap();
|
|
|
|
QVariantMap results = results_page["results"].toMap();
|
|
|
|
QVariantList events = results["event"].toList();
|
2012-08-27 11:49:54 +02:00
|
|
|
|
|
|
|
if (events.isEmpty()) {
|
|
|
|
emit Finished(id);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2012-06-04 14:06:43 +02:00
|
|
|
foreach (const QVariant& v, events) {
|
|
|
|
QVariantMap event = v.toMap();
|
|
|
|
{
|
2012-06-20 14:12:37 +02:00
|
|
|
QString display_name = event["displayName"].toString();
|
|
|
|
QVariantMap venue = event["venue"].toMap();
|
|
|
|
const bool valid_latlng =
|
|
|
|
venue["lng"].isValid() && venue["lat"].isValid();
|
|
|
|
|
|
|
|
if (valid_latlng && latlng_.IsValid()) {
|
|
|
|
static const int kFilterDistanceMetres = 250 * 1e3; // 250km
|
|
|
|
Geolocator::LatLng latlng(
|
|
|
|
venue["lat"].toString(), venue["lng"].toString());
|
|
|
|
if (latlng_.IsValid() && latlng.IsValid()) {
|
|
|
|
int distance_metres = latlng_.Distance(latlng);
|
|
|
|
if (distance_metres > kFilterDistanceMetres) {
|
|
|
|
qLog(Debug) << "Filtered concert:"
|
|
|
|
<< display_name
|
|
|
|
<< "as too far away:"
|
|
|
|
<< distance_metres;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-06-04 14:06:43 +02:00
|
|
|
writer.writeStartElement("div");
|
|
|
|
{
|
|
|
|
writer.writeStartElement("a");
|
|
|
|
writer.writeAttribute("href", event["uri"].toString());
|
2012-06-20 14:12:37 +02:00
|
|
|
writer.writeCharacters(display_name);
|
2012-06-04 14:06:43 +02:00
|
|
|
writer.writeEndElement();
|
|
|
|
}
|
2012-06-20 14:12:37 +02:00
|
|
|
if (valid_latlng) {
|
2012-06-04 14:06:43 +02:00
|
|
|
writer.writeStartElement("img");
|
|
|
|
QString maps_url = QString(kStaticMapUrl).arg(
|
|
|
|
venue["lat"].toString(),
|
|
|
|
venue["lng"].toString());
|
|
|
|
writer.writeAttribute("src", maps_url);
|
|
|
|
writer.writeEndElement();
|
2012-06-04 14:40:08 +02:00
|
|
|
|
|
|
|
// QTextDocument does not support loading remote images, so we load
|
|
|
|
// them here and then inject them into the document later.
|
|
|
|
QNetworkRequest request(maps_url);
|
|
|
|
QNetworkReply* reply = network_.get(request);
|
|
|
|
NewClosure(reply, SIGNAL(finished()), this,
|
|
|
|
SLOT(InjectImage(QNetworkReply*, SongInfoTextView*)),
|
|
|
|
reply, text_view);
|
2012-06-04 14:06:43 +02:00
|
|
|
}
|
|
|
|
writer.writeEndElement();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
CollapsibleInfoPane::Data data;
|
|
|
|
data.type_ = CollapsibleInfoPane::Data::Type_Biography;
|
|
|
|
data.id_ = QString("songkick/%1").arg(id);
|
|
|
|
data.title_ = tr("Upcoming Concerts");
|
2012-06-04 14:40:08 +02:00
|
|
|
data.icon_ = QIcon(":providers/songkick.png");
|
2012-06-04 14:06:43 +02:00
|
|
|
|
|
|
|
text_view->SetHtml(html);
|
|
|
|
data.contents_ = text_view;
|
|
|
|
|
|
|
|
emit InfoReady(id, data);
|
|
|
|
emit Finished(id);
|
2012-05-30 01:31:27 +02:00
|
|
|
}
|
2012-06-04 14:40:08 +02:00
|
|
|
|
|
|
|
void SongkickConcerts::InjectImage(
|
|
|
|
QNetworkReply* reply, SongInfoTextView* text_view) {
|
2012-06-04 15:00:10 +02:00
|
|
|
reply->deleteLater();
|
2012-06-04 14:40:08 +02:00
|
|
|
QImage image;
|
|
|
|
image.load(reply, "png");
|
|
|
|
text_view->document()->addResource(
|
|
|
|
QTextDocument::ImageResource,
|
|
|
|
reply->request().url(),
|
|
|
|
QVariant(image));
|
|
|
|
}
|
2012-06-20 14:12:37 +02:00
|
|
|
|
|
|
|
void SongkickConcerts::GeolocateFinished(Geolocator::LatLng latlng) {
|
|
|
|
latlng_ = latlng;
|
|
|
|
}
|