/*************************************************************************** * 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 #include #include "charset.h" #include "global.h" #include "helpers.h" #include "playlist.h" #include "tag_editor.h" using namespace MPD; using Global::Config; using Global::Mpd; using Global::wFooter; using std::string; namespace { inline void remove_the_word(string &s) { size_t the_pos = s.find("the "); if (the_pos == 0 && the_pos != string::npos) s = s.substr(4); } inline string extract_top_directory(const string &s) { size_t slash = s.rfind("/"); return slash != string::npos ? s.substr(++slash) : s; } } bool ConnectToMPD() { if (!Mpd->Connect()) { std::cout << "Cannot connect to mpd: " << Mpd->GetErrorMessage() << std::endl; return false; } return true; } void ParseArgv(int argc, char **argv) { using std::cout; using std::endl; bool quit = 0; for (int i = 1; i < argc; i++) { if (strcmp(argv[i], "-v") == 0 || strcmp(argv[i], "--version") == 0) { cout << "ncmpcpp version: " << VERSION << endl << "built with support for:" # ifdef HAVE_CURL_CURL_H << " curl" # endif # ifdef HAVE_TAGLIB_H << " taglib" # endif # ifdef _UTF8 << " unicode" # endif << endl; exit(0); } else if (strcmp(argv[i], "-?") == 0 || strcmp(argv[i], "--help") == 0) { cout << "Usage: ncmpcpp [OPTION]...\n" << " -?, --help show this help message\n" << " -v, --version display version information\n\n" << " play start playing\n" << " pause pause the currently playing song\n" << " toggle toggle play/pause mode\n" << " stop stop playing\n" << " next play the next song\n" << " prev play the previous song\n" << " volume [+-] adjusts volume by [+-]\n" ; exit(0); } if (!ConnectToMPD()) exit(0); if (strcmp(argv[i], "play") == 0) { Mpd->Play(); quit = 1; } else if (strcmp(argv[i], "pause") == 0) { Mpd->Execute("pause \"1\"\n"); quit = 1; } else if (strcmp(argv[i], "toggle") == 0) { Mpd->UpdateStatus(); switch (Mpd->GetState()) { case psPause: case psPlay: Mpd->Pause(); break; case psStop: Mpd->Play(); break; default: break; } quit = 1; } else if (strcmp(argv[i], "stop") == 0) { Mpd->Stop(); quit = 1; } else if (strcmp(argv[i], "next") == 0) { Mpd->Next(); quit = 1; } else if (strcmp(argv[i], "prev") == 0) { Mpd->Prev(); quit = 1; } else if (strcmp(argv[i], "volume") == 0) { i++; Mpd->UpdateStatus(); if (!Mpd->GetErrorMessage().empty()) { cout << "Error: " << Mpd->GetErrorMessage() << endl; exit(0); } if (i != argc) Mpd->SetVolume(Mpd->GetVolume()+atoi(argv[i])); quit = 1; } else { cout << "ncmpcpp: invalid option: " << argv[i] << endl; exit(0); } if (!Mpd->GetErrorMessage().empty()) { cout << "Error: " << Mpd->GetErrorMessage() << endl; exit(0); } } if (quit) exit(0); } bool CaseInsensitiveSorting::operator()(string a, string b) { ToLower(a); ToLower(b); if (Config.ignore_leading_the) { remove_the_word(a); remove_the_word(b); } return a < b; } bool CaseInsensitiveSorting::operator()(Song *sa, Song *sb) { string a = sa->GetName(); string b = sb->GetName(); ToLower(a); ToLower(b); if (Config.ignore_leading_the) { remove_the_word(a); remove_the_word(b); } return a < b; } bool CaseInsensitiveSorting::operator()(const Item &a, const Item &b) { if (a.type == b.type) { switch (a.type) { case itDirectory: return operator()(extract_top_directory(a.name), extract_top_directory(b.name)); case itPlaylist: return operator()(a.name, b.name); case itSong: return operator()(a.song, b.song); default: // there's no other type, just silence compiler. return 0; } } else return a.type < b.type; } void UpdateSongList(Menu *menu) { bool bold = 0; for (size_t i = 0; i < menu->Size(); i++) { for (size_t j = 0; j < myPlaylist->Main()->Size(); j++) { if (myPlaylist->Main()->at(j).GetHash() == menu->at(i).GetHash()) { bold = 1; break; } } menu->BoldOption(i, bold); bold = 0; } menu->Refresh(); } bool Keypressed(int in, const int *key) { return in == key[0] || in == key[1]; } bool SortSongsByTrack(Song *a, Song *b) { if (a->GetDisc() == b->GetDisc()) return StrToInt(a->GetTrack()) < StrToInt(b->GetTrack()); else return StrToInt(a->GetDisc()) < StrToInt(b->GetDisc()); } string FindSharedDir(const string &one, const string &two) { if (one == two) return one; string result; size_t i = 1; while (one.substr(0, i) == two.substr(0, i)) i++; result = one.substr(0, i); i = result.rfind("/"); return i != string::npos ? result.substr(0, i) : "/"; } string GetLineValue(string &line, char a, char b, bool once) { int i = 0; int begin = -1, end = -1; for (string::iterator it = line.begin(); it != line.end() && (begin == -1 || end == -1); i++, it++) { if (*it == a || *it == b) { if (once) *it = 0; if (begin < 0) begin = i+1; else end = i; } } if (begin >= 0 && end >= 0) return line.substr(begin, end-begin); else return ""; } Window &Statusbar() { wFooter->GotoXY(0, Config.statusbar_visibility); wclrtoeol(wFooter->Raw()); return *wFooter; } const Buffer &ShowTag(const string &tag) { static Buffer result; result.Clear(); if (tag.empty()) result << Config.empty_tags_color << Config.empty_tag << clEnd; else result << tag; return result; } const std::basic_string &Scroller(const string &string, size_t width, size_t &pos) { static std::basic_string result; result.clear(); std::basic_string s = TO_WSTRING(string); size_t len; # ifdef _UTF8 len = Window::Length(s); # else len = s.length(); # endif if (len > width) { # ifdef _UTF8 s += L" ** "; # else s += " ** "; # endif len = 0; std::basic_string::const_iterator b = s.begin(), e = s.end(); for (std::basic_string::const_iterator it = b+pos; it < e && len < width; it++) { # ifdef _UTF8 len += wcwidth(*it); # else len++; # endif result += *it; } if (++pos >= s.length()) pos = 0; for (; len < width; b++) { # ifdef _UTF8 len += wcwidth(*b); # else len++; # endif result += *b; } } else result = s; return result; } #ifdef HAVE_CURL_CURL_H size_t write_data(char *buffer, size_t size, size_t nmemb, string data) { size_t result = size * nmemb; data.append(buffer, result); return result; } #endif // HAVE_CURL_CURL_H