2c7b127dd7
git-svn-id: svn://svn.rockbox.org/rockbox/trunk@12811 a1c6a512-1295-4272-9138-f99709370657
2656 lines
88 KiB
C
2656 lines
88 KiB
C
/***************************************************************************
|
|
* __________ __ ___.
|
|
* Open \______ \ ____ ____ | | _\_ |__ _______ ___
|
|
* Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
|
|
* Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
|
|
* Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
|
|
* \/ \/ \/ \/ \/
|
|
* $Id$
|
|
*
|
|
* Copyright (C) 2002 Björn 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 "gwps-common.h"
|
|
#include "font.h"
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <stdlib.h>
|
|
#include "system.h"
|
|
#include "settings.h"
|
|
#include "rtc.h"
|
|
#include "audio.h"
|
|
#include "status.h"
|
|
#include "power.h"
|
|
#include "powermgmt.h"
|
|
#include "sound.h"
|
|
#include "debug.h"
|
|
#ifdef HAVE_LCD_CHARCELLS
|
|
#include "hwcompat.h"
|
|
#endif
|
|
#include "abrepeat.h"
|
|
#include "mp3_playback.h"
|
|
#include "backlight.h"
|
|
#include "lang.h"
|
|
#include "misc.h"
|
|
#include "splash.h"
|
|
#include "scrollbar.h"
|
|
#include "led.h"
|
|
#include "lcd.h"
|
|
#ifdef HAVE_LCD_BITMAP
|
|
#include "peakmeter.h"
|
|
/* Image stuff */
|
|
#include "bmp.h"
|
|
#include "atoi.h"
|
|
#endif
|
|
#if LCD_DEPTH > 1
|
|
#include "backdrop.h"
|
|
#endif
|
|
#include "dsp.h"
|
|
#include "action.h"
|
|
#include "cuesheet.h"
|
|
|
|
#ifdef HAVE_LCD_CHARCELLS
|
|
static bool draw_player_progress(struct gui_wps *gwps);
|
|
static void draw_player_fullbar(struct gui_wps *gwps,
|
|
char* buf, int buf_size);
|
|
#endif
|
|
|
|
#define FF_REWIND_MAX_PERCENT 3 /* cap ff/rewind step size at max % of file */
|
|
/* 3% of 30min file == 54s step size */
|
|
#define MIN_FF_REWIND_STEP 500
|
|
|
|
/* Skip leading UTF-8 BOM, if present. */
|
|
static char* skip_utf8_bom(char* buf)
|
|
{
|
|
unsigned char* s = (unsigned char*) buf;
|
|
|
|
if(s[0] == 0xef && s[1] == 0xbb && s[2] == 0xbf)
|
|
{
|
|
buf += 3;
|
|
}
|
|
|
|
return buf;
|
|
}
|
|
|
|
/*
|
|
* returns the image_id between
|
|
* a..z and A..Z
|
|
*/
|
|
#ifdef HAVE_LCD_BITMAP
|
|
static int get_image_id(int c)
|
|
{
|
|
if(c >= 'a' && c <= 'z')
|
|
return c - 'a';
|
|
else if(c >= 'A' && c <= 'Z')
|
|
return c - 'A' + 26;
|
|
else
|
|
return -1;
|
|
}
|
|
#endif
|
|
|
|
/*
|
|
* parse the given buffer for following static tags:
|
|
* %x - load image for always display
|
|
* %X - load backdrop image
|
|
* %xl - preload image
|
|
* %we - enable statusbar on wps regardless of the global setting
|
|
* %wd - disable statusbar on wps regardless of the global setting
|
|
* and also for:
|
|
* # - a comment line
|
|
*
|
|
* it returns true if one of these tags is found and handled
|
|
* false otherwise
|
|
*/
|
|
bool wps_data_preload_tags(struct wps_data *data, char *buf,
|
|
const char *bmpdir, size_t bmpdirlen)
|
|
{
|
|
if(!data || !buf) return false;
|
|
|
|
char c;
|
|
#ifndef HAVE_LCD_BITMAP
|
|
/* no bitmap-lcd == no bitmap loading */
|
|
(void)bmpdir;
|
|
(void)bmpdirlen;
|
|
#endif
|
|
buf = skip_utf8_bom(buf);
|
|
|
|
if(*buf == '#')
|
|
return true;
|
|
if('%' != *buf)
|
|
return false;
|
|
buf++;
|
|
|
|
c = *buf;
|
|
switch (c)
|
|
{
|
|
#ifdef HAVE_LCD_BITMAP
|
|
case 'w':
|
|
/*
|
|
* if tag found then return because these two tags must be on
|
|
* must be on their own line
|
|
*/
|
|
if(*(buf+1) == 'd' || *(buf+1) == 'e')
|
|
{
|
|
data->wps_sb_tag = true;
|
|
if( *(buf+1) == 'e' )
|
|
data->show_sb_on_wps = true;
|
|
return true;
|
|
}
|
|
break;
|
|
|
|
#if LCD_DEPTH > 1
|
|
case 'X':
|
|
/* Backdrop image - must be the same size as the LCD */
|
|
{
|
|
char *ptr = buf+2;
|
|
char *pos = NULL;
|
|
char imgname[MAX_PATH];
|
|
|
|
/* format: %X|filename.bmp| */
|
|
|
|
/* get filename */
|
|
pos = strchr(ptr, '|');
|
|
if ((pos - ptr) <
|
|
(int)sizeof(imgname)-ROCKBOX_DIR_LEN-2)
|
|
{
|
|
memcpy(imgname, bmpdir, bmpdirlen);
|
|
imgname[bmpdirlen] = '/';
|
|
memcpy(&imgname[bmpdirlen+1],
|
|
ptr, pos - ptr);
|
|
imgname[bmpdirlen+1+pos-ptr] = 0;
|
|
}
|
|
else
|
|
{
|
|
/* filename too long */
|
|
imgname[0] = 0;
|
|
}
|
|
|
|
/* load the image */
|
|
return load_wps_backdrop(imgname);
|
|
}
|
|
|
|
break;
|
|
#endif
|
|
|
|
case 'P':
|
|
/* progress bar image */
|
|
{
|
|
int ret = 0;
|
|
char *ptr = buf+2;
|
|
char *pos = NULL;
|
|
char imgname[MAX_PATH];
|
|
|
|
/* format: %P|filename.bmp| */
|
|
{
|
|
/* get filename */
|
|
pos = strchr(ptr, '|');
|
|
if ((pos - ptr) <
|
|
(int)sizeof(imgname)-ROCKBOX_DIR_LEN-2)
|
|
{
|
|
memcpy(imgname, bmpdir, bmpdirlen);
|
|
imgname[bmpdirlen] = '/';
|
|
memcpy(&imgname[bmpdirlen+1],
|
|
ptr, pos - ptr);
|
|
imgname[bmpdirlen+1+pos-ptr] = 0;
|
|
}
|
|
else
|
|
/* filename too long */
|
|
imgname[0] = 0;
|
|
|
|
ptr = pos+1;
|
|
|
|
/* load the image */
|
|
data->progressbar.bm.data=data->img_buf_ptr;
|
|
ret = read_bmp_file(imgname, &data->progressbar.bm,
|
|
data->img_buf_free,
|
|
FORMAT_ANY|FORMAT_TRANSPARENT);
|
|
|
|
if (ret > 0)
|
|
{
|
|
#if LCD_DEPTH == 16
|
|
if (ret % 2) ret++;
|
|
/* Always consume an even number of bytes */
|
|
#endif
|
|
|
|
data->img_buf_ptr += ret;
|
|
data->img_buf_free -= ret;
|
|
|
|
if (data->progressbar.bm.width <= LCD_WIDTH) {
|
|
data->progressbar.have_bitmap_pb=true;
|
|
return true;
|
|
} else
|
|
return false;
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
break;
|
|
|
|
case 'x':
|
|
/* Preload images so the %xd# tag can display it */
|
|
{
|
|
int ret = 0;
|
|
int n;
|
|
char *ptr = buf+1;
|
|
char *pos = NULL;
|
|
char imgname[MAX_PATH];
|
|
char qual = *ptr;
|
|
|
|
if (qual == 'l' || qual == '|') /* format:
|
|
%x|n|filename.bmp|x|y|
|
|
or
|
|
%xl|n|filename.bmp|x|y|
|
|
*/
|
|
{
|
|
ptr = strchr(ptr, '|') + 1;
|
|
pos = strchr(ptr, '|');
|
|
if (pos)
|
|
{
|
|
/* get the image ID */
|
|
n = get_image_id(*ptr);
|
|
|
|
if(n < 0 || n >= MAX_IMAGES)
|
|
{
|
|
/* Skip the rest of the line */
|
|
while(*buf != '\n')
|
|
buf++;
|
|
return false;
|
|
}
|
|
ptr = pos+1;
|
|
|
|
/* check the image number and load state */
|
|
if (data->img[n].loaded)
|
|
{
|
|
/* Skip the rest of the line */
|
|
while(*buf != '\n')
|
|
buf++;
|
|
return false;
|
|
}
|
|
else
|
|
{
|
|
/* get filename */
|
|
pos = strchr(ptr, '|');
|
|
|
|
if (pos == NULL)
|
|
return false;
|
|
|
|
if ((pos - ptr) <
|
|
(int)sizeof(imgname)-ROCKBOX_DIR_LEN-2)
|
|
{
|
|
memcpy(imgname, bmpdir, bmpdirlen);
|
|
imgname[bmpdirlen] = '/';
|
|
memcpy(&imgname[bmpdirlen+1],
|
|
ptr, pos - ptr);
|
|
imgname[bmpdirlen+1+pos-ptr] = 0;
|
|
}
|
|
else
|
|
/* filename too long */
|
|
imgname[0] = 0;
|
|
|
|
ptr = pos+1;
|
|
|
|
/* get x-position */
|
|
pos = strchr(ptr, '|');
|
|
if (pos)
|
|
data->img[n].x = atoi(ptr);
|
|
else
|
|
{
|
|
/* weird syntax, bail out */
|
|
buf++;
|
|
return false;
|
|
}
|
|
|
|
/* get y-position */
|
|
ptr = pos+1;
|
|
pos = strchr(ptr, '|');
|
|
if (pos)
|
|
data->img[n].y = atoi(ptr);
|
|
else
|
|
{
|
|
/* weird syntax, bail out */
|
|
buf++;
|
|
return false;
|
|
}
|
|
|
|
/* load the image */
|
|
data->img[n].bm.data = data->img_buf_ptr;
|
|
ret = read_bmp_file(imgname, &data->img[n].bm,
|
|
data->img_buf_free,
|
|
FORMAT_ANY|FORMAT_TRANSPARENT);
|
|
|
|
if (ret > 0)
|
|
{
|
|
#if LCD_DEPTH == 16
|
|
if (ret % 2) ret++;
|
|
/* Always consume an even number of bytes */
|
|
#endif
|
|
|
|
data->img_buf_ptr += ret;
|
|
data->img_buf_free -= ret;
|
|
data->img[n].loaded = true;
|
|
if(qual == '|')
|
|
data->img[n].always_display = true;
|
|
}
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
break;
|
|
#endif
|
|
}
|
|
/* no of these tags found */
|
|
return false;
|
|
}
|
|
|
|
|
|
/* draws the statusbar on the given wps-screen */
|
|
#ifdef HAVE_LCD_BITMAP
|
|
static void gui_wps_statusbar_draw(struct gui_wps *wps, bool force)
|
|
{
|
|
bool draw = global_settings.statusbar;
|
|
|
|
if(wps->data->wps_sb_tag
|
|
&& wps->data->show_sb_on_wps)
|
|
draw = true;
|
|
else if(wps->data->wps_sb_tag)
|
|
draw = false;
|
|
if(draw)
|
|
gui_statusbar_draw(wps->statusbar, force);
|
|
}
|
|
#else
|
|
#define gui_wps_statusbar_draw(wps, force) \
|
|
gui_statusbar_draw((wps)->statusbar, (force))
|
|
#endif
|
|
|
|
/* Extract a part from a path.
|
|
*
|
|
* buf - buffer extract part to.
|
|
* buf_size - size of buffer.
|
|
* path - path to extract from.
|
|
* level - what to extract. 0 is file name, 1 is parent of file, 2 is
|
|
* parent of parent, etc.
|
|
*
|
|
* Returns buf if the desired level was found, NULL otherwise.
|
|
*/
|
|
static char* get_dir(char* buf, int buf_size, const char* path, int level)
|
|
{
|
|
const char* sep;
|
|
const char* last_sep;
|
|
int len;
|
|
|
|
sep = path + strlen(path);
|
|
last_sep = sep;
|
|
|
|
while (sep > path)
|
|
{
|
|
if ('/' == *(--sep))
|
|
{
|
|
if (!level)
|
|
{
|
|
break;
|
|
}
|
|
|
|
level--;
|
|
last_sep = sep - 1;
|
|
}
|
|
}
|
|
|
|
if (level || (last_sep <= sep))
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
len = MIN(last_sep - sep, buf_size - 1);
|
|
strncpy(buf, sep + 1, len);
|
|
buf[len] = 0;
|
|
return buf;
|
|
}
|
|
|
|
/* Get the tag specified by the two characters at fmt.
|
|
*
|
|
* cid3 - ID3 data to get tag values from.
|
|
* nid3 - next-song ID3 data to get tag values from.
|
|
* tag - string (of two characters) specifying the tag to get.
|
|
* buf - buffer to certain tags, such as track number, play time or
|
|
* directory name.
|
|
* buf_size - size of buffer.
|
|
* flags - returns the type of the line. See constants i wps-display.h
|
|
*
|
|
* Returns the tag. NULL indicates the tag wasn't available.
|
|
*/
|
|
static char* get_tag(struct wps_data* wps_data,
|
|
struct mp3entry* cid3,
|
|
struct mp3entry* nid3,
|
|
const char* tag,
|
|
char* buf,
|
|
int buf_size,
|
|
unsigned char* tag_len,
|
|
unsigned short* subline_time_mult,
|
|
unsigned char* flags,
|
|
int *intval)
|
|
{
|
|
struct mp3entry *id3 = cid3; /* default to current song */
|
|
int limit = *intval;
|
|
#ifndef HAVE_LCD_CHARCELLS
|
|
(void)wps_data;
|
|
#endif
|
|
if ((0 == tag[0]) || (0 == tag[1]))
|
|
{
|
|
*tag_len = 0;
|
|
return NULL;
|
|
}
|
|
|
|
*tag_len = 2;
|
|
|
|
*intval = 0;
|
|
|
|
switch (tag[0])
|
|
{
|
|
case 'I': /* ID3 Information */
|
|
id3 = nid3; /* display next-song data */
|
|
*flags |= WPS_REFRESH_DYNAMIC;
|
|
if(!id3)
|
|
return NULL; /* no such info (yet) */
|
|
/* fall-through */
|
|
case 'i': /* ID3 Information */
|
|
*flags |= WPS_REFRESH_STATIC;
|
|
switch (tag[1])
|
|
{
|
|
case 't': /* ID3 Title */
|
|
return id3->title;
|
|
|
|
case 'a': /* ID3 Artist */
|
|
return id3->artist;
|
|
|
|
case 'n': /* ID3 Track Number */
|
|
if (id3->track_string)
|
|
return id3->track_string;
|
|
|
|
if (id3->tracknum) {
|
|
snprintf(buf, buf_size, "%d", id3->tracknum);
|
|
return buf;
|
|
}
|
|
return NULL;
|
|
|
|
case 'd': /* ID3 Album/Disc */
|
|
return id3->album;
|
|
|
|
case 'c': /* ID3 Composer */
|
|
return id3->composer;
|
|
|
|
case 'C': /* ID3 Comment */
|
|
return id3->comment;
|
|
|
|
case 'A': /* ID3 Albumartist */
|
|
return id3->albumartist;
|
|
|
|
case 'y': /* year */
|
|
if( id3->year_string )
|
|
return id3->year_string;
|
|
|
|
if (id3->year) {
|
|
snprintf(buf, buf_size, "%d", id3->year);
|
|
return buf;
|
|
}
|
|
return NULL;
|
|
|
|
case 'g': /* genre */
|
|
return id3->genre_string;
|
|
|
|
case 'v': /* id3 version */
|
|
switch (id3->id3version)
|
|
{
|
|
case ID3_VER_1_0:
|
|
return "1";
|
|
|
|
case ID3_VER_1_1:
|
|
return "1.1";
|
|
|
|
case ID3_VER_2_2:
|
|
return "2.2";
|
|
|
|
case ID3_VER_2_3:
|
|
return "2.3";
|
|
|
|
case ID3_VER_2_4:
|
|
return "2.4";
|
|
|
|
default:
|
|
return NULL;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case 'F': /* File Information */
|
|
id3 = nid3;
|
|
*flags |= WPS_REFRESH_DYNAMIC;
|
|
if(!id3)
|
|
return NULL; /* no such info (yet) */
|
|
/* fall-through */
|
|
case 'f': /* File Information */
|
|
*flags |= WPS_REFRESH_STATIC;
|
|
switch(tag[1])
|
|
{
|
|
case 'v': /* VBR file? */
|
|
return id3->vbr ? "(avg)" : NULL;
|
|
|
|
case 'b': /* File Bitrate */
|
|
if(id3->bitrate)
|
|
snprintf(buf, buf_size, "%d", id3->bitrate);
|
|
else
|
|
snprintf(buf, buf_size, "?");
|
|
return buf;
|
|
|
|
case 'f': /* File Frequency */
|
|
snprintf(buf, buf_size, "%ld", id3->frequency);
|
|
return buf;
|
|
|
|
case 'p': /* File Path */
|
|
return id3->path;
|
|
|
|
case 'm': /* File Name - With Extension */
|
|
return get_dir(buf, buf_size, id3->path, 0);
|
|
|
|
case 'n': /* File Name */
|
|
if (get_dir(buf, buf_size, id3->path, 0))
|
|
{
|
|
/* Remove extension */
|
|
char* sep = strrchr(buf, '.');
|
|
|
|
if (NULL != sep)
|
|
{
|
|
*sep = 0;
|
|
}
|
|
|
|
return buf;
|
|
}
|
|
else
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
case 's': /* File Size (in kilobytes) */
|
|
snprintf(buf, buf_size, "%ld", id3->filesize / 1024);
|
|
return buf;
|
|
|
|
case 'c': /* File Codec */
|
|
if(id3->codectype == AFMT_UNKNOWN)
|
|
*intval = AFMT_NUM_CODECS;
|
|
else
|
|
*intval = id3->codectype;
|
|
return id3_get_codec(id3);
|
|
}
|
|
break;
|
|
|
|
case 'p': /* Playlist/Song Information */
|
|
switch(tag[1])
|
|
{
|
|
case 'b': /* progress bar */
|
|
*flags |= WPS_REFRESH_PLAYER_PROGRESS;
|
|
#ifdef HAVE_LCD_CHARCELLS
|
|
snprintf(buf, buf_size, "%c",
|
|
wps_data->wps_progress_pat[0]);
|
|
wps_data->full_line_progressbar=0;
|
|
return buf;
|
|
#else
|
|
/* default values : */
|
|
wps_data->progress_top = -1;
|
|
wps_data->progress_height = 6;
|
|
wps_data->progress_start = 0;
|
|
wps_data->progress_end = 0;
|
|
|
|
char *prev=strchr(tag, '|');
|
|
if (prev) {
|
|
char *p=strchr(prev+1, '|');
|
|
if (p) {
|
|
wps_data->progress_height=atoi(++prev);
|
|
prev=strchr(prev, '|');
|
|
p=strchr(++p, '|');
|
|
if (p) {
|
|
wps_data->progress_start=atoi(++prev);
|
|
prev=strchr(prev, '|');
|
|
p=strchr(++p, '|');
|
|
if (p) {
|
|
wps_data->progress_end=atoi(++prev);
|
|
prev=strchr(prev, '|');
|
|
p=strchr(++p, '|');
|
|
if(p)
|
|
wps_data->progress_top = atoi(++prev);
|
|
}
|
|
|
|
if (wps_data->progress_height<3)
|
|
wps_data->progress_height=3;
|
|
if (wps_data->progress_end<wps_data->progress_start+3)
|
|
wps_data->progress_end=0;
|
|
}
|
|
}
|
|
}
|
|
return "\x01";
|
|
#endif
|
|
case 'f': /* full-line progress bar */
|
|
#ifdef HAVE_LCD_CHARCELLS
|
|
if(is_new_player()) {
|
|
*flags |= WPS_REFRESH_PLAYER_PROGRESS;
|
|
*flags |= WPS_REFRESH_DYNAMIC;
|
|
wps_data->full_line_progressbar=1;
|
|
/* we need 11 characters (full line) for
|
|
progress-bar */
|
|
snprintf(buf, buf_size, " ");
|
|
}
|
|
else
|
|
{
|
|
/* Tell the user if we have an OldPlayer */
|
|
snprintf(buf, buf_size, " <Old LCD> ");
|
|
}
|
|
return buf;
|
|
#endif
|
|
case 'p': /* Playlist Position */
|
|
*flags |= WPS_REFRESH_STATIC;
|
|
snprintf(buf, buf_size, "%d",
|
|
playlist_get_display_index());
|
|
return buf;
|
|
|
|
case 'n': /* Playlist Name (without path) */
|
|
*flags |= WPS_REFRESH_STATIC;
|
|
return playlist_name(NULL, buf, buf_size);
|
|
|
|
case 'e': /* Playlist Total Entries */
|
|
*flags |= WPS_REFRESH_STATIC;
|
|
snprintf(buf, buf_size, "%d", playlist_amount());
|
|
return buf;
|
|
|
|
case 'c': /* Current Time in Song */
|
|
*flags |= WPS_REFRESH_DYNAMIC;
|
|
format_time(buf, buf_size,
|
|
id3->elapsed + wps_state.ff_rewind_count);
|
|
return buf;
|
|
|
|
case 'r': /* Remaining Time in Song */
|
|
*flags |= WPS_REFRESH_DYNAMIC;
|
|
format_time(buf, buf_size,
|
|
id3->length - id3->elapsed -
|
|
wps_state.ff_rewind_count);
|
|
return buf;
|
|
|
|
case 't': /* Total Time */
|
|
*flags |= WPS_REFRESH_STATIC;
|
|
format_time(buf, buf_size, id3->length);
|
|
return buf;
|
|
|
|
#ifdef HAVE_LCD_BITMAP
|
|
case 'm': /* Peak Meter */
|
|
*flags |= WPS_REFRESH_PEAK_METER;
|
|
return "\x01";
|
|
#endif
|
|
case 's': /* shuffle */
|
|
*flags |= WPS_REFRESH_DYNAMIC;
|
|
if ( global_settings.playlist_shuffle )
|
|
return "s";
|
|
else
|
|
return NULL;
|
|
break;
|
|
|
|
case 'v': /* volume */
|
|
*flags |= WPS_REFRESH_DYNAMIC;
|
|
snprintf(buf, buf_size, "%d", global_settings.volume);
|
|
*intval = limit * (global_settings.volume
|
|
- sound_min(SOUND_VOLUME))
|
|
/ (sound_max(SOUND_VOLUME)
|
|
- sound_min(SOUND_VOLUME)) + 1;
|
|
return buf;
|
|
|
|
}
|
|
break;
|
|
|
|
#if (CONFIG_CODEC == SWCODEC)
|
|
case 'S': /* DSP/Equalizer/Sound settings */
|
|
switch (tag[1])
|
|
{
|
|
case 'p': /* pitch */
|
|
*intval = sound_get_pitch();
|
|
snprintf(buf, buf_size, "%d.%d",
|
|
*intval / 10, *intval % 10);
|
|
return buf;
|
|
}
|
|
break;
|
|
#endif
|
|
|
|
case 'm':
|
|
switch (tag[1])
|
|
{
|
|
case 'm': /* playback repeat mode */
|
|
*flags |= WPS_REFRESH_DYNAMIC;
|
|
*intval = global_settings.repeat_mode + 1;
|
|
snprintf(buf, buf_size, "%d", *intval);
|
|
return buf;
|
|
|
|
/* playback status */
|
|
case 'p': /* play */
|
|
*flags |= WPS_REFRESH_DYNAMIC;
|
|
int status = audio_status();
|
|
*intval = 1;
|
|
if (status == AUDIO_STATUS_PLAY && \
|
|
!(status & AUDIO_STATUS_PAUSE))
|
|
*intval = 2;
|
|
if (audio_status() & AUDIO_STATUS_PAUSE && \
|
|
(! status_get_ffmode()))
|
|
*intval = 3;
|
|
if (status_get_ffmode() == STATUS_FASTFORWARD)
|
|
*intval = 4;
|
|
if (status_get_ffmode() == STATUS_FASTBACKWARD)
|
|
*intval = 5;
|
|
snprintf(buf, buf_size, "%d", *intval);
|
|
return buf;
|
|
|
|
#ifdef HAS_BUTTON_HOLD
|
|
case 'h': /* hold */
|
|
*flags |= WPS_REFRESH_DYNAMIC;
|
|
if (button_hold())
|
|
return "h";
|
|
else
|
|
return NULL;
|
|
#endif
|
|
#ifdef HAS_REMOTE_BUTTON_HOLD
|
|
case 'r': /* remote hold */
|
|
*flags |= WPS_REFRESH_DYNAMIC;
|
|
if (remote_button_hold())
|
|
return "r";
|
|
else
|
|
return NULL;
|
|
#endif
|
|
}
|
|
break;
|
|
|
|
case 'b': /* battery info */
|
|
*flags |= WPS_REFRESH_DYNAMIC;
|
|
switch (tag[1])
|
|
{
|
|
case 'l': /* battery level */
|
|
{
|
|
int l = battery_level();
|
|
limit = MAX(limit, 2);
|
|
if (l > -1)
|
|
{
|
|
snprintf(buf, buf_size, "%d", l);
|
|
/* First enum is used for "unknown level". */
|
|
*intval = (limit - 1) * l / 100 + 1 + 1;
|
|
}
|
|
else
|
|
{
|
|
*intval = 1;
|
|
return "?";
|
|
}
|
|
return buf;
|
|
}
|
|
|
|
case 'v': /* battery voltage */
|
|
{
|
|
unsigned int v = battery_voltage();
|
|
snprintf(buf, buf_size, "%d.%02d", v/100, v%100);
|
|
return buf;
|
|
}
|
|
|
|
case 't': /* estimated battery time */
|
|
{
|
|
int t = battery_time();
|
|
if (t >= 0)
|
|
snprintf(buf, buf_size, "%dh %dm", t / 60, t % 60);
|
|
else
|
|
strncpy(buf, "?h ?m", buf_size);
|
|
return buf;
|
|
}
|
|
|
|
case 's': /* sleep timer */
|
|
{
|
|
if (get_sleep_timer() == 0)
|
|
{
|
|
return NULL;
|
|
}
|
|
else
|
|
{
|
|
format_time(buf, buf_size, \
|
|
get_sleep_timer() * 1000);
|
|
return buf;
|
|
}
|
|
}
|
|
|
|
#if CONFIG_CHARGING
|
|
case 'p': /* External power plugged in? */
|
|
{
|
|
if(charger_input_state==CHARGER)
|
|
return "p";
|
|
else
|
|
return NULL;
|
|
}
|
|
#endif
|
|
#if CONFIG_CHARGING >= CHARGING_MONITOR
|
|
case 'c': /* Charging */
|
|
{
|
|
if (charge_state == CHARGING || charge_state == TOPOFF) {
|
|
return "c";
|
|
} else {
|
|
return NULL;
|
|
}
|
|
}
|
|
#endif
|
|
}
|
|
break;
|
|
|
|
#if (CONFIG_LED == LED_VIRTUAL) || defined(HAVE_REMOTE_LCD)
|
|
case 'l': /* VIRTUAL_LED */
|
|
{
|
|
switch(tag[1])
|
|
{
|
|
case 'h': /* Only one we have so far HDD LED */
|
|
*flags |= WPS_REFRESH_DYNAMIC;
|
|
if(led_read(HZ/2))
|
|
return "h";
|
|
else
|
|
return NULL;
|
|
}
|
|
}
|
|
break;
|
|
#endif
|
|
|
|
case 'D': /* Directory path information */
|
|
id3 = nid3; /* next song please! */
|
|
*flags |= WPS_REFRESH_DYNAMIC;
|
|
if(!id3)
|
|
return NULL; /* no such info (yet) */
|
|
/* fall-through */
|
|
case 'd': /* Directory path information */
|
|
{
|
|
int level = tag[1] - '0';
|
|
*flags |= WPS_REFRESH_STATIC;
|
|
/* d1 through d9 */
|
|
if ((0 < level) && (9 > level))
|
|
{
|
|
return get_dir(buf, buf_size, id3->path, level);
|
|
}
|
|
}
|
|
break;
|
|
|
|
case 't': /* set sub line time multiplier */
|
|
{
|
|
int d = 1;
|
|
int time_mult = 0;
|
|
bool have_point = false;
|
|
bool have_tenth = false;
|
|
|
|
while (((tag[d] >= '0') &&
|
|
(tag[d] <= '9')) ||
|
|
(tag[d] == '.'))
|
|
{
|
|
if (tag[d] != '.')
|
|
{
|
|
time_mult = time_mult * 10;
|
|
time_mult = time_mult + tag[d] - '0';
|
|
if (have_point)
|
|
{
|
|
have_tenth = true;
|
|
d++;
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
have_point = true;
|
|
}
|
|
d++;
|
|
}
|
|
|
|
if (have_tenth == false)
|
|
time_mult *= 10;
|
|
|
|
*subline_time_mult = time_mult;
|
|
*tag_len = d;
|
|
|
|
buf[0] = 0;
|
|
return buf;
|
|
}
|
|
break;
|
|
case 'r': /* Runtime database Information and Replaygain */
|
|
switch(tag[1])
|
|
{
|
|
case 'p': /* Playcount */
|
|
*flags |= WPS_REFRESH_DYNAMIC;
|
|
*intval = cid3->playcount+1;
|
|
snprintf(buf, buf_size, "%ld", cid3->playcount);
|
|
return buf;
|
|
case 'r': /* Rating */
|
|
*flags |= WPS_REFRESH_DYNAMIC;
|
|
*intval = cid3->rating+1;
|
|
snprintf(buf, buf_size, "%d", cid3->rating);
|
|
return buf;
|
|
#if CONFIG_CODEC == SWCODEC
|
|
case 'g': /* ReplayGain */
|
|
*flags |= WPS_REFRESH_STATIC;
|
|
if (global_settings.replaygain == 0)
|
|
*intval = 1; /* off */
|
|
else
|
|
{
|
|
int type = get_replaygain_mode(
|
|
id3->track_gain_string != NULL,
|
|
id3->album_gain_string != NULL);
|
|
|
|
if (type < 0)
|
|
*intval = 6; /* no tag */
|
|
else
|
|
*intval = type + 2;
|
|
|
|
if (global_settings.replaygain_type == REPLAYGAIN_SHUFFLE)
|
|
*intval += 2;
|
|
}
|
|
|
|
switch (*intval)
|
|
{
|
|
case 1:
|
|
case 6:
|
|
return "+0.00 dB";
|
|
break;
|
|
case 2:
|
|
case 4:
|
|
strncpy(buf, id3->track_gain_string, buf_size);
|
|
break;
|
|
case 3:
|
|
case 5:
|
|
strncpy(buf, id3->album_gain_string, buf_size);
|
|
break;
|
|
}
|
|
return buf;
|
|
#endif
|
|
}
|
|
break;
|
|
#if CONFIG_RTC
|
|
case 'c': /* Real Time Clock display */
|
|
*flags |= WPS_REFRESH_DYNAMIC;
|
|
{
|
|
int value;
|
|
char *format = 0;
|
|
char *bufptr = buf;
|
|
struct tm* tm = get_time();
|
|
int i;
|
|
for (i=1;/*break*/;i++) {
|
|
switch(tag[i])
|
|
{
|
|
case 'a': /* abbreviated weekday name (Sun..Sat) */
|
|
value = tm->tm_wday;
|
|
if (value > 6 || value < 0) continue;
|
|
value = snprintf(
|
|
bufptr,buf_size,"%s",str(dayname[value]));
|
|
bufptr += value;
|
|
buf_size -= value;
|
|
continue;
|
|
case 'b': /* abbreviated month name (Jan..Dec) */
|
|
value = tm->tm_mon;
|
|
if (value > 11 || value < 0) continue;
|
|
value = snprintf(
|
|
bufptr,buf_size,"%s",str(monthname[value]));
|
|
bufptr += value;
|
|
buf_size -= value;
|
|
continue;
|
|
case 'd': /* day of month (01..31) */
|
|
value = tm->tm_mday;
|
|
if (value > 31 || value < 1) continue;
|
|
format = "%02d";
|
|
break;
|
|
case 'e': /* day of month, blank padded ( 1..31) */
|
|
value = tm->tm_mday;
|
|
if (value > 31 || value < 1) continue;
|
|
format = "%2d";
|
|
break;
|
|
case 'H': /* hour (00..23) */
|
|
value = tm->tm_hour;
|
|
if (value > 23) continue;
|
|
format = "%02d";
|
|
break;
|
|
case 'k': /* hour ( 0..23) */
|
|
value = tm->tm_hour;
|
|
if (value > 23) continue;
|
|
format = "%2d";
|
|
break;
|
|
case 'I': /* hour (01..12) */
|
|
value = tm->tm_hour;
|
|
if (value > 23) continue;
|
|
value %= 12;
|
|
if (value == 0) value = 12;
|
|
format = "%02d";
|
|
break;
|
|
case 'l': /* hour ( 1..12) */
|
|
value = tm->tm_hour;
|
|
if (value > 23 || value < 0) continue;
|
|
value %= 12;
|
|
if (value == 0) value = 12;
|
|
format = "%2d";
|
|
break;
|
|
case 'm': /* month (01..12) */
|
|
value = tm->tm_mon;
|
|
if (value > 11 || value < 0) continue;
|
|
value++;
|
|
format = "%02d";
|
|
break;
|
|
case 'M': /* minute (00..59) */
|
|
value = tm->tm_min;
|
|
if (value > 59 || value < 0) continue;
|
|
format = "%02d";
|
|
break;
|
|
case 'S': /* second (00..59) */
|
|
value = tm->tm_sec;
|
|
if (value > 59 || value < 0) continue;
|
|
format = "%02d";
|
|
break;
|
|
case 'y': /* last two digits of year (00..99) */
|
|
value = tm->tm_year;
|
|
value %= 100;
|
|
format = "%02d";
|
|
break;
|
|
case 'Y': /* year (1970...) */
|
|
value = tm->tm_year;
|
|
if (value > 199 || value < 100) continue;
|
|
value += 1900;
|
|
format = "%04d";
|
|
break;
|
|
case 'p': /* upper case AM or PM indicator */
|
|
if (tm->tm_hour/12 == 0) format = "AM";
|
|
else format = "PM";
|
|
snprintf(bufptr,buf_size,"%s",format);
|
|
bufptr += 2;
|
|
buf_size -= 2;
|
|
continue;
|
|
case 'P': /* lower case am or pm indicator */
|
|
if (tm->tm_hour/12 == 0) format = "am";
|
|
else format = "pm";
|
|
snprintf(bufptr,buf_size,"%s",format);
|
|
bufptr += 2;
|
|
buf_size -= 2;
|
|
continue;
|
|
case 'u': /* day of week (1..7); 1 is Monday */
|
|
value = tm->tm_wday;
|
|
if (value < 0 || value > 6) continue;
|
|
value++;
|
|
format = "%1d";
|
|
break;
|
|
case 'w': /* day of week (0..6); 0 is Sunday */
|
|
value = tm->tm_wday;
|
|
if (value < 0 || value > 6) continue;
|
|
format = "%1d";
|
|
break;
|
|
default:
|
|
if (tag[i] == 'c') {
|
|
i++;
|
|
value = -1;
|
|
break;
|
|
} else if (tag[i] == '\n') {
|
|
value = -1;
|
|
break;
|
|
}
|
|
snprintf(bufptr,buf_size,"%c",tag[i]);
|
|
bufptr++;
|
|
buf_size--;
|
|
continue;
|
|
} /* switch */
|
|
if (value < 0) break;
|
|
|
|
value = snprintf(bufptr, buf_size, format, value);
|
|
bufptr += value;
|
|
buf_size -= value;
|
|
} /* while */
|
|
*tag_len = i;
|
|
return buf;
|
|
}
|
|
#endif /* CONFIG_RTC */
|
|
#if CONFIG_CODEC == SWCODEC
|
|
case 'x':
|
|
*flags |= WPS_REFRESH_DYNAMIC;
|
|
switch(tag[1])
|
|
{
|
|
case 'd': /* crossfeed */
|
|
if(global_settings.crossfeed)
|
|
return "d";
|
|
else
|
|
return NULL;
|
|
case 'f': /* crossfade */
|
|
*intval = global_settings.crossfade+1;
|
|
snprintf(buf, buf_size, "%d", global_settings.crossfade);
|
|
return buf;
|
|
}
|
|
break;
|
|
#endif
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
#ifdef HAVE_LCD_BITMAP
|
|
/* clears the area where the image was shown */
|
|
static void clear_image_pos(struct gui_wps *gwps, int n)
|
|
{
|
|
if(!gwps)
|
|
return;
|
|
struct wps_data *data = gwps->data;
|
|
gwps->display->set_drawmode(DRMODE_SOLID|DRMODE_INVERSEVID);
|
|
gwps->display->fillrect(data->img[n].x, data->img[n].y,
|
|
data->img[n].bm.width, data->img[n].bm.height);
|
|
gwps->display->set_drawmode(DRMODE_SOLID);
|
|
}
|
|
#endif
|
|
|
|
/* Skip to the end of the current %? conditional.
|
|
*
|
|
* fmt - string to skip it. Should point to somewhere after the leading
|
|
* "<" char (and before or at the last ">").
|
|
* num - number of |'s to skip, or 0 to skip to the end (the ">").
|
|
* enums - If not NULL, set to the number of |'s found in the current
|
|
* conditional (sub-conditionals are ignored). num should be 0
|
|
* to find all |'s.
|
|
*
|
|
* Returns the new position in fmt.
|
|
*/
|
|
static const char* skip_conditional(struct gui_wps *gwps, const char* fmt,
|
|
int num, int *enums)
|
|
{
|
|
int level = 1;
|
|
int count = num;
|
|
const char *last_alternative = NULL;
|
|
#ifdef HAVE_LCD_BITMAP
|
|
struct wps_data *data = NULL;
|
|
int last_x=-1, last_y=-1, last_w=-1, last_h=-1;
|
|
if(gwps)
|
|
data = gwps->data;
|
|
if (enums)
|
|
*enums = 0;
|
|
#else
|
|
(void)gwps;
|
|
#endif
|
|
while (*fmt)
|
|
{
|
|
switch (*fmt++)
|
|
{
|
|
case '%':
|
|
#ifdef HAVE_LCD_BITMAP
|
|
if(data && *(fmt) == 'x' && *(fmt+1) == 'd' )
|
|
{
|
|
fmt +=2;
|
|
int n = *fmt;
|
|
if(n >= 'a' && n <= 'z')
|
|
n -= 'a';
|
|
if(n >= 'A' && n <= 'Z')
|
|
n = n - 'A' + 26;
|
|
if(last_x != data->img[n].x || last_y != data->img[n].y
|
|
|| last_w != data->img[n].bm.width
|
|
|| last_h != data->img[n].bm.height)
|
|
{
|
|
last_x = data->img[n].x;
|
|
last_y = data->img[n].y;
|
|
last_w = data->img[n].bm.width;
|
|
last_h = data->img[n].bm.height;
|
|
clear_image_pos(gwps,n);
|
|
}
|
|
}
|
|
#endif
|
|
break;
|
|
|
|
case '|':
|
|
if(1 == level) {
|
|
if (enums)
|
|
(*enums)++;
|
|
last_alternative = fmt;
|
|
if(num) {
|
|
count--;
|
|
if(count == 0)
|
|
return fmt;
|
|
continue;
|
|
}
|
|
}
|
|
continue;
|
|
|
|
case '>':
|
|
if (0 == --level)
|
|
{
|
|
/* We're just skipping to the end */
|
|
if(num == 0)
|
|
return fmt;
|
|
|
|
/* If we are parsing an enum, we'll return the selected
|
|
item. If there weren't enough items in the enum, we'll
|
|
return the last one found. */
|
|
if(count && last_alternative)
|
|
{
|
|
return last_alternative;
|
|
}
|
|
return fmt - 1;
|
|
}
|
|
continue;
|
|
|
|
default:
|
|
continue;
|
|
}
|
|
|
|
switch (*fmt++)
|
|
{
|
|
case 0:
|
|
case '%':
|
|
case '|':
|
|
case '<':
|
|
case '>':
|
|
break;
|
|
|
|
case '?':
|
|
while (*fmt && ('<' != *fmt))
|
|
fmt++;
|
|
|
|
if ('<' == *fmt)
|
|
fmt++;
|
|
|
|
level++;
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
return fmt;
|
|
}
|
|
|
|
/* Generate the display based on id3 information and format string.
|
|
*
|
|
* buf - char buffer to write the display to.
|
|
* buf_size - the size of buffer.
|
|
* id3 - the ID3 data to format with.
|
|
* nid3 - the ID3 data of the next song (might by NULL)
|
|
* fmt - format description.
|
|
* flags - returns the type of the line. See constants i wps-display.h
|
|
*/
|
|
static void format_display(struct gui_wps *gwps, char* buf,
|
|
int buf_size,
|
|
struct mp3entry* id3,
|
|
struct mp3entry* nid3, /* next song's id3 */
|
|
const char* fmt,
|
|
struct align_pos* align,
|
|
unsigned short* subline_time_mult,
|
|
unsigned char* flags)
|
|
{
|
|
char temp_buf[128];
|
|
char* buf_start = buf;
|
|
char* buf_end = buf + buf_size - 1; /* Leave room for end null */
|
|
char* value = NULL;
|
|
int level = 0;
|
|
unsigned char tag_length;
|
|
int intval;
|
|
int cur_align;
|
|
char* cur_align_start;
|
|
#ifdef HAVE_LCD_BITMAP
|
|
struct gui_img *img = gwps->data->img;
|
|
int n;
|
|
#endif
|
|
|
|
cur_align_start = buf;
|
|
cur_align = WPS_ALIGN_LEFT;
|
|
*subline_time_mult = DEFAULT_SUBLINE_TIME_MULTIPLIER;
|
|
|
|
align->left = 0;
|
|
align->center = 0;
|
|
align->right = 0;
|
|
|
|
while (fmt && *fmt && buf < buf_end)
|
|
{
|
|
switch (*fmt)
|
|
{
|
|
case '%':
|
|
++fmt;
|
|
break;
|
|
|
|
case '|':
|
|
case '>':
|
|
if (level > 0)
|
|
{
|
|
fmt = skip_conditional(NULL, fmt, 0, NULL);
|
|
level--;
|
|
continue;
|
|
}
|
|
/* Else fall through */
|
|
|
|
default:
|
|
*buf++ = *fmt++;
|
|
continue;
|
|
}
|
|
|
|
switch (*fmt)
|
|
{
|
|
case 0:
|
|
*buf++ = '%';
|
|
break;
|
|
case 'a':
|
|
++fmt;
|
|
/* remember where the current aligned text started */
|
|
switch (cur_align)
|
|
{
|
|
case WPS_ALIGN_LEFT:
|
|
align->left = cur_align_start;
|
|
break;
|
|
|
|
case WPS_ALIGN_CENTER:
|
|
align->center = cur_align_start;
|
|
break;
|
|
|
|
case WPS_ALIGN_RIGHT:
|
|
align->right = cur_align_start;
|
|
break;
|
|
}
|
|
/* start a new alignment */
|
|
switch (*fmt)
|
|
{
|
|
case 'l':
|
|
cur_align = WPS_ALIGN_LEFT;
|
|
break;
|
|
case 'c':
|
|
cur_align = WPS_ALIGN_CENTER;
|
|
break;
|
|
case 'r':
|
|
cur_align = WPS_ALIGN_RIGHT;
|
|
break;
|
|
}
|
|
*buf++=0;
|
|
cur_align_start = buf;
|
|
++fmt;
|
|
break;
|
|
case 's':
|
|
*flags |= WPS_REFRESH_SCROLL;
|
|
++fmt;
|
|
break;
|
|
|
|
case 'x': /* image support */
|
|
#ifdef HAVE_LCD_BITMAP
|
|
if ('d' == *(fmt+1) )
|
|
{
|
|
fmt+=2;
|
|
|
|
/* get the image ID */
|
|
n = *fmt;
|
|
if(n >= 'a' && n <= 'z')
|
|
n -= 'a';
|
|
if(n >= 'A' && n <= 'Z')
|
|
n = n - 'A' + 26;
|
|
if (n >= 0 && n < MAX_IMAGES && img[n].loaded) {
|
|
img[n].display = true;
|
|
}
|
|
}
|
|
|
|
#endif
|
|
fmt++;
|
|
break;
|
|
|
|
|
|
case '%':
|
|
case '|':
|
|
case '<':
|
|
case '>':
|
|
case ';':
|
|
*buf++ = *fmt++;
|
|
break;
|
|
|
|
case '?':
|
|
fmt++;
|
|
/* Get number of "|" chars in the current conditional;
|
|
* used by get_tag when calculating levels.
|
|
*/
|
|
skip_conditional(gwps, fmt, 0, &intval);
|
|
value = get_tag(gwps->data, id3, nid3, fmt, temp_buf,
|
|
sizeof(temp_buf),&tag_length,
|
|
subline_time_mult, flags, &intval);
|
|
|
|
while (*fmt && ('<' != *fmt))
|
|
fmt++;
|
|
|
|
if ('<' == *fmt)
|
|
fmt++;
|
|
|
|
/* No value, so skip to else part, using a sufficiently high
|
|
value to "hit" the last part of the conditional */
|
|
if ((!value) || (!strlen(value)))
|
|
fmt = skip_conditional(NULL, fmt, 1000, NULL);
|
|
else
|
|
if(intval > 1) /* enum */
|
|
fmt = skip_conditional(NULL, fmt, intval - 1, NULL);
|
|
|
|
level++;
|
|
break;
|
|
|
|
default:
|
|
intval = 1;
|
|
value = get_tag(gwps->data, id3, nid3, fmt, temp_buf,
|
|
sizeof(temp_buf), &tag_length,
|
|
subline_time_mult, flags,&intval);
|
|
fmt += tag_length;
|
|
|
|
if (value)
|
|
{
|
|
while (*value && (buf < buf_end))
|
|
*buf++ = *value++;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* remember where the current aligned text started */
|
|
switch (cur_align)
|
|
{
|
|
case WPS_ALIGN_LEFT:
|
|
align->left = cur_align_start;
|
|
break;
|
|
|
|
case WPS_ALIGN_CENTER:
|
|
align->center = cur_align_start;
|
|
break;
|
|
|
|
case WPS_ALIGN_RIGHT:
|
|
align->right = cur_align_start;
|
|
break;
|
|
}
|
|
|
|
*buf = 0;
|
|
|
|
/* if resulting line is an empty line, set the subline time to 0 */
|
|
if (buf - buf_start == 0)
|
|
*subline_time_mult = 0;
|
|
|
|
/* If no flags have been set, the line didn't contain any format codes.
|
|
We still want to refresh it. */
|
|
if(*flags == 0)
|
|
*flags = WPS_REFRESH_STATIC;
|
|
}
|
|
|
|
/* fades the volume */
|
|
void fade(bool fade_in)
|
|
{
|
|
int fp_global_vol = global_settings.volume << 8;
|
|
int fp_min_vol = sound_min(SOUND_VOLUME) << 8;
|
|
int fp_step = (fp_global_vol - fp_min_vol) / 30;
|
|
|
|
if (fade_in) {
|
|
/* fade in */
|
|
int fp_volume = fp_min_vol;
|
|
|
|
/* zero out the sound */
|
|
sound_set_volume(fp_min_vol >> 8);
|
|
|
|
sleep(HZ/10); /* let audio thread run */
|
|
audio_resume();
|
|
|
|
while (fp_volume < fp_global_vol - fp_step) {
|
|
fp_volume += fp_step;
|
|
sound_set_volume(fp_volume >> 8);
|
|
sleep(1);
|
|
}
|
|
sound_set_volume(global_settings.volume);
|
|
}
|
|
else {
|
|
/* fade out */
|
|
int fp_volume = fp_global_vol;
|
|
|
|
while (fp_volume > fp_min_vol + fp_step) {
|
|
fp_volume -= fp_step;
|
|
sound_set_volume(fp_volume >> 8);
|
|
sleep(1);
|
|
}
|
|
audio_pause();
|
|
#ifndef SIMULATOR
|
|
/* let audio thread run and wait for the mas to run out of data */
|
|
while (!mp3_pause_done())
|
|
#endif
|
|
sleep(HZ/10);
|
|
|
|
/* reset volume to what it was before the fade */
|
|
sound_set_volume(global_settings.volume);
|
|
}
|
|
}
|
|
|
|
/* Set format string to use for WPS, splitting it into lines */
|
|
void gui_wps_format(struct wps_data *data)
|
|
{
|
|
char* buf = data->format_buffer;
|
|
char* start_of_line = data->format_buffer;
|
|
int line = 0;
|
|
int subline;
|
|
char c;
|
|
if(!data)
|
|
return;
|
|
|
|
for (line=0; line<WPS_MAX_LINES; line++)
|
|
{
|
|
for (subline=0; subline<WPS_MAX_SUBLINES; subline++)
|
|
{
|
|
data->format_lines[line][subline] = 0;
|
|
data->time_mult[line][subline] = 0;
|
|
}
|
|
data->subline_expire_time[line] = 0;
|
|
data->curr_subline[line] = SUBLINE_RESET;
|
|
}
|
|
|
|
line = 0;
|
|
subline = 0;
|
|
buf = skip_utf8_bom(buf);
|
|
data->format_lines[line][subline] = buf;
|
|
|
|
while ((*buf) && (line < WPS_MAX_LINES))
|
|
{
|
|
c = *buf;
|
|
|
|
switch (c)
|
|
{
|
|
/*
|
|
* skip % sequences so "%;" doesn't start a new subline
|
|
* don't skip %x lines (pre-load bitmaps)
|
|
*/
|
|
case '%':
|
|
buf++;
|
|
break;
|
|
|
|
case '\r': /* CR */
|
|
*buf = 0;
|
|
break;
|
|
|
|
case '\n': /* LF */
|
|
*buf = 0;
|
|
|
|
if (*start_of_line != '#') /* A comment? */
|
|
line++;
|
|
|
|
if (line < WPS_MAX_LINES)
|
|
{
|
|
/* the next line starts on the next byte */
|
|
subline = 0;
|
|
data->format_lines[line][subline] = buf+1;
|
|
start_of_line = data->format_lines[line][subline];
|
|
}
|
|
break;
|
|
|
|
case ';': /* start a new subline */
|
|
*buf = 0;
|
|
subline++;
|
|
if (subline < WPS_MAX_SUBLINES)
|
|
{
|
|
data->format_lines[line][subline] = buf+1;
|
|
}
|
|
else /* exceeded max sublines, skip rest of line */
|
|
{
|
|
while (*(++buf))
|
|
{
|
|
if ((*buf == '\r') || (*buf == '\n'))
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
buf--;
|
|
subline = 0;
|
|
}
|
|
break;
|
|
}
|
|
buf++;
|
|
}
|
|
}
|
|
|
|
#ifdef HAVE_LCD_BITMAP
|
|
/* Display images */
|
|
static void wps_draw_image(struct gui_wps *gwps, int n)
|
|
{
|
|
struct screen *display = gwps->display;
|
|
struct wps_data *data = gwps->data;
|
|
if(data->img[n].always_display)
|
|
display->set_drawmode(DRMODE_FG);
|
|
else
|
|
display->set_drawmode(DRMODE_SOLID);
|
|
|
|
#if LCD_DEPTH > 1
|
|
if(data->img[n].bm.format == FORMAT_MONO) {
|
|
#endif
|
|
display->mono_bitmap(data->img[n].bm.data, data->img[n].x,
|
|
data->img[n].y, data->img[n].bm.width,
|
|
data->img[n].bm.height);
|
|
#if LCD_DEPTH > 1
|
|
} else {
|
|
display->transparent_bitmap((fb_data *)data->img[n].bm.data,
|
|
data->img[n].x,
|
|
data->img[n].y, data->img[n].bm.width,
|
|
data->img[n].bm.height);
|
|
}
|
|
#endif
|
|
}
|
|
static void wps_display_images(struct gui_wps *gwps, bool always)
|
|
{
|
|
if(!gwps || !gwps->data || !gwps->display) return;
|
|
int n;
|
|
struct wps_data *data = gwps->data;
|
|
struct screen *display = gwps->display;
|
|
for (n = 0; n < MAX_IMAGES; n++) {
|
|
if (data->img[n].loaded) {
|
|
if( (!always && data->img[n].display)
|
|
|| (always && data->img[n].always_display) )
|
|
wps_draw_image(gwps, n);
|
|
}
|
|
}
|
|
display->set_drawmode(DRMODE_SOLID);
|
|
}
|
|
#endif
|
|
|
|
#if 0 /* currently unused */
|
|
void gui_wps_reset(struct gui_wps *gui_wps)
|
|
{
|
|
if(!gui_wps || !gui_wps->data)
|
|
return;
|
|
gui_wps->data->wps_loaded = false;
|
|
memset(&gui_wps->data->format_buffer, 0,
|
|
sizeof(gui_wps->data->format_buffer));
|
|
}
|
|
#endif
|
|
|
|
bool gui_wps_refresh(struct gui_wps *gwps, int ffwd_offset,
|
|
unsigned char refresh_mode)
|
|
{
|
|
char buf[MAX_PATH];
|
|
unsigned char flags;
|
|
int i;
|
|
bool update_line;
|
|
bool only_one_subline;
|
|
bool new_subline_refresh;
|
|
bool reset_subline;
|
|
int search;
|
|
int search_start;
|
|
struct align_pos format_align;
|
|
struct wps_data *data = gwps->data;
|
|
struct wps_state *state = gwps->state;
|
|
struct screen *display = gwps->display;
|
|
if(!gwps || !data || !state || !display)
|
|
{
|
|
return false;
|
|
}
|
|
#ifdef HAVE_LCD_BITMAP
|
|
int h = font_get(FONT_UI)->height;
|
|
int offset = 0;
|
|
gui_wps_statusbar_draw(gwps, true);
|
|
if(data->wps_sb_tag && data->show_sb_on_wps)
|
|
offset = STATUSBAR_HEIGHT;
|
|
else if ( global_settings.statusbar && !data->wps_sb_tag)
|
|
offset = STATUSBAR_HEIGHT;
|
|
|
|
/* to find out wether the peak meter is enabled we
|
|
assume it wasn't until we find a line that contains
|
|
the peak meter. We can't use peak_meter_enabled itself
|
|
because that would mean to turn off the meter thread
|
|
temporarily. (That shouldn't matter unless yield
|
|
or sleep is called but who knows...)
|
|
*/
|
|
bool enable_pm = false;
|
|
|
|
/* Set images to not to be displayed */
|
|
for (i = 0; i < MAX_IMAGES; i++) {
|
|
data->img[i].display = false;
|
|
}
|
|
#endif
|
|
/* reset to first subline if refresh all flag is set */
|
|
if (refresh_mode == WPS_REFRESH_ALL)
|
|
{
|
|
for (i=0; i<WPS_MAX_LINES; i++)
|
|
{
|
|
data->curr_subline[i] = SUBLINE_RESET;
|
|
}
|
|
}
|
|
|
|
#ifdef HAVE_LCD_CHARCELLS
|
|
for (i=0; i<8; i++) {
|
|
if (data->wps_progress_pat[i]==0)
|
|
data->wps_progress_pat[i]=display->get_locked_pattern();
|
|
}
|
|
#endif
|
|
|
|
if (!state->id3)
|
|
{
|
|
display->stop_scroll();
|
|
return false;
|
|
}
|
|
|
|
state->ff_rewind_count = ffwd_offset;
|
|
|
|
for (i = 0; i < WPS_MAX_LINES; i++)
|
|
{
|
|
reset_subline = (data->curr_subline[i] == SUBLINE_RESET);
|
|
new_subline_refresh = false;
|
|
only_one_subline = false;
|
|
|
|
/* if time to advance to next sub-line */
|
|
if (TIME_AFTER(current_tick, data->subline_expire_time[i] - 1) ||
|
|
reset_subline)
|
|
{
|
|
/* search all sublines until the next subline with time > 0
|
|
is found or we get back to the subline we started with */
|
|
if (reset_subline)
|
|
search_start = 0;
|
|
else
|
|
search_start = data->curr_subline[i];
|
|
for (search=0; search<WPS_MAX_SUBLINES; search++)
|
|
{
|
|
data->curr_subline[i]++;
|
|
|
|
/* wrap around if beyond last defined subline or WPS_MAX_SUBLINES */
|
|
if ((!data->format_lines[i][data->curr_subline[i]]) ||
|
|
(data->curr_subline[i] == WPS_MAX_SUBLINES))
|
|
{
|
|
if (data->curr_subline[i] == 1)
|
|
only_one_subline = true;
|
|
data->curr_subline[i] = 0;
|
|
}
|
|
|
|
/* if back where we started after search or
|
|
only one subline is defined on the line */
|
|
if (((search > 0) && (data->curr_subline[i] == search_start)) ||
|
|
only_one_subline)
|
|
{
|
|
/* no other subline with a time > 0 exists */
|
|
data->subline_expire_time[i] = (reset_subline?
|
|
current_tick : data->subline_expire_time[i]) + 100 * HZ;
|
|
break;
|
|
}
|
|
else
|
|
{
|
|
/* get initial time multiplier and
|
|
line type flags for this subline */
|
|
format_display(gwps, buf, sizeof(buf),
|
|
state->id3, state->nid3,
|
|
data->format_lines[i][data->curr_subline[i]],
|
|
&format_align,
|
|
&data->time_mult[i][data->curr_subline[i]],
|
|
&data->line_type[i][data->curr_subline[i]]);
|
|
|
|
/* only use this subline if subline time > 0 */
|
|
if (data->time_mult[i][data->curr_subline[i]] > 0)
|
|
{
|
|
new_subline_refresh = true;
|
|
data->subline_expire_time[i] = (reset_subline?
|
|
current_tick : data->subline_expire_time[i]) +
|
|
BASE_SUBLINE_TIME * data->time_mult[i][data->curr_subline[i]];
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
update_line = false;
|
|
|
|
if ( !data->format_lines[i][data->curr_subline[i]] )
|
|
break;
|
|
|
|
if ((data->line_type[i][data->curr_subline[i]] & refresh_mode) ||
|
|
(refresh_mode == WPS_REFRESH_ALL) ||
|
|
new_subline_refresh)
|
|
{
|
|
flags = 0;
|
|
#ifdef HAVE_LCD_BITMAP
|
|
int left_width, left_xpos;
|
|
int center_width, center_xpos;
|
|
int right_width, right_xpos;
|
|
int space_width;
|
|
int string_height;
|
|
int ypos;
|
|
#endif
|
|
|
|
format_display(gwps, buf, sizeof(buf),
|
|
state->id3, state->nid3,
|
|
data->format_lines[i][data->curr_subline[i]],
|
|
&format_align,
|
|
&data->time_mult[i][data->curr_subline[i]],
|
|
&flags);
|
|
data->line_type[i][data->curr_subline[i]] = flags;
|
|
|
|
#ifdef HAVE_LCD_BITMAP
|
|
/* progress */
|
|
if (flags & refresh_mode & WPS_REFRESH_PLAYER_PROGRESS)
|
|
{
|
|
int sb_y;
|
|
if (data->progress_top == -1)
|
|
sb_y = i*h + offset + ((h > data->progress_height + 1)
|
|
? (h - data->progress_height) / 2 : 1);
|
|
else
|
|
sb_y = data->progress_top;
|
|
|
|
if (!data->progress_end)
|
|
data->progress_end=display->width;
|
|
|
|
if (gwps->data->progressbar.have_bitmap_pb)
|
|
gui_bitmap_scrollbar_draw(display, data->progressbar.bm,
|
|
data->progress_start, sb_y,
|
|
data->progress_end-data->progress_start,
|
|
data->progressbar.bm.height,
|
|
state->id3->length?state->id3->length:1, 0,
|
|
state->id3->length?state->id3->elapsed + state->ff_rewind_count:0,
|
|
HORIZONTAL);
|
|
else
|
|
gui_scrollbar_draw(display, data->progress_start, sb_y,
|
|
data->progress_end-data->progress_start,
|
|
data->progress_height,
|
|
state->id3->length?state->id3->length:1, 0,
|
|
state->id3->length?state->id3->elapsed + state->ff_rewind_count:0,
|
|
HORIZONTAL);
|
|
#ifdef AB_REPEAT_ENABLE
|
|
if ( ab_repeat_mode_enabled() )
|
|
ab_draw_markers(display, state->id3->length,
|
|
data->progress_start, data->progress_end, sb_y,
|
|
data->progress_height);
|
|
#endif
|
|
|
|
if (cuesheet_is_enabled() && state->id3->cuesheet_type)
|
|
{
|
|
cue_draw_markers(display, state->id3->length,
|
|
data->progress_start, data->progress_end,
|
|
sb_y+1, data->progress_height-2);
|
|
}
|
|
|
|
update_line = true;
|
|
}
|
|
if (flags & refresh_mode & WPS_REFRESH_PEAK_METER) {
|
|
/* peak meter */
|
|
int peak_meter_y;
|
|
|
|
update_line = true;
|
|
peak_meter_y = i * h + offset;
|
|
|
|
/* The user might decide to have the peak meter in the last
|
|
line so that it is only displayed if no status bar is
|
|
visible. If so we neither want do draw nor enable the
|
|
peak meter. */
|
|
if (peak_meter_y + h <= display->height) {
|
|
/* found a line with a peak meter -> remember that we must
|
|
enable it later */
|
|
enable_pm = true;
|
|
peak_meter_screen(gwps->display, 0, peak_meter_y,
|
|
MIN(h, display->height - peak_meter_y));
|
|
}
|
|
}
|
|
#else
|
|
/* progress */
|
|
if (flags & refresh_mode & WPS_REFRESH_PLAYER_PROGRESS) {
|
|
if (data->full_line_progressbar)
|
|
draw_player_fullbar(gwps, buf, sizeof(buf));
|
|
else
|
|
draw_player_progress(gwps);
|
|
}
|
|
#endif
|
|
#ifdef HAVE_LCD_BITMAP
|
|
/* calculate different string sizes and positions */
|
|
display->getstringsize((unsigned char *)" ", &space_width, &string_height);
|
|
if (format_align.left != 0) {
|
|
display->getstringsize((unsigned char *)format_align.left,
|
|
&left_width, &string_height);
|
|
}
|
|
else {
|
|
left_width = 0;
|
|
}
|
|
left_xpos = 0;
|
|
|
|
if (format_align.center != 0) {
|
|
display->getstringsize((unsigned char *)format_align.center,
|
|
¢er_width, &string_height);
|
|
}
|
|
else {
|
|
center_width = 0;
|
|
}
|
|
center_xpos=(display->width - center_width) / 2;
|
|
|
|
if (format_align.right != 0) {
|
|
display->getstringsize((unsigned char *)format_align.right,
|
|
&right_width, &string_height);
|
|
}
|
|
else {
|
|
right_width = 0;
|
|
}
|
|
right_xpos = (display->width - right_width);
|
|
|
|
/* Checks for overlapping strings.
|
|
If needed the overlapping strings will be merged, separated by a
|
|
space */
|
|
|
|
/* CASE 1: left and centered string overlap */
|
|
/* there is a left string, need to merge left and center */
|
|
if ((left_width != 0 && center_width != 0) &&
|
|
(left_xpos + left_width + space_width > center_xpos)) {
|
|
/* replace the former separator '\0' of left and
|
|
center string with a space */
|
|
*(--format_align.center) = ' ';
|
|
/* calculate the new width and position of the merged string */
|
|
left_width = left_width + space_width + center_width;
|
|
left_xpos = 0;
|
|
/* there is no centered string anymore */
|
|
center_width = 0;
|
|
}
|
|
/* there is no left string, move center to left */
|
|
if ((left_width == 0 && center_width != 0) &&
|
|
(left_xpos + left_width > center_xpos)) {
|
|
/* move the center string to the left string */
|
|
format_align.left = format_align.center;
|
|
/* calculate the new width and position of the string */
|
|
left_width = center_width;
|
|
left_xpos = 0;
|
|
/* there is no centered string anymore */
|
|
center_width = 0;
|
|
}
|
|
|
|
/* CASE 2: centered and right string overlap */
|
|
/* there is a right string, need to merge center and right */
|
|
if ((center_width != 0 && right_width != 0) &&
|
|
(center_xpos + center_width + space_width > right_xpos)) {
|
|
/* replace the former separator '\0' of center and
|
|
right string with a space */
|
|
*(--format_align.right) = ' ';
|
|
/* move the center string to the right after merge */
|
|
format_align.right = format_align.center;
|
|
/* calculate the new width and position of the merged string */
|
|
right_width = center_width + space_width + right_width;
|
|
right_xpos = (display->width - right_width);
|
|
/* there is no centered string anymore */
|
|
center_width = 0;
|
|
}
|
|
/* there is no right string, move center to right */
|
|
if ((center_width != 0 && right_width == 0) &&
|
|
(center_xpos + center_width > right_xpos)) {
|
|
/* move the center string to the right string */
|
|
format_align.right = format_align.center;
|
|
/* calculate the new width and position of the string */
|
|
right_width = center_width;
|
|
right_xpos = (display->width - right_width);
|
|
/* there is no centered string anymore */
|
|
center_width = 0;
|
|
}
|
|
|
|
/* CASE 3: left and right overlap
|
|
There is no center string anymore, either there never
|
|
was one or it has been merged in case 1 or 2 */
|
|
/* there is a left string, need to merge left and right */
|
|
if ((left_width != 0 && center_width == 0 && right_width != 0) &&
|
|
(left_xpos + left_width + space_width > right_xpos)) {
|
|
/* replace the former separator '\0' of left and
|
|
right string with a space */
|
|
*(--format_align.right) = ' ';
|
|
/* calculate the new width and position of the string */
|
|
left_width = left_width + space_width + right_width;
|
|
left_xpos = 0;
|
|
/* there is no right string anymore */
|
|
right_width = 0;
|
|
}
|
|
/* there is no left string, move right to left */
|
|
if ((left_width == 0 && center_width == 0 && right_width != 0) &&
|
|
(left_xpos + left_width > right_xpos)) {
|
|
/* move the right string to the left string */
|
|
format_align.left = format_align.right;
|
|
/* calculate the new width and position of the string */
|
|
left_width = right_width;
|
|
left_xpos = 0;
|
|
/* there is no right string anymore */
|
|
right_width = 0;
|
|
}
|
|
|
|
#endif
|
|
|
|
if (flags & WPS_REFRESH_SCROLL) {
|
|
|
|
/* scroll line */
|
|
if ((refresh_mode & WPS_REFRESH_SCROLL) ||
|
|
new_subline_refresh) {
|
|
#ifdef HAVE_LCD_BITMAP
|
|
ypos = (i*string_height)+display->getymargin();
|
|
update_line = true;
|
|
|
|
if (left_width>display->width) {
|
|
display->puts_scroll(0, i,
|
|
(unsigned char *)format_align.left);
|
|
} else {
|
|
/* clear the line first */
|
|
display->set_drawmode(DRMODE_SOLID|DRMODE_INVERSEVID);
|
|
display->fillrect(0, ypos, display->width, string_height);
|
|
display->set_drawmode(DRMODE_SOLID);
|
|
|
|
/* Nasty hack: we output an empty scrolling string,
|
|
which will reset the scroller for that line */
|
|
display->puts_scroll(0, i, (unsigned char *)"");
|
|
|
|
/* print aligned strings */
|
|
if (left_width != 0)
|
|
{
|
|
display->putsxy(left_xpos, ypos,
|
|
(unsigned char *)format_align.left);
|
|
}
|
|
if (center_width != 0)
|
|
{
|
|
display->putsxy(center_xpos, ypos,
|
|
(unsigned char *)format_align.center);
|
|
}
|
|
if (right_width != 0)
|
|
{
|
|
display->putsxy(right_xpos, ypos,
|
|
(unsigned char *)format_align.right);
|
|
}
|
|
}
|
|
#else
|
|
display->puts_scroll(0, i, buf);
|
|
update_line = true;
|
|
#endif
|
|
}
|
|
}
|
|
else if (flags & (WPS_REFRESH_DYNAMIC | WPS_REFRESH_STATIC))
|
|
{
|
|
/* dynamic / static line */
|
|
if ((refresh_mode & (WPS_REFRESH_DYNAMIC|WPS_REFRESH_STATIC)) ||
|
|
new_subline_refresh)
|
|
{
|
|
#ifdef HAVE_LCD_BITMAP
|
|
ypos = (i*string_height)+display->getymargin();
|
|
update_line = true;
|
|
|
|
/* clear the line first */
|
|
display->set_drawmode(DRMODE_SOLID|DRMODE_INVERSEVID);
|
|
display->fillrect(0, ypos, display->width, string_height);
|
|
display->set_drawmode(DRMODE_SOLID);
|
|
|
|
/* Nasty hack: we output an empty scrolling string,
|
|
which will reset the scroller for that line */
|
|
display->puts_scroll(0, i, (unsigned char *)"");
|
|
|
|
/* print aligned strings */
|
|
if (left_width != 0)
|
|
{
|
|
display->putsxy(left_xpos, ypos,
|
|
(unsigned char *)format_align.left);
|
|
}
|
|
if (center_width != 0)
|
|
{
|
|
display->putsxy(center_xpos, ypos,
|
|
(unsigned char *)format_align.center);
|
|
}
|
|
if (right_width != 0)
|
|
{
|
|
display->putsxy(right_xpos, ypos,
|
|
(unsigned char *)format_align.right);
|
|
}
|
|
#else
|
|
update_line = true;
|
|
display->puts(0, i, buf);
|
|
#endif
|
|
}
|
|
}
|
|
}
|
|
#ifdef HAVE_LCD_BITMAP
|
|
if (update_line) {
|
|
wps_display_images(gwps,false);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
#ifdef HAVE_LCD_BITMAP
|
|
/* Display all images */
|
|
wps_display_images(gwps,true);
|
|
display->update();
|
|
/* Now we know wether the peak meter is used.
|
|
So we can enable / disable the peak meter thread */
|
|
data->peak_meter_enabled = enable_pm;
|
|
#endif
|
|
|
|
#if CONFIG_BACKLIGHT
|
|
if (global_settings.caption_backlight && state->id3) {
|
|
/* turn on backlight n seconds before track ends, and turn it off n
|
|
seconds into the new track. n == backlight_timeout, or 5s */
|
|
int n = backlight_timeout_value[global_settings.backlight_timeout]
|
|
* 1000;
|
|
|
|
if ( n < 1000 )
|
|
n = 5000; /* use 5s if backlight is always on or off */
|
|
|
|
if (((state->id3->elapsed < 1000) ||
|
|
((state->id3->length - state->id3->elapsed) < (unsigned)n)) &&
|
|
(state->paused == false))
|
|
backlight_on();
|
|
}
|
|
#endif
|
|
#ifdef HAVE_REMOTE_LCD
|
|
if (global_settings.remote_caption_backlight && state->id3) {
|
|
/* turn on remote backlight n seconds before track ends, and turn it
|
|
off n seconds into the new track. n == remote_backlight_timeout,
|
|
or 5s */
|
|
int n = backlight_timeout_value[global_settings.remote_backlight_timeout]
|
|
* 1000;
|
|
|
|
if ( n < 1000 )
|
|
n = 5000; /* use 5s if backlight is always on or off */
|
|
|
|
if (((state->id3->elapsed < 1000) ||
|
|
((state->id3->length - state->id3->elapsed) < (unsigned)n)) &&
|
|
(state->paused == false))
|
|
remote_backlight_on();
|
|
}
|
|
#endif
|
|
return true;
|
|
}
|
|
|
|
#ifdef HAVE_LCD_CHARCELLS
|
|
static bool draw_player_progress(struct gui_wps *gwps)
|
|
{
|
|
char player_progressbar[7];
|
|
char binline[36];
|
|
int songpos = 0;
|
|
int i,j;
|
|
struct wps_state *state = gwps->state;
|
|
struct screen *display = gwps->display;
|
|
if (!state->id3)
|
|
return false;
|
|
|
|
memset(binline, 1, sizeof binline);
|
|
memset(player_progressbar, 1, sizeof player_progressbar);
|
|
|
|
if(state->id3->elapsed >= state->id3->length)
|
|
songpos = 0;
|
|
else
|
|
{
|
|
if(state->wps_time_countup == false)
|
|
songpos = ((state->id3->elapsed - state->ff_rewind_count) * 36) /
|
|
state->id3->length;
|
|
else
|
|
songpos = ((state->id3->elapsed + state->ff_rewind_count) * 36) /
|
|
state->id3->length;
|
|
}
|
|
for (i=0; i < songpos; i++)
|
|
binline[i] = 0;
|
|
|
|
for (i=0; i<=6; i++) {
|
|
for (j=0;j<5;j++) {
|
|
player_progressbar[i] <<= 1;
|
|
player_progressbar[i] += binline[i*5+j];
|
|
}
|
|
}
|
|
display->define_pattern(gwps->data->wps_progress_pat[0], player_progressbar);
|
|
return true;
|
|
}
|
|
|
|
static char map_fullbar_char(char ascii_val)
|
|
{
|
|
if (ascii_val >= '0' && ascii_val <= '9') {
|
|
return(ascii_val - '0');
|
|
}
|
|
else if (ascii_val == ':') {
|
|
return(10);
|
|
}
|
|
else
|
|
return(11); /* anything besides a number or ':' is mapped to <blank> */
|
|
}
|
|
|
|
static void draw_player_fullbar(struct gui_wps *gwps, char* buf, int buf_size)
|
|
{
|
|
int i,j,lcd_char_pos;
|
|
|
|
char player_progressbar[7];
|
|
char binline[36];
|
|
static const char numbers[12][4][3]={
|
|
{{1,1,1},{1,0,1},{1,0,1},{1,1,1}},/*0*/
|
|
{{0,1,0},{1,1,0},{0,1,0},{0,1,0}},/*1*/
|
|
{{1,1,1},{0,0,1},{0,1,0},{1,1,1}},/*2*/
|
|
{{1,1,1},{0,0,1},{0,1,1},{1,1,1}},/*3*/
|
|
{{1,0,0},{1,1,0},{1,1,1},{0,1,0}},/*4*/
|
|
{{1,1,1},{1,1,0},{0,0,1},{1,1,0}},/*5*/
|
|
{{1,1,1},{1,0,0},{1,1,1},{1,1,1}},/*6*/
|
|
{{1,1,1},{0,0,1},{0,1,0},{1,0,0}},/*7*/
|
|
{{1,1,1},{1,1,1},{1,0,1},{1,1,1}},/*8*/
|
|
{{1,1,1},{1,1,1},{0,0,1},{1,1,1}},/*9*/
|
|
{{0,0,0},{0,1,0},{0,0,0},{0,1,0}},/*:*/
|
|
{{0,0,0},{0,0,0},{0,0,0},{0,0,0}} /*<blank>*/
|
|
};
|
|
|
|
int songpos = 0;
|
|
int digits[6];
|
|
int time;
|
|
char timestr[7];
|
|
|
|
struct wps_state *state = gwps->state;
|
|
struct screen *display = gwps->display;
|
|
struct wps_data *data = gwps->data;
|
|
|
|
for (i=0; i < buf_size; i++)
|
|
buf[i] = ' ';
|
|
|
|
if(state->id3->elapsed >= state->id3->length)
|
|
songpos = 55;
|
|
else {
|
|
if(state->wps_time_countup == false)
|
|
songpos = ((state->id3->elapsed - state->ff_rewind_count) * 55) /
|
|
state->id3->length;
|
|
else
|
|
songpos = ((state->id3->elapsed + state->ff_rewind_count) * 55) /
|
|
state->id3->length;
|
|
}
|
|
|
|
time=(state->id3->elapsed + state->ff_rewind_count);
|
|
|
|
memset(timestr, 0, sizeof(timestr));
|
|
format_time(timestr, sizeof(timestr), time);
|
|
for(lcd_char_pos=0; lcd_char_pos<6; lcd_char_pos++) {
|
|
digits[lcd_char_pos] = map_fullbar_char(timestr[lcd_char_pos]);
|
|
}
|
|
|
|
/* build the progressbar-icons */
|
|
for (lcd_char_pos=0; lcd_char_pos<6; lcd_char_pos++) {
|
|
memset(binline, 0, sizeof binline);
|
|
memset(player_progressbar, 0, sizeof player_progressbar);
|
|
|
|
/* make the character (progressbar & digit)*/
|
|
for (i=0; i<7; i++) {
|
|
for (j=0;j<5;j++) {
|
|
/* make the progressbar */
|
|
if (lcd_char_pos==(songpos/5)) {
|
|
/* partial */
|
|
if ((j<(songpos%5))&&(i>4))
|
|
binline[i*5+j] = 1;
|
|
else
|
|
binline[i*5+j] = 0;
|
|
}
|
|
else {
|
|
if (lcd_char_pos<(songpos/5)) {
|
|
/* full character */
|
|
if (i>4)
|
|
binline[i*5+j] = 1;
|
|
}
|
|
}
|
|
/* insert the digit */
|
|
if ((j<3)&&(i<4)) {
|
|
if (numbers[digits[lcd_char_pos]][i][j]==1)
|
|
binline[i*5+j] = 1;
|
|
}
|
|
}
|
|
}
|
|
|
|
for (i=0; i<=6; i++) {
|
|
for (j=0;j<5;j++) {
|
|
player_progressbar[i] <<= 1;
|
|
player_progressbar[i] += binline[i*5+j];
|
|
}
|
|
}
|
|
|
|
display->define_pattern(data->wps_progress_pat[lcd_char_pos+1],
|
|
player_progressbar);
|
|
buf[lcd_char_pos]=data->wps_progress_pat[lcd_char_pos+1];
|
|
|
|
}
|
|
|
|
/* make rest of the progressbar if necessary */
|
|
if (songpos/5>5) {
|
|
|
|
/* set the characters positions that use the full 5 pixel wide bar */
|
|
for (lcd_char_pos=6; lcd_char_pos < (songpos/5); lcd_char_pos++)
|
|
buf[lcd_char_pos] = 0x86; /* '_' */
|
|
|
|
/* build the partial bar character for the tail character position */
|
|
memset(binline, 0, sizeof binline);
|
|
memset(player_progressbar, 0, sizeof player_progressbar);
|
|
|
|
for (i=5; i<7; i++) {
|
|
for (j=0;j<5;j++) {
|
|
if (j<(songpos%5)) {
|
|
binline[i*5+j] = 1;
|
|
}
|
|
}
|
|
}
|
|
|
|
for (i=0; i<7; i++) {
|
|
for (j=0;j<5;j++) {
|
|
player_progressbar[i] <<= 1;
|
|
player_progressbar[i] += binline[i*5+j];
|
|
}
|
|
}
|
|
|
|
display->define_pattern(data->wps_progress_pat[7],player_progressbar);
|
|
|
|
buf[songpos/5]=data->wps_progress_pat[7];
|
|
}
|
|
}
|
|
#endif
|
|
|
|
/* set volume */
|
|
void setvol(void)
|
|
{
|
|
if (global_settings.volume < sound_min(SOUND_VOLUME))
|
|
global_settings.volume = sound_min(SOUND_VOLUME);
|
|
if (global_settings.volume > sound_max(SOUND_VOLUME))
|
|
global_settings.volume = sound_max(SOUND_VOLUME);
|
|
sound_set_volume(global_settings.volume);
|
|
settings_save();
|
|
}
|
|
/* return true if screen restore is needed
|
|
return false otherwise
|
|
*/
|
|
bool update_onvol_change(struct gui_wps * gwps)
|
|
{
|
|
gui_wps_statusbar_draw(gwps, false);
|
|
gui_wps_refresh(gwps, 0, WPS_REFRESH_NON_STATIC);
|
|
|
|
#ifdef HAVE_LCD_CHARCELLS
|
|
gui_splash(gwps->display, 0, "Vol: %3d dB",
|
|
sound_val2phys(SOUND_VOLUME, global_settings.volume));
|
|
return true;
|
|
#endif
|
|
return false;
|
|
}
|
|
|
|
bool ffwd_rew(int button)
|
|
{
|
|
static const int ff_rew_steps[] = {
|
|
1000, 2000, 3000, 4000,
|
|
5000, 6000, 8000, 10000,
|
|
15000, 20000, 25000, 30000,
|
|
45000, 60000
|
|
};
|
|
|
|
unsigned int step = 0; /* current ff/rewind step */
|
|
unsigned int max_step = 0; /* maximum ff/rewind step */
|
|
int ff_rewind_count = 0; /* current ff/rewind count (in ticks) */
|
|
int direction = -1; /* forward=1 or backward=-1 */
|
|
long accel_tick = 0; /* next time at which to bump the step size */
|
|
bool exit = false;
|
|
bool usb = false;
|
|
int i = 0;
|
|
|
|
if (button == ACTION_NONE)
|
|
{
|
|
status_set_ffmode(0);
|
|
return usb;
|
|
}
|
|
while (!exit)
|
|
{
|
|
switch ( button )
|
|
{
|
|
case ACTION_WPS_SEEKFWD:
|
|
direction = 1;
|
|
case ACTION_WPS_SEEKBACK:
|
|
if (wps_state.ff_rewind)
|
|
{
|
|
if (direction == 1)
|
|
{
|
|
/* fast forwarding, calc max step relative to end */
|
|
max_step = (wps_state.id3->length -
|
|
(wps_state.id3->elapsed +
|
|
ff_rewind_count)) *
|
|
FF_REWIND_MAX_PERCENT / 100;
|
|
}
|
|
else
|
|
{
|
|
/* rewinding, calc max step relative to start */
|
|
max_step = (wps_state.id3->elapsed + ff_rewind_count) *
|
|
FF_REWIND_MAX_PERCENT / 100;
|
|
}
|
|
|
|
max_step = MAX(max_step, MIN_FF_REWIND_STEP);
|
|
|
|
if (step > max_step)
|
|
step = max_step;
|
|
|
|
ff_rewind_count += step * direction;
|
|
|
|
if (global_settings.ff_rewind_accel != 0 &&
|
|
current_tick >= accel_tick)
|
|
{
|
|
step *= 2;
|
|
accel_tick = current_tick +
|
|
global_settings.ff_rewind_accel*HZ;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if ( (audio_status() & AUDIO_STATUS_PLAY) &&
|
|
wps_state.id3 && wps_state.id3->length )
|
|
{
|
|
if (!wps_state.paused)
|
|
#if (CONFIG_CODEC == SWCODEC)
|
|
audio_pre_ff_rewind();
|
|
#else
|
|
audio_pause();
|
|
#endif
|
|
#if CONFIG_KEYPAD == PLAYER_PAD
|
|
FOR_NB_SCREENS(i)
|
|
gui_wps[i].display->stop_scroll();
|
|
#endif
|
|
if (direction > 0)
|
|
status_set_ffmode(STATUS_FASTFORWARD);
|
|
else
|
|
status_set_ffmode(STATUS_FASTBACKWARD);
|
|
|
|
wps_state.ff_rewind = true;
|
|
|
|
step = ff_rew_steps[global_settings.ff_rewind_min_step];
|
|
|
|
accel_tick = current_tick +
|
|
global_settings.ff_rewind_accel*HZ;
|
|
}
|
|
else
|
|
break;
|
|
}
|
|
|
|
if (direction > 0) {
|
|
if ((wps_state.id3->elapsed + ff_rewind_count) >
|
|
wps_state.id3->length)
|
|
ff_rewind_count = wps_state.id3->length -
|
|
wps_state.id3->elapsed;
|
|
}
|
|
else {
|
|
if ((int)(wps_state.id3->elapsed + ff_rewind_count) < 0)
|
|
ff_rewind_count = -wps_state.id3->elapsed;
|
|
}
|
|
|
|
FOR_NB_SCREENS(i)
|
|
gui_wps_refresh(&gui_wps[i],
|
|
(wps_state.wps_time_countup == false)?
|
|
ff_rewind_count:-ff_rewind_count,
|
|
WPS_REFRESH_PLAYER_PROGRESS |
|
|
WPS_REFRESH_DYNAMIC);
|
|
|
|
break;
|
|
|
|
case ACTION_WPS_STOPSEEK:
|
|
wps_state.id3->elapsed = wps_state.id3->elapsed+ff_rewind_count;
|
|
audio_ff_rewind(wps_state.id3->elapsed);
|
|
ff_rewind_count = 0;
|
|
wps_state.ff_rewind = false;
|
|
status_set_ffmode(0);
|
|
#if (CONFIG_CODEC != SWCODEC)
|
|
if (!wps_state.paused)
|
|
audio_resume();
|
|
#endif
|
|
#ifdef HAVE_LCD_CHARCELLS
|
|
gui_wps_display();
|
|
#endif
|
|
exit = true;
|
|
break;
|
|
|
|
default:
|
|
if(default_event_handler(button) == SYS_USB_CONNECTED) {
|
|
status_set_ffmode(0);
|
|
usb = true;
|
|
exit = true;
|
|
}
|
|
break;
|
|
}
|
|
if (!exit)
|
|
button = get_action(CONTEXT_WPS,TIMEOUT_BLOCK);
|
|
}
|
|
action_signalscreenchange();
|
|
return usb;
|
|
}
|
|
|
|
bool gui_wps_display(void)
|
|
{
|
|
int i;
|
|
if (!wps_state.id3 && !(audio_status() & AUDIO_STATUS_PLAY))
|
|
{
|
|
global_status.resume_index = -1;
|
|
#ifdef HAVE_LCD_CHARCELLS
|
|
gui_syncsplash(HZ, str(LANG_END_PLAYLIST_PLAYER));
|
|
#else
|
|
gui_syncstatusbar_draw(&statusbars, true);
|
|
gui_syncsplash(HZ, str(LANG_END_PLAYLIST_RECORDER));
|
|
#endif
|
|
return true;
|
|
}
|
|
else
|
|
{
|
|
FOR_NB_SCREENS(i)
|
|
{
|
|
gui_wps[i].display->clear_display();
|
|
if (!gui_wps[i].data->wps_loaded) {
|
|
if ( !gui_wps[i].data->format_buffer[0] ) {
|
|
/* set the default wps for the main-screen */
|
|
if(i == 0)
|
|
{
|
|
#ifdef HAVE_LCD_BITMAP
|
|
#if LCD_DEPTH > 1
|
|
unload_wps_backdrop();
|
|
#endif
|
|
wps_data_load(gui_wps[i].data,
|
|
"%s%?it<%?in<%in. |>%it|%fn>\n"
|
|
"%s%?ia<%ia|%?d2<%d2|(root)>>\n"
|
|
"%s%?id<%id|%?d1<%d1|(root)>> %?iy<(%iy)|>\n"
|
|
"\n"
|
|
"%al%pc/%pt%ar[%pp:%pe]\n"
|
|
"%fbkBit %?fv<avg|> %?iv<(id3v%iv)|(no id3)>\n"
|
|
"%pb\n"
|
|
"%pm\n", false);
|
|
#else
|
|
wps_data_load(gui_wps[i].data,
|
|
"%s%pp/%pe: %?it<%it|%fn> - %?ia<%ia|%d2> - %?id<%id|%d1>\n"
|
|
"%pc%?ps<*|/>%pt\n", false);
|
|
#endif
|
|
}
|
|
#if NB_SCREENS == 2
|
|
/* set the default wps for the remote-screen */
|
|
else if(i == 1)
|
|
{
|
|
wps_data_load(gui_wps[i].data,
|
|
"%s%?ia<%ia|%?d2<%d2|(root)>>\n"
|
|
"%s%?it<%?in<%in. |>%it|%fn>\n"
|
|
"%al%pc/%pt%ar[%pp:%pe]\n"
|
|
"%fbkBit %?fv<avg|> %?iv<(id3v%iv)|(no id3)>\n"
|
|
"%pb", false);
|
|
}
|
|
#endif
|
|
}
|
|
}
|
|
}
|
|
}
|
|
yield();
|
|
FOR_NB_SCREENS(i)
|
|
{
|
|
gui_wps_refresh(&gui_wps[i], 0, WPS_REFRESH_ALL);
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool update(struct gui_wps *gwps)
|
|
{
|
|
bool track_changed = audio_has_changed_track();
|
|
bool retcode = false;
|
|
|
|
gwps->state->nid3 = audio_next_track();
|
|
if (track_changed)
|
|
{
|
|
gwps->display->stop_scroll();
|
|
gwps->state->id3 = audio_current_track();
|
|
|
|
if (cuesheet_is_enabled() && gwps->state->id3->cuesheet_type
|
|
&& strcmp(gwps->state->id3->path, curr_cue->audio_filename))
|
|
{
|
|
/* the current cuesheet isn't the right one any more */
|
|
|
|
if (!strcmp(gwps->state->id3->path, temp_cue->audio_filename)) {
|
|
/* We have the new cuesheet in memory (temp_cue),
|
|
let's make it the current one ! */
|
|
memcpy(curr_cue, temp_cue, sizeof(struct cuesheet));
|
|
}
|
|
else {
|
|
/* We need to parse the new cuesheet */
|
|
|
|
char cuepath[MAX_PATH];
|
|
strncpy(cuepath, gwps->state->id3->path, MAX_PATH);
|
|
char *dot = strrchr(cuepath, '.');
|
|
strcpy(dot, ".cue");
|
|
|
|
if (parse_cuesheet(cuepath, curr_cue))
|
|
{
|
|
gwps->state->id3->cuesheet_type = 1;
|
|
strcpy(curr_cue->audio_filename, gwps->state->id3->path);
|
|
}
|
|
}
|
|
|
|
cue_spoof_id3(curr_cue, gwps->state->id3);
|
|
}
|
|
|
|
if (gui_wps_display())
|
|
retcode = true;
|
|
else{
|
|
gui_wps_refresh(gwps, 0, WPS_REFRESH_ALL);
|
|
}
|
|
|
|
if (gwps->state->id3)
|
|
memcpy(gwps->state->current_track_path, gwps->state->id3->path,
|
|
sizeof(gwps->state->current_track_path));
|
|
}
|
|
|
|
if (gwps->state->id3)
|
|
{
|
|
if (cuesheet_is_enabled() && gwps->state->id3->cuesheet_type
|
|
&& (gwps->state->id3->elapsed < curr_cue->curr_track->offset
|
|
|| (curr_cue->curr_track_idx < curr_cue->track_count - 1
|
|
&& gwps->state->id3->elapsed >= (curr_cue->curr_track+1)->offset)))
|
|
{
|
|
/* We've changed tracks within the cuesheet :
|
|
we need to update the ID3 info and refresh the WPS */
|
|
|
|
cue_find_current_track(curr_cue, gwps->state->id3->elapsed);
|
|
cue_spoof_id3(curr_cue, gwps->state->id3);
|
|
|
|
gwps->display->stop_scroll();
|
|
if (gui_wps_display())
|
|
retcode = true;
|
|
else
|
|
gui_wps_refresh(gwps, 0, WPS_REFRESH_ALL);
|
|
}
|
|
else
|
|
gui_wps_refresh(gwps, 0, WPS_REFRESH_NON_STATIC);
|
|
}
|
|
|
|
gui_wps_statusbar_draw(gwps, false);
|
|
|
|
return retcode;
|
|
}
|
|
|
|
|
|
void display_keylock_text(bool locked)
|
|
{
|
|
char* s;
|
|
int i;
|
|
FOR_NB_SCREENS(i)
|
|
gui_wps[i].display->stop_scroll();
|
|
|
|
#ifdef HAVE_LCD_CHARCELLS
|
|
if(locked)
|
|
s = str(LANG_KEYLOCK_ON_PLAYER);
|
|
else
|
|
s = str(LANG_KEYLOCK_OFF_PLAYER);
|
|
#else
|
|
if(locked)
|
|
s = str(LANG_KEYLOCK_ON_RECORDER);
|
|
else
|
|
s = str(LANG_KEYLOCK_OFF_RECORDER);
|
|
#endif
|
|
gui_syncsplash(HZ, s);
|
|
}
|
|
|