diff options
author | Lars-Dominik Braun <lars@6xq.net> | 2011-03-25 16:24:29 +0100 |
---|---|---|
committer | Lars-Dominik Braun <lars@6xq.net> | 2011-03-26 21:44:52 +0100 |
commit | 5488adb613ca623a8ef4cb37b6a802effaaf50ec (patch) | |
tree | a0b644092d337f27317b69a72c3d73c2da7623db /src/ui.c | |
parent | 6d3009b31e079498919a0678d48b6a911d1c4325 (diff) | |
download | pianobar-5488adb613ca623a8ef4cb37b6a802effaaf50ec.tar.gz pianobar-5488adb613ca623a8ef4cb37b6a802effaaf50ec.tar.bz2 pianobar-5488adb613ca623a8ef4cb37b6a802effaaf50ec.zip |
Filter-/searchable song list
Diffstat (limited to 'src/ui.c')
-rw-r--r-- | src/ui.c | 87 |
1 files changed, 71 insertions, 16 deletions
@@ -34,6 +34,7 @@ THE SOFTWARE. #include <errno.h> #include <strings.h> #include <assert.h> +#include <ctype.h> /* tolower() */ /* waitpid () */ #include <sys/types.h> @@ -337,6 +338,21 @@ PianoStation_t *BarUiSelectStation (PianoHandle_t *ph, const char *prompt, return retStation; } +/* is string a number? + */ +static bool isnumeric (const char *s) { + if (*s == '\0') { + return false; + } + while (*s != '\0') { + if (!isdigit ((unsigned char) *s)) { + return false; + } + ++s; + } + return true; +} + /* let user pick one song * @param pianobar settings * @param song list @@ -346,20 +362,27 @@ PianoStation_t *BarUiSelectStation (PianoHandle_t *ph, const char *prompt, PianoSong_t *BarUiSelectSong (const BarSettings_t *settings, PianoSong_t *startSong, BarReadlineFds_t *input) { PianoSong_t *tmpSong = NULL; - int i = 0; + char buf[100]; - i = BarUiListSongs (settings, startSong); + memset (buf, 0, sizeof (buf)); - BarUiMsg (MSG_QUESTION, "Select song: "); - if (BarReadlineInt (&i, input) == 0) { - return NULL; - } + do { + BarUiListSongs (settings, startSong, buf); - tmpSong = startSong; - while (tmpSong != NULL && i > 0) { - tmpSong = tmpSong->next; - i--; - } + BarUiMsg (MSG_QUESTION, "Select song: "); + if (BarReadlineStr (buf, sizeof (buf), input, BAR_RL_DEFAULT) == 0) { + return NULL; + } + + if (isnumeric (buf)) { + unsigned long i = strtoul (buf, NULL, 0); + tmpSong = startSong; + while (tmpSong != NULL && i > 0) { + tmpSong = tmpSong->next; + i--; + } + } + } while (tmpSong == NULL); return tmpSong; } @@ -566,20 +589,52 @@ inline void BarUiPrintSong (const BarSettings_t *settings, station != NULL ? station->name : ""); } +/* find needle in haystack, ignoring case, and return first position + */ +const char *strcasestr (const char *haystack, const char *needle) { + const char *needlePos = needle; + + assert (haystack != NULL); + assert (needle != NULL); + + if (*needle == '\0') { + return haystack; + } + + while (*haystack != '\0') { + if (tolower ((unsigned char) *haystack) == tolower ((unsigned char) *needlePos)) { + ++needlePos; + } else { + needlePos = needle; + } + ++haystack; + if (*needlePos == '\0') { + return haystack - strlen (needle); + } + } + + return NULL; +} + /* Print list of songs * @param pianobar settings * @param linked list of songs + * @param artist/song filter string * @return # of songs */ size_t BarUiListSongs (const BarSettings_t *settings, - const PianoSong_t *song) { + const PianoSong_t *song, const char *filter) { size_t i = 0; while (song != NULL) { - BarUiMsg (MSG_LIST, "%2lu) %s - %s %s%s\n", i, song->artist, - song->title, - (song->rating == PIANO_RATE_LOVE) ? settings->loveIcon : "", - (song->rating == PIANO_RATE_BAN) ? settings->banIcon : ""); + if (filter == NULL || + (filter != NULL && (strcasestr (song->artist, filter) != NULL || + strcasestr (song->title, filter) != NULL))) { + BarUiMsg (MSG_LIST, "%2lu) %s - %s %s%s\n", i, song->artist, + song->title, + (song->rating == PIANO_RATE_LOVE) ? settings->loveIcon : "", + (song->rating == PIANO_RATE_BAN) ? settings->banIcon : ""); + } song = song->next; i++; } |