2009 lines
		
	
	
		
			64 KiB
		
	
	
	
		
			C
		
	
	
	
			
		
		
	
	
			2009 lines
		
	
	
		
			64 KiB
		
	
	
	
		
			C
		
	
	
	
/*
 | 
						|
 *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
 | 
						|
 */
 | 
						|
 | 
						|
/* X headers */
 | 
						|
#ifdef HAVE_XWIN_CONFIG_H
 | 
						|
#include <xwin-config.h>
 | 
						|
#endif
 | 
						|
#include <stdio.h>
 | 
						|
#include <stdlib.h>
 | 
						|
#include <unistd.h>
 | 
						|
#ifdef __CYGWIN__
 | 
						|
#include <sys/select.h>
 | 
						|
#endif
 | 
						|
#include <fcntl.h>
 | 
						|
#include <setjmp.h>
 | 
						|
#define HANDLE void *
 | 
						|
#include <pthread.h>
 | 
						|
#undef HANDLE
 | 
						|
#include <xcb/xcb.h>
 | 
						|
#include <xcb/xcb_icccm.h>
 | 
						|
#include <xcb/xcb_ewmh.h>
 | 
						|
#include <xcb/xcb_aux.h>
 | 
						|
#include <xcb/composite.h>
 | 
						|
 | 
						|
#include <X11/Xwindows.h>
 | 
						|
 | 
						|
/* Local headers */
 | 
						|
#include "X11/Xdefs.h" // for Bool type
 | 
						|
#include "winwindow.h"
 | 
						|
#include "winprefs.h"
 | 
						|
#include "window.h"
 | 
						|
#include "pixmapstr.h"
 | 
						|
#include "windowstr.h"
 | 
						|
#include "winglobals.h"
 | 
						|
#include "windisplay.h"
 | 
						|
#include "winmultiwindowicons.h"
 | 
						|
#include "winauth.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);
 | 
						|
 | 
						|
#ifndef ENABLE_DEBUG
 | 
						|
#define ENABLE_DEBUG 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
 | 
						|
 | 
						|
/*
 | 
						|
 * 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_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
 | 
						|
 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);
 | 
						|
 | 
						|
void
 | 
						|
 winUpdateWindowPosition(HWND hWnd, HWND * zstyle);
 | 
						|
 | 
						|
/*
 | 
						|
 * Local globals
 | 
						|
 */
 | 
						|
 | 
						|
static Bool g_shutdown = FALSE;
 | 
						|
 | 
						|
/*
 | 
						|
 * Translate msg id to text, for debug purposes
 | 
						|
 */
 | 
						|
 | 
						|
#if ENABLE_DEBUG
 | 
						|
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_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;
 | 
						|
    default:
 | 
						|
      return "Unknown Message";
 | 
						|
      break;
 | 
						|
    }
 | 
						|
}
 | 
						|
#endif
 | 
						|
 | 
						|
 | 
						|
/*
 | 
						|
 * 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 ENABLE_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 '@<clientmachine>' 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)
 | 
						|
        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)
 | 
						|
{
 | 
						|
    HWND hWnd;
 | 
						|
    HWND zstyle = HWND_NOTOPMOST;
 | 
						|
    UINT flags;
 | 
						|
 | 
						|
    hWnd = getHwnd(pWMInfo, iWindow);
 | 
						|
    if (!hWnd)
 | 
						|
        return;
 | 
						|
 | 
						|
    /* Determine the Window style, which determines borders and clipping region... */
 | 
						|
    winApplyHints(pWMInfo, iWindow, hWnd, &zstyle);
 | 
						|
    winUpdateWindowPosition(hWnd, &zstyle);
 | 
						|
 | 
						|
    /* Apply the updated window style, without changing its 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);
 | 
						|
}
 | 
						|
 | 
						|
/*
 | 
						|
 * Updates the state of a HWND
 | 
						|
 * (only minimization supported at the moment)
 | 
						|
 */
 | 
						|
 | 
						|
static void
 | 
						|
UpdateState(WMInfoPtr pWMInfo, xcb_window_t iWindow)
 | 
						|
{
 | 
						|
    HWND hWnd;
 | 
						|
 | 
						|
    winDebug("UpdateState: iWindow 0x%08x\n", (int)iWindow);
 | 
						|
 | 
						|
    hWnd = getHwnd(pWMInfo, iWindow);
 | 
						|
    if (!hWnd)
 | 
						|
        return;
 | 
						|
 | 
						|
    ShowWindow(hWnd, SW_MINIMIZE);
 | 
						|
}
 | 
						|
 | 
						|
