Я узнал, что кодер IImagingFactory не работает, потому что мой CBitmap - это DDB, а не DIB. Я использовал CreateDIBSection для создания растрового изображения вместо CBitmap.Create () и смог кодировать CBitmap в PNG с помощью IImagingFactory.
Надеюсь, что это поможет следующему человеку, который пойдет по этому пути (проверено только с 1BPP):
HRESULT CSignatureControl::Imaging_GetCLSID(IImagingFactory *pFactory, LPCTSTR mimeType, CLSID *clsid)
{
if (pFactory == NULL)
return E_INVALIDARG;
if (mimeType == NULL || mimeType[0] == 0)
return E_INVALIDARG;
if (clsid == NULL)
return E_INVALIDARG;
UINT count = 0;
ImageCodecInfo *pCodecInfo = NULL;
HRESULT hr = pFactory->GetInstalledEncoders(&count, &pCodecInfo);
if (SUCCEEDED(hr))
{
for (UINT i = 0; i < count; i++)
{
if (!_tcscmp(pCodecInfo[i].MimeType, mimeType))
{
*clsid = pCodecInfo[i].Clsid;
break;
}
}
CoTaskMemFree(pCodecInfo);
}
return hr;
}
HRESULT CSignatureControl::Imaging_IBitmapImageFromHBITMAP(IImagingFactory *pFactory, BitmapData *bmData, HBITMAP hBitmap, IBitmapImage **pBitmapImage)
{
if (pFactory == NULL)
return E_INVALIDARG;
if (hBitmap == NULL)
return E_INVALIDARG;
BITMAP bm;
GetObject(hBitmap, sizeof(BITMAP), &bm);
// Map between different pixel formats
PixelFormatID pixelFormat;
if (bm.bmBitsPixel == 1)
pixelFormat = PixelFormat1bppIndexed;
else if (bm.bmBitsPixel == 4)
pixelFormat = PixelFormat4bppIndexed;
else if (bm.bmBitsPixel == 8)
pixelFormat = PixelFormat8bppIndexed;
else if (bm.bmBitsPixel == 24)
pixelFormat = PixelFormat24bppRGB;
bmData->Height = bm.bmHeight;
bmData->Width = bm.bmWidth;
bmData->Scan0 = bm.bmBits;
bmData->PixelFormat = pixelFormat;
bmData->Stride = ((bm.bmWidth * bm.bmBitsPixel + 31) & ~31) >> 3;
return pFactory->CreateBitmapFromBuffer(bmData, pBitmapImage);
}
HRESULT CSignatureControl::Imaging_EncodeBitmapToFile(IImagingFactory *pFactory, IBitmapImage *pBitmap, CLSID *clsid, LPCTSTR filename)
{
if (pFactory == NULL)
return E_INVALIDARG;
CComPtr<IImageEncoder> imageEncoder;
HRESULT hr = pFactory->CreateImageEncoderToFile(clsid, filename, &imageEncoder);
if (SUCCEEDED(hr))
{
CComPtr<IImageSink> imageSink;
hr = imageEncoder->GetEncodeSink(&imageSink);
if (SUCCEEDED(hr))
{
CComPtr<IImage> pImage;
hr = pBitmap->QueryInterface(IID_IImage, (void**)&pImage);
if (SUCCEEDED(hr))
{
hr = pImage->PushIntoSink(imageSink);
}
}
hr = imageEncoder->TerminateEncoder();
}
return hr;
}
Связывая все вместе:
CoInitializeEx(NULL, COINIT_MULTITHREADED);
{
CComPtr<IImagingFactory> pImgFactory;
HRESULT hr = pImgFactory.CoCreateInstance(CLSID_ImagingFactory);
if (SUCCEEDED(hr))
{
CLSID pngClsid = {0};
hr = Imaging_GetCLSID(pImgFactory.p, TEXT("image/png"), &pngClsid);
if (SUCCEEDED(hr))
{
CComPtr<IBitmapImage> pBitmapImg;
BitmapData *bmData = new BitmapData();
hr = Imaging_IBitmapImageFromHBITMAP(pImgFactory.p, bmData, m_offscreenBitmap, &pBitmapImg);
if (SUCCEEDED(hr))
{
hr = Imaging_EncodeBitmapToFile(pImgFactory.p, pBitmapImg, &pngClsid, fileName);
}
delete bmData;
}
}
}
CoUninitialize();