需要帮助将具有透明背景的图像设置为剪贴板
Posted
技术标签:
【中文标题】需要帮助将具有透明背景的图像设置为剪贴板【英文标题】:Need Help Setting an Image with Transparent Background to Clipboard 【发布时间】:2011-02-17 11:19:22 【问题描述】:我需要帮助将透明图像设置到剪贴板。我不断收到“句柄无效”。基本上,我需要“第二双眼睛”来查看以下代码。 (完整的工作项目ftp://missico.net/ImageVisualizer.zip。)
这是一个图像调试可视化器类库,但我将包含的项目作为可执行文件运行以进行测试。 (注意 window 是一个工具箱窗口,在任务栏中显示设置为 false。)我厌倦了必须在工具箱窗口上执行屏幕截图,用图像编辑器打开屏幕截图,然后删除添加的背景,因为它是一个屏幕截图。所以我想我会很快把透明图像放到剪贴板上。好吧,问题是……Clipboard.SetImage 不支持透明度。谷歌来救援……不完全是。
这是我目前所拥有的。我从许多来源中提取。请参阅代码以获取主要参考。我的问题是使用 CF_DIBV5 时的“无效句柄”。我需要使用 BITMAPV5HEADER 和 CreateDIBitmap 吗?
非常感谢您 GDI/GDI+ 向导的任何帮助。
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();
【问题讨论】:
您能澄清一下哪个步骤给您带来了无效句柄错误吗? 【参考方案1】:Bitmap.GetHbitmap()
实际上会将位图合成为不透明背景,从而丢失 Alpha 通道。 This question 解决了如何获取具有完整 alpha 通道的 HBITMAP。
【讨论】:
同样的“无效句柄”问题。此外,如果我使用 CF_BITMAP,则背景是黑色,而不是 PictureBox 控件使用的“控制灰色”。 剪贴板格式的文档(请参阅msdn.microsoft.com/en-us/library/ms649013.aspx)指出,当使用 CF_DIB 时,句柄是使用 GlobalAlloc 分配的缓冲区,其中包含 BITMAPINFO 结构,后跟位图位。我希望您必须手动编组。使用 CF_BITMAP 时,可以传递一个 HBITMAP,但是没有关联的元数据来指定使用 alpha 通道,所以它被忽略了。【参考方案2】:我看到三个问题:
无效句柄错误可能来自将memBM
选择为memDC
。在将位图传递到其他任何地方之前,您应该始终从 DC 中选择位图。
BitBlt
是 GDI 调用(不是 GDI+)。 GDI 不保留 Alpha 通道。在较新版本的 Windows 上,您可以使用 AlphaBlend
将带有 Alpha 的位图合成到背景上,但合成中没有 Alpha 通道。
您已经创建了一个兼容的位图,这意味着该位图与您传入的 DC 具有相同的颜色格式(我假设它与屏幕相同)。因此,您的位图最终可能是 16 位、24 位、32 位甚至 8 位,具体取决于屏幕的设置方式。如果BitBlt
保留了原始的 alpha 通道,那么在转换为屏幕格式时可能会丢失它。
【讨论】:
AlphaBlend 已经存在了很长一段时间。【参考方案3】:您可以做一些事情来稍微收紧代码库,我已经完成了一些关于 CF_DIBV5 的功课,您可能会觉得有帮助。
首先,在__copyMenuItem_Click()
中,我们有您图片的四份完整副本,这远远超出了必要的范围。
__pictureBox.Image
Image imag = Image.FromStream(ms);
new Bitmap(imag)
Bitmap img = new Bitmap(new Bitmap(imag));
(外位图)
此外,您的 MemoryStream
、imag
、new 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);
我相当肯定,在使用 CF_DIBV5 时,您必须整理出 BITMAPV5HEADER 结构以将位传递到剪贴板。我以前错了,但我会验证 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
祝你好运!
【讨论】:
我没有时间试用您的代码,但我会为您的辛勤工作提供奖励。发布的代码是我从众多来源收集的一次性代码,以便让任何东西都能正常工作。所以你的“第二眼”和建议是很有帮助的。一旦我赶上,我会尝试你的建议。 我发现很多应用程序(ab)使用“位域”(压缩=3)类型的 32 位 v1 DIB(40 字节标头)作为 ARGB,而根据其规格它只是RGB。其中,Windows 10 printscreen(将屏幕截图中的“alpha”位设置为 255,除了我的双显示器设置中不存在的部分)和 Google Chrome。以上是关于需要帮助将具有透明背景的图像设置为剪贴板的主要内容,如果未能解决你的问题,请参考以下文章