From d9f992b23dd8f447fedb0eb74e983be5246e3efd Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?P=C5=99emysl=20Janouch?= Date: Mon, 10 Oct 2016 16:02:57 +0200 Subject: Add a dumb Current tab --- README.adoc | 2 +- nncmpp.c | 227 ++++++++++++++++++++++++++++++++++++++++++++++++++---------- 2 files changed, 191 insertions(+), 38 deletions(-) diff --git a/README.adoc b/README.adoc index f7ece38..9d52445 100644 --- a/README.adoc +++ b/README.adoc @@ -11,7 +11,7 @@ names, and should be pronounced as "nincompoop". Features -------- -Currently it's under development and doesn't do much else than show the status. +Currently it's under development and doesn't show the music library yet. Since I only use the filesystem browsing mode, that's also the only thing I care to implement. Anything else will have to be user-contributed while diff --git a/nncmpp.c b/nncmpp.c index 7d91490..876c99b 100644 --- a/nncmpp.c +++ b/nncmpp.c @@ -142,6 +142,13 @@ clock_msec (clockid_t clock) return (int64_t) tp.tv_sec * 1000 + (int64_t) tp.tv_nsec / 1000000; } +static bool +xstrtoul_map (const struct str_map *map, const char *key, unsigned long *out) +{ + const char *field = str_map_find (map, key); + return field && xstrtoul (out, field, 10); +} + static char * latin1_to_utf8 (const char *latin1) { @@ -455,6 +462,13 @@ struct tab // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - +struct playlist +{ + struct str_map *items; ///< Current playlist + size_t len; ///< Length + size_t alloc; ///< Allocated items +}; + struct attrs { short fg; ///< Foreground colour index @@ -491,10 +505,14 @@ static struct app_context int64_t elapsed_since; ///< Time of the next tick // TODO: initialize these to -1 + int song; ///< Current song index int song_elapsed; ///< Song elapsed in seconds int song_duration; ///< Song duration in seconds int volume; ///< Current volume + struct playlist playlist; ///< Current playlist + uint32_t playlist_version; ///< Playlist version + // Data: struct config config; ///< Program configuration @@ -549,6 +567,68 @@ tab_free (struct tab *self) free (self->name); } +// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - + +static void +playlist_init (struct playlist *self) +{ + memset (self, 0, sizeof *self); + self->items = xcalloc (sizeof *self->items, (self->alloc = 16)); +} + +static void +playlist_free (struct playlist *self) +{ + for (size_t i = 0; i < self->len; i++) + str_map_free (&self->items[i]); + free (self->items); +} + +static bool +playlist_set (struct playlist *self, int i, struct str_map *item) +{ + if (i < 0 || (size_t) i >= self->len) + return false; + + str_map_free (&self->items[i]); + self->items[i] = *item; + return true; +} + +static struct str_map * +playlist_get (struct playlist *self, int i) +{ + if (i < 0 || (size_t) i >= self->len) + return false; + + return &self->items[i]; +} + +static void +playlist_resize (struct playlist *self, size_t len) +{ + // Make the allocated array big enough but not too large + size_t new_alloc = self->alloc; + while (new_alloc < len) + new_alloc <<= 1; + while (len < (new_alloc >> 2) + && new_alloc >= (STR_MAP_MIN_ALLOC << 1)) + new_alloc >>= 1; + + // Dispose of items that are out of range and resize the array if needed + for (size_t i = len; i < self->len; i++) + str_map_free (&self->items[i]); + if (new_alloc != self->alloc) + self->items = xreallocarray (self->items, + sizeof *self->items, new_alloc); + + // We need to initialize placeholders so that str_map_find() succeeds + for (size_t i = self->len; i < len; i++) + str_map_init (&self->items[i]); + + self->len = len; +} + // --- Configuration ----------------------------------------------------------- static struct config_schema g_config_settings[] = @@ -730,6 +810,7 @@ app_init_context (void) mpd_client_init (&g_ctx.client, &g_ctx.poller); config_init (&g_ctx.config); str_vector_init (&g_ctx.streams); + playlist_init (&g_ctx.playlist); // This is also approximately what libunistring does internally, // since the locale name is canonicalized by locale_charset(). @@ -782,6 +863,7 @@ app_free_context (void) mpd_client_free (&g_ctx.client); str_map_free (&g_ctx.playback_info); str_vector_free (&g_ctx.streams); + playlist_free (&g_ctx.playlist); config_free (&g_ctx.config); poller_free (&g_ctx.poller); @@ -1031,9 +1113,9 @@ app_flush_header (struct row_buffer *buf, chtype attrs) static void app_draw_song_info (void) { - // The map doesn't need to be initialized at all, so we need to check - struct str_map *map = &g_ctx.playback_info; - if (!soft_assert (map->len != 0)) + struct str_map *map; + if (!(map = playlist_get (&g_ctx.playlist, g_ctx.song)) + || !soft_assert (map->len != 0)) return; // XXX: can we get rid of this and still make it look acceptable? @@ -1805,8 +1887,45 @@ app_process_termo_event (termo_key_t *event) return false; } +// --- Current tab ------------------------------------------------------------- + +// TODO: remove the useless wrapper struct? +static struct +{ + struct tab super; ///< Parent class +} +g_current_tab; + +static void +current_tab_on_item_draw (size_t item_index, struct row_buffer *buffer, + int width) +{ + // TODO: better output + struct str_map *map = playlist_get (&g_ctx.playlist, item_index); + row_buffer_append (buffer, str_map_find (map, "file"), + (int) item_index == g_ctx.song ? A_BOLD : 0); +} + +static void +current_tab_update (void) +{ + g_current_tab.super.item_count = g_ctx.playlist.len; + app_invalidate (); +} + +static struct tab * +current_tab_init (void) +{ + struct tab *super = &g_current_tab.super; + tab_init (super, "Current"); + super->on_item_draw = current_tab_on_item_draw; + return super; +} + // --- Streams ----------------------------------------------------------------- +// MPD can only parse m3u8 playlists, and only when it feels like doing so + struct stream_tab_task { struct poller_curl_task curl; ///< Superclass @@ -2101,13 +2220,16 @@ info_tab_update (void) str_vector_reset (&g_info_tab.values); g_info_tab.super.item_count = 0; - struct str_map *map = &g_ctx.playback_info; - info_tab_add (map, "Title"); - info_tab_add (map, "Artist"); - info_tab_add (map, "Album"); - info_tab_add (map, "Track"); - info_tab_add (map, "Genre"); - info_tab_add (map, "File"); + struct str_map *map; + if ((map = playlist_get (&g_ctx.playlist, g_ctx.song))) + { + info_tab_add (map, "Title"); + info_tab_add (map, "Artist"); + info_tab_add (map, "Album"); + info_tab_add (map, "Track"); + info_tab_add (map, "Genre"); + info_tab_add (map, "file"); + } } static struct tab * @@ -2257,11 +2379,9 @@ mpd_update_playback_state (void) } } - unsigned long tmp; - if (time && xstrtoul (&tmp, time, 10)) - g_ctx.song_elapsed = tmp; - if (duration && xstrtoul (&tmp, duration, 10)) - g_ctx.song_duration = tmp; + unsigned long n; + if (time && xstrtoul (&n, time, 10)) g_ctx.song_elapsed = n; + if (duration && xstrtoul (&n, duration, 10)) g_ctx.song_duration = n; // We could also just poll the server each half a second but let's not int msec_past_second = 0; @@ -2271,11 +2391,11 @@ mpd_update_playback_state (void) { // For some reason this is much more precise *period++ = '\0'; - if (xstrtoul (&tmp, elapsed, 10)) - g_ctx.song_elapsed = tmp; + if (xstrtoul (&n, elapsed, 10)) + g_ctx.song_elapsed = n; - if (xstrtoul (&tmp, period, 10)) - msec_past_second = tmp; + if (xstrtoul (&n, period, 10)) + msec_past_second = n; } if (g_ctx.state == PLAYER_PLAYING) { @@ -2284,29 +2404,63 @@ mpd_update_playback_state (void) } // The server sends -1 when nothing is being played right now - char *volume = str_map_find (map, "volume"); - if (volume && xstrtoul (&tmp, volume, 10)) - g_ctx.volume = tmp; + if (xstrtoul_map (map, "volume", &n)) g_ctx.volume = n; + + if (xstrtoul_map (map, "playlist", &n)) g_ctx.playlist_version = n; + if (xstrtoul_map (map, "song", &n)) g_ctx.song = n; } // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -// Sometimes it's not that easy and there can be repeating entries static void -mpd_vector_to_map (const struct str_vector *data, struct str_map *map) +mpd_init_response_map (struct str_map *map) { str_map_init (map); map->key_xfrm = tolower_ascii_strxfrm; map->free = free; +} - char *key, *value; +static void +mpd_process_info_chunk (struct str_map *map, char *file) +{ + unsigned long n; + if (!file) + { + if (xstrtoul_map (map, "playlistlength", &n)) + playlist_resize (&g_ctx.playlist, n); + g_ctx.playback_info = *map; + } + else if (!xstrtoul_map (map, "pos", &n) + || !playlist_set (&g_ctx.playlist, n, map)) + str_map_free (map); + + mpd_init_response_map (map); +} + +static void +mpd_process_info (const struct str_vector *data) +{ + struct str_map map; + mpd_init_response_map (&map); + + // First there's the status, followed by playlist items chunked by "file" + char *key, *value, *file = NULL; for (size_t i = 0; i < data->len; i++) { - if ((key = mpd_client_parse_kv (data->vector[i], &value))) - str_map_set (map, key, xstrdup (value)); - else + if (!(key = mpd_client_parse_kv (data->vector[i], &value))) + { print_debug ("%s: %s", "erroneous MPD output", data->vector[i]); + continue; + } + if (!strcasecmp_ascii (key, "file")) + { + mpd_process_info_chunk (&map, file); + file = value; + } + str_map_set (&map, key, xstrdup (value)); } + mpd_process_info_chunk (&map, file); + str_map_free (&map); } static void @@ -2316,21 +2470,17 @@ mpd_on_info_response (const struct mpd_response *response, (void) user_data; // TODO: do this also on disconnect + g_ctx.song = -1; g_ctx.song_elapsed = -1; g_ctx.song_duration = -1; g_ctx.volume = -1; str_map_free (&g_ctx.playback_info); poller_timer_reset (&g_ctx.elapsed_event); + g_ctx.playlist_version = 0; // TODO: preset an error player state? if (response->success) - { - // Note that we may receive a "time" field twice, however the right - // one wins here due to the order we send the commands in - struct str_map map; - mpd_vector_to_map (data, &map); - g_ctx.playback_info = map; - } + mpd_process_info (data); else { print_debug ("%s: %s", @@ -2338,6 +2488,7 @@ mpd_on_info_response (const struct mpd_response *response, } mpd_update_playback_state (); + current_tab_update (); info_tab_update (); app_invalidate (); } @@ -2363,11 +2514,12 @@ mpd_request_info (void) struct mpd_client *c = &g_ctx.client; mpd_client_list_begin (c); - mpd_client_send_command (c, "currentsong", NULL); mpd_client_send_command (c, "status", NULL); + char *last_version = xstrdup_printf ("%" PRIu32, g_ctx.playlist_version); + mpd_client_send_command (c, "plchanges", last_version, NULL); + free (last_version); mpd_client_list_end (c); mpd_client_add_task (c, mpd_on_info_response, NULL); - mpd_client_idle (c, 0); } @@ -2747,6 +2899,7 @@ main (int argc, char *argv[]) app_prepend_tab (info_tab_init ()); if (g_ctx.streams.len) app_prepend_tab (streams_tab_init ()); + app_prepend_tab (current_tab_init ()); app_switch_tab ((g_ctx.help_tab = help_tab_init ())); g_ctx.polling = true; -- cgit v1.2.3