forked from len0rd/rockbox
		
	git-svn-id: svn://svn.rockbox.org/rockbox/trunk@3409 a1c6a512-1295-4272-9138-f99709370657
		
			
				
	
	
		
			1520 lines
		
	
	
	
		
			46 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			1520 lines
		
	
	
	
		
			46 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /***************************************************************************
 | |
|  *             __________               __   ___.
 | |
|  *   Open      \______   \ ____   ____ |  | _\_ |__   _______  ___
 | |
|  *   Source     |       _//  _ \_/ ___\|  |/ /| __ \ /  _ \  \/  /
 | |
|  *   Jukebox    |    |   (  <_> )  \___|    < | \_\ (  <_> > <  <
 | |
|  *   Firmware   |____|_  /\____/ \___  >__|_ \|___  /\____/__/\_ \
 | |
|  *                     \/            \/     \/    \/            \/
 | |
|  * $Id$
 | |
|  *
 | |
|  * Copyright (C) 2002 Daniel Stenberg
 | |
|  *
 | |
|  * All files in this archive are subject to the GNU General Public License.
 | |
|  * See the file COPYING in the source tree root for full license agreement.
 | |
|  *
 | |
|  * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
 | |
|  * KIND, either express or implied.
 | |
|  *
 | |
|  ****************************************************************************/
 | |
| 
 | |
| #include <stdio.h>
 | |
| #include <string.h>
 | |
| #include <stdlib.h>
 | |
| #include <stdbool.h>
 | |
| 
 | |
| #include "applimits.h"
 | |
| #include "dir.h"
 | |
| #include "file.h"
 | |
| #include "lcd.h"
 | |
| #include "font.h"
 | |
| #include "backlight.h"
 | |
| #include "button.h"
 | |
| #include "kernel.h"
 | |
| #include "usb.h"
 | |
| #include "tree.h"
 | |
| #include "main_menu.h"
 | |
| #include "sprintf.h"
 | |
| #include "mpeg.h"
 | |
| #include "playlist.h"
 | |
| #include "menu.h"
 | |
| #include "wps.h"
 | |
| #include "wps-display.h"
 | |
| #include "settings.h"
 | |
| #include "status.h"
 | |
| #include "debug.h"
 | |
| #include "ata.h"
 | |
| #include "rolo.h"
 | |
| #include "icons.h"
 | |
| #include "lang.h"
 | |
| #include "viewer.h"
 | |
| #include "language.h"
 | |
| #include "screens.h"
 | |
| #include "keyboard.h"
 | |
| 
 | |
| #ifdef HAVE_LCD_BITMAP
 | |
| #include "widgets.h"
 | |
| #endif
 | |
| 
 | |
| #define NAME_BUFFER_SIZE (AVERAGE_FILENAME_LENGTH * MAX_FILES_IN_DIR)
 | |
| 
 | |
| static char name_buffer[NAME_BUFFER_SIZE];
 | |
| static int name_buffer_length;
 | |
| struct entry {
 | |
|     short attr; /* FAT attributes + file type flags */
 | |
|     char *name;
 | |
| };
 | |
| 
 | |
| static struct entry dircache[MAX_FILES_IN_DIR];
 | |
| static int dircursor;
 | |
| static int dirstart;
 | |
| static int dirlevel;
 | |
| static int filesindir;
 | |
| static int dirpos[MAX_DIR_LEVELS];
 | |
| static int cursorpos[MAX_DIR_LEVELS];
 | |
| static char lastdir[MAX_PATH];
 | |
| static char lastfile[MAX_PATH];
 | |
| static char currdir[MAX_PATH];
 | |
| static bool reload_dir = false;
 | |
| 
 | |
| void browse_root(void)
 | |
| {
 | |
| #ifndef SIMULATOR
 | |
|     dirbrowse("/");
 | |
| #else
 | |
|     if (!dirbrowse("/")) {
 | |
|         DEBUGF("No filesystem found. Have you forgotten to create it?\n");
 | |
|     }
 | |
| #endif
 | |
| }
 | |
| 
 | |
| 
 | |
| #ifdef HAVE_LCD_BITMAP
 | |
| 
 | |
| /* pixel margins */
 | |
| #define MARGIN_X (global_settings.scrollbar && \
 | |
|                   filesindir > tree_max_on_screen ? SCROLLBAR_WIDTH : 0) + \
 | |
|                   CURSOR_WIDTH + ICON_WIDTH
 | |
| #define MARGIN_Y (global_settings.statusbar ? STATUSBAR_HEIGHT : 0)
 | |
| 
 | |
| /* position the entry-list starts at */
 | |
| #define LINE_X   0 
 | |
| #define LINE_Y   (global_settings.statusbar ? 1 : 0)
 | |
| 
 | |
| #define CURSOR_X (global_settings.scrollbar && \
 | |
|                   filesindir > tree_max_on_screen ? 1 : 0)
 | |
| #define CURSOR_Y 0 /* the cursor is not positioned in regard to
 | |
|                       the margins, so this is the amount of lines
 | |
|                       we add to the cursor Y position to position
 | |
|                       it on a line */
 | |
| #define CURSOR_WIDTH  4
 | |
| 
 | |
| #define ICON_WIDTH    6
 | |
| 
 | |
| #define SCROLLBAR_X      0
 | |
| #define SCROLLBAR_Y      lcd_getymargin()
 | |
| #define SCROLLBAR_WIDTH  6
 | |
| 
 | |
| extern unsigned char bitmap_icons_6x8[LastIcon][6];
 | |
| 
 | |
| #else /* HAVE_LCD_BITMAP */
 | |
| 
 | |
| #define TREE_MAX_ON_SCREEN   2
 | |
| #define TREE_MAX_LEN_DISPLAY 11 /* max length that fits on screen */
 | |
| #define LINE_X      2 /* X position the entry-list starts at */
 | |
| #define LINE_Y      0 /* Y position the entry-list starts at */
 | |
| 
 | |
| #define CURSOR_X    0
 | |
| #define CURSOR_Y    0 /* not really used for players */
 | |
| 
 | |
| #endif /* HAVE_LCD_BITMAP */
 | |
| 
 | |
| #ifdef HAVE_RECORDER_KEYPAD
 | |
| #define TREE_NEXT  BUTTON_DOWN
 | |
| #define TREE_PREV  BUTTON_UP
 | |
| #define TREE_EXIT  BUTTON_LEFT
 | |
| #define TREE_ENTER BUTTON_RIGHT
 | |
| #define TREE_MENU  BUTTON_F1
 | |
| #else
 | |
| #define TREE_NEXT  BUTTON_RIGHT
 | |
| #define TREE_PREV  BUTTON_LEFT
 | |
| #define TREE_EXIT  BUTTON_STOP
 | |
| #define TREE_ENTER BUTTON_PLAY
 | |
| #define TREE_MENU  BUTTON_MENU
 | |
| #endif /* HAVE_RECORDER_KEYPAD */
 | |
| 
 | |
| /* using attribute not used by FAT */
 | |
| #define TREE_ATTR_MPA 0x40 /* mpeg audio file */
 | |
| #define TREE_ATTR_M3U 0x80 /* playlist */
 | |
| #define TREE_ATTR_WPS 0x100 /* wps config file */
 | |
| #define TREE_ATTR_MOD 0x200 /* firmware file */
 | |
| #define TREE_ATTR_CFG 0x400 /* config file */
 | |
| #define TREE_ATTR_TXT 0x500 /* text file */
 | |
| #define TREE_ATTR_FONT 0x800 /* font file */
 | |
| #define TREE_ATTR_LNG  0x1000 /* binary lang file */
 | |
| #define TREE_ATTR_MASK 0xffd0 /* which bits tree.c uses (above + DIR) */
 | |
| 
 | |
| static int build_playlist(int start_index)
 | |
