我确信有一种将 alpha 存储在 CF_DIBV5 中的正确方法,但这并不重要。应用程序已经不一致地处理它,所以如果您希望您的应用程序与其他应用程序很好地配合使用,您不能使用 CF_DIBV5。
不久前我研究了复制和粘贴透明位图。我的目标是成功地将透明位图粘贴到两个版本的 Office 和 GIMP 中。我查看了几种可能的格式:
CF_BITMAP
透明度总是被忽略。
CF_DIB
使用通常的 0xAARRGGBB 格式的 32bpp BI_RGB。GIMP 支持这一点,但没有其他支持。
CF_DIBV5
GIMP 不支持这个。
“PNG”
支持粘贴:GIMP、Word 2000、Excel 2000、Excel 2007 和 PowerPoint 2007。
不支持粘贴:Word 2007 和 OneNote 2007。
如果您复制位图,所有这些应用程序都会成功导出“PNG”。
但是,Word 和 OneNote 2007将粘贴从资源管理器复制的 PNG 文件。所以我想出了以下内容:
复制解决方案
将您的透明位图转换为 PNG 格式。
宣传以下剪贴板格式:
“PNG” - 原始 PNG 数据。
CF_DIB - 用于不处理透明度的应用程序(如油漆)。
CFSTR_FILEDESCRIPTOR - 使 PNG 看起来像一个文件。文件描述符应该有一个带有“.png”扩展名的发明文件名。
CFSTR_FILECONTENTS - 内容必须公开为IStream
; 仅使用 anHGLOBAL
似乎不起作用。该数据与“PNG”数据相同。
完成此操作后,我可以成功地将透明位图粘贴到 GIMP、Office 2000 和 Office 2007 中。您还可以将 PNG 直接粘贴到资源管理器文件夹中。
更新
我意识到我只回答了一半的问题。这对于复制非常有用,但如果您想从仅复制 CF_DIBV5(如 Firefox)的应用程序粘贴,则没有用。
如果可用,我建议您使用“PNG”,否则回退到 CF_DIBV5,将其视为预乘。这将正确处理 Word 2010(导出“PNG”)、Firefox 和 Chrome。XnView 只导出非乘法的 CF_DIBV5,所以这不能正常工作。我不确定你能不能做得更好。
lscf - 探索剪贴板格式的工具
这是用于显示可用剪贴板格式列表的工具的来源。它也可以将一个写入文件。我叫它lscf
。在 Visual Studio 中创建一个 win32 控制台应用程序并将此源代码粘贴到 main 函数上。它有一个非常小的错误:如果您输入错误的格式名称,它永远不会显示“未知格式”错误。
#include <Windows.h>
#include <stdio.h>
#include <tchar.h>
#define ARRAY_SIZE(a) (sizeof(a) / sizeof(a[0]))
LPCTSTR cfNames[] = {
_T("CF_TEXT"),
_T("CF_BITMAP"),
_T("CF_METAFILEPICT"),
_T("CF_SYLK"),
_T("CF_DIF"),
_T("CF_TIFF"),
_T("CF_OEMTEXT"),
_T("CF_DIB"),
_T("CF_PALETTE"),
_T("CF_PENDATA"),
_T("CF_RIFF"),
_T("CF_WAVE"),
_T("CF_UNICODETEXT"),
_T("CF_ENHMETAFILE"),
_T("CF_HDROP"),
_T("CF_LOCALE"),
_T("CF_DIBV5")
};
int LookupFormat(LPCTSTR name)
{
for (int i = 0; i != ARRAY_SIZE(cfNames); ++i)
{
if (_tcscmp(cfNames[i], name) == 0)
return i + 1;
}
return RegisterClipboardFormat(name);
}
void PrintFormatName(int format)
{
if (!format)
return;
if ((format > 0) && (format <= ARRAY_SIZE(cfNames)))
{
_tprintf(_T("%s\n"), cfNames[format - 1]);
}
else
{
TCHAR buffer[100];
if (GetClipboardFormatName(format, buffer, ARRAY_SIZE(buffer)))
_tprintf(_T("%s\n"), buffer);
else
_tprintf(_T("#%i\n"), format);
}
}
void WriteFormats()
{
int count = 0;
int format = 0;
do
{
format = EnumClipboardFormats(format);
if (format)
{
++count;
PrintFormatName(format);
}
}
while (format != 0);
if (!count)
_tprintf(_T("Clipboard is empty!\n"));
}
void SaveFormat(int format, LPCTSTR filename)
{
HGLOBAL hData = (HGLOBAL)GetClipboardData(format);
LPVOID data = GlobalLock(hData);
HANDLE hFile = CreateFile(filename, GENERIC_WRITE, 0, 0, CREATE_ALWAYS, 0, 0);
if (hFile != INVALID_HANDLE_VALUE)
{
DWORD bytesWritten;
WriteFile(hFile, data, GlobalSize(hData), &bytesWritten, 0);
CloseHandle(hFile);
}
GlobalUnlock(hData);
}
int _tmain(int argc, _TCHAR* argv[])
{
if (!OpenClipboard(0))
{
_tprintf(_T("Cannot open clipboard\n"));
return 1;
}
if (argc == 1)
{
WriteFormats();
}
else if (argc == 3)
{
int format = LookupFormat(argv[1]);
if (format == 0)
{
_tprintf(_T("Unknown format\n"));
return 1;
}
SaveFormat(format, argv[2]);
}
else
{
_tprintf(_T("lscf\n"));
_tprintf(_T("List available clipboard formats\n\n"));
_tprintf(_T("lscf CF_NAME filename\n"));
_tprintf(_T("Write format CF_NAME to file filename\n\n"));
}
CloseClipboard();
return 0;
}