386 lines
9.9 KiB
C++
386 lines
9.9 KiB
C++
/***************************************************************************
|
|
* Copyright (C) 2008-2009 by Andrzej Rybczak *
|
|
* electricityispower@gmail.com *
|
|
* *
|
|
* 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, write to the *
|
|
* Free Software Foundation, Inc., *
|
|
* 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA. *
|
|
***************************************************************************/
|
|
|
|
#include "info.h"
|
|
|
|
#ifdef HAVE_CURL_CURL_H
|
|
# include <fstream>
|
|
# ifdef WIN32
|
|
# include <io.h>
|
|
# else
|
|
# include <sys/stat.h>
|
|
# endif // WIN32
|
|
# include "curl/curl.h"
|
|
# include "helpers.h"
|
|
#endif
|
|
|
|
#include "browser.h"
|
|
#include "charset.h"
|
|
#include "global.h"
|
|
#include "media_library.h"
|
|
#include "playlist.h"
|
|
#include "playlist_editor.h"
|
|
#include "search_engine.h"
|
|
#include "status.h"
|
|
#include "tag_editor.h"
|
|
|
|
using namespace Global;
|
|
|
|
#ifdef HAVE_CURL_CURL_H
|
|
const std::string Info::Folder = home_path + HOME_FOLDER"artists";
|
|
bool Info::ArtistReady = 0;
|
|
|
|
#ifdef HAVE_PTHREAD_H
|
|
pthread_t *Info::Downloader = 0;
|
|
#endif // HAVE_PTHREAD_H
|
|
|
|
#endif // HAVE_CURL_CURL_H
|
|
|
|
Info *myInfo = new Info;
|
|
|
|
void Info::Init()
|
|
{
|
|
w = new Scrollpad(0, MainStartY, COLS, MainHeight, "", Config.main_color, brNone);
|
|
w->SetTimeout(ncmpcpp_window_timeout);
|
|
isInitialized = 1;
|
|
}
|
|
|
|
void Info::Resize()
|
|
{
|
|
w->Resize(COLS, MainHeight);
|
|
w->MoveTo(0, MainStartY);
|
|
hasToBeResized = 0;
|
|
}
|
|
|
|
std::basic_string<my_char_t> Info::Title()
|
|
{
|
|
return TO_WSTRING(itsTitle);
|
|
}
|
|
|
|
#if defined(HAVE_CURL_CURL_H) && defined(HAVE_PTHREAD_H)
|
|
void Info::Update()
|
|
{
|
|
if (!ArtistReady)
|
|
return;
|
|
|
|
pthread_join(*Downloader, NULL);
|
|
w->Flush();
|
|
delete Downloader;
|
|
Downloader = 0;
|
|
ArtistReady = 0;
|
|
}
|
|
#endif // HAVE_CURL_CURL_H && HAVE_PTHREAD_H
|
|
|
|
void Info::GetSong()
|
|
{
|
|
if (myScreen == this)
|
|
{
|
|
myOldScreen->SwitchTo();
|
|
}
|
|
else
|
|
{
|
|
if (!isInitialized)
|
|
Init();
|
|
|
|
MPD::Song *s = myScreen->CurrentSong();
|
|
|
|
if (!s)
|
|
return;
|
|
|
|
if (hasToBeResized)
|
|
Resize();
|
|
|
|
myOldScreen = myScreen;
|
|
myScreen = this;
|
|
RedrawHeader = 1;
|
|
itsTitle = "Song info";
|
|
w->Clear(0);
|
|
PrepareSong(*s);
|
|
w->Window::Clear();
|
|
w->Flush();
|
|
}
|
|
}
|
|
|
|
#ifdef HAVE_CURL_CURL_H
|
|
|
|
void Info::GetArtist()
|
|
{
|
|
if (myScreen == this)
|
|
{
|
|
myOldScreen->SwitchTo();
|
|
}
|
|
else
|
|
{
|
|
if (!isInitialized)
|
|
Init();
|
|
|
|
# ifdef HAVE_PTHREAD_H
|
|
if (Downloader && !ArtistReady)
|
|
{
|
|
ShowMessage("Artist's info is being downloaded...");
|
|
return;
|
|
}
|
|
else if (ArtistReady)
|
|
Update();
|
|
# endif // HAVE_PTHREAD_H
|
|
|
|
MPD::Song *s = myScreen->CurrentSong();
|
|
|
|
if (!s && myScreen->ActiveWindow() != myLibrary->Artists)
|
|
return;
|
|
|
|
itsArtist = !s ? myLibrary->Artists->Current() : s->GetArtist();
|
|
|
|
if (itsArtist.empty())
|
|
return;
|
|
|
|
if (hasToBeResized)
|
|
Resize();
|
|
myOldScreen = myScreen;
|
|
myScreen = this;
|
|
RedrawHeader = 1;
|
|
itsTitle = "Artist's info - " + itsArtist;
|
|
w->Clear();
|
|
static_cast<Window &>(*w) << "Fetching artist's info...";
|
|
w->Window::Refresh();
|
|
# ifdef HAVE_PTHREAD_H
|
|
if (!Downloader)
|
|
# endif // HAVE_PTHREAD_H
|
|
{
|
|
locale_to_utf(itsArtist);
|
|
|
|
std::string file = itsArtist + ".txt";
|
|
ToLower(file);
|
|
EscapeUnallowedChars(file);
|
|
|
|
itsFilenamePath = Folder + "/" + file;
|
|
|
|
mkdir(Folder.c_str()
|
|
# ifndef WIN32
|
|
, 0755
|
|
# endif // !WIN32
|
|
);
|
|
|
|
std::ifstream input(itsFilenamePath.c_str());
|
|
if (input.is_open())
|
|
{
|
|
bool first = 1;
|
|
std::string line;
|
|
while (getline(input, line))
|
|
{
|
|
if (!first)
|
|
*w << "\n";
|
|
utf_to_locale(line);
|
|
*w << line;
|
|
first = 0;
|
|
}
|
|
input.close();
|
|
w->SetFormatting(fmtBold, "\n\nSimilar artists:\n", fmtBoldEnd, 0);
|
|
w->SetFormatting(Config.color2, "\n * ", clEnd);
|
|
// below is used so format won't be removed using RemoveFormatting() by accident.
|
|
w->ForgetFormatting();
|
|
w->Flush();
|
|
}
|
|
else
|
|
{
|
|
# ifdef HAVE_PTHREAD_H
|
|
Downloader = new pthread_t;
|
|
pthread_create(Downloader, 0, PrepareArtist, this);
|
|
# else
|
|
PrepareArtist(this);
|
|
w->Flush();
|
|
# endif // HAVE_PTHREAD_H
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void *Info::PrepareArtist(void *screen_void_ptr)
|
|
{
|
|
Info *screen = static_cast<Info *>(screen_void_ptr);
|
|
|
|
char *c_artist = curl_easy_escape(0, screen->itsArtist.c_str(), screen->itsArtist.length());
|
|
std::string url = "http://ws.audioscrobbler.com/2.0/?method=artist.getinfo&artist=";
|
|
url += c_artist;
|
|
url += "&api_key=d94e5b6e26469a2d1ffae8ef20131b79";
|
|
|
|
std::string result;
|
|
CURLcode code;
|
|
|
|
pthread_mutex_lock(&CurlLock);
|
|
CURL *info = curl_easy_init();
|
|
curl_easy_setopt(info, CURLOPT_URL, url.c_str());
|
|
curl_easy_setopt(info, CURLOPT_WRITEFUNCTION, write_data);
|
|
curl_easy_setopt(info, CURLOPT_WRITEDATA, &result);
|
|
curl_easy_setopt(info, CURLOPT_CONNECTTIMEOUT, 10);
|
|
curl_easy_setopt(info, CURLOPT_NOSIGNAL, 1);
|
|
code = curl_easy_perform(info);
|
|
curl_easy_cleanup(info);
|
|
pthread_mutex_unlock(&CurlLock);
|
|
|
|
curl_free(c_artist);
|
|
|
|
if (code != CURLE_OK)
|
|
{
|
|
*screen->w << "Error while fetching artist's info: " << curl_easy_strerror(code);
|
|
ArtistReady = 1;
|
|
pthread_exit(0);
|
|
}
|
|
|
|
size_t a, b;
|
|
bool save = 1;
|
|
|
|
a = result.find("status=\"failed\"");
|
|
|
|
if (a != std::string::npos)
|
|
{
|
|
EscapeHtml(result);
|
|
*screen->w << "Last.fm returned an error message: " << result;
|
|
ArtistReady = 1;
|
|
pthread_exit(0);
|
|
}
|
|
|
|
std::vector<std::string> similar;
|
|
for (size_t i = result.find("<name>"); i != std::string::npos; i = result.find("<name>"))
|
|
{
|
|
result[i] = '.';
|
|
size_t j = result.find("</name>");
|
|
result[j] = '.';
|
|
i += 6;
|
|
similar.push_back(result.substr(i, j-i));
|
|
EscapeHtml(similar.back());
|
|
}
|
|
std::vector<std::string> urls;
|
|
for (size_t i = result.find("<url>"); i != std::string::npos; i = result.find("<url>"))
|
|
{
|
|
result[i] = '.';
|
|
size_t j = result.find("</url>");
|
|
result[j] = '.';
|
|
i += 5;
|
|
urls.push_back(result.substr(i, j-i));
|
|
}
|
|
|
|
a = result.find("<content>")+9;
|
|
b = result.find("</content>");
|
|
|
|
if (a == b)
|
|
{
|
|
result = "No description available for this artist.";
|
|
save = 0;
|
|
}
|
|
else
|
|
{
|
|
a += 9; // for <![CDATA[
|
|
b -= 3; // for ]]>
|
|
result = result.substr(a, b-a);
|
|
}
|
|
|
|
EscapeHtml(result);
|
|
Trim(result);
|
|
|
|
std::ostringstream filebuffer;
|
|
if (save)
|
|
filebuffer << result;
|
|
utf_to_locale(result);
|
|
*screen->w << result;
|
|
|
|
if (save)
|
|
filebuffer << "\n\nSimilar artists:\n";
|
|
*screen->w << fmtBold << "\n\nSimilar artists:\n" << fmtBoldEnd;
|
|
for (size_t i = 1; i < similar.size(); ++i)
|
|
{
|
|
if (save)
|
|
filebuffer << "\n * " << similar[i] << " (" << urls[i] << ")";
|
|
utf_to_locale(similar[i]);
|
|
utf_to_locale(urls[i]);
|
|
*screen->w << "\n" << Config.color2 << " * " << clEnd << similar[i] << " (" << urls[i] << ")";
|
|
}
|
|
|
|
if (save)
|
|
filebuffer << "\n\n" << urls.front();
|
|
utf_to_locale(urls.front());
|
|
*screen->w << "\n\n" << urls.front();
|
|
|
|
if (save)
|
|
{
|
|
std::ofstream output(screen->itsFilenamePath.c_str());
|
|
if (output.is_open())
|
|
{
|
|
output << filebuffer.str();
|
|
output.close();
|
|
}
|
|
}
|
|
ArtistReady = 1;
|
|
pthread_exit(0);
|
|
}
|
|
#endif // HVAE_CURL_CURL_H
|
|
|
|
void Info::PrepareSong(MPD::Song &s)
|
|
{
|
|
# ifdef HAVE_TAGLIB_H
|
|
std::string path_to_file;
|
|
if (s.isFromDB())
|
|
path_to_file += Config.mpd_music_dir;
|
|
path_to_file += s.GetFile();
|
|
TagLib::FileRef f(path_to_file.c_str());
|
|
if (!f.isNull())
|
|
s.SetComment(f.tag()->comment().to8Bit(1));
|
|
# endif // HAVE_TAGLIB_H
|
|
|
|
*w << fmtBold << Config.color1 << "Filename: " << fmtBoldEnd << Config.color2 << s.GetName() << "\n" << clEnd;
|
|
*w << fmtBold << "Directory: " << fmtBoldEnd << Config.color2 << ShowTag(s.GetDirectory()) << "\n\n" << clEnd;
|
|
*w << fmtBold << "Length: " << fmtBoldEnd << Config.color2 << s.GetLength() << "\n" << clEnd;
|
|
# ifdef HAVE_TAGLIB_H
|
|
if (!f.isNull())
|
|
{
|
|
*w << fmtBold << "Bitrate: " << fmtBoldEnd << Config.color2 << f.audioProperties()->bitrate() << " kbps\n" << clEnd;
|
|
*w << fmtBold << "Sample rate: " << fmtBoldEnd << Config.color2 << f.audioProperties()->sampleRate() << " Hz\n" << clEnd;
|
|
*w << fmtBold << "Channels: " << fmtBoldEnd << Config.color2 << (f.audioProperties()->channels() == 1 ? "Mono" : "Stereo") << "\n" << clDefault;
|
|
}
|
|
# endif // HAVE_TAGLIB_H
|
|
*w << clDefault;
|
|
|
|
*w << fmtBold << "\nTitle: " << fmtBoldEnd << ShowTag(s.GetTitle());
|
|
*w << fmtBold << "\nArtist: " << fmtBoldEnd << ShowTag(s.GetArtist());
|
|
*w << fmtBold << "\nAlbum: " << fmtBoldEnd << ShowTag(s.GetAlbum());
|
|
*w << fmtBold << "\nYear: " << fmtBoldEnd << ShowTag(s.GetDate());
|
|
*w << fmtBold << "\nTrack: " << fmtBoldEnd << ShowTag(s.GetTrack());
|
|
*w << fmtBold << "\nGenre: " << fmtBoldEnd << ShowTag(s.GetGenre());
|
|
*w << fmtBold << "\nComposer: " << fmtBoldEnd << ShowTag(s.GetComposer());
|
|
*w << fmtBold << "\nPerformer: " << fmtBoldEnd << ShowTag(s.GetPerformer());
|
|
*w << fmtBold << "\nDisc: " << fmtBoldEnd << ShowTag(s.GetDisc());
|
|
*w << fmtBold << "\nComment: " << fmtBoldEnd << ShowTag(s.GetComment());
|
|
}
|
|
|
|
basic_buffer<my_char_t> Info::ShowTag(const std::string &tag)
|
|
{
|
|
# ifdef _UTF8
|
|
WBuffer result;
|
|
if (tag.empty())
|
|
result << Config.empty_tags_color << ToWString(Config.empty_tag) << clEnd;
|
|
else
|
|
result << ToWString(tag);
|
|
return result;
|
|
# else
|
|
return ::ShowTag(tag);
|
|
# endif
|
|
}
|
|
|