/* * X11 event driver * * Copyright 1993 Alexandre Julliard * 1999 Noel Borthwick */ #include "config.h" #include <X11/Xatom.h> #include <X11/keysym.h> #include "ts_xlib.h" #include "ts_xresource.h" #include "ts_xutil.h" #ifdef HAVE_LIBXXSHM #include "ts_xshm.h" #endif #ifdef HAVE_LIBXXF86DGA2 #include "ts_xf86dga2.h" #endif #include <assert.h> #include <string.h> #include "wine/winuser16.h" #include "shlobj.h" /* DROPFILES */ #include "clipboard.h" #include "dce.h" #include "debugtools.h" #include "drive.h" #include "heap.h" #include "input.h" #include "keyboard.h" #include "message.h" #include "mouse.h" #include "options.h" #include "queue.h" #include "win.h" #include "winpos.h" #include "services.h" #include "file.h" #include "windef.h" #include "x11drv.h" DEFAULT_DEBUG_CHANNEL(event); DECLARE_DEBUG_CHANNEL(win); /* X context to associate a hwnd to an X window */ extern XContext winContext; extern Atom wmProtocols; extern Atom wmDeleteWindow; extern Atom dndProtocol; extern Atom dndSelection; extern void X11DRV_KEYBOARD_UpdateState(void); extern void X11DRV_KEYBOARD_HandleEvent(WND *pWnd, XKeyEvent *event); #define NB_BUTTONS 5 /* Windows can handle 3 buttons and the wheel too */ #define DndNotDnd -1 /* OffiX drag&drop */ #define DndUnknown 0 #define DndRawData 1 #define DndFile 2 #define DndFiles 3 #define DndText 4 #define DndDir 5 #define DndLink 6 #define DndExe 7 #define DndEND 8 #define DndURL 128 /* KDE drag&drop */ /* The last X window which had the focus */ static Window glastXFocusWin = 0; static const char * const event_names[] = { "", "", "KeyPress", "KeyRelease", "ButtonPress", "ButtonRelease", "MotionNotify", "EnterNotify", "LeaveNotify", "FocusIn", "FocusOut", "KeymapNotify", "Expose", "GraphicsExpose", "NoExpose", "VisibilityNotify", "CreateNotify", "DestroyNotify", "UnmapNotify", "MapNotify", "MapRequest", "ReparentNotify", "ConfigureNotify", "ConfigureRequest", "GravityNotify", "ResizeRequest", "CirculateNotify", "CirculateRequest", "PropertyNotify", "SelectionClear", "SelectionRequest", "SelectionNotify", "ColormapNotify", "ClientMessage", "MappingNotify" }; static void CALLBACK EVENT_Flush( ULONG_PTR arg ); static void CALLBACK EVENT_ProcessAllEvents( ULONG_PTR arg ); static void EVENT_ProcessEvent( XEvent *event ); BOOL X11DRV_CheckFocus(void); /* Event handlers */ static void EVENT_Key( HWND hWnd, XKeyEvent *event ); static void EVENT_ButtonPress( HWND hWnd, XButtonEvent *event ); static void EVENT_ButtonRelease( HWND hWnd, XButtonEvent *event ); static void EVENT_MotionNotify( HWND hWnd, XMotionEvent *event ); static void EVENT_FocusIn( HWND hWnd, XFocusChangeEvent *event ); static void EVENT_FocusOut( HWND hWnd, XFocusChangeEvent *event ); static void EVENT_Expose( HWND hWnd, XExposeEvent *event ); static void EVENT_GraphicsExpose( HWND hWnd, XGraphicsExposeEvent *event ); static void EVENT_ConfigureNotify( HWND hWnd, XConfigureEvent *event ); static void EVENT_SelectionRequest( HWND hWnd, XSelectionRequestEvent *event, BOOL bIsMultiple ); static void EVENT_SelectionClear( HWND hWnd, XSelectionClearEvent *event); static void EVENT_PropertyNotify( XPropertyEvent *event ); static void EVENT_ClientMessage( HWND hWnd, XClientMessageEvent *event ); static void EVENT_MapNotify( HWND pWnd, XMapEvent *event ); static void EVENT_UnmapNotify( HWND pWnd, XUnmapEvent *event ); static void EVENT_MappingNotify( XMappingEvent *event ); #ifdef HAVE_LIBXXSHM static void EVENT_ShmCompletion( XShmCompletionEvent *event ); static int ShmAvailable, ShmCompletionType; extern int XShmGetEventBase( Display * );/* Missing prototype for function in libXext. */ #endif #ifdef HAVE_LIBXXF86DGA2 static int DGAMotionEventType; static int DGAButtonPressEventType; static int DGAButtonReleaseEventType; static int DGAKeyPressEventType; static int DGAKeyReleaseEventType; static BOOL DGAUsed = FALSE; static HWND DGAhwnd = 0; static void EVENT_DGAMotionEvent( XDGAMotionEvent *event ); static void EVENT_DGAButtonPressEvent( XDGAButtonEvent *event ); static void EVENT_DGAButtonReleaseEvent( XDGAButtonEvent *event ); #endif /* Usable only with OLVWM - compile option perhaps? static void EVENT_EnterNotify( HWND hWnd, XCrossingEvent *event ); */ static void EVENT_GetGeometry( Window win, int *px, int *py, unsigned int *pwidth, unsigned int *pheight ); static BOOL bUserRepaintDisabled = TRUE; /* Static used for the current input method */ static INPUT_TYPE current_input_type = X11DRV_INPUT_ABSOLUTE; static BOOL in_transition = FALSE; /* This is not used as for today */ /*********************************************************************** * EVENT_Init */ BOOL X11DRV_EVENT_Init(void) { #ifdef HAVE_LIBXXSHM ShmAvailable = XShmQueryExtension( display ); if (ShmAvailable) { ShmCompletionType = XShmGetEventBase( display ) + ShmCompletion; } #endif /* Install the X event processing callback */ SERVICE_AddObject( FILE_DupUnixHandle( ConnectionNumber(display), GENERIC_READ | SYNCHRONIZE ), EVENT_ProcessAllEvents, 0 ); /* Install the XFlush timer callback */ if ( Options.synchronous ) TSXSynchronize( display, True ); else SERVICE_AddTimer( 200, EVENT_Flush, 0 ); return TRUE; } /*********************************************************************** * EVENT_Flush */ static void CALLBACK EVENT_Flush( ULONG_PTR arg ) { TSXFlush( display ); } /*********************************************************************** * EVENT_ProcessAllEvents */ static void CALLBACK EVENT_ProcessAllEvents( ULONG_PTR arg ) { XEvent event; TRACE( "called (thread %lx).\n", GetCurrentThreadId() ); EnterCriticalSection( &X11DRV_CritSection ); while ( XPending( display ) ) { XNextEvent( display, &event ); LeaveCriticalSection( &X11DRV_CritSection ); EVENT_ProcessEvent( &event ); EnterCriticalSection( &X11DRV_CritSection ); } LeaveCriticalSection( &X11DRV_CritSection ); } /*********************************************************************** * X11DRV_Synchronize * * Synchronize with the X server. Should not be used too often. */ void X11DRV_Synchronize( void ) { TSXSync( display, False ); EVENT_ProcessAllEvents( 0 ); } /*********************************************************************** * X11DRV_UserRepaintDisable */ void X11DRV_UserRepaintDisable( BOOL bDisabled ) { bUserRepaintDisabled = bDisabled; } /*********************************************************************** * EVENT_ProcessEvent * * Process an X event. */ static void EVENT_ProcessEvent( XEvent *event ) { HWND hWnd; TRACE( "called.\n" ); switch (event->type) { case SelectionNotify: /* all of these should be caught by XCheckTypedWindowEvent() */ FIXME("Got SelectionNotify - must not happen!\n"); /* fall through */ /* We get all these because of StructureNotifyMask. This check is placed here to avoid getting error messages below, as X might send some of these even for windows that have already been deleted ... */ case CirculateNotify: case CreateNotify: case DestroyNotify: case GravityNotify: case ReparentNotify: return; } #ifdef HAVE_LIBXXSHM if (ShmAvailable && (event->type == ShmCompletionType)) { EVENT_ShmCompletion( (XShmCompletionEvent*)event ); return; } #endif #ifdef HAVE_LIBXXF86DGA2 if (DGAUsed) { if (event->type == DGAMotionEventType) { TRACE("DGAMotionEvent received.\n"); EVENT_DGAMotionEvent((XDGAMotionEvent *) event); return; } if (event->type == DGAButtonPressEventType) { TRACE("DGAButtonPressEvent received.\n"); EVENT_DGAButtonPressEvent((XDGAButtonEvent *) event); return; } if (event->type == DGAButtonReleaseEventType) { TRACE("DGAButtonReleaseEvent received.\n"); EVENT_DGAButtonReleaseEvent((XDGAButtonEvent *) event); return; } if ((event->type == DGAKeyPressEventType) || (event->type == DGAKeyReleaseEventType)) { /* Fill a XKeyEvent to send to EVENT_Key */ XKeyEvent ke; XDGAKeyEvent *evt = (XDGAKeyEvent *) event; TRACE("DGAKeyPress/ReleaseEvent received.\n"); if (evt->type == DGAKeyReleaseEventType) ke.type = KeyRelease; else ke.type = KeyPress; ke.serial = evt->serial; ke.send_event = FALSE; ke.display = evt->display; ke.window = 0; ke.root = 0; ke.subwindow = 0; ke.time = evt->time; ke.x = PosX; ke.y = PosY; ke.x_root = -1; ke.y_root = -1; ke.state = evt->state; ke.keycode = evt->keycode; ke.same_screen = TRUE; X11DRV_KEYBOARD_HandleEvent(NULL, &ke); return; } } #endif if ( TSXFindContext( display, event->xany.window, winContext, (char **)&hWnd ) != 0) { if ( event->type == ClientMessage) { /* query window (drag&drop event contains only drag window) */ Window root, child; int root_x, root_y, child_x, child_y; unsigned u; TSXQueryPointer( display, X11DRV_GetXRootWindow(), &root, &child, &root_x, &root_y, &child_x, &child_y, &u); if (TSXFindContext( display, child, winContext, (char **)&hWnd ) != 0) return; } else { hWnd = 0; /* Not for a registered window */ } } if ( !hWnd && event->xany.window != X11DRV_GetXRootWindow() && event->type != PropertyNotify && event->type != MappingNotify) ERR("Got event %s for unknown Window %08lx\n", event_names[event->type], event->xany.window ); else TRACE("Got event %s for hwnd %04x\n", event_names[event->type], hWnd ); switch(event->type) { case KeyPress: case KeyRelease: EVENT_Key( hWnd, (XKeyEvent*)event ); break; case ButtonPress: EVENT_ButtonPress( hWnd, (XButtonEvent*)event ); break; case ButtonRelease: EVENT_ButtonRelease( hWnd, (XButtonEvent*)event ); break; case MotionNotify: /* Wine between two fast machines across the overloaded campus ethernet gets very boged down in MotionEvents. The following simply finds the last motion event in the queue and drops the rest. On a good link events are servered before they build up so this doesn't take place. On a slow link this may cause problems if the event order is important. I'm not yet seen of any problems. Jon 7/6/96. */ if ((current_input_type == X11DRV_INPUT_ABSOLUTE) && (in_transition == FALSE)) /* Only cumulate events if in absolute mode */ while (TSXCheckTypedWindowEvent(display,((XAnyEvent *)event)->window, MotionNotify, event)); EVENT_MotionNotify( hWnd, (XMotionEvent*)event ); break; case FocusIn: { WND *pWndLastFocus = 0; XWindowAttributes win_attr; BOOL bIsDisabled; XFocusChangeEvent *xfocChange = (XFocusChangeEvent*)event; if (!hWnd || bUserRepaintDisabled) return; bIsDisabled = GetWindowLongA( hWnd, GWL_STYLE ) & WS_DISABLED; /* If the window has been disabled and we are in managed mode, * revert the X focus back to the last focus window. This is to disallow * the window manager from switching focus away while the app is * in a modal state. */ if ( Options.managed && bIsDisabled && glastXFocusWin) { /* Change focus only if saved focus window is registered and viewable */ if ( TSXFindContext( xfocChange->display, glastXFocusWin, winContext, (char **)&pWndLastFocus ) == 0 ) { if ( TSXGetWindowAttributes( display, glastXFocusWin, &win_attr ) && (win_attr.map_state == IsViewable) ) { TSXSetInputFocus( xfocChange->display, glastXFocusWin, RevertToParent, CurrentTime ); EVENT_Synchronize(); break; } } } EVENT_FocusIn( hWnd, xfocChange ); break; } case FocusOut: { /* Save the last window which had the focus */ XFocusChangeEvent *xfocChange = (XFocusChangeEvent*)event; glastXFocusWin = xfocChange->window; if (!hWnd || bUserRepaintDisabled) return; if (GetWindowLongA( hWnd, GWL_STYLE ) & WS_DISABLED) glastXFocusWin = 0; EVENT_FocusOut( hWnd, (XFocusChangeEvent*)event ); break; } case Expose: if (bUserRepaintDisabled) return; EVENT_Expose( hWnd, (XExposeEvent *)event ); break; case GraphicsExpose: if (bUserRepaintDisabled) return; EVENT_GraphicsExpose( hWnd, (XGraphicsExposeEvent *)event ); break; case ConfigureNotify: if (!hWnd || bUserRepaintDisabled) return; EVENT_ConfigureNotify( hWnd, (XConfigureEvent*)event ); break; case SelectionRequest: if (!hWnd || bUserRepaintDisabled) return; EVENT_SelectionRequest( hWnd, (XSelectionRequestEvent *)event, FALSE ); break; case SelectionClear: if (!hWnd || bUserRepaintDisabled) return; EVENT_SelectionClear( hWnd, (XSelectionClearEvent*) event ); break; case PropertyNotify: EVENT_PropertyNotify( (XPropertyEvent *)event ); break; case ClientMessage: if (!hWnd || bUserRepaintDisabled) return; EVENT_ClientMessage( hWnd, (XClientMessageEvent *) event ); break; #if 0 case EnterNotify: EVENT_EnterNotify( hWnd, (XCrossingEvent *) event ); break; #endif case NoExpose: break; case MapNotify: if (!hWnd || bUserRepaintDisabled) return; EVENT_MapNotify( hWnd, (XMapEvent *)event ); break; case UnmapNotify: if (!hWnd || bUserRepaintDisabled) return; EVENT_UnmapNotify( hWnd, (XUnmapEvent *)event ); break; case MappingNotify: EVENT_MappingNotify( (XMappingEvent *) event ); break; default: WARN("Unprocessed event %s for hwnd %04x\n", event_names[event->type], hWnd ); break; } TRACE( "returns.\n" ); } /*********************************************************************** * EVENT_QueryZOrder * * Synchronize internal z-order with the window manager's. */ static BOOL __check_query_condition( WND** pWndA, WND** pWndB ) { /* return TRUE if we have at least two managed windows */ for( *pWndB = NULL; *pWndA; *pWndA = (*pWndA)->next ) if( ((*pWndA)->dwExStyle & WS_EX_MANAGED) && ((*pWndA)->dwStyle & WS_VISIBLE )) break; if( *pWndA ) for( *pWndB = (*pWndA)->next; *pWndB; *pWndB = (*pWndB)->next ) if( ((*pWndB)->dwExStyle & WS_EX_MANAGED) && ((*pWndB)->dwStyle & WS_VISIBLE )) break; return ((*pWndB) != NULL); } static Window __get_common_ancestor( Window A, Window B, Window** children, unsigned* total ) { /* find the real root window */ Window root, *childrenB; unsigned totalB; do { TSXQueryTree( display, A, &root, &A, children, total ); TSXQueryTree( display, B, &root, &B, &childrenB, &totalB ); if( childrenB ) TSXFree( childrenB ); if( *children ) TSXFree( *children ), *children = NULL; } while( A != B && A && B ); if( A && B ) { TSXQueryTree( display, A, &root, &B, children, total ); return A; } return 0 ; } static Window __get_top_decoration( Window w, Window ancestor ) { Window* children, root, prev = w, parent = w; unsigned total; do { w = parent; TSXQueryTree( display, w, &root, &parent, &children, &total ); if( children ) TSXFree( children ); } while( parent && parent != ancestor ); TRACE("\t%08x -> %08x\n", (unsigned)prev, (unsigned)w ); return ( parent ) ? w : 0 ; } static unsigned __td_lookup( Window w, Window* list, unsigned max ) { unsigned i; for( i = max - 1; i >= 0; i-- ) if( list[i] == w ) break; return i; } static HWND EVENT_QueryZOrder( HWND hWndCheck) { HWND hwndInsertAfter = HWND_TOP; WND *pWndCheck = WIN_FindWndPtr(hWndCheck); WND *pDesktop = WIN_GetDesktop(); WND *pWnd, *pWndZ = WIN_LockWndPtr(pDesktop->child); Window w, parent, *children = NULL; unsigned total, check, pos, best; if( !__check_query_condition(&pWndZ, &pWnd) ) { WIN_ReleaseWndPtr(pWndCheck); WIN_ReleaseWndPtr(pDesktop->child); WIN_ReleaseDesktop(); return hwndInsertAfter; } WIN_LockWndPtr(pWndZ); WIN_LockWndPtr(pWnd); WIN_ReleaseWndPtr(pDesktop->child); WIN_ReleaseDesktop(); parent = __get_common_ancestor( X11DRV_WND_GetXWindow(pWndZ), X11DRV_WND_GetXWindow(pWnd), &children, &total ); if( parent && children ) { /* w is the ancestor if pWndCheck that is a direct descendant of 'parent' */ w = __get_top_decoration( X11DRV_WND_GetXWindow(pWndCheck), parent ); if( w != children[total-1] ) /* check if at the top */ { /* X child at index 0 is at the bottom, at index total-1 is at the top */ check = __td_lookup( w, children, total ); best = total; for( WIN_UpdateWndPtr(&pWnd,pWndZ); pWnd;WIN_UpdateWndPtr(&pWnd,pWnd->next)) { /* go through all windows in Wine z-order... */ if( pWnd != pWndCheck ) { if( !(pWnd->dwExStyle & WS_EX_MANAGED) || !(w = __get_top_decoration( X11DRV_WND_GetXWindow(pWnd), parent )) ) continue; pos = __td_lookup( w, children, total ); if( pos < best && pos > check ) { /* find a nearest Wine window precedes * pWndCheck in the real z-order... */ best = pos; hwndInsertAfter = pWnd->hwndSelf; } if( best - check == 1 ) break; } } } } if( children ) TSXFree( children ); WIN_ReleaseWndPtr(pWnd); WIN_ReleaseWndPtr(pWndZ); WIN_ReleaseWndPtr(pWndCheck); return hwndInsertAfter; } /*********************************************************************** * X11DRV_EVENT_XStateToKeyState * * Translate a X event state (Button1Mask, ShiftMask, etc...) to * a Windows key state (MK_SHIFT, MK_CONTROL, etc...) */ WORD X11DRV_EVENT_XStateToKeyState( int state ) { int kstate = 0; if (state & Button1Mask) kstate |= MK_LBUTTON; if (state & Button2Mask) kstate |= MK_MBUTTON; if (state & Button3Mask) kstate |= MK_RBUTTON; if (state & ShiftMask) kstate |= MK_SHIFT; if (state & ControlMask) kstate |= MK_CONTROL; return kstate; } /*********************************************************************** * EVENT_Expose */ static void EVENT_Expose( HWND hWnd, XExposeEvent *event ) { RECT rect; int offx = 0,offy = 0; WND *pWnd = WIN_FindWndPtr(hWnd); /* Make position relative to client area instead of window */ offx = (pWnd? (pWnd->rectClient.left - pWnd->rectWindow.left) : 0); offy = (pWnd? (pWnd->rectClient.top - pWnd->rectWindow.top) : 0); rect.left = event->x - offx; rect.top = event->y - offy; rect.right = rect.left + event->width; rect.bottom = rect.top + event->height; WIN_ReleaseWndPtr(pWnd); RedrawWindow( hWnd, &rect, 0, RDW_INVALIDATE | RDW_FRAME | RDW_ALLCHILDREN | RDW_ERASE ); /* FIXME: We should use SendNotifyMessage here, but this function is not implemented correctly, so for now we used SendMessage */ /*SendNotifyMessageA(hWnd,WM_SYNCPAINT, 0, 0);*/ if (event->count == 0) SendMessageA(hWnd,WM_SYNCPAINT, 0, 0); } /*********************************************************************** * EVENT_GraphicsExpose * * This is needed when scrolling area is partially obscured * by non-Wine X window. */ static void EVENT_GraphicsExpose( HWND hWnd, XGraphicsExposeEvent *event ) { RECT rect; int offx = 0,offy = 0; WND *pWnd = WIN_FindWndPtr(hWnd); /* Make position relative to client area instead of window */ offx = (pWnd? (pWnd->rectClient.left - pWnd->rectWindow.left) : 0); offy = (pWnd? (pWnd->rectClient.top - pWnd->rectWindow.top) : 0); rect.left = event->x - offx; rect.top = event->y - offy; rect.right = rect.left + event->width; rect.bottom = rect.top + event->height; WIN_ReleaseWndPtr(pWnd); RedrawWindow( hWnd, &rect, 0, RDW_INVALIDATE | RDW_ALLCHILDREN | RDW_ERASE ); /* FIXME: We should use SendNotifyMessage here, but this function is not implemented correctly, so for now we used SendMessage */ /*SendNotifyMessageA(hWnd,WM_SYNCPAINT, 0, 0);*/ if (event->count == 0) SendMessageA(hWnd,WM_SYNCPAINT, 0, 0); } /*********************************************************************** * EVENT_Key * * Handle a X key event */ static void EVENT_Key( HWND hWnd, XKeyEvent *event ) { WND *pWnd = WIN_FindWndPtr(hWnd); X11DRV_KEYBOARD_HandleEvent( pWnd, event ); WIN_ReleaseWndPtr(pWnd); } /*********************************************************************** * EVENT_MotionNotify */ static void EVENT_MotionNotify( HWND hWnd, XMotionEvent *event ) { if (current_input_type == X11DRV_INPUT_ABSOLUTE) { WND *pWnd = WIN_FindWndPtr(hWnd); int xOffset = pWnd? pWnd->rectWindow.left : 0; int yOffset = pWnd? pWnd->rectWindow.top : 0; WIN_ReleaseWndPtr(pWnd); X11DRV_SendEvent( MOUSEEVENTF_MOVE | MOUSEEVENTF_ABSOLUTE, xOffset + event->x, yOffset + event->y, X11DRV_EVENT_XStateToKeyState( event->state ), event->time - X11DRV_server_startticks, hWnd); } else { X11DRV_SendEvent( MOUSEEVENTF_MOVE, event->x_root, event->y_root, X11DRV_EVENT_XStateToKeyState( event->state ), event->time - X11DRV_server_startticks, hWnd); } } /*********************************************************************** * EVENT_ButtonPress */ static void EVENT_ButtonPress( HWND hWnd, XButtonEvent *event ) { static WORD statusCodes[NB_BUTTONS] = { MOUSEEVENTF_LEFTDOWN, MOUSEEVENTF_MIDDLEDOWN, MOUSEEVENTF_RIGHTDOWN, MOUSEEVENTF_WHEEL, MOUSEEVENTF_WHEEL}; int buttonNum = event->button - 1; WND *pWnd = WIN_FindWndPtr(hWnd); int xOffset = pWnd? pWnd->rectWindow.left : 0; int yOffset = pWnd? pWnd->rectWindow.top : 0; WORD keystate,wData = 0; WIN_ReleaseWndPtr(pWnd); if (buttonNum >= NB_BUTTONS) return; /* * Get the compatible keystate */ keystate = X11DRV_EVENT_XStateToKeyState( event->state ); /* * Make sure that the state of the button that was just * pressed is "down". */ switch (buttonNum) { case 0: keystate |= MK_LBUTTON; break; case 1: keystate |= MK_MBUTTON; break; case 2: keystate |= MK_RBUTTON; break; case 3: wData = WHEEL_DELTA; break; case 4: wData = -WHEEL_DELTA; break; } X11DRV_SendEvent( statusCodes[buttonNum], xOffset + event->x, yOffset + event->y, MAKEWPARAM(keystate,wData), event->time - X11DRV_server_startticks, hWnd); } /*********************************************************************** * EVENT_ButtonRelease */ static void EVENT_ButtonRelease( HWND hWnd, XButtonEvent *event ) { static WORD statusCodes[NB_BUTTONS] = { MOUSEEVENTF_LEFTUP, MOUSEEVENTF_MIDDLEUP, MOUSEEVENTF_RIGHTUP }; int buttonNum = event->button - 1; WND *pWnd = WIN_FindWndPtr(hWnd); int xOffset = pWnd? pWnd->rectWindow.left : 0; int yOffset = pWnd? pWnd->rectWindow.top : 0; WORD keystate; WIN_ReleaseWndPtr(pWnd); if (buttonNum >= NB_BUTTONS) return; /* * Get the compatible keystate */ keystate = X11DRV_EVENT_XStateToKeyState( event->state ); /* * Make sure that the state of the button that was just * released is "up". */ switch (buttonNum) { case 0: keystate &= ~MK_LBUTTON; break; case 1: keystate &= ~MK_MBUTTON; break; case 2: keystate &= ~MK_RBUTTON; break; default: return; } X11DRV_SendEvent( statusCodes[buttonNum], xOffset + event->x, yOffset + event->y, keystate, event->time - X11DRV_server_startticks, hWnd); } /********************************************************************** * EVENT_FocusIn */ static void EVENT_FocusIn( HWND hWnd, XFocusChangeEvent *event ) { if (event->detail != NotifyPointer) if (hWnd != GetForegroundWindow()) { SetForegroundWindow( hWnd ); X11DRV_KEYBOARD_UpdateState(); } } /********************************************************************** * EVENT_FocusOut * * Note: only top-level override-redirect windows get FocusOut events. */ static void EVENT_FocusOut( HWND hWnd, XFocusChangeEvent *event ) { if (event->detail != NotifyPointer) if (hWnd == GetForegroundWindow()) { SendMessageA( hWnd, WM_CANCELMODE, 0, 0 ); /* don't reset the foreground window, if the window who's getting the focus is a Wine window */ if (!X11DRV_CheckFocus()) { /* Abey : 6-Oct-99. Check again if the focus out window is the Foreground window, because in most cases the messages sent above must have already changed the foreground window, in which case we don't have to change the foreground window to 0 */ if (hWnd == GetForegroundWindow()) SetForegroundWindow( 0 ); } } } /********************************************************************** * X11DRV_CheckFocus */ BOOL X11DRV_CheckFocus(void) { HWND hWnd; Window xW; int state; TSXGetInputFocus(display, &xW, &state); if( xW == None || TSXFindContext(display, xW, winContext, (char **)&hWnd) ) return FALSE; return TRUE; } /********************************************************************** * EVENT_GetGeometry * * Helper function for ConfigureNotify handling. * Get the new geometry of a window relative to the root window. */ static void EVENT_GetGeometry( Window win, int *px, int *py, unsigned int *pwidth, unsigned int *pheight ) { Window root, top; int x, y, width, height, border, depth; EnterCriticalSection( &X11DRV_CritSection ); /* Get the geometry of the window */ XGetGeometry( display, win, &root, &x, &y, &width, &height, &border, &depth ); /* Translate the window origin to root coordinates */ XTranslateCoordinates( display, win, root, 0, 0, &x, &y, &top ); LeaveCriticalSection( &X11DRV_CritSection ); *px = x; *py = y; *pwidth = width; *pheight = height; } /********************************************************************** * EVENT_ConfigureNotify * * The ConfigureNotify event is only selected on top-level windows * when the -managed flag is used. */ static void EVENT_ConfigureNotify( HWND hWnd, XConfigureEvent *event ) { RECT rectWindow; int x, y, flags = 0; unsigned int width, height; HWND newInsertAfter, oldInsertAfter; /* Get geometry and Z-order according to X */ EVENT_GetGeometry( event->window, &x, &y, &width, &height ); newInsertAfter = EVENT_QueryZOrder( hWnd ); /* Get geometry and Z-order according to Wine */ /* * Needs to find the first Visible Window above the current one */ oldInsertAfter = hWnd; for (;;) { oldInsertAfter = GetWindow( oldInsertAfter, GW_HWNDPREV ); if (!oldInsertAfter) { oldInsertAfter = HWND_TOP; break; } if (GetWindowLongA( oldInsertAfter, GWL_STYLE ) & WS_VISIBLE) break; } /* Compare what has changed */ GetWindowRect( hWnd, &rectWindow ); if ( rectWindow.left == x && rectWindow.top == y ) flags |= SWP_NOMOVE; else TRACE_(win)( "%04x moving from (%d,%d) to (%d,%d)\n", hWnd, rectWindow.left, rectWindow.top, x, y ); if ( rectWindow.right - rectWindow.left == width && rectWindow.bottom - rectWindow.top == height ) flags |= SWP_NOSIZE; else TRACE_(win)( "%04x resizing from (%d,%d) to (%d,%d)\n", hWnd, rectWindow.right - rectWindow.left, rectWindow.bottom - rectWindow.top, width, height ); if ( newInsertAfter == oldInsertAfter ) flags |= SWP_NOZORDER; else TRACE_(win)( "%04x restacking from after %04x to after %04x\n", hWnd, oldInsertAfter, newInsertAfter ); /* If anything changed, call SetWindowPos */ if ( flags != (SWP_NOMOVE | SWP_NOSIZE | SWP_NOZORDER) ) SetWindowPos( hWnd, newInsertAfter, x, y, width, height, flags | SWP_NOACTIVATE | SWP_WINE_NOHOSTMOVE ); } /*********************************************************************** * EVENT_SelectionRequest_TARGETS * Service a TARGETS selection request event */ static Atom EVENT_SelectionRequest_TARGETS( Window requestor, Atom target, Atom rprop ) { Atom xaTargets = TSXInternAtom(display, "TARGETS", False); Atom* targets; Atom prop; UINT wFormat; unsigned long cTargets; BOOL bHavePixmap; int xRc; TRACE("Request for %s\n", TSXGetAtomName(display, target)); /* * Count the number of items we wish to expose as selection targets. * We include the TARGETS item, and a PIXMAP if we have CF_DIB or CF_BITMAP */ cTargets = CountClipboardFormats() + 1; if ( CLIPBOARD_IsPresent(CF_DIB) || CLIPBOARD_IsPresent(CF_BITMAP) ) cTargets++; /* Allocate temp buffer */ targets = (Atom*)HeapAlloc( GetProcessHeap(), 0, cTargets * sizeof(Atom)); if(targets == NULL) return None; /* Create TARGETS property list (First item in list is TARGETS itself) */ for ( targets[0] = xaTargets, cTargets = 1, wFormat = 0, bHavePixmap = FALSE; (wFormat = EnumClipboardFormats( wFormat )); ) { if ( (prop = X11DRV_CLIPBOARD_MapFormatToProperty(wFormat)) != None ) { /* Scan through what we have so far to avoid duplicates */ int i; BOOL bExists; for (i = 0, bExists = FALSE; i < cTargets; i++) { if (targets[i] == prop) { bExists = TRUE; break; } } if (!bExists) { targets[cTargets++] = prop; /* Add PIXMAP prop for bitmaps additionally */ if ( (wFormat == CF_DIB || wFormat == CF_BITMAP ) && !bHavePixmap ) { targets[cTargets++] = XA_PIXMAP; bHavePixmap = TRUE; } } } } if (TRACE_ON(event)) { int i; for ( i = 0; i < cTargets; i++) { if (targets[i]) { char *itemFmtName = TSXGetAtomName(display, targets[i]); TRACE("\tAtom# %d: Type %s\n", i, itemFmtName); TSXFree(itemFmtName); } } } /* Update the X property */ TRACE("\tUpdating property %s...", TSXGetAtomName(display, rprop)); /* We may want to consider setting the type to xaTargets instead, * in case some apps expect this instead of XA_ATOM */ xRc = TSXChangeProperty(display, requestor, rprop, XA_ATOM, 32, PropModeReplace, (unsigned char *)targets, cTargets); TRACE("(Rc=%d)\n", xRc); HeapFree( GetProcessHeap(), 0, targets ); return rprop; } /*********************************************************************** * EVENT_SelectionRequest_STRING * Service a STRING selection request event */ static Atom EVENT_SelectionRequest_STRING( Window requestor, Atom target, Atom rprop ) { HANDLE16 hText; LPSTR text; int size,i,j; char* lpstr = 0; char *itemFmtName; int xRc; /* * Map the requested X selection property type atom name to a * windows clipboard format ID. */ itemFmtName = TSXGetAtomName(display, target); TRACE("Request for %s (wFormat=%x %s)\n", itemFmtName, CF_TEXT, CLIPBOARD_GetFormatName(CF_TEXT)); TSXFree(itemFmtName); hText = GetClipboardData16(CF_TEXT); if ( !hText ) return None; text = GlobalLock16(hText); if (!text) return None; size = GlobalSize16(hText); /* remove carriage returns */ lpstr = (char*)HeapAlloc( GetProcessHeap(), 0, size-- ); if(lpstr == NULL) return None; for(i=0,j=0; i < size && text[i]; i++ ) { if( text[i] == '\r' && (text[i+1] == '\n' || text[i+1] == '\0') ) continue; lpstr[j++] = text[i]; } lpstr[j]='\0'; /* Update the X property */ TRACE("\tUpdating property %s...\n", TSXGetAtomName(display, rprop)); xRc = TSXChangeProperty(display, requestor, rprop, XA_STRING, 8, PropModeReplace, lpstr, j); TRACE("(Rc=%d)\n", xRc); GlobalUnlock16(hText); HeapFree( GetProcessHeap(), 0, lpstr ); return rprop; } /*********************************************************************** * EVENT_SelectionRequest_PIXMAP * Service a PIXMAP selection request event */ static Atom EVENT_SelectionRequest_PIXMAP( Window requestor, Atom target, Atom rprop ) { HANDLE hClipData = 0; Pixmap pixmap = 0; UINT wFormat; char * itemFmtName; int xRc; #if(0) XSetWindowAttributes win_attr; XWindowAttributes win_attr_src; #endif /* * Map the requested X selection property type atom name to a * windows clipboard format ID. */ itemFmtName = TSXGetAtomName(display, target); wFormat = X11DRV_CLIPBOARD_MapPropertyToFormat(itemFmtName); TRACE("Request for %s (wFormat=%x %s)\n", itemFmtName, wFormat, CLIPBOARD_GetFormatName( wFormat)); TSXFree(itemFmtName); hClipData = GetClipboardData(wFormat); if ( !hClipData ) { TRACE("Could not retrieve a Pixmap compatible format from clipboard!\n"); rprop = None; /* Fail the request */ goto END; } if (wFormat == CF_DIB) { HWND hwnd = GetOpenClipboardWindow(); HDC hdc = GetDC(hwnd); /* For convert from packed DIB to Pixmap */ pixmap = X11DRV_DIB_CreatePixmapFromDIB(hClipData, hdc); ReleaseDC(hdc, hwnd); } else if (wFormat == CF_BITMAP) { HWND hwnd = GetOpenClipboardWindow(); HDC hdc = GetDC(hwnd); pixmap = X11DRV_BITMAP_CreatePixmapFromBitmap(hClipData, hdc); ReleaseDC(hdc, hwnd); } else { FIXME("%s to PIXMAP conversion not yet implemented!\n", CLIPBOARD_GetFormatName(wFormat)); rprop = None; goto END; } TRACE("\tUpdating property %s on Window %ld with %s %ld...\n", TSXGetAtomName(display, rprop), (long)requestor, TSXGetAtomName(display, target), pixmap); /* Store the Pixmap handle in the property */ xRc = TSXChangeProperty(display, requestor, rprop, target, 32, PropModeReplace, (unsigned char *)&pixmap, 1); TRACE("(Rc=%d)\n", xRc); /* Enable the code below if you want to handle destroying Pixmap resources * in response to property notify events. Clients like XPaint don't * appear to be duplicating Pixmaps so they don't like us deleting, * the resource in response to the property being deleted. */ #if(0) /* Express interest in property notify events so that we can delete the * pixmap when the client deletes the property atom. */ xRc = TSXGetWindowAttributes(display, requestor, &win_attr_src); TRACE("Turning on PropertyChangeEvent notifications from window %ld\n", (long)requestor); win_attr.event_mask = win_attr_src.your_event_mask | PropertyChangeMask; TSXChangeWindowAttributes(display, requestor, CWEventMask, &win_attr); /* Register the Pixmap we created with the request property Atom. * When this property is destroyed we also destroy the Pixmap in * response to the PropertyNotify event. */ X11DRV_CLIPBOARD_RegisterPixmapResource( rprop, pixmap ); #endif END: return rprop; } /*********************************************************************** * EVENT_SelectionRequest_WCF * Service a Wine Clipboard Format selection request event. * For <WCF>* data types we simply copy the data to X without conversion. */ static Atom EVENT_SelectionRequest_WCF( Window requestor, Atom target, Atom rprop ) { HANDLE hClipData = 0; void* lpClipData; UINT wFormat; char * itemFmtName; int cBytes; int xRc; /* * Map the requested X selection property type atom name to a * windows clipboard format ID. */ itemFmtName = TSXGetAtomName(display, target); wFormat = X11DRV_CLIPBOARD_MapPropertyToFormat(itemFmtName); TRACE("Request for %s (wFormat=%x %s)\n", itemFmtName, wFormat, CLIPBOARD_GetFormatName( wFormat)); TSXFree(itemFmtName); hClipData = GetClipboardData16(wFormat); if( hClipData && (lpClipData = GlobalLock16(hClipData)) ) { cBytes = GlobalSize16(hClipData); TRACE("\tUpdating property %s, %d bytes...\n", TSXGetAtomName(display, rprop), cBytes); xRc = TSXChangeProperty(display, requestor, rprop, target, 8, PropModeReplace, (unsigned char *)lpClipData, cBytes); TRACE("(Rc=%d)\n", xRc); GlobalUnlock16(hClipData); } else { TRACE("\tCould not retrieve native format!\n"); rprop = None; /* Fail the request */ } return rprop; } /*********************************************************************** * EVENT_SelectionRequest_MULTIPLE * Service a MULTIPLE selection request event * rprop contains a list of (target,property) atom pairs. * The first atom names a target and the second names a property. * The effect is as if we have received a sequence of SelectionRequest events * (one for each atom pair) except that: * 1. We reply with a SelectionNotify only when all the requested conversions * have been performed. * 2. If we fail to convert the target named by an atom in the MULTIPLE property, * we replace the atom in the property by None. */ static Atom EVENT_SelectionRequest_MULTIPLE( HWND hWnd, XSelectionRequestEvent *pevent ) { Atom rprop; Atom atype=AnyPropertyType; int aformat; unsigned long remain; Atom* targetPropList=NULL; unsigned long cTargetPropList = 0; /* Atom xAtomPair = TSXInternAtom(display, "ATOM_PAIR", False); */ /* If the specified property is None the requestor is an obsolete client. * We support these by using the specified target atom as the reply property. */ rprop = pevent->property; if( rprop == None ) rprop = pevent->target; if (!rprop) goto END; /* Read the MULTIPLE property contents. This should contain a list of * (target,property) atom pairs. */ if(TSXGetWindowProperty(display, pevent->requestor, rprop, 0, 0x3FFF, False, AnyPropertyType, &atype,&aformat, &cTargetPropList, &remain, (unsigned char**)&targetPropList) != Success) TRACE("\tCouldn't read MULTIPLE property\n"); else { TRACE("\tType %s,Format %d,nItems %ld, Remain %ld\n", TSXGetAtomName(display, atype), aformat, cTargetPropList, remain); /* * Make sure we got what we expect. * NOTE: According to the X-ICCCM Version 2.0 documentation the property sent * in a MULTIPLE selection request should be of type ATOM_PAIR. * However some X apps(such as XPaint) are not compliant with this and return * a user defined atom in atype when XGetWindowProperty is called. * The data *is* an atom pair but is not denoted as such. */ if(aformat == 32 /* atype == xAtomPair */ ) { int i; /* Iterate through the ATOM_PAIR list and execute a SelectionRequest * for each (target,property) pair */ for (i = 0; i < cTargetPropList; i+=2) { char *targetName = TSXGetAtomName(display, targetPropList[i]); char *propName = TSXGetAtomName(display, targetPropList[i+1]); XSelectionRequestEvent event; TRACE("MULTIPLE(%d): Target='%s' Prop='%s'\n", i/2, targetName, propName); TSXFree(targetName); TSXFree(propName); /* We must have a non "None" property to service a MULTIPLE target atom */ if ( !targetPropList[i+1] ) { TRACE("\tMULTIPLE(%d): Skipping target with empty property!", i); continue; } /* Set up an XSelectionRequestEvent for this (target,property) pair */ memcpy( &event, pevent, sizeof(XSelectionRequestEvent) ); event.target = targetPropList[i]; event.property = targetPropList[i+1]; /* Fire a SelectionRequest, informing the handler that we are processing * a MULTIPLE selection request event. */ EVENT_SelectionRequest( hWnd, &event, TRUE ); } } /* Free the list of targets/properties */ TSXFree(targetPropList); } END: return rprop; } /*********************************************************************** * EVENT_SelectionRequest * Process an event selection request event. * The bIsMultiple flag is used to signal when EVENT_SelectionRequest is called * recursively while servicing a "MULTIPLE" selection target. * * Note: We only receive this event when WINE owns the X selection */ static void EVENT_SelectionRequest( HWND hWnd, XSelectionRequestEvent *event, BOOL bIsMultiple ) { XSelectionEvent result; Atom rprop = None; Window request = event->requestor; BOOL couldOpen = FALSE; Atom xaClipboard = TSXInternAtom(display, "CLIPBOARD", False); Atom xaTargets = TSXInternAtom(display, "TARGETS", False); Atom xaMultiple = TSXInternAtom(display, "MULTIPLE", False); /* * We can only handle the selection request if : * The selection is PRIMARY or CLIPBOARD, AND we can successfully open the clipboard. * Don't do these checks or open the clipboard while recursively processing MULTIPLE, * since this has been already done. */ if ( !bIsMultiple ) { if ( ( (event->selection != XA_PRIMARY) && (event->selection != xaClipboard) ) || !(couldOpen = OpenClipboard(hWnd)) ) goto END; } /* If the specified property is None the requestor is an obsolete client. * We support these by using the specified target atom as the reply property. */ rprop = event->property; if( rprop == None ) rprop = event->target; if(event->target == xaTargets) /* Return a list of all supported targets */ { /* TARGETS selection request */ rprop = EVENT_SelectionRequest_TARGETS( request, event->target, rprop ); } else if(event->target == xaMultiple) /* rprop contains a list of (target, property) atom pairs */ { /* MULTIPLE selection request */ rprop = EVENT_SelectionRequest_MULTIPLE( hWnd, event ); } else if(event->target == XA_STRING) /* treat CF_TEXT as Unix text */ { /* XA_STRING selection request */ rprop = EVENT_SelectionRequest_STRING( request, event->target, rprop ); } else if(event->target == XA_PIXMAP) /* Convert DIB's to Pixmaps */ { /* XA_PIXMAP selection request */ rprop = EVENT_SelectionRequest_PIXMAP( request, event->target, rprop ); } else if(event->target == XA_BITMAP) /* Convert DIB's to 1-bit Pixmaps */ { /* XA_BITMAP selection request - TODO: create a monochrome Pixmap */ rprop = EVENT_SelectionRequest_PIXMAP( request, XA_PIXMAP, rprop ); } else if(X11DRV_CLIPBOARD_IsNativeProperty(event->target)) /* <WCF>* */ { /* All <WCF> selection requests */ rprop = EVENT_SelectionRequest_WCF( request, event->target, rprop ); } else rprop = None; /* Don't support this format */ END: /* close clipboard only if we opened before */ if(couldOpen) CloseClipboard(); if( rprop == None) TRACE("\tRequest ignored\n"); /* reply to sender * SelectionNotify should be sent only at the end of a MULTIPLE request */ if ( !bIsMultiple ) { result.type = SelectionNotify; result.display = display; result.requestor = request; result.selection = event->selection; result.property = rprop; result.target = event->target; result.time = event->time; TRACE("Sending SelectionNotify event...\n"); TSXSendEvent(display,event->requestor,False,NoEventMask,(XEvent*)&result); } } /*********************************************************************** * EVENT_SelectionClear */ static void EVENT_SelectionClear( HWND hWnd, XSelectionClearEvent *event ) { Atom xaClipboard = TSXInternAtom(display, "CLIPBOARD", False); if (event->selection == XA_PRIMARY || event->selection == xaClipboard) X11DRV_CLIPBOARD_ReleaseSelection( event->selection, event->window, hWnd ); } /*********************************************************************** * EVENT_PropertyNotify * We use this to release resources like Pixmaps when a selection * client no longer needs them. */ static void EVENT_PropertyNotify( XPropertyEvent *event ) { /* Check if we have any resources to free */ TRACE("Received PropertyNotify event: "); switch(event->state) { case PropertyDelete: { TRACE("\tPropertyDelete for atom %s on window %ld\n", TSXGetAtomName(event->display, event->atom), (long)event->window); if (X11DRV_IsSelectionOwner()) X11DRV_CLIPBOARD_FreeResources( event->atom ); break; } case PropertyNewValue: { TRACE("\tPropertyNewValue for atom %s on window %ld\n\n", TSXGetAtomName(event->display, event->atom), (long)event->window); break; } default: break; } } /********************************************************************** * EVENT_DropFromOffix * * don't know if it still works (last Changlog is from 96/11/04) */ static void EVENT_DropFromOffiX( HWND hWnd, XClientMessageEvent *event ) { unsigned long data_length; unsigned long aux_long; unsigned char* p_data = NULL; union { Atom atom_aux; struct { int x; int y; } pt_aux; int i; } u; int x, y; BOOL16 bAccept; HGLOBAL16 hDragInfo = GlobalAlloc16( GMEM_SHARE | GMEM_ZEROINIT, sizeof(DRAGINFO16)); LPDRAGINFO16 lpDragInfo = (LPDRAGINFO16) GlobalLock16(hDragInfo); SEGPTR spDragInfo = (SEGPTR) WIN16_GlobalLock16(hDragInfo); Window w_aux_root, w_aux_child; WND* pDropWnd; WND* pWnd; if( !lpDragInfo || !spDragInfo ) return; pWnd = WIN_FindWndPtr(hWnd); TSXQueryPointer( display, X11DRV_WND_GetXWindow(pWnd), &w_aux_root, &w_aux_child, &x, &y, (int *) &u.pt_aux.x, (int *) &u.pt_aux.y, (unsigned int*)&aux_long); lpDragInfo->hScope = hWnd; lpDragInfo->pt.x = (INT16)x; lpDragInfo->pt.y = (INT16)y; /* find out drop point and drop window */ if( x < 0 || y < 0 || x > (pWnd->rectWindow.right - pWnd->rectWindow.left) || y > (pWnd->rectWindow.bottom - pWnd->rectWindow.top) ) { bAccept = pWnd->dwExStyle & WS_EX_ACCEPTFILES; x = y = 0; } else { bAccept = DRAG_QueryUpdate( hWnd, spDragInfo, TRUE ); x = lpDragInfo->pt.x; y = lpDragInfo->pt.y; } pDropWnd = WIN_FindWndPtr( lpDragInfo->hScope ); WIN_ReleaseWndPtr(pWnd); GlobalFree16( hDragInfo ); if( bAccept ) { TSXGetWindowProperty( display, DefaultRootWindow(display), dndSelection, 0, 65535, FALSE, AnyPropertyType, &u.atom_aux, (int *) &u.pt_aux.y, &data_length, &aux_long, &p_data); if( !aux_long && p_data) /* don't bother if > 64K */ { char *p = (char*) p_data; char *p_drop; aux_long = 0; while( *p ) /* calculate buffer size */ { p_drop = p; if((u.i = *p) != -1 ) u.i = DRIVE_FindDriveRoot( (const char **)&p_drop ); if( u.i == -1 ) *p = -1; /* mark as "bad" */ else { INT len = GetShortPathNameA( p, NULL, 0 ); if (len) aux_long += len + 1; else *p = -1; } p += strlen(p) + 1; } if( aux_long && aux_long < 65535 ) { HDROP hDrop; DROPFILES *lpDrop; aux_long += sizeof(DROPFILES) + 1; hDrop = GlobalAlloc( GMEM_SHARE, aux_long ); lpDrop = (DROPFILES*)GlobalLock( hDrop ); if( lpDrop ) { lpDrop->pFiles = sizeof(DROPFILES); lpDrop->pt.x = x; lpDrop->pt.y = y; lpDrop->fNC = ( x < (pDropWnd->rectClient.left - pDropWnd->rectWindow.left) || y < (pDropWnd->rectClient.top - pDropWnd->rectWindow.top) || x > (pDropWnd->rectClient.right - pDropWnd->rectWindow.left) || y > (pDropWnd->rectClient.bottom - pDropWnd->rectWindow.top) ); lpDrop->fWide = FALSE; p_drop = (char *)(lpDrop + 1); p = p_data; while(*p) { if( *p != -1 ) /* use only "good" entries */ { GetShortPathNameA( p, p_drop, 65535 ); p_drop += strlen( p_drop ) + 1; } p += strlen(p) + 1; } *p_drop = '\0'; PostMessageA( hWnd, WM_DROPFILES, hDrop, 0L ); } } } if( p_data ) TSXFree(p_data); } /* WS_EX_ACCEPTFILES */ WIN_ReleaseWndPtr(pDropWnd); } /********************************************************************** * EVENT_DropURLs * * drop items are separated by \n * each item is prefixed by its mime type * * event->data.l[3], event->data.l[4] contains drop x,y position */ static void EVENT_DropURLs( HWND hWnd, XClientMessageEvent *event ) { WND *pDropWnd; WND *pWnd; unsigned long data_length; unsigned long aux_long, drop_len = 0; unsigned char *p_data = NULL; /* property data */ char *p_drop = NULL; char *p, *next; int x, y; DROPFILES *lpDrop; HDROP hDrop; union { Atom atom_aux; int i; Window w_aux; } u; /* unused */ pWnd = WIN_FindWndPtr(hWnd); if (!(pWnd->dwExStyle & WS_EX_ACCEPTFILES)) { WIN_ReleaseWndPtr(pWnd); return; } WIN_ReleaseWndPtr(pWnd); TSXGetWindowProperty( display, DefaultRootWindow(display), dndSelection, 0, 65535, FALSE, AnyPropertyType, &u.atom_aux, &u.i, &data_length, &aux_long, &p_data); if (aux_long) WARN("property too large, truncated!\n"); TRACE("urls=%s\n", p_data); if( !aux_long && p_data) { /* don't bother if > 64K */ /* calculate length */ p = p_data; next = strchr(p, '\n'); while (p) { if (next) *next=0; if (strncmp(p,"file:",5) == 0 ) { INT len = GetShortPathNameA( p+5, NULL, 0 ); if (len) drop_len += len + 1; } if (next) { *next = '\n'; p = next + 1; next = strchr(p, '\n'); } else { p = NULL; } } if( drop_len && drop_len < 65535 ) { TSXQueryPointer( display, X11DRV_GetXRootWindow(), &u.w_aux, &u.w_aux, &x, &y, &u.i, &u.i, &u.i); pDropWnd = WIN_FindWndPtr( hWnd ); drop_len += sizeof(DROPFILES) + 1; hDrop = (HDROP)GlobalAlloc( GMEM_SHARE, drop_len ); lpDrop = (DROPFILES *) GlobalLock( hDrop ); if( lpDrop ) { lpDrop->pFiles = sizeof(DROPFILES); lpDrop->pt.x = (INT)x; lpDrop->pt.y = (INT)y; lpDrop->fNC = ( x < (pDropWnd->rectClient.left - pDropWnd->rectWindow.left) || y < (pDropWnd->rectClient.top - pDropWnd->rectWindow.top) || x > (pDropWnd->rectClient.right - pDropWnd->rectWindow.left) || y > (pDropWnd->rectClient.bottom - pDropWnd->rectWindow.top) ); lpDrop->fWide = FALSE; p_drop = (char*)(lpDrop + 1); } /* create message content */ if (p_drop) { p = p_data; next = strchr(p, '\n'); while (p) { if (next) *next=0; if (strncmp(p,"file:",5) == 0 ) { INT len = GetShortPathNameA( p+5, p_drop, 65535 ); if (len) { TRACE("drop file %s as %s\n", p+5, p_drop); p_drop += len+1; } else { WARN("can't convert file %s to dos name \n", p+5); } } else { WARN("unknown mime type %s\n", p); } if (next) { *next = '\n'; p = next + 1; next = strchr(p, '\n'); } else { p = NULL; } *p_drop = '\0'; } GlobalUnlock(hDrop); PostMessageA( hWnd, WM_DROPFILES, hDrop, 0L ); } WIN_ReleaseWndPtr(pDropWnd); } if( p_data ) TSXFree(p_data); } } /********************************************************************** * EVENT_ClientMessage */ static void EVENT_ClientMessage( HWND hWnd, XClientMessageEvent *event ) { if (event->message_type != None && event->format == 32) { if ((event->message_type == wmProtocols) && (((Atom) event->data.l[0]) == wmDeleteWindow)) { /* Ignore the delete window request if the window has been disabled * and we are in managed mode. This is to disallow applications from * being closed by the window manager while in a modal state. */ BOOL bIsDisabled; bIsDisabled = GetWindowLongA( hWnd, GWL_STYLE ) & WS_DISABLED; if ( !Options.managed || !bIsDisabled ) PostMessage16( hWnd, WM_SYSCOMMAND, SC_CLOSE, 0 ); } else if ( event->message_type == dndProtocol && (event->data.l[0] == DndFile || event->data.l[0] == DndFiles) ) EVENT_DropFromOffiX(hWnd, event); else if ( event->message_type == dndProtocol && event->data.l[0] == DndURL ) EVENT_DropURLs(hWnd, event); else { #if 0 /* enable this if you want to see the message */ unsigned char* p_data = NULL; union { unsigned long l; int i; Atom atom; } u; /* unused */ TSXGetWindowProperty( display, DefaultRootWindow(display), dndSelection, 0, 65535, FALSE, AnyPropertyType, &u.atom, &u.i, &u.l, &u.l, &p_data); TRACE("message_type=%ld, data=%ld,%ld,%ld,%ld,%ld, msg=%s\n", event->message_type, event->data.l[0], event->data.l[1], event->data.l[2], event->data.l[3], event->data.l[4], p_data); #endif TRACE("unrecognized ClientMessage\n" ); } } } /********************************************************************** * EVENT_EnterNotify * * Install colormap when Wine window is focused in * self-managed mode with private colormap */ #if 0 void EVENT_EnterNotify( HWND hWnd, XCrossingEvent *event ) { if( !Options.managed && X11DRV_GetXRootWindow() == DefaultRootWindow(display) && (COLOR_GetSystemPaletteFlags() & COLOR_PRIVATE) && GetFocus() ) TSXInstallColormap( display, X11DRV_PALETTE_GetColormap() ); } #endif /********************************************************************** * EVENT_MapNotify */ void EVENT_MapNotify( HWND hWnd, XMapEvent *event ) { HWND hwndFocus = GetFocus(); WND *wndFocus = WIN_FindWndPtr(hwndFocus); WND *pWnd = WIN_FindWndPtr(hWnd); if (pWnd && (pWnd->dwExStyle & WS_EX_MANAGED)) { DCE_InvalidateDCE( pWnd, &pWnd->rectWindow ); pWnd->dwStyle &= ~WS_MINIMIZE; pWnd->dwStyle |= WS_VISIBLE; WIN_InternalShowOwnedPopups(hWnd,TRUE,TRUE); } WIN_ReleaseWndPtr(pWnd); if (hwndFocus && IsChild( hWnd, hwndFocus )) X11DRV_WND_SetFocus(wndFocus); WIN_ReleaseWndPtr(wndFocus); return; } /********************************************************************** * EVENT_UnmapNotify */ void EVENT_UnmapNotify( HWND hWnd, XUnmapEvent *event ) { WND *pWnd = WIN_FindWndPtr(hWnd); if (pWnd && (pWnd->dwExStyle & WS_EX_MANAGED)) { EndMenu(); if( pWnd->dwStyle & WS_VISIBLE ) { pWnd->dwStyle |= WS_MINIMIZE; pWnd->dwStyle &= ~WS_VISIBLE; WIN_InternalShowOwnedPopups(hWnd,FALSE,TRUE); } } WIN_ReleaseWndPtr(pWnd); } /*********************************************************************** * EVENT_MappingNotify */ static void EVENT_MappingNotify( XMappingEvent *event ) { TSXRefreshKeyboardMapping(event); /* reinitialize Wine-X11 driver keyboard table */ X11DRV_InitKeyboard(); } /********************************************************************** * X11DRV_EVENT_SetInputMethod */ INPUT_TYPE X11DRV_EVENT_SetInputMethod(INPUT_TYPE type) { INPUT_TYPE prev = current_input_type; /* Flag not used yet */ in_transition = FALSE; current_input_type = type; return prev; } #ifdef HAVE_LIBXXF86DGA2 /********************************************************************** * X11DRV_EVENT_SetDGAStatus */ void X11DRV_EVENT_SetDGAStatus(HWND hwnd, int event_base) { if (event_base < 0) { DGAUsed = FALSE; DGAhwnd = 0; } else { DGAUsed = TRUE; DGAhwnd = hwnd; DGAMotionEventType = event_base + MotionNotify; DGAButtonPressEventType = event_base + ButtonPress; DGAButtonReleaseEventType = event_base + ButtonRelease; DGAKeyPressEventType = event_base + KeyPress; DGAKeyReleaseEventType = event_base + KeyRelease; } } /* DGA2 event handlers */ static void EVENT_DGAMotionEvent( XDGAMotionEvent *event ) { X11DRV_SendEvent( MOUSEEVENTF_MOVE, event->dx, event->dy, X11DRV_EVENT_XStateToKeyState( event->state ), event->time - X11DRV_server_startticks, DGAhwnd ); } static void EVENT_DGAButtonPressEvent( XDGAButtonEvent *event ) { static WORD statusCodes[NB_BUTTONS] = { MOUSEEVENTF_LEFTDOWN, MOUSEEVENTF_MIDDLEDOWN, MOUSEEVENTF_RIGHTDOWN }; int buttonNum = event->button - 1; WORD keystate; if (buttonNum >= NB_BUTTONS) return; keystate = X11DRV_EVENT_XStateToKeyState( event->state ); switch (buttonNum) { case 0: keystate |= MK_LBUTTON; break; case 1: keystate |= MK_MBUTTON; break; case 2: keystate |= MK_RBUTTON; break; } X11DRV_SendEvent( statusCodes[buttonNum], 0, 0, keystate, event->time - X11DRV_server_startticks, DGAhwnd ); } static void EVENT_DGAButtonReleaseEvent( XDGAButtonEvent *event ) { static WORD statusCodes[NB_BUTTONS] = { MOUSEEVENTF_LEFTUP, MOUSEEVENTF_MIDDLEUP, MOUSEEVENTF_RIGHTUP }; int buttonNum = event->button - 1; WORD keystate; if (buttonNum >= NB_BUTTONS) return; keystate = X11DRV_EVENT_XStateToKeyState( event->state ); switch (buttonNum) { case 0: keystate &= ~MK_LBUTTON; break; case 1: keystate &= ~MK_MBUTTON; break; case 2: keystate &= ~MK_RBUTTON; break; } X11DRV_SendEvent( statusCodes[buttonNum], 0, 0, keystate, event->time - X11DRV_server_startticks, DGAhwnd ); } #endif #ifdef HAVE_LIBXXSHM /* Normal XShm operation: X11 service thread app thread ------------- ----------------- ------------------------ (idle) ddraw calls XShmPutImage (copies data) (waiting for shm_event) ShmCompletion -> (waiting for shm_event) (idle) signal shm_event -> (idle) returns to app However, this situation can occur for some reason: X11 service thread app thread ------------- ----------------- ------------------------ Expose -> WM_ERASEBKGND? -> (waiting for app) ddraw calls XShmPutImage (copies data) (waiting for app) (waiting for shm_event) ShmCompletion (waiting for app) (waiting for shm_event) (idle) DEADLOCK DEADLOCK which is why I also wait for shm_read and do XCheckTypedEvent() calls in the wait loop. This results in: X11 service thread app thread ------------- ----------------- ------------------------ ShmCompletion (waiting for app) waking up on shm_read (idle) (waiting for app) XCheckTypedEvent() -> signal shm_event (waiting for app) returns (idle) */ typedef struct { Drawable draw; LONG state, waiter; HANDLE sema; } shm_qs; /* FIXME: this is not pretty */ static HANDLE shm_read = 0; #define SHM_MAX_Q 4 static volatile shm_qs shm_q[SHM_MAX_Q]; static void EVENT_ShmCompletion( XShmCompletionEvent *event ) { int n; TRACE("Got ShmCompletion for drawable %ld (time %ld)\n", event->drawable, GetTickCount() ); for (n=0; n<SHM_MAX_Q; n++) if ((shm_q[n].draw == event->drawable) && (shm_q[n].state == 0)) { HANDLE sema = shm_q[n].sema; if (!InterlockedCompareExchange((PVOID*)&shm_q[n].state, (PVOID)1, (PVOID)0)) { ReleaseSemaphore(sema, 1, NULL); TRACE("Signaling ShmCompletion (#%d) (semaphore %x)\n", n, sema); } return; } ERR("Got ShmCompletion for unknown drawable %ld\n", event->drawable ); } int X11DRV_EVENT_PrepareShmCompletion( Drawable dw ) { int n; if (!shm_read) shm_read = FILE_DupUnixHandle( ConnectionNumber(display), GENERIC_READ | SYNCHRONIZE ); for (n=0; n<SHM_MAX_Q; n++) if (!shm_q[n].draw) if (!InterlockedCompareExchange((PVOID*)&shm_q[n].draw, (PVOID)dw, (PVOID)0)) break; if (n>=SHM_MAX_Q) { ERR("Maximum number of outstanding ShmCompletions exceeded!\n"); return 0; } shm_q[n].state = 0; if (!shm_q[n].sema) { shm_q[n].sema = CreateSemaphoreA( NULL, 0, 256, NULL ); TRACE("Allocated ShmCompletion slots have been increased to %d, new semaphore is %x\n", n+1, shm_q[n].sema); } TRACE("Prepared ShmCompletion (#%d) wait for drawable %ld (thread %lx) (time %ld)\n", n, dw, GetCurrentThreadId(), GetTickCount() ); return n+1; } static void X11DRV_EVENT_WaitReplaceShmCompletionInternal( int *compl, Drawable dw, int creat ) { int n = *compl; LONG nn, st; HANDLE sema; if ((!n) || (creat && (!shm_q[n-1].draw))) { nn = X11DRV_EVENT_PrepareShmCompletion(dw); if (!(n=(LONG)InterlockedCompareExchange((PVOID*)compl, (PVOID)nn, (PVOID)n))) return; /* race for compl lost, clear slot */ shm_q[nn-1].draw = 0; return; } if (dw && (shm_q[n-1].draw != dw)) { /* this shouldn't happen with the current ddraw implementation */ FIXME("ShmCompletion replace with different drawable!\n"); return; } sema = shm_q[n-1].sema; if (!sema) { /* nothing to wait on (PrepareShmCompletion not done yet?), so probably nothing to wait for */ return; } nn = InterlockedExchangeAdd((PLONG)&shm_q[n-1].waiter, 1); if ((!shm_q[n-1].draw) || (shm_q[n-1].state == 2)) { /* too late, the wait was just cleared (wait complete) */ TRACE("Wait skip for ShmCompletion (#%d) (thread %lx) (time %ld) (semaphore %x)\n", n-1, GetCurrentThreadId(), GetTickCount(), sema); } else { TRACE("Waiting for ShmCompletion (#%d) (thread %lx) (time %ld) (semaphore %x)\n", n-1, GetCurrentThreadId(), GetTickCount(), sema); if (nn) { /* another thread is already waiting, let the primary waiter do the dirty work * (to avoid TSX critical section contention - that could get really slow) */ WaitForSingleObject( sema, INFINITE ); } else /* we're primary waiter - first check if it's already triggered */ if ( WaitForSingleObject( sema, 0 ) != WAIT_OBJECT_0 ) { /* nope, may need to poll X event queue, in case the service thread is blocked */ XEvent event; HANDLE hnd[2]; hnd[0] = sema; hnd[1] = shm_read; do { /* check X event queue */ if (TSXCheckTypedEvent( display, ShmCompletionType, &event)) { EVENT_ProcessEvent( &event ); } } while ( WaitForMultipleObjects(2, hnd, FALSE, INFINITE) > WAIT_OBJECT_0 ); } TRACE("Wait complete (thread %lx) (time %ld)\n", GetCurrentThreadId(), GetTickCount() ); /* clear wait */ st = InterlockedExchange((LPLONG)&shm_q[n-1].state, 2); if (st != 2) { /* first waiter to return, release all other waiters */ nn = shm_q[n-1].waiter; TRACE("Signaling %ld additional ShmCompletion (#%d) waiter(s), semaphore %x\n", nn-1, n-1, sema); ReleaseSemaphore(sema, nn-1, NULL); } } nn = InterlockedDecrement((LPLONG)&shm_q[n-1].waiter); if (!nn) { /* last waiter to return, replace drawable and prepare new wait */ shm_q[n-1].draw = dw; shm_q[n-1].state = 0; } } void X11DRV_EVENT_WaitReplaceShmCompletion( int *compl, Drawable dw ) { X11DRV_EVENT_WaitReplaceShmCompletionInternal( compl, dw, 1 ); } void X11DRV_EVENT_WaitShmCompletion( int compl ) { if (!compl) return; X11DRV_EVENT_WaitReplaceShmCompletionInternal( &compl, 0, 0 ); } void X11DRV_EVENT_WaitShmCompletions( Drawable dw ) { int n; for (n=0; n<SHM_MAX_Q; n++) if (shm_q[n].draw == dw) X11DRV_EVENT_WaitShmCompletion( n+1 ); } #endif /* defined(HAVE_LIBXXSHM) */