using System; using System.Collections.Generic; using System.Text; using CPF.Input; using System.Threading; using System.Runtime.InteropServices; using System.Runtime.InteropServices.ComTypes; using System.IO; using CPF.Drawing; using System.Linq; namespace CPF.Windows { internal class ClipboardImpl : IClipboard { static int htmlId; static int[] Format; static ClipboardImpl() { //CF_BITMAP 位图的句柄(HBITMAP)。 //CF_DIB 包含BITMAPINFO结构和位图位的内存对象。 htmlId = UnmanagedMethods.RegisterClipboardFormat(Html); Format = new int[] { (int)UnmanagedMethods.ClipboardFormat.CF_UNICODETEXT, htmlId, (int)UnmanagedMethods.ClipboardFormat.CF_DIB, (int)UnmanagedMethods.ClipboardFormat.CF_HDROP, 0 }; } private void OpenClipboard() { while (!UnmanagedMethods.OpenClipboard(IntPtr.Zero)) { Thread.Sleep(100); //await Task.Delay(100); } } //public void SetText(string text) //{ // if (text == null) // { // throw new ArgumentNullException(nameof(text)); // } // OpenClipboard(); // UnmanagedMethods.EmptyClipboard(); // try // { // var hGlobal = Marshal.StringToHGlobalUni(text); // UnmanagedMethods.SetClipboardData(UnmanagedMethods.ClipboardFormat.CF_UNICODETEXT, hGlobal); // } // finally // { // UnmanagedMethods.CloseClipboard(); // } //} public void Clear() { OpenClipboard(); try { UnmanagedMethods.EmptyClipboard(); } finally { UnmanagedMethods.CloseClipboard(); } } //List intPtrs = new List(); public void SetData(params ValueTuple[] data) { if (data == null) { throw new ArgumentNullException(nameof(data)); } //foreach (var item in intPtrs) //{ // Marshal.FreeHGlobal(item); //} OpenClipboard(); UnmanagedMethods.EmptyClipboard(); try { IntPtr hGlobal = IntPtr.Zero; IntPtr ptr = IntPtr.Zero; foreach (var item in data) { switch (item.Item1) { case DataFormat.Text: hGlobal = Marshal.StringToHGlobalUni(item.Item2.ToString()); UnmanagedMethods.SetClipboardData(UnmanagedMethods.ClipboardFormat.CF_UNICODETEXT, hGlobal); break; case DataFormat.Html: var html = item.Item2.ToString(); Encoding enc = Encoding.UTF8; string begin = "Version:0.9\r\nStartHTML:{0:000000}\r\nEndHTML:{1:000000}" + "\r\nStartFragment:{2:000000}\r\nEndFragment:{3:000000}\r\n"; string html_begin = "\r\n\r\n" + "\r\n" + "HTML clipboard\r\n\r\n\r\n" + ""; string html_end = "\r\n\r\n\r\n"; string begin_sample = String.Format(begin, 0, 0, 0, 0); int count_begin = enc.GetByteCount(begin_sample); int count_html_begin = enc.GetByteCount(html_begin); int count_html = enc.GetByteCount(html); int count_html_end = enc.GetByteCount(html_end); string html_total = String.Format( begin , count_begin , count_begin + count_html_begin + count_html + count_html_end , count_begin + count_html_begin , count_begin + count_html_begin + count_html ) + html_begin + html + html_end; //var h = Marshal.StringToCoTaskMemUTF8(html_total); var l = enc.GetByteCount(html_total); var h = Marshal.AllocHGlobal(l); //intPtrs.Add(h); Marshal.Copy(enc.GetBytes(html_total), 0, h, l); UnmanagedMethods.SetClipboardData(htmlId, h); //Marshal.FreeHGlobal(h); break; case DataFormat.Image: var img = item.Item2 as Image; IntPtr screenDC = UnmanagedMethods.GetDC(IntPtr.Zero); IntPtr memDc = UnmanagedMethods.CreateCompatibleDC(screenDC); UnmanagedMethods.BITMAPINFOHEADER info = new UnmanagedMethods.BITMAPINFOHEADER(); info.biSize = (uint)Marshal.SizeOf(typeof(UnmanagedMethods.BITMAPINFOHEADER)); info.biBitCount = 24; info.biHeight = img.Height; info.biWidth = img.Width; info.biPlanes = 1; info.biSizeImage = (uint)(img.Width * img.Height * 3); var dibHbitmap = UnmanagedMethods.CreateDIBSection(memDc, ref info, 0, out IntPtr ppvBits, IntPtr.Zero, 0); var oldBits = UnmanagedMethods.SelectObject(memDc, dibHbitmap); IntPtr hBitmap; var stream = img.SaveToStream(ImageFormat.Png); var states = UnmanagedMethods.GdipCreateBitmapFromStream(new GPStream(stream), out IntPtr bitmap); stream.Dispose(); UnmanagedMethods.GdipCreateHBITMAPFromBitmap(bitmap, out hBitmap, UnmanagedMethods.ToWin32(Color.White)); UnmanagedMethods.GdipDisposeImage(bitmap); UnmanagedMethods.SetClipboardData(UnmanagedMethods.ClipboardFormat.CF_BITMAP, hBitmap); IntPtr sdc = UnmanagedMethods.CreateCompatibleDC(screenDC); var sob = UnmanagedMethods.SelectObject(sdc, hBitmap); UnmanagedMethods.BitBlt(memDc, 0, 0, img.Width, img.Height, sdc, 0, 0, TernaryRasterOperations.SRCCOPY); var dib = UnmanagedMethods.GlobalAlloc(UnmanagedMethods.GMEM_MOVEABLE | UnmanagedMethods.GMEM_ZEROINIT, (int)info.biSize + (int)info.biSizeImage); ptr = UnmanagedMethods.GlobalLock(dib); Marshal.StructureToPtr(info, ptr, true); var d = new byte[info.biSizeImage]; Marshal.Copy(ppvBits, d, 0, d.Length); Marshal.Copy(d, 0, ptr + (int)info.biSize, d.Length); UnmanagedMethods.SetClipboardData(UnmanagedMethods.ClipboardFormat.CF_DIB, dib); UnmanagedMethods.GlobalUnlock(dib); UnmanagedMethods.SelectObject(sdc, sob); UnmanagedMethods.DeleteDC(sdc); UnmanagedMethods.SelectObject(memDc, oldBits); UnmanagedMethods.DeleteDC(memDc); UnmanagedMethods.ReleaseDC(IntPtr.Zero, screenDC); //UnmanagedMethods.DeleteObject(hBitmap); break; case DataFormat.FileNames: var files = item.Item2 as IEnumerable; if (files != null) { char[] filesStr = (string.Join("\0", files) + "\0\0").ToCharArray(); _DROPFILES df = new _DROPFILES(); df.fWide = true; #if Net4 df.pFiles = Marshal.SizeOf(typeof(_DROPFILES)); int required = (filesStr.Length * sizeof(char)) + Marshal.SizeOf(typeof(_DROPFILES)); #else df.pFiles = Marshal.SizeOf<_DROPFILES>(); int required = (filesStr.Length * sizeof(char)) + Marshal.SizeOf<_DROPFILES>(); #endif hGlobal = UnmanagedMethods.GlobalAlloc(UnmanagedMethods.GMEM_MOVEABLE | UnmanagedMethods.GMEM_ZEROINIT, required); long available = UnmanagedMethods.GlobalSize(hGlobal).ToInt64(); if (required > available) { break; } ptr = UnmanagedMethods.GlobalLock(hGlobal); try { Marshal.StructureToPtr(df, ptr, false); #if Net4 Marshal.Copy(filesStr, 0, ptr + Marshal.SizeOf(typeof(_DROPFILES)), filesStr.Length); #else Marshal.Copy(filesStr, 0, ptr + Marshal.SizeOf<_DROPFILES>(), filesStr.Length); #endif UnmanagedMethods.SetClipboardData(UnmanagedMethods.ClipboardFormat.CF_HDROP, hGlobal); } finally { UnmanagedMethods.GlobalUnlock(hGlobal); } } break; } } } finally { UnmanagedMethods.CloseClipboard(); } } public bool Contains(DataFormat dataFormat) { OpenClipboard(); uint LastRetrievedFormat = 0; List list = new List(); while (0 != (LastRetrievedFormat = UnmanagedMethods.EnumClipboardFormats(LastRetrievedFormat))) { list.Add(LastRetrievedFormat); } UnmanagedMethods.CloseClipboard(); //var f = Format[(int)dataFormat - 1]; var f = GetFormatId(dataFormat); if (list.IndexOf((uint)f) < 0) { if (dataFormat == DataFormat.Text) { if (list.IndexOf((uint)UnmanagedMethods.ClipboardFormat.CF_TEXT) >= 0) { return true; } if (list.IndexOf((uint)UnmanagedMethods.ClipboardFormat.CF_OEMTEXT) >= 0) { return true; } } return false; } return true; } public static DataFormat GetFormat(int format) { ushort f = (ushort)(format & 0xFFFF); if (format == 2 || format == 8) { return DataFormat.Image; } for (int i = 0; i < Format.Length; i++) { if (Format[i] == f) { if (i == 0) { return DataFormat.Text; } else if (i == 1) { return DataFormat.Html; } else if (i == 2) { return DataFormat.Image; } else if (i == 3) { return DataFormat.FileNames; } } } if (format == (ushort)UnmanagedMethods.ClipboardFormat.CF_TEXT || format == (ushort)UnmanagedMethods.ClipboardFormat.CF_OEMTEXT) { return DataFormat.Text; } return DataFormat.Unknown; } public static int GetFormatId(DataFormat dataFormat) { //return Format[(int)dataFormat]; switch (dataFormat) { case DataFormat.Text: return Format[0]; case DataFormat.Html: return Format[1]; case DataFormat.Image: return Format[2]; case DataFormat.FileNames: return Format[3]; default: return 0; } } public static Image ImageFormHBitmap(IntPtr hBitmap) { var r = UnmanagedMethods.GdipCreateBitmapFromHBITMAP(hBitmap, IntPtr.Zero, out IntPtr b); if (r != 0) { return null; } MemoryStream ms = new MemoryStream(); Guid png = new Guid("{b96b3caf-0728-11d3-9d7b-0000f81ef32e}"); Guid jpg = new Guid("{b96b3cae-0728-11d3-9d7b-0000f81ef32e}"); int numDecoders; int status = UnmanagedMethods.GdipGetImageDecodersSize(out numDecoders, out int size); if (status != 0) { return null; } IntPtr memory = Marshal.AllocHGlobal(size); status = UnmanagedMethods.GdipGetImageDecoders(numDecoders, size, memory); int index; List codes = new List(); for (index = 0; index < numDecoders; index++) { IntPtr curcodec = (IntPtr)((long)memory + (int)Marshal.SizeOf(typeof(ImageCodecInfoPrivate)) * index); ImageCodecInfoPrivate codecp = new ImageCodecInfoPrivate(); Marshal.PtrToStructure(curcodec, codecp); codes.Add(codecp); } var pngCode = codes.Find(a => a.FormatID == png); UnmanagedMethods.GdipSaveImageToStream(b, new GPStream(ms), ref pngCode.Clsid, IntPtr.Zero); ms.Position = 0; var img = new Bitmap(ms); UnmanagedMethods.GdipDisposeImage(b); //ms.Position = 0; //File.WriteAllBytes(Path.Combine(CPF.Platform.Application.StartupPath, "test.png"), ms.ToArray()); ms.Dispose(); return img; } public object GetData(DataFormat dataFormat) { //if (dataFormat != DataFormat.Image) { OpenClipboard(); try { int formatId = (int)UnmanagedMethods.ClipboardFormat.CF_UNICODETEXT; switch (dataFormat) { case DataFormat.Text: uint LastRetrievedFormat = 0; while (0 != (LastRetrievedFormat = UnmanagedMethods.EnumClipboardFormats(LastRetrievedFormat))) { if (LastRetrievedFormat == (uint)UnmanagedMethods.ClipboardFormat.CF_OEMTEXT) { formatId = (int)UnmanagedMethods.ClipboardFormat.CF_OEMTEXT; break; } else if (LastRetrievedFormat == (uint)UnmanagedMethods.ClipboardFormat.CF_TEXT) { formatId = (int)UnmanagedMethods.ClipboardFormat.CF_TEXT; break; } } break; case DataFormat.Html: formatId = htmlId; break; case DataFormat.Image: formatId = (int)UnmanagedMethods.ClipboardFormat.CF_DIB; break; case DataFormat.FileNames: formatId = (int)UnmanagedMethods.ClipboardFormat.CF_HDROP; break; } IntPtr hText = UnmanagedMethods.GetClipboardData(formatId); if (hText == IntPtr.Zero) { return null; } string stringData = null; int size; IntPtr ptr = UnmanagedMethods.GlobalLock(hText); try { switch (dataFormat) { case DataFormat.Text: if (formatId == (int)UnmanagedMethods.ClipboardFormat.CF_TEXT) { var rv = Marshal.PtrToStringAnsi(hText); return rv; } else { var rv = Marshal.PtrToStringUni(hText); return rv; } case DataFormat.Html: size = (int)UnmanagedMethods.GlobalSize(hText); //if (unicode) //{ //stringData = new string((char*)ptr); //} //else //{ // stringData = new string((sbyte*)ptr); //} byte[] bytes = new byte[size]; Marshal.Copy(ptr, bytes, 0, size); stringData = Encoding.UTF8.GetString(bytes); var start = stringData.IndexOf("