make media library show more songs (with incomplete tags etc.)
This commit is contained in:
@@ -57,13 +57,13 @@ void MediaLibrary::Init()
|
|||||||
Artists->SetSelectSuffix(&Config.selected_item_suffix);
|
Artists->SetSelectSuffix(&Config.selected_item_suffix);
|
||||||
Artists->SetItemDisplayer(Display::Generic);
|
Artists->SetItemDisplayer(Display::Generic);
|
||||||
|
|
||||||
Albums = new Menu< std::pair<std::string, SearchConstraints> >(itsMiddleColStartX, MainStartY, itsMiddleColWidth, MainHeight, "Albums", Config.main_color, brNone);
|
Albums = new Menu<SearchConstraints>(itsMiddleColStartX, MainStartY, itsMiddleColWidth, MainHeight, "Albums", Config.main_color, brNone);
|
||||||
Albums->HighlightColor(Config.main_highlight_color);
|
Albums->HighlightColor(Config.main_highlight_color);
|
||||||
Albums->CyclicScrolling(Config.use_cyclic_scrolling);
|
Albums->CyclicScrolling(Config.use_cyclic_scrolling);
|
||||||
Albums->SetSelectPrefix(&Config.selected_item_prefix);
|
Albums->SetSelectPrefix(&Config.selected_item_prefix);
|
||||||
Albums->SetSelectSuffix(&Config.selected_item_suffix);
|
Albums->SetSelectSuffix(&Config.selected_item_suffix);
|
||||||
Albums->SetItemDisplayer(Display::Pairs);
|
Albums->SetItemDisplayer(DisplayAlbums);
|
||||||
Albums->SetGetStringFunction(StringPairToString);
|
Albums->SetGetStringFunction(AlbumToString);
|
||||||
|
|
||||||
Songs = new Menu<Song>(itsRightColStartX, MainStartY, itsRightColWidth, MainHeight, "Songs", Config.main_color, brNone);
|
Songs = new Menu<Song>(itsRightColStartX, MainStartY, itsRightColWidth, MainHeight, "Songs", Config.main_color, brNone);
|
||||||
Songs->HighlightColor(Config.main_highlight_color);
|
Songs->HighlightColor(Config.main_highlight_color);
|
||||||
@@ -188,48 +188,27 @@ void MediaLibrary::Update()
|
|||||||
Albums->Reset();
|
Albums->Reset();
|
||||||
TagList list;
|
TagList list;
|
||||||
locale_to_utf(Artists->Current());
|
locale_to_utf(Artists->Current());
|
||||||
if (Config.media_lib_primary_tag == MPD_TAG_ARTIST)
|
Mpd.StartFieldSearch(MPD_TAG_ALBUM);
|
||||||
Mpd.GetAlbums(Artists->Current(), list);
|
Mpd.AddSearch(Config.media_lib_primary_tag, Artists->Current());
|
||||||
else
|
Mpd.CommitSearch(list);
|
||||||
{
|
|
||||||
Mpd.StartFieldSearch(MPD_TAG_ALBUM);
|
|
||||||
Mpd.AddSearch(Config.media_lib_primary_tag, Artists->Current());
|
|
||||||
Mpd.CommitSearch(list);
|
|
||||||
}
|
|
||||||
|
|
||||||
// <mpd-0.14 doesn't support searching for empty tag
|
for (TagList::iterator it = list.begin(); it != list.end(); ++it)
|
||||||
if (Mpd.Version() > 13)
|
|
||||||
{
|
|
||||||
SongList noalbum_list;
|
|
||||||
Mpd.StartSearch(1);
|
|
||||||
Mpd.AddSearch(Config.media_lib_primary_tag, Artists->Current());
|
|
||||||
Mpd.AddSearch(MPD_TAG_ALBUM, "");
|
|
||||||
Mpd.CommitSearch(noalbum_list);
|
|
||||||
if (!noalbum_list.empty())
|
|
||||||
Albums->AddOption(std::make_pair("<no album>", SearchConstraints("", "")));
|
|
||||||
FreeSongList(noalbum_list);
|
|
||||||
}
|
|
||||||
|
|
||||||
for (TagList::const_iterator it = list.begin(); it != list.end(); ++it)
|
|
||||||
{
|
{
|
||||||
TagList l;
|
TagList l;
|
||||||
Mpd.StartFieldSearch(MPD_TAG_DATE);
|
Mpd.StartFieldSearch(MPD_TAG_DATE);
|
||||||
Mpd.AddSearch(Config.media_lib_primary_tag, Artists->Current());
|
Mpd.AddSearch(Config.media_lib_primary_tag, Artists->Current());
|
||||||
Mpd.AddSearch(MPD_TAG_ALBUM, *it);
|
Mpd.AddSearch(MPD_TAG_ALBUM, *it);
|
||||||
Mpd.CommitSearch(l);
|
Mpd.CommitSearch(l);
|
||||||
if (l.empty())
|
utf_to_locale(*it);
|
||||||
|
for (TagList::iterator j = l.begin(); j != l.end(); ++j)
|
||||||
{
|
{
|
||||||
Albums->AddOption(std::make_pair(*it, SearchConstraints(*it, "")));
|
utf_to_locale(*j);
|
||||||
continue;
|
Albums->AddOption(SearchConstraints(*it, *j));
|
||||||
}
|
}
|
||||||
for (TagList::const_iterator j = l.begin(); j != l.end(); ++j)
|
|
||||||
Albums->AddOption(std::make_pair("(" + *j + ") " + *it, SearchConstraints(*it, *j)));
|
|
||||||
}
|
}
|
||||||
utf_to_locale(Artists->Current());
|
utf_to_locale(Artists->Current());
|
||||||
for (size_t i = 0; i < Albums->Size(); ++i)
|
|
||||||
utf_to_locale((*Albums)[i].first);
|
|
||||||
if (!Albums->Empty())
|
if (!Albums->Empty())
|
||||||
Albums->Sort<CaseInsensitiveSorting>((*Albums)[0].first == "<no album>");
|
Albums->Sort<SearchConstraintsSorting>();
|
||||||
Albums->Refresh();
|
Albums->Refresh();
|
||||||
}
|
}
|
||||||
else if (hasTwoColumns && Albums->Empty())
|
else if (hasTwoColumns && Albums->Empty())
|
||||||
@@ -239,13 +218,15 @@ void MediaLibrary::Update()
|
|||||||
*Albums << XY(0, 0) << "Fetching albums...";
|
*Albums << XY(0, 0) << "Fetching albums...";
|
||||||
Albums->Window::Refresh();
|
Albums->Window::Refresh();
|
||||||
Mpd.GetList(artists, Config.media_lib_primary_tag);
|
Mpd.GetList(artists, Config.media_lib_primary_tag);
|
||||||
for (TagList::const_iterator i = artists.begin(); i != artists.end(); ++i)
|
for (TagList::iterator i = artists.begin(); i != artists.end(); ++i)
|
||||||
{
|
{
|
||||||
|
if (i->empty())
|
||||||
|
continue;
|
||||||
TagList albums;
|
TagList albums;
|
||||||
Mpd.StartFieldSearch(MPD_TAG_ALBUM);
|
Mpd.StartFieldSearch(MPD_TAG_ALBUM);
|
||||||
Mpd.AddSearch(Config.media_lib_primary_tag, *i);
|
Mpd.AddSearch(Config.media_lib_primary_tag, *i);
|
||||||
Mpd.CommitSearch(albums);
|
Mpd.CommitSearch(albums);
|
||||||
for (TagList::const_iterator j = albums.begin(); j != albums.end(); ++j)
|
for (TagList::iterator j = albums.begin(); j != albums.end(); ++j)
|
||||||
{
|
{
|
||||||
if (Config.media_lib_primary_tag != MPD_TAG_DATE)
|
if (Config.media_lib_primary_tag != MPD_TAG_DATE)
|
||||||
{
|
{
|
||||||
@@ -254,24 +235,24 @@ void MediaLibrary::Update()
|
|||||||
Mpd.AddSearch(Config.media_lib_primary_tag, *i);
|
Mpd.AddSearch(Config.media_lib_primary_tag, *i);
|
||||||
Mpd.AddSearch(MPD_TAG_ALBUM, *j);
|
Mpd.AddSearch(MPD_TAG_ALBUM, *j);
|
||||||
Mpd.CommitSearch(years);
|
Mpd.CommitSearch(years);
|
||||||
if (!years.empty())
|
utf_to_locale(*i);
|
||||||
|
utf_to_locale(*j);
|
||||||
|
for (TagList::iterator k = years.begin(); k != years.end(); ++k)
|
||||||
{
|
{
|
||||||
for (TagList::const_iterator k = years.begin(); k != years.end(); ++k)
|
utf_to_locale(*k);
|
||||||
{
|
Albums->AddOption(SearchConstraints(*i, *j, *k));
|
||||||
Albums->AddOption(std::make_pair(*i + " - (" + *k + ") " + *j, SearchConstraints(*i, *j, *k)));
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
else
|
|
||||||
Albums->AddOption(std::make_pair(*i + " - " + *j, SearchConstraints(*i, *j, "")));
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
Albums->AddOption(std::make_pair(*i + " - " + *j, SearchConstraints(*i, *j, *i)));
|
{
|
||||||
|
utf_to_locale(*i);
|
||||||
|
utf_to_locale(*j);
|
||||||
|
Albums->AddOption(SearchConstraints(*i, *j, *i));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
for (size_t i = 0; i < Albums->Size(); ++i)
|
|
||||||
utf_to_locale((*Albums)[i].first);
|
|
||||||
if (!Albums->Empty())
|
if (!Albums->Empty())
|
||||||
Albums->Sort<CaseInsensitiveSorting>();
|
Albums->Sort<SearchConstraintsSorting>();
|
||||||
Albums->Refresh();
|
Albums->Refresh();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -288,7 +269,7 @@ void MediaLibrary::Update()
|
|||||||
SongList list;
|
SongList list;
|
||||||
|
|
||||||
Mpd.StartSearch(1);
|
Mpd.StartSearch(1);
|
||||||
Mpd.AddSearch(Config.media_lib_primary_tag, hasTwoColumns ? Albums->Current().second.Artist : locale_to_utf_cpy(Artists->Current()));
|
Mpd.AddSearch(Config.media_lib_primary_tag, locale_to_utf_cpy(hasTwoColumns ? Albums->Current().Artist : Artists->Current()));
|
||||||
if (Albums->Empty()) // left for compatibility with <mpd-0.14
|
if (Albums->Empty()) // left for compatibility with <mpd-0.14
|
||||||
{
|
{
|
||||||
*Albums << XY(0, 0) << "No albums found.";
|
*Albums << XY(0, 0) << "No albums found.";
|
||||||
@@ -296,9 +277,8 @@ void MediaLibrary::Update()
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
Mpd.AddSearch(MPD_TAG_ALBUM, Albums->Current().second.Album);
|
Mpd.AddSearch(MPD_TAG_ALBUM, locale_to_utf_cpy(Albums->Current().Album));
|
||||||
if (!Albums->Current().second.Album.empty()) // for <no album>
|
Mpd.AddSearch(MPD_TAG_DATE, locale_to_utf_cpy(Albums->Current().Year));
|
||||||
Mpd.AddSearch(MPD_TAG_DATE, Albums->Current().second.Year);
|
|
||||||
}
|
}
|
||||||
Mpd.CommitSearch(list);
|
Mpd.CommitSearch(list);
|
||||||
|
|
||||||
@@ -473,11 +453,10 @@ void MediaLibrary::GetSelectedSongs(MPD::SongList &v)
|
|||||||
MPD::SongList list;
|
MPD::SongList list;
|
||||||
Mpd.StartSearch(1);
|
Mpd.StartSearch(1);
|
||||||
Mpd.AddSearch(Config.media_lib_primary_tag, hasTwoColumns
|
Mpd.AddSearch(Config.media_lib_primary_tag, hasTwoColumns
|
||||||
? Albums->at(*it).second.Artist
|
? Albums->at(*it).Artist
|
||||||
: locale_to_utf_cpy(Artists->Current()));
|
: locale_to_utf_cpy(Artists->Current()));
|
||||||
Mpd.AddSearch(MPD_TAG_ALBUM, Albums->at(*it).second.Album);
|
Mpd.AddSearch(MPD_TAG_ALBUM, Albums->at(*it).Album);
|
||||||
if (!Albums->at(*it).second.Album.empty()) // for <no album>
|
Mpd.AddSearch(MPD_TAG_DATE, Albums->at(*it).Year);
|
||||||
Mpd.AddSearch(MPD_TAG_DATE, Albums->at(*it).second.Year);
|
|
||||||
Mpd.CommitSearch(list);
|
Mpd.CommitSearch(list);
|
||||||
for (MPD::SongList::const_iterator sIt = list.begin(); sIt != list.end(); ++sIt)
|
for (MPD::SongList::const_iterator sIt = list.begin(); sIt != list.end(); ++sIt)
|
||||||
v.push_back(new MPD::Song(**sIt));
|
v.push_back(new MPD::Song(**sIt));
|
||||||
@@ -602,15 +581,15 @@ void MediaLibrary::LocateSong(const MPD::Song &s)
|
|||||||
|
|
||||||
std::string album = s.GetAlbum();
|
std::string album = s.GetAlbum();
|
||||||
std::string date = s.GetDate();
|
std::string date = s.GetDate();
|
||||||
if ((hasTwoColumns && Albums->Current().second.Artist != primary_tag)
|
if ((hasTwoColumns && Albums->Current().Artist != primary_tag)
|
||||||
|| album != Albums->Current().second.Album
|
|| album != Albums->Current().Album
|
||||||
|| date != Albums->Current().second.Year)
|
|| date != Albums->Current().Year)
|
||||||
{
|
{
|
||||||
for (size_t i = 0; i < Albums->Size(); ++i)
|
for (size_t i = 0; i < Albums->Size(); ++i)
|
||||||
{
|
{
|
||||||
if ((!hasTwoColumns || (*Albums)[i].second.Artist == primary_tag)
|
if ((!hasTwoColumns || (*Albums)[i].Artist == primary_tag)
|
||||||
&& album == (*Albums)[i].second.Album
|
&& album == (*Albums)[i].Album
|
||||||
&& date == (*Albums)[i].second.Year)
|
&& date == (*Albums)[i].Year)
|
||||||
{
|
{
|
||||||
Albums->Highlight(i);
|
Albums->Highlight(i);
|
||||||
Songs->Clear();
|
Songs->Clear();
|
||||||
@@ -660,7 +639,7 @@ void MediaLibrary::AddToPlaylist(bool add_n_play)
|
|||||||
ShowMessage("Adding songs of %s \"%s\"", tag_type.c_str(), Artists->Current().c_str());
|
ShowMessage("Adding songs of %s \"%s\"", tag_type.c_str(), Artists->Current().c_str());
|
||||||
}
|
}
|
||||||
else if (w == Albums)
|
else if (w == Albums)
|
||||||
ShowMessage("Adding songs from album \"%s\"", Albums->Current().second.Album.c_str());
|
ShowMessage("Adding songs from album \"%s\"", Albums->Current().Album.c_str());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -682,6 +661,36 @@ std::string MediaLibrary::SongToString(const MPD::Song &s, void *)
|
|||||||
return s.toString(Config.song_library_format);
|
return s.toString(Config.song_library_format);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
std::string MediaLibrary::AlbumToString(const SearchConstraints &sc, void *)
|
||||||
|
{
|
||||||
|
std::string result;
|
||||||
|
if (!sc.Artist.empty())
|
||||||
|
(result += sc.Artist) += " - ";
|
||||||
|
if (!sc.Year.empty())
|
||||||
|
((result += "(") += sc.Year) += ") ";
|
||||||
|
result += sc.Album.empty() ? "<no album>" : sc.Album;
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
void MediaLibrary::DisplayAlbums(const SearchConstraints &sc, void *, Menu<SearchConstraints> *menu)
|
||||||
|
{
|
||||||
|
*menu << AlbumToString(sc, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool MediaLibrary::SearchConstraintsSorting::operator()(const SearchConstraints &a, const SearchConstraints &b) const
|
||||||
|
{
|
||||||
|
int result;
|
||||||
|
CaseInsensitiveStringComparison cmp;
|
||||||
|
if (!a.Artist.empty() || b.Artist.empty())
|
||||||
|
{
|
||||||
|
result = cmp(a.Artist, b.Artist);
|
||||||
|
if (result != 0)
|
||||||
|
return result < 0;
|
||||||
|
}
|
||||||
|
result = cmp(a.Year, b.Year);
|
||||||
|
return (result == 0 ? cmp(a.Album, b.Album) : result) < 0;
|
||||||
|
}
|
||||||
|
|
||||||
bool MediaLibrary::SortSongsByYear(Song *a, Song *b)
|
bool MediaLibrary::SortSongsByYear(Song *a, Song *b)
|
||||||
{
|
{
|
||||||
return a->GetDate() < b->GetDate();
|
return a->GetDate() < b->GetDate();
|
||||||
|
|||||||
@@ -36,6 +36,11 @@ class MediaLibrary : public Screen<Window>
|
|||||||
std::string Year;
|
std::string Year;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
struct SearchConstraintsSorting
|
||||||
|
{
|
||||||
|
bool operator()(const SearchConstraints &a, const SearchConstraints &b) const;
|
||||||
|
};
|
||||||
|
|
||||||
public:
|
public:
|
||||||
virtual void SwitchTo();
|
virtual void SwitchTo();
|
||||||
virtual void Resize();
|
virtual void Resize();
|
||||||
@@ -67,7 +72,7 @@ class MediaLibrary : public Screen<Window>
|
|||||||
void LocateSong(const MPD::Song &);
|
void LocateSong(const MPD::Song &);
|
||||||
|
|
||||||
Menu<std::string> *Artists;
|
Menu<std::string> *Artists;
|
||||||
Menu< std::pair<std::string, SearchConstraints> > *Albums;
|
Menu<SearchConstraints> *Albums;
|
||||||
Menu<MPD::Song> *Songs;
|
Menu<MPD::Song> *Songs;
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
@@ -78,6 +83,9 @@ class MediaLibrary : public Screen<Window>
|
|||||||
|
|
||||||
static std::string SongToString(const MPD::Song &s, void *);
|
static std::string SongToString(const MPD::Song &s, void *);
|
||||||
|
|
||||||
|
static std::string AlbumToString(const SearchConstraints &, void *);
|
||||||
|
static void DisplayAlbums(const SearchConstraints &, void *, Menu<SearchConstraints> *);
|
||||||
|
|
||||||
static bool SortSongsByTrack(MPD::Song *, MPD::Song *);
|
static bool SortSongsByTrack(MPD::Song *, MPD::Song *);
|
||||||
static bool SortSongsByYear(MPD::Song *, MPD::Song *);
|
static bool SortSongsByYear(MPD::Song *, MPD::Song *);
|
||||||
|
|
||||||
|
|||||||
@@ -1022,28 +1022,7 @@ void Connection::GetList(TagList &v, mpd_tag_type type)
|
|||||||
mpd_search_commit(itsConnection);
|
mpd_search_commit(itsConnection);
|
||||||
while (mpd_pair *item = mpd_recv_pair_tag(itsConnection, type))
|
while (mpd_pair *item = mpd_recv_pair_tag(itsConnection, type))
|
||||||
{
|
{
|
||||||
if (item->value[0] != 0) // do not push empty item
|
v.push_back(item->value);
|
||||||
v.push_back(item->value);
|
|
||||||
mpd_return_pair(itsConnection, item);
|
|
||||||
}
|
|
||||||
mpd_response_finish(itsConnection);
|
|
||||||
GoIdle();
|
|
||||||
}
|
|
||||||
|
|
||||||
void Connection::GetAlbums(const std::string &artist, TagList &v)
|
|
||||||
{
|
|
||||||
if (!itsConnection)
|
|
||||||
return;
|
|
||||||
assert(!isCommandsListEnabled);
|
|
||||||
GoBusy();
|
|
||||||
mpd_search_db_tags(itsConnection, MPD_TAG_ALBUM);
|
|
||||||
if (!artist.empty())
|
|
||||||
mpd_search_add_tag_constraint(itsConnection, MPD_OPERATOR_DEFAULT, MPD_TAG_ARTIST, artist.c_str());
|
|
||||||
mpd_search_commit(itsConnection);
|
|
||||||
while (mpd_pair *item = mpd_recv_pair_tag(itsConnection, MPD_TAG_ALBUM))
|
|
||||||
{
|
|
||||||
if (item->value[0] != 0) // do not push empty item
|
|
||||||
v.push_back(item->value);
|
|
||||||
mpd_return_pair(itsConnection, item);
|
mpd_return_pair(itsConnection, item);
|
||||||
}
|
}
|
||||||
mpd_response_finish(itsConnection);
|
mpd_response_finish(itsConnection);
|
||||||
@@ -1096,8 +1075,7 @@ void Connection::CommitSearch(TagList &v)
|
|||||||
mpd_search_commit(itsConnection);
|
mpd_search_commit(itsConnection);
|
||||||
while (mpd_pair *tag = mpd_recv_pair_tag(itsConnection, itsSearchedField))
|
while (mpd_pair *tag = mpd_recv_pair_tag(itsConnection, itsSearchedField))
|
||||||
{
|
{
|
||||||
if (tag->value[0] != 0) // do not push empty item
|
v.push_back(tag->value);
|
||||||
v.push_back(tag->value);
|
|
||||||
mpd_return_pair(itsConnection, tag);
|
mpd_return_pair(itsConnection, tag);
|
||||||
}
|
}
|
||||||
mpd_response_finish(itsConnection);
|
mpd_response_finish(itsConnection);
|
||||||
|
|||||||
@@ -197,7 +197,6 @@ namespace MPD
|
|||||||
|
|
||||||
void GetPlaylists(TagList &);
|
void GetPlaylists(TagList &);
|
||||||
void GetList(TagList &, mpd_tag_type);
|
void GetList(TagList &, mpd_tag_type);
|
||||||
void GetAlbums(const std::string &, TagList &);
|
|
||||||
void GetDirectory(const std::string &, ItemList &);
|
void GetDirectory(const std::string &, ItemList &);
|
||||||
void GetDirectoryRecursive(const std::string &, SongList &);
|
void GetDirectoryRecursive(const std::string &, SongList &);
|
||||||
void GetSongs(const std::string &, SongList &);
|
void GetSongs(const std::string &, SongList &);
|
||||||
|
|||||||
@@ -1420,9 +1420,9 @@ int main(int argc, char *argv[])
|
|||||||
{
|
{
|
||||||
LockStatusbar();
|
LockStatusbar();
|
||||||
Statusbar() << fmtBold << "Album: " << fmtBoldEnd;
|
Statusbar() << fmtBold << "Album: " << fmtBoldEnd;
|
||||||
std::string new_album = wFooter->GetString(myLibrary->Albums->Current().second.Album);
|
std::string new_album = wFooter->GetString(myLibrary->Albums->Current().Album);
|
||||||
UnlockStatusbar();
|
UnlockStatusbar();
|
||||||
if (!new_album.empty() && new_album != myLibrary->Albums->Current().second.Album)
|
if (!new_album.empty() && new_album != myLibrary->Albums->Current().Album)
|
||||||
{
|
{
|
||||||
bool success = 1;
|
bool success = 1;
|
||||||
ShowMessage("Updating tags...");
|
ShowMessage("Updating tags...");
|
||||||
|
|||||||
@@ -235,7 +235,7 @@ void TagEditor::Update()
|
|||||||
{
|
{
|
||||||
*Albums << XY(0, 0) << "Fetching albums...";
|
*Albums << XY(0, 0) << "Fetching albums...";
|
||||||
Albums->Window::Refresh();
|
Albums->Window::Refresh();
|
||||||
Mpd.GetAlbums("", list);
|
Mpd.GetList(list, MPD_TAG_ALBUM);
|
||||||
for (MPD::TagList::const_iterator it = list.begin(); it != list.end(); ++it)
|
for (MPD::TagList::const_iterator it = list.begin(); it != list.end(); ++it)
|
||||||
{
|
{
|
||||||
MPD::SongList l;
|
MPD::SongList l;
|
||||||
|
|||||||
Reference in New Issue
Block a user