需要帮助设置具有透明背景的图像到剪贴板

时间:2010-05-09 22:32:47

标签: winforms gdi+ transparency gdi clipboard

我需要帮助将透明图像设置到剪贴板。我一直得到“处理无效”。基本上,我需要一个“第二组眼睛”来查看以下代码。 (ftp://missico.net/ImageVisualizer.zip的完整工作项目。)

这是一个Debug Visualizer类库的图像,但是我将包含的项目作为可执行文件运行进行测试。 (请注意,窗口是工具箱窗口,任务栏中的显示设置为false。)我厌倦了必须在工具箱窗口上执行屏幕捕获,使用图像编辑器打开屏幕捕获,然后删除添加的背景,因为它是一个屏幕截图。所以我想我会很快将透明图像放到剪贴板上。好吧,问题是......没有对Clipboard.SetImage的透明度支持。谷歌救援......并不完全。

这是我到目前为止所拥有的。我从众多消息来源中撤出。请参阅主要参考的代码。我的问题是使用CF_DIBV5时的“无效句柄”。我需要使用BITMAPV5HEADER和CreateDIBitmap吗?

GDI / GDI + Wizards的任何帮助都将不胜感激。

    public static void SetClipboardData(Bitmap bitmap, IntPtr hDC) {

        const uint SRCCOPY = 0x00CC0020;
        const int CF_DIBV5 = 17;
        const int CF_BITMAP = 2;

        //'reference
        //'http://social.msdn.microsoft.com/Forums/en-US/winforms/thread/816a35f6-9530-442b-9647-e856602cc0e2

        IntPtr memDC = CreateCompatibleDC(hDC);
        IntPtr memBM = CreateCompatibleBitmap(hDC, bitmap.Width, bitmap.Height);

        SelectObject(memDC, memBM);

        using (Graphics g = Graphics.FromImage(bitmap)) {

            IntPtr hBitmapDC = g.GetHdc();
            IntPtr hBitmap = bitmap.GetHbitmap();

            SelectObject(hBitmapDC, hBitmap);

            BitBlt(memDC, 0, 0, bitmap.Width, bitmap.Height, hBitmapDC, 0, 0, SRCCOPY);

            if (!OpenClipboard(IntPtr.Zero)) {
                throw new System.Runtime.InteropServices.ExternalException("Could not open Clipboard", new Win32Exception());
            }

            if (!EmptyClipboard()) {
                throw new System.Runtime.InteropServices.ExternalException("Unable to empty Clipboard", new Win32Exception());
            }

            //IntPtr hClipboard = SetClipboardData(CF_BITMAP, memBM); //works but image is not transparent

            //all my attempts result in SetClipboardData returning hClipboard = IntPtr.Zero
            IntPtr hClipboard = SetClipboardData(CF_DIBV5, memBM);


            //because 
            if (hClipboard == IntPtr.Zero) {

                //    InnerException: System.ComponentModel.Win32Exception
                //         Message="The handle is invalid"
                //         ErrorCode=-2147467259
                //         NativeErrorCode=6
                //         InnerException: 

                throw new System.Runtime.InteropServices.ExternalException("Could not put data on Clipboard", new Win32Exception());
            }

            if (!CloseClipboard()) {
                throw new System.Runtime.InteropServices.ExternalException("Could not close Clipboard", new Win32Exception());
            }

            g.ReleaseHdc(hBitmapDC);

        }

    }

    private void __copyMenuItem_Click(object sender, EventArgs e) {

        using (Graphics g = __pictureBox.CreateGraphics()) {

            IntPtr hDC = g.GetHdc();

            MemoryStream ms = new MemoryStream();

            __pictureBox.Image.Save(ms, ImageFormat.Png);

            ms.Seek(0, SeekOrigin.Begin);

            Image imag = Image.FromStream(ms);

            // Derive BitMap object using Image instance, so that you can avoid the issue 
            //"a graphics object cannot be created from an image that has an indexed pixel format"

            Bitmap img = new Bitmap(new Bitmap(imag));

            SetClipboardData(img, hDC);

            g.ReleaseHdc();

        }

    }

3 个答案:

答案 0 :(得分:4)

您可以采取一些措施来加强代码库,我已经在CF_DIBV5上完成了一些您可能会觉得有用的功课。

首先,在__copyMenuItem_Click()中,我们有四张完整的图片副本,这远远超过必要。

  1. __pictureBox.Image
  2. Image imag = Image.FromStream(ms);
  3. new Bitmap(imag)
  4. Bitmap img = new Bitmap(new Bitmap(imag));(外部位图)
  5. 此外,您的MemoryStreamimagnew Bitmap(imag)img不会被处置,这可能会导致问题。

    不改变代码的意图(可能没有解决句柄问题),你可以重写这样的方法:

    private void __copyMenuItem_Click(object sender, EventArgs e)
    {
        var image =  __pictureBox.Image;
        using (var g = __pictureBox.CreateGraphics())
        using (var bmp = new Bitmap(image.Width, image.Height, PixelFormat.Format32bppArgb))
        using (var bmpg = Graphics.FromImage(bmp))
        {
            IntPtr hDC = g.GetHdc();
            bmpg.DrawImage(image, 0, 0, image.Width, image.Height);
            SetClipboardData(bmp, hDC);
            g.ReleaseHdc();
        }
    }
    

    下一件看起来需要引起注意的事情是:

    IntPtr hClipboard = SetClipboardData(CF_DIBV5, memBM);
    

    我相当肯定你必须编组BITMAPV5HEADER结构,以便在使用CF_DIBV5时将位传递给剪贴板。我以前错了,但我会验证memBM实际上是否包含标题。一个好的指标是第一个DWORD(UInt32)是否具有值124(以字节为单位的头的大小)。

    我的最后评论比第二双眼睛更有推荐。我发现像GIMP,Paint.NET,Fireworks和PhotoScape这样的照片应用程序似乎对CF_DIBV5(Format17)粘贴的支持很差或者不存在。您可以考虑将PNG格式复制到剪贴板,并使用不透明的位图作为备份,以防目标应用程序不支持PNG。我使用扩展方法来促进这一点:

    public static void CopyMultiFormatBitmapToClipboard(this Image image)
    {
        using (var opaque = image.CreateOpaqueBitmap(Color.White))
        using (var stream = new MemoryStream())
        {
            image.Save(stream, ImageFormat.Png);
    
            Clipboard.Clear();
            var data = new DataObject();
            data.SetData(DataFormats.Bitmap, true, opaque);
            data.SetData("PNG", true, stream);
            Clipboard.SetDataObject(data, true);
        }
    }
    

    使用扩展方法,您的__copyMenuItem_Click()方法可以简化为以下方法,并且可以完全删除SetClipboardData()方法:

    private void __copyMenuItem_Click(object sender, EventArgs e)
    {
        __pictureBox.Image.CopyMultiFormatBitmapToClipboard();
    }
    

    现在,正如我们已经在另一个主题上讨论的那样,PNG支持可能不会为你削减它。我已经在一些应用程序上测试了这种方法;但是,您需要确定这是否足以满足您的要求。

    • GIMP:支持透明度
    • Fireworks(3.0):支持透明度
    • PhotoScape:白色背景
    • Paint.NET:白色背景
    • MS PowerPoint 2007:支持透明度
    • MS Word 2007:白色背景
    • MS Paint(Win7):白色背景

    对于我所研究的所有内容的讨论对于Stack Overflow而言过于冗长。我在我的博客上提供了其他示例代码和讨论:http://www.notesoncode.com/articles/2010/08/16/HandlingTransparentImagesOnTheClipboardIsForSomeReasonHard.aspx

    祝你好运!

答案 1 :(得分:1)

我看到三个问题:

  1. 无效的句柄错误可能来自将memBM选入memDC。您应该始终从DC中选择位图,然后再将其传递到其他位置。

  2. BitBlt是GDI调用(不是GDI +)。 GDI不保留alpha通道。在较新版本的Windows上,您可以使用AlphaBlend将带有alpha的位图合成到背景上,但复合材料不会有Alpha通道。

  3. 您已经创建了一个兼容的位图,这意味着位图的颜色格式与您传入的DC的颜色格式相同(我假设它与屏幕相同)。因此,您的位图可能最终为16位,24位,32位甚至8位,具体取决于屏幕的设置方式。如果BitBlt保留了原始的Alpha通道,则转换为屏幕格式时可能会丢失它。

答案 2 :(得分:0)

Bitmap.GetHbitmap()实际上会将位图​​合成到不透明背景,从而丢失Alpha通道。 This question解决了如何使alpha通道完整的HBITMAP。