我需要帮助将透明图像设置到剪贴板。我一直得到“处理无效”。基本上,我需要一个“第二组眼睛”来查看以下代码。 (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();
}
}
答案 0 :(得分:4)
您可以采取一些措施来加强代码库,我已经在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);
我相当肯定你必须编组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支持可能不会为你削减它。我已经在一些应用程序上测试了这种方法;但是,您需要确定这是否足以满足您的要求。
对于我所研究的所有内容的讨论对于Stack Overflow而言过于冗长。我在我的博客上提供了其他示例代码和讨论:http://www.notesoncode.com/articles/2010/08/16/HandlingTransparentImagesOnTheClipboardIsForSomeReasonHard.aspx
祝你好运!答案 1 :(得分:1)
我看到三个问题:
无效的句柄错误可能来自将memBM
选入memDC
。您应该始终从DC中选择位图,然后再将其传递到其他位置。
BitBlt
是GDI调用(不是GDI +)。 GDI不保留alpha通道。在较新版本的Windows上,您可以使用AlphaBlend
将带有alpha的位图合成到背景上,但复合材料不会有Alpha通道。
您已经创建了一个兼容的位图,这意味着位图的颜色格式与您传入的DC的颜色格式相同(我假设它与屏幕相同)。因此,您的位图可能最终为16位,24位,32位甚至8位,具体取决于屏幕的设置方式。如果BitBlt
保留了原始的Alpha通道,则转换为屏幕格式时可能会丢失它。
答案 2 :(得分:0)
Bitmap.GetHbitmap()
实际上会将位图合成到不透明背景,从而丢失Alpha通道。 This question解决了如何使alpha通道完整的HBITMAP。