我一直试图让它工作一段时间,但我似乎无法弄清楚,而且谷歌搜索数小时还没有显示任何有用的结果。
我有一个 RGBA 顺序的 32 位像素数组,我想从它们创建一个与设备无关的位图,并使用SetClipboardData(CF_DIBV5, dib)
或类似方法将其放置在剪贴板上(理想情况下,我想保留 alpha 通道)。注册自定义剪贴板类型不是一种选择,因为将其放在剪贴板上的目的是可以将其粘贴到另一个程序中。如果我不必手动将像素数据转换为其他格式(例如平面 BGRA),则可以加分。
我当前的代码是这样的(都在 set_clipboard_img 函数中):
if(!OpenClipboard(hwnd)) return;
BITMAPV5HEADER* info = (BITMAPV5HEADER*) GlobalAlloc(GMEM_MOVEABLE, sizeof(BITMAPV5HEADER));
info->bV5Size = sizeof(BITMAPV5HEADER);
info->bV5Width = img_width;
info->bV5Height = -img_height;
info->bV5Planes = 1; // The docs say this is the only valid value here.
info->bV5BitCount = 32;
info->bV5Compression = BI_BITFIELDS;
info->bV5SizeImage = img_width * img_height * 4;
info->bV5RedMask = 0xff000000;
info->bV5GreenMask = 0x00ff0000;
info->bV5BlueMask = 0x0000ff00;
info->bV5AlphaMask = 0x000000ff;
unsigned char* buf;
// One of the sources I found said that I can pass a BITMAPV5HEADER in place of the BITMAPINFO, hence the first reinterpret_cast.
HBITMAP dib = CreateDIBSection(NULL, reinterpret_cast<BITMAPINFO*>(info), DIB_RGB_COLORS, reinterpret_cast<void**>(&buf), NULL, 0);
if(dib == NULL) {
CloseClipboard();
return;
}
// img_pixels_ptr is a unsigned char* to the pixels in non-planar RGBA format
std::copy_n(img_pixels_ptr, info->bV5SizeImage, buf);
EmptyClipboard();
auto result = SetClipboardData(CF_DIBV5, dib);
if(result == NULL) {
char str[256];
str[255] = 0;
FormatMessageA(FORMAT_MESSAGE_FROM_SYSTEM, NULL, GetLastError(), 0, str, 255, NULL);
std::cerr << "Error setting clipboard: " << str << std::endl;
// Here I get "handle not valid". I have no idea _why_ it's not valid, though.
}
CloseClipboard();
最终,我还需要能够扭转这个过程(从剪贴板上获取一个可能透明的位图),但一次只做一件事。