本主題相當老,但我只是有同樣的問題,花了很多小時找到解決方案,保留在PNG文件中的透明度。
既然問題可以在Java迎刃而解......
sun.awt.shell.ShellFolder sf = sun.awt.shell.ShellFolder.getShellFolder(file);
ImageIcon icon = new ImageIcon(sf.getIcon(true));
FileOutputStream bos = new FileOutputStream("d:\\icons\\icon.png");
ImageIO.write((BufferedImage)icon.getImage(), "PNG", bos);
......我看了看一個JDK的源代碼。在函數「Java_sun_awt_shell_Win32ShellFolder2_getIconBits」的文件「\ jdk \ src \ windows \ native \ sun \ windows \ ShellFolder2.cpp」中,我發現了我需要的有價值的提示。
該函數從HICON中檢索顏色位圖並調用GetDIBits來獲取圖像數據。繪製圖標並不是必需的 - 透明性將會丟失。
非常感謝JDK開發人員。
這是我最終的代碼:
static CLSID g_pngClsid = GUID_NULL;
// http://msdn.microsoft.com/en-us/library/windows/desktop/ms533843(v=vs.85).aspx
extern int GetEncoderClsid(const WCHAR* format, CLSID* pClsid);
static HICON getShellIconByIndex(int shilsize, int iImage)
{
IImageListPtr spiml;
SHGetImageList(shilsize, IID_PPV_ARGS(&spiml));
HICON hico;
spiml->GetIcon(iImage, ILD_TRANSPARENT, &hico);
return hico;
}
static HICON getShellIcon(int shilsize, const std::wstring& fname) {
UINT flags = SHGFI_SYSICONINDEX;
SHFILEINFO fi = {0};
HICON hIcon = NULL;
if (SHGetFileInfo(fname.c_str(), 0, &fi, sizeof(fi), flags) != 0) {
hIcon = getShellIconByIndex(shilsize, fi.iIcon);
}
return hIcon;
}
struct BITMAP_AND_BYTES {
Gdiplus::Bitmap* bmp;
int32_t* bytes;
};
static BITMAP_AND_BYTES createAlphaChannelBitmapFromIcon(HICON hIcon) {
// Get the icon info
ICONINFO iconInfo = {0};
GetIconInfo(hIcon, &iconInfo);
// Get the screen DC
HDC dc = GetDC(NULL);
// Get icon size info
BITMAP bm = {0};
GetObject(iconInfo.hbmColor, sizeof(BITMAP), &bm);
// Set up BITMAPINFO
BITMAPINFO bmi = {0};
bmi.bmiHeader.biSize = sizeof(BITMAPINFOHEADER);
bmi.bmiHeader.biWidth = bm.bmWidth;
bmi.bmiHeader.biHeight = -bm.bmHeight;
bmi.bmiHeader.biPlanes = 1;
bmi.bmiHeader.biBitCount = 32;
bmi.bmiHeader.biCompression = BI_RGB;
// Extract the color bitmap
int nBits = bm.bmWidth * bm.bmHeight;
int32_t* colorBits = new int32_t[nBits];
GetDIBits(dc, iconInfo.hbmColor, 0, bm.bmHeight, colorBits, &bmi, DIB_RGB_COLORS);
// Check whether the color bitmap has an alpha channel.
// (On my Windows 7, all file icons I tried have an alpha channel.)
BOOL hasAlpha = FALSE;
for (int i = 0; i < nBits; i++) {
if ((colorBits[i] & 0xff000000) != 0) {
hasAlpha = TRUE;
break;
}
}
// If no alpha values available, apply the mask bitmap
if (!hasAlpha) {
// Extract the mask bitmap
int32_t* maskBits = new int32_t[nBits];
GetDIBits(dc, iconInfo.hbmMask, 0, bm.bmHeight, maskBits, &bmi, DIB_RGB_COLORS);
// Copy the mask alphas into the color bits
for (int i = 0; i < nBits; i++) {
if (maskBits[i] == 0) {
colorBits[i] |= 0xff000000;
}
}
delete[] maskBits;
}
// Release DC and GDI bitmaps
ReleaseDC(NULL, dc);
::DeleteObject(iconInfo.hbmColor);
::DeleteObject(iconInfo.hbmMask);
// Create GDI+ Bitmap
Gdiplus::Bitmap* bmp = new Gdiplus::Bitmap(bm.bmWidth, bm.bmHeight, bm.bmWidth*4, PixelFormat32bppARGB, (BYTE*)colorBits);
BITMAP_AND_BYTES ret = {bmp, colorBits};
return ret;
}
static void saveFileIconAsPng(int shilsize, const std::wstring& fname, const std::wstring& pngFile) {
HICON hIcon = getShellIcon(shilsize, fname);
BITMAP_AND_BYTES bbs = createAlphaChannelBitmapFromIcon(hIcon);
IStream* fstrm = NULL;
SHCreateStreamOnFile(pngFile.c_str(), STGM_WRITE|STGM_CREATE, &fstrm);
bbs.bmp->Save(fstrm, &g_pngClsid, NULL);
fstrm->Release();
delete bbs.bmp;
delete[] bbs.bytes;
DestroyIcon(hIcon);
}
例invokation:
GdiplusStartup(...);
GetEncoderClsid(L"image/png", &g_pngClsid);
wstring fname = L"d:\\index.html";
wstring pngFile = L"d:\\icons\\index.html.png";
saveFileIconAsPng(SHIL_JUMBO, fname, pngFile);
GdiplusShutdown(...);
無法看到在鏈接的圖像的任何差異。 – 2009-11-30 10:30:12
看看兩幅圖像的漸變,第二幅獲得了一些黑色。 – Manoj 2009-11-30 10:38:19