2009-11-30 153 views
2

我正在使用IShellItemImageFactory來提取文件的圖標。我能夠成功地提取它並使用 SendDlgItemMessage(hDlg,IDC_STATIC2,STM_SETIMAGE,IMAGE_ICON,(LPARAM)hicon)在對話框中顯示它。將HICON保存爲PNG

看到輸出:click here

的問題是,當我使用GDI保存此作爲一個文件(PNG格式)+梯度不保留糾正。找到我正在使用的代碼。

 
GdiplusStartupInput gdiplusStartupInput; 
ULONG_PTR gdiplusToken; 
GdiplusStartup(&gdiplusToken, &gdiplusStartupInput, NULL); 
Bitmap *h = new Bitmap(256, 256, PixelFormat32bppARGB); 
Graphics* g = Graphics::FromImage(copy); 
HDC copyHdc = g->GetHDC(); 
DrawIconEx(copyHdc, 0, 0, hicon, 256, 256, 0, NULL, DI_NORMAL); 
g->ReleaseHDC(copyHdc);; 
CLSID encoderClsid; 
GetEncoderClsid(L"image/png", &encoderClsid); 
h->Save(L"D:\\mynew.png", &encoderClsid, NULL); 
GdiplusShutdown(gdiplusToken); 

,我的文本文件的提取後得到的輸出:click here

誰能幫我解決這個?

問候, 馬諾

+0

無法看到在鏈接的圖像的任何差異。 – 2009-11-30 10:30:12

+0

看看兩幅圖像的漸變,第二幅獲得了一些黑色。 – Manoj 2009-11-30 10:38:19

回答

2

我錯過了從代碼中「複製」:

Graphics* g = Graphics::FromImage(copy); 

望着,似乎在設備上下文你繪製圖標,沒有按」圖像沒有32位顏色(缺少Alpha通道)。

嘗試創建DC這樣的:

HDC hDC = CreateCompatibleDC(NULL); 

,然後選擇一個32位的彩色(空)的位圖成直流。之後,您可以繪製圖標並保存。

+0

是的,它應該是Graphics * g = Graphics :: FromImage(h); – Manoj 2009-12-01 04:32:48

+0

好的,謝謝。我實現了這樣的示例程序。 – Manoj 2009-12-01 06:40:49

6

本主題相當老,但我只是有同樣的問題,花了很多小時找到解決方案,保留在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(...); 
+0

不錯,謝謝。 – noober 2014-10-06 11:40:37