X-Git-Url: https://git.tokkee.org/?a=blobdiff_plain;f=src%2Fscreen_file.c;h=ab2af071321c506c03d55ff164a58e2f7a421f25;hb=08c055113f2eacd42fde57b952b548ca45e29eb2;hp=8b7577fdfde71c5a4cefe41b0a1dc4b0342edf10;hpb=a14b635343ebdcb711885d840687d181ebd2bfbb;p=ncmpc.git diff --git a/src/screen_file.c b/src/screen_file.c index 8b7577f..ab2af07 100644 --- a/src/screen_file.c +++ b/src/screen_file.c @@ -18,485 +18,752 @@ * */ -#include -#include -#include -#include -#include - #include "config.h" #include "ncmpc.h" +#include "options.h" #include "support.h" -#include "libmpdclient.h" -#include "mpc.h" +#include "mpdclient.h" +#include "strfsong.h" #include "command.h" #include "screen.h" #include "screen_utils.h" +#include "screen_browse.h" #include "screen_play.h" -#include "screen_file.h" -#define BUFSIZE 1024 -#define TITLESIZE 256 +#include +#include +#include +#include +#include #define USE_OLD_LAYOUT +#undef USE_OLD_ADD -static list_window_t *lw; -static mpd_client_t *mpc = NULL; +#define BUFSIZE 1024 -static char * -list_callback(int index, int *highlight, void *data) -{ - static char buf[BUFSIZE]; - mpd_client_t *c = (mpd_client_t *) data; - filelist_entry_t *entry; - mpd_InfoEntity *entity; +#define HIGHLIGHT (0x01) - *highlight = 0; - if( (entry=(filelist_entry_t *) g_list_nth_data(c->filelist, index))==NULL ) - return NULL; - entity = entry->entity; - *highlight = entry->selected; +static list_window_t *lw = NULL; +static list_window_state_t *lw_state = NULL; +static mpdclient_filelist_t *filelist = NULL; - if( entity == NULL ) - { -#ifdef USE_OLD_LAYOUT - return "[..]"; -#else - return "d .."; -#endif - } - if( entity->type==MPD_INFO_ENTITY_TYPE_DIRECTORY ) - { - mpd_Directory *dir = entity->info.directory; - char *dirname = utf8_to_locale(basename(dir->path)); -#ifdef USE_OLD_LAYOUT - snprintf(buf, BUFSIZE, "[%s]", dirname); -#else - snprintf(buf, BUFSIZE, "d %s", dirname); -#endif - g_free(dirname); - return buf; - } - else if( entity->type==MPD_INFO_ENTITY_TYPE_SONG ) - { - mpd_Song *song = entity->info.song; - -#ifdef USE_OLD_LAYOUT - return mpc_get_song_name(song); -#else - snprintf(buf, BUFSIZE, "m %s", mpc_get_song_name(song)); - return buf; -#endif +/* clear the highlight flag for all items in the filelist */ +void +clear_highlights(mpdclient_filelist_t *fl) +{ + GList *list = g_list_first(fl->list); - } - else if( entity->type==MPD_INFO_ENTITY_TYPE_PLAYLISTFILE ) - { - mpd_PlaylistFile *plf = entity->info.playlistFile; - char *filename = utf8_to_locale(basename(plf->path)); + while( list ) { + filelist_entry_t *entry = list->data; -#ifdef USE_OLD_LAYOUT - snprintf(buf, BUFSIZE, "*%s*", filename); -#else - snprintf(buf, BUFSIZE, "p %s", filename); -#endif - g_free(filename); - return buf; - } - return "Error: Unknow entry!"; + entry->flags &= ~HIGHLIGHT; + list = list->next; + } } -static int -change_directory(screen_t *screen, mpd_client_t *c, filelist_entry_t *entry) +/* change the highlight flag for a song */ +void +set_highlight(mpdclient_filelist_t *fl, mpd_Song *song, int highlight) { - mpd_InfoEntity *entity = entry->entity; + GList *list = g_list_first(fl->list); + + if( !song ) + return; + + while( list ) { + filelist_entry_t *entry = list->data; + mpd_InfoEntity *entity = entry->entity; + + if( entity && entity->type==MPD_INFO_ENTITY_TYPE_SONG ) { + mpd_Song *song2 = entity->info.song; + + if( strcmp(song->file, song2->file) == 0 ) { + if(highlight) + entry->flags |= HIGHLIGHT; + else + entry->flags &= ~HIGHLIGHT; + } + } + list = list->next; + } +} - if( entity==NULL ) - { - char *parent = g_path_get_dirname(c->cwd); +/* sync highlight flags with playlist */ +void +sync_highlights(mpdclient_t *c, mpdclient_filelist_t *fl) +{ + GList *list = g_list_first(fl->list); - if( strcmp(parent,".") == 0 ) - { - parent[0] = '\0'; + while(list) { + filelist_entry_t *entry = list->data; + mpd_InfoEntity *entity = entry->entity; + + if( entity && entity->type==MPD_INFO_ENTITY_TYPE_SONG ) { + mpd_Song *song = entity->info.song; + + if( playlist_get_index_from_file(c, song->file) >= 0 ) + entry->flags |= HIGHLIGHT; + else + entry->flags &= ~HIGHLIGHT; + } + list=list->next; } - if( c->cwd ) - g_free(c->cwd); - c->cwd = parent; - } - else - if( entity->type==MPD_INFO_ENTITY_TYPE_DIRECTORY) - { - mpd_Directory *dir = entity->info.directory; - if( c->cwd ) - g_free(c->cwd); - c->cwd = g_strdup(dir->path); - } - else - return -1; - - mpc_update_filelist(c); - list_window_reset(lw); - return 0; } -static int -load_playlist(screen_t *screen, mpd_client_t *c, filelist_entry_t *entry) +/* the db have changed -> update the filelist */ +static void +file_changed_callback(mpdclient_t *c, int event, gpointer data) { - mpd_InfoEntity *entity = entry->entity; - mpd_PlaylistFile *plf = entity->info.playlistFile; - char *filename = utf8_to_locale(basename(plf->path)); + D("screen_file.c> filelist_callback() [%d]\n", event); + filelist = mpdclient_filelist_update(c, filelist); + sync_highlights(c, filelist); + list_window_check_selected(lw, filelist->length); +} - mpd_sendLoadCommand(c->connection, plf->path); - mpd_finishCommand(c->connection); +/* the playlist have been updated -> fix highlights */ +static void +playlist_changed_callback(mpdclient_t *c, int event, gpointer data) +{ + D("screen_file.c> playlist_callback() [%d]\n", event); + switch(event) { + case PLAYLIST_EVENT_CLEAR: + clear_highlights(filelist); + break; + case PLAYLIST_EVENT_ADD: + set_highlight(filelist, (mpd_Song *) data, 1); + break; + case PLAYLIST_EVENT_DELETE: + set_highlight(filelist, (mpd_Song *) data, 0); + break; + case PLAYLIST_EVENT_MOVE: + break; + default: + sync_highlights(c, filelist); + break; + } +} - screen_status_printf(_("Loading playlist %s..."), filename); - g_free(filename); - return 0; +/* list_window callback */ +const char * +browse_lw_callback(int idx, int *highlight, void *data) +{ + static char buf[BUFSIZE]; + mpdclient_filelist_t *fl = (mpdclient_filelist_t *) data; + filelist_entry_t *entry; + mpd_InfoEntity *entity; + + *highlight = 0; + if( (entry=(filelist_entry_t *)g_list_nth_data(fl->list,idx))==NULL ) + return NULL; + + entity = entry->entity; + *highlight = (entry->flags & HIGHLIGHT); + + if( entity == NULL ) + return "[..]"; + + if( entity->type==MPD_INFO_ENTITY_TYPE_DIRECTORY ) { + mpd_Directory *dir = entity->info.directory; + char *directory = utf8_to_locale(basename(dir->path)); + + g_snprintf(buf, BUFSIZE, "[%s]", directory); + g_free(directory); + return buf; + } else if( entity->type==MPD_INFO_ENTITY_TYPE_SONG ) { + mpd_Song *song = entity->info.song; + + strfsong(buf, BUFSIZE, LIST_FORMAT, song); + return buf; + } else if( entity->type==MPD_INFO_ENTITY_TYPE_PLAYLISTFILE ) { + mpd_PlaylistFile *plf = entity->info.playlistFile; + char *filename = utf8_to_locale(basename(plf->path)); + +#ifdef USE_OLD_LAYOUT + g_snprintf(buf, BUFSIZE, "*%s*", filename); +#else + g_snprintf(buf, BUFSIZE, " %s", filename); +#endif + g_free(filename); + return buf; + } + + return "Error: Unknown entry!"; } -static int -handle_delete(screen_t *screen, mpd_client_t *c) +/* chdir */ +static int +change_directory(screen_t *screen, mpdclient_t *c, filelist_entry_t *entry, + const char *new_path) { - filelist_entry_t *entry; - mpd_InfoEntity *entity; - mpd_PlaylistFile *plf; - char *str, buf[BUFSIZE]; - int key; - - entry = ( filelist_entry_t *) g_list_nth_data(c->filelist, lw->selected); - if( entry==NULL || entry->entity==NULL ) - return -1; - - entity = entry->entity; - - if( entity->type!=MPD_INFO_ENTITY_TYPE_PLAYLISTFILE ) - { - screen_status_printf(_("You can only delete playlists!")); - beep(); - return -1; - } - - plf = entity->info.playlistFile; - str = utf8_to_locale(basename(plf->path)); - snprintf(buf, BUFSIZE, _("Delete playlist %s [y/n] ? "), str); - g_free(str); - key = tolower(screen_getch(screen->status_window.w, buf)); - if( key==KEY_RESIZE ) - screen_resize(); - if( key!='y' ) - { - screen_status_printf(_("Aborted!")); - return 0; - } - - mpd_sendRmCommand(c->connection, plf->path); - mpd_finishCommand(c->connection); - if( mpc_error(c)) - { - str = utf8_to_locale(mpc_error_str(c)); - screen_status_printf("Error: %s", str); - g_free(str); - beep(); - return -1; - } - screen_status_printf(_("Playlist deleted!")); - mpc_update_filelist(c); - list_window_check_selected(lw, c->filelist_length); - return 0; + mpd_InfoEntity *entity = NULL; + gchar *path = NULL; + + if( entry!=NULL ) + entity = entry->entity; + else if( new_path==NULL ) + return -1; + + if( entity==NULL ) { + if( entry || 0==strcmp(new_path, "..") ) { + /* return to parent */ + char *parent = g_path_get_dirname(filelist->path); + if( strcmp(parent, ".") == 0 ) + parent[0] = '\0'; + path = g_strdup(parent); + list_window_reset(lw); + /* restore previous list window state */ + list_window_pop_state(lw_state,lw); + } else { + /* entry==NULL, then new_path ("" is root) */ + path = g_strdup(new_path); + list_window_reset(lw); + /* restore first list window state (pop while returning true) */ + while(list_window_pop_state(lw_state,lw)); + } + } else if( entity->type==MPD_INFO_ENTITY_TYPE_DIRECTORY) { + /* enter sub */ + mpd_Directory *dir = entity->info.directory; + path = utf8_to_locale(dir->path); + /* save current list window state */ + list_window_push_state(lw_state,lw); + } else + return -1; + + filelist = mpdclient_filelist_free(filelist); + filelist = mpdclient_filelist_get(c, path); + sync_highlights(c, filelist); + list_window_check_selected(lw, filelist->length); + g_free(path); + return 0; } +static int +load_playlist(screen_t *screen, mpdclient_t *c, filelist_entry_t *entry) +{ + mpd_InfoEntity *entity = entry->entity; + mpd_PlaylistFile *plf = entity->info.playlistFile; + char *filename = utf8_to_locale(plf->path); + + if( mpdclient_cmd_load_playlist_utf8(c, plf->path) == 0 ) + screen_status_printf(_("Loading playlist %s..."), basename(filename)); + g_free(filename); + return 0; +} static int -handle_play_cmd(screen_t *screen, mpd_client_t *c) +handle_save(screen_t *screen, mpdclient_t *c) { - filelist_entry_t *entry; - mpd_InfoEntity *entity; - - entry = ( filelist_entry_t *) g_list_nth_data(c->filelist, lw->selected); - if( entry==NULL ) - return -1; - - entity = entry->entity; - if( entity==NULL || entity->type==MPD_INFO_ENTITY_TYPE_DIRECTORY ) - return change_directory(screen, c, entry); - else if( entity->type==MPD_INFO_ENTITY_TYPE_PLAYLISTFILE ) - return load_playlist(screen, c, entry); - return -1; + filelist_entry_t *entry; + char *defaultname = NULL; + + + entry=( filelist_entry_t *) g_list_nth_data(filelist->list,lw->selected); + if( entry && entry->entity ) { + mpd_InfoEntity *entity = entry->entity; + if( entity->type==MPD_INFO_ENTITY_TYPE_PLAYLISTFILE ) { + mpd_PlaylistFile *plf = entity->info.playlistFile; + defaultname = plf->path; + } + } + + return playlist_save(screen, c, NULL, defaultname); } static int -add_directory(mpd_client_t *c, char *dir) +handle_delete(screen_t *screen, mpdclient_t *c) { - mpd_InfoEntity *entity; - GList *subdir_list = NULL; - GList *list = NULL; - char *dirname; - - dirname = utf8_to_locale(dir); - screen_status_printf(_("Adding directory %s...\n"), dirname); - doupdate(); - g_free(dirname); - dirname = NULL; - - mpd_sendLsInfoCommand(c->connection, dir); - mpd_sendCommandListBegin(c->connection); - while( (entity=mpd_getNextInfoEntity(c->connection)) ) - { - if( entity->type==MPD_INFO_ENTITY_TYPE_SONG ) - { - mpd_Song *song = entity->info.song; - mpd_sendAddCommand(c->connection, song->file); - mpd_freeInfoEntity(entity); + filelist_entry_t *entry; + mpd_InfoEntity *entity; + mpd_PlaylistFile *plf; + char *str, *buf; + int key; + + entry=( filelist_entry_t *) g_list_nth_data(filelist->list,lw->selected); + if( entry==NULL || entry->entity==NULL ) + return -1; + + entity = entry->entity; + + if( entity->type!=MPD_INFO_ENTITY_TYPE_PLAYLISTFILE ) { + screen_status_printf(_("You can only delete playlists!")); + screen_bell(); + return -1; } - else if( entity->type==MPD_INFO_ENTITY_TYPE_DIRECTORY ) - { - subdir_list = g_list_append(subdir_list, (gpointer) entity); + + plf = entity->info.playlistFile; + str = utf8_to_locale(basename(plf->path)); + buf = g_strdup_printf(_("Delete playlist %s [%s/%s] ? "), str, YES, NO); + g_free(str); + key = tolower(screen_getch(screen->status_window.w, buf)); + g_free(buf); + if( key==KEY_RESIZE ) + screen_resize(); + if( key != YES[0] ) { + screen_status_printf(_("Aborted!")); + return 0; } - else - mpd_freeInfoEntity(entity); - } - mpd_sendCommandListEnd(c->connection); - mpd_finishCommand(c->connection); - - list = g_list_first(subdir_list); - while( list!=NULL ) - { - mpd_Directory *dir; - - entity = list->data; - dir = entity->info.directory; - add_directory(c, dir->path); - mpd_freeInfoEntity(entity); - list->data=NULL; - list=list->next; - } - g_list_free(subdir_list); - return 0; + + if( mpdclient_cmd_delete_playlist_utf8(c, plf->path) ) + return -1; + + screen_status_printf(_("Playlist deleted!")); + return 0; } static int -handle_select(screen_t *screen, mpd_client_t *c) +enqueue_and_play(screen_t *screen, mpdclient_t *c, filelist_entry_t *entry) { - filelist_entry_t *entry; + int idx; + mpd_InfoEntity *entity = entry->entity; + mpd_Song *song = entity->info.song; + + if(!( entry->flags & HIGHLIGHT )) { + if( mpdclient_cmd_add(c, song) == 0 ) { + char buf[BUFSIZE]; + + entry->flags |= HIGHLIGHT; + strfsong(buf, BUFSIZE, LIST_FORMAT, song); + screen_status_printf(_("Adding \'%s\' to playlist\n"), buf); + mpdclient_update(c); /* get song id */ + } else + return -1; + } - entry = ( filelist_entry_t *) g_list_nth_data(c->filelist, lw->selected); - if( entry==NULL || entry->entity==NULL) - return -1; + idx = playlist_get_index_from_file(c, song->file); + mpdclient_cmd_play(c, idx); + return 0; +} - if( entry->entity->type==MPD_INFO_ENTITY_TYPE_DIRECTORY ) - { - mpd_Directory *dir = entry->entity->info.directory; - add_directory(c, dir->path); - return 0; - } +int +browse_handle_enter(screen_t *screen, + mpdclient_t *c, + list_window_t *local_lw, + mpdclient_filelist_t *fl) +{ + filelist_entry_t *entry; + mpd_InfoEntity *entity; + + if ( fl==NULL ) + return -1; + entry = ( filelist_entry_t *) g_list_nth_data(fl->list, local_lw->selected); + if( entry==NULL ) + return -1; + + entity = entry->entity; + if( entity==NULL || entity->type==MPD_INFO_ENTITY_TYPE_DIRECTORY ) + return change_directory(screen, c, entry, NULL); + else if( entity->type==MPD_INFO_ENTITY_TYPE_PLAYLISTFILE ) + return load_playlist(screen, c, entry); + else if( entity->type==MPD_INFO_ENTITY_TYPE_SONG ) + return enqueue_and_play(screen, c, entry); + return -1; +} - if( entry->entity->type!=MPD_INFO_ENTITY_TYPE_SONG ) - return -1; - entry->selected = !entry->selected; +#ifdef USE_OLD_ADD +/* NOTE - The add_directory functions should move to mpdclient.c */ +extern gint mpdclient_finish_command(mpdclient_t *c); - if( entry->selected ) - { - if( entry->entity->type==MPD_INFO_ENTITY_TYPE_SONG ) - { - mpd_Song *song = entry->entity->info.song; +static int +add_directory(mpdclient_t *c, char *dir) +{ + mpd_InfoEntity *entity; + GList *subdir_list = NULL; + GList *list = NULL; + char *dirname; + + dirname = utf8_to_locale(dir); + screen_status_printf(_("Adding directory %s...\n"), dirname); + doupdate(); + g_free(dirname); + dirname = NULL; + + mpd_sendLsInfoCommand(c->connection, dir); + mpd_sendCommandListBegin(c->connection); + while( (entity=mpd_getNextInfoEntity(c->connection)) ) { + if( entity->type==MPD_INFO_ENTITY_TYPE_SONG ) { + mpd_Song *song = entity->info.song; + mpd_sendAddCommand(c->connection, song->file); + mpd_freeInfoEntity(entity); + } else if( entity->type==MPD_INFO_ENTITY_TYPE_DIRECTORY ) { + subdir_list = g_list_append(subdir_list, (gpointer) entity); + } else + mpd_freeInfoEntity(entity); + } + mpd_sendCommandListEnd(c->connection); + mpdclient_finish_command(c); + c->need_update = TRUE; + + list = g_list_first(subdir_list); + while( list!=NULL ) { + mpd_Directory *dir; + + entity = list->data; + dir = entity->info.directory; + add_directory(c, dir->path); + mpd_freeInfoEntity(entity); + list->data=NULL; + list=list->next; + } + g_list_free(subdir_list); + return 0; +} +#endif - playlist_add_song(c, song); +int +browse_handle_select(screen_t *screen, + mpdclient_t *c, + list_window_t *local_lw, + mpdclient_filelist_t *fl) +{ + filelist_entry_t *entry; + + if ( fl==NULL ) + return -1; + entry=( filelist_entry_t *) g_list_nth_data(fl->list, + local_lw->selected); + if( entry==NULL || entry->entity==NULL) + return -1; + + if( entry->entity->type==MPD_INFO_ENTITY_TYPE_PLAYLISTFILE ) + return load_playlist(screen, c, entry); + + if( entry->entity->type==MPD_INFO_ENTITY_TYPE_DIRECTORY ) { + mpd_Directory *dir = entry->entity->info.directory; +#ifdef USE_OLD_ADD + add_directory(c, tmp); +#else + if( mpdclient_cmd_add_path_utf8(c, dir->path) == 0 ) { + char *tmp = utf8_to_locale(dir->path); + + screen_status_printf(_("Adding \'%s\' to playlist\n"), tmp); + g_free(tmp); + } +#endif + return 0; + } - screen_status_printf(_("Adding \'%s\' to playlist\n"), - mpc_get_song_name(song)); + if( entry->entity->type!=MPD_INFO_ENTITY_TYPE_SONG ) + return -1; + + if( entry->flags & HIGHLIGHT ) + entry->flags &= ~HIGHLIGHT; + else + entry->flags |= HIGHLIGHT; + + if( entry->flags & HIGHLIGHT ) { + if( entry->entity->type==MPD_INFO_ENTITY_TYPE_SONG ) { + mpd_Song *song = entry->entity->info.song; + + if( mpdclient_cmd_add(c, song) == 0 ) { + char buf[BUFSIZE]; + + strfsong(buf, BUFSIZE, LIST_FORMAT, song); + screen_status_printf(_("Adding \'%s\' to playlist\n"), buf); + } + } + } else { + /* remove song from playlist */ + if( entry->entity->type==MPD_INFO_ENTITY_TYPE_SONG ) { + mpd_Song *song = entry->entity->info.song; + + if( song ) { + int idx = playlist_get_index_from_file(c, song->file); + + while( (idx=playlist_get_index_from_file(c, song->file))>=0 ) + mpdclient_cmd_delete(c, idx); + } + } } - } - else - { - /* remove song from playlist */ - if( entry->entity->type==MPD_INFO_ENTITY_TYPE_SONG ) - { - mpd_Song *song = entry->entity->info.song; - - if( song ) - { - int index = mpc_playlist_get_song_index(c, song->file); - - while( (index=mpc_playlist_get_song_index(c, song->file))>=0 ) - playlist_delete_song(c, index); - } + return 0; +} + +int +browse_handle_select_all (screen_t *screen, + mpdclient_t *c, + list_window_t *local_lw, + mpdclient_filelist_t *fl) +{ + filelist_entry_t *entry; + GList *temp = fl->list; + + if ( fl==NULL ) + return -1; + for (fl->list = g_list_first(fl->list); + fl->list; + fl->list = g_list_next(fl->list)) { + entry=( filelist_entry_t *) fl->list->data; + if( entry==NULL || entry->entity==NULL) + return -1; + + if( entry->entity->type==MPD_INFO_ENTITY_TYPE_PLAYLISTFILE ) + load_playlist(screen, c, entry); + + if( entry->entity->type==MPD_INFO_ENTITY_TYPE_DIRECTORY ) { + mpd_Directory *dir = entry->entity->info.directory; +#ifdef USE_OLD_ADD + add_directory(c, tmp); +#else + if (mpdclient_cmd_add_path_utf8(c, dir->path) == 0) { + char *tmp = utf8_to_locale(dir->path); + + screen_status_printf(_("Adding \'%s\' to playlist\n"), tmp); + g_free(tmp); + } +#endif + } + + if( entry->entity->type!=MPD_INFO_ENTITY_TYPE_SONG ) + continue; + + entry->flags |= HIGHLIGHT; + + if( entry->flags & HIGHLIGHT ) { + if( entry->entity->type==MPD_INFO_ENTITY_TYPE_SONG ) { + mpd_Song *song = entry->entity->info.song; + + if( mpdclient_cmd_add(c, song) == 0 ) { + char buf[BUFSIZE]; + + strfsong(buf, BUFSIZE, LIST_FORMAT, song); + screen_status_printf(_("Adding \'%s\' to playlist\n"), buf); + } + } + } + /* + else { + //remove song from playlist + if( entry->entity->type==MPD_INFO_ENTITY_TYPE_SONG ) { + mpd_Song *song = entry->entity->info.song; + + if( song ) { + int idx = playlist_get_index_from_file(c, song->file); + + while( (idx=playlist_get_index_from_file(c, song->file))>=0 ) + mpdclient_cmd_delete(c, idx); + } + } + } + */ + return 0; } - } - return 0; + + fl->list = temp; + return 0; } static void -file_init(WINDOW *w, int cols, int rows) +browse_init(WINDOW *w, int cols, int rows) { - lw = list_window_init(w, cols, rows); + lw = list_window_init(w, cols, rows); + lw_state = list_window_init_state(); } static void -file_resize(int cols, int rows) +browse_resize(int cols, int rows) { - lw->cols = cols; - lw->rows = rows; + lw->cols = cols; + lw->rows = rows; } static void -file_exit(void) +browse_exit(void) { - list_window_free(lw); + if( filelist ) + filelist = mpdclient_filelist_free(filelist); + lw = list_window_free(lw); + lw_state = list_window_free_state(lw_state); } -static void -file_open(screen_t *screen, mpd_client_t *c) +static void +browse_open(screen_t *screen, mpdclient_t *c) { - if( c->filelist == NULL ) - { - mpc_update_filelist(c); - } - mpc = c; + if( filelist == NULL ) { + filelist = mpdclient_filelist_get(c, ""); + mpdclient_install_playlist_callback(c, playlist_changed_callback); + mpdclient_install_browse_callback(c, file_changed_callback); + } } static void -file_close(void) +browse_close(void) { } -static char * -file_title(char *str, size_t size) +static const char * +browse_title(char *str, size_t size) { - char *tmp; + char *pathcopy; + char *parentdir; - tmp = utf8_to_locale(basename(mpc->cwd)); - snprintf(str, size, _("Browse: %s"), tmp); - g_free(tmp); + pathcopy = strdup(filelist->path); + parentdir = dirname(pathcopy); + parentdir = basename(parentdir); - return str; -} + if( parentdir[0] == '.' && strlen(parentdir) == 1 ) { + parentdir = NULL; + } -static void -file_paint(screen_t *screen, mpd_client_t *c) -{ - lw->clear = 1; - - list_window_paint(lw, list_callback, (void *) c); - wnoutrefresh(lw->w); + g_snprintf(str, size, _("Browse: %s%s%s"), + parentdir ? parentdir : "", + parentdir ? "/" : "", + basename(filelist->path)); + free(pathcopy); + return str; } -static void -file_update(screen_t *screen, mpd_client_t *c) +static void +browse_paint(screen_t *screen, mpdclient_t *c) { - if( c->filelist_updated ) - { - file_paint(screen, c); - c->filelist_updated = 0; - return; - } - list_window_paint(lw, list_callback, (void *) c); - wnoutrefresh(lw->w); -} + lw->clear = 1; + list_window_paint(lw, browse_lw_callback, (void *) filelist); + wnoutrefresh(lw->w); +} -static int -file_cmd(screen_t *screen, mpd_client_t *c, command_t cmd) +static void +browse_update(screen_t *screen, mpdclient_t *c) { - switch(cmd) - { - case CMD_PLAY: - handle_play_cmd(screen, c); - return 1; - case CMD_SELECT: - if( handle_select(screen, c) == 0 ) - { - /* continue and select next item... */ - cmd = CMD_LIST_NEXT; + if( filelist->updated ) { + browse_paint(screen, c); + filelist->updated = FALSE; + return; } - break; - case CMD_DELETE: - handle_delete(screen, c); - break; - case CMD_SCREEN_UPDATE: - mpc_update_filelist(c); - list_window_check_selected(lw, c->filelist_length); - screen_status_printf(_("Screen updated!")); - return 1; - case CMD_LIST_FIND: - case CMD_LIST_RFIND: - case CMD_LIST_FIND_NEXT: - case CMD_LIST_RFIND_NEXT: - return screen_find(screen, c, - lw, c->filelist_length, - cmd, list_callback); - default: - break; - } - return list_window_cmd(lw, c->filelist_length, cmd); + + list_window_paint(lw, browse_lw_callback, (void *) filelist); + wnoutrefresh(lw->w); } -list_window_t * -get_filelist_window() +#ifdef HAVE_GETMOUSE +int +browse_handle_mouse_event(screen_t *screen, + mpdclient_t *c, + list_window_t *local_lw, + mpdclient_filelist_t *fl) { - return lw; -} + int row; + int prev_selected = local_lw->selected; + unsigned long bstate; + int length; + + if ( fl ) + length = fl->length; + else + length = 0; + + if( screen_get_mouse_event(c, local_lw, length, &bstate, &row) ) + return 1; + + local_lw->selected = local_lw->start+row; + list_window_check_selected(local_lw, length); + + if( bstate & BUTTON1_CLICKED ) { + if( prev_selected == local_lw->selected ) + browse_handle_enter(screen, c, local_lw, fl); + } else if( bstate & BUTTON3_CLICKED ) { + if( prev_selected == local_lw->selected ) + browse_handle_select(screen, c, local_lw, fl); + } + return 1; +} +#endif -void -file_clear_highlights(mpd_client_t *c) +static int +browse_cmd(screen_t *screen, mpdclient_t *c, command_t cmd) { - GList *list = g_list_first(c->filelist); - - while( list ) - { - filelist_entry_t *entry = list->data; - - entry->selected = 0; - list = list->next; - } + switch(cmd) + { + case CMD_PLAY: + browse_handle_enter(screen, c, lw, filelist); + return 1; + case CMD_GO_ROOT_DIRECTORY: + return change_directory(screen, c, NULL, ""); + break; + case CMD_GO_PARENT_DIRECTORY: + return change_directory(screen, c, NULL, ".."); + break; + case CMD_SELECT: + if( browse_handle_select(screen, c, lw, filelist) == 0 ) + { + /* continue and select next item... */ + cmd = CMD_LIST_NEXT; + } + break; + case CMD_DELETE: + handle_delete(screen, c); + break; + case CMD_SAVE_PLAYLIST: + handle_save(screen, c); + break; + case CMD_SCREEN_UPDATE: + screen->painted = 0; + lw->clear = 1; + lw->repaint = 1; + filelist = mpdclient_filelist_update(c, filelist); + list_window_check_selected(lw, filelist->length); + screen_status_printf(_("Screen updated!")); + return 1; + case CMD_DB_UPDATE: + if( !c->status->updatingDb ) + { + if( mpdclient_cmd_db_update_utf8(c,filelist->path)==0 ) + { + if(strcmp(filelist->path,"")) { + screen_status_printf(_("Database update of %s started!"), + filelist->path); + } else { + screen_status_printf(_("Database update started!")); + } + /* set updatingDb to make shure the browse callback gets called + * even if the updated has finished before status is updated */ + c->status->updatingDb = 1; + } + } + else + screen_status_printf(_("Database update running...")); + return 1; + case CMD_LIST_FIND: + case CMD_LIST_RFIND: + case CMD_LIST_FIND_NEXT: + case CMD_LIST_RFIND_NEXT: + return screen_find(screen, + lw, filelist->length, + cmd, browse_lw_callback, (void *) filelist); + case CMD_MOUSE_EVENT: + return browse_handle_mouse_event(screen,c,lw,filelist); + default: + break; + } + return list_window_cmd(lw, filelist->length, cmd); } -void -file_set_highlight(mpd_client_t *c, mpd_Song *song, int highlight) +static list_window_t * +get_filelist_window(void) { - GList *list = g_list_first(c->filelist); - - if( !song ) - return; - - while( list ) - { - filelist_entry_t *entry = list->data; - mpd_InfoEntity *entity = entry->entity; - - if( entity && entity->type==MPD_INFO_ENTITY_TYPE_SONG ) - { - mpd_Song *song2 = entity->info.song; - - if( strcmp(song->file, song2->file) == 0 ) - { - entry->selected = highlight; - } - } - list = list->next; - } + return lw; } screen_functions_t * -get_screen_file(void) +get_screen_browse(void) { - static screen_functions_t functions; - - memset(&functions, 0, sizeof(screen_functions_t)); - functions.init = file_init; - functions.exit = file_exit; - functions.open = file_open; - functions.close = file_close; - functions.resize = file_resize; - functions.paint = file_paint; - functions.update = file_update; - functions.cmd = file_cmd; - functions.get_lw = get_filelist_window; - functions.get_title = file_title; - - return &functions; + static screen_functions_t functions; + + memset(&functions, 0, sizeof(screen_functions_t)); + functions.init = browse_init; + functions.exit = browse_exit; + functions.open = browse_open; + functions.close = browse_close; + functions.resize = browse_resize; + functions.paint = browse_paint; + functions.update = browse_update; + functions.cmd = browse_cmd; + functions.get_lw = get_filelist_window; + functions.get_title = browse_title; + + return &functions; }