X-Git-Url: https://git.tokkee.org/?a=blobdiff_plain;f=src%2Fscreen_file.c;h=d55684c3d06cd86d6641e236a9026350516af0d1;hb=63fc23f1c83f52eb47b5511e97bb1fe7cf7d82d5;hp=3aaade12a03896d727d92e34cbc8062c40dc7f6f;hpb=3b68d63a1f1dc53e642bc269bbdd1148ede52a10;p=ncmpc.git diff --git a/src/screen_file.c b/src/screen_file.c index 3aaade1..d55684c 100644 --- a/src/screen_file.c +++ b/src/screen_file.c @@ -1,35 +1,36 @@ /* ncmpc (Ncurses MPD Client) - * (c) 2004-2009 The Music Player Daemon Project + * (c) 2004-2017 The Music Player Daemon Project * Project homepage: http://musicpd.org - + * * 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 Street, Fifth Floor, Boston, MA 02110-1301 USA. -*/ + */ #include "screen_file.h" +#include "screen_browser.h" #include "screen_interface.h" +#include "screen_status.h" +#include "save_playlist.h" +#include "screen.h" #include "config.h" #include "i18n.h" -#include "options.h" #include "charset.h" #include "mpdclient.h" #include "filelist.h" -#include "command.h" #include "screen_utils.h" -#include "screen_browser.h" -#include "screen_play.h" #include "screen_client.h" +#include "options.h" #include @@ -42,13 +43,19 @@ static struct screen_browser browser; static char *current_path; static void -screen_file_paint(void); - -static void -screen_file_repaint(void) +screen_file_load_list(struct mpdclient *c, struct filelist *filelist) { - screen_file_paint(); - wrefresh(browser.lw->w); + struct mpd_connection *connection; + + connection = mpdclient_get_connection(c); + if (connection == NULL) + return; + + mpd_send_list_meta(connection, current_path); + filelist_recv(filelist, connection); + + if (mpdclient_finish_command(c)) + filelist_sort_dir_play(filelist, compare_filelist_entry_path); } static void @@ -57,13 +64,15 @@ screen_file_reload(struct mpdclient *c) if (browser.filelist != NULL) filelist_free(browser.filelist); - browser.filelist = mpdclient_filelist_get(c, current_path); - if (browser.filelist == NULL) - browser.filelist = filelist_new(); - + browser.filelist = filelist_new(); if (*current_path != 0) /* add a dummy entry for ./.. */ - filelist_prepend(browser.filelist, NULL); + filelist_append(browser.filelist, NULL); + + screen_file_load_list(c, browser.filelist); + + list_window_set_length(browser.lw, + filelist_length(browser.filelist)); } /** @@ -77,9 +86,7 @@ change_directory(struct mpdclient *c, const char *new_path) screen_file_reload(c); -#ifndef NCMPC_MINI - sync_highlights(c, browser.filelist); -#endif + screen_browser_sync_highlights(browser.filelist, &c->playlist); list_window_reset(browser.lw); @@ -93,29 +100,24 @@ static bool change_to_parent(struct mpdclient *c) { char *parent = g_path_get_dirname(current_path); - char *old_path; - int idx; - bool success; - if (strcmp(parent, ".") == 0) parent[0] = '\0'; - old_path = current_path; + char *old_path = current_path; current_path = NULL; - success = change_directory(c, parent); + bool success = change_directory(c, parent); g_free(parent); - idx = success + int idx = success ? filelist_find_directory(browser.filelist, old_path) : -1; g_free(old_path); if (success && idx >= 0) { /* set the cursor on the previous working directory */ - list_window_set_selected(browser.lw, idx); - list_window_center(browser.lw, - filelist_length(browser.filelist), idx); + list_window_set_cursor(browser.lw, idx); + list_window_center(browser.lw, idx); } return success; @@ -149,21 +151,19 @@ screen_file_handle_enter(struct mpdclient *c) return change_to_entry(c, entry); } -static int +static void handle_save(struct mpdclient *c) { - struct filelist_entry *entry; + struct list_window_range range; const char *defaultname = NULL; - char *defaultname_utf8 = NULL; - int ret; - unsigned selected; - if (browser.lw->selected >= filelist_length(browser.filelist)) - return -1; + list_window_get_range(browser.lw, &range); + if (range.start == range.end) + return; - for(selected = browser.lw->selected_start; selected <= browser.lw->selected_end; ++selected) - { - entry = filelist_get(browser.filelist, selected); + for (unsigned i = range.start; i < range.end; ++i) { + struct filelist_entry *entry = + filelist_get(browser.filelist, i); if( entry && entry->entity ) { struct mpd_entity *entity = entry->entity; if (mpd_entity_get_type(entity) == MPD_ENTITY_TYPE_PLAYLIST) { @@ -174,34 +174,30 @@ handle_save(struct mpdclient *c) } } + char *defaultname_utf8 = NULL; if(defaultname) defaultname_utf8 = utf8_to_locale(defaultname); - ret = playlist_save(c, NULL, defaultname_utf8); + playlist_save(c, NULL, defaultname_utf8); g_free(defaultname_utf8); - - return ret; } -static int +static void handle_delete(struct mpdclient *c) { - struct filelist_entry *entry; - struct mpd_entity *entity; - const struct mpd_playlist *playlist; - char *str, *buf; - int key; - unsigned selected; - - for(selected = browser.lw->selected_start; selected <= browser.lw->selected_end; ++selected) - { - if (selected >= filelist_length(browser.filelist)) - return -1; - - entry = filelist_get(browser.filelist, selected); + struct mpd_connection *connection = mpdclient_get_connection(c); + + if (connection == NULL) + return; + + struct list_window_range range; + list_window_get_range(browser.lw, &range); + for (unsigned i = range.start; i < range.end; ++i) { + struct filelist_entry *entry = + filelist_get(browser.filelist, i); if( entry==NULL || entry->entity==NULL ) continue; - entity = entry->entity; + struct mpd_entity *entity = entry->entity; if (mpd_entity_get_type(entity) != MPD_ENTITY_TYPE_PLAYLIST) { /* translators: the "delete" command is only possible @@ -212,41 +208,45 @@ handle_delete(struct mpdclient *c) continue; } - playlist = mpd_entity_get_playlist(entity); - str = utf8_to_locale(g_basename(mpd_playlist_get_path(playlist))); - buf = g_strdup_printf(_("Delete playlist %s [%s/%s] ? "), str, YES, NO); + const struct mpd_playlist *playlist = mpd_entity_get_playlist(entity); + char *str = utf8_to_locale(g_basename(mpd_playlist_get_path(playlist))); + char *buf = g_strdup_printf(_("Delete playlist %s?"), str); g_free(str); - key = tolower(screen_getch(buf)); + bool delete = screen_get_yesno(buf, false); g_free(buf); - if( key != YES[0] ) { + + if (!delete) { /* translators: a dialog was aborted by the user */ screen_status_printf(_("Aborted")); - return 0; + return; } - if (mpdclient_cmd_delete_playlist(c, mpd_playlist_get_path(playlist))) - continue; + if (!mpd_run_rm(connection, mpd_playlist_get_path(playlist))) { + mpdclient_handle_error(c); + break; + } + + c->events |= MPD_IDLE_STORED_PLAYLIST; /* translators: MPD deleted the playlist, as requested by the user */ screen_status_printf(_("Playlist deleted")); } - return 0; } static void -screen_file_init(WINDOW *w, int cols, int rows) +screen_file_init(WINDOW *w, unsigned cols, unsigned rows) { current_path = g_strdup(""); browser.lw = list_window_init(w, cols, rows); + browser.song_format = options.list_format; } static void -screen_file_resize(int cols, int rows) +screen_file_resize(unsigned cols, unsigned rows) { - browser.lw->cols = cols; - browser.lw->rows = rows; + list_window_resize(browser.lw, cols, rows); } static void @@ -263,13 +263,13 @@ static void screen_file_open(struct mpdclient *c) { screen_file_reload(c); + screen_browser_sync_highlights(browser.filelist, &c->playlist); } static const char * screen_file_get_title(char *str, size_t size) { const char *path = NULL, *prev = NULL, *slash = current_path; - char *path_locale; /* determine the last 2 parts of the path */ while ((slash = strchr(slash, '/')) != NULL) { @@ -281,7 +281,7 @@ screen_file_get_title(char *str, size_t size) /* fall back to full path */ path = current_path; - path_locale = utf8_to_locale(path); + char *path_locale = utf8_to_locale(path); g_snprintf(str, size, "%s: %s", /* translators: caption of the browser screen */ _("Browse"), path_locale); @@ -292,7 +292,7 @@ screen_file_get_title(char *str, size_t size) static void screen_file_paint(void) { - list_window_paint(browser.lw, browser_lw_callback, browser.filelist); + screen_browser_paint(&browser); } static void @@ -301,22 +301,16 @@ screen_file_update(struct mpdclient *c) if (c->events & (MPD_IDLE_DATABASE | MPD_IDLE_STORED_PLAYLIST)) { /* the db has changed -> update the filelist */ screen_file_reload(c); - list_window_check_selected(browser.lw, - filelist_length(browser.filelist)); } -#ifndef NCMPC_MINI - if (c->events & (MPD_IDLE_DATABASE | MPD_IDLE_STORED_PLAYLIST | - MPD_IDLE_PLAYLIST)) - sync_highlights(c, browser.filelist); -#endif - if (c->events & (MPD_IDLE_DATABASE | MPD_IDLE_STORED_PLAYLIST #ifndef NCMPC_MINI - | MPD_IDLE_PLAYLIST + | MPD_IDLE_QUEUE #endif - )) - screen_file_repaint(); + )) { + screen_browser_sync_highlights(browser.filelist, &c->playlist); + screen_file_paint(); + } } static bool @@ -325,7 +319,7 @@ screen_file_cmd(struct mpdclient *c, command_t cmd) switch(cmd) { case CMD_PLAY: if (screen_file_handle_enter(c)) { - screen_file_repaint(); + screen_file_paint(); return true; } @@ -333,11 +327,11 @@ screen_file_cmd(struct mpdclient *c, command_t cmd) case CMD_GO_ROOT_DIRECTORY: change_directory(c, ""); - screen_file_repaint(); + screen_file_paint(); return true; case CMD_GO_PARENT_DIRECTORY: change_to_parent(c); - screen_file_repaint(); + screen_file_paint(); return true; case CMD_LOCATE: @@ -346,27 +340,36 @@ screen_file_cmd(struct mpdclient *c, command_t cmd) segmentation fault in the current implementation */ return false; + case CMD_SCREEN_UPDATE: + screen_file_reload(c); + screen_browser_sync_highlights(browser.filelist, &c->playlist); + screen_file_paint(); + return false; + + default: + break; + } + + if (browser_cmd(&browser, c, cmd)) { + if (screen_is_visible(&screen_browse)) + screen_file_paint(); + return true; + } + + if (!mpdclient_is_connected(c)) + return false; + + switch(cmd) { case CMD_DELETE: handle_delete(c); - screen_file_repaint(); + screen_file_paint(); break; + case CMD_SAVE_PLAYLIST: handle_save(c); break; - case CMD_SCREEN_UPDATE: - screen_file_reload(c); -#ifndef NCMPC_MINI - sync_highlights(c, browser.filelist); -#endif - list_window_check_selected(browser.lw, - filelist_length(browser.filelist)); - screen_file_repaint(); - return false; case CMD_DB_UPDATE: - if (c->status == NULL) - return true; - screen_database_update(c, current_path); return true; @@ -374,14 +377,22 @@ screen_file_cmd(struct mpdclient *c, command_t cmd) break; } - if (browser_cmd(&browser, c, cmd)) { + return false; +} + +#ifdef HAVE_GETMOUSE +static bool +screen_file_mouse(struct mpdclient *c, int x, int y, mmask_t bstate) +{ + if (browser_mouse(&browser, c, x, y, bstate)) { if (screen_is_visible(&screen_browse)) - screen_file_repaint(); + screen_file_paint(); return true; } return false; } +#endif const struct screen_functions screen_browse = { .init = screen_file_init, @@ -391,6 +402,9 @@ const struct screen_functions screen_browse = { .paint = screen_file_paint, .update = screen_file_update, .cmd = screen_file_cmd, +#ifdef HAVE_GETMOUSE + .mouse = screen_file_mouse, +#endif .get_title = screen_file_get_title, }; @@ -399,8 +413,6 @@ screen_file_goto_song(struct mpdclient *c, const struct mpd_song *song) { const char *uri, *slash, *parent; char *allocated = NULL; - bool ret; - int i; assert(song != NULL); @@ -418,18 +430,18 @@ screen_file_goto_song(struct mpdclient *c, const struct mpd_song *song) else parent = ""; - ret = change_directory(c, parent); + bool ret = change_directory(c, parent); g_free(allocated); if (!ret) return false; /* select the specified song */ - i = filelist_find_song(browser.filelist, song); + int i = filelist_find_song(browser.filelist, song); if (i < 0) i = 0; - list_window_set_selected(browser.lw, i); + list_window_set_cursor(browser.lw, i); /* finally, switch to the file screen */ screen_switch(&screen_browse, c);