new feature: add selected items to playlist at given position

This commit is contained in:
Andrzej Rybczak
2009-11-26 21:41:30 +01:00
parent d2b9aee650
commit 03129784d4
4 changed files with 138 additions and 45 deletions

View File

@@ -37,10 +37,23 @@ SelectedItemsAdder *mySelectedItemsAdder = new SelectedItemsAdder;
void SelectedItemsAdder::Init() void SelectedItemsAdder::Init()
{ {
SetDimensions(); SetDimensions();
w = new Menu<std::string>((COLS-itsWidth)/2, (MainHeight-itsHeight)/2+MainStartY, itsWidth, itsHeight, "Add selected items to...", Config.main_color, Config.window_border); itsPlaylistSelector = new Menu<std::string>((COLS-itsWidth)/2, (MainHeight-itsHeight)/2+MainStartY, itsWidth, itsHeight, "Add selected items to...", Config.main_color, Config.window_border);
w->CyclicScrolling(Config.use_cyclic_scrolling); itsPlaylistSelector->CyclicScrolling(Config.use_cyclic_scrolling);
w->HighlightColor(Config.main_highlight_color); itsPlaylistSelector->HighlightColor(Config.main_highlight_color);
w->SetItemDisplayer(Display::Generic); itsPlaylistSelector->SetItemDisplayer(Display::Generic);
itsPositionSelector = new Menu<std::string>((COLS-itsPSWidth)/2, (MainHeight-itsPSHeight)/2+MainStartY, itsPSWidth, itsPSHeight, "Where?", Config.main_color, Config.window_border);
itsPositionSelector->CyclicScrolling(Config.use_cyclic_scrolling);
itsPositionSelector->HighlightColor(Config.main_highlight_color);
itsPositionSelector->SetItemDisplayer(Display::Generic);
itsPositionSelector->AddOption("At the end of playlist");
itsPositionSelector->AddOption("At the beginning of playlist");
itsPositionSelector->AddOption("After current track");
itsPositionSelector->AddOption("After current album");
itsPositionSelector->AddSeparator();
itsPositionSelector->AddOption("Cancel");
w = itsPlaylistSelector;
isInitialized = 1; isInitialized = 1;
} }
@@ -67,6 +80,9 @@ void SelectedItemsAdder::SwitchTo()
if (!isInitialized) if (!isInitialized)
Init(); Init();
// default to main window
w = itsPlaylistSelector;
// Resize() can fall back to old screen, so we need it updated // Resize() can fall back to old screen, so we need it updated
myOldScreen = myScreen; myOldScreen = myScreen;
@@ -102,8 +118,12 @@ void SelectedItemsAdder::Resize()
SetDimensions(); SetDimensions();
if (itsHeight < 5) // screen too low to display this window if (itsHeight < 5) // screen too low to display this window
return myOldScreen->SwitchTo(); return myOldScreen->SwitchTo();
w->Resize(itsWidth, itsHeight); itsPlaylistSelector->Resize(itsWidth, itsHeight);
w->MoveTo((COLS-itsWidth)/2, (MainHeight-itsHeight)/2+MainStartY); itsPlaylistSelector->MoveTo((COLS-itsWidth)/2, (MainHeight-itsHeight)/2+MainStartY);
size_t poss_width = std::min(itsPSWidth, size_t(COLS));
size_t poss_height = std::min(itsPSHeight, size_t(MainHeight));
itsPositionSelector->Resize(poss_width, poss_height);
itsPositionSelector->MoveTo((COLS-poss_width)/2, (MainHeight-poss_height)/2+MainStartY);
if (myOldScreen && myOldScreen->hasToBeResized) // resize background window if (myOldScreen && myOldScreen->hasToBeResized) // resize background window
{ {
myOldScreen->Resize(); myOldScreen->Resize();
@@ -112,6 +132,17 @@ void SelectedItemsAdder::Resize()
hasToBeResized = 0; hasToBeResized = 0;
} }
void SelectedItemsAdder::Refresh()
{
if (w == itsPositionSelector)
{
itsPlaylistSelector->Display();
itsPositionSelector->Display();
}
else
itsPlaylistSelector->Display();
}
std::basic_string<my_char_t> SelectedItemsAdder::Title() std::basic_string<my_char_t> SelectedItemsAdder::Title()
{ {
return myOldScreen->Title(); return myOldScreen->Title();
@@ -122,46 +153,89 @@ void SelectedItemsAdder::EnterPressed()
size_t pos = w->Choice(); size_t pos = w->Choice();
MPD::SongList list; MPD::SongList list;
if (pos != w->Size()-1) if ((w != itsPlaylistSelector || pos != 0) && pos != w->Size()-1)
myOldScreen->GetSelectedSongs(list); myOldScreen->GetSelectedSongs(list);
if (pos == 0) // add to mpd playlist if (w == itsPlaylistSelector)
{ {
if (myPlaylist->Add(list, 0)) if (pos == 0) // add to mpd playlist
ShowMessage("Selected items added!");
}
else if (pos == 1) // create new playlist
{
LockStatusbar();
Statusbar() << "Save playlist as: ";
std::string playlist = Global::wFooter->GetString();
UnlockStatusbar();
if (!playlist.empty())
{ {
std::string utf_playlist = locale_to_utf_cpy(playlist); w = itsPositionSelector;
itsPositionSelector->Reset();
return;
}
else if (pos == 1) // create new playlist
{
LockStatusbar();
Statusbar() << "Save playlist as: ";
std::string playlist = Global::wFooter->GetString();
UnlockStatusbar();
if (!playlist.empty())
{
std::string utf_playlist = locale_to_utf_cpy(playlist);
Mpd.StartCommandsList();
for (MPD::SongList::const_iterator it = list.begin(); it != list.end(); ++it)
Mpd.AddToPlaylist(utf_playlist, **it);
if (Mpd.CommitCommandsList())
ShowMessage("Selected items added to playlist \"%s\"!", playlist.c_str());
}
}
else if (pos > 1 && pos < w->Size()-1) // add items to existing playlist
{
std::string playlist = locale_to_utf_cpy(w->Current());
Mpd.StartCommandsList(); Mpd.StartCommandsList();
for (MPD::SongList::const_iterator it = list.begin(); it != list.end(); ++it) for (MPD::SongList::const_iterator it = list.begin(); it != list.end(); ++it)
Mpd.AddToPlaylist(utf_playlist, **it); Mpd.AddToPlaylist(playlist, **it);
if (Mpd.CommitCommandsList()) if (Mpd.CommitCommandsList())
ShowMessage("Selected items added to playlist \"%s\"!", playlist.c_str()); ShowMessage("Selected items added to playlist \"%s\"!", w->Current().c_str());
}
if (pos != w->Size()-1)
{
// refresh playlist's lists
if (myBrowser->Main() && !myBrowser->isLocal() && myBrowser->CurrentDir() == "/")
myBrowser->GetDirectory("/");
if (myPlaylistEditor->Main())
myPlaylistEditor->Playlists->Clear(); // make playlist editor update itself
} }
} }
else if (pos > 1 && pos < w->Size()-1) // add items to existing playlist else
{ {
std::string playlist = locale_to_utf_cpy(w->Current()); if (pos > 0 && pos < 3 && !Mpd.isPlaying())
Mpd.StartCommandsList(); {
for (MPD::SongList::const_iterator it = list.begin(); it != list.end(); ++it) ShowMessage("Player is stopped!");
Mpd.AddToPlaylist(playlist, **it); return;
if (Mpd.CommitCommandsList()) }
ShowMessage("Selected items added to playlist \"%s\"!", w->Current().c_str());
} bool successful_operation;
if (pos != w->Size()-1) if (pos == 0) // end of playlist
{ {
// refresh playlist's lists successful_operation = myPlaylist->Add(list, 0);
if (myBrowser->Main() && !myBrowser->isLocal() && myBrowser->CurrentDir() == "/") }
myBrowser->GetDirectory("/"); else if (pos == 1) // beginning of playlist
if (myPlaylistEditor->Main()) {
myPlaylistEditor->Playlists->Clear(); // make playlist editor update itself successful_operation = myPlaylist->Add(list, 0, 0);
}
else if (pos == 2) // after currently playing track
{
successful_operation = myPlaylist->Add(list, 0, Mpd.GetCurrentSongPos()+1);
}
else if (pos == 3) // after currently playing album
{
std::string album = myPlaylist->CurrentSong()->GetAlbum();
int i;
for (i = Mpd.GetCurrentSongPos()+1; i < int(myPlaylist->Items->Size()); ++i)
if ((*myPlaylist->Items)[i].GetAlbum() != album)
break;
successful_operation = myPlaylist->Add(list, 0, i);
}
else
{
w = itsPlaylistSelector;
return;
}
if (successful_operation)
ShowMessage("Selected items added!");
} }
MPD::FreeSongList(list); MPD::FreeSongList(list);
SwitchTo(); SwitchTo();

View File

@@ -27,8 +27,11 @@
class SelectedItemsAdder : public Screen< Menu<std::string> > class SelectedItemsAdder : public Screen< Menu<std::string> >
{ {
public: public:
SelectedItemsAdder() : itsPSWidth(35), itsPSHeight(10) { }
virtual void SwitchTo(); virtual void SwitchTo();
virtual void Resize(); virtual void Resize();
virtual void Refresh();
virtual std::basic_string<my_char_t> Title(); virtual std::basic_string<my_char_t> Title();
@@ -46,6 +49,12 @@ class SelectedItemsAdder : public Screen< Menu<std::string> >
private: private:
void SetDimensions(); void SetDimensions();
Menu<std::string> *itsPlaylistSelector;
Menu<std::string> *itsPositionSelector;
size_t itsPSWidth;
size_t itsPSHeight;
size_t itsWidth; size_t itsWidth;
size_t itsHeight; size_t itsHeight;
}; };

View File

@@ -419,7 +419,7 @@ std::string Playlist::SongInColumnsToString(const MPD::Song &s, void *)
return s.toString(result); return s.toString(result);
} }
bool Playlist::Add(const MPD::Song &s, bool in_playlist, bool play) bool Playlist::Add(const MPD::Song &s, bool in_playlist, bool play, int position)
{ {
BlockItemListUpdate = 1; BlockItemListUpdate = 1;
if (Config.ncmpc_like_songs_adding && in_playlist) if (Config.ncmpc_like_songs_adding && in_playlist)
@@ -457,7 +457,7 @@ bool Playlist::Add(const MPD::Song &s, bool in_playlist, bool play)
} }
else else
{ {
int id = Mpd.AddSong(s); int id = Mpd.AddSong(s, position);
if (id >= 0) if (id >= 0)
{ {
ShowMessage("Added to playlist: %s", s.toString(Config.song_status_format_no_colors).c_str()); ShowMessage("Added to playlist: %s", s.toString(Config.song_status_format_no_colors).c_str());
@@ -470,7 +470,7 @@ bool Playlist::Add(const MPD::Song &s, bool in_playlist, bool play)
} }
} }
bool Playlist::Add(const MPD::SongList &l, bool play) bool Playlist::Add(const MPD::SongList &l, bool play, int position)
{ {
if (l.empty()) if (l.empty())
return false; return false;
@@ -479,15 +479,25 @@ bool Playlist::Add(const MPD::SongList &l, bool play)
Mpd.StartCommandsList(); Mpd.StartCommandsList();
MPD::SongList::const_iterator it = l.begin(); MPD::SongList::const_iterator it = l.begin();
for (; it != l.end(); ++it) if (position < 0)
if (Mpd.AddSong(**it) < 0) {
break; for (; it != l.end(); ++it)
if (Mpd.AddSong(**it) < 0)
break;
}
else
{
MPD::SongList::const_reverse_iterator j = l.rbegin();
for (; j != l.rend(); ++j)
if (Mpd.AddSong(**j, position) < 0)
break;
}
Mpd.CommitCommandsList(); Mpd.CommitCommandsList();
if (play && old_playlist_size < Items->Size()) if (play && old_playlist_size < Items->Size())
Mpd.Play(old_playlist_size); Mpd.Play(old_playlist_size);
if (Items->Back().GetHash() != l.back()->GetHash()) if (position < 0 && Items->Back().GetHash() != l.back()->GetHash())
{ {
if (it != l.begin()) if (it != l.begin())
ShowMessage("%s", MPD::Message::PartOfSongsAdded); ShowMessage("%s", MPD::Message::PartOfSongsAdded);

View File

@@ -64,8 +64,8 @@ class Playlist : public Screen<Window>
void UpdateTimer() { time(&itsTimer); } void UpdateTimer() { time(&itsTimer); }
time_t Timer() const { return itsTimer; } time_t Timer() const { return itsTimer; }
bool Add(const MPD::Song &s, bool in_playlist, bool play); bool Add(const MPD::Song &s, bool in_playlist, bool play, int position = -1);
bool Add(const MPD::SongList &l, bool play); bool Add(const MPD::SongList &l, bool play, int position = -1);
static std::string SongToString(const MPD::Song &, void *); static std::string SongToString(const MPD::Song &, void *);
static std::string SongInColumnsToString(const MPD::Song &, void *); static std::string SongInColumnsToString(const MPD::Song &, void *);