mirror of
				https://github.com/Rockbox/rockbox.git
				synced 2025-10-24 23:47:38 -04:00 
			
		
		
		
	Allow independent sorting of playlists in the playlist catalog, using the context menu. You may want to usually sort playlists by their date (modified), but sort audio tracks alphabetically. This is now possible without having to constantly adjust a setting. The fact that 'Sort Files' applied to both the file browser and playlist catalog seems more like a side effect of the implementation, rather than behavior that is wanted. Additionally, the 'By Type' sorting makes no sense when only a single type is displayed. Since many of the other File View settings don't apply to the playlist catalog, hide the menu there. Change-Id: Ic35038015d0860998ae117f472ce23ce3bc80cfa
		
			
				
	
	
		
			905 lines
		
	
	
	
		
			28 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			905 lines
		
	
	
	
		
			28 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /***************************************************************************
 | |
|  *             __________               __   ___.
 | |
|  *   Open      \______   \ ____   ____ |  | _\_ |__   _______  ___
 | |
|  *   Source     |       _//  _ \_/ ___\|  |/ /| __ \ /  _ \  \/  /
 | |
|  *   Jukebox    |    |   (  <_> )  \___|    < | \_\ (  <_> > <  <
 | |
|  *   Firmware   |____|_  /\____/ \___  >__|_ \|___  /\____/__/\_ \
 | |
|  *                     \/            \/     \/    \/            \/
 | |
|  * $Id$
 | |
|  *
 | |
|  * Copyright (C) 2005 by Björn Stenberg
 | |
|  *
 | |
|  * 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 software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
 | |
|  * KIND, either express or implied.
 | |
|  *
 | |
|  ****************************************************************************/
 | |
| #include <stdlib.h>
 | |
| #include <file.h>
 | |
| #include <dir.h>
 | |
| #include <string.h>
 | |
| #include <kernel.h>
 | |
| #include <lcd.h>
 | |
| #include <debug.h>
 | |
| #include <font.h>
 | |
| #include <limits.h>
 | |
| #include "bookmark.h"
 | |
| #include "tree.h"
 | |
| #include "core_alloc.h"
 | |
| #include "settings.h"
 | |
| #include "filetypes.h"
 | |
| #include "talk.h"
 | |
| #include "playlist.h"
 | |
| #include "lang.h"
 | |
| #include "language.h"
 | |
| #include "screens.h"
 | |
| #include "plugin.h"
 | |
| #include "rolo.h"
 | |
| #include "splash.h"
 | |
| #include "cuesheet.h"
 | |
| #include "filetree.h"
 | |
| #include "misc.h"
 | |
| #include "strnatcmp.h"
 | |
| #include "keyboard.h"
 | |
| 
 | |
| #ifdef HAVE_MULTIVOLUME
 | |
| #include "mv.h"
 | |
| #endif
 | |
| 
 | |
| #if CONFIG_TUNER
 | |
| #include "radio.h"
 | |
| #endif
 | |
| #include "wps.h"
 | |
| 
 | |
| static struct compare_data
 | |
| {
 | |
|     int sort_dir; /* qsort key for sorting directories */
 | |
|     int sort_file; /*       ...for sorting files       */
 | |
|     int(*_compar)(const char*, const char*, size_t);
 | |
| } cmp_data;
 | |
| 
 | |
| /* dummmy functions to allow compatibility with strncmp & strncasecmp */
 | |
| static int strnatcmp_n(const char *a, const char *b, size_t n)
 | |
| {
 | |
|     (void)n;
 | |
|      return strnatcmp(a, b);
 | |
| }
 | |
| static int strnatcasecmp_n(const char *a, const char *b, size_t n)
 | |
| {
 | |
|     (void)n;
 | |
|      return strnatcasecmp(a, b);
 | |
| }
 | |
| 
 | |
| int ft_build_playlist(struct tree_context* c, int start_index)
 | |
| {
 | |
|     int i;
 | |
|     int start=start_index;
 | |
|     int res;
 | |
|     struct playlist_info *playlist = playlist_get_current();
 | |
| 
 | |
|     tree_lock_cache(c);
 | |
|     struct entry *entries = tree_get_entries(c);
 | |
|     bool exceeds_pl = false;
 | |
|     if (c->filesindir > playlist->max_playlist_size)
 | |
|     {
 | |
|         exceeds_pl = true;
 | |
|         start_index = 0;
 | |
|     }
 | |
|     struct playlist_insert_context pl_context;
 | |
| 
 | |
|     res = playlist_insert_context_create(playlist, &pl_context,
 | |
|                                         PLAYLIST_REPLACE, false, false);
 | |
|     if (res >= 0)
 | |
|     {
 | |
|         cpu_boost(true);
 | |
|         for(i = 0;i < c->filesindir;i++)
 | |
|         {
 | |
|             int item = i;
 | |
|             if (exceeds_pl)
 | |
|                 item = (i + start) % c->filesindir;
 | |
| #if 0 /*only needed if displaying progress */
 | |
|             /* user abort */
 | |
|             if (action_userabort(TIMEOUT_NOBLOCK))
 | |
|             {
 | |
|                 break;
 | |
|             }
 | |
| #endif
 | |
|             if((entries[item].attr & FILE_ATTR_MASK) == FILE_ATTR_AUDIO)
 | |
|             {
 | |
|                 res = playlist_insert_context_add(&pl_context, entries[item].name);
 | |
|                 if (res < 0)
 | |
|                     break;
 | |
|             }
 | |
|             else if (!exceeds_pl)
 | |
|             {
 | |
|                 /* Adjust the start index when se skip non-MP3 entries */
 | |
|                 if(i < start)
 | |
|                     start_index--;
 | |
|             }
 | |
|         }
 | |
|         cpu_boost(false);
 | |
|     }
 | |
| 
 | |
|     playlist_insert_context_release(&pl_context);
 | |
| 
 | |
|     tree_unlock_cache(c);
 | |
|     return start_index;
 | |
| }
 | |
