246 lines
5.9 KiB
C++
246 lines
5.9 KiB
C++
/***************************************************************************
|
|
* Copyright (C) 2008-2014 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 <algorithm>
|
|
#include <time.h>
|
|
|
|
#include "enums.h"
|
|
#include "helpers.h"
|
|
#include "playlist.h"
|
|
#include "statusbar.h"
|
|
#include "utility/functional.h"
|
|
|
|
const MPD::Song *currentSong(const BaseScreen *screen)
|
|
{
|
|
const MPD::Song *ptr = nullptr;
|
|
const auto *list = dynamic_cast<const SongList *>(screen->activeWindow());
|
|
if (list != nullptr)
|
|
{
|
|
const auto it = list->currentS();
|
|
if (it != list->endS())
|
|
ptr = it->get<Bit::Song>();
|
|
}
|
|
return ptr;
|
|
}
|
|
|
|
typedef std::vector<MPD::Song>::const_iterator VectorSongIterator;
|
|
bool addSongsToPlaylist(VectorSongIterator first, VectorSongIterator last, bool play, int position)
|
|
{
|
|
bool result = true;
|
|
auto addSongNoError = [&](VectorSongIterator song) -> int {
|
|
try
|
|
{
|
|
return Mpd.AddSong(*song, position);
|
|
}
|
|
catch (MPD::ServerError &e)
|
|
{
|
|
Status::handleServerError(e);
|
|
result = false;
|
|
return -1;
|
|
}
|
|
};
|
|
|
|
if (last-first >= 1)
|
|
{
|
|
int id;
|
|
while (true)
|
|
{
|
|
id = addSongNoError(first);
|
|
if (id >= 0)
|
|
break;
|
|
++first;
|
|
if (first == last)
|
|
return result;
|
|
}
|
|
|
|
if (position == -1)
|
|
{
|
|
++first;
|
|
for(; first != last; ++first)
|
|
addSongNoError(first);
|
|
}
|
|
else
|
|
{
|
|
++position;
|
|
--last;
|
|
for (; first != last; --last)
|
|
addSongNoError(last);
|
|
}
|
|
if (play)
|
|
Mpd.PlayID(id);
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
void removeSongFromPlaylist(const SongMenu &playlist, const MPD::Song &s)
|
|
{
|
|
Mpd.StartCommandsList();
|
|
for (auto &item : reverse_iteration(playlist))
|
|
if (item.value() == s)
|
|
Mpd.Delete(item.value().getPosition());
|
|
Mpd.CommitCommandsList();
|
|
}
|
|
|
|
bool addSongToPlaylist(const MPD::Song &s, bool play, int position)
|
|
{
|
|
bool result = false;
|
|
if (Config.space_add_mode == SpaceAddMode::AddRemove
|
|
&& myPlaylist->checkForSong(s)
|
|
)
|
|
{
|
|
result = true;
|
|
if (play)
|
|
{
|
|
const auto begin = myPlaylist->main().beginV(), end = myPlaylist->main().endV();
|
|
auto it = find_map_first(begin, end, s, [](const MPD::Song &found) {
|
|
Mpd.PlayID(found.getID());
|
|
});
|
|
assert(it != end);
|
|
}
|
|
else
|
|
removeSongFromPlaylist(myPlaylist->main(), s);
|
|
return result;
|
|
}
|
|
int id = Mpd.AddSong(s, position);
|
|
if (id >= 0)
|
|
{
|
|
Statusbar::printf("Added to playlist: %s",
|
|
Format::stringify<char>(Config.song_status_format, &s)
|
|
);
|
|
if (play)
|
|
Mpd.PlayID(id);
|
|
result = true;
|
|
}
|
|
return result;
|
|
}
|
|
|
|
std::string timeFormat(const char *format, time_t t)
|
|
{
|
|
char result[32];
|
|
tm tinfo;
|
|
localtime_r(&t, &tinfo);
|
|
strftime(result, sizeof(result), format, &tinfo);
|
|
return result;
|
|
}
|
|
|
|
std::string Timestamp(time_t t)
|
|
{
|
|
char result[32];
|
|
tm info;
|
|
result[strftime(result, 31, "%x %X", localtime_r(&t, &info))] = 0;
|
|
return result;
|
|
}
|
|
|
|
void markSongsInPlaylist(SongList &list)
|
|
{
|
|
MPD::Song *s;
|
|
for (auto &p : list)
|
|
{
|
|
s = p.get<Bit::Song>();
|
|
if (s != nullptr)
|
|
p.get<Bit::Properties>().setBold(myPlaylist->checkForSong(*s));
|
|
}
|
|
}
|
|
|
|
std::wstring Scroller(const std::wstring &str, size_t &pos, size_t width)
|
|
{
|
|
std::wstring s(str);
|
|
if (!Config.header_text_scrolling)
|
|
return s;
|
|
std::wstring result;
|
|
size_t len = wideLength(s);
|
|
|
|
if (len > width)
|
|
{
|
|
s += L" ** ";
|
|
len = 0;
|
|
auto b = s.begin(), e = s.end();
|
|
for (auto it = b+pos; it < e && len < width; ++it)
|
|
{
|
|
if ((len += wcwidth(*it)) > width)
|
|
break;
|
|
result += *it;
|
|
}
|
|
if (++pos >= s.length())
|
|
pos = 0;
|
|
for (; len < width; ++b)
|
|
{
|
|
if ((len += wcwidth(*b)) > width)
|
|
break;
|
|
result += *b;
|
|
}
|
|
}
|
|
else
|
|
result = s;
|
|
return result;
|
|
}
|
|
|
|
void writeCyclicBuffer(const NC::WBuffer &buf, NC::Window &w, size_t &start_pos,
|
|
size_t width, const std::wstring &separator)
|
|
{
|
|
const auto &s = buf.str();
|
|
size_t len = wideLength(s);
|
|
if (len > width)
|
|
{
|
|
len = 0;
|
|
const auto &ps = buf.properties();
|
|
auto p = ps.begin();
|
|
|
|
// load attributes from before starting pos
|
|
for (; p != ps.end() && p->position() < start_pos; ++p)
|
|
w << *p;
|
|
|
|
auto write_buffer = [&](size_t start) {
|
|
for (size_t i = start; i < s.length() && len < width; ++i)
|
|
{
|
|
for (; p != ps.end() && p->position() == i; ++p)
|
|
w << *p;
|
|
len += wcwidth(s[i]);
|
|
if (len > width)
|
|
break;
|
|
w << s[i];
|
|
}
|
|
for (; p != ps.end(); ++p)
|
|
w << *p;
|
|
p = ps.begin();
|
|
};
|
|
|
|
write_buffer(start_pos);
|
|
size_t i = 0;
|
|
if (start_pos > s.length())
|
|
i = start_pos - s.length();
|
|
for (; i < separator.length() && len < width; ++i)
|
|
{
|
|
len += wcwidth(separator[i]);
|
|
if (len > width)
|
|
break;
|
|
w << separator[i];
|
|
}
|
|
write_buffer(0);
|
|
|
|
++start_pos;
|
|
if (start_pos >= s.length() + separator.length())
|
|
start_pos = 0;
|
|
}
|
|
else
|
|
w << buf;
|
|
}
|