xserver-multidpi/hw/xwin/winmultiwindowwm.c

1669 lines
40 KiB
C
Raw Normal View History

2003-11-14 17:48:57 +01:00
/*
*Copyright (C) 1994-2000 The XFree86 Project, Inc. All Rights Reserved.
*Copyright (C) Colin Harrison 2005-2009
2003-11-14 17:48:57 +01:00
*
*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
2003-11-14 17:48:57 +01:00
*/
/* X headers */
#ifdef HAVE_XWIN_CONFIG_H
#include <xwin-config.h>
#endif
2003-11-14 17:48:57 +01:00
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#ifdef __CYGWIN__
2003-11-14 17:48:57 +01:00
#include <sys/select.h>
#endif
2003-11-14 17:48:57 +01:00
#include <fcntl.h>
#include <setjmp.h>
2005-05-25 14:14:29 +02:00
#define HANDLE void *
2003-11-14 17:48:57 +01:00
#include <pthread.h>
2005-05-25 14:14:29 +02:00
#undef HANDLE
2003-11-14 17:48:57 +01:00
#include <X11/X.h>
#include <X11/Xatom.h>
#include <X11/Xlib.h>
#include <X11/Xlocale.h>
#include <X11/Xproto.h>
#include <X11/Xutil.h>
#include <X11/cursorfont.h>
#include <X11/Xwindows.h>
2003-11-14 17:48:57 +01:00
/* Local headers */
#include "objbase.h"
2004-06-21 15:19:32 +02:00
#include "ddraw.h"
2003-11-14 17:48:57 +01:00
#include "winwindow.h"
#include "winprefs.h"
#include "window.h"
#include "pixmapstr.h"
#include "windowstr.h"
#ifdef XWIN_MULTIWINDOWEXTWM
#include <X11/extensions/windowswmstr.h>
#else
/* We need the native HWND atom for intWM, so for consistency use the
same name as extWM would if we were building with enabled... */
#define WINDOWSWM_NATIVE_HWND "_WINDOWSWM_NATIVE_HWND"
#endif
2003-11-14 17:48:57 +01:00
2004-06-21 15:19:32 +02:00
extern void winDebug(const char *format, ...);
extern void winReshapeMultiWindow(WindowPtr pWin);
extern void winUpdateRgnMultiWindow(WindowPtr pWin);
2004-06-21 15:19:32 +02:00
2004-11-04 12:52:22 +01:00
#ifndef CYGDEBUG
2004-06-21 15:19:32 +02:00
#define CYGDEBUG NO
#endif
2003-11-14 17:48:57 +01:00
/*
* Constant defines
*/
#define WIN_CONNECT_RETRIES 5
#define WIN_CONNECT_DELAY 5
#ifdef HAS_DEVWINDOWS
# define WIN_MSG_QUEUE_FNAME "/dev/windows"
#endif
2003-11-14 17:48:57 +01:00
#define WIN_JMP_OKAY 0
#define WIN_JMP_ERROR_IO 2
/*
* 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;
2004-06-21 15:19:32 +02:00
int nQueueSize;
2003-11-14 17:48:57 +01:00
} WMMsgQueueRec, *WMMsgQueuePtr;
typedef struct _WMInfo {
Display *pDisplay;
WMMsgQueueRec wmMsgQueue;
Atom atmWmProtos;
Atom atmWmDelete;
Atom atmPrivMap;
Bool fAllowOtherWM;
2003-11-14 17:48:57 +01:00
} WMInfoRec, *WMInfoPtr;
typedef struct _WMProcArgRec {
DWORD dwScreen;
WMInfoPtr pWMInfo;
pthread_mutex_t *ppmServerStarted;
} WMProcArgRec, *WMProcArgPtr;
typedef struct _XMsgProcArgRec {
2004-06-21 15:19:32 +02:00
Display *pDisplay;
DWORD dwScreen;
WMInfoPtr pWMInfo;
pthread_mutex_t *ppmServerStarted;
2004-11-04 12:52:22 +01:00
HWND hwndScreen;
} XMsgProcArgRec, *XMsgProcArgPtr;
2003-11-14 17:48:57 +01:00
/*
* References to external symbols
*/
extern char *display;
extern void ErrorF (const char* /*f*/, ...);
/*
* Prototypes for local functions
*/
static void
PushMessage (WMMsgQueuePtr pQueue, WMMsgNodePtr pNode);
static WMMsgNodePtr
PopMessage (WMMsgQueuePtr pQueue, WMInfoPtr pWMInfo);
2003-11-14 17:48:57 +01:00
static Bool
InitQueue (WMMsgQueuePtr pQueue);
static void
GetWindowName (Display * pDpy, Window iWin, char **ppName);
static int
SendXMessage (Display *pDisplay, Window iWin, Atom atmType, long nData);
static void
UpdateName (WMInfoPtr pWMInfo, Window iWindow);
2003-11-14 17:48:57 +01:00
static void*
winMultiWindowWMProc (void* pArg);
static int
2004-06-21 15:19:32 +02:00
winMultiWindowWMErrorHandler (Display *pDisplay, XErrorEvent *pErr);
static int
winMultiWindowWMIOErrorHandler (Display *pDisplay);
2003-11-14 17:48:57 +01:00
static void *
winMultiWindowXMsgProc (void *pArg);
2004-06-21 15:19:32 +02:00
static int
winMultiWindowXMsgProcErrorHandler (Display *pDisplay, XErrorEvent *pErr);
static int
winMultiWindowXMsgProcIOErrorHandler (Display *pDisplay);
static int
winRedirectErrorHandler (Display *pDisplay, XErrorEvent *pErr);
2003-11-14 17:48:57 +01:00
static void
winInitMultiWindowWM (WMInfoPtr pWMInfo, WMProcArgPtr pProcArg);
2004-06-21 15:19:32 +02:00
#if 0
static void
PreserveWin32Stack(WMInfoPtr pWMInfo, Window iWindow, UINT direction);
#endif
2003-11-14 17:48:57 +01:00
2004-11-04 12:52:22 +01:00
static Bool
CheckAnotherWindowManager (Display *pDisplay, DWORD dwScreen);
static void
winApplyHints (Display *pDisplay, Window iWindow, HWND hWnd, HWND *zstyle);
void
winUpdateWindowPosition (HWND hWnd, Bool reshape, HWND *zstyle);
2003-11-14 17:48:57 +01:00
/*
* Local globals
*/
2004-06-21 15:19:32 +02:00
static jmp_buf g_jmpWMEntry;
static jmp_buf g_jmpXMsgProcEntry;
2004-11-04 12:52:22 +01:00
static Bool g_shutdown = FALSE;
2004-06-21 15:19:32 +02:00
static Bool redirectError = FALSE;
2004-11-04 12:52:22 +01:00
static Bool g_fAnotherWMRunnig = FALSE;
2003-11-14 17:48:57 +01:00
/*
* 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;
}
#if 0
switch (pNode->msg.msg)
{
case WM_WM_MOVE:
ErrorF ("\tWM_WM_MOVE\n");
break;
2004-11-04 12:52:22 +01:00
case WM_WM_SIZE:
ErrorF ("\tWM_WM_SIZE\n");
break;
2003-11-14 17:48:57 +01:00
case WM_WM_RAISE:
ErrorF ("\tWM_WM_RAISE\n");
break;
case WM_WM_LOWER:
2004-11-04 12:52:22 +01:00
ErrorF ("\tWM_WM_LOWER\n");
2003-11-14 17:48:57 +01:00
break;
case WM_WM_MAP:
ErrorF ("\tWM_WM_MAP\n");
break;
case WM_WM_MAP2:
ErrorF ("\tWM_WM_MAP2\n");
break;
case WM_WM_MAP3:
ErrorF ("\tWM_WM_MAP3\n");
break;
2003-11-14 17:48:57 +01:00
case WM_WM_UNMAP:
ErrorF ("\tWM_WM_UNMAP\n");
break;
case WM_WM_KILL:
ErrorF ("\tWM_WM_KILL\n");
break;
2004-11-04 12:52:22 +01:00
case WM_WM_ACTIVATE:
ErrorF ("\tWM_WM_ACTIVATE\n");
break;
2003-11-14 17:48:57 +01:00
default:
2004-11-04 12:52:22 +01:00
ErrorF ("\tUnknown Message.\n");
2003-11-14 17:48:57 +01:00
break;
}
#endif
/* Increase the count of elements in the queue by one */
2004-06-21 15:19:32 +02:00
++(pQueue->nQueueSize);
2003-11-14 17:48:57 +01:00
/* Release the queue mutex */
pthread_mutex_unlock (&pQueue->pmMutex);
/* Signal that the queue is not empty */
pthread_cond_signal (&pQueue->pcNotEmpty);
}
#if CYGMULTIWINDOW_DEBUG
2003-11-14 17:48:57 +01:00
/*
* QueueSize - Return the size of the queue
*/
static int
QueueSize (WMMsgQueuePtr pQueue)
{
WMMsgNodePtr pNode;
int nSize = 0;
/* Loop through all elements in the queue */
for (pNode = pQueue->pHead; pNode != NULL; pNode = pNode->pNext)
++nSize;
return nSize;
}
#endif
/*
* PopMessage - Pop a message from the queue
2003-11-14 17:48:57 +01:00
*/
static WMMsgNodePtr
PopMessage (WMMsgQueuePtr pQueue, WMInfoPtr pWMInfo)
2003-11-14 17:48:57 +01:00
{
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;
}
/* Drop the number of elements in the queue by one */
2004-06-21 15:19:32 +02:00
--(pQueue->nQueueSize);
2003-11-14 17:48:57 +01:00
#if CYGMULTIWINDOW_DEBUG
2004-06-21 15:19:32 +02:00
ErrorF ("Queue Size %d %d\n", pQueue->nQueueSize, QueueSize(pQueue));
2003-11-14 17:48:57 +01:00
#endif
/* Release the queue mutex */
pthread_mutex_unlock (&pQueue->pmMutex);
return pNode;
}
#if 0
/*
* HaveMessage -
*/
static Bool
HaveMessage (WMMsgQueuePtr pQueue, UINT msg, Window 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;
/* There are no elements initially */
2004-06-21 15:19:32 +02:00
pQueue->nQueueSize = 0;
2003-11-14 17:48:57 +01:00
#if CYGMULTIWINDOW_DEBUG
2004-06-21 15:19:32 +02:00
ErrorF ("InitQueue - Queue Size %d %d\n", pQueue->nQueueSize,
QueueSize(pQueue));
2003-11-14 17:48:57 +01:00
#endif
ErrorF ("InitQueue - Calling pthread_mutex_init\n");
/* Create synchronization objects */
pthread_mutex_init (&pQueue->pmMutex, NULL);
ErrorF ("InitQueue - pthread_mutex_init returned\n");
ErrorF ("InitQueue - Calling pthread_cond_init\n");
pthread_cond_init (&pQueue->pcNotEmpty, NULL);
ErrorF ("InitQueue - pthread_cond_init returned\n");
return TRUE;
}
/*
* GetWindowName - Retrieve the title of an X Window
2003-11-14 17:48:57 +01:00
*/
static void
GetWindowName (Display *pDisplay, Window iWin, char **ppName)
{
int nResult, nNum;
char **ppList;
XTextProperty xtpName;
#if CYGMULTIWINDOW_DEBUG
ErrorF ("GetWindowName\n");
#endif
/* Intialize ppName to NULL */
*ppName = NULL;
/* Try to get --- */
nResult = XGetWMName (pDisplay, iWin, &xtpName);
if (!nResult || !xtpName.value || !xtpName.nitems)
{
#if CYGMULTIWINDOW_DEBUG
2003-11-14 17:48:57 +01:00
ErrorF ("GetWindowName - XGetWMName failed. No name.\n");
#endif
2003-11-14 17:48:57 +01:00
return;
}
/* */
if (xtpName.encoding == XA_STRING)
{
/* */
if (xtpName.value)
{
int size = xtpName.nitems * (xtpName.format >> 3);
*ppName = malloc(size + 1);
strncpy(*ppName, xtpName.value, size);
(*ppName)[size] = 0;
2003-11-14 17:48:57 +01:00
XFree (xtpName.value);
}
#if CYGMULTIWINDOW_DEBUG
ErrorF ("GetWindowName - XA_STRING %s\n", *ppName);
2003-11-14 17:48:57 +01:00
#endif
}
else
{
if (XmbTextPropertyToTextList (pDisplay, &xtpName, &ppList, &nNum) >= Success && nNum > 0 && *ppList)
2003-11-14 17:48:57 +01:00
{
*ppName = strdup (*ppList);
XFreeStringList (ppList);
}
2004-06-21 15:19:32 +02:00
XFree (xtpName.value);
2003-11-14 17:48:57 +01:00
#if CYGMULTIWINDOW_DEBUG
ErrorF ("GetWindowName - %s %s\n",
XGetAtomName (pDisplay, xtpName.encoding), *ppName);
2003-11-14 17:48:57 +01:00
#endif
}
#if CYGMULTIWINDOW_DEBUG
ErrorF ("GetWindowName - Returning\n");
2003-11-14 17:48:57 +01:00
#endif
}
/*
* Send a message to the X server from the WM thread
*/
static int
SendXMessage (Display *pDisplay, Window iWin, Atom atmType, long nData)
{
XEvent e;
/* Prepare the X event structure */
e.type = ClientMessage;
e.xclient.window = iWin;
e.xclient.message_type = atmType;
e.xclient.format = 32;
e.xclient.data.l[0] = nData;
e.xclient.data.l[1] = CurrentTime;
/* Send the event to X */
return XSendEvent (pDisplay, iWin, False, NoEventMask, &e);
}
/*
* Updates the name of a HWND according to its X WM_NAME property
*/
static void
UpdateName (WMInfoPtr pWMInfo, Window iWindow)
{
char *pszName;
Atom atmType;
int fmtRet;
unsigned long items, remain;
HWND *retHwnd, hWnd;
XWindowAttributes attr;
hWnd = 0;
/* See if we can get the cached HWND for this window... */
if (XGetWindowProperty (pWMInfo->pDisplay,
iWindow,
pWMInfo->atmPrivMap,
0,
1,
False,
2004-11-04 12:52:22 +01:00
XA_INTEGER,//pWMInfo->atmPrivMap,
&atmType,
&fmtRet,
&items,
&remain,
(unsigned char **) &retHwnd) == Success)
{
if (retHwnd)
{
hWnd = *retHwnd;
XFree (retHwnd);
}
}
/* Some sanity checks */
if (!hWnd) return;
if (!IsWindow (hWnd)) return;
2004-11-04 12:52:22 +01:00
/* Set the Windows window name */
GetWindowName (pWMInfo->pDisplay, iWindow, &pszName);
if (pszName)
{
/* Get the window attributes */
XGetWindowAttributes (pWMInfo->pDisplay,
iWindow,
&attr);
if (!attr.override_redirect)
{
SetWindowText (hWnd, pszName);
winUpdateIcon (iWindow);
}
free (pszName);
}
}
2004-06-21 15:19:32 +02:00
#if 0
/*
* Fix up any differences between the X11 and Win32 window stacks
* starting at the window passed in
*/
static void
PreserveWin32Stack(WMInfoPtr pWMInfo, Window iWindow, UINT direction)
{
Atom atmType;
int fmtRet;
unsigned long items, remain;
HWND hWnd, *retHwnd;
DWORD myWinProcID, winProcID;
Window xWindow;
WINDOWPLACEMENT wndPlace;
hWnd = NULL;
/* See if we can get the cached HWND for this window... */
if (XGetWindowProperty (pWMInfo->pDisplay,
iWindow,
pWMInfo->atmPrivMap,
0,
1,
False,
2004-11-04 12:52:22 +01:00
XA_INTEGER,//pWMInfo->atmPrivMap,
2004-06-21 15:19:32 +02:00
&atmType,
&fmtRet,
&items,
&remain,
(unsigned char **) &retHwnd) == Success)
{
if (retHwnd)
{
hWnd = *retHwnd;
XFree (retHwnd);
}
}
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 */
2003-11-14 17:48:57 +01:00
/*
* winMultiWindowWMProc
*/
static void *
winMultiWindowWMProc (void *pArg)
{
WMProcArgPtr pProcArg = (WMProcArgPtr)pArg;
WMInfoPtr pWMInfo = pProcArg->pWMInfo;
/* Initialize the Window Manager */
winInitMultiWindowWM (pWMInfo, pProcArg);
#if CYGMULTIWINDOW_DEBUG
ErrorF ("winMultiWindowWMProc ()\n");
#endif
/* Loop until we explicity break out */
for (;;)
{
WMMsgNodePtr pNode;
2004-11-04 12:52:22 +01:00
if(g_fAnotherWMRunnig)/* Another Window manager exists. */
{
Sleep (1000);
continue;
}
2003-11-14 17:48:57 +01:00
/* Pop a message off of our queue */
pNode = PopMessage (&pWMInfo->wmMsgQueue, pWMInfo);
2003-11-14 17:48:57 +01:00
if (pNode == NULL)
{
/* Bail if PopMessage returns without a message */
/* NOTE: Remember that PopMessage is a blocking function. */
2004-06-21 15:19:32 +02:00
ErrorF ("winMultiWindowWMProc - Queue is Empty? Exiting.\n");
2003-11-14 17:48:57 +01:00
pthread_exit (NULL);
}
#if CYGMULTIWINDOW_DEBUG
ErrorF ("winMultiWindowWMProc - %d ms MSG: %d ID: %d\n",
GetTickCount (), (int)pNode->msg.msg, (int)pNode->msg.dwID);
#endif
2004-11-04 12:52:22 +01:00
2003-11-14 17:48:57 +01:00
/* Branch on the message type */
switch (pNode->msg.msg)
{
#if 0
case WM_WM_MOVE:
ErrorF ("\tWM_WM_MOVE\n");
break;
case WM_WM_SIZE:
ErrorF ("\tWM_WM_SIZE\n");
break;
#endif
case WM_WM_RAISE:
#if CYGMULTIWINDOW_DEBUG
ErrorF ("\tWM_WM_RAISE\n");
#endif
/* Raise the window */
XRaiseWindow (pWMInfo->pDisplay, pNode->msg.iWindow);
2004-06-21 15:19:32 +02:00
#if 0
PreserveWin32Stack (pWMInfo, pNode->msg.iWindow, GW_HWNDPREV);
#endif
2003-11-14 17:48:57 +01:00
break;
case WM_WM_LOWER:
#if CYGMULTIWINDOW_DEBUG
ErrorF ("\tWM_WM_LOWER\n");
#endif
/* Lower the window */
XLowerWindow (pWMInfo->pDisplay, pNode->msg.iWindow);
break;
case WM_WM_MAP:
#if CYGMULTIWINDOW_DEBUG
ErrorF ("\tWM_WM_MAP\n");
#endif
/* Put a note as to the HWND associated with this Window */
XChangeProperty (pWMInfo->pDisplay,
pNode->msg.iWindow,
pWMInfo->atmPrivMap,
2004-11-04 12:52:22 +01:00
XA_INTEGER,//pWMInfo->atmPrivMap,
32,
PropModeReplace,
(unsigned char *) &(pNode->msg.hwndWindow),
1);
UpdateName (pWMInfo, pNode->msg.iWindow);
winUpdateIcon (pNode->msg.iWindow);
break;
case WM_WM_MAP2:
#if CYGMULTIWINDOW_DEBUG
ErrorF ("\tWM_WM_MAP2\n");
2004-06-21 15:19:32 +02:00
#endif
XChangeProperty (pWMInfo->pDisplay,
pNode->msg.iWindow,
pWMInfo->atmPrivMap,
XA_INTEGER,//pWMInfo->atmPrivMap,
32,
PropModeReplace,
(unsigned char *) &(pNode->msg.hwndWindow),
1);
break;
case WM_WM_MAP3:
#if CYGMULTIWINDOW_DEBUG
ErrorF ("\tWM_WM_MAP3\n");
#endif
/* Put a note as to the HWND associated with this Window */
XChangeProperty (pWMInfo->pDisplay,
pNode->msg.iWindow,
pWMInfo->atmPrivMap,
XA_INTEGER,//pWMInfo->atmPrivMap,
32,
PropModeReplace,
(unsigned char *) &(pNode->msg.hwndWindow),
1);
UpdateName (pWMInfo, pNode->msg.iWindow);
winUpdateIcon (pNode->msg.iWindow);
{
HWND zstyle = HWND_NOTOPMOST;
winApplyHints (pWMInfo->pDisplay, pNode->msg.iWindow, pNode->msg.hwndWindow, &zstyle);
winUpdateWindowPosition (pNode->msg.hwndWindow, TRUE, &zstyle);
}
2003-11-14 17:48:57 +01:00
break;
case WM_WM_UNMAP:
#if CYGMULTIWINDOW_DEBUG
ErrorF ("\tWM_WM_UNMAP\n");
#endif
/* Unmap the window */
XUnmapWindow (pWMInfo->pDisplay, pNode->msg.iWindow);
2003-11-14 17:48:57 +01:00
break;
case WM_WM_KILL:
#if CYGMULTIWINDOW_DEBUG
ErrorF ("\tWM_WM_KILL\n");
#endif
{
int i, n, found = 0;
Atom *protocols;
/* --- */
if (XGetWMProtocols (pWMInfo->pDisplay,
pNode->msg.iWindow,
&protocols,
&n))
{
for (i = 0; i < n; ++i)
if (protocols[i] == pWMInfo->atmWmDelete)
++found;
XFree (protocols);
}
/* --- */
if (found)
SendXMessage (pWMInfo->pDisplay,
pNode->msg.iWindow,
pWMInfo->atmWmProtos,
pWMInfo->atmWmDelete);
else
XKillClient (pWMInfo->pDisplay,
pNode->msg.iWindow);
}
break;
case WM_WM_ACTIVATE:
#if CYGMULTIWINDOW_DEBUG
ErrorF ("\tWM_WM_ACTIVATE\n");
#endif
/* Set the input focus */
XSetInputFocus (pWMInfo->pDisplay,
pNode->msg.iWindow,
RevertToPointerRoot,
CurrentTime);
break;
case WM_WM_NAME_EVENT:
UpdateName (pWMInfo, pNode->msg.iWindow);
break;
case WM_WM_HINTS_EVENT:
winUpdateIcon (pNode->msg.iWindow);
break;
case WM_WM_CHANGE_STATE:
/* Minimize the window in Windows */
winMinimizeWindow (pNode->msg.iWindow);
2003-11-14 17:48:57 +01:00
break;
default:
2004-06-21 15:19:32 +02:00
ErrorF ("winMultiWindowWMProc - Unknown Message. Exiting.\n");
2003-11-14 17:48:57 +01:00
pthread_exit (NULL);
break;
}
/* Free the retrieved message */
free (pNode);
/* Flush any pending events on our display */
XFlush (pWMInfo->pDisplay);
}
/* 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
return NULL;
2003-11-14 17:48:57 +01:00
}
/*
2004-06-21 15:19:32 +02:00
* X message procedure
2003-11-14 17:48:57 +01:00
*/
static void *
winMultiWindowXMsgProc (void *pArg)
2003-11-14 17:48:57 +01:00
{
winWMMessageRec msg;
XMsgProcArgPtr pProcArg = (XMsgProcArgPtr) pArg;
char pszDisplay[512];
int iRetries;
XEvent event;
Atom atmWmName;
Atom atmWmHints;
Atom atmWmChange;
int iReturn;
XIconSize *xis;
2003-11-14 17:48:57 +01:00
ErrorF ("winMultiWindowXMsgProc - Hello\n");
/* Check that argument pointer is not invalid */
if (pProcArg == NULL)
2003-11-14 17:48:57 +01:00
{
2004-06-21 15:19:32 +02:00
ErrorF ("winMultiWindowXMsgProc - pProcArg is NULL. Exiting.\n");
2003-11-14 17:48:57 +01:00
pthread_exit (NULL);
}
ErrorF ("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)
{
2004-06-21 15:19:32 +02:00
ErrorF ("winMultiWindowXMsgProc - pthread_mutex_lock () failed: %d. "
"Exiting.\n",
iReturn);
pthread_exit (NULL);
}
ErrorF ("winMultiWindowXMsgProc - pthread_mutex_lock () returned.\n");
2004-06-21 15:19:32 +02:00
/* Allow multiple threads to access Xlib */
if (XInitThreads () == 0)
{
2004-06-21 15:19:32 +02:00
ErrorF ("winMultiWindowXMsgProc - XInitThreads () failed. Exiting.\n");
pthread_exit (NULL);
}
2004-06-21 15:19:32 +02:00
/* See if X supports the current locale */
if (XSupportsLocale () == False)
{
ErrorF ("winMultiWindowXMsgProc - Locale not supported by X. "
"Exiting.\n");
pthread_exit (NULL);
}
/* Release the server started mutex */
pthread_mutex_unlock (pProcArg->ppmServerStarted);
ErrorF ("winMultiWindowXMsgProc - pthread_mutex_unlock () returned.\n");
2004-06-21 15:19:32 +02:00
/* Set jump point for IO Error exits */
iReturn = setjmp (g_jmpXMsgProcEntry);
/* Check if we should continue operations */
if (iReturn != WIN_JMP_ERROR_IO
&& iReturn != WIN_JMP_OKAY)
{
/* setjmp returned an unknown value, exit */
ErrorF ("winInitMultiWindowXMsgProc - setjmp returned: %d. Exiting.\n",
iReturn);
pthread_exit (NULL);
}
else if (iReturn == WIN_JMP_ERROR_IO)
{
ErrorF ("winInitMultiWindowXMsgProc - Caught IO Error. Exiting.\n");
pthread_exit (NULL);
}
/* Install our error handler */
XSetErrorHandler (winMultiWindowXMsgProcErrorHandler);
XSetIOErrorHandler (winMultiWindowXMsgProcIOErrorHandler);
/* Setup the display connection string x */
snprintf (pszDisplay,
512, "127.0.0.1:%s.%d", display, (int)pProcArg->dwScreen);
/* Print the display connection string */
ErrorF ("winMultiWindowXMsgProc - DISPLAY=%s\n", pszDisplay);
2003-11-14 17:48:57 +01:00
2004-06-21 15:19:32 +02:00
/* Initialize retry count */
iRetries = 0;
2003-11-14 17:48:57 +01:00
/* Open the X display */
do
2003-11-14 17:48:57 +01:00
{
/* Try to open the display */
pProcArg->pDisplay = XOpenDisplay (pszDisplay);
if (pProcArg->pDisplay == NULL)
{
ErrorF ("winMultiWindowXMsgProc - Could not open display, try: %d, "
"sleeping: %d\n\f",
iRetries + 1, WIN_CONNECT_DELAY);
++iRetries;
sleep (WIN_CONNECT_DELAY);
continue;
}
else
break;
}
while (pProcArg->pDisplay == NULL && iRetries < WIN_CONNECT_RETRIES);
/* Make sure that the display opened */
if (pProcArg->pDisplay == NULL)
{
2004-06-21 15:19:32 +02:00
ErrorF ("winMultiWindowXMsgProc - Failed opening the display. "
"Exiting.\n");
2003-11-14 17:48:57 +01:00
pthread_exit (NULL);
}
ErrorF ("winMultiWindowXMsgProc - XOpenDisplay () returned and "
"successfully opened the display.\n");
2004-06-21 15:19:32 +02:00
/* Check if another window manager is already running */
if (pProcArg->pWMInfo->fAllowOtherWM)
{
g_fAnotherWMRunnig = CheckAnotherWindowManager (pProcArg->pDisplay, pProcArg->dwScreen);
} else {
redirectError = FALSE;
XSetErrorHandler (winRedirectErrorHandler);
XSelectInput(pProcArg->pDisplay,
RootWindow (pProcArg->pDisplay, pProcArg->dwScreen),
SubstructureNotifyMask | ButtonPressMask);
XSync (pProcArg->pDisplay, 0);
XSetErrorHandler (winMultiWindowXMsgProcErrorHandler);
if (redirectError)
{
ErrorF ("winMultiWindowXMsgProc - "
"another window manager is running. Exiting.\n");
pthread_exit (NULL);
}
g_fAnotherWMRunnig = FALSE;
}
/* Set up the supported icon sizes */
xis = XAllocIconSize ();
if (xis)
{
xis->min_width = xis->min_height = 16;
xis->max_width = xis->max_height = 48;
xis->width_inc = xis->height_inc = 16;
XSetIconSizes (pProcArg->pDisplay,
RootWindow (pProcArg->pDisplay, pProcArg->dwScreen),
xis,
1);
XFree (xis);
}
atmWmName = XInternAtom (pProcArg->pDisplay,
"WM_NAME",
False);
atmWmHints = XInternAtom (pProcArg->pDisplay,
"WM_HINTS",
False);
atmWmChange = XInternAtom (pProcArg->pDisplay,
"WM_CHANGE_STATE",
False);
/* Loop until we explicitly break out */
while (1)
{
2005-07-05 20:25:44 +02:00
if (g_shutdown)
break;
if (pProcArg->pWMInfo->fAllowOtherWM && !XPending (pProcArg->pDisplay))
2004-11-04 12:52:22 +01:00
{
if (CheckAnotherWindowManager (pProcArg->pDisplay, pProcArg->dwScreen))
{
if (!g_fAnotherWMRunnig)
{
g_fAnotherWMRunnig = TRUE;
SendMessage(*(HWND*)pProcArg->hwndScreen, WM_UNMANAGE, 0, 0);
}
}
else
{
if (g_fAnotherWMRunnig)
{
g_fAnotherWMRunnig = FALSE;
SendMessage(*(HWND*)pProcArg->hwndScreen, WM_MANAGE, 0, 0);
}
}
Sleep (500);
continue;
}
/* Fetch next event */
XNextEvent (pProcArg->pDisplay, &event);
/* Branch on event type */
if (event.type == CreateNotify)
{
2004-11-04 12:52:22 +01:00
XWindowAttributes attr;
XSelectInput (pProcArg->pDisplay,
event.xcreatewindow.window,
PropertyChangeMask);
2004-11-04 12:52:22 +01:00
/* Get the window attributes */
XGetWindowAttributes (pProcArg->pDisplay,
event.xcreatewindow.window,
&attr);
if (!attr.override_redirect)
XSetWindowBorderWidth(pProcArg->pDisplay,
event.xcreatewindow.window,
0);
}
else if (event.type == PropertyNotify
&& event.xproperty.atom == atmWmName)
{
memset (&msg, 0, sizeof (msg));
msg.msg = WM_WM_NAME_EVENT;
msg.iWindow = event.xproperty.window;
/* Other fields ignored */
winSendMessageToWM (pProcArg->pWMInfo, &msg);
}
else if (event.type == PropertyNotify
&& event.xproperty.atom == atmWmHints)
{
memset (&msg, 0, sizeof (msg));
msg.msg = WM_WM_HINTS_EVENT;
msg.iWindow = event.xproperty.window;
/* Other fields ignored */
winSendMessageToWM (pProcArg->pWMInfo, &msg);
}
else if (event.type == ClientMessage
&& event.xclient.message_type == atmWmChange
&& event.xclient.data.l[0] == IconicState)
{
ErrorF ("winMultiWindowXMsgProc - WM_CHANGE_STATE - IconicState\n");
memset (&msg, 0, sizeof (msg));
msg.msg = WM_WM_CHANGE_STATE;
msg.iWindow = event.xclient.window;
winSendMessageToWM (pProcArg->pWMInfo, &msg);
}
}
2005-07-05 20:25:44 +02:00
XCloseDisplay (pProcArg->pDisplay);
pthread_exit (NULL);
return NULL;
2003-11-14 17:48:57 +01:00
}
/*
* 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,
2003-11-14 17:48:57 +01:00
pthread_mutex_t *ppmServerStarted,
2004-11-04 12:52:22 +01:00
int dwScreen,
HWND hwndScreen,
BOOL allowOtherWM)
2003-11-14 17:48:57 +01:00
{
WMProcArgPtr pArg = (WMProcArgPtr) malloc (sizeof(WMProcArgRec));
WMInfoPtr pWMInfo = (WMInfoPtr) malloc (sizeof(WMInfoRec));
XMsgProcArgPtr pXMsgArg = (XMsgProcArgPtr) malloc (sizeof(XMsgProcArgRec));
2003-11-14 17:48:57 +01:00
/* Bail if the input parameters are bad */
if (pArg == NULL || pWMInfo == NULL)
{
2004-06-21 15:19:32 +02:00
ErrorF ("winInitWM - malloc failed.\n");
2003-11-14 17:48:57 +01:00
return FALSE;
}
2004-06-21 15:19:32 +02:00
/* Zero the allocated memory */
ZeroMemory (pArg, sizeof (WMProcArgRec));
ZeroMemory (pWMInfo, sizeof (WMInfoRec));
ZeroMemory (pXMsgArg, sizeof (XMsgProcArgRec));
2003-11-14 17:48:57 +01:00
/* Set a return pointer to the Window Manager info structure */
*ppWMInfo = pWMInfo;
pWMInfo->fAllowOtherWM = allowOtherWM;
2003-11-14 17:48:57 +01:00
/* 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;
2004-11-04 12:52:22 +01:00
pXMsgArg->hwndScreen = hwndScreen;
if (pthread_create (ptXMsgProc, NULL, winMultiWindowXMsgProc, pXMsgArg))
{
/* Bail if thread creation failed */
ErrorF ("winInitWM - pthread_create failed on XMSG.\n");
2003-11-14 17:48:57 +01:00
return FALSE;
}
#if CYGDEBUG || YES
2004-06-21 15:19:32 +02:00
winDebug ("winInitWM - Returning.\n");
2003-11-14 17:48:57 +01:00
#endif
return TRUE;
}
/*
2004-06-21 15:19:32 +02:00
* Window manager thread - setup
2003-11-14 17:48:57 +01:00
*/
static void
winInitMultiWindowWM (WMInfoPtr pWMInfo, WMProcArgPtr pProcArg)
{
int iRetries = 0;
char pszDisplay[512];
int iReturn;
ErrorF ("winInitMultiWindowWM - Hello\n");
/* Check that argument pointer is not invalid */
if (pProcArg == NULL)
{
2004-06-21 15:19:32 +02:00
ErrorF ("winInitMultiWindowWM - pProcArg is NULL. Exiting.\n");
2003-11-14 17:48:57 +01:00
pthread_exit (NULL);
}
ErrorF ("winInitMultiWindowWM - Calling pthread_mutex_lock ()\n");
/* Grab our garbage mutex to satisfy pthread_cond_wait */
iReturn = pthread_mutex_lock (pProcArg->ppmServerStarted);
if (iReturn != 0)
{
2004-06-21 15:19:32 +02:00
ErrorF ("winInitMultiWindowWM - pthread_mutex_lock () failed: %d. "
"Exiting.\n",
2003-11-14 17:48:57 +01:00
iReturn);
pthread_exit (NULL);
}
ErrorF ("winInitMultiWindowWM - pthread_mutex_lock () returned.\n");
2004-06-21 15:19:32 +02:00
/* Allow multiple threads to access Xlib */
if (XInitThreads () == 0)
2003-11-14 17:48:57 +01:00
{
2004-06-21 15:19:32 +02:00
ErrorF ("winInitMultiWindowWM - XInitThreads () failed. Exiting.\n");
pthread_exit (NULL);
2003-11-14 17:48:57 +01:00
}
2004-06-21 15:19:32 +02:00
/* See if X supports the current locale */
if (XSupportsLocale () == False)
{
2004-06-21 15:19:32 +02:00
ErrorF ("winInitMultiWindowWM - Locale not supported by X. Exiting.\n");
pthread_exit (NULL);
}
/* Release the server started mutex */
2003-11-14 17:48:57 +01:00
pthread_mutex_unlock (pProcArg->ppmServerStarted);
ErrorF ("winInitMultiWindowWM - pthread_mutex_unlock () returned.\n");
/* Set jump point for IO Error exits */
2004-06-21 15:19:32 +02:00
iReturn = setjmp (g_jmpWMEntry);
2003-11-14 17:48:57 +01:00
/* Check if we should continue operations */
if (iReturn != WIN_JMP_ERROR_IO
&& iReturn != WIN_JMP_OKAY)
{
/* setjmp returned an unknown value, exit */
2004-06-21 15:19:32 +02:00
ErrorF ("winInitMultiWindowWM - setjmp returned: %d. Exiting.\n",
2003-11-14 17:48:57 +01:00
iReturn);
pthread_exit (NULL);
}
else if (iReturn == WIN_JMP_ERROR_IO)
{
2004-06-21 15:19:32 +02:00
ErrorF ("winInitMultiWindowWM - Caught IO Error. Exiting.\n");
pthread_exit (NULL);
2003-11-14 17:48:57 +01:00
}
2004-06-21 15:19:32 +02:00
/* Install our error handler */
XSetErrorHandler (winMultiWindowWMErrorHandler);
XSetIOErrorHandler (winMultiWindowWMIOErrorHandler);
2003-11-14 17:48:57 +01:00
/* Setup the display connection string x */
snprintf (pszDisplay,
512,
"127.0.0.1:%s.%d",
display,
(int) pProcArg->dwScreen);
2003-11-14 17:48:57 +01:00
/* Print the display connection string */
ErrorF ("winInitMultiWindowWM - DISPLAY=%s\n", pszDisplay);
/* Open the X display */
do
{
/* Try to open the display */
pWMInfo->pDisplay = XOpenDisplay (pszDisplay);
if (pWMInfo->pDisplay == NULL)
{
ErrorF ("winInitMultiWindowWM - Could not open display, try: %d, "
"sleeping: %d\n\f",
iRetries + 1, WIN_CONNECT_DELAY);
++iRetries;
sleep (WIN_CONNECT_DELAY);
continue;
}
else
break;
}
while (pWMInfo->pDisplay == NULL && iRetries < WIN_CONNECT_RETRIES);
/* Make sure that the display opened */
if (pWMInfo->pDisplay == NULL)
{
2004-06-21 15:19:32 +02:00
ErrorF ("winInitMultiWindowWM - Failed opening the display. "
"Exiting.\n");
2003-11-14 17:48:57 +01:00
pthread_exit (NULL);
}
ErrorF ("winInitMultiWindowWM - XOpenDisplay () returned and "
"successfully opened the display.\n");
/* Create some atoms */
pWMInfo->atmWmProtos = XInternAtom (pWMInfo->pDisplay,
"WM_PROTOCOLS",
False);
pWMInfo->atmWmDelete = XInternAtom (pWMInfo->pDisplay,
"WM_DELETE_WINDOW",
False);
pWMInfo->atmPrivMap = XInternAtom (pWMInfo->pDisplay,
2004-11-04 12:52:22 +01:00
WINDOWSWM_NATIVE_HWND,
False);
if (1) {
Cursor cursor = XCreateFontCursor (pWMInfo->pDisplay, XC_left_ptr);
if (cursor)
{
XDefineCursor (pWMInfo->pDisplay, DefaultRootWindow(pWMInfo->pDisplay), cursor);
XFreeCursor (pWMInfo->pDisplay, cursor);
}
}
2003-11-14 17:48:57 +01:00
}
/*
* 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 ()\n");
#endif
pNode = (WMMsgNodePtr)malloc(sizeof(WMMsgNodeRec));
if (pNode != NULL)
{
memcpy (&pNode->msg, pMsg, sizeof(winWMMessageRec));
PushMessage (&((WMInfoPtr)pWMInfo)->wmMsgQueue, pNode);
}
}
/*
2004-06-21 15:19:32 +02:00
* Window manager error handler
*/
static int
winMultiWindowWMErrorHandler (Display *pDisplay, XErrorEvent *pErr)
{
char pszErrorMsg[100];
if (pErr->request_code == X_ChangeWindowAttributes
&& pErr->error_code == BadAccess)
{
ErrorF ("winMultiWindowWMErrorHandler - ChangeWindowAttributes "
"BadAccess.\n");
return 0;
}
XGetErrorText (pDisplay,
pErr->error_code,
pszErrorMsg,
sizeof (pszErrorMsg));
ErrorF ("winMultiWindowWMErrorHandler - ERROR: %s\n", pszErrorMsg);
return 0;
}
/*
* Window manager IO error handler
2003-11-14 17:48:57 +01:00
*/
static int
2004-06-21 15:19:32 +02:00
winMultiWindowWMIOErrorHandler (Display *pDisplay)
2003-11-14 17:48:57 +01:00
{
2004-06-21 15:19:32 +02:00
ErrorF ("\nwinMultiWindowWMIOErrorHandler!\n\n");
2003-11-14 17:48:57 +01:00
2005-07-05 20:25:44 +02:00
if (g_shutdown)
pthread_exit(NULL);
2003-11-14 17:48:57 +01:00
/* Restart at the main entry point */
2004-06-21 15:19:32 +02:00
longjmp (g_jmpWMEntry, WIN_JMP_ERROR_IO);
2003-11-14 17:48:57 +01:00
return 0;
}
2004-06-21 15:19:32 +02:00
/*
* X message procedure error handler
*/
static int
winMultiWindowXMsgProcErrorHandler (Display *pDisplay, XErrorEvent *pErr)
{
char pszErrorMsg[100];
XGetErrorText (pDisplay,
pErr->error_code,
pszErrorMsg,
sizeof (pszErrorMsg));
#if CYGMULTIWINDOW_DEBUG
2004-06-21 15:19:32 +02:00
ErrorF ("winMultiWindowXMsgProcErrorHandler - ERROR: %s\n", pszErrorMsg);
#endif
2004-06-21 15:19:32 +02:00
return 0;
}
/*
* X message procedure IO error handler
*/
static int
winMultiWindowXMsgProcIOErrorHandler (Display *pDisplay)
{
ErrorF ("\nwinMultiWindowXMsgProcIOErrorHandler!\n\n");
/* Restart at the main entry point */
longjmp (g_jmpXMsgProcEntry, WIN_JMP_ERROR_IO);
return 0;
}
/*
* Catch RedirectError to detect other window manager running
*/
static int
winRedirectErrorHandler (Display *pDisplay, XErrorEvent *pErr)
{
redirectError = TRUE;
return 0;
}
2004-11-04 12:52:22 +01:00
/*
* Check if another window manager is running
*/
static Bool
CheckAnotherWindowManager (Display *pDisplay, DWORD dwScreen)
{
redirectError = FALSE;
XSetErrorHandler (winRedirectErrorHandler);
XSelectInput(pDisplay, RootWindow (pDisplay, dwScreen),
// SubstructureNotifyMask | ButtonPressMask
ColormapChangeMask | EnterWindowMask | PropertyChangeMask |
SubstructureRedirectMask | KeyPressMask |
ButtonPressMask | ButtonReleaseMask);
XSync (pDisplay, 0);
XSetErrorHandler (winMultiWindowXMsgProcErrorHandler);
XSelectInput(pDisplay, RootWindow (pDisplay, dwScreen),
SubstructureNotifyMask);
XSync (pDisplay, 0);
if (redirectError)
{
//ErrorF ("CheckAnotherWindowManager() - another window manager is running. Exiting.\n");
return TRUE;
}
else
{
return FALSE;
}
}
/*
* 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)
/* These two are used on their own */
#define HINT_MAX (1L<<0)
#define HINT_MIN (1L<<1)
static void
winApplyHints (Display *pDisplay, Window iWindow, HWND hWnd, HWND *zstyle)
{
static Atom windowState, motif_wm_hints, windowType;
static Atom hiddenState, fullscreenState, belowState, aboveState;
static Atom dockWindow;
static int generation;
Atom type, *pAtom = NULL;
int format;
unsigned long hint = 0, maxmin = 0, style, nitems = 0 , left = 0;
WindowPtr pWin = GetProp (hWnd, WIN_WINDOW_PROP);
if (!hWnd) return;
if (!IsWindow (hWnd)) return;
if (generation != serverGeneration) {
generation = serverGeneration;
windowState = XInternAtom(pDisplay, "_NET_WM_STATE", False);
motif_wm_hints = XInternAtom(pDisplay, "_MOTIF_WM_HINTS", False);
windowType = XInternAtom(pDisplay, "_NET_WM_WINDOW_TYPE", False);
hiddenState = XInternAtom(pDisplay, "_NET_WM_STATE_HIDDEN", False);
fullscreenState = XInternAtom(pDisplay, "_NET_WM_STATE_FULLSCREEN", False);
belowState = XInternAtom(pDisplay, "_NET_WM_STATE_BELOW", False);
aboveState = XInternAtom(pDisplay, "_NET_WM_STATE_ABOVE", False);
dockWindow = XInternAtom(pDisplay, "_NET_WM_WINDOW_TYPE_DOCK", False);
}
if (XGetWindowProperty(pDisplay, iWindow, windowState, 0L,
1L, False, XA_ATOM, &type, &format,
&nitems, &left, (unsigned char **)&pAtom) == Success)
{
if (pAtom && nitems == 1)
{
if (*pAtom == hiddenState) maxmin |= HINT_MIN;
else if (*pAtom == fullscreenState) maxmin |= HINT_MAX;
if (*pAtom == belowState) *zstyle = HWND_BOTTOM;
else if (*pAtom == aboveState) *zstyle = HWND_TOPMOST;
}
if (pAtom) XFree(pAtom);
}
nitems = left = 0;
MwmHints *mwm_hint = NULL;
if (XGetWindowProperty(pDisplay, iWindow, motif_wm_hints, 0L,
PropMwmHintsElements, False, motif_wm_hints, &type, &format,
&nitems, &left, (unsigned char **)&mwm_hint) == Success)
{
if (mwm_hint && nitems == PropMwmHintsElements && (mwm_hint->flags & MwmHintsDecorations))
{
if (!mwm_hint->decorations) hint |= HINT_NOFRAME;
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) XFree(mwm_hint);
}
nitems = left = 0;
pAtom = NULL;
if (XGetWindowProperty(pDisplay, iWindow, windowType, 0L,
1L, False, XA_ATOM, &type, &format,
&nitems, &left, (unsigned char **)&pAtom) == Success)
{
if (pAtom && nitems == 1)
{
if (*pAtom == dockWindow)
{
hint = (hint & ~HINT_NOFRAME) | HINT_SIZEBOX; /* Xming puts a sizebox on dock windows */
*zstyle = HWND_TOPMOST;
}
}
if (pAtom) XFree(pAtom);
}
/* Apply Styles, overriding hint settings from above */
style = winOverrideStyle((unsigned long)pWin);
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;
style = GetWindowLongPtr(hWnd, GWL_STYLE) & ~WS_CAPTION & ~WS_SIZEBOX; /* Just in case */
if (!style) return;
if (!hint) /* All on, but no resize of children is allowed */
style = style | WS_CAPTION | (GetParent(hWnd) ? 0 : WS_SIZEBOX);
else if (hint & HINT_NOFRAME); /* All off, so do nothing */
else style = style | ((hint & HINT_BORDER) ? WS_BORDER : 0) |
((hint & HINT_SIZEBOX) ? (GetParent(hWnd) ? 0 : WS_SIZEBOX) : 0) |
((hint & HINT_CAPTION) ? WS_CAPTION : 0);
SetWindowLongPtr (hWnd, GWL_STYLE, style);
}
void
winUpdateWindowPosition (HWND hWnd, Bool reshape, 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);
#if 0
ErrorF ("winUpdateWindowPosition - (%d, %d)-(%d, %d)\n",
rcNew.left, rcNew.top,
rcNew.right, rcNew.bottom);
#endif
AdjustWindowRectEx (&rcNew, GetWindowLongPtr (hWnd, GWL_STYLE), FALSE, WS_EX_APPWINDOW);
/* 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;
}
#if 0
ErrorF ("winUpdateWindowPosition - (%d, %d)-(%d, %d)\n",
rcNew.left, rcNew.top,
rcNew.right, rcNew.bottom);
#endif
/* Position the Windows window */
SetWindowPos (hWnd, *zstyle, rcNew.left, rcNew.top,
rcNew.right - rcNew.left, rcNew.bottom - rcNew.top,
0);
if (reshape)
{
winReshapeMultiWindow(pWin);
winUpdateRgnMultiWindow(pWin);
}
}