c - Win32 剪贴板和 alpha channel 图像

标签 c windows winapi 2d

我的应用程序应该能够将 32 位图像(RGB + alpha channel )复制到剪贴板并从剪贴板粘贴这些图像。为此,我计划使用 CF_DIBV5,因为 BITMAPV5HEADER 结构有一个字段 bV5AlphaMask

问题是对于图像数据应该如何准确地存储在剪贴板中似乎没有达成共识。在进行一些测试时,我发现应用程序之间存在一些差异,因此几乎不可能得出一个通用的解决方案。

这是我的观察:

  1. 当我将 alpha channel 图像从 Word 2010 或 XnView 复制到剪贴板时,它的存储没有预乘像素数据。

  2. 但是,当我使用 Firefox 或 Chrome 复制图像时,像素数据似乎预乘了 alpha channel 。

  3. Firefox 将 bV5AlphaMask 设置为 0xff000000 而大多数其他应用程序根本不设置它而是将其保持为 0。这很奇怪,因为这些应用程序将 DIB 放到剪贴板上,而实际上包含一个 alpha channel 在最高 8 位中,但它们仍然将 bV5AlphaMask 设置为 0。因此,必须假设如果位深度为 32,则即使 bV5AlphaMask 为0.

长话短说,我的基本问题是:是否有关于应如何将 alpha channel 数据存储在剪贴板上的官方信息?我特别想知道数据是否必须预乘。正如您在上面看到的,Word 2010 和 XnView 没有预乘,而 Firefox 和 Chrome 有。但是了解颜色 channel 是否应该预乘至关重要。

非常感谢您阐明这一点!

更新 2 粘贴到 Paint.NET 现在可以正常工作了。这是由我的代码中的一个错误引起的,如果 alpha channel 为 0,该错误没有将颜色 channel 设置为 0,即在这种情况下预乘没有正确完成,这似乎混淆了 Paint.NET。

Internet Explorer 10 的问题仍未解决。当将带有 alpha channel 的 PNG 复制到剪贴板时,IE 10 只是将 24 位 CF_DIBV5 放在剪贴板上,但 Paint.NET 可以粘贴带有 alpha channel 的位图,因此必须是 IE 10 向剪贴板公开的另一种格式。也许它公开了一个使用 CFSTR_FILECONTENTS 和 CFSTR_FILEDESCRIPTOR 的 PNG。

更新 我现在已经按照下面 arx 描述的方式实现了它,并且效果很好。但是,还有两点让我不解:

1) 将我的应用程序中的 alpha channel 图像粘贴到 Paint.NET 中不会保留 alpha channel 。图像在 Paint.NET 中显示为不透明。但是,从 Firefox 和 Chrome 粘贴到 Paint.NET 中效果很好,alpha channel 被保留了!我已经放弃了完整的 DIBV5,它与我的应用程序相同,但它仍然适用于 FF 和 Chrome,但不适用于我的应用程序,所以它一定有其他东西! Firefox 和 Chrome 一定在做我的应用没有做的其他事情!?

2) Internet Explorer 10 也是如此。将 alpha channel 图像从 IE 10 粘贴到我的应用程序根本不起作用...我得到一个位深度为 24 的 DIB,即没有alpha channel 。但是,当从 IE 10 粘贴到 Paint.NET 时,alpha channel 就在那里!所以这里一定还有更多的东西......

最佳答案

我确信在 CF_DIBV5 中存储 alpha 有正确的方法,但这并不重要。应用程序已经不一致地处理它,所以如果您希望您的应用程序与其他应用程序很好地协同工作,则不能使用 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 公开;仅使用 HGLOBAL 似乎不起作用。数据与“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;
}

关于c - Win32 剪贴板和 alpha channel 图像,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/15689541/

相关文章:

c++ - 如何捕获来自某些进程的所有 HTTP 请求?

c - pthread_mutex_unlock 不是原子的

c++ - 如何使用 Win32 解码 JPEG?

c - 跳过位图到字符串转换器中的填充?

c - 如何获取数组大小

c - Intel Galileo 裸机 UART

c - 在单个 DNS 查询中请求 A 和 AAAA 记录

c++ - 如何在 C++/WinAPI 中通过网络适配器获取发送/接收的字节数

python - os.system(cmd) 和 subprocess.call(cmd, shell=True) 的执行与 CMD.EXE 不同

c++ - C++ Win32 中的哪个控件等于 C# WPF 中的 "Run"控件?