554 lines
14 KiB
C++
554 lines
14 KiB
C++
#include <precomp.h>
|
|
#include <setjmp.h>
|
|
#include <bfc/bfc_assert.h>
|
|
#include <api.h>
|
|
|
|
#include <bfc/wasabi_std.h>
|
|
#include <tataki/bitmap/bitmap.h>
|
|
#include <api/skin/skinfilter.h> // ApplySkinFilters
|
|
|
|
#include "imgldr.h"
|
|
#ifdef _WIN32
|
|
#include <api/imgldr/winbmp.h>
|
|
#endif
|
|
#include <api/imgldr/skinbmps.h>
|
|
|
|
#include <api/skin/skinparse.h>
|
|
#include <api/skin/skinelem.h>
|
|
#include <api/skin/gammamgr.h>
|
|
|
|
#include <api/service/svcs/svc_skinfilter.h>
|
|
#include <api/service/svcs/svc_imgload.h>
|
|
#include <api/service/svcs/svc_imggen.h>
|
|
#include "ImgLoaderEnum.h"
|
|
#include <api/memmgr/api_memmgr.h>
|
|
#include <bfc/string/PathString.h>
|
|
#include <api/locales/localesmgr.h>
|
|
|
|
//#define DEBUG_OUTPUT
|
|
|
|
#define IMAGEHEADERLEN 256
|
|
|
|
#ifdef _WIN32
|
|
ARGB32 *imageLoader::makeBmp(OSMODULEHANDLE hInst, int id, int *has_alpha, int *w, int *h, const wchar_t *forcegroup)
|
|
{
|
|
ARGB32 *bits = makeBmp(StringPrintfW(L"res://%u,%i", hInst, id), NULL, has_alpha, w, h, NULL, TRUE, NULL);
|
|
if (bits && *w > 0 && *h > 0)
|
|
{
|
|
ApplySkinFilters::apply(StringPrintfW(L"resource:%x,%d", hInst, id), forcegroup, bits, *w, *h);
|
|
}
|
|
return bits;
|
|
}
|
|
#endif
|
|
|
|
StringW imageLoader::getWallpaper()
|
|
{
|
|
StringW ret(L"");
|
|
#ifdef WIN32
|
|
HKEY hkey;
|
|
static wchar_t file[MAX_PATH];
|
|
file[0] = 0;
|
|
if (RegOpenKey(HKEY_CURRENT_USER, TEXT("Control Panel\\Desktop"), &hkey) == ERROR_SUCCESS)
|
|
{
|
|
unsigned long len = MAX_PATH;
|
|
RegQueryValueExW(hkey, L"Wallpaper", 0, NULL, (unsigned char *)&file, &len);
|
|
RegCloseKey(hkey);
|
|
}
|
|
if (file[0] && GetFileAttributesW(file) != (DWORD) - 1) ret = file;
|
|
#endif
|
|
return ret;
|
|
}
|
|
extern StringW g_resourcepath;
|
|
|
|
ARGB32 *imageLoader::makeBmp(const wchar_t *_filename, const wchar_t *path, int *has_alpha, int *w, int *h, ifc_xmlreaderparams *params, bool addMem, int *force_nocache)
|
|
{
|
|
if (!_filename || !*_filename)
|
|
return 0;
|
|
|
|
|
|
ARGB32 *ret = NULL;
|
|
|
|
if (has_alpha != NULL)
|
|
*has_alpha = 0; //BU
|
|
|
|
// test image generator services FIRST
|
|
ImgGeneratorEnum ige(_filename);
|
|
svc_imageGenerator *gen;
|
|
while ((gen = ige.getNext()) != NULL)
|
|
{
|
|
ret = gen->genImage(_filename, has_alpha, w, h, params);
|
|
int cacheable = gen->outputCacheable();
|
|
ige.release(gen);
|
|
if (ret != NULL)
|
|
{
|
|
ApplySkinFilters::apply(params->getItemValue(L"id"), params->getItemValue(L"gammagroup"), ret, *w, *h);
|
|
if (addMem) addMemUsage(_filename, (*w) * (*h) * sizeof(int));
|
|
optimizeHasAlpha(ret, *w * *h, has_alpha);
|
|
// don't try to cache generated images
|
|
if (force_nocache) *force_nocache = !cacheable;
|
|
return ret;
|
|
}
|
|
}
|
|
|
|
StringW wallpaper;
|
|
if (!WCSICMP(_filename, L"$wallpaper"))
|
|
{
|
|
wallpaper = getWallpaper();
|
|
_filename = wallpaper.getValue();
|
|
}
|
|
|
|
MemBlock<uint8_t> mem;
|
|
|
|
wchar_t olddir[PATH_MAX] = {0};
|
|
Wasabi::Std::getCurDir(olddir, PATH_MAX);
|
|
Wasabi::Std::setCurDir(WASABI_API_APP->path_getAppPath());
|
|
|
|
StringW file;
|
|
|
|
// benski> try language pack first
|
|
StringPathCombine skinLocalePath(LocalesManager::getLocaleRoot(), WASABI_API_SKIN->getSkinName());
|
|
file.swap(StringPathCombine(skinLocalePath, _filename));
|
|
OSFILETYPE in = WFOPEN(file, WF_READONLY_BINARY);
|
|
|
|
if (in == OPEN_FAILED)
|
|
{
|
|
// try the language pack's folder itself before falling back to the resource path
|
|
file.swap(StringPathCombine(LocalesManager::getLocaleRoot(), _filename));
|
|
in = WFOPEN(file, WF_READONLY_BINARY);
|
|
}
|
|
|
|
if (in == OPEN_FAILED)
|
|
{
|
|
if (path)
|
|
{
|
|
file.swap(StringPathCombine(path, _filename));
|
|
in = WFOPEN(file, WF_READONLY_BINARY);
|
|
}
|
|
else
|
|
in = WFOPEN(file=_filename, WF_READONLY_BINARY);
|
|
}
|
|
|
|
#ifdef WASABI_COMPILE_SKIN
|
|
if (in == OPEN_FAILED)
|
|
{
|
|
file.swap(StringPathCombine(WASABI_API_SKIN->getSkinPath(), _filename));
|
|
in = WFOPEN(file, WF_READONLY_BINARY);
|
|
}
|
|
|
|
#if 0 // this isn't used in gen_ff, basically makes it look in C:/Program Files/Winamp/Skins/Default/
|
|
if (in == OPEN_FAILED)
|
|
{
|
|
file.swap(StringPathCombine(Skin::getDefaultSkinPath(), _filename));
|
|
in = WFOPEN(file, WF_READONLY_BINARY);
|
|
}
|
|
#endif
|
|
|
|
// look in the fallback stuff (in Winamp5, this is c:/program files/winamp/plugins/freeform/xml)
|
|
if (in == OPEN_FAILED)
|
|
{
|
|
file.swap(StringPathCombine(g_resourcepath, _filename));
|
|
in = WFOPEN(file, WF_READONLY_BINARY);
|
|
}
|
|
#endif
|
|
|
|
if (in == OPEN_FAILED && path)
|
|
{
|
|
in = WFOPEN(file = _filename, WF_READONLY_BINARY);
|
|
}
|
|
|
|
Wasabi::Std::setCurDir(olddir);
|
|
|
|
if (in != OPEN_FAILED)
|
|
{
|
|
int filelen = (int)FGETSIZE(in);
|
|
if (filelen > 0)
|
|
{
|
|
mem.setSize(filelen);
|
|
int len = FREAD(mem, 1, mem.getSizeInBytes(), in);
|
|
if (len == filelen)
|
|
{
|
|
svc_imageLoader *loader = ImgLoaderEnum(mem, len).getNext();
|
|
if (loader != NULL)
|
|
{
|
|
ret = loader->loadImage(mem, mem.getSizeInBytes(), w, h, params);
|
|
if (ret != NULL)
|
|
{
|
|
if (addMem)
|
|
addMemUsage(file, (*w) * (*h) * sizeof(ARGB32));
|
|
optimizeHasAlpha(ret, *w * *h, has_alpha);
|
|
}
|
|
SvcEnum::release(loader);
|
|
}
|
|
}
|
|
} // filelen > 0
|
|
FCLOSE(in);
|
|
} // if file opened
|
|
if (ret != NULL)
|
|
{
|
|
return ret;
|
|
}
|
|
else if (in != OPEN_FAILED && mem)
|
|
{
|
|
int m = getFileType(mem);
|
|
#ifdef WIN32
|
|
switch (m)
|
|
{
|
|
case FT_BMP:
|
|
{
|
|
wchar_t tmpname[WA_MAX_PATH] = L"";
|
|
|
|
// FG> if loading bmp from disk, no need to do the copy to disk
|
|
HBITMAP hbmp = (HBITMAP)LoadImageW(0, file, IMAGE_BITMAP, 0, 0, LR_LOADFROMFILE | LR_CREATEDIBSECTION);
|
|
|
|
if (!hbmp)
|
|
{
|
|
// CT> extract/copy the file into temp directory (so we don't have any trouble if the file
|
|
// is in a ZIP file). this whole copying thing will go away as soon as we'll get rid of
|
|
// the LoadImage win32 function and use our own bmp loading functions
|
|
|
|
GetTempPathW(WA_MAX_PATH, tmpname);
|
|
wcscat(tmpname, L"wa3tmp");
|
|
OSFILETYPE fs = WFOPEN(file, WF_READONLY_BINARY);
|
|
if (fs != OPEN_FAILED)
|
|
{
|
|
OSFILETYPE fd = WFOPEN(tmpname, L"wb");
|
|
int l;
|
|
do
|
|
{
|
|
char buf[1024] = {0};
|
|
l = FREAD(buf, 1, sizeof(buf), fs);
|
|
if (l > 0) FWRITE(buf, 1, l, fd);
|
|
}
|
|
while (l > 0);
|
|
FCLOSE(fs);
|
|
FCLOSE(fd);
|
|
hbmp = (HBITMAP)LoadImageW(0, tmpname, IMAGE_BITMAP, 0, 0, LR_LOADFROMFILE | LR_CREATEDIBSECTION);
|
|
}
|
|
if (!hbmp)
|
|
{
|
|
#ifdef WASABI_COMPILE_SKIN_WA2
|
|
// bitmap not found or broken (like in the netscape skin)
|
|
// try to find it in the Classic skin (wa2)
|
|
StringW wa2skinFn = WASABI_API_APP->getSkinsPath();
|
|
wa2skinFn.AppendPath("Classic");
|
|
wa2skinFn.AppendPath(_filename);
|
|
fs = WFOPEN(wa2skinFn), WF_READONLY_BINARY);
|
|
if (fs != OPEN_FAILED)
|
|
{
|
|
OSFILETYPE fd = WFOPEN(tmpname, L"wb");
|
|
int l;
|
|
do
|
|
{
|
|
char buf[1024] = {0};
|
|
l = FREAD(buf, 1, sizeof(buf), fs);
|
|
if (l > 0) FWRITE(buf, 1, l, fd);
|
|
}
|
|
while (l > 0);
|
|
FCLOSE(fs);
|
|
FCLOSE(fd);
|
|
hbmp = (HBITMAP)LoadImageW(0, tmpname, IMAGE_BITMAP, 0, 0, LR_LOADFROMFILE | LR_CREATEDIBSECTION);
|
|
}
|
|
#endif //WASABI_COMPILE_SKIN_WA2
|
|
}
|
|
if (!hbmp)
|
|
{
|
|
// no luck :(
|
|
_wunlink(tmpname);
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
BITMAP bm;
|
|
HDC hMemDC, hMemDC2;
|
|
HBITMAP hprev, hprev2;
|
|
HBITMAP hsrcdib;
|
|
void *srcdib;
|
|
BITMAPINFO srcbmi = {0, };
|
|
int r = GetObject(hbmp, sizeof(BITMAP), &bm);
|
|
ASSERT(r != 0);
|
|
*w = bm.bmWidth;
|
|
*h = ABS(bm.bmHeight);
|
|
|
|
ARGB32 *newbits;
|
|
srcbmi.bmiHeader.biSize = sizeof(srcbmi.bmiHeader);
|
|
srcbmi.bmiHeader.biWidth = *w;
|
|
srcbmi.bmiHeader.biHeight = -*h;
|
|
srcbmi.bmiHeader.biPlanes = 1;
|
|
srcbmi.bmiHeader.biBitCount = 32;
|
|
srcbmi.bmiHeader.biCompression = BI_RGB;
|
|
|
|
hMemDC = CreateCompatibleDC(NULL);
|
|
hMemDC2 = CreateCompatibleDC(NULL);
|
|
hsrcdib = CreateDIBSection(hMemDC, &srcbmi, DIB_RGB_COLORS, &srcdib, NULL, 0);
|
|
ASSERTPR(hsrcdib != 0, "CreateDIBSection() failed #69");
|
|
hprev2 = (HBITMAP) SelectObject(hMemDC2, hbmp);
|
|
hprev = (HBITMAP) SelectObject(hMemDC, hsrcdib);
|
|
BitBlt(hMemDC, 0, 0, *w, *h, hMemDC2, 0, 0, SRCCOPY);
|
|
newbits = (ARGB32*)MALLOC((*w) * (*h) * sizeof(ARGB32));
|
|
MEMCPY32(newbits, srcdib, (*w)*(*h) /**sizeof(ARGB32)*/);
|
|
{
|
|
// put the alpha channel to 255
|
|
unsigned char *b = (unsigned char *)newbits;
|
|
int l = (*w) * (*h);
|
|
for (int i = 0;i < l;i++)
|
|
b[(i*4) + 3] = 0xff;
|
|
}
|
|
SelectObject(hMemDC, hprev);
|
|
SelectObject(hMemDC2, hprev2);
|
|
DeleteObject(hsrcdib);
|
|
DeleteDC(hMemDC2);
|
|
DeleteDC(hMemDC);
|
|
|
|
DeleteObject(hbmp);
|
|
|
|
if (tmpname[0])
|
|
_wunlink(tmpname); // destroy temp extraction
|
|
|
|
if (addMem)
|
|
addMemUsage(file, (*w)*(*h)*4);
|
|
return newbits;
|
|
}
|
|
}
|
|
#endif
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
int imageLoader::getFileType(unsigned char *pData)
|
|
{
|
|
// Bmp ?
|
|
#ifdef WIN32
|
|
WINBITMAPFILEHEADER * pBFH;
|
|
pBFH = (WINBITMAPFILEHEADER *) pData;
|
|
#ifdef _WINDOWS
|
|
if (pBFH->bfType == 0x4d42)
|
|
#else
|
|
if (pBFH->bfType == 0x424d)
|
|
#endif
|
|
return FT_BMP;
|
|
#endif
|
|
return FT_UNKNOWN;
|
|
}
|
|
|
|
#ifdef WASABI_COMPILE_SKIN
|
|
|
|
ARGB32 *imageLoader::requestSkinBitmap(const wchar_t *id, int *has_alpha, int *x, int *y, int *subw, int *subh, int *w, int *h, int cached)
|
|
{
|
|
ifc_xmlreaderparams *params = NULL;
|
|
const wchar_t *rootpath = NULL;
|
|
|
|
const wchar_t *aliastarget = WASABI_API_PALETTE->getElementAlias(id);
|
|
|
|
if (aliastarget)
|
|
id = aliastarget;
|
|
|
|
const wchar_t *efile = WASABI_API_PALETTE->getSkinBitmapFilename(id, x, y, subw, subh, &rootpath, ¶ms);
|
|
|
|
if (x && *x == -1) *x = 0;
|
|
if (y && *y == -1) *y = 0;
|
|
|
|
if (!efile)
|
|
efile = id;
|
|
|
|
if (cached)
|
|
{
|
|
StringPathCombine f(rootpath, efile);
|
|
f.toupper();
|
|
f.FixSlashes();
|
|
|
|
int pos = -1;
|
|
/*skinCacheEntry *entry = */skinCacheList.findItem(f.getValue(), &pos);
|
|
if (pos != -1)
|
|
{
|
|
//find first one
|
|
while (pos > 0 && !wcscmp(skinCacheList[pos - 1]->fullpathfilename, f)) pos--;
|
|
do
|
|
{
|
|
skinCacheEntry *entry = skinCacheList[pos];
|
|
if (GammaMgr::gammaEqual(entry->original_element_id, id) && layerEqual(entry->original_element_id, id))
|
|
{
|
|
entry->usageCount++;
|
|
if (w) *w = entry->width;
|
|
if (h) *h = entry->height;
|
|
if (has_alpha) *has_alpha = entry->has_alpha;
|
|
return entry->bitmapbits;
|
|
}
|
|
pos++;
|
|
if (pos >= skinCacheList.getNumItems()) break;
|
|
}
|
|
while (!wcscmp(skinCacheList[pos]->fullpathfilename, f));
|
|
}
|
|
}
|
|
|
|
int force_nocache = 0;
|
|
int t_has_alpha = 0;
|
|
ARGB32 *bits = makeBmp(efile, rootpath, &t_has_alpha, w, h, params, TRUE, &force_nocache);
|
|
if (has_alpha != NULL) *has_alpha = t_has_alpha;
|
|
|
|
if (!bits)
|
|
return NULL;
|
|
|
|
if (force_nocache || !cached) return bits;
|
|
|
|
skinCacheEntry *cachedbmp = new skinCacheEntry;
|
|
|
|
if (params)
|
|
{
|
|
for (size_t i = 0;i != params->getNbItems();i++)
|
|
cachedbmp->params.addItem(params->getItemName(i), params->getItemValue(i));
|
|
}
|
|
|
|
cachedbmp->usageCount = 1;
|
|
cachedbmp->bitmapbits = bits;
|
|
cachedbmp->filename = efile;
|
|
cachedbmp->has_alpha = !!t_has_alpha;
|
|
cachedbmp->width = *w;
|
|
cachedbmp->height = *h;
|
|
cachedbmp->original_element_id = id;
|
|
|
|
//needed for findItem above
|
|
StringPathCombine b(rootpath, efile);
|
|
b.toupper();
|
|
b.FixSlashes();
|
|
cachedbmp->fullpathfilename.swap(b);
|
|
|
|
applySkinFilters(cachedbmp);
|
|
skinCacheList.addItem(cachedbmp);
|
|
|
|
return cachedbmp->bitmapbits;
|
|
}
|
|
/*
|
|
int imageLoader::paramsMatch(ifc_xmlreaderparams *a, ifc_xmlreaderparams *b)
|
|
{
|
|
if (!a && !b) return 1;
|
|
if ((!a && b) || (!b && a)) return 0;
|
|
for (int i = 0;i < a->getNbItems();i++)
|
|
{
|
|
const wchar_t *name = a->getItemName(i);
|
|
if (!_wcsicmp(name, L"w") || !_wcsicmp(name, L"h") || !_wcsicmp(name, L"x") || !_wcsicmp(name, L"y")) continue;
|
|
if (_wcsicmp(a->getItemValue(i), b->getItemValue(name)))
|
|
return 0;
|
|
}
|
|
return 1;
|
|
}
|
|
*/
|
|
int imageLoader::layerEqual(const wchar_t *id1, const wchar_t *id2)
|
|
{
|
|
int a = WASABI_API_PALETTE->getLayerFromId(id1);
|
|
int b = WASABI_API_PALETTE->getLayerFromId(id2);
|
|
return (a == b);
|
|
}
|
|
|
|
void imageLoader::releaseSkinBitmap(ARGB32 *bitmapbits)
|
|
{ //FG
|
|
int i;
|
|
// TODO: add critical sections
|
|
|
|
int ni = skinCacheList.getNumItems();
|
|
for (i = 0;i < ni;i++)
|
|
{
|
|
skinCacheEntry *entry = skinCacheList.enumItem(i);
|
|
if (entry->bitmapbits == bitmapbits)
|
|
{
|
|
entry->usageCount--;
|
|
if (entry->usageCount == 0)
|
|
{
|
|
subMemUsage(entry->width*entry->height*sizeof(int));
|
|
WASABI_API_MEMMGR->sysFree(entry->bitmapbits);
|
|
skinCacheList.removeByPos(i);
|
|
delete entry;
|
|
if (skinCacheList.getNumItems() == 0) skinCacheList.removeAll();
|
|
}
|
|
return ;
|
|
}
|
|
}
|
|
// bitmap was not a cached skin bitmap, simply free it
|
|
release(bitmapbits);
|
|
}
|
|
|
|
#endif //WASABI_COMPILE_SKIN
|
|
|
|
void imageLoader::release(ARGB32 *bitmapbits)
|
|
{
|
|
WASABI_API_MEMMGR->sysFree(bitmapbits);
|
|
}
|
|
|
|
void imageLoader::optimizeHasAlpha(ARGB32 *bits, int len, int *has_alpha)
|
|
{
|
|
if (len <= 0 || has_alpha == NULL) return ;
|
|
for (*has_alpha = 0; len; len--, bits++)
|
|
{
|
|
ARGB32 v = *bits;
|
|
unsigned int alpha = v >> 24;
|
|
if (alpha != 255)
|
|
{
|
|
*has_alpha = 1;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
#ifdef WASABI_COMPILE_SKIN
|
|
|
|
void imageLoader::applySkinFilters()
|
|
{ //FG
|
|
int i;
|
|
|
|
Skin::unloadResources();
|
|
for (i = 0; i < skinCacheList.getNumItems(); i++)
|
|
{
|
|
skinCacheEntry *entry = skinCacheList.q(i);
|
|
applySkinFilters(entry);
|
|
}
|
|
|
|
WASABI_API_PALETTE->newSkinPart();
|
|
Skin::reloadResources();
|
|
}
|
|
|
|
void imageLoader::applySkinFilters(skinCacheEntry *entry)
|
|
{
|
|
ASSERT(entry != NULL);
|
|
|
|
int w = entry->width, h = entry->height;
|
|
|
|
ApplySkinFilters::apply(entry->original_element_id, NULL, (ARGB32*)entry->bitmapbits, w, h);
|
|
}
|
|
|
|
ARGB32 imageLoader::filterSkinColor(ARGB32 color, const wchar_t *element_id, const wchar_t *groupname)
|
|
{
|
|
SkinFilterEnum sfe;
|
|
|
|
svc_skinFilter *obj;
|
|
while (1)
|
|
{
|
|
obj = sfe.getNext(FALSE);
|
|
if (!obj) break;
|
|
color = obj->filterColor(color, element_id, groupname);
|
|
sfe.getLastFactory()->releaseInterface(obj);
|
|
}
|
|
|
|
return color;
|
|
}
|
|
|
|
#endif //WASABI_COMPILE_SKIN
|
|
|
|
void imageLoader::addMemUsage(const wchar_t *filename, int size)
|
|
{
|
|
totalMemUsage += size;
|
|
#ifdef DEBUG_OUTPUT
|
|
DebugStringW("Bitmaps memory usage : %s - %d\n", filename, totalMemUsage);
|
|
#endif
|
|
}
|
|
|
|
void imageLoader::subMemUsage(int size)
|
|
{
|
|
totalMemUsage -= size;
|
|
}
|
|
|
|
#ifdef WASABI_COMPILE_SKIN
|
|
|
|
//PtrList<skinCacheEntry> imageLoader::skinCacheList;
|
|
PtrListInsertMultiSorted<skinCacheEntry, skinCacheComp> imageLoader::skinCacheList;
|
|
#endif //WASABI_COMPILE_SKIN
|
|
|
|
int imageLoader::totalMemUsage = 0; |