#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;
 | 
						|
 | 
						|
    /* Initialize the Window Manager */
 | 
						|
    winInitMultiWindowWM(pWMInfo, pProcArg);
 | 
						|
 | 
						|
#if ENABLE_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 ENABLE_DEBUG
 | 
						|
        ErrorF("winMultiWindowWMProc - MSG: %s (%d) ID: %d\n",
 | 
						|
               MessageName(&(pNode->msg)), (int)pNode->msg.msg, (int)pNode->msg.dwID);
 | 
						|
#endif
 | 
						|
 | 
						|
        /* Branch on the message type */
 | 
						|
        switch (pNode->msg.msg) {
 | 
						|
#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_UNMANAGED:
 | 
						|
            /* 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));
 | 
						|
 | 
						|
            break;
 | 
						|
 | 
						|
        case WM_WM_MAP_MANAGED:
 | 
						|
            /* 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));
 | 
						|
 | 
						|
            UpdateName(pWMInfo, pNode->msg.iWindow);
 | 
						|
            UpdateIcon(pWMInfo, pNode->msg.iWindow);
 | 
						|
            UpdateStyle(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:
 | 
						|
            {
 | 
						|
            /* Don't do anything if this is an override-redirect window */
 | 
						|
            if (IsOverrideRedirect(pWMInfo->conn, pNode->msg.iWindow))
 | 
						|
              break;
 | 
						|
 | 
						|
            UpdateStyle(pWMInfo, pNode->msg.iWindow);
 | 
						|
            }
 | 
						|
            break;
 | 
						|
 | 
						|
        case WM_WM_CHANGE_STATE:
 | 
						|
            UpdateState(pWMInfo, pNode->msg.iWindow);
 | 
						|
            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;
 | 
						|
                    ErrorF("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);
 | 
						|
                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 ENABLE_DEBUG
 | 
						|
    ErrorF("-winMultiWindowWMProc ()\n");
 | 
						|
#endif
 | 
						|
    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;
 | 
						|
 | 
						|
    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");
 | 
						|
 | 
						|
    /*
 | 
						|
       iiimxcf had a bug until 2009-04-27, assuming that the
 | 
						|
       WM_STATE atom exists, causing clients to fail with
 | 
						|
       a BadAtom X error if it doesn't.
 | 
						|
 | 
						|
       Since this is on in the default Solaris 10 install,
 | 
						|
       workaround this by making sure it does exist...
 | 
						|
     */
 | 
						|
    intern_atom(pProcArg->conn, "WM_STATE");
 | 
						|
 | 
						|
    /*
 | 
						|
      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);
 | 
						|
            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;
 | 
						|
            ErrorF("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 = XCB_REPARENT_NOTIFY;
 | 
						|
                    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_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;
 | 
						|
 | 
						|
            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;
 | 
						|
 | 
						|
            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;
 | 
						|
 | 
						|
                winSendMessageToWM(pProcArg->pWMInfo, &msg);
 | 
						|
            }
 | 
						|
        }
 | 
						|
 | 
						|
        /* Free the event */
 | 
						|
        free(event);
 | 
						|
    }
 | 
						|
 | 
						|
    xcb_disconnect(pProcArg->conn);
 | 
						|
    pthread_exit(NULL);
 | 
						|
    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 = calloc(1, sizeof(WMProcArgRec));
 | 
						|
    WMInfoPtr pWMInfo = calloc(1, sizeof(WMInfoRec));
 | 
						|
    XMsgProcArgPtr pXMsgArg = calloc(1, 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;
 | 
						|
    }
 | 
						|
 | 
						|
    /* 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;
 | 
						|
 | 
						|
    /* Initialize 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 ENABLE_DEBUG || 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");
 | 
						|
 | 
						|
    /* 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,
 | 
						|
                };
 | 
						|
 | 
						|
            xcb_ewmh_set_supported(&pWMInfo->ewmh, pProcArg->dwScreen,
 | 
						|
                                   ARRAY_SIZE(supported), 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 its 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 ENABLE_DEBUG
 | 
						|
    ErrorF("winSendMessageToWM %s\n", MessageName(pMsg));
 | 
						|
#endif
 | 
						|
 | 
						|
    pNode = malloc(sizeof(WMMsgNodeRec));
 | 
						|
    if (pNode != NULL) {
 | 
						|
        memcpy(&pNode->msg, pMsg, sizeof(winWMMessageRec));
 | 
						|
        PushMessage(&((WMInfoPtr) pWMInfo)->wmMsgQueue, pNode);
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
/*
 | 
						|
 * 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;
 | 
						|
}
 | 
						|
 | 
						|
/* 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)
 | 
						|
/* These two are used on their own */
 | 
						|
