我希望将CF_DIBV5
格式从剪贴板转换为BitmapSource
对象。到目前为止,这是我的代码,基于之前用于将普通DIB转换为BitmapSource
的代码。为了转换普通的DIB,我发现我只需要获取头部,并将其添加到位图的原始字节中。
不幸的是,当我尝试使代码DIBV5兼容时,程序会抛出以下异常。
没有找到适合完成此操作的成像组件。
当我使用BitmapFrame
的{{1}}方法时会发生这种情况。完整的代码可以在下面看到。
Create
关于我做错了什么,或者我做得不对的任何想法?我在失明中摸索着,不幸的是,像我这样的C#程序员的文档很少。我理解了一些C ++,但还不足以从C ++转换为C#。
答案 0 :(得分:1)
将此image复制到剪贴板。您还需要将一个Image控件添加到MainWindow。这需要一段时间来弄清楚所以我可能错过了解释添加所需的参考。
using System;
using System.Runtime.InteropServices;
using System.Windows;
using System.Windows.Interop;
using System.Windows.Media;
using System.Windows.Media.Imaging;
namespace WpfApp1
{
public partial class MainWindow : Window
{
[DllImport("User32.dll", SetLastError = true)]
[return: MarshalAs(UnmanagedType.Bool)]
private static extern bool IsClipboardFormatAvailable(uint format);
[DllImport("User32.dll", SetLastError = true)]
private static extern IntPtr GetClipboardData(uint uFormat);
[DllImport("User32.dll", SetLastError = true)]
[return: MarshalAs(UnmanagedType.Bool)]
private static extern bool OpenClipboard(IntPtr hWndNewOwner);
[DllImport("User32.dll", SetLastError = true)]
[return: MarshalAs(UnmanagedType.Bool)]
private static extern bool CloseClipboard();
[DllImport("Kernel32.dll", SetLastError = true)]
private static extern IntPtr GlobalLock(IntPtr hMem);
[DllImport("Kernel32.dll", SetLastError = true)]
[return: MarshalAs(UnmanagedType.Bool)]
private static extern bool GlobalUnlock(IntPtr hMem);
const uint CF_DIBV5 = 17;
public enum BitmapCompressionMode : uint
{
BI_RGB = 0,
BI_RLE8 = 1,
BI_RLE4 = 2,
BI_BITFIELDS = 3,
BI_JPEG = 4,
BI_PNG = 5
}
[StructLayout(LayoutKind.Sequential)]
public struct BITMAPV5HEADER
{
public uint bV5Size;
public int bV5Width;
public int bV5Height;
public UInt16 bV5Planes;
public UInt16 bV5BitCount;
public uint bV5Compression;
public uint bV5SizeImage;
public int bV5XPelsPerMeter;
public int bV5YPelsPerMeter;
public UInt16 bV5ClrUsed;
public UInt16 bV5ClrImportant;
public UInt16 bV5RedMask;
public UInt16 bV5GreenMask;
public UInt16 bV5BlueMask;
public UInt16 bV5AlphaMask;
public UInt16 bV5CSType;
public IntPtr bV5Endpoints;
public UInt16 bV5GammaRed;
public UInt16 bV5GammaGreen;
public UInt16 bV5GammaBlue;
public UInt16 bV5Intent;
public UInt16 bV5ProfileData;
public UInt16 bV5ProfileSize;
public UInt16 bV5Reserved;
}
[StructLayout(LayoutKind.Sequential, Pack = 1)]
public struct RGBQUAD
{
public byte rgbBlue;
public byte rgbGreen;
public byte rgbRed;
public byte rgbReserved;
}
public MainWindow()
{
InitializeComponent();
AttemptConversion();
}
private void AttemptConversion()
{
WindowInteropHelper helper = new WindowInteropHelper(this);
bool gotIt = OpenClipboard(helper.Handle);
if (!gotIt) return;
bool formatAvail = IsClipboardFormatAvailable(CF_DIBV5);
if (!formatAvail) return;
IntPtr hBitmap = IntPtr.Zero;
hBitmap = GetClipboardData(CF_DIBV5);
IntPtr ptr = GlobalLock(hBitmap);
BitmapSource bmpSrc = CF_DIBV5ToBitmapSource(hBitmap);
img.Width = (int)bmpSrc.Width;
img.Height = (int)bmpSrc.Height;
img.Source = bmpSrc;
if (ptr != IntPtr.Zero) GlobalUnlock(hBitmap);
if (gotIt) CloseClipboard();
}
private BitmapSource CF_DIBV5ToBitmapSource(IntPtr hBitmap)
{
IntPtr scan0 = IntPtr.Zero;
var bmi = (BITMAPV5HEADER)Marshal.PtrToStructure(hBitmap, typeof(BITMAPV5HEADER));
int stride = (int)(bmi.bV5SizeImage / bmi.bV5Height);
long offset = bmi.bV5Size + bmi.bV5ClrUsed * Marshal.SizeOf<RGBQUAD>();
if (bmi.bV5Compression == (uint)BitmapCompressionMode.BI_BITFIELDS)
{
offset += 12; //bit masks follow the header
}
scan0 = new IntPtr(hBitmap.ToInt64() + offset);
BitmapSource bmpSource = BitmapSource.Create(
bmi.bV5Width, bmi.bV5Height,
bmi.bV5XPelsPerMeter, bmi.bV5YPelsPerMeter,
PixelFormats.Bgra32, null,
scan0, (int)bmi.bV5SizeImage, stride);
return bmpSource;
}
}
}