Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

lyricwiki-qt: Add support for chartlyrics.com. Closes: #1217 #149

Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion configure.ac
Original file line number Diff line number Diff line change
Expand Up @@ -859,7 +859,7 @@ if test "x$USE_QT" = "xyes" ; then
echo " Winamp Classic Interface: yes"
echo " Album Art: yes"
echo " Blur Scope: yes"
echo " LyricWiki viewer: yes"
echo " Lyrics Viewer: yes"
echo " OpenGL Spectrum Analyzer: $have_qtglspectrum"
echo " Playlist Manager: yes"
echo " Search Tool: yes"
Expand Down
253 changes: 236 additions & 17 deletions src/lyricwiki-qt/lyricwiki.cc
Original file line number Diff line number Diff line change
Expand Up @@ -32,13 +32,9 @@
#include <QMenu>
#include <QRegularExpression>
#include <QTextCursor>
#include <QTextDocument>
#include <QTextEdit>

#include <libxml/parser.h>
#include <libxml/tree.h>
#include <libxml/HTMLparser.h>
#include <libxml/xpath.h>

#define AUD_GLIB_INTEGRATION
#include <libaudcore/drct.h>
Expand All @@ -63,7 +59,8 @@ struct LyricsState {
Embedded,
Local,
LyricWiki,
LyricsOVH
LyricsOVH,
ChartLyrics
} source = None;