| {
 | |
|     int i;
 | |
|     int start=start_index;
 | |
| 
 | |
|     playlist_clear();
 | |
| 
 | |
|     for(i = 0;i < filesindir;i++)
 | |
|     {
 | |
|         if(dircache[i].attr & TREE_ATTR_MPA)
 | |
|         {
 | |
|             DEBUGF("Adding %s\n", dircache[i].name);
 | |
|             playlist_add(dircache[i].name);
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             /* Adjust the start index when se skip non-MP3 entries */
 | |
|             if(i < start)
 | |
|                 start_index--;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     return start_index;
 | |
| }
 | |
| 
 | |
| static int compare(const void* p1, const void* p2)
 | |
| {
 | |
|     struct entry* e1 = (struct entry*)p1;
 | |
|     struct entry* e2 = (struct entry*)p2;
 | |
|     
 | |
|     if (( e1->attr & ATTR_DIRECTORY ) == ( e2->attr & ATTR_DIRECTORY ))
 | |
|         if (global_settings.sort_case)
 | |
|             return strncmp(e1->name, e2->name, MAX_PATH);
 | |
|         else
 | |
|             return strncasecmp(e1->name, e2->name, MAX_PATH);
 | |
|     else 
 | |
|         return ( e2->attr & ATTR_DIRECTORY ) - ( e1->attr & ATTR_DIRECTORY );
 | |
| }
 | |
| 
 | |
| static void showfileline(int line, int direntry, bool scroll)
 | |
| {
 | |
|     /* if any file filter is on, strip the extension */
 | |
|     if (global_settings.dirfilter != SHOW_ALL && 
 | |
|         !(dircache[direntry].attr & ATTR_DIRECTORY))
 | |
|     {
 | |
|         char* dotpos = strrchr(dircache[direntry].name, '.');
 | |
|         char temp=0;
 | |
|         if (dotpos) {
 | |
|             temp = *dotpos;
 | |
|             *dotpos = 0;
 | |
|         }
 | |
|         if(scroll)
 | |
|             lcd_puts_scroll(LINE_X, line, dircache[direntry].name);
 | |
|         else
 | |
|             lcd_puts(LINE_X, line, dircache[direntry].name);
 | |
|         if (temp)
 | |
|             *dotpos = temp;
 | |
|     }
 | |
|     else {
 | |
|         if(scroll)
 | |
|             lcd_puts_scroll(LINE_X, line, dircache[direntry].name);
 | |
|         else
 | |
|             lcd_puts(LINE_X, line, dircache[direntry].name);
 | |
|     }
 | |
| }
 | |
| 
 | |
| 
 | |
| static int showdir(char *path, int start)
 | |
| {
 | |
|     int icon_type = 0;
 | |
|     int i;
 | |
|     int tree_max_on_screen;
 | |
|     bool dir_buffer_full;
 | |
| 
 | |
| #ifdef HAVE_LCD_BITMAP
 | |
|     int line_height;
 | |
|     int fw, fh;
 | |
|     lcd_setfont(FONT_UI);
 | |
|     lcd_getstringsize("A", &fw, &fh);
 | |
|     tree_max_on_screen = (LCD_HEIGHT - MARGIN_Y) / fh;
 | |
|     line_height = fh;
 | |
| #else
 | |
|     tree_max_on_screen = TREE_MAX_ON_SCREEN;
 | |
| #endif
 | |
| 
 | |
|     /* new dir? cache it */
 | |
|     if (strncmp(path,lastdir,sizeof(lastdir)) || reload_dir) {
 | |
|         DIR *dir = opendir(path);
 | |
|         if(!dir)
 | |
|             return -1; /* not a directory */
 | |
| 
 | |
|         name_buffer_length = 0;
 | |
|         dir_buffer_full = false;
 | |
|         
 | |
|         for ( i=0; i<MAX_FILES_IN_DIR; i++ ) {
 | |
|             int len;
 | |
|             struct dirent *entry = readdir(dir);
 | |
|             struct entry* dptr = &dircache[i];
 | |
|             if (!entry)
 | |
|                 break;
 | |
| 
 | |
|             len = strlen(entry->d_name);
 | |
| 
 | |
|             /* skip directories . and .. */
 | |
|             if ((entry->attribute & ATTR_DIRECTORY) &&
 | |
|                 (((len == 1) && 
 | |
|                   (!strncmp(entry->d_name, ".", 1))) ||
 | |
|                  ((len == 2) && 
 | |
|                   (!strncmp(entry->d_name, "..", 2))))) {
 | |
|                 i--;
 | |
|                 continue;
 | |
|             }
 | |
| 
 | |
|             /* Skip FAT volume ID */
 | |
|             if (entry->attribute & ATTR_VOLUME_ID) {
 | |
|                 i--;
 | |
|                 continue;
 | |
|             }
 | |
| 
 | |
|             /* filter out dotfiles and hidden files */
 | |
|             if (global_settings.dirfilter != SHOW_ALL &&
 | |
|                 ((entry->d_name[0]=='.') ||
 | |
|                  (entry->attribute & ATTR_HIDDEN))) {
 | |
|                 i--;
 | |
|                 continue;
 | |
|             }
 | |
| 
 | |
|             dptr->attr = entry->attribute;
 | |
| 
 | |
|             /* mark mp? and m3u files as such */
 | |
|             if ( !(dptr->attr & ATTR_DIRECTORY) && (len > 4) ) {
 | |
|                 if (!strcasecmp(&entry->d_name[len-4], ".mp3") ||
 | |
|                     (!strcasecmp(&entry->d_name[len-4], ".mp2")) ||
 | |
|                     (!strcasecmp(&entry->d_name[len-4], ".mpa")))
 | |
|                     dptr->attr |= TREE_ATTR_MPA;
 | |
|                 else if (!strcasecmp(&entry->d_name[len-4], ".m3u"))
 | |
|                     dptr->attr |= TREE_ATTR_M3U;
 | |
|                 else if (!strcasecmp(&entry->d_name[len-4], ".cfg"))
 | |
|                     dptr->attr |= TREE_ATTR_CFG;
 | |
|                 else if (!strcasecmp(&entry->d_name[len-4], ".wps"))
 | |
|                     dptr->attr |= TREE_ATTR_WPS;
 | |
|                 else if (!strcasecmp(&entry->d_name[len-4], ".txt"))
 | |
|                     dptr->attr |= TREE_ATTR_TXT;
 | |
|                 else if (!strcasecmp(&entry->d_name[len-4], ".lng"))
 | |
|                     dptr->attr |= TREE_ATTR_LNG;
 | |
| #ifdef HAVE_RECORDER_KEYPAD
 | |
|                 else if (!strcasecmp(&entry->d_name[len-4], ".fnt"))
 | |
|                     dptr->attr |= TREE_ATTR_FONT;
 | |
|                 else if (!strcasecmp(&entry->d_name[len-4], ".ajz"))
 | |
| #else
 | |
|                 else if (!strcasecmp(&entry->d_name[len-4], ".mod"))
 | |
| #endif
 | |
|                     dptr->attr |= TREE_ATTR_MOD;
 | |
|             }
 | |
| 
 | |
|             /* filter out all non-playlist files */
 | |
|             if ( global_settings.dirfilter == SHOW_PLAYLIST &&
 | |
|                  (!(dptr->attr &
 | |
|                     (ATTR_DIRECTORY|TREE_ATTR_M3U))) ) {
 | |
|                 i--;
 | |
|                 continue;
 | |
|             }
 | |
|             
 | |
|             /* filter out non-music files */
 | |
|             if ( global_settings.dirfilter == SHOW_MUSIC &&
 | |
|                  (!(dptr->attr &
 | |
|                     (ATTR_DIRECTORY|TREE_ATTR_MPA|TREE_ATTR_M3U))) ) {
 | |
|                 i--;
 | |
|                 continue;
 | |
|             }
 | |
| 
 | |
|             /* filter out non-supported files */
 | |
|             if ( global_settings.dirfilter == SHOW_SUPPORTED &&
 | |
|                  (!(dptr->attr & TREE_ATTR_MASK)) ) {
 | |
|                 i--;
 | |
|                 continue;
 | |
|             }
 | |
| 
 | |
|             if (len > NAME_BUFFER_SIZE - name_buffer_length - 1) {
 | |
|                 /* Tell the world that we ran out of buffer space */
 | |
|                 dir_buffer_full = true;
 | |
|                 break;
 | |
|             }
 | |
|             dptr->name = &name_buffer[name_buffer_length];
 | |
|             strcpy(dptr->name,entry->d_name);
 | |
|             name_buffer_length += len + 1;
 | |
|         }
 | |
|         filesindir = i;
 | |
|         closedir(dir);
 | |
|         strncpy(lastdir,path,sizeof(lastdir));
 | |
|         lastdir[sizeof(lastdir)-1] = 0;
 | |
|         qsort(dircache,filesindir,sizeof(struct entry),compare);
 | |
| 
 | |
|         if ( dir_buffer_full || filesindir == MAX_FILES_IN_DIR ) {
 | |
| #ifdef HAVE_LCD_CHARCELLS
 | |
|             lcd_double_height(false);
 | |
| #endif
 | |
|             lcd_clear_display();
 | |
|             lcd_puts(0,0,str(LANG_SHOWDIR_ERROR_BUFFER));
 | |
|             lcd_puts(0,1,str(LANG_SHOWDIR_ERROR_FULL));
 | |
|             lcd_update();
 | |
|             sleep(HZ*2);
 | |
|             lcd_clear_display();
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (start == -1)
 | |
|     {
 | |
|         int diff_files;
 | |
| 
 | |
|         /* use lastfile to determine start (default=0) */
 | |
|         start = 0;
 | |
| 
 | |
|         for (i=0; i<filesindir; i++)
 | |
|         {
 | |
|             if (!strcasecmp(dircache[i].name, lastfile))
 | |
|             {
 | |
|                 start = i;
 | |
|                 break;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         diff_files = filesindir - start;
 | |
|         if (diff_files < tree_max_on_screen)
 | |
|         {
 | |
|             int oldstart = start;
 | |
| 
 | |
|             start -= (tree_max_on_screen - diff_files);
 | |
|             if (start < 0)
 | |
|                 start = 0;
 | |
| 
 | |
|             dircursor = oldstart - start;
 | |
|         }
 | |
| 
 | |
|         dirstart = start;
 | |
|     }
 | |
| 
 | |
| #ifdef HAVE_LCD_CHARCELLS
 | |
|     lcd_stop_scroll();
 | |
|     lcd_double_height(false);
 | |
| #endif
 | |
|     lcd_clear_display();
 | |
| #ifdef HAVE_LCD_BITMAP
 | |
|     lcd_setmargins(MARGIN_X,MARGIN_Y); /* leave room for cursor and icon */
 | |
|     lcd_setfont(FONT_UI);
 | |
| #endif
 | |
| 
 | |
|     for ( i=start; i < start+tree_max_on_screen; i++ ) {
 | |
|         int len;
 | |
| 
 | |
|         if ( i >= filesindir )
 | |
|             break;
 | |
| 
 | |
|         len = strlen(dircache[i].name);
 | |
| 
 | |
|         switch ( dircache[i].attr & TREE_ATTR_MASK ) {
 | |
|             case ATTR_DIRECTORY:
 | |
|                 icon_type = Folder;
 | |
|                 break;
 | |
| 
 | |
|             case TREE_ATTR_M3U:
 | |
|                 icon_type = Playlist;
 | |
|                 break;
 | |
| 
 | |
|             case TREE_ATTR_MPA:
 | |
|                 icon_type = File;
 | |
|                 break;
 | |
| 
 | |
|             case TREE_ATTR_WPS:
 | |
|                 icon_type = Wps;
 | |
|                 break;
 | |
| 
 | |
|             case TREE_ATTR_CFG:
 | |
|                 icon_type = Config;
 | |
|                 break;
 | |
| 
 | |
|             case TREE_ATTR_TXT:
 | |
|                 icon_type = Text;
 | |
|                 break;
 | |
| 
 | |
|             case TREE_ATTR_LNG:
 | |
|                 icon_type = Language;
 | |
|                 break;
 | |
| 
 | |
|             case TREE_ATTR_MOD:
 | |
|                 icon_type = Mod_Ajz;
 | |
|                 break;
 | |
| 
 | |
| #ifdef HAVE_LCD_BITMAP
 | |
|             case TREE_ATTR_FONT:
 | |
|                 icon_type = Font;
 | |
|                 break;
 | |
| #endif
 | |
|             default:
 | |
| #ifdef HAVE_LCD_BITMAP
 | |
|                 icon_type = 0;
 | |
| #else
 | |
|                 icon_type = Unknown;
 | |
| #endif
 | |
|         }
 | |
| 
 | |
|         if (icon_type) {
 | |
| #ifdef HAVE_LCD_BITMAP
 | |
|             int offset=0;
 | |
|             if ( line_height > 8 )
 | |
|                 offset = (line_height - 8) / 2;
 | |
|             lcd_bitmap(bitmap_icons_6x8[icon_type], 
 | |
|                        CURSOR_X * 6 + CURSOR_WIDTH, 
 | |
|                        MARGIN_Y+(i-start)*line_height + offset,
 | |
|                        6, 8, true);
 | |
| #else
 | |
|             lcd_putc(LINE_X-1, i-start, icon_type);
 | |
| #endif
 | |
|         }
 | |
| 
 | |
|         showfileline(i-start, i, false); /* no scroll */
 | |
|     }
 | |
| 
 | |
| #ifdef HAVE_LCD_BITMAP
 | |
|     if (global_settings.scrollbar && filesindir > tree_max_on_screen) 
 | |
|         scrollbar(SCROLLBAR_X, SCROLLBAR_Y, SCROLLBAR_WIDTH - 1,
 | |
|                   LCD_HEIGHT - SCROLLBAR_Y, filesindir, start,
 | |
|                   start + tree_max_on_screen, VERTICAL);
 | |
| #endif
 | |
|     status_draw();
 | |
|     return filesindir;
 | |
| }
 | |
| 
 | |
| bool ask_resume(void)
 | |
| {
 | |
| #ifdef HAVE_LCD_CHARCELLS
 | |
|     lcd_double_height(false);
 | |
| #endif
 | |
| 
 | |
|     /* always resume? */
 | |
|     if ( global_settings.resume == RESUME_ON )
 | |
|         return true;
 | |
| 
 | |
|     lcd_clear_display();
 | |
|     lcd_puts(0,0,str(LANG_RESUME_ASK));
 | |
| #ifdef HAVE_LCD_CHARCELLS
 | |
|     status_draw();
 | |
|     lcd_puts(0,1,str(LANG_RESUME_CONFIRM_PLAYER));
 | |
| #else
 | |
|     lcd_puts(0,1,str(LANG_RESUME_CONFIRM_RECORDER));
 | |
|     lcd_puts(0,2,str(LANG_RESUME_CANCEL_RECORDER));
 | |
| #endif
 | |
|     lcd_update();
 | |
| 
 | |
|     switch (button_get(true)) {
 | |
|         case BUTTON_PLAY:
 | |
|             return true;
 | |
| 
 | |
|         case SYS_USB_CONNECTED:
 | |
|             usb_screen();
 | |
|             break;
 | |
|     }
 | |
| 
 | |
|     if ( global_settings.resume == RESUME_ASK_ONCE) {
 | |
|         global_settings.resume_index = -1;
 | |
|         settings_save();
 | |
|     }
 | |
| 
 | |
|     return false;
 | |
| }
 | |
| 
 | |
| void start_resume(void)
 | |
| {
 | |
|     if ( global_settings.resume &&
 | |
|          global_settings.resume_index != -1 ) {
 | |
|         int len = strlen(global_settings.resume_file);
 | |
| 
 | |
|         DEBUGF("Resume file %s\n",global_settings.resume_file);
 | |
|         DEBUGF("Resume index %X offset %X\n",
 | |
|                global_settings.resume_index,
 | |
|                global_settings.resume_offset);
 | |
|         DEBUGF("Resume shuffle %s seed %X\n",
 | |
|                global_settings.playlist_shuffle?"on":"off",
 | |
|                global_settings.resume_seed);
 | |
| 
 | |
|         /* playlist? */
 | |
|         if (!strcasecmp(&global_settings.resume_file[len-4], ".m3u")) {
 | |
|             char* slash;
 | |
| 
 | |
|             /* check that the file exists */
 | |
|             int fd = open(global_settings.resume_file, O_RDONLY);
 | |
|             if(fd<0)
 | |
|                 return;
 | |
|             close(fd);
 | |
| 
 | |
|             if (!ask_resume())
 | |
|                 return;
 | |
|             
 | |
|             slash = strrchr(global_settings.resume_file,'/');
 | |
|             if (slash) {
 | |
|                 *slash=0;
 | |
|                 play_list(global_settings.resume_file,
 | |
|                           slash+1,
 | |
|                           global_settings.resume_index,
 | |
|                           true, /* the index is AFTER shuffle */
 | |
|                           global_settings.resume_offset,
 | |
|                           global_settings.resume_seed,
 | |
|                           global_settings.resume_first_index,
 | |
|                           global_settings.queue_resume,
 | |
|                           global_settings.queue_resume_index);
 | |
|                 *slash='/';
 | |
|             }
 | |
|             else {
 | |
|                 /* check that the dir exists */
 | |
|                 DIR* dir = opendir(global_settings.resume_file);
 | |
|                 if(!dir)
 | |
|                     return;
 | |
|                 closedir(dir);
 | |
| 
 | |
|                 if (!ask_resume())
 | |
|                     return;
 | |
| 
 | |
|                 play_list("/",
 | |
|                           global_settings.resume_file,
 | |
|                           global_settings.resume_index,
 | |
|                           true,
 | |
|                           global_settings.resume_offset,
 | |
|                           global_settings.resume_seed,
 | |
|                           global_settings.resume_first_index,
 | |
|                           global_settings.queue_resume,
 | |
|                           global_settings.queue_resume_index);
 | |
|             }
 | |
|         }
 | |
|         else {
 | |
|             if (!ask_resume())
 | |
|                 return;
 | |
| 
 | |
|             if (showdir(global_settings.resume_file, 0) < 0 )
 | |
|                 return;
 | |
| 
 | |
|             lastdir[0] = '\0';
 | |
| 
 | |
|             build_playlist(global_settings.resume_index);
 | |
|             play_list(global_settings.resume_file,
 | |
|                       NULL, 
 | |
|                       global_settings.resume_index,
 | |
|                       true,
 | |
|                       global_settings.resume_offset,
 | |
|                       global_settings.resume_seed,
 | |
|                       global_settings.resume_first_index,
 | |
|                       global_settings.queue_resume,
 | |
|                       global_settings.queue_resume_index);
 | |
|         }
 | |
| 
 | |
|         status_set_playmode(STATUS_PLAY);
 | |
|         status_draw();
 | |
|         wps_show();
 | |
|     }
 | |
| }
 | |
| 
 | |
| void set_current_file(char *path)
 | |
| {
 | |
|     char *name;
 | |
|     unsigned int i;
 | |
| 
 | |
|     /* separate directory from filename */
 | |
|     name = strrchr(path+1,'/');
 | |
|     if (name)
 | |
|     {
 | |
|         *name = 0;
 | |
|         strcpy(currdir, path);
 | |
|         *name = '/';
 | |
|         name++;
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         strcpy(currdir, "/");
 | |
|         name = path+1;
 | |
|     }
 | |
| 
 | |
|     strcpy(lastfile, name);
 | |
| 
 | |
|     dircursor    =  0;
 | |
|     dirstart     = -1;
 | |
| 
 | |
|     if (strncmp(currdir,lastdir,sizeof(lastdir)))
 | |
|     {
 | |
|         dirlevel            =  0;
 | |
|         dirpos[dirlevel]    = -1;
 | |
|         cursorpos[dirlevel] =  0;
 | |
|         
 | |
|         /* use '/' to calculate dirlevel */
 | |
|         for (i=1; i<strlen(path)+1; i++)
 | |
|         {
 | |
|             if (path[i] == '/')
 | |
|             {
 | |
|                 dirlevel++;
 | |
|                 dirpos[dirlevel]    = -1;
 | |
|                 cursorpos[dirlevel] =  0;
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| #ifdef HAVE_LCD_BITMAP
 | |
| static int onplay_screen(char* dir, char* file)
 | |
| {
 | |
|     bool exit = false;
 | |
|     bool used = false;
 | |
|     bool playing = mpeg_status() & MPEG_STATUS_PLAY;
 | |
|     char buf[MAX_PATH];
 | |
|     struct entry* f = &dircache[dirstart + dircursor];
 | |
|     bool isdir = f->attr & ATTR_DIRECTORY;
 | |
|         
 | |
|     if ((dir[0]=='/') && (dir[1]==0))
 | |
|         snprintf(buf, sizeof buf, "%s%s", dir, file);
 | |
|     else
 | |
|         snprintf(buf, sizeof buf, "%s/%s", dir, file);
 | |
| 
 | |
|     lcd_clear_display();
 | |
| 
 | |
|     {
 | |
|         int w,h;
 | |
|         char* ptr;
 | |
| 
 | |
|         lcd_setfont(FONT_SYSFIXED);
 | |
| 
 | |
|         if (playing) {
 | |
|             ptr = str(LANG_QUEUE);
 | |
|             lcd_getstringsize(ptr,&w,&h);
 | |
|             lcd_putsxy((LCD_WIDTH-w)/2, h*2, ptr);
 | |
|             lcd_bitmap(bitmap_icons_7x8[Icon_Play],
 | |
|                        LCD_WIDTH/2 - 3, LCD_HEIGHT/2 - 4, 7, 8, true);
 | |
|         }
 | |
| 
 | |
|         /* don't delete directories */
 | |
|         if (!isdir) {
 | |
|             ptr = str(LANG_DELETE);
 | |
|             lcd_getstringsize(ptr,&w,&h);
 | |
|             lcd_putsxy(LCD_WIDTH - w, LCD_HEIGHT/2 - h/2, ptr);
 | |
|             lcd_bitmap(bitmap_icons_7x8[Icon_FastForward], 
 | |
|                        LCD_WIDTH/2 + 8, LCD_HEIGHT/2 - 4, 7, 8, true);
 | |
|         }
 | |
| 
 | |
|         ptr = str(LANG_RENAME);
 | |
|         lcd_getstringsize(ptr,&w,&h);
 | |
|         lcd_putsxy(0, LCD_HEIGHT/2 - h/2, ptr);
 | |
|         lcd_bitmap(bitmap_icons_7x8[Icon_FastBackward], 
 | |
|                    LCD_WIDTH/2 - 16, LCD_HEIGHT/2 - 4, 7, 8, true);
 | |
|     }
 | |
|     lcd_update();
 | |
| 
 | |
|     
 | |
|     while (!exit) {
 | |
|         switch (button_get(true)) {
 | |
|             case BUTTON_LEFT:
 | |
|             case BUTTON_ON | BUTTON_LEFT: {
 | |
|                 char newname[MAX_PATH];
 | |
|                 char* ptr = strrchr(buf, '/') + 1;
 | |
|                 int pathlen = (ptr - buf);
 | |
|                 strncpy(newname, buf, sizeof newname);
 | |
|                 if (!kbd_input(newname + pathlen, (sizeof newname)-pathlen)) {
 | |
|                     if (!strlen(buf+pathlen) || (rename(buf, newname) < 0)) {
 | |
|                         lcd_clear_display();
 | |
|                         lcd_puts(0,0,str(LANG_RENAME));
 | |
|                         lcd_puts(0,1,str(LANG_FAILED));
 | |
|                         lcd_update();
 | |
|                         sleep(HZ*2);
 | |
|                     }
 | |
|                     else
 | |
|                         reload_dir = true;
 | |
|                 }
 | |
|                 exit = true;
 | |
|                 break;
 | |
|             }
 | |
| 
 | |
|             case BUTTON_RIGHT:
 | |
|             case BUTTON_ON | BUTTON_RIGHT:
 | |
|                 /* don't delete directories */
 | |
|                 if (isdir)
 | |
|                     break;
 | |
|                 lcd_clear_display();
 | |
|                 lcd_puts(0,0,str(LANG_REALLY_DELETE));
 | |
|                 lcd_puts_scroll(0,1,file);
 | |
|                 lcd_puts(0,3,str(LANG_RESUME_CONFIRM_RECORDER));
 | |
|                 lcd_puts(0,4,str(LANG_RESUME_CANCEL_RECORDER));
 | |
|                 lcd_update();
 | |
|                 while (!exit) {
 | |
|                     int btn = button_get(true);
 | |
|                     switch (btn) {
 | |
|                         case BUTTON_PLAY:
 | |
|                         case BUTTON_PLAY | BUTTON_REL:
 | |
|                             if (!remove(buf)) {
 | |
|                                 reload_dir = true;
 | |
|                                 lcd_clear_display();
 | |
|                                 lcd_puts_scroll(0,0,file);
 | |
|                                 lcd_puts(0,1,str(LANG_DELETED));
 | |
|                                 lcd_update();
 | |
|                                 sleep(HZ);
 | |
|                                 exit = true;
 | |
|                             }
 | |
|                             break;
 | |
| 
 | |
|                         default:
 | |
|                             /* ignore button releases */
 | |
|                             if (!(btn & BUTTON_REL))
 | |
|                                 exit = true;
 | |
|                             break;
 | |
|                     }
 | |
|                 }
 | |
|                 break;
 | |
| 
 | |
|             case BUTTON_PLAY:
 | |
|             case BUTTON_ON | BUTTON_PLAY: {
 | |
|                 if (playing)
 | |
|                     queue_add(buf);
 | |
|                 exit = true;
 | |
|                 break;
 | |
|             }
 | |
| 
 | |
|             case BUTTON_ON | BUTTON_REL:
 | |
|                 used = true;
 | |
|                 break;
 | |
| 
 | |
|             case BUTTON_ON:
 | |
|                 if (used)
 | |
|                     exit = true;
 | |
|                 break;
 | |
| 
 | |
|             case BUTTON_OFF:
 | |
|                 exit = true;
 | |
|                 break;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     lcd_setfont(FONT_UI);
 | |
| 
 | |
|     return false;
 | |
| }
 | |
| 
 | |
| #else
 | |
| 
 | |
| static int onplay_screen(char* dir, char* file)
 | |
| {
 | |
|     bool exit = false;
 | |
|     bool playing = mpeg_status() & MPEG_STATUS_PLAY;
 | |
|     char buf[MAX_PATH];
 | |
|     struct entry* f = &dircache[dirstart + dircursor];
 | |
|     bool isdir = f->attr & ATTR_DIRECTORY;
 | |
|     struct menu_items items[3];
 | |
|     int ids[3];
 | |
|     int lastitem=0;
 | |
|     int m_handle;
 | |
|     int selected;
 | |
| 
 | |
|     if ((dir[0]=='/') && (dir[1]==0))
 | |
|         snprintf(buf, sizeof buf, "%s%s", dir, file);
 | |
|     else
 | |
|         snprintf(buf, sizeof buf, "%s/%s", dir, file);
 | |
| 
 | |
|     if (playing) {
 | |
|         items[lastitem].desc=str(LANG_QUEUE);
 | |
|         ids[lastitem]=1;
 | |
|         lastitem++;
 | |
|     }
 | |
|     
 | |
|     items[lastitem].desc=str(LANG_RENAME);
 | |
|     ids[lastitem]=2;
 | |
|     lastitem++;
 | |
| 
 | |
|     /* don't delete directories */
 | |
|     if (!isdir) {
 | |
|         items[lastitem].desc=str(LANG_DELETE);
 | |
|         ids[lastitem]=3;
 | |
|         lastitem++;
 | |
|     }
 | |
|     m_handle=menu_init(items, lastitem);
 | |
| 
 | |
|     selected=menu_show(m_handle);
 | |
|     if (selected>=0) {
 | |
|         switch(ids[selected]) {
 | |
|             case 1:
 | |
|                 if (playing)
 | |
|                     queue_add(buf);
 | |
|                 break;
 | |
| 
 | |
|             case 2: {
 | |
|                 char newname[MAX_PATH];
 | |
|                 char* ptr = strrchr(buf, '/') + 1;
 | |
|                 int pathlen = (ptr - buf);
 | |
|                 strncpy(newname, buf, sizeof newname);
 | |
|                 if (!kbd_input(newname + pathlen, (sizeof newname)-pathlen)) {
 | |
|                     if (rename(buf, newname) < 0) {
 | |
|                         lcd_clear_display();
 | |
|                         lcd_puts(0,0,str(LANG_RENAME));
 | |
|                         lcd_puts(0,1,str(LANG_FAILED));
 | |
|                         lcd_update();
 | |
|                         sleep(HZ*2);
 | |
|                     }
 | |
|                     else
 | |
|                         reload_dir = true;
 | |
|                 }
 | |
|             }
 | |
|             break;
 | |
| 
 | |
|             case 3:
 | |
|                 lcd_clear_display();
 | |
|                 lcd_puts_scroll(0,0,file);
 | |
|                 lcd_puts(0,1,str(LANG_REALLY_DELETE));
 | |
|                 lcd_update();
 | |
|                 while (!exit) {
 | |
|                     int btn = button_get(true);
 | |
|                     switch (btn) {
 | |
|                         case BUTTON_PLAY:
 | |
|                             if (!remove(buf)) {
 | |
|                                 reload_dir = true;
 | |
|                                 lcd_clear_display();
 | |
|                                 lcd_puts_scroll(0,0,file);
 | |
|                                 lcd_puts(0,1,str(LANG_DELETED));
 | |
|                                 lcd_update();
 | |
|                                 sleep(HZ);
 | |
|                                 exit = true;
 | |
|                             }
 | |
|                             break;
 | |
|           
 | |
|                         default:
 | |
|                             /* ignore button releases */
 | |
|                             if (!(btn & BUTTON_REL))
 | |
|                                 exit = true;
 | |
|                             break;
 | |
|                     }
 | |
|                 }
 | |
|                 break;
 | |
|         }
 | |
|     }
 | |
|     menu_exit(m_handle);
 | |
|     return false;
 | |
| }
 | |
| #endif
 | |
| 
 | |
| static bool handle_on(int* ds, int* dc, int numentries, int tree_max_on_screen)
 | |
| {
 | |
|     bool exit = false;
 | |
|     bool used = false;
 | |
| 
 | |
|     int dirstart = *ds;
 | |
|     int dircursor = *dc;
 | |
| 
 | |
|     while (!exit) {
 | |
|         switch (button_get(true)) {
 | |
|             case TREE_PREV:
 | |
|             case BUTTON_ON | TREE_PREV:
 | |
|             case BUTTON_ON | TREE_PREV | BUTTON_REPEAT:
 | |
|                 used = true;
 | |
|                 if ( dirstart ) {
 | |
|                     dirstart -= tree_max_on_screen;
 | |
|                     if ( dirstart < 0 )
 | |
|                         dirstart = 0;
 | |
|                 }
 | |
|                 else
 | |
|                     dircursor = 0;
 | |
|                 break;
 | |
|                 
 | |
|             case TREE_NEXT:
 | |
|             case BUTTON_ON | TREE_NEXT:
 | |
|             case BUTTON_ON | TREE_NEXT | BUTTON_REPEAT:
 | |
|                 used = true;
 | |
|                 if ( dirstart < numentries - tree_max_on_screen ) {
 | |
|                     dirstart += tree_max_on_screen;
 | |
|                     if ( dirstart > 
 | |
|                          numentries - tree_max_on_screen )
 | |
|                         dirstart = numentries - tree_max_on_screen;
 | |
|                 }
 | |
|                 else
 | |
|                     dircursor = numentries - dirstart - 1;
 | |
|                 break;
 | |
| 
 | |
| 
 | |
|             case BUTTON_PLAY:
 | |
|             case BUTTON_ON | BUTTON_PLAY:
 | |
|                 onplay_screen(currdir, dircache[dircursor+dirstart].name);
 | |
|                 exit = true;
 | |
|                 used = true;
 | |
|                 break;
 | |
|                 
 | |
|             case BUTTON_ON | BUTTON_REL:
 | |
|             case BUTTON_ON | TREE_PREV | BUTTON_REL:
 | |
|             case BUTTON_ON | TREE_NEXT | BUTTON_REL:
 | |
|                 exit = true;
 | |
|                 break;
 | |
|         }
 | |
|         if ( used && !exit ) {
 | |
|             showdir(currdir, dirstart);
 | |
|             put_cursorxy(CURSOR_X, CURSOR_Y + dircursor, true);
 | |
|             lcd_update();
 | |
|         }
 | |
|     }
 | |
|     *ds = dirstart;
 | |
|     *dc = dircursor;
 | |
| 
 | |
|     return used;
 | |
| }
 | |
| 
 | |
| bool dirbrowse(char *root)
 | |
| {
 | |
|     int numentries=0;
 | |
|     char buf[MAX_PATH];
 | |
|     int i;
 | |
|     int lasti=-1;
 | |
|     int button;
 | |
|     int tree_max_on_screen;
 | |
|     bool reload_root = false;
 | |
|     int lastfilter = global_settings.dirfilter;
 | |
|     bool lastsortcase = global_settings.sort_case;
 | |
|     int lastdircursor=-1;
 | |
|     bool need_update = true;
 | |
| 
 | |
|     bool update_all = false; /* set this to true when the whole file list
 | |
|                                 has been refreshed on screen */
 | |
| 
 | |
| #ifdef HAVE_LCD_BITMAP
 | |
|     int fw, fh;
 | |
|     lcd_getstringsize("A", &fw, &fh);
 | |
|     tree_max_on_screen = (LCD_HEIGHT - MARGIN_Y) / fh;
 | |
| #else
 | |
|     tree_max_on_screen = TREE_MAX_ON_SCREEN;
 | |
| #endif
 | |
| 
 | |
|     dircursor=0;
 | |
|     dirstart=0;
 | |
|     dirlevel=0;
 | |
| 
 | |
|     memcpy(currdir,root,sizeof(currdir));
 | |
| 
 | |
|     start_resume();
 | |
| 
 | |
|     numentries = showdir(currdir, dirstart);
 | |
|     if (numentries == -1) 
 | |
|         return false;  /* currdir is not a directory */
 | |
|     update_all = true;
 | |
| 
 | |
|     put_cursorxy(CURSOR_X, CURSOR_Y + dircursor, true);
 | |
| 
 | |
|     while(1) {
 | |
|         struct entry* file = &dircache[dircursor+dirstart];
 | |
| 
 | |
|         bool restore = false;
 | |
| 
 | |
|         button = button_get_w_tmo(HZ/5);
 | |
|         switch ( button ) {
 | |
|             case TREE_EXIT:
 | |
|             case TREE_EXIT | BUTTON_REPEAT:
 | |
|                 i=strlen(currdir);
 | |
|                 if (i>1) {
 | |
|                     while (currdir[i-1]!='/')
 | |
|                         i--;
 | |
|                     strcpy(buf,&currdir[i]);
 | |
|                     if (i==1)
 | |
|                         currdir[i]=0;
 | |
|                     else
 | |
|                         currdir[i-1]=0;
 | |
| 
 | |
|                     dirlevel--;
 | |
|                     if ( dirlevel < MAX_DIR_LEVELS ) {
 | |
|                         dirstart = dirpos[dirlevel];
 | |
|                         dircursor = cursorpos[dirlevel];
 | |
|                     }
 | |
|                     else
 | |
|                         dirstart = dircursor = 0;
 | |
| 
 | |
|                     if (dirstart == -1)
 | |
|                         strcpy(lastfile, buf);
 | |
| 
 | |
|                     restore = true;
 | |
|                 }
 | |
|                 break;
 | |
| 
 | |
| #ifdef HAVE_RECORDER_KEYPAD
 | |
|             case BUTTON_OFF:
 | |
|                 mpeg_stop();
 | |
|                 status_set_playmode(STATUS_STOP);
 | |
|                 status_draw();
 | |
|                 restore = true;
 | |
|                 break;
 | |
| 
 | |
|             case BUTTON_OFF | BUTTON_REL:
 | |
| #else
 | |
|             case BUTTON_STOP | BUTTON_REL:
 | |
| #endif
 | |
|                 settings_save();
 | |
|                 break;
 | |
|                 
 | |
| 
 | |
|             case TREE_ENTER:
 | |
|             case TREE_ENTER | BUTTON_REPEAT:
 | |
| #ifdef HAVE_RECORDER_KEYPAD
 | |
|             case BUTTON_PLAY:
 | |
|             case BUTTON_PLAY | BUTTON_REPEAT: 
 | |
| #endif
 | |
|                 if ( !numentries )
 | |
|                     break;
 | |
|                 if ((currdir[0]=='/') && (currdir[1]==0)) {
 | |
|                     snprintf(buf,sizeof(buf),"%s%s",currdir, file->name);
 | |
|                 } else {
 | |
|                     snprintf(buf,sizeof(buf),"%s/%s",currdir, file->name);
 | |
|                 }
 | |
| 
 | |
|                 if (file->attr & ATTR_DIRECTORY) {
 | |
|                     memcpy(currdir,buf,sizeof(currdir));
 | |
|                     if ( dirlevel < MAX_DIR_LEVELS ) {
 | |
|                         dirpos[dirlevel] = dirstart;
 | |
|                         cursorpos[dirlevel] = dircursor;
 | |
|                     }
 | |
|                     dirlevel++;
 | |
|                     dircursor=0;
 | |
|                     dirstart=0;
 | |
|                 }
 | |
|                 else {
 | |
|                     int seed = current_tick;
 | |
|                     bool play = false;
 | |
|                     int start_index=0;
 | |
| 
 | |
|                     lcd_stop_scroll();
 | |
|                     switch ( file->attr & TREE_ATTR_MASK ) {
 | |
|                         case TREE_ATTR_M3U:
 | |
|                             if ( global_settings.resume )
 | |
|                                 snprintf(global_settings.resume_file,
 | |
|                                          MAX_PATH, "%s/%s",
 | |
|                                          currdir, file->name);
 | |
|                             play_list(currdir, file->name, 0, false, 0,
 | |
|                                       seed, 0, 0, -1);
 | |
|                             start_index = 0;
 | |
|                             play = true;
 | |
|                             break;
 | |
| 
 | |
|                         case TREE_ATTR_MPA:
 | |
|                             if ( global_settings.resume )
 | |
|                                 strncpy(global_settings.resume_file,
 | |
|                                         currdir, MAX_PATH);
 | |
|                             start_index =
 | |
|                                 build_playlist(dircursor+dirstart);
 | |
|                             
 | |
|                                 /* it is important that we get back the index
 | |
|                                    in the (shuffled) list and store that */
 | |
|                             start_index = play_list(currdir, NULL,
 | |
|                                                     start_index, false,
 | |
|                                                     0, seed, 0, 0, -1);
 | |
|                             play = true;
 | |
|                             break;
 | |
| 
 | |
|                             /* wps config file */
 | |
|                         case TREE_ATTR_WPS:
 | |
|                             wps_load(buf,true);
 | |
|                             set_file(buf, global_settings.wps_file,
 | |
|                                      MAX_FILENAME);
 | |
|                             restore = true;
 | |
|                             break;
 | |
| 
 | |
|                         case TREE_ATTR_CFG:
 | |
|                             if (!settings_load_config(buf))
 | |
|                                 break;
 | |
|                             lcd_clear_display();
 | |
|                             lcd_puts(0,0,str(LANG_SETTINGS_LOADED1));
 | |
|                             lcd_puts(0,1,str(LANG_SETTINGS_LOADED2));
 | |
| #ifdef HAVE_LCD_BITMAP
 | |
|                             lcd_update();
 | |
| 
 | |
|                             /* maybe we have a new font */
 | |
|                             lcd_getstringsize("A", &fw, &fh);
 | |
|                             tree_max_on_screen = (LCD_HEIGHT - MARGIN_Y) / fh;
 | |
|                             /* make sure cursor is on screen */
 | |
|                             while ( dircursor > tree_max_on_screen ) 
 | |
|                             {
 | |
|                                 dircursor--;
 | |
|                                 dirstart++;
 | |
|                             }
 | |
| #endif
 | |
|                             sleep(HZ/2);
 | |
|                             restore = true;
 | |
|                             break;
 | |
| 
 | |
|                         case TREE_ATTR_TXT:
 | |
|                             viewer_run(buf);
 | |
|                             restore = true;
 | |
|                             break;
 | |
| 
 | |
|                         case TREE_ATTR_LNG:
 | |
|                             if(!lang_load(buf)) {
 | |
|                                 set_file(buf, global_settings.lang_file,
 | |
|                                          MAX_FILENAME);
 | |
| 
 | |
|                                 lcd_clear_display();
 | |
| #ifdef HAVE_LCD_CHARCELLS
 | |
|                                 lcd_puts(0, 0, str(LANG_LANGUAGE_LOADED));
 | |
| #else
 | |
|                                 lcd_getstringsize(str(LANG_LANGUAGE_LOADED),
 | |
|                                                   &fw, &fh);
 | |
|                                 if(fw>LCD_WIDTH)
 | |
|                                     fw=0;
 | |
|                                 else
 | |
|                                     fw=LCD_WIDTH/2 - fw/2;
 | |
|                                 
 | |
|                                 lcd_putsxy(fw, LCD_HEIGHT/2 - fh/2,
 | |
|                                            str(LANG_LANGUAGE_LOADED));
 | |
| #endif
 | |
|                                 lcd_update();
 | |
|                                 sleep(HZ);            
 | |
|                                 restore = true;
 | |
|                             }
 | |
|                             break;
 | |
| 
 | |
| #ifdef HAVE_LCD_BITMAP
 | |
|                         case TREE_ATTR_FONT:
 | |
|                             font_load(buf);
 | |
|                             set_file(buf, global_settings.font_file,
 | |
|                                      MAX_FILENAME);
 | |
| 
 | |
|                             lcd_getstringsize("A", &fw, &fh);
 | |
|                             tree_max_on_screen = (LCD_HEIGHT - MARGIN_Y) / fh;
 | |
|                             /* make sure cursor is on screen */
 | |
|                             while ( dircursor > tree_max_on_screen ) {
 | |
|                                 dircursor--;
 | |
|                                 dirstart++;
 | |
|                             }
 | |
|                             restore = true;
 | |
|                             break;
 | |
| #endif
 | |
| 
 | |
| #ifndef SIMULATOR
 | |
|                             /* firmware file */
 | |
|                         case TREE_ATTR_MOD:
 | |
|                             rolo_load(buf);
 | |
|                             break;
 | |
| #endif
 | |
|                     }
 | |
| 
 | |
|                     if ( play ) {
 | |
|                         if ( global_settings.resume ) {
 | |
|                             /* the resume_index must always be the index in the
 | |
|                                shuffled list in case shuffle is enabled */
 | |
|                             global_settings.resume_index = start_index;
 | |
|                             global_settings.resume_offset = 0;
 | |
|                             global_settings.resume_first_index =
 | |
|                                 playlist_first_index();
 | |
|                             global_settings.resume_seed = seed;
 | |
|                             settings_save();
 | |
|                         }
 | |
| 
 | |
|                         status_set_playmode(STATUS_PLAY);
 | |
|                         status_draw();
 | |
|                         lcd_stop_scroll();
 | |
|                         if ( wps_show() == SYS_USB_CONNECTED ) {
 | |
|                             reload_root = true;
 | |
|                         }
 | |
| #ifdef HAVE_LCD_BITMAP
 | |
|                         tree_max_on_screen = (LCD_HEIGHT - MARGIN_Y) / fh;
 | |
| #endif
 | |
|                     }
 | |
|                 }
 | |
|                 restore = true;
 | |
|                 break;
 | |
| 
 | |
|             case TREE_PREV:
 | |
|             case TREE_PREV | BUTTON_REPEAT:
 | |
|             case BUTTON_VOL_UP:
 | |
|                 if(filesindir) {
 | |
|                     if(dircursor) {
 | |
|                         put_cursorxy(CURSOR_X, CURSOR_Y + dircursor, false);
 | |
|                         dircursor--;
 | |
|                         put_cursorxy(CURSOR_X, CURSOR_Y + dircursor, true);
 | |
|                     }
 | |
|                     else {
 | |
|                         if (dirstart) {
 | |
|                             dirstart--;
 | |
|                             numentries = showdir(currdir, dirstart);
 | |
|                             update_all=true;
 | |
|                             put_cursorxy(CURSOR_X, CURSOR_Y + dircursor, true);
 | |
|                         }
 | |
|                         else {
 | |
|                             if (numentries < tree_max_on_screen) {
 | |
|                                 put_cursorxy(CURSOR_X, CURSOR_Y + dircursor,
 | |
|                                              false);
 | |
|                                 dircursor = numentries - 1;
 | |
|                                 put_cursorxy(CURSOR_X, CURSOR_Y + dircursor,
 | |
|                                              true);
 | |
|                             }
 | |
|                             else {
 | |
|                                 dirstart = numentries - tree_max_on_screen;
 | |
|                                 dircursor = tree_max_on_screen - 1;
 | |
|                                 numentries = showdir(currdir, dirstart);
 | |
|                                 update_all = true;
 | |
|                                 put_cursorxy(CURSOR_X, CURSOR_Y +
 | |
|                                              tree_max_on_screen - 1, true);
 | |
|                             }
 | |
|                         }
 | |
|                     }
 | |
|                     need_update = true;
 | |
|                 }
 | |
|                 break;
 | |
| 
 | |
|             case TREE_NEXT:
 | |
|             case TREE_NEXT | BUTTON_REPEAT:
 | |
|             case BUTTON_VOL_DOWN:
 | |
|                 if(filesindir)
 | |
|                 {
 | |
|                     if (dircursor + dirstart + 1 < numentries ) {
 | |
|                         if(dircursor+1 < tree_max_on_screen) {
 | |
|                             put_cursorxy(CURSOR_X, CURSOR_Y + dircursor,
 | |
|                                          false);
 | |
|                             dircursor++;
 | |
|                             put_cursorxy(CURSOR_X, CURSOR_Y + dircursor, true);
 | |
|                         } 
 | |
|                         else {
 | |
|                             dirstart++;
 | |
|                             numentries = showdir(currdir, dirstart);
 | |
|                             update_all = true;
 | |
|                             put_cursorxy(CURSOR_X, CURSOR_Y + dircursor, true);
 | |
|                         }
 | |
|                     }
 | |
|                     else {
 | |
|                         if(numentries < tree_max_on_screen) {
 | |
|                             put_cursorxy(CURSOR_X, CURSOR_Y + dircursor,
 | |
|                                          false);
 | |
|                             dirstart = dircursor = 0;
 | |
|                             put_cursorxy(CURSOR_X, CURSOR_Y + dircursor, true);
 | |
|                         } 
 | |
|                         else {
 | |
|                             dirstart = dircursor = 0;
 | |
|                             numentries = showdir(currdir, dirstart);
 | |
|                             update_all=true;
 | |
|                             put_cursorxy(CURSOR_X, CURSOR_Y + dircursor, true);
 | |
|                         }
 | |
|                     }
 | |
|                     need_update = true;
 | |
|                 }
 | |
|                 break;
 | |
| 
 | |
|             case TREE_MENU:
 | |
|                 lcd_stop_scroll();
 | |
|                 if (main_menu())
 | |
|                     reload_root = true;
 | |
|                 restore = true;
 | |
|                 break;
 | |
| 
 | |
|             case BUTTON_ON:
 | |
|                 if (handle_on(&dirstart, &dircursor, numentries,
 | |
|                               tree_max_on_screen))
 | |
|                 {
 | |
|                     /* start scroll */
 | |
|                     reload_dir = true;
 | |
|                     restore = true;
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     if (mpeg_status() & MPEG_STATUS_PLAY)
 | |
|                     {
 | |
|                         lcd_stop_scroll();
 | |
|                         if (wps_show() == SYS_USB_CONNECTED)
 | |
|                             reload_root = true;
 | |
| #ifdef HAVE_LCD_BITMAP
 | |
|                         tree_max_on_screen = (LCD_HEIGHT - MARGIN_Y) / fh;
 | |
| #endif
 | |
|                         restore = true;
 | |
|                     }
 | |
|                 }
 | |
|                 break;
 | |
| 
 | |
| #ifdef HAVE_RECORDER_KEYPAD
 | |
|             case BUTTON_F2:
 | |
|                 if (f2_screen())
 | |
|                     reload_root = true;
 | |
|                 restore = true;
 | |
|                 break;
 | |
| 
 | |
|             case BUTTON_F3:
 | |
|                 if (f3_screen())
 | |
|                     reload_root = true;
 | |
| 
 | |
| #ifdef HAVE_LCD_BITMAP
 | |
|                 tree_max_on_screen = (LCD_HEIGHT - MARGIN_Y) / fh;
 | |
| #endif
 | |
|                 restore = true;
 | |
|                 break;
 | |
| #endif
 | |
| 
 | |
|             case SYS_USB_CONNECTED:
 | |
|                 status_set_playmode(STATUS_STOP);
 | |
|                 usb_screen();
 | |
|                 reload_root = true;
 | |
|                 break;
 | |
| 
 | |
|             case BUTTON_NONE:
 | |
|                 status_draw();
 | |
|                 break;
 | |
|         }
 | |
| 
 | |
|         if ( button )
 | |
|             ata_spin();
 | |
|         
 | |
|         /* do we need to rescan dir? */
 | |
|         if (reload_dir || reload_root ||
 | |
|             lastfilter != global_settings.dirfilter ||
 | |
|             lastsortcase != global_settings.sort_case)
 | |
|         {
 | |
|             if ( reload_root ) {
 | |
|                 strcpy(currdir, "/");
 | |
|                 dirlevel = 0;
 | |
|                 reload_root = false;
 | |
|             }
 | |
|             if (! reload_dir )
 | |
|             {
 | |
|                 dircursor = 0;
 | |
|                 dirstart = 0;
 | |
|                 lastdir[0] = 0;
 | |
|             }
 | |
|             lastfilter = global_settings.dirfilter;
 | |
|             lastsortcase = global_settings.sort_case;
 | |
|             restore = true;
 | |
|         }
 | |
| 
 | |
|         if (restore || reload_dir) {
 | |
|             /* restore display */
 | |
|             /* We need to adjust if the number of lines on screen have
 | |
|                changed because of a status bar change */
 | |
|             if(CURSOR_Y+LINE_Y+dircursor>tree_max_on_screen) {
 | |
|                 dirstart++;
 | |
|                 dircursor--;
 | |
|             }
 | |
| #ifdef HAVE_LCD_BITMAP
 | |
|             /* the sub-screen might've ruined the margins */
 | |
|             lcd_setmargins(MARGIN_X,MARGIN_Y); /* leave room for cursor and
 | |
|                                                   icon */
 | |
| #endif
 | |
|             numentries = showdir(currdir, dirstart);
 | |
|             update_all = true;
 | |
|             put_cursorxy(CURSOR_X, CURSOR_Y + dircursor, true);
 | |
|             
 | |
|             need_update = true;
 | |
|             reload_dir = false;
 | |
|         }
 | |
| 
 | |
|         if ( numentries && need_update) {
 | |
|             i = dirstart+dircursor;
 | |
| 
 | |
|             /* if MP3 filter is on, cut off the extension */
 | |
|             if(lasti!=i || restore) {
 | |
|                 lcd_stop_scroll();
 | |
| 
 | |
|                 /* So if lastdircursor and dircursor differ, and then full
 | |
|                    screen was not refreshed, restore the previous line */
 | |
|                 if ((lastdircursor != dircursor) && !update_all ) {
 | |
|                     showfileline(lastdircursor, lasti, false); /* no scroll */
 | |
|                 }
 | |
|                 lasti=i;
 | |
|                 lastdircursor=dircursor;
 | |
| 
 | |
|                 showfileline(dircursor, i, true); /* scroll please */
 | |
|                 need_update = true;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         if(need_update) {
 | |
|             lcd_update();
 | |
| 
 | |
|             need_update = false;
 | |
|             update_all = false;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| static int plsize = 0;
 | |
| static bool add_dir(char* dirname, int fd)
 | |
| {
 | |
|     bool abort = false;
 | |
|     char buf[MAX_PATH/2]; /* saving a little stack... */
 | |
|     DIR* dir;
 | |
|     
 | |
|     /* check for user abort */
 | |
| #ifdef HAVE_PLAYER_KEYPAD
 | |
|     if (button_get(false) == BUTTON_STOP)
 | |
| #else
 | |
|     if (button_get(false) == BUTTON_OFF)
 | |
| #endif
 | |
|         return true;
 | |
|     
 | |
|     dir = opendir(dirname);
 | |
|     if(!dir)
 | |
|         return true;
 | |
| 
 | |
|     while (true) {
 | |
|         struct dirent *entry;
 | |
| 
 | |
|         entry = readdir(dir);
 | |
|         if (!entry)
 | |
|             break;
 | |
|         if (entry->attribute & ATTR_DIRECTORY) {
 | |
|             if (!strcmp(entry->d_name, ".") ||
 | |
|                 !strcmp(entry->d_name, ".."))
 | |
|                 continue;
 | |
|             snprintf(buf, sizeof buf, "%s/%s", dirname, entry->d_name);
 | |
|             if (add_dir(buf,fd)) {
 | |
|                 abort = true;
 | |
|                 break;
 | |
|             }
 | |
|         }
 | |
|         else {
 | |
|             int x = strlen(entry->d_name);
 | |
|             if ((!strcasecmp(&entry->d_name[x-4], ".mp3")) ||
 | |
|                 (!strcasecmp(&entry->d_name[x-4], ".mp2")) ||
 | |
|                 (!strcasecmp(&entry->d_name[x-4], ".mpa")))
 | |
|             {
 | |
|                 write(fd, dirname, strlen(dirname));
 | |
|                 write(fd, "/", 1);
 | |
|                 write(fd, entry->d_name, x);
 | |
|                 write(fd, "\n", 1);
 | |
| 
 | |
|                 plsize++;
 | |
|                 snprintf(buf, sizeof buf, "%d", plsize);
 | |
| #ifdef HAVE_LCD_BITMAP
 | |
|                 lcd_puts(0,4,buf);
 | |
|                 lcd_update();
 | |
| #else
 | |
|                 x = 10;
 | |
|                 if (plsize > 999)
 | |
|                     x=7;
 | |
|                 else {
 | |
|                     if (plsize > 99)
 | |
|                         x=8;
 | |
|                     else {
 | |
|                         if (plsize > 9)
 | |
|                             x=9;
 | |
|                     }
 | |
|                 }
 | |
|                 lcd_puts(x,0,buf);
 | |
| #endif
 | |
|             }
 | |
|         }
 | |
|     }
 | |
|     closedir(dir);
 | |
| 
 | |
|     return abort;
 | |
| }
 | |
| 
 | |
| bool create_playlist(void)
 | |
| {
 | |
|     int fd;
 | |
|     char filename[MAX_PATH];
 | |
| 
 | |
|     snprintf(filename, sizeof filename, "%s.m3u",
 | |
|              currdir[1] ? currdir : "/root");
 | |
| 
 | |
|     lcd_clear_display();
 | |
|     lcd_puts(0,0,str(LANG_CREATING));
 | |
|     lcd_puts_scroll(0,1,filename);
 | |
|     lcd_update();
 | |
| 
 | |
|     fd = creat(filename,0);
 | |
|     if (!fd)
 | |
|         return false;
 | |
| 
 | |
|     plsize = 0;
 | |
|     add_dir(currdir[1] ? currdir : "/", fd);
 | |
|     close(fd);
 | |
|     sleep(HZ);
 | |
|     
 | |
|     return false;
 | |
| }
 | |
| 
 |