#define HINT_MAX	(1L<<0)
 | 
						|
#define HINT_MIN	(1L<<1)
 | 
						|
 | 
						|
static void
 | 
						|
winApplyHints(WMInfoPtr pWMInfo, xcb_window_t iWindow, HWND hWnd, HWND * zstyle)
 | 
						|
{
 | 
						|
 | 
						|
    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 = 0, maxmin = 0;
 | 
						|
    unsigned long style, exStyle;
 | 
						|
 | 
						|
    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);
 | 
						|
 | 
						|
            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;
 | 
						|
            }
 | 
						|
 | 
						|
            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_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 & ~HINT_NOFRAME) | 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_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)) {
 | 
						|
            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 & ~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 (maxmin & HINT_MAX)
 | 
						|
        SendMessage(hWnd, WM_SYSCOMMAND, SC_MAXIMIZE, 0);
 | 
						|
    else if (maxmin & HINT_MIN)
 | 
						|
        SendMessage(hWnd, WM_SYSCOMMAND, SC_MINIMIZE, 0);
 | 
						|
 | 
						|
    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;
 | 
						|
 | 
						|
    /* 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 (!(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);
 | 
						|
}
 | 
						|
 | 
						|
void
 | 
						|
winUpdateWindowPosition(HWND hWnd, HWND * zstyle)
 | 
						|
{
 | 
						|
    int iX, iY, iWidth, iHeight;
 | 
						|
    int iDx, iDy;
 | 
						|
    RECT rcNew;
 | 
						|
    WindowPtr pWin = GetProp(hWnd, WIN_WINDOW_PROP);
 | 
						|
    DrawablePtr pDraw = NULL;
 | 
						|
 | 
						|
    if (!pWin)
 | 
						|
        return;
 | 
						|
    pDraw = &pWin->drawable;
 | 
						|
    if (!pDraw)
 | 
						|
        return;
 | 
						|
 | 
						|
    /* Get the X and Y location of the X window */
 | 
						|
    iX = pWin->drawable.x + GetSystemMetrics(SM_XVIRTUALSCREEN);
 | 
						|
    iY = pWin->drawable.y + GetSystemMetrics(SM_YVIRTUALSCREEN);
 | 
						|
 | 
						|
    /* Get the height and width of the X window */
 | 
						|
    iWidth = pWin->drawable.width;
 | 
						|
    iHeight = pWin->drawable.height;
 | 
						|
 | 
						|
    /* Setup a rectangle with the X window position and size */
 | 
						|
    SetRect(&rcNew, iX, iY, iX + iWidth, iY + iHeight);
 | 
						|
 | 
						|
    winDebug("winUpdateWindowPosition - drawable extent (%d, %d)-(%d, %d)\n",
 | 
						|
             rcNew.left, rcNew.top, rcNew.right, rcNew.bottom);
 | 
						|
 | 
						|
    AdjustWindowRectEx(&rcNew, GetWindowLongPtr(hWnd, GWL_STYLE), FALSE,
 | 
						|
                       GetWindowLongPtr(hWnd, GWL_EXSTYLE));
 | 
						|
 | 
						|
    /* Don't allow window decoration to disappear off to top-left as a result of this adjustment */
 | 
						|
    if (rcNew.left < GetSystemMetrics(SM_XVIRTUALSCREEN)) {
 | 
						|
        iDx = GetSystemMetrics(SM_XVIRTUALSCREEN) - rcNew.left;
 | 
						|
        rcNew.left += iDx;
 | 
						|
        rcNew.right += iDx;
 | 
						|
    }
 | 
						|
 | 
						|
    if (rcNew.top < GetSystemMetrics(SM_YVIRTUALSCREEN)) {
 | 
						|
        iDy = GetSystemMetrics(SM_YVIRTUALSCREEN) - rcNew.top;
 | 
						|
        rcNew.top += iDy;
 | 
						|
        rcNew.bottom += iDy;
 | 
						|
    }
 | 
						|
 | 
						|
    winDebug("winUpdateWindowPosition - Window extent (%d, %d)-(%d, %d)\n",
 | 
						|
             rcNew.left, rcNew.top, rcNew.right, rcNew.bottom);
 | 
						|
 | 
						|
    /* Position the Windows window */
 | 
						|
    SetWindowPos(hWnd, *zstyle, rcNew.left, rcNew.top,
 | 
						|
                 rcNew.right - rcNew.left, rcNew.bottom - rcNew.top, 0);
 | 
						|
 | 
						|
}
 |