bool error = false;
Expand Down Expand Up @@ -116,7 +113,8 @@ const char * const LyricWikiQt::defaults[] = {

static const ComboItem remote_sources[] = {
ComboItem(N_("Nowhere"), "nowhere"),
ComboItem(N_("lyrics.ovh"), "lyrics.ovh")
ComboItem("chartlyrics.com", "chartlyrics.com"),
ComboItem("lyrics.ovh", "lyrics.ovh")
};

static const PreferencesWidget truncate_elements[] = {
Expand Down Expand Up @@ -165,7 +163,7 @@ static void update_lyrics_window_error (const char * message);
static void update_lyrics_window_notfound (LyricsState state);

// LyricProvider encapsulates an entire strategy for fetching lyrics,
// for example from LyricWiki or local storage.
// for example from chartlyrics.com, lyrics.ovh or local storage.
class LyricProvider {
public:
virtual bool match (LyricsState state) = 0;
Expand All @@ -179,9 +177,10 @@ class FileProvider : public LyricProvider {
public:
FileProvider() {};

bool match (LyricsState state);
void fetch (LyricsState state);
String edit_uri (LyricsState state) { return String (); }
bool match (LyricsState state) override;
void fetch (LyricsState state) override;
String edit_uri (LyricsState state) override { return String (); }

void save (LyricsState state);
void cache (LyricsState state);
void cache_fetch (LyricsState state);
Expand Down Expand Up @@ -335,15 +334,218 @@ void FileProvider::save (LyricsState state)
VFSFile::write_file (path, state.lyrics, strlen (state.lyrics));
}

// ChartLyricsProvider provides a strategy for fetching lyrics using the API
// from chartlyrics.com. It uses the two-step approach since the endpoint
// "SearchLyricDirect" may sometimes return incorrect data. One example is
// "Metallica - Unforgiven II" which leads to the lyrics of "Unforgiven".
class ChartLyricsProvider : public LyricProvider
{
public:
ChartLyricsProvider () {};

bool match (LyricsState state) override;
void fetch (LyricsState state) override;
String edit_uri (LyricsState state) override { return m_lyric_url; }

private:
String match_uri (LyricsState state);
String fetch_uri (LyricsState state);

void reset_lyric_metadata ();
bool has_match (LyricsState state, xmlNodePtr node);

int m_lyric_id = -1;
String m_lyric_checksum, m_lyric_url, m_lyrics;

const char * m_base_url = "http://api.chartlyrics.com/apiv1.asmx";
};

void ChartLyricsProvider::reset_lyric_metadata ()
{
m_lyric_id = -1;
m_lyric_checksum = String ();
m_lyric_url = String ();
m_lyrics = String ();
}

String ChartLyricsProvider::match_uri (LyricsState state)
{
auto artist = str_copy (state.artist);
artist = str_encode_percent (artist, -1);

auto title = str_copy (state.title);
title = str_encode_percent (title, -1);

return String (str_concat ({m_base_url, "/SearchLyric?artist=", artist, "&song=", title}));
}

bool ChartLyricsProvider::has_match (LyricsState state, xmlNodePtr node)
{
bool match_found = false;
String lyric_id, checksum, url, artist, title;

for (xmlNodePtr cur_node = node->xmlChildrenNode; cur_node; cur_node = cur_node->next)
{
if (cur_node->type != XML_ELEMENT_NODE)
continue;

xmlChar * content = xmlNodeGetContent (cur_node);

if (xmlStrEqual (cur_node->name, (xmlChar *) "LyricId"))
lyric_id = String ((const char *) content);
else if (xmlStrEqual (cur_node->name, (xmlChar *) "LyricChecksum"))
checksum = String ((const char *) content);
else if (xmlStrEqual (cur_node->name, (xmlChar *) "SongUrl"))
url = String ((const char *) content);
else if (xmlStrEqual (cur_node->name, (xmlChar *) "Artist"))
artist = String ((const char *) content);
else if (xmlStrEqual (cur_node->name, (xmlChar *) "Song"))
title = String ((const char *) content);

xmlFree (content);
}

if (lyric_id && checksum && artist && title) // url is optional
{
int id = str_to_int (lyric_id);

if (id > 0 &&
! strcmp_nocase (artist, state.artist) &&
! strcmp_nocase (title, state.title))
{
m_lyric_id = id;
m_lyric_checksum = checksum;
m_lyric_url = url;

match_found = true;
}
}

return match_found;
}

bool ChartLyricsProvider::match (LyricsState state)
{
reset_lyric_metadata ();

auto handle_result_cb = [=] (const char * uri, const Index<char> & buf) {
if (! buf.len ())
{
update_lyrics_window_error (str_printf (_("Unable to fetch %s"), uri));
return;
}

xmlDocPtr doc = xmlReadMemory (buf.begin (), buf.len (), nullptr, nullptr, 0);
if (! doc)
{
update_lyrics_window_error (str_printf (_("Unable to parse %s"), uri));
return;
}

xmlNodePtr root = xmlDocGetRootElement (doc);

for (xmlNodePtr cur_node = root->xmlChildrenNode; cur_node; cur_node = cur_node->next)
{
if (cur_node->type != XML_ELEMENT_NODE)
continue;

if (has_match (state, cur_node))
break;
}

xmlFreeDoc (doc);

fetch (state);
};

vfs_async_file_get_contents (match_uri (state), handle_result_cb);
update_lyrics_window_message (state, _("Looking for lyrics ..."));

return true;
}

String ChartLyricsProvider::fetch_uri (LyricsState state)
{
if (m_lyric_id <= 0 || ! m_lyric_checksum)
return String ();

auto id = int_to_str (m_lyric_id);
auto checksum = str_copy (m_lyric_checksum);
checksum = str_encode_percent (checksum, -1);

return String (str_concat ({m_base_url, "/GetLyric?lyricId=", id, "&lyricCheckSum=", checksum}));
}

void ChartLyricsProvider::fetch (LyricsState state)
{
String _fetch_uri = fetch_uri (state);
if (! _fetch_uri)
{
update_lyrics_window_notfound (state);
return;
}

auto handle_result_cb = [=] (const char * uri, const Index<char> & buf) {
if (! buf.len ())
{
update_lyrics_window_error (str_printf (_("Unable to fetch %s"), uri));
return;
}

xmlDocPtr doc = xmlReadMemory (buf.begin (), buf.len (), nullptr, nullptr, 0);
if (! doc)
{
update_lyrics_window_error (str_printf (_("Unable to parse %s"), uri));
return;
}

xmlNodePtr root = xmlDocGetRootElement (doc);

for (xmlNodePtr cur_node = root->xmlChildrenNode; cur_node; cur_node = cur_node->next)
{
if (cur_node->type == XML_ELEMENT_NODE &&
xmlStrEqual (cur_node->name, (xmlChar *) "Lyric"))
{
xmlChar * content = xmlNodeGetContent (cur_node);
m_lyrics = String ((const char *) content);
xmlFree (content);
break;
}
}

xmlFreeDoc (doc);

LyricsState new_state = g_state;
new_state.lyrics = String ();

if (! m_lyrics || ! m_lyrics[0])
{
update_lyrics_window_notfound (new_state);
return;
}

new_state.lyrics = m_lyrics;
new_state.source = LyricsState::Source::ChartLyrics;

update_lyrics_window (new_state.title, new_state.artist, new_state.lyrics);
persist_state (new_state);
};

vfs_async_file_get_contents (_fetch_uri, handle_result_cb);
update_lyrics_window_message (state, _("Looking for lyrics ..."));
}

static ChartLyricsProvider chart_lyrics_provider;

// LyricsOVHProvider provides a strategy for fetching lyrics using the
// lyrics.ovh search engine.
class LyricsOVHProvider : public LyricProvider {
public:
LyricsOVHProvider() {};

bool match (LyricsState state);
void fetch (LyricsState state);
String edit_uri (LyricsState state) { return String (); }
bool match (LyricsState state) override;
void fetch (LyricsState state) override;
String edit_uri (LyricsState state) override { return String (); }
};

bool LyricsOVHProvider::match (LyricsState state)
Expand Down Expand Up @@ -413,6 +615,9 @@ static LyricProvider * remote_source ()
{
auto source = aud_get_str ("lyricwiki", "remote-source");

if (! strcmp (source, "chartlyrics.com"))
return &chart_lyrics_provider;

if (! strcmp (source, "lyrics.ovh"))
return &lyrics_ovh_provider;

Expand Down Expand Up @@ -568,11 +773,26 @@ void TextEdit::contextMenuEvent (QContextMenuEvent * event)
if (! g_state.artist || ! g_state.title)
return QTextEdit::contextMenuEvent (event);

LyricProvider * remote_provider = remote_source ();

QMenu * menu = createStandardContextMenu ();
menu->addSeparator ();

if (g_state.lyrics && g_state.source != LyricsState::Source::Local && ! g_state.error)
{
if (remote_provider)
{
String edit_uri = remote_provider->edit_uri (g_state);

if (edit_uri && edit_uri[0])
{
QAction * edit = menu->addAction (_("Edit Lyrics ..."));
QObject::connect (edit, & QAction::triggered, [edit_uri] () {
QDesktopServices::openUrl (QUrl ((const char *) edit_uri));
});
}
}

QAction * save = menu->addAction (_("Save Locally"));
QObject::connect (save, & QAction::triggered, [] () {
file_provider.save (g_state);
Expand All @@ -582,10 +802,9 @@ void TextEdit::contextMenuEvent (QContextMenuEvent * event)
if (g_state.source == LyricsState::Source::Local || g_state.error)
{
QAction * refresh = menu->addAction (_("Refresh"));
QObject::connect (refresh, & QAction::triggered, [] () {
auto rsrc = remote_source ();
if (rsrc)
rsrc->match (g_state);
QObject::connect (refresh, & QAction::triggered, [remote_provider] () {
if (remote_provider)
remote_provider->match (g_state);
});
}

Expand Down