| 
 | |
| /* Start playback of a playlist, checking for bookmark autoload, modified
 | |
|  * playlists, etc., as required. Returns false if playback wasn't started,
 | |
|  * or started via bookmark autoload, true otherwise.
 | |
|  *
 | |
|  * Pointers to both the full pathname and the separated parts needed to
 | |
|  * avoid allocating yet another path buffer on the stack (and save some
 | |
|  * code; the caller typically needs to create the full pathname anyway)...
 | |
|  */
 | |
| bool ft_play_playlist(char* pathname, char* dirname, char* filename)
 | |
| {
 | |
|     if (global_settings.party_mode && audio_status())
 | |
|     {
 | |
|         splash(HZ, ID2P(LANG_PARTY_MODE));
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     int res =  bookmark_autoload(pathname);
 | |
|     if (res == BOOKMARK_CANCEL || res == BOOKMARK_DO_RESUME || !warn_on_pl_erase())
 | |
|         return false;
 | |
| 
 | |
|     splash(0, ID2P(LANG_WAIT));
 | |
| 
 | |
|     if (playlist_create(dirname, filename) != -1)
 | |
|     {
 | |
|         if (global_settings.playlist_shuffle)
 | |
|             playlist_shuffle(current_tick, -1);
 | |
| 
 | |
|         playlist_start(0, 0, 0);
 | |
|         return true;
 | |
|     }
 | |
| 
 | |
|     return false;
 | |
| }
 | |
| 
 | |
| /* walk a directory and check all entries if a .talk file exists */
 | |
| static void check_file_thumbnails(struct tree_context* c)
 | |
| {
 | |
|     int i;
 | |
|     struct dirent *entry;
 | |
|     struct entry* entries;
 | |
|     DIR *dir;
 | |
| 
 | |
|     dir = opendir(c->currdir);
 | |
|     if(!dir)
 | |
|         return;
 | |
|     /* mark all files as non talking, except the .talk ones */
 | |
|     tree_lock_cache(c);
 | |
|     entries = tree_get_entries(c);
 | |
| 
 | |
|     for (i=0; i < c->filesindir; i++)
 | |
|     {
 | |
|         if (entries[i].attr & ATTR_DIRECTORY)
 | |
|             continue; /* we're not touching directories */
 | |
| 
 | |
|         if (strcasecmp(file_thumbnail_ext,
 | |
|             &entries[i].name[strlen(entries[i].name)
 | |
|                               - strlen(file_thumbnail_ext)]))
 | |
|         {   /* no .talk file */
 | |
|             entries[i].attr &= ~FILE_ATTR_THUMBNAIL; /* clear */
 | |
|         }
 | |
|         else
 | |
|         {   /* .talk file, we later let them speak themselves */
 | |
|             entries[i].attr |= FILE_ATTR_THUMBNAIL; /* set */
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     while((entry = readdir(dir)) != 0) /* walk directory */
 | |
|     {
 | |
|         int ext_pos;
 | |
|         struct dirinfo info = dir_get_info(dir, entry);
 | |
|         ext_pos = strlen((char *)entry->d_name) - strlen(file_thumbnail_ext);
 | |
|         if (ext_pos <= 0 /* too short to carry ".talk" */
 | |
|             || (info.attribute & ATTR_DIRECTORY) /* no file */
 | |
|             || strcasecmp((char *)&entry->d_name[ext_pos], file_thumbnail_ext))
 | |
|         {   /* or doesn't end with ".talk", no candidate */
 | |
|             continue;
 | |
|         }
 | |
| 
 | |
|         /* terminate the (disposable) name in dir buffer,
 | |
|            this truncates off the ".talk" without needing an extra buffer */
 | |
|         entry->d_name[ext_pos] = '\0';
 | |
| 
 | |
|         /* search corresponding file in dir cache */
 | |
|         for (i=0; i < c->filesindir; i++)
 | |
|         {
 | |
|             if (!strcasecmp(entries[i].name, (char *)entry->d_name))
 | |
|             {   /* match */
 | |
|                 entries[i].attr |= FILE_ATTR_THUMBNAIL; /* set the flag */
 | |
|                 break; /* exit search loop, because we found it */
 | |
|             }
 | |
|         }
 | |
|     }
 | |
|     tree_unlock_cache(c);
 | |
|     closedir(dir);
 | |
| }
 | |
| 
 | |
| /* support function for qsort() */
 | |
| static int compare(const void* p1, const void* p2)
 | |
| {
 | |
|     struct entry* e1 = (struct entry*)p1;
 | |
|     struct entry* e2 = (struct entry*)p2;
 | |
|     int criteria;
 | |
| 
 | |
|     if (cmp_data.sort_dir == SORT_AS_FILE)
 | |
|     {   /* treat as two files */
 | |
|         criteria = cmp_data.sort_file;
 | |
|     }
 | |
|     else if (e1->attr & ATTR_DIRECTORY && e2->attr & ATTR_DIRECTORY)
 | |
|     {   /* two directories */
 | |
|         criteria = cmp_data.sort_dir;
 | |
| 
 | |
| #ifdef HAVE_MULTIVOLUME
 | |
|         if (e1->attr & ATTR_VOLUME || e2->attr & ATTR_VOLUME)
 | |
|         {   /* a volume identifier is involved */
 | |
|             if (e1->attr & ATTR_VOLUME && e2->attr & ATTR_VOLUME)
 | |
|                 criteria = SORT_ALPHA; /* two volumes: sort alphabetically */
 | |
|             else /* only one is a volume: volume first */
 | |
|                 return (e2->attr & ATTR_VOLUME) - (e1->attr & ATTR_VOLUME);
 | |
|         }
 | |
| #endif
 | |
| 
 | |
|     }
 | |
|     else if (!(e1->attr & ATTR_DIRECTORY) && !(e2->attr & ATTR_DIRECTORY))
 | |
|     {   /* two files */
 | |
|         criteria = cmp_data.sort_file;
 | |
|     }
 | |
|     else /* dir and file, dir goes first */
 | |
|         return (e2->attr & ATTR_DIRECTORY) - (e1->attr & ATTR_DIRECTORY);
 | |
| 
 | |
|     switch(criteria)
 | |
|     {
 | |
|         case SORT_TYPE:
 | |
|         case SORT_TYPE_REVERSED:
 | |
|         {
 | |
|             int t1 = e1->attr & FILE_ATTR_MASK;
 | |
|             int t2 = e2->attr & FILE_ATTR_MASK;
 | |
| 
 | |
|             if (!t1) /* unknown type */
 | |
|                 t1 = INT_MAX; /* gets a high number, to sort after known */
 | |
|             if (!t2) /* unknown type */
 | |
|                 t2 = INT_MAX; /* gets a high number, to sort after known */
 | |
| 
 | |
|             if (t1 != t2) /* if different */
 | |
|                 return (t1 - t2) * (criteria == SORT_TYPE_REVERSED ? -1 : 1);
 | |
|             /* else alphabetical sorting */
 | |
|             return cmp_data._compar(e1->name, e2->name, MAX_PATH);
 | |
|         }
 | |
| 
 | |
|         case SORT_DATE:
 | |
|         case SORT_DATE_REVERSED:
 | |
|         {
 | |
|             if (e1->time_write != e2->time_write)
 | |
|                 return (e1->time_write - e2->time_write)
 | |
|                        * (criteria == SORT_DATE_REVERSED ? -1 : 1);
 | |
|             /* else fall through to alphabetical sorting */
 | |
|         }
 | |
|         case SORT_ALPHA:
 | |
|         case SORT_ALPHA_REVERSED:
 | |
|         {
 | |
|             return cmp_data._compar(e1->name, e2->name, MAX_PATH) *
 | |
|                 (criteria == SORT_ALPHA_REVERSED ? -1 : 1);
 | |
|         }
 | |
| 
 | |
|     }
 | |
|     return 0; /* never reached */
 | |
| }
 | |
| 
 | |
| /* load and sort directory into the tree's cache. returns NULL on failure. */
 | |
| int ft_load(struct tree_context* c, const char* tempdir)
 | |
| {
 | |
|     if (c->out_of_tree > 0) /* something else is loaded */
 | |
|         return 0;
 | |
| 
 | |
|     int files_in_dir = 0;
 | |
|     int name_buffer_used = 0;
 | |
|     struct dirent *entry;
 | |
|     bool (*callback_show_item)(char *, int, struct tree_context *) = NULL;
 | |
|     DIR *dir;
 | |
| 
 | |
|     if (!c->is_browsing)
 | |
|         c->browse = NULL;
 | |
| 
 | |
|     if (tempdir)
 | |
|         dir = opendir(tempdir);
 | |
|     else
 | |
|     {
 | |
|         dir = opendir(c->currdir);
 | |
|         callback_show_item = c->browse? c->browse->callback_show_item: NULL;
 | |
|     }
 | |
|     if(!dir)
 | |
|         return -1; /* not a directory */
 | |
| 
 | |
|     c->dirsindir = 0;
 | |
|     c->dirfull = false;
 | |
| 
 | |
|     tree_lock_cache(c);
 | |
|     while ((entry = readdir(dir))) {
 | |
|         int len;
 | |
|         struct dirinfo info;
 | |
|         struct entry* dptr = tree_get_entry_at(c, files_in_dir);
 | |
|         if (!dptr)
 | |
|         {
 | |
|             c->dirfull = true;
 | |
|             break;
 | |
|         }
 | |
| 
 | |
|         info = dir_get_info(dir, entry);
 | |
|         len = strlen((char *)entry->d_name);
 | |
| 
 | |
|         /* Skip FAT volume ID */
 | |
|         if (info.attribute & ATTR_VOLUME_ID) {
 | |
|             continue;
 | |
|         }
 | |
| 
 | |
|         dptr->attr = info.attribute;
 | |
|         int dir_attr = (dptr->attr & ATTR_DIRECTORY);
 | |
|         /* skip directories . and .. */
 | |
|         if (dir_attr && is_dotdir_name(entry->d_name))
 | |
|             continue;
 | |
| 
 | |
|         /* filter out dotfiles and hidden files */
 | |
|         if (*c->dirfilter != SHOW_ALL &&
 | |
|             ((entry->d_name[0]=='.') ||
 | |
|             (info.attribute & ATTR_HIDDEN))) {
 | |
|             continue;
 | |
|         }
 | |
| 
 | |
|         if (*c->dirfilter == SHOW_PLUGINS && (dptr->attr & ATTR_DIRECTORY) &&
 | |
|             (dptr->attr &
 | |
|             (ATTR_HIDDEN | ATTR_SYSTEM | ATTR_VOLUME_ID | ATTR_VOLUME)) != 0) {
 | |
|             continue; /* skip non plugin folders */
 | |
|         }
 | |
| 
 | |
|         /* check for known file types */
 | |
|         if ( !(dir_attr) )
 | |
|             dptr->attr |= filetype_get_attr((char *)entry->d_name);
 | |
| 
 | |
|         int file_attr = (dptr->attr & FILE_ATTR_MASK);
 | |
| 
 | |
| #define CHK_FT(show,attr) (*c->dirfilter == (show) && file_attr != (attr))
 | |
|         /* filter out non-visible files */
 | |
|         if ((!(dir_attr) && (CHK_FT(SHOW_PLAYLIST, FILE_ATTR_M3U) ||
 | |
|             (CHK_FT(SHOW_MUSIC, FILE_ATTR_AUDIO) && file_attr != FILE_ATTR_M3U) ||
 | |
|             (*c->dirfilter == SHOW_SUPPORTED && !filetype_supported(dptr->attr)))) ||
 | |
|             CHK_FT(SHOW_WPS,  FILE_ATTR_WPS)  ||
 | |
|             CHK_FT(SHOW_FONT, FILE_ATTR_FONT) ||
 | |
|             CHK_FT(SHOW_SBS,  FILE_ATTR_SBS)  ||
 | |
| #if CONFIG_TUNER
 | |
|             CHK_FT(SHOW_FMS, FILE_ATTR_FMS) ||
 | |
|             CHK_FT(SHOW_FMR, FILE_ATTR_FMR) ||
 | |
| #endif
 | |
| #ifdef HAVE_REMOTE_LCD
 | |
|             CHK_FT(SHOW_RWPS, FILE_ATTR_RWPS) ||
 | |
|             CHK_FT(SHOW_RSBS, FILE_ATTR_RSBS) ||
 | |
| #if CONFIG_TUNER
 | |
|             CHK_FT(SHOW_RFMS, FILE_ATTR_RFMS) ||
 | |
| #endif
 | |
| #endif
 | |
|             CHK_FT(SHOW_M3U, FILE_ATTR_M3U) ||
 | |
|             CHK_FT(SHOW_CFG, FILE_ATTR_CFG) ||
 | |
|             CHK_FT(SHOW_LNG, FILE_ATTR_LNG) ||
 | |
|             CHK_FT(SHOW_MOD, FILE_ATTR_MOD) ||
 | |
|            /* show first level directories */
 | |
|            ((!(dir_attr) || c->dirlevel > 0)       &&
 | |
|             CHK_FT(SHOW_PLUGINS, FILE_ATTR_ROCK)   &&
 | |
|                        file_attr != FILE_ATTR_LUA  &&
 | |
|                        file_attr != FILE_ATTR_OPX) ||
 | |
|             (callback_show_item && !callback_show_item(entry->d_name, dptr->attr, c)))
 | |
|         {
 | |
|             continue;
 | |
|         }
 | |
| #undef CHK_FT
 | |
| 
 | |
|         if (len > c->cache.name_buffer_size - name_buffer_used - 1) {
 | |
|             /* Tell the world that we ran out of buffer space */
 | |
|             c->dirfull = true;
 | |
|             break;
 | |
|         }
 | |
| 
 | |
|         ++files_in_dir;
 | |
| 
 | |
|         dptr->name = core_get_data(c->cache.name_buffer_handle)+name_buffer_used;
 | |
|         dptr->time_write = info.mtime;
 | |
|         strcpy(dptr->name, (char *)entry->d_name);
 | |
|         name_buffer_used += len + 1;
 | |
| 
 | |
|         if (dir_attr) /* count the remaining dirs */
 | |
|             c->dirsindir++;
 | |
|     }
 | |
|     c->filesindir = files_in_dir;
 | |
|     c->dirlength = files_in_dir;
 | |
|     closedir(dir);
 | |
| 
 | |
|     /* allow directories to be sorted into file list */
 | |
|     cmp_data.sort_dir = (*c->dirfilter == SHOW_PLUGINS) ? SORT_AS_FILE : c->sort_dir;
 | |
| 
 | |
|     /* playlist catalog uses sorting independent from file browser */
 | |
|     cmp_data.sort_file = (*c->dirfilter == SHOW_M3U) ?
 | |
|                          global_settings.sort_playlists : global_settings.sort_file;
 | |
| 
 | |
|     if (global_settings.sort_case)
 | |
|     {
 | |
|         if (global_settings.interpret_numbers == SORT_INTERPRET_AS_NUMBER)
 | |
|             cmp_data._compar = strnatcmp_n;
 | |
|         else
 | |
|             cmp_data._compar = strncmp;
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         if (global_settings.interpret_numbers == SORT_INTERPRET_AS_NUMBER)
 | |
|             cmp_data._compar = strnatcasecmp_n;
 | |
|         else
 | |
|             cmp_data._compar = strncasecmp;
 | |
|     }
 | |
| 
 | |
|     qsort(tree_get_entries(c), files_in_dir, sizeof(struct entry), compare);
 | |
| 
 | |
|     /* If thumbnail talking is enabled, make an extra run to mark files with
 | |
|        associated thumbnails, so we don't do unsuccessful spinups later. */
 | |
|     if (global_settings.talk_file_clip)
 | |
|         check_file_thumbnails(c); /* map .talk to ours */
 | |
| 
 | |
|     tree_unlock_cache(c);
 | |
|     return 0;
 | |
| }
 | |
| static void ft_load_font(char *file)
 | |
| {
 | |
|     int current_font_id;
 | |
|     enum screen_type screen = SCREEN_MAIN;
 | |
| #if NB_SCREENS > 1
 | |
|     MENUITEM_STRINGLIST(menu, ID2P(LANG_CUSTOM_FONT), NULL,
 | |
|                         ID2P(LANG_MAIN_SCREEN), ID2P(LANG_REMOTE_SCREEN))
 | |
|     switch (do_menu(&menu, NULL, NULL, false))
 | |
|     {
 | |
|         case 0: /* main lcd */
 | |
|             screen = SCREEN_MAIN;
 | |
|             set_file(file, (char *)global_settings.font_file);
 | |
|             break;
 | |
|         case 1: /* remote */
 | |
|             screen = SCREEN_REMOTE;
 | |
|             set_file(file, (char *)global_settings.remote_font_file);
 | |
|             break;
 | |
|     }
 | |
| #else
 | |
|     set_file(file, (char *)global_settings.font_file);
 | |
| #endif
 | |
|     splash(0, ID2P(LANG_WAIT));
 | |
|     current_font_id = screens[screen].getuifont();
 | |
|     if (current_font_id >= 0)
 | |
|         font_unload(current_font_id);
 | |
|     screens[screen].setuifont(
 | |
|         font_load_ex(file,0,global_settings.glyphs_to_cache));
 | |
|     viewportmanager_theme_changed(THEME_UI_VIEWPORT);
 | |
| }
 | |
| 
 | |
| static void ft_apply_skin_file(char *buf, char *file)
 | |
| {
 | |
|     splash(0, ID2P(LANG_WAIT));
 | |
|     set_file(buf, file);
 | |
|     settings_apply_skins();
 | |
| }
 | |
| 
 | |
| static const char *strip_slash(const char *path, const char *def)
 | |
| {
 | |
|     if (path)
 | |
|     {
 | |
|         while (*path == PATH_SEPCH)
 | |
|             path++; /* we don't want this treated as an absolute path */
 | |
|         return path;
 | |
|     }
 | |
|     return def;
 | |
| }
 | |
| 
 | |
| int ft_assemble_path(char *buf, size_t bufsz, const char* currdir, const char* filename)
 | |
| {
 | |
|     size_t len;
 | |
|     const char *cd = strip_slash(currdir, "");
 | |
|     filename = strip_slash(filename, "");
 | |
|     /* remove slashes and NULL strings to make logic below simpler */
 | |
| 
 | |
| #ifdef HAVE_MULTIVOLUME
 | |
|     /* Multi-volume device drives might be enumerated in root so everything
 | |
|        should be an absolute qualified path with <drive>/ prepended */
 | |
|     if (*cd != '\0') /* Not in / */
 | |
|     {
 | |
|         if (*cd == VOL_START_TOK)
 | |
|         {
 | |
|           /* use currdir, here we want the slash as it already contains the <drive> */
 | |
|             len = path_append(buf, currdir, filename, bufsz);
 | |
|         } /* buf => /currdir/filename */
 | |
|         else
 | |
|         {
 | |
|             len = path_append(buf, root_realpath(), cd, bufsz); /* /<drive>/currdir */
 | |
|             if(len < bufsz)
 | |
|                 len += path_append(buf + len, PA_SEP_HARD, filename, bufsz - len);
 | |
|         } /* buf => /<drive>/currdir/filename */
 | |
|     }
 | |
|     else /* In / */
 | |
|     {
 | |
|         if (*filename == VOL_START_TOK)
 | |
|         {
 | |
|             len = path_append(buf, PATH_SEPSTR, filename, bufsz);
 | |
|         } /* buf => /filename */
 | |
|         else
 | |
|         {
 | |
|             len = path_append(buf, root_realpath(), filename, bufsz);
 | |
|         } /* buf => /<drive>/filename */
 | |
|     }
 | |
| #else
 | |
|     /* Other devices might need a specific drive/dir prepended but its usually '/' */
 | |
|     if (*cd != '\0') /* Not in / */
 | |
|     {
 | |
|         len = path_append(buf, root_realpath(), cd, bufsz);/* /currdir */
 | |
|         if(len < bufsz)
 | |
|             len += path_append(buf + len, PA_SEP_HARD, filename, bufsz - len);
 | |
|     } /* buf => /currdir/filename */
 | |
|     else /* In / */
 | |
|     {
 | |
|         len = path_append(buf, root_realpath(), filename, bufsz);
 | |
|     }  /* buf => /filename */
 | |
| #endif
 | |
| 
 | |
|     if (len > bufsz)
 | |
|         splash(HZ, ID2P(LANG_PLAYLIST_DIRECTORY_ACCESS_ERROR));
 | |
|     return (int)len;
 | |
| }
 | |
| 
 | |
| int ft_enter(struct tree_context* c)
 | |
| {
 | |
|     int rc = GO_TO_PREVIOUS;
 | |
|     char buf[MAX_PATH];
 | |
| 
 | |
|     struct entry* file = tree_get_entry_at(c, c->selected_item);
 | |
|     if (!file)
 | |
|     {
 | |
|         splashf(HZ, ID2P(LANG_READ_FAILED), str(LANG_UNKNOWN));
 | |
|         return rc;
 | |
|     }
 | |
| 
 | |
|     int file_attr = file->attr;
 | |
|     ft_assemble_path(buf, sizeof(buf), c->currdir, file->name);
 | |
|     if (file_attr & ATTR_DIRECTORY) {
 | |
|         memcpy(c->currdir, buf, sizeof(c->currdir));
 | |
|         if ( c->dirlevel < MAX_DIR_LEVELS )
 | |
|             c->selected_item_history[c->dirlevel] = c->selected_item;
 | |
|         c->dirlevel++;
 | |
|         c->selected_item=0;
 | |
|     }
 | |
|     else {
 | |
|         int seed = current_tick;
 | |
|         bool play = false;
 | |
|         int start_index=0;
 | |
| 
 | |
|         switch ( file_attr & FILE_ATTR_MASK ) {
 | |
|             case FILE_ATTR_M3U:
 | |
|                 play = ft_play_playlist(buf, c->currdir, file->name);
 | |
| 
 | |
|                 if (play)
 | |
|                 {
 | |
|                     start_index = 0;
 | |
|                 }
 | |
| 
 | |
|                 break;
 | |
| 
 | |
|             case FILE_ATTR_AUDIO:
 | |
|             {
 | |
|                 int res = bookmark_autoload(c->currdir);
 | |
|                 if (res == BOOKMARK_CANCEL || res == BOOKMARK_DO_RESUME)
 | |
|                     break;
 | |
| 
 | |
|                 splash(0, ID2P(LANG_WAIT));
 | |
| 
 | |
|                 /* about to create a new current playlist...
 | |
|                    allow user to cancel the operation */
 | |
|                 if (!warn_on_pl_erase())
 | |
|                     break;
 | |
| 
 | |
|                 if (global_settings.party_mode && audio_status())
 | |
|                 {
 | |
|                     playlist_insert_track(NULL, buf,
 | |
|                                           PLAYLIST_INSERT_LAST, true, true);
 | |
|                     splash(HZ, ID2P(LANG_QUEUE_LAST));
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     /* use the assembled path sans filename */
 | |
|                     char * fp = strrchr(buf, PATH_SEPCH);
 | |
|                     if (fp)
 | |
|                         *fp = '\0';
 | |
|                     if (playlist_create(buf, NULL) != -1)
 | |
|                     {
 | |
|                         start_index = ft_build_playlist(c, c->selected_item);
 | |
|                         if (global_settings.playlist_shuffle)
 | |
|                         {
 | |
|                             start_index = playlist_shuffle(seed, start_index);
 | |
|                             /* when shuffling dir.: play all files
 | |
|                                even if the file selected by user is
 | |
|                                not the first one */
 | |
|                             if (!global_settings.play_selected)
 | |
|                                 start_index = 0;
 | |
|                         }
 | |
|                         playlist_start(start_index, 0, 0);
 | |
|                         play = true;
 | |
|                     }
 | |
|                 }
 | |
|                 break;
 | |
|             }
 | |
| #if CONFIG_TUNER
 | |
|                 /* fmr preset file */
 | |
|             case FILE_ATTR_FMR:
 | |
|                 radio_load_presets(buf);
 | |
|                 rc = GO_TO_FM;
 | |
|                 break;
 | |
|             case FILE_ATTR_FMS:
 | |
|                 ft_apply_skin_file(buf, global_settings.fms_file);
 | |
|                 break;
 | |
| #ifdef HAVE_REMOTE_LCD
 | |
|             case FILE_ATTR_RFMS:
 | |
|                 ft_apply_skin_file(buf, global_settings.rfms_file);
 | |
|                 break;
 | |
| #endif
 | |
| #endif
 | |
|             case FILE_ATTR_SBS:
 | |
|                 ft_apply_skin_file(buf, global_settings.sbs_file);
 | |
|                 break;
 | |
| #ifdef HAVE_REMOTE_LCD
 | |
|             case FILE_ATTR_RSBS:
 | |
|                 ft_apply_skin_file(buf, global_settings.rsbs_file);
 | |
|                 break;
 | |
| #endif
 | |
|                 /* wps config file */
 | |
|             case FILE_ATTR_WPS:
 | |
|                 ft_apply_skin_file(buf, global_settings.wps_file);
 | |
|                 break;
 | |
| #if defined(HAVE_REMOTE_LCD) && (NB_SCREENS > 1)
 | |
|                 /* remote-wps config file */
 | |
|             case FILE_ATTR_RWPS:
 | |
|                 ft_apply_skin_file(buf, global_settings.rwps_file);
 | |
|                 break;
 | |
| #endif
 | |
|             case FILE_ATTR_CFG:
 | |
|                 splash(0, ID2P(LANG_WAIT));
 | |
|                 if (!settings_load_config(buf,true))
 | |
|                     break;
 | |
|                 splash(HZ, ID2P(LANG_SETTINGS_LOADED));
 | |
|                 break;
 | |
| 
 | |
|             case FILE_ATTR_BMARK:
 | |
|                 splash(0, ID2P(LANG_WAIT));
 | |
|                 bookmark_load(buf, false);
 | |
|                 rc = GO_TO_FILEBROWSER;
 | |
|                 break;
 | |
| 
 | |
|             case FILE_ATTR_LNG:
 | |
|                 splash(0, ID2P(LANG_WAIT));
 | |
|                 if (lang_core_load(buf))
 | |
|                 {
 | |
|                     splash(HZ, ID2P(LANG_FAILED));
 | |
|                     break;
 | |
|                 }
 | |
|                 set_file(buf, (char *)global_settings.lang_file);
 | |
|                 talk_init(); /* use voice of same language */
 | |
|                 viewportmanager_theme_changed(THEME_LANGUAGE);
 | |
|                 settings_apply_skins();
 | |
|                 splash(HZ, ID2P(LANG_LANGUAGE_LOADED));
 | |
|                 break;
 | |
| 
 | |
|             case FILE_ATTR_FONT:
 | |
|                 ft_load_font(buf);
 | |
|                 break;
 | |
| 
 | |
|             case FILE_ATTR_KBD:
 | |
|                 splash(0, ID2P(LANG_WAIT));
 | |
|                 if (!load_kbd(buf))
 | |
|                     splash(HZ, ID2P(LANG_KEYBOARD_LOADED));
 | |
|                 set_file(buf, (char *)global_settings.kbd_file);
 | |
|                 break;
 | |
| 
 | |
| #if defined(HAVE_ROLO)
 | |
|                 /* firmware file */
 | |
|             case FILE_ATTR_MOD:
 | |
|                 splash(0, ID2P(LANG_WAIT));
 | |
|                 audio_hard_stop();
 | |
|                 rolo_load(buf);
 | |
|                 break;
 | |
| #endif
 | |
|             case FILE_ATTR_CUE:
 | |
|                 display_cuesheet_content(buf);
 | |
|                 break;
 | |
| 
 | |
|                 /* plugin file */
 | |
|             case FILE_ATTR_ROCK:
 | |
|             {
 | |
|                 char *plugin = buf, *argument = NULL;
 | |
|                 if (global_settings.party_mode && audio_status()) {
 | |
|                     splash(HZ, ID2P(LANG_PARTY_MODE));
 | |
|                     break;
 | |
|                 }
 | |
| 
 | |
| #ifdef PLUGINS_RUN_IN_BROWSER /* Stay in the filetree to run a plugin */
 | |
|                 switch (plugin_load(plugin, argument))
 | |
|                 {
 | |
|                     case PLUGIN_GOTO_WPS:
 | |
|                         play = true;
 | |
|                         break;
 | |
|                     case PLUGIN_GOTO_PLUGIN:
 | |
|                         rc = GO_TO_PLUGIN;
 | |
|                         break;
 | |
|                     case PLUGIN_USB_CONNECTED:
 | |
|                         if(*c->dirfilter > NUM_FILTER_MODES)
 | |
|                             /* leave sub-browsers after usb, doing
 | |
|                                otherwise might be confusing to the user */
 | |
|                             rc = GO_TO_ROOT;
 | |
|                         else
 | |
|                             rc = GO_TO_FILEBROWSER;
 | |
|                         break;
 | |
|                     /*
 | |
|                     case PLUGIN_ERROR:
 | |
|                     case PLUGIN_OK:
 | |
|                     */
 | |
|                     default:
 | |
|                         break;
 | |
|                 }
 | |
| #else /* Exit the filetree to run a plugin */
 | |
|                 plugin_open(plugin, argument);
 | |
|                 rc = GO_TO_PLUGIN;
 | |
| #endif
 | |
|                 break;
 | |
|             }
 | |
| 
 | |
|             default:
 | |
|             {
 | |
|                 const char* plugin;
 | |
|                 char plugin_path[MAX_PATH];
 | |
|                 const char *argument = buf;
 | |
|                 if (global_settings.party_mode && audio_status()) {
 | |
|                     splash(HZ, ID2P(LANG_PARTY_MODE));
 | |
|                     break;
 | |
|                 }
 | |
| 
 | |
|                 file = tree_get_entry_at(c, c->selected_item);
 | |
|                 if (!file)
 | |
|                 {
 | |
|                     splashf(HZ, ID2P(LANG_READ_FAILED), str(LANG_UNKNOWN));
 | |
|                     return rc;
 | |
|                 }
 | |
| 
 | |
|                 plugin = filetype_get_plugin(file->attr, plugin_path, sizeof(plugin_path));
 | |
|                 if (plugin)
 | |
|                 {
 | |
| #ifdef PLUGINS_RUN_IN_BROWSER /* Stay in the filetree to run a plugin */
 | |
|                     switch (plugin_load(plugin, argument))
 | |
|                     {
 | |
|                         case PLUGIN_USB_CONNECTED:
 | |
|                             rc = GO_TO_FILEBROWSER;
 | |
|                             break;
 | |
|                         case PLUGIN_GOTO_PLUGIN:
 | |
|                             rc = GO_TO_PLUGIN;
 | |
|                             break;
 | |
|                         case PLUGIN_GOTO_WPS:
 | |
|                             rc = GO_TO_WPS;
 | |
|                             break;
 | |
|                         /*
 | |
|                         case PLUGIN_OK:
 | |
|                         case PLUGIN_ERROR:
 | |
|                         */
 | |
|                         default:
 | |
|                             break;
 | |
|                     }
 | |
| #else /* Exit the filetree to run a plugin */
 | |
|                     plugin_open(plugin, argument);
 | |
|                     rc = GO_TO_PLUGIN;
 | |
| #endif
 | |
|                 }
 | |
|                 break;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         if ( play ) {
 | |
|             /* the resume_index must always be the index in the
 | |
|                shuffled list in case shuffle is enabled */
 | |
|             global_status.resume_index = start_index;
 | |
|             global_status.resume_crc32 =
 | |
|                 playlist_get_filename_crc32(NULL, start_index);
 | |
|             global_status.resume_elapsed = 0;
 | |
|             global_status.resume_offset = 0;
 | |
|             status_save(false);
 | |
|             rc = GO_TO_WPS;
 | |
|         }
 | |
|         else {
 | |
|             if (*c->dirfilter > NUM_FILTER_MODES &&
 | |
|                 *c->dirfilter != SHOW_CFG &&
 | |
|                 *c->dirfilter != SHOW_FONT &&
 | |
|                 *c->dirfilter != SHOW_PLUGINS)
 | |
|             {
 | |
|                 rc = GO_TO_ROOT;
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     return rc;
 | |
| }
 | |
| 
 | |
| int ft_exit(struct tree_context* c)
 | |
| {
 | |
|     extern char lastfile[]; /* from tree.c */
 | |
|     char buf[MAX_PATH];
 | |
|     int rc = 0;
 | |
|     bool exit_func = false;
 | |
|     int i = strlen(c->currdir);
 | |
| 
 | |
|     /* strip trailing slashes */
 | |
|     while (c->currdir[i-1] == PATH_SEPCH)
 | |
|         i--;
 | |
| 
 | |
|     if (i>1) {
 | |
|         while (c->currdir[i-1]!=PATH_SEPCH)
 | |
|             i--;
 | |
|         strcpy(buf,&c->currdir[i]);
 | |
|         if (i==1)
 | |
|             c->currdir[i]='\0';
 | |
|         else
 | |
|             c->currdir[i-1]='\0';
 | |
| 
 | |
| #ifdef HAVE_MULTIVOLUME /* un-redirect the realpath */
 | |
|         if ((unsigned)c->dirlevel<=2) /* only expect redirect two levels max */
 | |
|         {
 | |
|             char *currdir = c->currdir;
 | |
|             const char *root = root_realpath();
 | |
|             int len = i-1;
 | |
|             /* compare to the root path bail if they don't match except single '/' */
 | |
|             for (; len > 0 && *root != '\0' && *root == *currdir; len--)
 | |
|             {
 | |
|                 root++;
 | |
|                 currdir++;
 | |
|             }
 | |
|             if (*root == PATH_SEPCH) /* root may have trailing slash */
 | |
|                 root++;
 | |
|             if (*root == '\0' &&
 | |
|                 (len == 0 || (len == 1 && *currdir == PATH_SEPCH)))
 | |
|             {
 | |
|                 strcpy(c->currdir, PATH_ROOTSTR);
 | |
|                 c->dirlevel=1;
 | |
|             }
 | |
|         }
 | |
| #endif
 | |
| 
 | |
|         if (*c->dirfilter > NUM_FILTER_MODES && c->dirlevel < 1)
 | |
|             exit_func = true;
 | |
| 
 | |
|         c->dirlevel--;
 | |
|         if ( c->dirlevel < MAX_DIR_LEVELS )
 | |
|             c->selected_item=c->selected_item_history[c->dirlevel];
 | |
|         else
 | |
|             c->selected_item=0;
 | |
| 
 | |
|         /* if undefined position */
 | |
|         if (c->selected_item == -1)
 | |
|             strcpy(lastfile, buf);
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         if (*c->dirfilter > NUM_FILTER_MODES && c->dirlevel < 1)
 | |
|             exit_func = true;
 | |
|     }
 | |
| 
 | |
|     if (exit_func)
 | |
|         rc = 3;
 | |
| 
 | |
|     c->out_of_tree = 0;
 | |
| 
 | |
|     return rc;
 | |
| }
 |