mirror of
				https://gitee.com/csharpui/CPF.git
				synced 2025-10-25 18:29:19 +08:00 
			
		
		
		
	
		
			
				
	
	
		
			967 lines
		
	
	
		
			36 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			967 lines
		
	
	
		
			36 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| using System;
 | |
| using System.Collections.Generic;
 | |
| using System.Text;
 | |
| using CPF.Input;
 | |
| using System.Linq;
 | |
| using System.Runtime.InteropServices;
 | |
| using CPF.Drawing;
 | |
| 
 | |
| namespace CPF.Linux
 | |
| {
 | |
|     class DataObject : IDataObject
 | |
|     {
 | |
|         public static IntPtr textAtom = XLib.XInternAtom(LinuxPlatform.Platform.Display, "text/plain", false);
 | |
|         public static IntPtr textUtf8Atom = XLib.XInternAtom(LinuxPlatform.Platform.Display, "text/plain;charset=utf-8", false);
 | |
|         public static IntPtr htmlAtom = XLib.XInternAtom(LinuxPlatform.Platform.Display, "text/html", false);
 | |
|         public static IntPtr fileNamesAtom = XLib.XInternAtom(LinuxPlatform.Platform.Display, "text/uri-list", false);
 | |
|         public static List<(IntPtr, IntPtr, DataFormat)> atomNonProtocols = new List<(IntPtr, IntPtr, DataFormat)>();
 | |
|         static DataObject()
 | |
|         {
 | |
|             var atoms = LinuxPlatform.Platform.Info.Atoms;
 | |
|             //atomNonProtocols.Add((textAtom, XLib.XInternAtom(LinuxPlatform.Platform.Display,
 | |
|             //            String.Concat("MWFNonP+", "text/plain"), false), DataFormat.Text));
 | |
|             //atomNonProtocols.Add((htmlAtom, XLib.XInternAtom(LinuxPlatform.Platform.Display,
 | |
|             //            String.Concat("MWFNonP+", "text/html"), false), DataFormat.Html));
 | |
|             //atomNonProtocols.Add((fileNamesAtom, XLib.XInternAtom(LinuxPlatform.Platform.Display,
 | |
|             //            String.Concat("MWFNonP+", "text/uri-list"), false), DataFormat.FileNames));
 | |
|             atomNonProtocols.Add((atoms.UTF8_STRING, atoms.UTF8_STRING, DataFormat.Text));
 | |
|             atomNonProtocols.Add((atoms.COMPOUND_TEXT, atoms.COMPOUND_TEXT, DataFormat.Text));
 | |
|             atomNonProtocols.Add((atoms.XA_STRING, atoms.XA_STRING, DataFormat.Text));
 | |
|             atomNonProtocols.Add((atoms.STRING, atoms.STRING, DataFormat.Text));
 | |
|             atomNonProtocols.Add((atoms.TEXT, atoms.TEXT, DataFormat.Text));
 | |
|         }
 | |
| 
 | |
|         string text;
 | |
|         string html;
 | |
|         List<string> fileNames;
 | |
|         X11Info x11Info;
 | |
|         public DataObject()
 | |
|         {
 | |
|             x11Info = LinuxPlatform.Platform.Info;
 | |
|         }
 | |
| 
 | |
| 
 | |
|         public bool Contains(DataFormat dataFormat)
 | |
|         {
 | |
|             switch (dataFormat)
 | |
|             {
 | |
|                 case DataFormat.Text:
 | |
|                     return text != null;
 | |
|                 case DataFormat.Html:
 | |
|                     return html != null;
 | |
|                 case DataFormat.FileNames:
 | |
|                     return fileNames != null;
 | |
|                 default:
 | |
|                     return false;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public object GetData(DataFormat dataFormat)
 | |
|         {
 | |
|             switch (dataFormat)
 | |
|             {
 | |
|                 case DataFormat.Text:
 | |
|                     return text;
 | |
|                 case DataFormat.Html:
 | |
|                     return html;
 | |
|                 case DataFormat.FileNames:
 | |
|                     return fileNames;
 | |
|                 default:
 | |
|                     return null;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public DragDropEffects EffectFromAction(IntPtr action)
 | |
|         {
 | |
|             if (action == x11Info.Atoms.XdndActionCopy)
 | |
|                 return DragDropEffects.Copy;
 | |
|             else if (action == x11Info.Atoms.XdndActionMove)
 | |
|                 return DragDropEffects.Move;
 | |
|             if (action == x11Info.Atoms.XdndActionLink)
 | |
|                 return DragDropEffects.Link;
 | |
| 
 | |
|             return DragDropEffects.None;
 | |
|         }
 | |
| 
 | |
|         private IntPtr ActionFromEffect(DragDropEffects effect)
 | |
|         {
 | |
|             IntPtr action = IntPtr.Zero;
 | |
| 
 | |
|             // We can't OR together actions on XDND so sadly the primary
 | |
|             // is the only one shown here
 | |
|             if ((effect & DragDropEffects.Copy) != 0)
 | |
|                 action = x11Info.Atoms.XdndActionCopy;
 | |
|             else if ((effect & DragDropEffects.Move) != 0)
 | |
|                 action = x11Info.Atoms.XdndActionMove;
 | |
|             else if ((effect & DragDropEffects.Link) != 0)
 | |
|                 action = x11Info.Atoms.XdndActionLink;
 | |
|             return action;
 | |
|         }
 | |
| 
 | |
|         //private IntPtr target;
 | |
|         //private IntPtr source;
 | |
|         //private IntPtr toplevel;
 | |
|         IntPtr[] SupportedTypes;
 | |
|         Point point;
 | |
|         CancelHandle cancelHandle;
 | |
|         IntPtr LastWindow;
 | |
|         IntPtr TargetWindow;
 | |
|         IntPtr SourceWindow;
 | |
|         DragState State;
 | |
|         IntPtr Action;
 | |
|         public DragDropEffects AllowedEffects;
 | |
|         //private bool dropped = false;
 | |
|         (DataFormat, object)[] data;
 | |
| 
 | |
|         public DragDropEffects StartDrag(DragDropEffects allowedEffects, params (DataFormat, object)[] data)
 | |
|         {
 | |
|             if (X11Window.mouseDownWindow == IntPtr.Zero)
 | |
|             {
 | |
|                 throw new Exception("必须是鼠标按下的时候调用");
 | |
|             }
 | |
|             SourceWindow = X11Window.mouseDownWindow;
 | |
|             State = DragState.Beginning;
 | |
|             //MouseState = XplatUIX11.MouseState;
 | |
|             AllowedEffects = allowedEffects;
 | |
|             Action = ActionFromEffect(allowedEffects);
 | |
|             this.data = data;
 | |
| 
 | |
|             var suc = XLib.XSetSelectionOwner(LinuxPlatform.Platform.Display, x11Info.Atoms.XdndSelection, X11Window.mouseDownWindow, IntPtr.Zero);
 | |
| 
 | |
|             List<IntPtr> types = new List<IntPtr>();
 | |
|             foreach (var item in data)
 | |
|             {
 | |
|                 switch (item.Item1)
 | |
|                 {
 | |
|                     case DataFormat.Text:
 | |
|                         types.Add(textAtom);
 | |
|                         types.Add(x11Info.Atoms.UTF8_STRING);
 | |
|                         types.Add(x11Info.Atoms.STRING);
 | |
|                         types.Add(x11Info.Atoms.TEXT);
 | |
|                         types.Add(x11Info.Atoms.COMPOUND_TEXT);
 | |
|                         types.Add(textUtf8Atom);
 | |
|                         break;
 | |
|                     case DataFormat.Html:
 | |
|                         types.Add(htmlAtom);
 | |
|                         break;
 | |
|                     case DataFormat.FileNames:
 | |
|                         types.Add(fileNamesAtom);
 | |
|                         break;
 | |
|                     default:
 | |
|                         break;
 | |
|                 }
 | |
|             }
 | |
|             SupportedTypes = types.ToArray();
 | |
| 
 | |
|             if (suc == 0)
 | |
|             {
 | |
|                 Console.Error.WriteLine("Could not take ownership of XdndSelection aborting drag.");
 | |
|                 //drag_data.Reset();
 | |
|                 return DragDropEffects.None;
 | |
|             }
 | |
|             //source = toplevel = target = X11Window.mouseDownWindow;
 | |
|             State = DragState.Dragging;
 | |
|             //dropped = false;
 | |
| 
 | |
|             TargetWindow = X11Window.mouseDownWindow;
 | |
|             SendEnter(X11Window.mouseDownWindow, X11Window.mouseDownWindow, SupportedTypes);
 | |
| 
 | |
|             if (cancelHandle != null)
 | |
|             {
 | |
|                 cancelHandle.Cancel = true;
 | |
|             }
 | |
|             cancelHandle = new CancelHandle();
 | |
|             LinuxPlatform.Platform.RunMainLoop(cancelHandle, OnEvent);
 | |
|             //if (!dropped)
 | |
|             return EffectFromAction(Action);
 | |
| 
 | |
|         }
 | |
|         bool WillAccept;
 | |
|         // This version seems to be the most common
 | |
|         private static readonly IntPtr[] XdndVersion = new IntPtr[] { new IntPtr(4) };
 | |
|         private unsafe void SendEnter(IntPtr handle, IntPtr from, IntPtr[] supported)
 | |
|         {
 | |
|             XEvent xevent = new XEvent();
 | |
| 
 | |
|             xevent.AnyEvent.type = XEventName.ClientMessage;
 | |
|             xevent.AnyEvent.display = LinuxPlatform.Platform.Display;
 | |
|             xevent.ClientMessageEvent.window = handle;
 | |
|             xevent.ClientMessageEvent.message_type = x11Info.Atoms.XdndEnter;
 | |
|             xevent.ClientMessageEvent.format = 32;
 | |
|             xevent.ClientMessageEvent.ptr1 = from;
 | |
|             xevent.ClientMessageEvent.ptr5 = Action;
 | |
| 
 | |
|             // (int) xevent.ClientMessageEvent.ptr2 & 0x1)
 | |
|             // int ptr2 = 0x1;
 | |
|             // xevent.ClientMessageEvent.ptr2 = (IntPtr) ptr2;
 | |
|             // (e)->xclient.data.l[1] = ((e)->xclient.data.l[1] & ~(0xFF << 24)) | ((v) << 24)
 | |
| 
 | |
|             //xevent.ClientMessageEvent.ptr2 = (IntPtr)((long)XdndVersion[0] << 24);
 | |
|             xevent.ClientMessageEvent.ptr2 = (IntPtr)67108865;
 | |
| 
 | |
|             //if (supported.Length > 0)
 | |
|             //    xevent.ClientMessageEvent.ptr3 = supported[0];
 | |
|             //if (supported.Length > 1)
 | |
|             //    xevent.ClientMessageEvent.ptr4 = supported[1];
 | |
|             //if (supported.Length > 2)
 | |
|             //    xevent.ClientMessageEvent.ptr5 = supported[2];
 | |
| 
 | |
|             //var list = supported.Select(a => (int)a).ToArray();
 | |
|             //fixed (void* data = list)
 | |
|             //{
 | |
|             XLib.XChangeProperty(x11Info.Display, from, x11Info.Atoms.XdndTypeList, (IntPtr)Atom.XA_ATOM, 32, PropertyMode.Replace, supported, supported.Length);
 | |
|             //}
 | |
| 
 | |
|             Console.WriteLine("SendEnter" + handle);
 | |
|             XLib.XSendEvent(LinuxPlatform.Platform.Display, handle, false, IntPtr.Zero, ref xevent);
 | |
|         }
 | |
| 
 | |
|         bool OnEvent(ref XEvent xEvent)
 | |
|         {
 | |
|             //Console.WriteLine(xEvent.type);
 | |
|             switch (xEvent.type)
 | |
|             {
 | |
|                 case XEventName.ButtonRelease:
 | |
|                     SetResult();
 | |
|                     return true;
 | |
|                 case XEventName.MotionNotify:
 | |
|                     point = new Point(xEvent.MotionEvent.x, xEvent.MotionEvent.y);
 | |
|                     HandleMouseOver();
 | |
|                     return true;
 | |
|                 case XEventName.GenericEvent:
 | |
|                     unsafe
 | |
|                     {
 | |
|                         fixed (void* data = &xEvent.GenericEventCookie)
 | |
|                         {
 | |
|                             XLib.XGetEventData(LinuxPlatform.Platform.Info.Display, data);
 | |
|                             try
 | |
|                             {
 | |
|                                 if (LinuxPlatform.Platform.Info.XInputOpcode ==
 | |
|                                     xEvent.GenericEventCookie.extension)
 | |
|                                 {
 | |
|                                     var xev = (XIEvent*)xEvent.GenericEventCookie.data;
 | |
|                                     if (LinuxPlatform.Platform.windows.TryGetValue(((XIDeviceEvent*)xev)->EventWindow, out var window))
 | |
|                                     {
 | |
|                                         if (xev->evtype == XiEventType.XI_ButtonRelease)
 | |
|                                         {
 | |
|                                             SetResult();
 | |
|                                         }
 | |
|                                         else if (xev->evtype == XiEventType.XI_Motion)
 | |
|                                         {
 | |
|                                             var ev = (XIDeviceEvent*)xev;
 | |
|                                             point = new Point((float)ev->event_x, (float)ev->event_y);
 | |
|                                             HandleMouseOver();
 | |
|                                         }
 | |
|                                     }
 | |
|                                 }
 | |
|                             }
 | |
|                             finally
 | |
|                             {
 | |
|                                 if (xEvent.GenericEventCookie.data != null)
 | |
|                                     XLib.XFreeEventData(LinuxPlatform.Platform.Info.Display, data);
 | |
|                             }
 | |
|                         }
 | |
|                         return true;
 | |
|                     }
 | |
|                 case XEventName.ClientMessage:
 | |
|                     if (xEvent.ClientMessageEvent.message_type == x11Info.Atoms.XdndStatus)
 | |
|                     {
 | |
|                         WillAccept = ((int)xEvent.ClientMessageEvent.ptr2 & 0x1) != 0;
 | |
|                         GiveFeedback(xEvent.ClientMessageEvent.ptr5);
 | |
|                         //Console.WriteLine("WillAccept" + WillAccept);
 | |
|                         return true;
 | |
|                     }
 | |
|                     else if (xEvent.ClientMessageEvent.message_type == x11Info.Atoms.XdndFinished)
 | |
|                     {
 | |
|                         cancelHandle.Cancel = true;
 | |
|                         return true;
 | |
|                     }
 | |
|                     break;
 | |
|                 case XEventName.SelectionRequest:
 | |
|                     if (xEvent.SelectionRequestEvent.selection != x11Info.Atoms.XdndSelection)
 | |
|                         break;
 | |
|                     //Console.WriteLine("requestor:" + xEvent.SelectionRequestEvent.requestor);
 | |
| 
 | |
|                     //foreach (var item in data)
 | |
|                     //{
 | |
|                     var type = xEvent.SelectionRequestEvent.target;
 | |
|                     var item = data.FirstOrDefault();
 | |
|                     if (type == textAtom || type == textUtf8Atom || type == x11Info.Atoms.STRING || type == x11Info.Atoms.TEXT)
 | |
|                     {
 | |
|                         item = data.FirstOrDefault(a => a.Item1 == DataFormat.Text);
 | |
|                     }
 | |
|                     else if (type == htmlAtom)
 | |
|                     {
 | |
|                         item = data.FirstOrDefault(a => a.Item1 == DataFormat.Html);
 | |
|                     }
 | |
|                     else if (type == fileNamesAtom)
 | |
|                     {
 | |
|                         item = data.FirstOrDefault(a => a.Item1 == DataFormat.FileNames);
 | |
|                     }
 | |
|                     if (item.Item2 != null)
 | |
|                     {
 | |
|                         switch (item.Item1)
 | |
|                         {
 | |
|                             case DataFormat.Text:
 | |
|                                 SetTextData(item.Item2, ref xEvent);
 | |
|                                 break;
 | |
|                             case DataFormat.Html:
 | |
|                                 SetHtmlData(item.Item2, ref xEvent);
 | |
|                                 break;
 | |
|                             case DataFormat.FileNames:
 | |
|                                 SetFileNameData(item.Item2, ref xEvent);
 | |
|                                 break;
 | |
|                         }
 | |
|                         //}
 | |
|                         XLib.XFlush(x11Info.Display);
 | |
|                     }
 | |
|                     return true;
 | |
|             }
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|         private void SetResult()
 | |
|         {
 | |
|             if (State == DragState.Beginning)
 | |
|             {
 | |
|                 //state = State.Accepting;
 | |
|             }
 | |
|             else if (State != DragState.None)
 | |
|             {
 | |
|                 if (WillAccept)
 | |
|                 {
 | |
|                     SendDrop(TargetWindow, SourceWindow, IntPtr.Zero);
 | |
|                     SendLeave(TargetWindow, SourceWindow);
 | |
|                     XLib.XChangeActivePointerGrab(x11Info.Display,
 | |
| EventMask.ButtonMotionMask |
 | |
| EventMask.PointerMotionMask |
 | |
| EventMask.ButtonPressMask |
 | |
| EventMask.ButtonReleaseMask,
 | |
| (IntPtr)((Cursor)Cursors.Arrow).PlatformCursor, IntPtr.Zero);
 | |
|                     //if (QueryContinue(false, DragAction.Drop))
 | |
|                     //    return;
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
| 
 | |
|                     cancelHandle.Cancel = true;
 | |
|                     SendLeave(TargetWindow, SourceWindow);
 | |
|                     //if (QueryContinue(false, DragAction.Cancel))
 | |
|                     //    return;
 | |
| 
 | |
|                     // fallback if no movement was detected, as .net does.
 | |
|                     //if (motion_poll == -1)
 | |
|                     //    DefaultEnterLeave(drag_data.Data);
 | |
|                 }
 | |
| 
 | |
|                 State = DragState.None;
 | |
|                 // WE can't reset the drag data yet as it is still
 | |
|                 // most likely going to be used by the SelectionRequest
 | |
|                 // handlers
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         DataFormat dataFormat;
 | |
|         DragDropEffects dragDropEffects;
 | |
|         public DragDropEffects DropEffects
 | |
|         {
 | |
|             get { return dragDropEffects; }
 | |
|         }
 | |
| 
 | |
|         public void SetDragDropEffects(DragDropEffects dragDropEffects)
 | |
|         {
 | |
|             if (dragDropEffects != this.dragDropEffects)
 | |
|             {
 | |
|                 SendStatus(TargetWindow, dragDropEffects);//发送消息,确定可以接受数据
 | |
|             }
 | |
|             this.dragDropEffects = dragDropEffects;
 | |
|         }
 | |
| 
 | |
|         Threading.DispatcherTimer timer;
 | |
|         public void StopTimer()
 | |
|         {
 | |
|             if (timer != null)
 | |
|             {
 | |
|                 timer.Stop();
 | |
|                 timer.Dispose();
 | |
|                 timer = null;
 | |
|             }
 | |
|             if (cancelHandle != null)
 | |
|             {
 | |
|                 cancelHandle.Cancel = true;
 | |
|                 cancelHandle = null;
 | |
|             }
 | |
| 
 | |
|         }
 | |
|         public void DragEnter(ref XEvent xEvent, EventHandler eventHandler)
 | |
|         {
 | |
|             dataFormat = DataFormat.Unknown;
 | |
|             text = null;
 | |
|             html = null;
 | |
|             fileNames = null;
 | |
|             TargetWindow = xEvent.AnyEvent.window;
 | |
|             SourceWindow = xEvent.ClientMessageEvent.ptr1;
 | |
|             SendStatus(SourceWindow, DropEffects);//发送消息,确定可以接受数据
 | |
| 
 | |
|             //确认可以接受数据的格式
 | |
|             foreach (IntPtr atom in SourceSupportedList(ref xEvent))
 | |
|             {
 | |
|                 //Console.WriteLine("格式:" + XLib.GetAtomName(x11Info.Display, atom));
 | |
|                 var f = atomNonProtocols.FirstOrDefault(a => a.Item1 == atom);
 | |
|                 Console.WriteLine(XLib.GetAtomName(LinuxPlatform.Platform.Display, atom));
 | |
|                 if (f.Item1 == IntPtr.Zero || dataFormat.HasFlag(f.Item3))
 | |
|                 {
 | |
|                     continue;
 | |
|                 }
 | |
|                 dataFormat |= f.Item3;
 | |
|                 XLib.XConvertSelection(x11Info.Display, x11Info.Atoms.XdndSelection, atom,
 | |
|                    f.Item2, TargetWindow, IntPtr.Zero /* CurrentTime */);
 | |
|             }
 | |
| 
 | |
|             if (cancelHandle != null)
 | |
|             {
 | |
|                 cancelHandle.Cancel = true;
 | |
|             }
 | |
|             cancelHandle = new CancelHandle();
 | |
|             LinuxPlatform.Platform.RunMainLoop(cancelHandle, OnAcceptEvent);
 | |
|             if (cancelHandle != null)
 | |
|             {
 | |
|                 if (timer == null)
 | |
|                 {
 | |
|                     timer = new Threading.DispatcherTimer { Interval = TimeSpan.FromMilliseconds(100) };
 | |
|                     timer.Tick += eventHandler;
 | |
|                 }
 | |
|                 timer.Start();
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public bool OnAcceptEvent(ref XEvent xEvent)
 | |
|         {
 | |
|             if (xEvent.type == XEventName.SelectionNotify)
 | |
|             {
 | |
|                 var df = dataFormat;
 | |
|                 while (true)
 | |
|                 {
 | |
|                     if (df == DataFormat.Unknown)
 | |
|                     {
 | |
|                         break;
 | |
|                     }
 | |
|                     if (df.HasFlag(DataFormat.Text))
 | |
|                     {
 | |
|                         df ^= DataFormat.Text;
 | |
|                         text = GetText(ref xEvent);
 | |
|                     }
 | |
|                     if (df.HasFlag(DataFormat.Html))
 | |
|                     {
 | |
|                         df ^= DataFormat.Html;
 | |
|                         html = GetText(ref xEvent);
 | |
|                     }
 | |
|                     if (df.HasFlag(DataFormat.FileNames))
 | |
|                     {
 | |
|                         if (text == null)
 | |
|                         {
 | |
|                             text = GetText(ref xEvent);
 | |
|                         }
 | |
|                         df ^= DataFormat.FileNames;
 | |
|                         List<string> uri_list = new List<string>();
 | |
|                         string[] lines = text.Split(new char[] { '\r', '\n' });
 | |
|                         foreach (string line in lines)
 | |
|                         {
 | |
|                             // # is a comment line (see RFC 2483)
 | |
|                             if (line.StartsWith("#"))
 | |
|                                 continue;
 | |
|                             try
 | |
|                             {
 | |
|                                 Uri uri = new Uri(line);
 | |
|                                 uri_list.Add(uri.LocalPath);
 | |
|                             }
 | |
|                             catch { }
 | |
|                         }
 | |
| 
 | |
|                         fileNames = uri_list;
 | |
|                         //string[] l = (string[])uri_list.ToArray(typeof(string));
 | |
|                     }
 | |
|                 }
 | |
|                 if (cancelHandle != null)
 | |
|                 {
 | |
|                     cancelHandle.Cancel = true;
 | |
|                 }
 | |
|                 return true;
 | |
|             }
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|         public unsafe string GetText(ref XEvent xevent)
 | |
|         {
 | |
|             int nread = 0;
 | |
|             IntPtr nitems;
 | |
|             IntPtr bytes_after;
 | |
| 
 | |
|             StringBuilder builder = new StringBuilder();
 | |
|             do
 | |
|             {
 | |
|                 IntPtr actual_type;
 | |
|                 int actual_fmt;
 | |
|                 IntPtr data = IntPtr.Zero;
 | |
| 
 | |
|                 if (0 != XLib.XGetWindowProperty(x11Info.Display,
 | |
|                             xevent.AnyEvent.window,
 | |
|                             (IntPtr)xevent.SelectionEvent.property,
 | |
|                             IntPtr.Zero, new IntPtr(0xffffff), false,
 | |
|                             (IntPtr)Atom.AnyPropertyType, out actual_type,
 | |
|                             out actual_fmt, out nitems, out bytes_after,
 | |
|                             out data))
 | |
|                 {
 | |
|                     XLib.XFree(data);
 | |
|                     break;
 | |
|                 }
 | |
| 
 | |
|                 //if (unicode)
 | |
|                 //    builder.Append(Marshal.PtrToStringUni(data));
 | |
|                 //else
 | |
|                 //    builder.Append(Marshal.PtrToStringAnsi(data));
 | |
| 
 | |
|                 Encoding textEnc = GetStringEncoding(xevent.SelectionEvent.property);
 | |
|                 if (textEnc != null)
 | |
|                 {
 | |
|                     var text = textEnc.GetString((byte*)data.ToPointer(), nitems.ToInt32());
 | |
|                     builder.Append(text);
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     builder.Append(Marshal.PtrToStringAnsi(data));
 | |
|                 }
 | |
| 
 | |
|                 nread += nitems.ToInt32();
 | |
| 
 | |
|                 XLib.XFree(data);
 | |
|             } while (bytes_after.ToInt32() > 0);
 | |
|             if (nread == 0)
 | |
|                 return null;
 | |
|             return builder.ToString();
 | |
|         }
 | |
|         Encoding GetStringEncoding(IntPtr atom)
 | |
|         {
 | |
|             return (atom == x11Info.Atoms.XA_STRING || atom == x11Info.Atoms.OEMTEXT)
 | |
|                 ? Encoding.ASCII
 | |
|                 : (atom == x11Info.Atoms.UTF8_STRING || atom == textUtf8Atom)
 | |
|                     ? Encoding.UTF8
 | |
|                     : (atom == x11Info.Atoms.UTF16_STRING || atom == x11Info.Atoms.UNICODETEXT || atomNonProtocols.First(a => a.Item1 == htmlAtom).Item2 == atom)
 | |
|                         ? Encoding.Unicode
 | |
|                         : null;
 | |
|         }
 | |
| 
 | |
|         public void SendStatus(IntPtr source, DragDropEffects effect)
 | |
|         {
 | |
|             XEvent xevent = new XEvent();
 | |
| 
 | |
|             xevent.AnyEvent.type = XEventName.ClientMessage;
 | |
|             xevent.AnyEvent.display = x11Info.Display;
 | |
|             xevent.ClientMessageEvent.window = source;
 | |
|             xevent.ClientMessageEvent.message_type = x11Info.Atoms.XdndStatus;
 | |
|             xevent.ClientMessageEvent.format = 32;
 | |
|             xevent.ClientMessageEvent.ptr1 = TargetWindow;
 | |
|             if (effect != DragDropEffects.None)
 | |
|                 xevent.ClientMessageEvent.ptr2 = (IntPtr)1;
 | |
| 
 | |
|             xevent.ClientMessageEvent.ptr5 = ActionFromEffect(effect);
 | |
|             XLib.XSendEvent(x11Info.Display, source, false, IntPtr.Zero, ref xevent);
 | |
|             //XLib.XSync(x11Info.Display,false);
 | |
|             Console.WriteLine("state:" + effect);
 | |
|         }
 | |
| 
 | |
|         private void GiveFeedback(IntPtr action)
 | |
|         {
 | |
|             //GiveFeedbackEventArgs gfe = new GiveFeedbackEventArgs(EffectFromAction(drag_data.Action), true);
 | |
| 
 | |
|             //Control c = MwfWindow(source);
 | |
|             //c.DndFeedback(gfe);
 | |
| 
 | |
|             //if (gfe.UseDefaultCursors)
 | |
|             {
 | |
|                 Cursor cursor = Cursors.No;
 | |
|                 if (WillAccept)
 | |
|                 {
 | |
|                     // Same order as on MS
 | |
|                     if (action == x11Info.Atoms.XdndActionCopy)
 | |
|                         cursor = Cursors.DragCopy;
 | |
|                     else if (action == x11Info.Atoms.XdndActionLink)
 | |
|                         cursor = Cursors.DragLink;
 | |
|                     else if (action == x11Info.Atoms.XdndActionMove)
 | |
|                         cursor = Cursors.DragMove;
 | |
|                 }
 | |
|                 // TODO: Try not to set the cursor so much
 | |
|                 //if (cursor.Handle != CurrentCursorHandle) {
 | |
|                 XLib.XChangeActivePointerGrab(x11Info.Display,
 | |
|                         EventMask.ButtonMotionMask |
 | |
|                         EventMask.PointerMotionMask |
 | |
|                         EventMask.ButtonPressMask |
 | |
|                         EventMask.ButtonReleaseMask,
 | |
|                         (IntPtr)cursor.PlatformCursor, IntPtr.Zero);
 | |
|                 //CurrentCursorHandle = cursor.Handle;
 | |
|                 //}
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public IntPtr[] SourceSupportedList(ref XEvent xevent)
 | |
|         {
 | |
|             IntPtr[] res;
 | |
| 
 | |
| 
 | |
|             if (((int)xevent.ClientMessageEvent.ptr2 & 0x1) == 0)
 | |
|             {
 | |
|                 res = new IntPtr[3];
 | |
|                 res[0] = xevent.ClientMessageEvent.ptr3;
 | |
|                 res[1] = xevent.ClientMessageEvent.ptr4;
 | |
|                 res[2] = xevent.ClientMessageEvent.ptr5;
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 IntPtr type;
 | |
|                 int format;
 | |
|                 IntPtr count;
 | |
|                 IntPtr remaining;
 | |
|                 IntPtr data;
 | |
|                 XLib.XGetWindowProperty(x11Info.Display, SourceWindow, x11Info.Atoms.XdndTypeList,
 | |
|                         IntPtr.Zero, new IntPtr(32), false, (IntPtr)Atom.XA_ATOM,
 | |
|                         out type, out format, out count,
 | |
|                         out remaining, out data);
 | |
| 
 | |
|                 res = new IntPtr[count.ToInt32()];
 | |
|                 for (int i = 0; i < count.ToInt32(); i++)
 | |
|                 {
 | |
|                     res[i] = (IntPtr)Marshal.ReadInt32(data, i *
 | |
|                             Marshal.SizeOf(typeof(IntPtr)));
 | |
|                 }
 | |
| 
 | |
|                 XLib.XFree(data);
 | |
|             }
 | |
| 
 | |
|             return res;
 | |
|         }
 | |
| 
 | |
|         public void SetTextData(object data, ref XEvent xevent)
 | |
|         {
 | |
|             IntPtr buffer;
 | |
|             int len;
 | |
|             string str = data as string;
 | |
| 
 | |
|             //if (str == null)
 | |
|             //{
 | |
|             //    IDataObject dobj = data as IDataObject;
 | |
|             //    if (dobj == null)
 | |
|             //        return;
 | |
|             //    str = (string)dobj.GetData("System.String", true);
 | |
|             //}
 | |
| 
 | |
|             if (xevent.SelectionRequestEvent.target == (IntPtr)Atom.XA_STRING)
 | |
|             {
 | |
|                 byte[] bytes = Encoding.ASCII.GetBytes(str);
 | |
|                 buffer = Marshal.AllocHGlobal(bytes.Length);
 | |
|                 len = bytes.Length;
 | |
|                 for (int i = 0; i < len; i++)
 | |
|                     Marshal.WriteByte(buffer, i, bytes[i]);
 | |
|             }
 | |
|             else if (xevent.SelectionRequestEvent.target == x11Info.Atoms.UTF8_STRING || xevent.SelectionRequestEvent.target == textUtf8Atom)
 | |
|             {
 | |
|                 byte[] bytes = Encoding.UTF8.GetBytes(str);
 | |
|                 buffer = Marshal.AllocHGlobal(bytes.Length);
 | |
|                 len = bytes.Length;
 | |
|                 for (int i = 0; i < len; i++)
 | |
|                     Marshal.WriteByte(buffer, i, bytes[i]);
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 buffer = Marshal.StringToHGlobalAnsi(str);
 | |
|                 len = 0;
 | |
|                 while (Marshal.ReadByte(buffer, len) != 0)
 | |
|                     len++;
 | |
|             }
 | |
| 
 | |
|             SetProperty(ref xevent, buffer, len);
 | |
| 
 | |
|             Marshal.FreeHGlobal(buffer);
 | |
|         }
 | |
|         public void SetHtmlData(object data, ref XEvent xevent)
 | |
|         {
 | |
|             IntPtr buffer;
 | |
|             int len;
 | |
|             string str = data as string;
 | |
| 
 | |
|             if (str == null)
 | |
|                 return;
 | |
| 
 | |
|             if (xevent.SelectionRequestEvent.target == (IntPtr)Atom.XA_STRING)
 | |
|             {
 | |
|                 byte[] bytes = Encoding.ASCII.GetBytes(str);
 | |
|                 buffer = Marshal.AllocHGlobal(bytes.Length);
 | |
|                 len = bytes.Length;
 | |
|                 for (int i = 0; i < len; i++)
 | |
|                     Marshal.WriteByte(buffer, i, bytes[i]);
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 buffer = Marshal.StringToHGlobalAnsi(str);
 | |
|                 len = 0;
 | |
|                 while (Marshal.ReadByte(buffer, len) != 0)
 | |
|                     len++;
 | |
|             }
 | |
| 
 | |
|             SetProperty(ref xevent, buffer, len);
 | |
| 
 | |
|             Marshal.FreeHGlobal(buffer);
 | |
|         }
 | |
| 
 | |
|         public void SetFileNameData(object data, ref XEvent xevent)
 | |
|         {
 | |
|             var uri_list = data as IEnumerable<string>;
 | |
| 
 | |
|             //if (uri_list == null)
 | |
|             //{
 | |
|             //    IDataObject dobj = data as IDataObject;
 | |
|             //    if (dobj == null)
 | |
|             //        return;
 | |
|             //    uri_list = dobj.GetData(DataFormats.FileDrop, true) as string[];
 | |
|             //}
 | |
| 
 | |
|             if (uri_list == null)
 | |
|                 return;
 | |
| 
 | |
|             StringBuilder res = new StringBuilder();
 | |
|             foreach (string uri_str in uri_list)
 | |
|             {
 | |
|                 Uri uri = new Uri(uri_str);
 | |
|                 res.Append(uri.ToString());
 | |
|                 res.Append("\r\n");
 | |
|             }
 | |
| 
 | |
|             IntPtr buffer = Marshal.StringToHGlobalAnsi((string)res.ToString());
 | |
|             int len = 0;
 | |
|             while (Marshal.ReadByte(buffer, len) != 0)
 | |
|                 len++;
 | |
| 
 | |
|             SetProperty(ref xevent, buffer, len);
 | |
|         }
 | |
|         private void SetProperty(ref XEvent xevent, IntPtr data, int length)
 | |
|         {
 | |
|             XEvent sel = new XEvent();
 | |
|             sel.SelectionEvent.type = XEventName.SelectionNotify;
 | |
|             sel.SelectionEvent.send_event = true;
 | |
|             sel.SelectionEvent.display = x11Info.Display;
 | |
|             sel.SelectionEvent.selection = xevent.SelectionRequestEvent.selection;
 | |
|             sel.SelectionEvent.target = xevent.SelectionRequestEvent.target;
 | |
|             sel.SelectionEvent.requestor = xevent.SelectionRequestEvent.requestor;
 | |
|             sel.SelectionEvent.time = xevent.SelectionRequestEvent.time;
 | |
|             sel.SelectionEvent.property = xevent.SelectionRequestEvent.property;
 | |
| 
 | |
|             XLib.XChangeProperty(x11Info.Display, xevent.SelectionRequestEvent.requestor,
 | |
|                     xevent.SelectionRequestEvent.property,
 | |
|                     xevent.SelectionRequestEvent.target,
 | |
|                     8, PropertyMode.Replace, data, length);
 | |
| 
 | |
|             XLib.XSendEvent(x11Info.Display, xevent.SelectionRequestEvent.requestor, true,
 | |
|                     (IntPtr)EventMask.NoEventMask, ref sel);
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         public bool HandleMouseOver()
 | |
|         {
 | |
|             IntPtr toplevel, window;
 | |
|             int x_root, y_root;
 | |
| 
 | |
|             GetWindowsUnderPointer(out window, out toplevel, out x_root, out y_root);
 | |
| 
 | |
|             if (window != LastWindow && State == DragState.Entered)
 | |
|             {
 | |
|                 State = DragState.Dragging;
 | |
| 
 | |
|                 // TODO: Send a Leave if this is an MWF window
 | |
| 
 | |
|                 if (toplevel != TargetWindow)
 | |
|                     SendLeave(TargetWindow, toplevel);
 | |
|             }
 | |
| 
 | |
|             State = DragState.Entered;
 | |
|             if (toplevel != TargetWindow)
 | |
|             {
 | |
|                 if (cancelHandle == null || cancelHandle.Cancel)
 | |
|                 {
 | |
|                     return true;
 | |
|                 }
 | |
|                 // Entering a new toplevel window
 | |
|                 SendEnter(toplevel, SourceWindow, SupportedTypes);
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 // Already in a toplevel window, so send a position
 | |
|                 SendPosition(toplevel, SourceWindow,
 | |
|                         Action,
 | |
|                         x_root, y_root,
 | |
|                         IntPtr.Zero);
 | |
|             }
 | |
| 
 | |
|             TargetWindow = toplevel;
 | |
|             LastWindow = window;
 | |
|             return true;
 | |
|         }
 | |
|         void GetWindowsUnderPointer(out IntPtr window, out IntPtr toplevel, out int x_root, out int y_root)
 | |
|         {
 | |
|             toplevel = IntPtr.Zero;
 | |
|             window = x11Info.RootWindow;
 | |
| 
 | |
|             IntPtr root, child;
 | |
|             bool dnd_aware = false;
 | |
|             int x_temp, y_temp;
 | |
|             int mask_return;
 | |
|             int x = x_root = (int)point.X;
 | |
|             int y = y_root = (int)point.Y;
 | |
| 
 | |
|             while (XLib.XQueryPointer(x11Info.Display, window, out root, out child,
 | |
|                            out x_temp, out y_temp, out x, out y, out mask_return))
 | |
|             {
 | |
| 
 | |
|                 if (!dnd_aware)
 | |
|                 {
 | |
|                     dnd_aware = IsWindowDndAware(window);
 | |
|                     if (dnd_aware)
 | |
|                     {
 | |
|                         toplevel = window;
 | |
|                         x_root = x_temp;
 | |
|                         y_root = y_temp;
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 if (child == IntPtr.Zero)
 | |
|                     break;
 | |
| 
 | |
|                 window = child;
 | |
|             }
 | |
|         }
 | |
| 
 | |
| 
 | |
|         private void SendDrop(IntPtr handle, IntPtr from, IntPtr time)
 | |
|         {
 | |
|             XEvent xevent = new XEvent();
 | |
| 
 | |
|             xevent.AnyEvent.type = XEventName.ClientMessage;
 | |
|             xevent.AnyEvent.display = x11Info.Display;
 | |
|             xevent.ClientMessageEvent.window = handle;
 | |
|             xevent.ClientMessageEvent.message_type = x11Info.Atoms.XdndDrop;
 | |
|             xevent.ClientMessageEvent.format = 32;
 | |
|             xevent.ClientMessageEvent.ptr1 = from;
 | |
|             xevent.ClientMessageEvent.ptr3 = time;
 | |
| 
 | |
|             XLib.XSendEvent(x11Info.Display, handle, false, IntPtr.Zero, ref xevent);
 | |
|             //dropped = true;
 | |
|         }
 | |
| 
 | |
|         private void SendPosition(IntPtr handle, IntPtr from, IntPtr action, int x, int y, IntPtr time)
 | |
|         {
 | |
|             XEvent xevent = new XEvent();
 | |
| 
 | |
|             xevent.AnyEvent.type = XEventName.ClientMessage;
 | |
|             xevent.AnyEvent.display = x11Info.Display;
 | |
|             xevent.ClientMessageEvent.window = handle;
 | |
|             xevent.ClientMessageEvent.message_type = x11Info.Atoms.XdndPosition;
 | |
|             xevent.ClientMessageEvent.format = 32;
 | |
|             xevent.ClientMessageEvent.ptr1 = from;
 | |
|             xevent.ClientMessageEvent.ptr3 = (IntPtr)((x << 16) | (y & 0xFFFF));
 | |
|             xevent.ClientMessageEvent.ptr4 = time;
 | |
|             xevent.ClientMessageEvent.ptr5 = action;
 | |
| 
 | |
|             XLib.XSendEvent(x11Info.Display, handle, false, IntPtr.Zero, ref xevent);
 | |
|         }
 | |
| 
 | |
|         private void SendLeave(IntPtr handle, IntPtr from)
 | |
|         {
 | |
|             XEvent xevent = new XEvent();
 | |
| 
 | |
|             xevent.AnyEvent.type = XEventName.ClientMessage;
 | |
|             xevent.AnyEvent.display = x11Info.Display;
 | |
|             xevent.ClientMessageEvent.window = handle;
 | |
|             xevent.ClientMessageEvent.message_type = x11Info.Atoms.XdndLeave;
 | |
|             xevent.ClientMessageEvent.format = 32;
 | |
|             xevent.ClientMessageEvent.ptr1 = from;
 | |
| 
 | |
|             XLib.XSendEvent(x11Info.Display, handle, false, IntPtr.Zero, ref xevent);
 | |
|         }
 | |
| 
 | |
|         public void SendFinished()
 | |
|         {
 | |
|             XEvent xevent = new XEvent();
 | |
| 
 | |
|             xevent.AnyEvent.type = XEventName.ClientMessage;
 | |
|             xevent.AnyEvent.display = x11Info.Display;
 | |
|             xevent.ClientMessageEvent.window = SourceWindow;
 | |
|             xevent.ClientMessageEvent.message_type = x11Info.Atoms.XdndFinished;
 | |
|             xevent.ClientMessageEvent.format = 32;
 | |
|             xevent.ClientMessageEvent.ptr1 = TargetWindow;
 | |
| 
 | |
|             XLib.XSendEvent(x11Info.Display, SourceWindow, false, IntPtr.Zero, ref xevent);
 | |
|             XLib.XFlush(x11Info.Display);
 | |
|         }
 | |
| 
 | |
|         private bool IsWindowDndAware(IntPtr handle)
 | |
|         {
 | |
|             bool res = true;
 | |
|             // Check the version number, we need greater than 3
 | |
|             IntPtr actual;
 | |
|             int format;
 | |
|             IntPtr count;
 | |
|             IntPtr remaining;
 | |
|             IntPtr data = IntPtr.Zero;
 | |
| 
 | |
|             XLib.XGetWindowProperty(x11Info.Display, handle, x11Info.Atoms.XdndAware, IntPtr.Zero, new IntPtr(0x8000000), false,
 | |
|                     (IntPtr)Atom.XA_ATOM, out actual, out format,
 | |
|                     out count, out remaining, out data);
 | |
| 
 | |
|             if (actual != (IntPtr)Atom.XA_ATOM || format != 32 ||
 | |
|                     count.ToInt32() == 0 || data == IntPtr.Zero)
 | |
|             {
 | |
|                 if (data != IntPtr.Zero)
 | |
|                     XLib.XFree(data);
 | |
|                 return false;
 | |
|             }
 | |
| 
 | |
|             int version = Marshal.ReadInt32(data, 0);
 | |
| 
 | |
|             if (version < 3)
 | |
|             {
 | |
|                 Console.Error.WriteLine("XDND Version too old (" + version + ").");
 | |
|                 XLib.XFree(data);
 | |
|                 return false;
 | |
|             }
 | |
| 
 | |
|             // First type is actually the XDND version
 | |
|             if (count.ToInt32() > 1)
 | |
|             {
 | |
|                 res = false;
 | |
|                 for (int i = 1; i < count.ToInt32(); i++)
 | |
|                 {
 | |
|                     IntPtr type = (IntPtr)Marshal.ReadInt32(data, i *
 | |
|                             Marshal.SizeOf(typeof(int)));
 | |
|                     for (int j = 0; j < SupportedTypes.Length; j++)
 | |
|                     {
 | |
|                         if (SupportedTypes[j] == type)
 | |
|                         {
 | |
|                             res = true;
 | |
|                             break;
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             XLib.XFree(data);
 | |
|             return res;
 | |
|         }
 | |
|         private enum DragState
 | |
|         {
 | |
|             None,
 | |
|             Beginning,
 | |
|             Dragging,
 | |
|             Entered
 | |
|         }
 | |
| 
 | |
|     }
 | |
| }
 | 
