2007-11-11 12:29:37 +00:00
|
|
|
/***************************************************************************
|
|
|
|
* __________ __ ___.
|
|
|
|
* Open \______ \ ____ ____ | | _\_ |__ _______ ___
|
|
|
|
* Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
|
|
|
|
* Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
|
|
|
|
* Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
|
|
|
|
* \/ \/ \/ \/ \/
|
|
|
|
* $Id$
|
|
|
|
*
|
|
|
|
* Copyright (C) 2007 Nicolas Pennequin
|
|
|
|
*
|
2008-06-28 18:10:04 +00:00
|
|
|
* This program is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU General Public License
|
|
|
|
* as published by the Free Software Foundation; either version 2
|
|
|
|
* of the License, or (at your option) any later version.
|
2007-11-11 12:29:37 +00:00
|
|
|
*
|
|
|
|
* This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
|
|
|
|
* KIND, either express or implied.
|
|
|
|
*
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
#include <string.h>
|
|
|
|
#include "sprintf.h"
|
|
|
|
#include "system.h"
|
|
|
|
#include "albumart.h"
|
|
|
|
#include "id3.h"
|
|
|
|
#include "gwps.h"
|
|
|
|
#include "buffering.h"
|
|
|
|
#include "dircache.h"
|
|
|
|
#include "debug.h"
|
2007-11-21 21:28:27 +00:00
|
|
|
#include "misc.h"
|
2008-01-19 18:05:06 +00:00
|
|
|
#include "settings.h"
|
2007-11-11 12:29:37 +00:00
|
|
|
|
|
|
|
|
|
|
|
/* Strip filename from a full path
|
|
|
|
*
|
|
|
|
* buf - buffer to extract directory to.
|
|
|
|
* buf_size - size of buffer.
|
|
|
|
* fullpath - fullpath to extract from.
|
|
|
|
*
|
|
|
|
* Split the directory part of the given fullpath and store it in buf
|
|
|
|
* (including last '/').
|
|
|
|
* The function return parameter is a pointer to the filename
|
|
|
|
* inside the given fullpath.
|
|
|
|
*/
|
|
|
|
static char* strip_filename(char* buf, int buf_size, const char* fullpath)
|
|
|
|
{
|
|
|
|
char* sep;
|
|
|
|
int len;
|
|
|
|
|
|
|
|
if (!buf || buf_size <= 0 || !fullpath)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
/* if 'fullpath' is only a filename return immediately */
|
|
|
|
sep = strrchr(fullpath, '/');
|
|
|
|
if (sep == NULL)
|
|
|
|
{
|
|
|
|
buf[0] = 0;
|
|
|
|
return (char*)fullpath;
|
|
|
|
}
|
|
|
|
|
|
|
|
len = MIN(sep - fullpath + 1, buf_size - 1);
|
|
|
|
strncpy(buf, fullpath, len);
|
|
|
|
buf[len] = 0;
|
|
|
|
return (sep + 1);
|
|
|
|
}
|
|
|
|
|
2007-11-15 20:22:03 +00:00
|
|
|
/* Make sure part of path only contain chars valid for a FAT32 long name.
|
|
|
|
* Double quotes are replaced with single quotes, other unsupported chars
|
|
|
|
* are replaced with an underscore.
|
|
|
|
*
|
|
|
|
* path - path to modify.
|
|
|
|
* offset - where in path to start checking.
|
|
|
|
* count - number of chars to check.
|
|
|
|
*/
|
|
|
|
static void fix_path_part(char* path, int offset, int count)
|
|
|
|
{
|
|
|
|
static const char invalid_chars[] = "*/:<>?\\|";
|
|
|
|
int i;
|
|
|
|
|
|
|
|
path += offset;
|
|
|
|
|
|
|
|
for (i = 0; i <= count; i++, path++)
|
|
|
|
{
|
2008-01-19 18:05:06 +00:00
|
|
|
if (*path == 0)
|
|
|
|
return;
|
2007-11-15 20:22:03 +00:00
|
|
|
if (*path == '"')
|
|
|
|
*path = '\'';
|
|
|
|
else if (strchr(invalid_chars, *path))
|
|
|
|
*path = '_';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-11-11 12:29:37 +00:00
|
|
|
/* Look for the first matching album art bitmap in the following list:
|
|
|
|
* ./<trackname><size>.bmp
|
|
|
|
* ./<albumname><size>.bmp
|
|
|
|
* ./cover<size>.bmp
|
|
|
|
* ../<albumname><size>.bmp
|
|
|
|
* ../cover<size>.bmp
|
2008-01-19 18:05:06 +00:00
|
|
|
* ROCKBOX_DIR/albumart/<artist>-<albumname><size>.bmp
|
2007-11-11 12:29:37 +00:00
|
|
|
* <size> is the value of the size_string parameter, <trackname> and
|
|
|
|
* <albumname> are read from the ID3 metadata.
|
|
|
|
* If a matching bitmap is found, its filename is stored in buf.
|
|
|
|
* Return value is true if a bitmap was found, false otherwise.
|
|
|
|
*/
|
2007-12-09 18:48:02 +00:00
|
|
|
bool search_albumart_files(const struct mp3entry *id3, const char *size_string,
|
|
|
|
char *buf, int buflen)
|
2007-11-11 12:29:37 +00:00
|
|
|
{
|
|
|
|
char path[MAX_PATH + 1];
|
|
|
|
char dir[MAX_PATH + 1];
|
|
|
|
bool found = false;
|
|
|
|
const char *trackname;
|
2008-04-19 15:21:09 +00:00
|
|
|
const char *artist;
|
2007-11-15 20:22:03 +00:00
|
|
|
int dirlen;
|
|
|
|
int albumlen;
|
2007-11-11 12:29:37 +00:00
|
|
|
|
|
|
|
if (!id3 || !buf)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
trackname = id3->path;
|
2008-02-05 11:56:21 +00:00
|
|
|
|
|
|
|
if (strcmp(trackname, "No file!") == 0)
|
|
|
|
return false;
|
|
|
|
|
2007-11-11 12:29:37 +00:00
|
|
|
strip_filename(dir, sizeof(dir), trackname);
|
2007-11-15 20:22:03 +00:00
|
|
|
dirlen = strlen(dir);
|
|
|
|
albumlen = id3->album ? strlen(id3->album) : 0;
|
2007-11-11 12:29:37 +00:00
|
|
|
|
|
|
|
/* the first file we look for is one specific to the track playing */
|
2008-02-06 19:51:19 +00:00
|
|
|
strip_extension(path, sizeof(path) - strlen(size_string) - 4, trackname);
|
2007-11-11 12:29:37 +00:00
|
|
|
strcat(path, size_string);
|
|
|
|
strcat(path, ".bmp");
|
|
|
|
found = file_exists(path);
|
2007-11-15 20:22:03 +00:00
|
|
|
if (!found && albumlen > 0)
|
2007-11-11 12:29:37 +00:00
|
|
|
{
|
|
|
|
/* if it doesn't exist,
|
|
|
|
* we look for a file specific to the track's album name */
|
2007-11-15 20:22:03 +00:00
|
|
|
snprintf(path, sizeof(path),
|
|
|
|
"%s%s%s.bmp", dir, id3->album, size_string);
|
|
|
|
fix_path_part(path, dirlen, albumlen);
|
2007-11-11 12:29:37 +00:00
|
|
|
found = file_exists(path);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!found)
|
|
|
|
{
|
|
|
|
/* if it still doesn't exist, we look for a generic file */
|
2007-11-15 20:22:03 +00:00
|
|
|
snprintf(path, sizeof(path),
|
|
|
|
"%scover%s.bmp", dir, size_string);
|
2007-11-11 12:29:37 +00:00
|
|
|
found = file_exists(path);
|
2008-01-19 18:05:06 +00:00
|
|
|
}
|
|
|
|
|
2008-04-19 15:21:09 +00:00
|
|
|
artist = id3->albumartist != NULL ? id3->albumartist : id3->artist;
|
|
|
|
|
|
|
|
if (!found && artist && id3->album)
|
2008-01-19 18:05:06 +00:00
|
|
|
{
|
|
|
|
/* look in the albumart subdir of .rockbox */
|
|
|
|
snprintf(path, sizeof(path),
|
|
|
|
ROCKBOX_DIR "/albumart/%s-%s%s.bmp",
|
2008-04-19 15:21:09 +00:00
|
|
|
artist,
|
2008-01-19 18:05:06 +00:00
|
|
|
id3->album,
|
|
|
|
size_string);
|
|
|
|
fix_path_part(path, strlen(ROCKBOX_DIR "/albumart/"), MAX_PATH);
|
|
|
|
found = file_exists(path);
|
2007-11-11 12:29:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!found)
|
|
|
|
{
|
|
|
|
/* if it still doesn't exist,
|
|
|
|
* we continue to search in the parent directory */
|
2007-11-15 20:22:03 +00:00
|
|
|
strcpy(path, dir);
|
|
|
|
path[dirlen - 1] = 0;
|
|
|
|
strip_filename(dir, sizeof(dir), path);
|
|
|
|
dirlen = strlen(dir);
|
2007-11-11 12:29:37 +00:00
|
|
|
}
|
|
|
|
|
2007-11-15 20:22:03 +00:00
|
|
|
/* only try parent if there is one */
|
|
|
|
if (dirlen > 0)
|
2007-11-11 12:29:37 +00:00
|
|
|
{
|
2007-11-15 20:22:03 +00:00
|
|
|
if (!found && albumlen > 0)
|
|
|
|
{
|
|
|
|
/* we look in the parent directory
|
|
|
|
* for a file specific to the track's album name */
|
|
|
|
snprintf(path, sizeof(path),
|
|
|
|
"%s%s%s.bmp", dir, id3->album, size_string);
|
|
|
|
fix_path_part(path, dirlen, albumlen);
|
|
|
|
found = file_exists(path);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!found)
|
|
|
|
{
|
|
|
|
/* if it still doesn't exist, we look in the parent directory
|
|
|
|
* for a generic file */
|
|
|
|
snprintf(path, sizeof(path),
|
|
|
|
"%scover%s.bmp", dir, size_string);
|
|
|
|
found = file_exists(path);
|
|
|
|
}
|
2007-11-11 12:29:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!found)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
strncpy(buf, path, buflen);
|
|
|
|
DEBUGF("Album art found: %s\n", path);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Look for albumart bitmap in the same dir as the track and in its parent dir.
|
|
|
|
* Stores the found filename in the buf parameter.
|
|
|
|
* Returns true if a bitmap was found, false otherwise */
|
|
|
|
bool find_albumart(const struct mp3entry *id3, char *buf, int buflen)
|
|
|
|
{
|
|
|
|
if (!id3 || !buf)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
char size_string[9];
|
|
|
|
struct wps_data *data = gui_wps[0].data;
|
|
|
|
|
|
|
|
if (!data)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
DEBUGF("Looking for album art for %s\n", id3->path);
|
|
|
|
|
|
|
|
/* Write the size string, e.g. ".100x100". */
|
|
|
|
snprintf(size_string, sizeof(size_string), ".%dx%d",
|
|
|
|
data->albumart_max_width, data->albumart_max_height);
|
|
|
|
|
|
|
|
/* First we look for a bitmap of the right size */
|
2007-12-09 18:48:02 +00:00
|
|
|
if (search_albumart_files(id3, size_string, buf, buflen))
|
2007-11-11 12:29:37 +00:00
|
|
|
return true;
|
|
|
|
|
|
|
|
/* Then we look for generic bitmaps */
|
|
|
|
*size_string = 0;
|
2007-12-09 18:48:02 +00:00
|
|
|
return search_albumart_files(id3, size_string, buf, buflen);
|
2007-11-11 12:29:37 +00:00
|
|
|
}
|
|
|
|
|
2007-11-12 01:31:42 +00:00
|
|
|
/* Draw the album art bitmap from the given handle ID onto the given WPS.
|
|
|
|
Call with clear = true to clear the bitmap instead of drawing it. */
|
|
|
|
void draw_album_art(struct gui_wps *gwps, int handle_id, bool clear)
|
2007-11-11 12:29:37 +00:00
|
|
|
{
|
|
|
|
if (!gwps || !gwps->data || !gwps->display || handle_id < 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
struct wps_data *data = gwps->data;
|
|
|
|
|
|
|
|
#ifdef HAVE_REMOTE_LCD
|
|
|
|
/* No album art on RWPS */
|
|
|
|
if (data->remote_wps)
|
|
|
|
return;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
struct bitmap *bmp;
|
2007-11-12 14:26:47 +00:00
|
|
|
if (bufgetdata(handle_id, 0, (void *)&bmp) <= 0)
|
|
|
|
return;
|
2007-11-11 12:29:37 +00:00
|
|
|
|
|
|
|
short x = data->albumart_x;
|
|
|
|
short y = data->albumart_y;
|
|
|
|
short width = bmp->width;
|
|
|
|
short height = bmp->height;
|
|
|
|
|
|
|
|
if (data->albumart_max_width > 0)
|
|
|
|
{
|
|
|
|
/* Crop if the bitmap is too wide */
|
|
|
|
width = MIN(bmp->width, data->albumart_max_width);
|
|
|
|
|
|
|
|
/* Align */
|
|
|
|
if (data->albumart_xalign & WPS_ALBUMART_ALIGN_RIGHT)
|
|
|
|
x += data->albumart_max_width - width;
|
|
|
|
else if (data->albumart_xalign & WPS_ALBUMART_ALIGN_CENTER)
|
|
|
|
x += (data->albumart_max_width - width) / 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (data->albumart_max_height > 0)
|
|
|
|
{
|
|
|
|
/* Crop if the bitmap is too high */
|
|
|
|
height = MIN(bmp->height, data->albumart_max_height);
|
|
|
|
|
|
|
|
/* Align */
|
|
|
|
if (data->albumart_yalign & WPS_ALBUMART_ALIGN_BOTTOM)
|
|
|
|
y += data->albumart_max_height - height;
|
|
|
|
else if (data->albumart_yalign & WPS_ALBUMART_ALIGN_CENTER)
|
|
|
|
y += (data->albumart_max_height - height) / 2;
|
|
|
|
}
|
|
|
|
|
2007-11-12 01:31:42 +00:00
|
|
|
if (!clear)
|
|
|
|
{
|
|
|
|
/* Draw the bitmap */
|
|
|
|
gwps->display->set_drawmode(DRMODE_FG);
|
|
|
|
gwps->display->bitmap_part((fb_data*)bmp->data, 0, 0, bmp->width,
|
|
|
|
x, y, width, height);
|
|
|
|
gwps->display->set_drawmode(DRMODE_SOLID);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Clear the bitmap */
|
|
|
|
gwps->display->set_drawmode(DRMODE_SOLID|DRMODE_INVERSEVID);
|
|
|
|
gwps->display->fillrect(x, y, width, height);
|
|
|
|
gwps->display->set_drawmode(DRMODE_SOLID);
|
|
|
|
}
|
2007-11-11 12:29:37 +00:00
|
|
|
}
|