/* *Copyright (C) 1994-2000 The XFree86 Project, Inc. All Rights Reserved. *Copyright (C) Colin Harrison 2005-2009 * *Permission is hereby granted, free of charge, to any person obtaining * a copy of this software and associated documentation files (the *"Software"), to deal in the Software without restriction, including *without limitation the rights to use, copy, modify, merge, publish, *distribute, sublicense, and/or sell copies of the Software, and to *permit persons to whom the Software is furnished to do so, subject to *the following conditions: * *The above copyright notice and this permission notice shall be *included in all copies or substantial portions of the Software. * *THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, *EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF *MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND *NONINFRINGEMENT. IN NO EVENT SHALL THE XFREE86 PROJECT BE LIABLE FOR *ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF *CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION *WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. * *Except as contained in this notice, the name of the XFree86 Project *shall not be used in advertising or otherwise to promote the sale, use *or other dealings in this Software without prior written authorization *from the XFree86 Project. * * Authors: Kensuke Matsuzaki * Colin Harrison */ #ifndef WINVER #define WINVER 0x0500 #endif /* X headers */ #ifdef HAVE_XWIN_CONFIG_H #include #endif /* * Including any server header might define the macro _XSERVER64 on 64 bit machines. * That macro must _NOT_ be defined for Xlib client code, otherwise bad things happen. * So let's undef that macro if necessary. */ #ifdef _XSERVER64 #undef _XSERVER64 #endif #include #include #include #ifdef __CYGWIN__ #include #endif #include #include #define HANDLE void * #include #undef HANDLE #include #include #include #include #include #include /* Local headers */ #include "X11/Xdefs.h" // for Bool type #include "winwindow.h" #include "winprefs.h" #include "window.h" #include "winglobals.h" #include "windisplay.h" #include "winmultiwindowicons.h" /* We need the native HWND atom for intWM, so for consistency use the same name as extWM does */ #define WINDOWSWM_NATIVE_HWND "_WINDOWSWM_NATIVE_HWND" #ifndef HOST_NAME_MAX #define HOST_NAME_MAX 255 #endif extern void winDebug(const char *format, ...); extern void winReshapeMultiWindow(WindowPtr pWin); extern void winUpdateRgnMultiWindow(WindowPtr pWin); extern xcb_auth_info_t *winGetXcbAuthInfo(void); extern void winUpdateWindowPosition(HWND hWnd, HWND * zstyle); #ifndef CYGDEBUG #define CYGDEBUG NO #endif /* * Constant defines */ #define WIN_CONNECT_RETRIES 5 #define WIN_CONNECT_DELAY 5 #ifdef HAS_DEVWINDOWS #define WIN_MSG_QUEUE_FNAME "/dev/windows" #endif #define HINT_MAX (1L<<0) #define HINT_MIN (1L<<1) /* * Local structures */ typedef struct _WMMsgNodeRec { winWMMessageRec msg; struct _WMMsgNodeRec *pNext; } WMMsgNodeRec, *WMMsgNodePtr; typedef struct _WMMsgQueueRec { struct _WMMsgNodeRec *pHead; struct _WMMsgNodeRec *pTail; pthread_mutex_t pmMutex; pthread_cond_t pcNotEmpty; } WMMsgQueueRec, *WMMsgQueuePtr; typedef struct _WMInfo { xcb_connection_t *conn; WMMsgQueueRec wmMsgQueue; xcb_atom_t atmWmProtos; xcb_atom_t atmWmDelete; xcb_atom_t atmWmTakeFocus; xcb_atom_t atmPrivMap; xcb_atom_t atmUtf8String; xcb_atom_t atmNetWmName; xcb_atom_t atmCurrentDesktop; xcb_atom_t atmNumberDesktops; xcb_atom_t atmDesktopNames; xcb_atom_t atmWmState; xcb_ewmh_connection_t ewmh; Bool fCompositeWM; } WMInfoRec, *WMInfoPtr; typedef struct _WMProcArgRec { DWORD dwScreen; WMInfoPtr pWMInfo; pthread_mutex_t *ppmServerStarted; } WMProcArgRec, *WMProcArgPtr; typedef struct _XMsgProcArgRec { xcb_connection_t *conn; DWORD dwScreen; WMInfoPtr pWMInfo; pthread_mutex_t *ppmServerStarted; HWND hwndScreen; } XMsgProcArgRec, *XMsgProcArgPtr; /* * Prototypes for local functions */ static void PushMessage(WMMsgQueuePtr pQueue, WMMsgNodePtr pNode); static WMMsgNodePtr PopMessage(WMMsgQueuePtr pQueue, WMInfoPtr pWMInfo); static Bool InitQueue(WMMsgQueuePtr pQueue); static void GetWindowName(WMInfoPtr pWMInfo, xcb_window_t iWin, char **ppWindowName); static void SendXMessage(xcb_connection_t *conn, xcb_window_t iWin, xcb_atom_t atmType, long nData); static void UpdateName(WMInfoPtr pWMInfo, xcb_window_t iWindow); static void *winMultiWindowWMProc(void *pArg); static void *winMultiWindowXMsgProc(void *pArg); static void winMultiWindowThreadExit(void *arg); static void winInitMultiWindowWM(WMInfoPtr pWMInfo, WMProcArgPtr pProcArg); #if 0 static void PreserveWin32Stack(WMInfoPtr pWMInfo, xcb_window_t iWindow, UINT direction); #endif static Bool CheckAnotherWindowManager(xcb_connection_t *conn, DWORD dwScreen); static void winApplyHints(WMInfoPtr pWMInfo, xcb_window_t iWindow, HWND hWnd, HWND * zstyle, unsigned long *maxmin); static void winApplyUrgency(WMInfoPtr pWMInfo, xcb_window_t iWindow, HWND hWnd); /* * Local globals */ static Bool g_shutdown = FALSE; /* * Translate msg id to text, for debug purposes */ static const char * MessageName(winWMMessagePtr msg) { switch (msg->msg) { case WM_WM_MOVE: return "WM_WM_MOVE"; break; case WM_WM_SIZE: return "WM_WM_SIZE"; break; case WM_WM_RAISE: return "WM_WM_RAISE"; break; case WM_WM_LOWER: return "WM_WM_LOWER"; break; case WM_WM_UNMAP: return "WM_WM_UNMAP"; break; case WM_WM_KILL: return "WM_WM_KILL"; break; case WM_WM_ACTIVATE: return "WM_WM_ACTIVATE"; break; case WM_WM_NAME_EVENT: return "WM_WM_NAME_EVENT"; break; case WM_WM_ICON_EVENT: return "WM_WM_ICON_EVENT"; break; case WM_WM_CHANGE_STATE: return "WM_WM_CHANGE_STATE"; break; case WM_WM_MAP: return "WM_WM_MAP"; break; case WM_WM_MAP_UNMANAGED: return "WM_WM_MAP_UNMANAGED"; break; case WM_WM_MAP_MANAGED: return "WM_WM_MAP_MANAGED"; break; case WM_WM_HINTS_EVENT: return "WM_WM_HINTS_EVENT"; break; case WM_WM_CREATE: return "WM_WM_CREATE"; break; default: return "Unknown Message"; break; } } /* * PushMessage - Push a message onto the queue */ static void PushMessage(WMMsgQueuePtr pQueue, WMMsgNodePtr pNode) { /* Lock the queue mutex */ pthread_mutex_lock(&pQueue->pmMutex); pNode->pNext = NULL; if (pQueue->pTail != NULL) { pQueue->pTail->pNext = pNode; } pQueue->pTail = pNode; if (pQueue->pHead == NULL) { pQueue->pHead = pNode; } /* Release the queue mutex */ pthread_mutex_unlock(&pQueue->pmMutex); /* Signal that the queue is not empty */ pthread_cond_signal(&pQueue->pcNotEmpty); } /* * PopMessage - Pop a message from the queue */ static WMMsgNodePtr PopMessage(WMMsgQueuePtr pQueue, WMInfoPtr pWMInfo) { WMMsgNodePtr pNode; /* Lock the queue mutex */ pthread_mutex_lock(&pQueue->pmMutex); /* Wait for --- */ while (pQueue->pHead == NULL) { pthread_cond_wait(&pQueue->pcNotEmpty, &pQueue->pmMutex); } pNode = pQueue->pHead; if (pQueue->pHead != NULL) { pQueue->pHead = pQueue->pHead->pNext; } if (pQueue->pTail == pNode) { pQueue->pTail = NULL; } /* Release the queue mutex */ pthread_mutex_unlock(&pQueue->pmMutex); return pNode; } #if 0 /* * HaveMessage - */ static Bool HaveMessage(WMMsgQueuePtr pQueue, UINT msg, xcb_window_t iWindow) { WMMsgNodePtr pNode; for (pNode = pQueue->pHead; pNode != NULL; pNode = pNode->pNext) { if (pNode->msg.msg == msg && pNode->msg.iWindow == iWindow) return True; } return False; } #endif /* * InitQueue - Initialize the Window Manager message queue */ static Bool InitQueue(WMMsgQueuePtr pQueue) { /* Check if the pQueue pointer is NULL */ if (pQueue == NULL) { ErrorF("InitQueue - pQueue is NULL. Exiting.\n"); return FALSE; } /* Set the head and tail to NULL */ pQueue->pHead = NULL; pQueue->pTail = NULL; winDebug("InitQueue - Calling pthread_mutex_init\n"); /* Create synchronization objects */ pthread_mutex_init(&pQueue->pmMutex, NULL); winDebug("InitQueue - pthread_mutex_init returned\n"); winDebug("InitQueue - Calling pthread_cond_init\n"); pthread_cond_init(&pQueue->pcNotEmpty, NULL); winDebug("InitQueue - pthread_cond_init returned\n"); return TRUE; } static char * Xutf8TextPropertyToString(WMInfoPtr pWMInfo, xcb_icccm_get_text_property_reply_t *xtp) { char *pszReturnData; if ((xtp->encoding == XCB_ATOM_STRING) || // Latin1 ISO 8859-1 (xtp->encoding == pWMInfo->atmUtf8String)) { // UTF-8 ISO 10646 pszReturnData = strndup(xtp->name, xtp->name_len); } else { // Converting from COMPOUND_TEXT to UTF-8 properly is complex to // implement, and not very much use unless you have an old // application which isn't UTF-8 aware. ErrorF("Xutf8TextPropertyToString: text encoding %d is not implemented\n", xtp->encoding); pszReturnData = strdup(""); } return pszReturnData; } /* * GetWindowName - Retrieve the title of an X Window */ static void GetWindowName(WMInfoPtr pWMInfo, xcb_window_t iWin, char **ppWindowName) { xcb_connection_t *conn = pWMInfo->conn; char *pszWindowName = NULL; #if CYGMULTIWINDOW_DEBUG ErrorF("GetWindowName\n"); #endif /* Try to get window name from _NET_WM_NAME */ { xcb_get_property_cookie_t cookie; xcb_get_property_reply_t *reply; cookie = xcb_get_property(pWMInfo->conn, FALSE, iWin, pWMInfo->atmNetWmName, XCB_GET_PROPERTY_TYPE_ANY, 0, INT_MAX); reply = xcb_get_property_reply(pWMInfo->conn, cookie, NULL); if (reply && (reply->type != XCB_NONE)) { pszWindowName = strndup(xcb_get_property_value(reply), xcb_get_property_value_length(reply)); free(reply); } } /* Otherwise, try to get window name from WM_NAME */ if (!pszWindowName) { xcb_get_property_cookie_t cookie; xcb_icccm_get_text_property_reply_t reply; cookie = xcb_icccm_get_wm_name(conn, iWin); if (!xcb_icccm_get_wm_name_reply(conn, cookie, &reply, NULL)) { ErrorF("GetWindowName - xcb_icccm_get_wm_name_reply failed. No name.\n"); *ppWindowName = NULL; return; } pszWindowName = Xutf8TextPropertyToString(pWMInfo, &reply); xcb_icccm_get_text_property_reply_wipe(&reply); } /* return the window name, unless... */ *ppWindowName = pszWindowName; if (g_fHostInTitle) { xcb_get_property_cookie_t cookie; xcb_icccm_get_text_property_reply_t reply; /* Try to get client machine name */ cookie = xcb_icccm_get_wm_client_machine(conn, iWin); if (xcb_icccm_get_wm_client_machine_reply(conn, cookie, &reply, NULL)) { char *pszClientMachine; char *pszClientHostname; char *dot; char hostname[HOST_NAME_MAX + 1]; pszClientMachine = Xutf8TextPropertyToString(pWMInfo, &reply); xcb_icccm_get_text_property_reply_wipe(&reply); /* If client machine name looks like a FQDN, find the hostname */ pszClientHostname = strdup(pszClientMachine); dot = strchr(pszClientHostname, '.'); if (dot) *dot = '\0'; /* If we have a client machine hostname and it's not the local hostname and it's not already in the window title... */ if (strlen(pszClientHostname) && !gethostname(hostname, HOST_NAME_MAX + 1) && strcmp(hostname, pszClientHostname) && (strstr(pszWindowName, pszClientHostname) == 0)) { /* ... add '@' to end of window name */ *ppWindowName = malloc(strlen(pszWindowName) + strlen(pszClientMachine) + 2); strcpy(*ppWindowName, pszWindowName); strcat(*ppWindowName, "@"); strcat(*ppWindowName, pszClientMachine); free(pszWindowName); } free(pszClientMachine); free(pszClientHostname); } } } /* * Does the client support the specified WM_PROTOCOLS protocol? */ static Bool IsWmProtocolAvailable(WMInfoPtr pWMInfo, xcb_window_t iWindow, xcb_atom_t atmProtocol) { int i, found = 0; xcb_get_property_cookie_t cookie; xcb_icccm_get_wm_protocols_reply_t reply; xcb_connection_t *conn = pWMInfo->conn; cookie = xcb_icccm_get_wm_protocols(conn, iWindow, pWMInfo->ewmh.WM_PROTOCOLS); if (xcb_icccm_get_wm_protocols_reply(conn, cookie, &reply, NULL)) { for (i = 0; i < reply.atoms_len; ++i) if (reply.atoms[i] == atmProtocol) { ++found; break; } xcb_icccm_get_wm_protocols_reply_wipe(&reply); } return found > 0; } /* * Send a message to the X server from the WM thread */ static void SendXMessage(xcb_connection_t *conn, xcb_window_t iWin, xcb_atom_t atmType, long nData) { xcb_client_message_event_t e; /* Prepare the X event structure */ memset(&e, 0, sizeof(e)); e.response_type = XCB_CLIENT_MESSAGE; e.window = iWin; e.type = atmType; e.format = 32; e.data.data32[0] = nData; e.data.data32[1] = XCB_CURRENT_TIME; /* Send the event to X */ xcb_send_event(conn, FALSE, iWin, XCB_EVENT_MASK_NO_EVENT, (const char *)&e); } /* * See if we can get the stored HWND for this window... */ static HWND getHwnd(WMInfoPtr pWMInfo, xcb_window_t iWindow) { HWND hWnd = NULL; xcb_get_property_cookie_t cookie; xcb_get_property_reply_t *reply; cookie = xcb_get_property(pWMInfo->conn, FALSE, iWindow, pWMInfo->atmPrivMap, XCB_ATOM_INTEGER, 0L, sizeof(HWND)/4L); reply = xcb_get_property_reply(pWMInfo->conn, cookie, NULL); if (reply) { int length = xcb_get_property_value_length(reply); HWND *value = xcb_get_property_value(reply); if (value && (length == sizeof(HWND))) { hWnd = *value; } free(reply); } /* Some sanity checks */ if (!hWnd) { winDebug("getHwnd: no HWND\n"); return NULL; } if (!IsWindow(hWnd)) return NULL; return hWnd; } /* * Helper function to check for override-redirect */ static Bool IsOverrideRedirect(xcb_connection_t *conn, xcb_window_t iWin) { Bool result = FALSE; xcb_get_window_attributes_reply_t *reply; xcb_get_window_attributes_cookie_t cookie; cookie = xcb_get_window_attributes(conn, iWin); reply = xcb_get_window_attributes_reply(conn, cookie, NULL); if (reply) { result = (reply->override_redirect != 0); free(reply); } else { ErrorF("IsOverrideRedirect: Failed to get window attributes\n"); } return result; } /* * Helper function to get class and window names */ static void GetClassNames(WMInfoPtr pWMInfo, xcb_window_t iWindow, char **res_name, char **res_class, char **window_name) { xcb_get_property_cookie_t cookie1; xcb_icccm_get_wm_class_reply_t reply1; xcb_get_property_cookie_t cookie2; xcb_icccm_get_text_property_reply_t reply2; cookie1 = xcb_icccm_get_wm_class(pWMInfo->conn, iWindow); if (xcb_icccm_get_wm_class_reply(pWMInfo->conn, cookie1, &reply1, NULL)) { *res_name = strdup(reply1.instance_name); *res_class = strdup(reply1.class_name); xcb_icccm_get_wm_class_reply_wipe(&reply1); } else { *res_name = strdup(""); *res_class = strdup(""); } cookie2 = xcb_icccm_get_wm_name(pWMInfo->conn, iWindow); if (xcb_icccm_get_wm_name_reply(pWMInfo->conn, cookie2, &reply2, NULL)) { *window_name = strndup(reply2.name, reply2.name_len); xcb_icccm_get_text_property_reply_wipe(&reply2); } else { *window_name = strdup(""); } } /* * Updates the name of a HWND according to its X WM_NAME property */ static void UpdateName(WMInfoPtr pWMInfo, xcb_window_t iWindow) { HWND hWnd; hWnd = getHwnd(pWMInfo, iWindow); if (!hWnd) return; /* If window isn't override-redirect */ if (!IsOverrideRedirect(pWMInfo->conn, iWindow)) { char *pszWindowName; /* Get the X windows window name */ GetWindowName(pWMInfo, iWindow, &pszWindowName); if (pszWindowName) { /* Convert from UTF-8 to wide char */ int iLen = MultiByteToWideChar(CP_UTF8, 0, pszWindowName, -1, NULL, 0); wchar_t *pwszWideWindowName = malloc(sizeof(wchar_t)*(iLen + 1)); MultiByteToWideChar(CP_UTF8, 0, pszWindowName, -1, pwszWideWindowName, iLen); /* Set the Windows window name */ SetWindowTextW(hWnd, pwszWideWindowName); free(pwszWideWindowName); free(pszWindowName); } } } /* * Updates the icon of a HWND according to its X icon properties */ static void UpdateIcon(WMInfoPtr pWMInfo, xcb_window_t iWindow) { HWND hWnd; HICON hIconNew = NULL; hWnd = getHwnd(pWMInfo, iWindow); if (!hWnd) return; /* If window isn't override-redirect */ if (!IsOverrideRedirect(pWMInfo->conn, iWindow)) { char *window_name = 0; char *res_name = 0; char *res_class = 0; GetClassNames(pWMInfo, iWindow, &res_name, &res_class, &window_name); hIconNew = winOverrideIcon(res_name, res_class, window_name); free(res_name); free(res_class); free(window_name); winUpdateIcon(hWnd, pWMInfo->conn, iWindow, hIconNew); } } /* * Updates the style of a HWND according to its X style properties */ static void UpdateStyle(WMInfoPtr pWMInfo, xcb_window_t iWindow, unsigned long *maxmin) { HWND hWnd; HWND zstyle = HWND_NOTOPMOST; UINT flags; hWnd = getHwnd(pWMInfo, iWindow); if (!hWnd) return; /* If window isn't override-redirect */ if (IsOverrideRedirect(pWMInfo->conn, iWindow)) return; /* Determine the Window style, which determines borders and clipping region... */ winApplyHints(pWMInfo, iWindow, hWnd, &zstyle, maxmin); winUpdateWindowPosition(hWnd, &zstyle); /* Apply the updated window style, without changing it's show or activation state */ flags = SWP_FRAMECHANGED | SWP_NOACTIVATE | SWP_NOMOVE | SWP_NOSIZE; if (zstyle == HWND_NOTOPMOST) flags |= SWP_NOZORDER | SWP_NOOWNERZORDER; SetWindowPos(hWnd, NULL, 0, 0, 0, 0, flags); /* Use the WS_EX_TOOLWINDOW style to remove window from Alt-Tab window switcher According to MSDN, this is supposed to remove the window from the taskbar as well, if we SW_HIDE before changing the style followed by SW_SHOW afterwards. But that doesn't seem to work reliably, and causes the window to flicker, so use the iTaskbarList interface to tell the taskbar to show or hide this window. */ winShowWindowOnTaskbar(hWnd, (GetWindowLongPtr(hWnd, GWL_EXSTYLE) & WS_EX_APPWINDOW) ? TRUE : FALSE); /* Check urgency hint */ winApplyUrgency(pWMInfo, iWindow, hWnd); } /* * Updates the state of a HWND */ static void UpdateState(WMInfoPtr pWMInfo, xcb_window_t iWindow, int state) { HWND hWnd; int current_state = -1; winDebug("UpdateState: iWindow 0x%08x %d\n", (int)iWindow, state); hWnd = getHwnd(pWMInfo, iWindow); if (hWnd) { // Keep track of the Window state, do nothing if it's not changing current_state = (intptr_t)GetProp(hWnd, WIN_STATE_PROP); if (current_state == state) return; SetProp(hWnd, WIN_STATE_PROP, (HANDLE)(intptr_t)state); switch (state) { case XCB_ICCCM_WM_STATE_ICONIC: ShowWindow(hWnd, SW_SHOWMINNOACTIVE); break; #define XCB_ICCCM_WM_STATE_ZOOM 2 case XCB_ICCCM_WM_STATE_ZOOM: // There doesn't seem to be a SW_SHOWMAXNOACTIVE. Hopefully // always activating a maximized window isn't so bad... ShowWindow(hWnd, SW_SHOWMAXIMIZED); break; case XCB_ICCCM_WM_STATE_NORMAL: ShowWindow(hWnd, SW_SHOWNOACTIVATE); break; case XCB_ICCCM_WM_STATE_WITHDRAWN: ShowWindow(hWnd, SW_HIDE); break; } } // Update WM_STATE property { // ZoomState is obsolete in ICCCM, so map it to NormalState int icccm_state = state; int icccm_current_state = current_state; if (icccm_state == XCB_ICCCM_WM_STATE_ZOOM) icccm_state = XCB_ICCCM_WM_STATE_NORMAL; if (icccm_current_state == XCB_ICCCM_WM_STATE_ZOOM) icccm_current_state = XCB_ICCCM_WM_STATE_NORMAL; // Don't change property unnecessarily // // (Note that we do not take notice of WM_STATE PropertyNotify, only // WM_CHANGE_STATE ClientMessage, so this should not cause the state to // change itself) if (icccm_current_state != icccm_state) { struct { CARD32 state; XID icon; } wmstate; wmstate.state = icccm_state; wmstate.icon = None; xcb_change_property(pWMInfo->conn, XCB_PROP_MODE_REPLACE, iWindow, pWMInfo->atmWmState, pWMInfo->atmWmState, 32, sizeof(wmstate)/sizeof(int), (unsigned char *) &wmstate); } } // Update _NET_WM_STATE property if (state == XCB_ICCCM_WM_STATE_WITHDRAWN) { xcb_delete_property(pWMInfo->conn, iWindow, pWMInfo->ewmh._NET_WM_STATE); } else { xcb_get_property_cookie_t cookie; xcb_get_property_reply_t *reply; cookie = xcb_get_property(pWMInfo->conn, FALSE, iWindow, pWMInfo->ewmh._NET_WM_STATE, XCB_ATOM_ATOM, 0, INT_MAX); reply = xcb_get_property_reply(pWMInfo->conn, cookie, NULL); if (reply) { int nitems = xcb_get_property_value_length(reply)/sizeof(xcb_atom_t); xcb_atom_t *pAtom = xcb_get_property_value(reply); unsigned long i, o = 0; xcb_atom_t netwmstate[nitems + 2]; Bool changed = FALSE; // Make a copy with _NET_WM_HIDDEN, _NET_WM_MAXIMIZED_{VERT,HORZ} // removed for (i = 0; i < nitems; i++) { if ((pAtom[i] != pWMInfo->ewmh._NET_WM_STATE_HIDDEN) && (pAtom[i] != pWMInfo->ewmh._NET_WM_STATE_MAXIMIZED_VERT) && (pAtom[i] != pWMInfo->ewmh._NET_WM_STATE_MAXIMIZED_HORZ)) netwmstate[o++] = pAtom[i]; } free(reply); // if iconized, add _NET_WM_HIDDEN if (state == XCB_ICCCM_WM_STATE_ICONIC) { netwmstate[o++] = pWMInfo->ewmh._NET_WM_STATE_HIDDEN; } // if maximized, add _NET_WM_MAXIMIZED_{VERT,HORZ} if (state == XCB_ICCCM_WM_STATE_ZOOM) { netwmstate[o++] = pWMInfo->ewmh._NET_WM_STATE_MAXIMIZED_VERT; netwmstate[o++] = pWMInfo->ewmh._NET_WM_STATE_MAXIMIZED_HORZ; } // Don't change property unnecessarily if (nitems != o) changed = TRUE; else for (i = 0; i < nitems; i++) { if (pAtom[i] != netwmstate[i]) { changed = TRUE; break; } } if (changed) xcb_change_property(pWMInfo->conn, XCB_PROP_MODE_REPLACE, iWindow, pWMInfo->ewmh._NET_WM_STATE, XCB_ATOM_ATOM, 32, o, (unsigned char *) &netwmstate); } } } #if 0 /* * Fix up any differences between the X11 and Win32 window stacks * starting at the window passed in */ static void PreserveWin32Stack(WMInfoPtr pWMInfo, xcb_window_t iWindow, UINT direction) { HWND hWnd; DWORD myWinProcID, winProcID; xcb_window_t xWindow; WINDOWPLACEMENT wndPlace; hWnd = getHwnd(pWMInfo, iWindow); if (!hWnd) return; GetWindowThreadProcessId(hWnd, &myWinProcID); hWnd = GetNextWindow(hWnd, direction); while (hWnd) { GetWindowThreadProcessId(hWnd, &winProcID); if (winProcID == myWinProcID) { wndPlace.length = sizeof(WINDOWPLACEMENT); GetWindowPlacement(hWnd, &wndPlace); if (!(wndPlace.showCmd == SW_HIDE || wndPlace.showCmd == SW_MINIMIZE)) { xWindow = (Window) GetProp(hWnd, WIN_WID_PROP); if (xWindow) { if (direction == GW_HWNDPREV) XRaiseWindow(pWMInfo->pDisplay, xWindow); else XLowerWindow(pWMInfo->pDisplay, xWindow); } } } hWnd = GetNextWindow(hWnd, direction); } } #endif /* PreserveWin32Stack */ /* * winMultiWindowWMProc */ static void * winMultiWindowWMProc(void *pArg) { WMProcArgPtr pProcArg = (WMProcArgPtr) pArg; WMInfoPtr pWMInfo = pProcArg->pWMInfo; pthread_cleanup_push(&winMultiWindowThreadExit, NULL); /* Initialize the Window Manager */ winInitMultiWindowWM(pWMInfo, pProcArg); #if CYGMULTIWINDOW_DEBUG ErrorF("winMultiWindowWMProc ()\n"); #endif /* Loop until we explicitly break out */ for (;;) { WMMsgNodePtr pNode; /* Pop a message off of our queue */ pNode = PopMessage(&pWMInfo->wmMsgQueue, pWMInfo); if (pNode == NULL) { /* Bail if PopMessage returns without a message */ /* NOTE: Remember that PopMessage is a blocking function. */ ErrorF("winMultiWindowWMProc - Queue is Empty? Exiting.\n"); pthread_exit(NULL); } #if CYGMULTIWINDOW_DEBUG ErrorF("winMultiWindowWMProc - MSG: %s (%d) Window: %08x ID: %d\n", MessageName(&(pNode->msg)), (int)pNode->msg.msg, pNode->msg.iWindow, (int)pNode->msg.dwID); #endif /* Branch on the message type */ switch (pNode->msg.msg) { case WM_WM_CREATE: { unsigned long maxmin = 0; #if CYGMULTIWINDOW_DEBUG ErrorF("\tWM_WM_CREATE\n"); #endif /* Put a note as to the HWND associated with this Window */ xcb_change_property(pWMInfo->conn, XCB_PROP_MODE_REPLACE, pNode->msg.iWindow, pWMInfo->atmPrivMap, XCB_ATOM_INTEGER, 32, sizeof(HWND)/4, &(pNode->msg.hwndWindow)); /* Determine the Window style, which determines borders and clipping region... */ UpdateStyle(pWMInfo, pNode->msg.iWindow, &maxmin); /* Display the window without activating it */ { xcb_get_window_attributes_cookie_t cookie; xcb_get_window_attributes_reply_t *reply; cookie = xcb_get_window_attributes(pWMInfo->conn, pNode->msg.iWindow); reply = xcb_get_window_attributes_reply(pWMInfo->conn, cookie, NULL); if (reply) { if (reply->_class != InputOnly) ShowWindow(pNode->msg.hwndWindow, SW_SHOWNA); free(reply); } } /* Send first paint message */ UpdateWindow(pNode->msg.hwndWindow); /* Establish initial state */ UpdateState(pWMInfo, pNode->msg.iWindow, XCB_ICCCM_WM_STATE_NORMAL); /* It only makes sense to apply minimize/maximize override as the initial state, otherwise that state can't be changed. */ if (maxmin & HINT_MAX) SendMessage(pNode->msg.hwndWindow, WM_SYSCOMMAND, SC_MAXIMIZE, 0); else if (maxmin & HINT_MIN) SendMessage(pNode->msg.hwndWindow, WM_SYSCOMMAND, SC_MINIMIZE, 0); } break; #if 0 case WM_WM_MOVE: break; case WM_WM_SIZE: break; #endif case WM_WM_RAISE: /* Raise the window */ { const static uint32_t values[] = { XCB_STACK_MODE_ABOVE }; xcb_configure_window(pWMInfo->conn, pNode->msg.iWindow, XCB_CONFIG_WINDOW_STACK_MODE, values); } #if 0 PreserveWin32Stack(pWMInfo, pNode->msg.iWindow, GW_HWNDPREV); #endif break; case WM_WM_LOWER: /* Lower the window */ { const static uint32_t values[] = { XCB_STACK_MODE_BELOW }; xcb_configure_window(pWMInfo->conn, pNode->msg.iWindow, XCB_CONFIG_WINDOW_STACK_MODE, values); } break; case WM_WM_MAP: UpdateName(pWMInfo, pNode->msg.iWindow); UpdateIcon(pWMInfo, pNode->msg.iWindow); break; case WM_WM_MAP_UNMANAGED: break; case WM_WM_MAP_MANAGED: { unsigned long maxmin = 0; UpdateName(pWMInfo, pNode->msg.iWindow); UpdateStyle(pWMInfo, pNode->msg.iWindow, &maxmin); UpdateIcon(pWMInfo, pNode->msg.iWindow); /* Reshape */ { WindowPtr pWin = GetProp(pNode->msg.hwndWindow, WIN_WINDOW_PROP); if (pWin) { winReshapeMultiWindow(pWin); winUpdateRgnMultiWindow(pWin); } } } break; case WM_WM_UNMAP: /* Unmap the window */ xcb_unmap_window(pWMInfo->conn, pNode->msg.iWindow); break; case WM_WM_KILL: { /* --- */ if (IsWmProtocolAvailable(pWMInfo, pNode->msg.iWindow, pWMInfo->atmWmDelete)) SendXMessage(pWMInfo->conn, pNode->msg.iWindow, pWMInfo->atmWmProtos, pWMInfo->atmWmDelete); else xcb_kill_client(pWMInfo->conn, pNode->msg.iWindow); } break; case WM_WM_ACTIVATE: /* Set the input focus */ /* ICCCM 4.1.7 is pretty opaque, but it appears that the rules are actually quite simple: -- the WM_HINTS input field determines whether the WM should call XSetInputFocus() -- independently, the WM_TAKE_FOCUS protocol determines whether the WM should send a WM_TAKE_FOCUS ClientMessage. */ { Bool neverFocus = FALSE; xcb_get_property_cookie_t cookie; xcb_icccm_wm_hints_t hints; cookie = xcb_icccm_get_wm_hints(pWMInfo->conn, pNode->msg.iWindow); if (xcb_icccm_get_wm_hints_reply(pWMInfo->conn, cookie, &hints, NULL)) { if (hints.flags & XCB_ICCCM_WM_HINT_INPUT) neverFocus = !hints.input; } if (!neverFocus) xcb_set_input_focus(pWMInfo->conn, XCB_INPUT_FOCUS_POINTER_ROOT, pNode->msg.iWindow, XCB_CURRENT_TIME); if (IsWmProtocolAvailable(pWMInfo, pNode->msg.iWindow, pWMInfo->atmWmTakeFocus)) SendXMessage(pWMInfo->conn, pNode->msg.iWindow, pWMInfo->atmWmProtos, pWMInfo->atmWmTakeFocus); } break; case WM_WM_NAME_EVENT: UpdateName(pWMInfo, pNode->msg.iWindow); break; case WM_WM_ICON_EVENT: UpdateIcon(pWMInfo, pNode->msg.iWindow); break; case WM_WM_HINTS_EVENT: { unsigned long maxmin = 0; /* Don't do anything if this is an override-redirect window */ if (IsOverrideRedirect(pWMInfo->conn, pNode->msg.iWindow)) break; UpdateStyle(pWMInfo, pNode->msg.iWindow, &maxmin); } break; case WM_WM_CHANGE_STATE: UpdateState(pWMInfo, pNode->msg.iWindow, pNode->msg.dwID); break; default: ErrorF("winMultiWindowWMProc - Unknown Message. Exiting.\n"); pthread_exit(NULL); break; } /* Free the retrieved message */ free(pNode); /* Flush any pending events on our display */ xcb_flush(pWMInfo->conn); /* This is just laziness rather than making sure we used _checked everywhere */ { xcb_generic_event_t *event = xcb_poll_for_event(pWMInfo->conn); if (event) { if ((event->response_type & ~0x80) == 0) { xcb_generic_error_t *err = (xcb_generic_error_t *)event; winDebug("winMultiWindowWMProc - Error code: %i, ID: 0x%08x, " "Major opcode: %i, Minor opcode: %i\n", err->error_code, err->resource_id, err->major_code, err->minor_code); } } } /* I/O errors etc. */ { int e = xcb_connection_has_error(pWMInfo->conn); if (e) { ErrorF("winMultiWindowWMProc - Fatal error %d on xcb connection\n", e); pthread_exit(NULL); break; } } } /* Free the condition variable */ pthread_cond_destroy(&pWMInfo->wmMsgQueue.pcNotEmpty); /* Free the mutex variable */ pthread_mutex_destroy(&pWMInfo->wmMsgQueue.pmMutex); /* Free the passed-in argument */ free(pProcArg); #if CYGMULTIWINDOW_DEBUG ErrorF("-winMultiWindowWMProc ()\n"); #endif pthread_cleanup_pop(0); return NULL; } static xcb_atom_t intern_atom(xcb_connection_t *conn, const char *atomName) { xcb_intern_atom_reply_t *atom_reply; xcb_intern_atom_cookie_t atom_cookie; xcb_atom_t atom = XCB_ATOM_NONE; atom_cookie = xcb_intern_atom(conn, 0, strlen(atomName), atomName); atom_reply = xcb_intern_atom_reply(conn, atom_cookie, NULL); if (atom_reply) { atom = atom_reply->atom; free(atom_reply); } return atom; } /* * X message procedure */ static void * winMultiWindowXMsgProc(void *pArg) { winWMMessageRec msg; XMsgProcArgPtr pProcArg = (XMsgProcArgPtr) pArg; char pszDisplay[512]; int iRetries; xcb_atom_t atmWmName; xcb_atom_t atmNetWmName; xcb_atom_t atmWmHints; xcb_atom_t atmWmChange; xcb_atom_t atmNetWmIcon; xcb_atom_t atmWindowState, atmMotifWmHints, atmWindowType, atmNormalHints; int iReturn; xcb_auth_info_t *auth_info; xcb_screen_t *root_screen; xcb_window_t root_window_id; pthread_cleanup_push(&winMultiWindowThreadExit, NULL); winDebug("winMultiWindowXMsgProc - Hello\n"); /* Check that argument pointer is not invalid */ if (pProcArg == NULL) { ErrorF("winMultiWindowXMsgProc - pProcArg is NULL. Exiting.\n"); pthread_exit(NULL); } winDebug("winMultiWindowXMsgProc - Calling pthread_mutex_lock ()\n"); /* Grab the server started mutex - pause until we get it */ iReturn = pthread_mutex_lock(pProcArg->ppmServerStarted); if (iReturn != 0) { ErrorF("winMultiWindowXMsgProc - pthread_mutex_lock () failed: %d. " "Exiting.\n", iReturn); pthread_exit(NULL); } winDebug("winMultiWindowXMsgProc - pthread_mutex_lock () returned.\n"); /* Release the server started mutex */ pthread_mutex_unlock(pProcArg->ppmServerStarted); winDebug("winMultiWindowXMsgProc - pthread_mutex_unlock () returned.\n"); /* Setup the display connection string x */ winGetDisplayName(pszDisplay, (int) pProcArg->dwScreen); /* Print the display connection string */ ErrorF("winMultiWindowXMsgProc - DISPLAY=%s\n", pszDisplay); /* Use our generated cookie for authentication */ auth_info = winGetXcbAuthInfo(); /* Initialize retry count */ iRetries = 0; /* Open the X display */ do { /* Try to open the display */ pProcArg->conn = xcb_connect_to_display_with_auth_info(pszDisplay, auth_info, NULL); if (xcb_connection_has_error(pProcArg->conn)) { ErrorF("winMultiWindowXMsgProc - Could not open display, try: %d, " "sleeping: %d\n", iRetries + 1, WIN_CONNECT_DELAY); ++iRetries; sleep(WIN_CONNECT_DELAY); continue; } else break; } while (xcb_connection_has_error(pProcArg->conn) && iRetries < WIN_CONNECT_RETRIES); /* Make sure that the display opened */ if (xcb_connection_has_error(pProcArg->conn)) { ErrorF("winMultiWindowXMsgProc - Failed opening the display. " "Exiting.\n"); pthread_exit(NULL); } ErrorF("winMultiWindowXMsgProc - xcb_connect() returned and " "successfully opened the display.\n"); /* Check if another window manager is already running */ if (CheckAnotherWindowManager(pProcArg->conn, pProcArg->dwScreen)) { ErrorF("winMultiWindowXMsgProc - " "another window manager is running. Exiting.\n"); pthread_exit(NULL); } /* Get root window id */ root_screen = xcb_aux_get_screen(pProcArg->conn, pProcArg->dwScreen); root_window_id = root_screen->root; { /* Set WM_ICON_SIZE property indicating desired icon sizes */ typedef struct { uint32_t min_width, min_height; uint32_t max_width, max_height; int32_t width_inc, height_inc; } xcb_wm_icon_size_hints_hints_t; xcb_wm_icon_size_hints_hints_t xis; xis.min_width = xis.min_height = 16; xis.max_width = xis.max_height = 48; xis.width_inc = xis.height_inc = 16; xcb_change_property(pProcArg->conn, XCB_PROP_MODE_REPLACE, root_window_id, XCB_ATOM_WM_ICON_SIZE, XCB_ATOM_WM_ICON_SIZE, 32, sizeof(xis)/4, &xis); } atmWmName = intern_atom(pProcArg->conn, "WM_NAME"); atmNetWmName = intern_atom(pProcArg->conn, "_NET_WM_NAME"); atmWmHints = intern_atom(pProcArg->conn, "WM_HINTS"); atmWmChange = intern_atom(pProcArg->conn, "WM_CHANGE_STATE"); atmNetWmIcon = intern_atom(pProcArg->conn, "_NET_WM_ICON"); atmWindowState = intern_atom(pProcArg->conn, "_NET_WM_STATE"); atmMotifWmHints = intern_atom(pProcArg->conn, "_MOTIF_WM_HINTS"); atmWindowType = intern_atom(pProcArg->conn, "_NET_WM_WINDOW_TYPE"); atmNormalHints = intern_atom(pProcArg->conn, "WM_NORMAL_HINTS"); /* Enable Composite extension and redirect subwindows of the root window */ if (pProcArg->pWMInfo->fCompositeWM) { const char *extension_name = "Composite"; xcb_query_extension_cookie_t cookie; xcb_query_extension_reply_t *reply; cookie = xcb_query_extension(pProcArg->conn, strlen(extension_name), extension_name); reply = xcb_query_extension_reply(pProcArg->conn, cookie, NULL); if (reply && (reply->present)) { xcb_composite_redirect_subwindows(pProcArg->conn, root_window_id, XCB_COMPOSITE_REDIRECT_AUTOMATIC); /* We use automatic updating of the root window for two reasons: 1) redirected window contents are mirrored to the root window so that the root window draws correctly when shown. 2) updating the root window causes damage against the shadow framebuffer, which ultimately causes WM_PAINT to be sent to the affected window(s) to cause the damage regions to be redrawn. */ ErrorF("Using Composite redirection\n"); free(reply); } } /* Loop until we explicitly break out */ while (1) { xcb_generic_event_t *event; uint8_t type; Bool send_event; if (g_shutdown) break; /* Fetch next event */ event = xcb_wait_for_event(pProcArg->conn); if (!event) { // returns NULL on I/O error int e = xcb_connection_has_error(pProcArg->conn); ErrorF("winMultiWindowXMsgProc - Fatal error %d on xcb connection\n", e); pthread_exit(NULL); break; } type = event->response_type & ~0x80; send_event = event->response_type & 0x80; winDebug("winMultiWindowXMsgProc - event %d\n", type); /* Branch on event type */ if (type == 0) { xcb_generic_error_t *err = (xcb_generic_error_t *)event; winDebug("winMultiWindowXMsgProc - Error code: %i, ID: 0x%08x, " "Major opcode: %i, Minor opcode: %i\n", err->error_code, err->resource_id, err->major_code, err->minor_code); } else if (type == XCB_CREATE_NOTIFY) { xcb_create_notify_event_t *notify = (xcb_create_notify_event_t *)event; /* Request property change events */ const static uint32_t mask_value[] = { XCB_EVENT_MASK_PROPERTY_CHANGE }; xcb_change_window_attributes (pProcArg->conn, notify->window, XCB_CW_EVENT_MASK, mask_value); /* If it's not override-redirect, set the border-width to 0 */ if (!IsOverrideRedirect(pProcArg->conn, notify->window)) { const static uint32_t width_value[] = { 0 }; xcb_configure_window(pProcArg->conn, notify->window, XCB_CONFIG_WINDOW_BORDER_WIDTH, width_value); } } else if (type == XCB_MAP_NOTIFY) { /* Fake a reparentNotify event as SWT/Motif expects a Window Manager to reparent a top-level window when it is mapped and waits until they do. We don't actually need to reparent, as the frame is a native window, not an X window We do this on MapNotify, not MapRequest like a real Window Manager would, so we don't have do get involved in actually mapping the window via it's (non-existent) parent... See sourceware bugzilla #9848 */ xcb_map_notify_event_t *notify = (xcb_map_notify_event_t *)event; xcb_get_geometry_cookie_t cookie; xcb_get_geometry_reply_t *reply; xcb_query_tree_cookie_t cookie_qt; xcb_query_tree_reply_t *reply_qt; cookie = xcb_get_geometry(pProcArg->conn, notify->window); cookie_qt = xcb_query_tree(pProcArg->conn, notify->window); reply = xcb_get_geometry_reply(pProcArg->conn, cookie, NULL); reply_qt = xcb_query_tree_reply(pProcArg->conn, cookie_qt, NULL); if (reply && reply_qt) { /* It's a top-level window if the parent window is a root window Only non-override_redirect windows can get reparented */ if ((reply->root == reply_qt->parent) && !notify->override_redirect) { xcb_reparent_notify_event_t event_send; event_send.response_type = ReparentNotify; event_send.event = notify->window; event_send.window = notify->window; event_send.parent = reply_qt->parent; event_send.x = reply->x; event_send.y = reply->y; xcb_send_event (pProcArg->conn, TRUE, notify->window, XCB_EVENT_MASK_STRUCTURE_NOTIFY, (const char *)&event_send); free(reply_qt); free(reply); } } } else if (type == XCB_UNMAP_NOTIFY) { xcb_unmap_notify_event_t *notify = (xcb_unmap_notify_event_t *)event; memset(&msg, 0, sizeof(msg)); msg.msg = WM_WM_CHANGE_STATE; msg.iWindow = notify->window; msg.dwID = XCB_ICCCM_WM_STATE_WITHDRAWN; winSendMessageToWM(pProcArg->pWMInfo, &msg); } else if (type == XCB_CONFIGURE_NOTIFY) { if (!send_event) { /* Java applications using AWT on JRE 1.6.0 break with non-reparenting WMs AWT doesn't explicitly know about (See sun bug #6434227) XDecoratedPeer.handleConfigureNotifyEvent() only processes non-synthetic ConfigureNotify events to update window location if it's identified the WM as a non-reparenting WM it knows about (compiz or lookingglass) Rather than tell all sorts of lies to get XWM to recognize us as one of those, simply send a synthetic ConfigureNotify for every non-synthetic one */ xcb_configure_notify_event_t *notify = (xcb_configure_notify_event_t *)event; xcb_configure_notify_event_t event_send = *notify; event_send.event = notify->window; xcb_send_event(pProcArg->conn, TRUE, notify->window, XCB_EVENT_MASK_STRUCTURE_NOTIFY, (const char *)&event_send); } } else if (type == XCB_PROPERTY_NOTIFY) { xcb_property_notify_event_t *notify = (xcb_property_notify_event_t *)event; xcb_get_atom_name_cookie_t cookie = xcb_get_atom_name(pProcArg->conn, notify->atom); xcb_get_atom_name_reply_t *reply = xcb_get_atom_name_reply(pProcArg->conn, cookie, NULL); if (reply) { winDebug("winMultiWindowXMsgProc: PropertyNotify %.*s\n", xcb_get_atom_name_name_length(reply), xcb_get_atom_name_name(reply)); free(reply); } if ((notify->atom == atmWmName) || (notify->atom == atmNetWmName)) { memset(&msg, 0, sizeof(msg)); msg.msg = WM_WM_NAME_EVENT; msg.iWindow = notify->window; /* Other fields ignored */ winSendMessageToWM(pProcArg->pWMInfo, &msg); } else { /* Several properties are considered for WM hints, check if this property change affects any of them... (this list needs to be kept in sync with winApplyHints()) */ if ((notify->atom == atmWmHints) || (notify->atom == atmWindowState) || (notify->atom == atmMotifWmHints) || (notify->atom == atmWindowType) || (notify->atom == atmNormalHints)) { memset(&msg, 0, sizeof(msg)); msg.msg = WM_WM_HINTS_EVENT; msg.iWindow = notify->window; /* Other fields ignored */ winSendMessageToWM(pProcArg->pWMInfo, &msg); } /* Not an else as WM_HINTS affects both style and icon */ if ((notify->atom == atmWmHints) || (notify->atom == atmNetWmIcon)) { memset(&msg, 0, sizeof(msg)); msg.msg = WM_WM_ICON_EVENT; msg.iWindow = notify->window; /* Other fields ignored */ winSendMessageToWM(pProcArg->pWMInfo, &msg); } } } else if (type == XCB_CLIENT_MESSAGE) { xcb_client_message_event_t *client_msg = (xcb_client_message_event_t *)event; winDebug("winMultiWindowXMsgProc: ClientMessage: type %d window 0x%08x data[0] %d\n", client_msg->type, client_msg->window, client_msg->data.data32[0]); if (client_msg->type == atmWmChange && client_msg->data.data32[0] == XCB_ICCCM_WM_STATE_ICONIC) { ErrorF("winMultiWindowXMsgProc - WM_CHANGE_STATE - IconicState\n"); memset(&msg, 0, sizeof(msg)); msg.msg = WM_WM_CHANGE_STATE; msg.iWindow = client_msg->window; msg.dwID = client_msg->data.data32[0]; winSendMessageToWM(pProcArg->pWMInfo, &msg); } else if (client_msg->type == pProcArg->pWMInfo->ewmh._NET_WM_STATE) { int action = client_msg->data.data32[0]; int state = -1; if (action == XCB_EWMH_WM_STATE_ADD) { if ((client_msg->data.data32[1] == pProcArg->pWMInfo->ewmh._NET_WM_STATE_MAXIMIZED_VERT) && (client_msg->data.data32[2] == pProcArg->pWMInfo->ewmh._NET_WM_STATE_MAXIMIZED_HORZ)) state = XCB_ICCCM_WM_STATE_ZOOM; if ((client_msg->data.data32[1] == pProcArg->pWMInfo->ewmh._NET_WM_STATE_MAXIMIZED_HORZ) && (client_msg->data.data32[2] == pProcArg->pWMInfo->ewmh._NET_WM_STATE_MAXIMIZED_VERT)) state = XCB_ICCCM_WM_STATE_ZOOM; if (client_msg->data.data32[1] == pProcArg->pWMInfo->ewmh._NET_WM_STATE_HIDDEN) state = XCB_ICCCM_WM_STATE_ICONIC; } else if (action == XCB_EWMH_WM_STATE_REMOVE) { if ((client_msg->data.data32[1] == pProcArg->pWMInfo->ewmh._NET_WM_STATE_MAXIMIZED_VERT) && (client_msg->data.data32[2] == pProcArg->pWMInfo->ewmh._NET_WM_STATE_MAXIMIZED_HORZ)) state = XCB_ICCCM_WM_STATE_NORMAL; if ((client_msg->data.data32[1] == pProcArg->pWMInfo->ewmh._NET_WM_STATE_MAXIMIZED_HORZ) && (client_msg->data.data32[2] == pProcArg->pWMInfo->ewmh._NET_WM_STATE_MAXIMIZED_VERT)) state = XCB_ICCCM_WM_STATE_NORMAL; if (client_msg->data.data32[1] == pProcArg->pWMInfo->ewmh._NET_WM_STATE_HIDDEN) state = XCB_ICCCM_WM_STATE_NORMAL; } else { ErrorF("winMultiWindowXMsgProc: ClientMEssage _NET_WM_STATE unsupported action %d\n", action); } if (state != -1) { memset(&msg, 0, sizeof(msg)); msg.msg = WM_WM_CHANGE_STATE; msg.iWindow = client_msg->window; msg.dwID = state; winSendMessageToWM(pProcArg->pWMInfo, &msg); } } } /* Free the event */ free(event); } xcb_disconnect(pProcArg->conn); pthread_cleanup_pop(0); return NULL; } /* * winInitWM - Entry point for the X server to spawn * the Window Manager thread. Called from * winscrinit.c/winFinishScreenInitFB (). */ Bool winInitWM(void **ppWMInfo, pthread_t * ptWMProc, pthread_t * ptXMsgProc, pthread_mutex_t * ppmServerStarted, int dwScreen, HWND hwndScreen, Bool compositeWM) { WMProcArgPtr pArg = malloc(sizeof(WMProcArgRec)); WMInfoPtr pWMInfo = malloc(sizeof(WMInfoRec)); XMsgProcArgPtr pXMsgArg = malloc(sizeof(XMsgProcArgRec)); /* Bail if the input parameters are bad */ if (pArg == NULL || pWMInfo == NULL || pXMsgArg == NULL) { ErrorF("winInitWM - malloc failed.\n"); free(pArg); free(pWMInfo); free(pXMsgArg); return FALSE; } /* Zero the allocated memory */ ZeroMemory(pArg, sizeof(WMProcArgRec)); ZeroMemory(pWMInfo, sizeof(WMInfoRec)); ZeroMemory(pXMsgArg, sizeof(XMsgProcArgRec)); /* Set a return pointer to the Window Manager info structure */ *ppWMInfo = pWMInfo; pWMInfo->fCompositeWM = compositeWM; /* Setup the argument structure for the thread function */ pArg->dwScreen = dwScreen; pArg->pWMInfo = pWMInfo; pArg->ppmServerStarted = ppmServerStarted; /* Intialize the message queue */ if (!InitQueue(&pWMInfo->wmMsgQueue)) { ErrorF("winInitWM - InitQueue () failed.\n"); return FALSE; } /* Spawn a thread for the Window Manager */ if (pthread_create(ptWMProc, NULL, winMultiWindowWMProc, pArg)) { /* Bail if thread creation failed */ ErrorF("winInitWM - pthread_create failed for Window Manager.\n"); return FALSE; } /* Spawn the XNextEvent thread, will send messages to WM */ pXMsgArg->dwScreen = dwScreen; pXMsgArg->pWMInfo = pWMInfo; pXMsgArg->ppmServerStarted = ppmServerStarted; pXMsgArg->hwndScreen = hwndScreen; if (pthread_create(ptXMsgProc, NULL, winMultiWindowXMsgProc, pXMsgArg)) { /* Bail if thread creation failed */ ErrorF("winInitWM - pthread_create failed on XMSG.\n"); return FALSE; } #if CYGDEBUG || YES winDebug("winInitWM - Returning.\n"); #endif return TRUE; } /* * Window manager thread - setup */ static void winInitMultiWindowWM(WMInfoPtr pWMInfo, WMProcArgPtr pProcArg) { int iRetries = 0; char pszDisplay[512]; int iReturn; xcb_auth_info_t *auth_info; xcb_screen_t *root_screen; xcb_window_t root_window_id; winDebug("winInitMultiWindowWM - Hello\n"); /* Check that argument pointer is not invalid */ if (pProcArg == NULL) { ErrorF("winInitMultiWindowWM - pProcArg is NULL. Exiting.\n"); pthread_exit(NULL); } winDebug("winInitMultiWindowWM - Calling pthread_mutex_lock ()\n"); /* Grab our garbage mutex to satisfy pthread_cond_wait */ iReturn = pthread_mutex_lock(pProcArg->ppmServerStarted); if (iReturn != 0) { ErrorF("winInitMultiWindowWM - pthread_mutex_lock () failed: %d. " "Exiting.\n", iReturn); pthread_exit(NULL); } winDebug("winInitMultiWindowWM - pthread_mutex_lock () returned.\n"); /* Release the server started mutex */ pthread_mutex_unlock(pProcArg->ppmServerStarted); winDebug("winInitMultiWindowWM - pthread_mutex_unlock () returned.\n"); /* Setup the display connection string x */ winGetDisplayName(pszDisplay, (int) pProcArg->dwScreen); /* Print the display connection string */ ErrorF("winInitMultiWindowWM - DISPLAY=%s\n", pszDisplay); /* Use our generated cookie for authentication */ auth_info = winGetXcbAuthInfo(); /* Open the X display */ do { /* Try to open the display */ pWMInfo->conn = xcb_connect_to_display_with_auth_info(pszDisplay, auth_info, NULL); if (xcb_connection_has_error(pWMInfo->conn)) { ErrorF("winInitMultiWindowWM - Could not open display, try: %d, " "sleeping: %d\n", iRetries + 1, WIN_CONNECT_DELAY); ++iRetries; sleep(WIN_CONNECT_DELAY); continue; } else break; } while (xcb_connection_has_error(pWMInfo->conn) && iRetries < WIN_CONNECT_RETRIES); /* Make sure that the display opened */ if (xcb_connection_has_error(pWMInfo->conn)) { ErrorF("winInitMultiWindowWM - Failed opening the display. " "Exiting.\n"); pthread_exit(NULL); } ErrorF("winInitMultiWindowWM - xcb_connect () returned and " "successfully opened the display.\n"); /* Create some atoms */ pWMInfo->atmWmProtos = intern_atom(pWMInfo->conn, "WM_PROTOCOLS"); pWMInfo->atmWmDelete = intern_atom(pWMInfo->conn, "WM_DELETE_WINDOW"); pWMInfo->atmWmTakeFocus = intern_atom(pWMInfo->conn, "WM_TAKE_FOCUS"); pWMInfo->atmPrivMap = intern_atom(pWMInfo->conn, WINDOWSWM_NATIVE_HWND); pWMInfo->atmUtf8String = intern_atom(pWMInfo->conn, "UTF8_STRING"); pWMInfo->atmNetWmName = intern_atom(pWMInfo->conn, "_NET_WM_NAME"); pWMInfo->atmCurrentDesktop = intern_atom(pWMInfo->conn, "_NET_CURRENT_DESKTOP"); pWMInfo->atmNumberDesktops = intern_atom(pWMInfo->conn, "_NET_NUMBER_OF_DESKTOPS"); pWMInfo->atmDesktopNames = intern_atom(pWMInfo->conn, "__NET_DESKTOP_NAMES"); pWMInfo->atmWmState = intern_atom(pWMInfo->conn, "WM_STATE"); /* Initialization for the xcb_ewmh and EWMH atoms */ { xcb_intern_atom_cookie_t *atoms_cookie; atoms_cookie = xcb_ewmh_init_atoms(pWMInfo->conn, &pWMInfo->ewmh); if (xcb_ewmh_init_atoms_replies(&pWMInfo->ewmh, atoms_cookie, NULL)) { /* Set the _NET_SUPPORTED atom for this context. TODO: Audit to ensure we implement everything defined as MUSTs for window managers in the EWMH standard.*/ xcb_atom_t supported[] = { pWMInfo->ewmh.WM_PROTOCOLS, pWMInfo->ewmh._NET_SUPPORTED, pWMInfo->ewmh._NET_SUPPORTING_WM_CHECK, pWMInfo->ewmh._NET_CLOSE_WINDOW, pWMInfo->ewmh._NET_WM_WINDOW_TYPE, pWMInfo->ewmh._NET_WM_WINDOW_TYPE_DOCK, pWMInfo->ewmh._NET_WM_WINDOW_TYPE_SPLASH, pWMInfo->ewmh._NET_WM_STATE, pWMInfo->ewmh._NET_WM_STATE_HIDDEN, pWMInfo->ewmh._NET_WM_STATE_ABOVE, pWMInfo->ewmh._NET_WM_STATE_BELOW, pWMInfo->ewmh._NET_WM_STATE_SKIP_TASKBAR, pWMInfo->ewmh._NET_WM_STATE_MAXIMIZED_VERT, pWMInfo->ewmh._NET_WM_STATE_MAXIMIZED_HORZ, }; xcb_ewmh_set_supported(&pWMInfo->ewmh, pProcArg->dwScreen, sizeof(supported)/sizeof(xcb_atom_t), supported); } else { ErrorF("winInitMultiWindowWM - xcb_ewmh_init_atoms() failed\n"); } } /* Get root window id */ root_screen = xcb_aux_get_screen(pWMInfo->conn, pProcArg->dwScreen); root_window_id = root_screen->root; /* Set root window properties for describing multiple desktops to describe the one desktop we have */ { int data; const char buf[] = "Desktop"; data = 0; xcb_change_property(pWMInfo->conn, XCB_PROP_MODE_REPLACE, root_window_id, pWMInfo->atmCurrentDesktop, XCB_ATOM_CARDINAL, 32, 1, &data); data = 1; xcb_change_property(pWMInfo->conn, XCB_PROP_MODE_REPLACE, root_window_id, pWMInfo->atmNumberDesktops, XCB_ATOM_CARDINAL, 32, 1, &data); xcb_change_property(pWMInfo->conn, XCB_PROP_MODE_REPLACE, root_window_id, pWMInfo->atmDesktopNames, pWMInfo->atmUtf8String, 8, strlen(buf), (unsigned char *) buf); } /* Set the root window cursor to left_ptr (this controls the cursor an application gets over it's windows when it doesn't set one) */ { #define XC_left_ptr 68 xcb_cursor_t cursor = xcb_generate_id(pWMInfo->conn); xcb_font_t font = xcb_generate_id(pWMInfo->conn); xcb_font_t *mask_font = &font; /* An alias to clarify */ int shape = XC_left_ptr; uint32_t mask = XCB_CW_CURSOR; uint32_t value_list = cursor; static const uint16_t fgred = 0, fggreen = 0, fgblue = 0; static const uint16_t bgred = 0xFFFF, bggreen = 0xFFFF, bgblue = 0xFFFF; xcb_open_font(pWMInfo->conn, font, sizeof("cursor"), "cursor"); xcb_create_glyph_cursor(pWMInfo->conn, cursor, font, *mask_font, shape, shape + 1, fgred, fggreen, fgblue, bgred, bggreen, bgblue); xcb_change_window_attributes(pWMInfo->conn, root_window_id, mask, &value_list); xcb_free_cursor(pWMInfo->conn, cursor); xcb_close_font(pWMInfo->conn, font); } } /* * winSendMessageToWM - Send a message from the X thread to the WM thread */ void winSendMessageToWM(void *pWMInfo, winWMMessagePtr pMsg) { WMMsgNodePtr pNode; #if CYGMULTIWINDOW_DEBUG ErrorF("winSendMessageToWM %s %08x %d\n", MessageName(pMsg), pMsg->iWindow, pMsg->dwID); #endif pNode = malloc(sizeof(WMMsgNodeRec)); if (pNode != NULL) { memcpy(&pNode->msg, pMsg, sizeof(winWMMessageRec)); PushMessage(&((WMInfoPtr) pWMInfo)->wmMsgQueue, pNode); } } /* * winMultiWindowThreadExit - Thread exit handler */ static void winMultiWindowThreadExit(void *arg) { /* multiwindow client thread has exited, stop server as well */ raise(SIGTERM); } /* * Check if another window manager is running */ static Bool CheckAnotherWindowManager(xcb_connection_t *conn, DWORD dwScreen) { Bool redirectError = FALSE; /* Get root window id */ xcb_screen_t *root_screen = xcb_aux_get_screen(conn, dwScreen); xcb_window_t root_window_id = root_screen->root; /* Try to select the events which only one client at a time is allowed to select. If this causes an error, another window manager is already running... */ const static uint32_t test_mask[] = { XCB_EVENT_MASK_RESIZE_REDIRECT | XCB_EVENT_MASK_SUBSTRUCTURE_REDIRECT | XCB_EVENT_MASK_BUTTON_PRESS }; xcb_void_cookie_t cookie = xcb_change_window_attributes_checked(conn, root_window_id, XCB_CW_EVENT_MASK, test_mask); xcb_generic_error_t *error; if ((error = xcb_request_check(conn, cookie))) { redirectError = TRUE; free(error); } /* Side effect: select the events we are actually interested in... Other WMs are not allowed, also select one of the events which only one client at a time is allowed to select, so other window managers won't start... */ { const uint32_t mask[] = { XCB_EVENT_MASK_SUBSTRUCTURE_NOTIFY | XCB_EVENT_MASK_BUTTON_PRESS }; xcb_change_window_attributes(conn, root_window_id, XCB_CW_EVENT_MASK, mask); } return redirectError; } /* * Notify the MWM thread we're exiting and not to reconnect */ void winDeinitMultiWindowWM(void) { ErrorF("winDeinitMultiWindowWM - Noting shutdown in progress\n"); g_shutdown = TRUE; } static void winApplyUrgency(WMInfoPtr pWMInfo, xcb_window_t iWindow, HWND hWnd) { xcb_get_property_cookie_t cookie; xcb_icccm_wm_hints_t hints; cookie = xcb_icccm_get_wm_hints(pWMInfo->conn, iWindow); if (xcb_icccm_get_wm_hints_reply(pWMInfo->conn, cookie, &hints, NULL)) { FLASHWINFO fwi; fwi.cbSize = sizeof(FLASHWINFO); fwi.hwnd = hWnd; winDebug("winApplyUrgency: window 0x%08x has urgency hint %s\n", iWindow, (hints.flags & XCB_ICCCM_WM_HINT_X_URGENCY) ? "on" : "off"); if (hints.flags & XCB_ICCCM_WM_HINT_X_URGENCY) { DWORD count = 3; SystemParametersInfo(SPI_GETFOREGROUNDFLASHCOUNT, 0, &count, 0); fwi.dwFlags = FLASHW_TRAY; fwi.uCount = count; fwi.dwTimeout = 0; } else { fwi.dwFlags = FLASHW_STOP; fwi.uCount = 0; fwi.dwTimeout = 0; } FlashWindowEx(&fwi); } } /* Windows window styles */ #define HINT_NOFRAME (1L<<0) #define HINT_BORDER (1L<<1) #define HINT_SIZEBOX (1L<<2) #define HINT_CAPTION (1L<<3) #define HINT_NOMAXIMIZE (1L<<4) #define HINT_NOMINIMIZE (1L<<5) #define HINT_NOSYSMENU (1L<<6) #define HINT_SKIPTASKBAR (1L<<7) static void winApplyHints(WMInfoPtr pWMInfo, xcb_window_t iWindow, HWND hWnd, HWND * zstyle, unsigned long *maxmin) { xcb_connection_t *conn = pWMInfo->conn; static xcb_atom_t windowState, motif_wm_hints; static xcb_atom_t hiddenState, fullscreenState, belowState, aboveState, skiptaskbarState; static xcb_atom_t splashType; static int generation; unsigned long hint = HINT_BORDER | HINT_SIZEBOX | HINT_CAPTION; unsigned long style, exStyle; *maxmin = 0; if (!hWnd) return; if (!IsWindow(hWnd)) return; if (generation != serverGeneration) { generation = serverGeneration; windowState = intern_atom(conn, "_NET_WM_STATE"); motif_wm_hints = intern_atom(conn, "_MOTIF_WM_HINTS"); hiddenState = intern_atom(conn, "_NET_WM_STATE_HIDDEN"); fullscreenState = intern_atom(conn, "_NET_WM_STATE_FULLSCREEN"); belowState = intern_atom(conn, "_NET_WM_STATE_BELOW"); aboveState = intern_atom(conn, "_NET_WM_STATE_ABOVE"); skiptaskbarState = intern_atom(conn, "_NET_WM_STATE_SKIP_TASKBAR"); splashType = intern_atom(conn, "_NET_WM_WINDOW_TYPE_SPLASHSCREEN"); } { xcb_get_property_cookie_t cookie_wm_state = xcb_get_property(conn, FALSE, iWindow, windowState, XCB_ATOM_ATOM, 0L, INT_MAX); xcb_get_property_reply_t *reply = xcb_get_property_reply(conn, cookie_wm_state, NULL); if (reply) { int i; int nitems = xcb_get_property_value_length(reply)/sizeof(xcb_atom_t); xcb_atom_t *pAtom = xcb_get_property_value(reply); Bool verMax = FALSE; Bool horMax = FALSE; for (i = 0; i < nitems; i++) { if (pAtom[i] == skiptaskbarState) hint |= HINT_SKIPTASKBAR; if (pAtom[i] == hiddenState) *maxmin |= HINT_MIN; else if (pAtom[i] == fullscreenState) *maxmin |= HINT_MAX; if (pAtom[i] == belowState) *zstyle = HWND_BOTTOM; else if (pAtom[i] == aboveState) *zstyle = HWND_TOPMOST; if (pAtom[i] == pWMInfo->ewmh._NET_WM_STATE_MAXIMIZED_VERT) verMax = TRUE; if (pAtom[i] == pWMInfo->ewmh._NET_WM_STATE_MAXIMIZED_HORZ) horMax = TRUE; } if (verMax && horMax) *maxmin |= HINT_MAX; free(reply); } } { xcb_get_property_cookie_t cookie_mwm_hint = xcb_get_property(conn, FALSE, iWindow, motif_wm_hints, motif_wm_hints, 0L, sizeof(MwmHints)); xcb_get_property_reply_t *reply = xcb_get_property_reply(conn, cookie_mwm_hint, NULL); if (reply) { int nitems = xcb_get_property_value_length(reply)/4; MwmHints *mwm_hint = xcb_get_property_value(reply); if (mwm_hint && (nitems >= PropMwmHintsElements) && (mwm_hint->flags & MwmHintsDecorations)) { if (!mwm_hint->decorations) { hint &= ~(HINT_BORDER | HINT_SIZEBOX | HINT_CAPTION); hint |= (HINT_NOFRAME | HINT_NOSYSMENU | HINT_NOMINIMIZE | HINT_NOMAXIMIZE); } else if (!(mwm_hint->decorations & MwmDecorAll)) { if (!(mwm_hint->decorations & MwmDecorBorder)) hint &= ~HINT_BORDER; if (!(mwm_hint->decorations & MwmDecorHandle)) hint &= ~HINT_SIZEBOX; if (!(mwm_hint->decorations & MwmDecorTitle)) hint &= ~HINT_CAPTION; if (!(mwm_hint->decorations & MwmDecorMenu)) hint |= HINT_NOSYSMENU; if (!(mwm_hint->decorations & MwmDecorMinimize)) hint |= HINT_NOMINIMIZE; if (!(mwm_hint->decorations & MwmDecorMaximize)) hint |= HINT_NOMAXIMIZE; } else { /* MwmDecorAll means all decorations *except* those specified by other flag bits that are set. Not yet implemented. */ } } free(reply); } } { int i; xcb_ewmh_get_atoms_reply_t type; xcb_get_property_cookie_t cookie = xcb_ewmh_get_wm_window_type(&pWMInfo->ewmh, iWindow); if (xcb_ewmh_get_wm_window_type_reply(&pWMInfo->ewmh, cookie, &type, NULL)) { for (i = 0; i < type.atoms_len; i++) { if (type.atoms[i] == pWMInfo->ewmh._NET_WM_WINDOW_TYPE_DOCK) { hint &= ~(HINT_BORDER | HINT_SIZEBOX | HINT_CAPTION | HINT_NOFRAME); hint |= (HINT_SKIPTASKBAR | HINT_SIZEBOX); *zstyle = HWND_TOPMOST; } else if ((type.atoms[i] == pWMInfo->ewmh._NET_WM_WINDOW_TYPE_SPLASH) || (type.atoms[i] == splashType)) { hint &= ~(HINT_BORDER | HINT_SIZEBOX | HINT_CAPTION); hint |= (HINT_SKIPTASKBAR | HINT_NOSYSMENU | HINT_NOMINIMIZE | HINT_NOMAXIMIZE); *zstyle = HWND_TOPMOST; } } } } { xcb_size_hints_t size_hints; xcb_get_property_cookie_t cookie; cookie = xcb_icccm_get_wm_normal_hints(conn, iWindow); if (xcb_icccm_get_wm_normal_hints_reply(conn, cookie, &size_hints, NULL)) { /* Notwithstanding MwmDecorHandle, if we have a border, and WM_NORMAL_HINTS indicates the window should be resizeable, let the window have a resizing border. This is necessary for windows with gtk3+ 3.14 csd. */ if (hint & HINT_BORDER) hint |= HINT_SIZEBOX; if (size_hints.flags & XCB_ICCCM_SIZE_HINT_P_MAX_SIZE) { /* Not maximizable if a maximum size is specified, and that size is smaller (in either dimension) than the screen size */ if ((size_hints.max_width < GetSystemMetrics(SM_CXVIRTUALSCREEN)) || (size_hints.max_height < GetSystemMetrics(SM_CYVIRTUALSCREEN))) hint |= HINT_NOMAXIMIZE; if (size_hints.flags & XCB_ICCCM_SIZE_HINT_P_MIN_SIZE) { /* If both minimum size and maximum size are specified and are the same, don't bother with a resizing frame */ if ((size_hints.min_width == size_hints.max_width) && (size_hints.min_height == size_hints.max_height)) { hint |= HINT_NOMAXIMIZE; hint = (hint & ~HINT_SIZEBOX); } } } } } /* Override hint settings from above with settings from config file and set application id for grouping. */ { char *application_id = 0; char *window_name = 0; char *res_name = 0; char *res_class = 0; GetClassNames(pWMInfo, iWindow, &res_name, &res_class, &window_name); style = STYLE_NONE; style = winOverrideStyle(res_name, res_class, window_name); #define APPLICATION_ID_FORMAT "%s.xwin.%s" #define APPLICATION_ID_UNKNOWN "unknown" if (res_class) { asprintf(&application_id, APPLICATION_ID_FORMAT, XVENDORNAME, res_class); } else { asprintf(&application_id, APPLICATION_ID_FORMAT, XVENDORNAME, APPLICATION_ID_UNKNOWN); } winSetAppUserModelID(hWnd, application_id); free(application_id); free(res_name); free(res_class); free(window_name); } if (style & STYLE_TOPMOST) *zstyle = HWND_TOPMOST; else if (style & STYLE_MAXIMIZE) *maxmin = (hint & ~HINT_MIN) | HINT_MAX; else if (style & STYLE_MINIMIZE) *maxmin = (hint & ~HINT_MAX) | HINT_MIN; else if (style & STYLE_BOTTOM) *zstyle = HWND_BOTTOM; if (style & STYLE_NOTITLE) hint = (hint & ~HINT_NOFRAME & ~HINT_BORDER & ~HINT_CAPTION) | HINT_SIZEBOX; else if (style & STYLE_OUTLINE) hint = (hint & ~HINT_NOFRAME & ~HINT_SIZEBOX & ~HINT_CAPTION) | HINT_BORDER; else if (style & STYLE_NOFRAME) hint = (hint & ~HINT_BORDER & ~HINT_CAPTION & ~HINT_SIZEBOX) | HINT_NOFRAME; if (style & STYLE_SKIPTASKBAR) hint |= HINT_SKIPTASKBAR; /* Now apply styles to window */ style = GetWindowLongPtr(hWnd, GWL_STYLE); if (!style) return; /* GetWindowLongPointer returns 0 on failure, we hope this isn't a valid style */ style &= ~WS_CAPTION & ~WS_SIZEBOX; /* Just in case */ if (GetParent(hWnd)) style |= WS_SYSMENU; else style |= WS_SYSMENU | WS_MAXIMIZEBOX | WS_MINIMIZEBOX; if (!(hint & ~HINT_SKIPTASKBAR)) /* No hints, default */ style = style | WS_CAPTION | WS_SIZEBOX; else if (hint & HINT_NOFRAME) /* No frame, no decorations */ style = style & ~WS_CAPTION & ~WS_SIZEBOX; else style = style | ((hint & HINT_BORDER) ? WS_BORDER : 0) | ((hint & HINT_SIZEBOX) ? WS_SIZEBOX : 0) | ((hint & HINT_CAPTION) ? WS_CAPTION : 0); if (hint & HINT_NOMAXIMIZE) style = style & ~WS_MAXIMIZEBOX; if (hint & HINT_NOMINIMIZE) style = style & ~WS_MINIMIZEBOX; if (hint & HINT_NOSYSMENU) style = style & ~WS_SYSMENU; if (hint & HINT_SKIPTASKBAR) style = style & ~WS_MINIMIZEBOX; /* window will become lost if minimized */ SetWindowLongPtr(hWnd, GWL_STYLE, style); exStyle = GetWindowLongPtr(hWnd, GWL_EXSTYLE); if (hint & HINT_SKIPTASKBAR) exStyle = (exStyle & ~WS_EX_APPWINDOW) | WS_EX_TOOLWINDOW; else exStyle = (exStyle & ~WS_EX_TOOLWINDOW) | WS_EX_APPWINDOW; SetWindowLongPtr(hWnd, GWL_EXSTYLE, exStyle); winDebug ("winApplyHints: iWindow 0x%08x hints 0x%08x style 0x%08x exstyle 0x%08x\n", iWindow, hint, style, exStyle); }