2909 lines
65 KiB
C
2909 lines
65 KiB
C
/*
|
||
* CDE - Common Desktop Environment
|
||
*
|
||
* Copyright (c) 1993-2012, The Open Group. All rights reserved.
|
||
*
|
||
* These libraries and programs are free software; you can
|
||
* redistribute them and/or modify them under the terms of the GNU
|
||
* Lesser General Public License as published by the Free Software
|
||
* Foundation; either version 2 of the License, or (at your option)
|
||
* any later version.
|
||
*
|
||
* These libraries and programs are distributed in the hope that
|
||
* they will be useful, but WITHOUT ANY WARRANTY; without even the
|
||
* implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
|
||
* PURPOSE. See the GNU Lesser General Public License for more
|
||
* details.
|
||
*
|
||
* You should have received a copy of the GNU Lesser General Public
|
||
* License along with these libraries and programs; if not, write
|
||
* to the Free Software Foundation, Inc., 51 Franklin Street, Fifth
|
||
* Floor, Boston, MA 02110-1301 USA
|
||
*/
|
||
/*
|
||
* (c) Copyright 1989, 1990, 1991, 1992, 1993, 1994 OPEN SOFTWARE FOUNDATION, INC.
|
||
* ALL RIGHTS RESERVED
|
||
*/
|
||
/*
|
||
* Motif Release 1.2.4
|
||
*/
|
||
/*
|
||
* (c) Copyright 1987, 1988, 1989, 1990, 1992, 1993 HEWLETT-PACKARD COMPANY
|
||
*/
|
||
|
||
/*
|
||
* Included Files:
|
||
*/
|
||
|
||
#include "WmGlobal.h"
|
||
#include "WmICCC.h"
|
||
/*
|
||
* include extern functions
|
||
*/
|
||
#include "WmCDecor.h"
|
||
#include "WmCEvent.h"
|
||
#include "WmColormap.h"
|
||
#include "WmError.h"
|
||
#include "WmEvent.h"
|
||
#include "WmFunction.h"
|
||
#include "WmGraphics.h"
|
||
#include "WmIDecor.h"
|
||
#include "WmIconBox.h"
|
||
#include "WmImage.h"
|
||
#include "WmInstantTitle.h"
|
||
#include "WmKeyFocus.h"
|
||
#include "WmPanelP.h" /* typedef needed in WmManage.h */
|
||
#include <Dt/Message.h>
|
||
#include "WmIPC.h"
|
||
#include "WmManage.h"
|
||
#include "WmMenu.h"
|
||
#include "WmProperty.h"
|
||
#include "WmProtocol.h"
|
||
#include "WmWinInfo.h"
|
||
#include "WmWinList.h"
|
||
#include "WmWinState.h"
|
||
#include "WmPresence.h"
|
||
#include "WmWrkspace.h"
|
||
#include "WmXSMP.h"
|
||
|
||
|
||
|
||
/*
|
||
* Function Declarations:
|
||
*/
|
||
|
||
Boolean IsEmbeddedClient (
|
||
ClientData *pCD,
|
||
WmFpEmbeddedClientData **ppECD);
|
||
Boolean ManageEmbeddedClient (
|
||
ClientData *pCD,
|
||
WmFpEmbeddedClientData *pECD,
|
||
long manageFlags);
|
||
Boolean IsPushRecallClient (
|
||
ClientData *pCD,
|
||
WmFpPushRecallClientData **ppPRCD);
|
||
static void HandleSubstructEvents(
|
||
Widget w,
|
||
caddr_t ptr,
|
||
XEvent *event );
|
||
Boolean UpdateEmbeddedClientsProperty(
|
||
WmScreenData *pSD );
|
||
static void ForceSubpanelWMState(Window win);
|
||
static void ReManageWindow (ClientData *pCD);
|
||
static void CheckPushRecallClient (ClientData *pCD);
|
||
|
||
|
||
/*
|
||
* Global Variables:
|
||
*/
|
||
|
||
|
||
static void ApplyPrematureClientMessages (ClientData *pCD)
|
||
{
|
||
unsigned long i, nitems, leftover;
|
||
int actualFormat;
|
||
Atom actualType;
|
||
Atom property = wmGD.xa_PREMATURE_XCLIENTMESSAGEEVENT_LIST;
|
||
XClientMessageEvent *messages = NULL;
|
||
|
||
if (!HasProperty (pCD, property)) goto err;
|
||
|
||
if (XGetWindowProperty (DISPLAY, pCD->client, property, 0L, 1000000L, True,
|
||
property, &actualType, &actualFormat, &nitems,
|
||
&leftover, (unsigned char **)&messages) != Success)
|
||
goto err;
|
||
|
||
if (actualType != property) goto err;
|
||
|
||
nitems /= sizeof (XClientMessageEvent);
|
||
|
||
if (!nitems) goto err;
|
||
|
||
for (i = 0; i < nitems; ++i) HandleClientMessage (pCD, &messages[i]);
|
||
|
||
err:
|
||
if (messages) XFree (messages);
|
||
}
|
||
|
||
|
||
|
||
/*************************************<->*************************************
|
||
*
|
||
* AdoptInitialClients (pSD)
|
||
*
|
||
* Inputs:
|
||
* -------
|
||
* pSD = pointer to screen data
|
||
*
|
||
*
|
||
* Description:
|
||
* -----------
|
||
* This function is called to find client windows that were mapped prior to
|
||
* starting (or restarting) the window manager. These windows are included
|
||
* in the set of windows managed by the window manager.
|
||
*
|
||
*************************************<->***********************************/
|
||
|
||
void AdoptInitialClients (WmScreenData *pSD)
|
||
{
|
||
Window root;
|
||
Window parent;
|
||
Window *clients;
|
||
int nAncillaries, iAnc;
|
||
Window *pAncillaryWindows, *pWin1;
|
||
WmWorkspaceData *pWS0;
|
||
unsigned int nclients;
|
||
ClientData *pcd = NULL;
|
||
PropWMState *wmStateProp;
|
||
Boolean manageOnRestart;
|
||
int i,j;
|
||
long manageFlags;
|
||
|
||
/*
|
||
* Generate list of ancillary windows (not to be managed)
|
||
*/
|
||
nAncillaries = 2 + pSD->numWorkspaces;
|
||
pAncillaryWindows = (Window *) XtMalloc (sizeof(Window)*(nAncillaries));
|
||
if (!pAncillaryWindows)
|
||
{
|
||
Warning (((char *)GETMESSAGE(46, 1, "Insufficient memory to adopt initial clients")));
|
||
ExitWM (WM_ERROR_EXIT_VALUE);
|
||
}
|
||
pWS0 = pSD->pWS;
|
||
pWin1 = pAncillaryWindows;
|
||
for (iAnc = 0; iAnc < pSD->numWorkspaces; iAnc++)
|
||
{
|
||
*pWin1 = XtWindow((pWS0)->workspaceTopLevelW);
|
||
pWin1++;
|
||
pWS0++;
|
||
}
|
||
*pWin1++ = XtWindow (pSD->screenTopLevelW);
|
||
*pWin1 = pSD->activeIconTextWin;
|
||
|
||
|
||
/*
|
||
* Look for mapped top-level windows and start managing them:
|
||
*/
|
||
|
||
if (XQueryTree (DISPLAY, pSD->rootWindow, &root, &parent, &clients,
|
||
&nclients))
|
||
{
|
||
/*
|
||
* Filter out icon windows so they don't get managed as a client
|
||
* window. Icon windows will be process in SetupClientIconWindow().
|
||
*/
|
||
XWMHints *tmphint;
|
||
|
||
for (i = 0; i < nclients; i++) {
|
||
if (clients[i]) {
|
||
if ((tmphint = XGetWMHints (DISPLAY, clients[i])) != NULL) {
|
||
if (tmphint->flags & IconWindowHint) {
|
||
for (j = 0; j < nclients; j++) {
|
||
if (clients[j] == tmphint->icon_window) {
|
||
clients[j] = None;
|
||
break;
|
||
}
|
||
}
|
||
}
|
||
XFree ((char *) tmphint);
|
||
}
|
||
}
|
||
}
|
||
|
||
for (i = 0; i < nclients; i++)
|
||
{
|
||
/* determine if the client window should be managed by wm */
|
||
if (InWindowList (clients[i], pAncillaryWindows, nAncillaries))
|
||
{
|
||
/* don't manage ancillary window manager windows */
|
||
continue;
|
||
}
|
||
if (!XFindContext (DISPLAY, clients[i], wmGD.windowContextType,
|
||
(caddr_t *)&pcd))
|
||
{
|
||
/* don't manage a window we've already established a
|
||
context for (e.g. icon windows) */
|
||
continue;
|
||
}
|
||
if (!WmGetWindowAttributes (clients[i]))
|
||
{
|
||
/* can't access the window; ignore it */
|
||
continue;
|
||
}
|
||
/* window attributes are put into the global cache */
|
||
|
||
/*
|
||
* Get the window WM_STATE property value to determine the
|
||
* initial window state if the wm is being restarted.
|
||
*/
|
||
|
||
manageFlags = MANAGEW_WM_STARTUP;
|
||
manageOnRestart = True;
|
||
|
||
if (wmGD.wmRestarted)
|
||
{
|
||
manageFlags |= MANAGEW_WM_RESTART;
|
||
if ((wmStateProp = GetWMState (clients[i])) != NULL)
|
||
{
|
||
if (wmStateProp->state == IconicState)
|
||
{
|
||
manageFlags |= MANAGEW_WM_RESTART_ICON;
|
||
}
|
||
else if (wmStateProp->state != NormalState)
|
||
{
|
||
manageOnRestart = False;
|
||
}
|
||
XFree ((char *)wmStateProp);
|
||
}
|
||
else
|
||
{
|
||
manageOnRestart = False;
|
||
}
|
||
}
|
||
|
||
/*
|
||
* Don't manage any override_redirect windows (mapped or not).
|
||
* Manage an unmapped window if it has a WM_STATE property
|
||
* and it is not Withdrawn.
|
||
* Manage any window that is mapped.
|
||
*/
|
||
|
||
if ((wmGD.windowAttributes.override_redirect != True) &&
|
||
((wmGD.wmRestarted && manageOnRestart) ||
|
||
(wmGD.windowAttributes.map_state != IsUnmapped)))
|
||
{
|
||
ManageWindow (pSD, clients[i], manageFlags);
|
||
}
|
||
}
|
||
|
||
if (nclients)
|
||
{
|
||
XFree ((char *)clients);
|
||
}
|
||
}
|
||
|
||
if (pAncillaryWindows)
|
||
{
|
||
XtFree ((char *) pAncillaryWindows);
|
||
}
|
||
|
||
} /* END OF FUNCTION AdoptInitialClients */
|
||
|
||
|
||
|
||
/*************************************<->*************************************
|
||
*
|
||
* ManageWindow (pSD, clientWindow, manageFlags)
|
||
*
|
||
*
|
||
* Description:
|
||
* -----------
|
||
* This is the highlevel function that is used to include a window in
|
||
* the set of windows that are managed by the window manager. The window
|
||
* gets reparented and decorated, gets an icon, is setup for window
|
||
* management event handling, etc. Client windows that are controlled
|
||
* by the window manager (e.g., the icon box) are also managed with
|
||
* this function.
|
||
*
|
||
*
|
||
* Inputs:
|
||
* ------
|
||
* clientWindow = window of the client that we should manage
|
||
*
|
||
* manageFlags = additional control information
|
||
*
|
||
*
|
||
* Outputs:
|
||
* -------
|
||
* pCD = initialized client data
|
||
*
|
||
*************************************<->***********************************/
|
||
|
||
void
|
||
ManageWindow (WmScreenData *pSD, Window clientWindow, long manageFlags)
|
||
{
|
||
ClientData *pCD;
|
||
int initialState;
|
||
int i;
|
||
Boolean sendConfigNotify;
|
||
WmWorkspaceData *pwsi;
|
||
WmFpEmbeddedClientData *pECD;
|
||
|
||
/*
|
||
* Get client information including window attributes and window
|
||
* property values. Use this information to determine how the window
|
||
* is to be managed.
|
||
*/
|
||
|
||
if (!(pCD = GetClientInfo (pSD, clientWindow, manageFlags)))
|
||
{
|
||
/* error getting client info; do not manage the client window */
|
||
return;
|
||
}
|
||
|
||
/*
|
||
* Handle case of transients that derive from embedded clients.
|
||
*/
|
||
if (wmGD.dtSD && (wmGD.dtSD == pCD->pSD))
|
||
{
|
||
if (pCD->transientLeader && pCD->transientLeader->pECD)
|
||
{
|
||
WmPanelistObject pPanelist;
|
||
ClientData *pCDfp = NULL;
|
||
|
||
pPanelist = (WmPanelistObject) pCD->pSD->wPanelist;
|
||
(void) XFindContext (DISPLAY, XtWindow(O_Shell (pPanelist)),
|
||
wmGD.windowContextType, (caddr_t *)&pCDfp);
|
||
|
||
pCD->transientLeader = pCDfp;
|
||
}
|
||
}
|
||
|
||
if (IsEmbeddedClient (pCD, &pECD))
|
||
{
|
||
/*
|
||
* This client is embedded in the front panel
|
||
*/
|
||
|
||
if (ManageEmbeddedClient(pCD, pECD, manageFlags))
|
||
{
|
||
/*
|
||
* ...then we've embedded it in the front
|
||
* panel--no further processing required.
|
||
*/
|
||
if (smAckState == SM_START_ACK)
|
||
{
|
||
SendClientMsg( wmGD.dtSmWindow,
|
||
(long) wmGD.xa_DT_SM_WM_PROTOCOL,
|
||
(long) wmGD.xa_DT_WM_WINDOW_ACK,
|
||
CurrentTime, NULL, 0);
|
||
}
|
||
return;
|
||
}
|
||
}
|
||
|
||
/*
|
||
* Handle case of transients that derive from embedded clients.
|
||
* !!!!
|
||
*/
|
||
#if 0
|
||
if (pCD->transientLeader && pCD->transientLeader->pAccessPanel)
|
||
{
|
||
pCD->transientLeader =
|
||
pCD->transientLeader->pAccessPanel->pCD_accessPanel;
|
||
}
|
||
#endif
|
||
|
||
if (pCD->inputMode == MWM_INPUT_SYSTEM_MODAL)
|
||
{
|
||
/*
|
||
* Put system modal windows in all workspaces to
|
||
* avoid the race condition of the window coming up
|
||
* just as the user switches workspaces.
|
||
*/
|
||
pCD->dtwmFunctions |= DtWM_FUNCTION_OCCUPY_WS;
|
||
F_AddToAllWorkspaces(0, pCD, 0);
|
||
pCD->dtwmFunctions &= ~DtWM_FUNCTION_OCCUPY_WS;
|
||
}
|
||
if (manageFlags & MANAGEW_WM_RESTART)
|
||
{
|
||
if (manageFlags & MANAGEW_WM_RESTART_ICON)
|
||
{
|
||
pCD->clientState = MINIMIZED_STATE;
|
||
}
|
||
else
|
||
{
|
||
pCD->clientState = NORMAL_STATE;
|
||
}
|
||
}
|
||
|
||
|
||
/*
|
||
* Setup the initial placement of the client window. Do interactive
|
||
* placement if configured.
|
||
*/
|
||
|
||
sendConfigNotify = InitClientPlacement (pCD, manageFlags);
|
||
|
||
|
||
/*
|
||
* Make a window frame for the client window and reparent the client
|
||
* window.
|
||
*/
|
||
|
||
if (!FrameWindow (pCD))
|
||
{
|
||
/*
|
||
* Error in framing the window; clean up the wm resources made
|
||
* up to this point for the client window. Do not manage the
|
||
* client window.
|
||
*/
|
||
|
||
UnManageWindow (pCD);
|
||
return;
|
||
}
|
||
|
||
/*
|
||
* Send config notify if the client's been moved/resized
|
||
*/
|
||
if (sendConfigNotify)
|
||
{
|
||
SendConfigureNotify (pCD);
|
||
}
|
||
|
||
/*
|
||
* Send client offset message if:
|
||
*
|
||
* 1. The client is interested.
|
||
* 2. The position we report to the user is not the client's real
|
||
* position.
|
||
* 3. There is a client offset to report.
|
||
*/
|
||
if ((pCD->protocolFlags & PROTOCOL_MWM_OFFSET) &&
|
||
(wmGD.positionIsFrame) &&
|
||
((pCD->clientOffset.x != 0) ||
|
||
(pCD->clientOffset.y != 0)))
|
||
{
|
||
SendClientOffsetMessage (pCD);
|
||
}
|
||
|
||
/*
|
||
* Make an icon for the client window if it is not a valid transient
|
||
* window.
|
||
*/
|
||
|
||
if ((pCD->clientFunctions & MWM_FUNC_MINIMIZE) &&
|
||
(pCD->transientLeader == NULL))
|
||
{
|
||
/*
|
||
* Make icons frames
|
||
* Only make one icon frame for root icons.
|
||
* Make one per workspace for icon box icons.
|
||
*/
|
||
for (i = 0; i < pCD->numInhabited; i++)
|
||
{
|
||
if ((pwsi = GetWorkspaceData(pCD->pSD, pCD->pWsList[i].wsID)))
|
||
{
|
||
|
||
if ((pCD->pSD->useIconBox &&
|
||
!(manageFlags & MANAGEW_WM_CLIENTS) &&
|
||
!(pCD->clientFlags & FRONT_PANEL_BOX)) || (i == 0))
|
||
{
|
||
/*
|
||
* Make icon inside an icon box for non-root case
|
||
*/
|
||
if (!MakeIcon (pwsi, pCD))
|
||
{
|
||
/*
|
||
* Error in making an icon for the client window;
|
||
* clean up the wm resources; do not manage the
|
||
* client window.
|
||
*/
|
||
|
||
UnManageWindow (pCD);
|
||
return;
|
||
}
|
||
else
|
||
{
|
||
XSaveContext (DISPLAY, pCD->pWsList[i].iconFrameWin,
|
||
wmGD.windowContextType, (caddr_t)pCD);
|
||
|
||
if (pCD->iconWindow && pCD->pWsList[i].iconFrameWin)
|
||
{
|
||
XGrabButton (DISPLAY, AnyButton, AnyModifier,
|
||
pCD->pWsList[i].iconFrameWin, True,
|
||
ButtonPressMask|ButtonReleaseMask|
|
||
ButtonMotionMask,
|
||
GrabModeAsync, GrabModeAsync, None,
|
||
wmGD.workspaceCursor);
|
||
}
|
||
}
|
||
}
|
||
else
|
||
{
|
||
/*
|
||
* Make root icons for a client
|
||
*/
|
||
if ((pCD->clientFunctions & MWM_FUNC_MINIMIZE) &&
|
||
(pCD->transientLeader == NULL))
|
||
{
|
||
if ((i == 0) &&
|
||
(!MakeIcon (pwsi, pCD)))
|
||
{
|
||
/*
|
||
* Error in making an icon for the client
|
||
* window; clean up the wm resources; do
|
||
* not manage the client window.
|
||
*/
|
||
|
||
UnManageWindow (pCD);
|
||
return;
|
||
}
|
||
else
|
||
{
|
||
/* copy root icon frame reference to other
|
||
* workspaces
|
||
*/
|
||
pCD->pWsList[i].iconFrameWin =
|
||
pCD->pWsList[0].iconFrameWin;
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
/*
|
||
* Register window contexts to facilitate event handling:
|
||
*/
|
||
|
||
XSaveContext (DISPLAY, pCD->clientFrameWin, wmGD.windowContextType,
|
||
(caddr_t)pCD);
|
||
|
||
XSaveContext (DISPLAY, pCD->clientBaseWin, wmGD.windowContextType,
|
||
(caddr_t)pCD);
|
||
|
||
if (DECOUPLE_TITLE_APPEARANCE(pCD) && pCD->clientTitleWin)
|
||
{
|
||
/*
|
||
* handle exposures on title bar if it has its own appearance
|
||
*/
|
||
XSaveContext (DISPLAY, pCD->clientTitleWin, wmGD.windowContextType,
|
||
(caddr_t)pCD);
|
||
}
|
||
|
||
if (pCD->clientCmapCount > 0)
|
||
{
|
||
for (i = 0; i < pCD->clientCmapCount; i++)
|
||
{
|
||
if (pCD->cmapWindows[i] != pCD->client)
|
||
{
|
||
#ifndef IBM_169380
|
||
AddColormapWindowReference(pCD, pCD->cmapWindows[i]);
|
||
#else
|
||
XSaveContext (DISPLAY, pCD->cmapWindows[i],
|
||
wmGD.windowContextType, (caddr_t)pCD);
|
||
#endif
|
||
}
|
||
}
|
||
}
|
||
|
||
pCD->clientFlags |= CLIENT_CONTEXT_SAVED;
|
||
|
||
|
||
/*
|
||
* Setup button binding handling for actions that apply to the client
|
||
* window.
|
||
*/
|
||
|
||
if (BUTTON_SPECS(pCD))
|
||
{
|
||
SetupCButtonBindings (pCD->clientBaseWin, BUTTON_SPECS(pCD));
|
||
}
|
||
|
||
/*
|
||
* Setup key binding handling for system menu accelerators.
|
||
*/
|
||
|
||
if (pCD->systemMenuSpec &&
|
||
(pCD->systemMenuSpec->accelKeySpecs))
|
||
{
|
||
SetupKeyBindings (pCD->systemMenuSpec->accelKeySpecs,
|
||
pCD->clientFrameWin, GrabModeSync, F_CONTEXT_ALL);
|
||
for (i = 0; i < pCD->numInhabited; i++)
|
||
{
|
||
if (!pCD->pWsList[i].pIconBox && pCD->pWsList[i].iconFrameWin)
|
||
{
|
||
SetupKeyBindings (pCD->systemMenuSpec->accelKeySpecs,
|
||
pCD->pWsList[i].iconFrameWin, GrabModeSync,
|
||
F_CONTEXT_ALL);
|
||
}
|
||
}
|
||
}
|
||
|
||
for (i = 0; i < pCD->numInhabited; i++)
|
||
{
|
||
if (!pCD->pWsList[i].pIconBox && pCD->pWsList[i].iconFrameWin)
|
||
{
|
||
static int iconKeySpec = 1;
|
||
static int iconAccelSpec = 1;
|
||
|
||
if ((iconKeySpec != 0) && KEY_SPECS(pCD))
|
||
{
|
||
iconKeySpec = SetupKeyBindings (KEY_SPECS(pCD),
|
||
pCD->pWsList[i].iconFrameWin,
|
||
GrabModeSync, F_CONTEXT_ICON);
|
||
}
|
||
|
||
if ((iconAccelSpec != 0) && ACCELERATOR_MENU_COUNT(pCD))
|
||
{
|
||
int n;
|
||
|
||
iconAccelSpec = 0;
|
||
for (n= 0; n < pSD->acceleratorMenuCount; n++)
|
||
{
|
||
iconAccelSpec += SetupKeyBindings (
|
||
ACCELERATOR_MENU_SPECS(pCD)[n]->accelKeySpecs,
|
||
pCD->pWsList[i].iconFrameWin, GrabModeSync,
|
||
F_CONTEXT_ICON);
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
|
||
/*
|
||
* Setup keyboard focus handling if policy is "explicit".
|
||
*/
|
||
|
||
if (wmGD.keyboardFocusPolicy == KEYBOARD_FOCUS_EXPLICIT)
|
||
{
|
||
DoExplicitSelectGrab (pCD->clientBaseWin);
|
||
}
|
||
|
||
UpdateWorkspacePresenceProperty(pCD);
|
||
|
||
|
||
/*
|
||
* Make sure the client window has been reparented ...
|
||
*/
|
||
|
||
if (!(manageFlags & MANAGEW_WM_CLIENTS))
|
||
{
|
||
XSync (DISPLAY, False);
|
||
|
||
if (pCD->clientFlags & CLIENT_DESTROYED)
|
||
{
|
||
UnManageWindow (pCD);
|
||
return;
|
||
}
|
||
}
|
||
|
||
/*
|
||
* Setup the initial display state for the client window:
|
||
*/
|
||
|
||
initialState = pCD->clientState;
|
||
if (!ClientInWorkspace (pSD->pActiveWS, pCD))
|
||
{
|
||
initialState |= UNSEEN_STATE;
|
||
}
|
||
pCD->clientState = WITHDRAWN_STATE;
|
||
pCD->clientFlags &= ~WM_INITIALIZATION;
|
||
|
||
/*
|
||
* Add to stacking list using the client's zero'th workspace
|
||
* instead of the current one because it may not be in
|
||
* the current one.
|
||
*/
|
||
AddClientToList (GetWorkspaceData (pSD, pCD->pWsList[0].wsID),
|
||
pCD, True /*on top*/);
|
||
SetClientState (pCD, initialState, GetTimestamp());
|
||
|
||
/*
|
||
* Set the keyboard input focus to the newly managed window if appropriate:
|
||
* - focus is automatically set only if the focus policy is explicit
|
||
* - if there is a system modal window active then set the focus only
|
||
* if the new window is in the system modal hierarchy
|
||
* - focus is automatically set if startupKeyFocus is selected or
|
||
* the new window is a system modal window or the current focus window
|
||
* has the new window as an application modal subordinate
|
||
* - don't automatically set the focus if the window is minimized or
|
||
* is a window that generally doesn't take input
|
||
*/
|
||
|
||
if ((wmGD.keyboardFocusPolicy == KEYBOARD_FOCUS_EXPLICIT) &&
|
||
((pCD->inputMode == MWM_INPUT_SYSTEM_MODAL) ||
|
||
((!wmGD.systemModalActive ||
|
||
(wmGD.systemModalClient == FindTransientTreeLeader (pCD))) &&
|
||
(wmGD.startupKeyFocus ||
|
||
(wmGD.keyboardFocus && (IS_APP_MODALIZED(wmGD.keyboardFocus)))) &&
|
||
!(manageFlags &
|
||
(MANAGEW_WM_STARTUP | MANAGEW_WM_RESTART | MANAGEW_WM_CLIENTS)) &&
|
||
(pCD->clientState != MINIMIZED_STATE) &&
|
||
!(pCD->clientState & UNSEEN_STATE) &&
|
||
(pCD->inputFocusModel ||
|
||
(pCD->protocolFlags & PROTOCOL_WM_TAKE_FOCUS)))))
|
||
{
|
||
Do_Focus_Key (pCD, GetTimestamp() , ALWAYS_SET_FOCUS);
|
||
}
|
||
else if ((pCD->inputMode == MWM_INPUT_SYSTEM_MODAL) ||
|
||
(wmGD.keyboardFocus && IS_APP_MODALIZED(wmGD.keyboardFocus)))
|
||
{
|
||
Do_Focus_Key ((ClientData *)NULL, GetTimestamp() , ALWAYS_SET_FOCUS);
|
||
}
|
||
|
||
if (smAckState == SM_START_ACK)
|
||
{
|
||
SendClientMsg( wmGD.dtSmWindow, (long) wmGD.xa_DT_SM_WM_PROTOCOL,
|
||
(long) wmGD.xa_DT_WM_WINDOW_ACK,
|
||
CurrentTime, NULL, 0);
|
||
}
|
||
|
||
ApplyPrematureClientMessages (pCD);
|
||
|
||
if (!HasProperty (pCD, wmGD.xa__NET_WM_STATE))
|
||
UpdateNetWmState (pCD->client, NULL, 0, _NET_WM_STATE_REMOVE);
|
||
|
||
/*
|
||
* Free the initial property list. This will force
|
||
* reads of properties that change after the initial
|
||
* management (see HasProperty() function.)
|
||
*/
|
||
DiscardInitialPropertyList (pCD);
|
||
|
||
CheckPushRecallClient (pCD);
|
||
|
||
} /* END OF FUNCTION ManageWindow */
|
||
|
||
|
||
|
||
/*************************************<->*************************************
|
||
*
|
||
* UnManageWindow (pCD)
|
||
*
|
||
*
|
||
* Description:
|
||
* -----------
|
||
* This function removes a top-level client window and it's transients
|
||
* from the set of windows that is managed by the window manager.
|
||
*
|
||
*
|
||
* Inputs:
|
||
* ------
|
||
* pCD - pointer to client data of window to unmanage
|
||
*
|
||
*************************************<->***********************************/
|
||
|
||
void UnManageWindow (ClientData *pCD)
|
||
{
|
||
if (pCD->pECD)
|
||
{
|
||
WmFpEmbeddedClientData *pECD;
|
||
|
||
pECD = (WmFpEmbeddedClientData *) pCD->pECD;
|
||
|
||
XtRemoveEventHandler(XtWindowToWidget (DISPLAY1, pECD->winParent),
|
||
(SubstructureRedirectMask | SubstructureNotifyMask),
|
||
False,
|
||
(XtEventHandler)HandleSubstructEvents,
|
||
(XtPointer)(pCD));
|
||
|
||
pECD->pCD = NULL;
|
||
UpdateEmbeddedClientsProperty (pCD->pSD);
|
||
}
|
||
|
||
if (pCD->pPRCD)
|
||
{
|
||
WmFpPushRecallClientData *pPRCD;
|
||
int j;
|
||
|
||
pPRCD = (WmFpPushRecallClientData *) pCD->pSD->pPRCD;
|
||
|
||
for (j = 0;
|
||
j < pCD->pSD->numPushRecallClients;
|
||
j++, pPRCD++)
|
||
{
|
||
/*
|
||
* Clean out all slots used by this client.
|
||
*/
|
||
if ((!strcmp ((char *)pCD->clientName,
|
||
(char *)(pPRCD->pchResName))) &&
|
||
(pPRCD->pCD == pCD))
|
||
{
|
||
pPRCD->pCD = NULL;
|
||
}
|
||
}
|
||
pCD->pPRCD = NULL;
|
||
}
|
||
|
||
/*
|
||
* Withdraw all the transient children of this window.
|
||
*/
|
||
|
||
if (pCD->transientChildren != NULL)
|
||
{
|
||
WithdrawTransientChildren (pCD);
|
||
}
|
||
|
||
|
||
/*
|
||
* If this is a transient window, then delete it from the leader's
|
||
* list of transients.
|
||
*/
|
||
|
||
if (pCD->transientLeader)
|
||
{
|
||
DeleteTransient (pCD);
|
||
|
||
/* If this was a modal dialog box, then replay the event. */
|
||
if ( wmGD.replayEnterEvent )
|
||
{
|
||
XPutBackEvent( DISPLAY, (XEvent*)&wmGD.savedEnterEvent );
|
||
/* Reset event flag to false */
|
||
wmGD.replayEnterEvent = False;
|
||
}
|
||
}
|
||
|
||
|
||
/*
|
||
* Withdraw this window
|
||
*/
|
||
|
||
WithdrawWindow (pCD);
|
||
|
||
} /* END OF FUNCTION UnManageWindow */
|
||
|
||
|
||
|
||
/*************************************<->*************************************
|
||
*
|
||
* WithdrawTransientChildren (pCD)
|
||
*
|
||
*
|
||
* Description:
|
||
* -----------
|
||
* This function withdraws all transient children of the specified window.
|
||
*
|
||
*
|
||
* Inputs:
|
||
* ------
|
||
* pCD = pointer to client data of the leader of the transient tree.
|
||
*
|
||
*************************************<->***********************************/
|
||
|
||
void WithdrawTransientChildren (ClientData *pCD)
|
||
{
|
||
ClientData *pcdNext;
|
||
ClientData *pcdThis;
|
||
|
||
|
||
pcdNext = pCD->transientChildren;
|
||
while (pcdNext)
|
||
{
|
||
if (pcdNext->transientChildren)
|
||
{
|
||
WithdrawTransientChildren (pcdNext);
|
||
}
|
||
pcdThis = pcdNext;
|
||
pcdNext = pcdThis->transientSiblings;
|
||
DeleteTransient(pcdThis);
|
||
WithdrawWindow (pcdThis);
|
||
}
|
||
|
||
} /* END OF FUNCTION WithdrawTransientChildren */
|
||
|
||
|
||
|
||
/*************************************<->*************************************
|
||
*
|
||
* WithdrawWindow (pCD)
|
||
*
|
||
*
|
||
* Description:
|
||
* -----------
|
||
* This function removes a top-level client window from the set of windows
|
||
* that is managed by the window manager. All window manager resources
|
||
* associtated with the client window are freed up (possibly cached for
|
||
* reuse). Any custom system menu is destroyed.
|
||
*
|
||
*
|
||
* Inputs:
|
||
* ------
|
||
* pCD - pointer to client data of window to withdraw
|
||
*
|
||
*************************************<->***********************************/
|
||
|
||
void WithdrawWindow (ClientData *pCD)
|
||
{
|
||
int x;
|
||
int y;
|
||
int i;
|
||
XWindowChanges xwc;
|
||
|
||
if (pCD->pSD->topClient == pCD) pCD->pSD->topClient = NULL;
|
||
if (pCD->pSD->bottomClient == pCD) pCD->pSD->bottomClient = NULL;
|
||
|
||
/*
|
||
* Put the client window into a withdrawn state:
|
||
*
|
||
* - remove the icon/client window from the screen
|
||
* - make sure the input focus no longer is associted with the window
|
||
* - free the icon placement (if necessary)
|
||
*/
|
||
SetClientWsIndex (pCD);
|
||
|
||
if (!(pCD->clientFlags & WM_INITIALIZATION))
|
||
{
|
||
if (!pCD->transientLeader)
|
||
{
|
||
DeleteClientFromList (pCD->pSD->pActiveWS, pCD);
|
||
}
|
||
ResetWithdrawnFocii (pCD);
|
||
if (pCD->clientState & MINIMIZED_STATE)
|
||
{
|
||
if (wmGD.iconAutoPlace && (!(P_ICON_BOX(pCD))))
|
||
{
|
||
WmWorkspaceData *pWsTmp;
|
||
WsClientData *pWsc;
|
||
int j;
|
||
|
||
/*
|
||
* Clean up icon placement data in all inhabited
|
||
* workspaces
|
||
*/
|
||
for (j = 0; j< pCD->numInhabited; j++)
|
||
{
|
||
pWsc = &(pCD->pWsList[j]);
|
||
|
||
if (pWsc->iconPlace != NO_ICON_PLACE)
|
||
{
|
||
if ((pWsTmp=GetWorkspaceData(pCD->pSD, pWsc->wsID)))
|
||
{
|
||
pWsTmp->IPData.placeList[pWsc->iconPlace].pCD
|
||
= NULL;
|
||
}
|
||
}
|
||
}
|
||
}
|
||
if (ICON_FRAME_WIN(pCD))
|
||
{
|
||
XUnmapWindow (DISPLAY, ICON_FRAME_WIN(pCD));
|
||
}
|
||
XFlush (DISPLAY);
|
||
}
|
||
else if ((pCD->clientState == NORMAL_STATE) ||
|
||
(pCD->clientState == MAXIMIZED_STATE))
|
||
{
|
||
XUnmapWindow (DISPLAY, pCD->clientFrameWin);
|
||
XFlush (DISPLAY);
|
||
}
|
||
}
|
||
/*
|
||
* Clean up the workspace presence dialog if it's
|
||
* connected to this client.
|
||
*/
|
||
if ((pCD->pSD->presence.shellW) &&
|
||
(pCD->pSD->presence.pCDforClient == pCD))
|
||
{
|
||
if (pCD->pSD->presence.onScreen)
|
||
{
|
||
HidePresenceBox (pCD->pSD, True);
|
||
}
|
||
pCD->pSD->presence.pCDforClient = NULL;
|
||
}
|
||
|
||
InstantTitleHideDialog (pCD);
|
||
|
||
/*
|
||
* Check to see if the window is being unmanaged because the window
|
||
* was destroyed.
|
||
*/
|
||
|
||
if (!(pCD->clientFlags & CLIENT_DESTROYED))
|
||
{
|
||
XEvent eventReturn;
|
||
|
||
if (XCheckTypedWindowEvent (DISPLAY, pCD->clientBaseWin, DestroyNotify,
|
||
&eventReturn))
|
||
{
|
||
pCD->clientFlags |= CLIENT_DESTROYED;
|
||
}
|
||
}
|
||
|
||
|
||
/*
|
||
* Reparent the client window back to root if the window has been
|
||
* reparented by the window manager. Remove the window from the
|
||
* window managers save-set if necessary.
|
||
*/
|
||
|
||
if ((pCD->clientFlags & CLIENT_REPARENTED) &&
|
||
!(pCD->clientFlags & CLIENT_DESTROYED))
|
||
{
|
||
SetWMState (pCD->client, WithdrawnSTATE,
|
||
pCD->pWsList[0].iconFrameWin);
|
||
|
||
if (pCD->maxConfig)
|
||
{
|
||
x = pCD->maxX;
|
||
y = pCD->maxY;
|
||
}
|
||
else
|
||
{
|
||
int xoff, yoff;
|
||
|
||
if(wmGD.positionIsFrame)
|
||
{
|
||
CalculateGravityOffset (pCD, &xoff, &yoff);
|
||
x = pCD->clientX - xoff;
|
||
y = pCD->clientY - yoff;
|
||
}
|
||
else
|
||
{
|
||
x = pCD->clientX;
|
||
y = pCD->clientY;
|
||
}
|
||
}
|
||
|
||
XDeleteProperty (DISPLAY, pCD->client, wmGD.xa__NET_WM_STATE);
|
||
XDeleteProperty (DISPLAY, pCD->client,
|
||
wmGD.xa_PREMATURE_XCLIENTMESSAGEEVENT_LIST);
|
||
XUnmapWindow (DISPLAY, pCD->client);
|
||
XReparentWindow (DISPLAY, pCD->client, ROOT_FOR_CLIENT(pCD), x, y);
|
||
|
||
/* give the window back it's X border */
|
||
xwc.border_width = pCD->xBorderWidth;
|
||
XConfigureWindow(DISPLAY, pCD->client, CWBorderWidth, &xwc);
|
||
|
||
if (pCD->iconWindow && (pCD->clientFlags & ICON_REPARENTED))
|
||
{
|
||
XUnmapWindow (DISPLAY, pCD->iconWindow);
|
||
XReparentWindow (DISPLAY, pCD->iconWindow, ROOT_FOR_CLIENT(pCD),
|
||
pCD->pWsList[0].iconX, pCD->pWsList[0].iconY);
|
||
}
|
||
}
|
||
|
||
|
||
if ((pCD->clientFlags & CLIENT_IN_SAVE_SET) &&
|
||
!(pCD->clientFlags & CLIENT_DESTROYED))
|
||
{
|
||
XRemoveFromSaveSet (DISPLAY, pCD->client);
|
||
|
||
if (pCD->iconWindow && (pCD->clientFlags & ICON_IN_SAVE_SET))
|
||
{
|
||
XRemoveFromSaveSet (DISPLAY, pCD->iconWindow);
|
||
}
|
||
}
|
||
|
||
/*
|
||
* Free a custom system menu if one was created.
|
||
*/
|
||
|
||
FreeCustomMenuSpec (pCD->systemMenuSpec);
|
||
|
||
/*
|
||
* Free the client window frame:
|
||
*/
|
||
|
||
if (pCD->clientFrameWin)
|
||
{
|
||
FreeClientFrame (pCD);
|
||
}
|
||
|
||
|
||
/*
|
||
* Free the icon associated with the client window:
|
||
*/
|
||
|
||
if ((pCD->iconFlags & ICON_HINTS_PIXMAP) && pCD->iconPixmap)
|
||
{
|
||
XFreePixmap (DISPLAY, pCD->iconPixmap);
|
||
}
|
||
|
||
if ((pCD->numInhabited > 0) && ICON_FRAME_WIN(pCD))
|
||
{
|
||
FreeIcon (pCD);
|
||
}
|
||
|
||
|
||
/*
|
||
* Free up the client protocol list:
|
||
*/
|
||
|
||
if (pCD->clientProtocols)
|
||
{
|
||
XtFree ((char *)pCD->clientProtocols);
|
||
}
|
||
|
||
|
||
/*
|
||
* Free up the mwm messages list:
|
||
*/
|
||
|
||
if (pCD->mwmMessages)
|
||
{
|
||
XtFree ((char *)pCD->mwmMessages);
|
||
}
|
||
|
||
|
||
/*
|
||
* Delete client window manager timers:
|
||
*/
|
||
|
||
DeleteClientWmTimers (pCD);
|
||
|
||
|
||
/*
|
||
* Free up window context associations.
|
||
*/
|
||
DeleteClientContext (pCD);
|
||
|
||
/*
|
||
* Count backward for efficiency --
|
||
* removes from end of list.
|
||
*/
|
||
for (i = pCD->numInhabited - 1; i >= 0; i--)
|
||
{
|
||
TakeClientOutOfWorkspace (
|
||
GetWorkspaceData(pCD->pSD, pCD->pWsList[i].wsID),
|
||
pCD);
|
||
}
|
||
|
||
/*
|
||
* Free up window manager resources:
|
||
*/
|
||
|
||
if (!(pCD->clientFlags & CLIENT_WM_CLIENTS))
|
||
{
|
||
if (pCD->clientName)
|
||
{
|
||
XFree ((char *) (pCD->clientName));
|
||
}
|
||
if (pCD->clientClass)
|
||
{
|
||
XFree ((char *) (pCD->clientClass));
|
||
}
|
||
}
|
||
|
||
if ((pCD->clientFlags & CLIENT_HINTS_TITLE) && pCD->clientTitle)
|
||
{
|
||
XmStringFree (pCD->clientTitle);
|
||
}
|
||
|
||
if ((pCD->iconFlags & ICON_HINTS_TITLE) && pCD->iconTitle)
|
||
{
|
||
XmStringFree (pCD->iconTitle);
|
||
}
|
||
|
||
if (pCD->instantTitle)
|
||
{
|
||
XmStringFree (pCD->instantTitle);
|
||
}
|
||
|
||
if (pCD->clientCmapCount > 0)
|
||
{
|
||
for (i = 0; i < pCD->clientCmapCount; i++)
|
||
{
|
||
if (pCD->cmapWindows[i] != pCD->client)
|
||
{
|
||
#ifndef IBM_169380
|
||
RemoveColormapWindowReference(pCD, pCD->cmapWindows[i]);
|
||
#else
|
||
XDeleteContext (DISPLAY, pCD->cmapWindows[i],
|
||
wmGD.windowContextType);
|
||
#endif
|
||
}
|
||
}
|
||
XtFree ((char *) (pCD->cmapWindows));
|
||
XtFree ((char *) (pCD->clientCmapList));
|
||
XtFree ((char *) (pCD->clientCmapFlags));
|
||
}
|
||
|
||
/*
|
||
* Insure list of initial properties has been freed.
|
||
*/
|
||
DiscardInitialPropertyList (pCD);
|
||
|
||
/*
|
||
* free up list of workspace specific data
|
||
*/
|
||
if ((pCD)->pWsList)
|
||
{
|
||
XtFree ((char *) (pCD->pWsList));
|
||
}
|
||
|
||
/*
|
||
* free up workspace hints
|
||
*/
|
||
if (pCD->pWorkspaceHints)
|
||
{
|
||
XtFree ((char *)pCD->pWorkspaceHints);
|
||
}
|
||
|
||
if (pCD->smClientID)
|
||
XFree (pCD->smClientID);
|
||
|
||
/*
|
||
* Clean up references to this data before we free it.
|
||
*/
|
||
if (wmGD.menuClient == pCD) {
|
||
wmGD.menuClient = NULL;
|
||
}
|
||
|
||
if (wmGD.gadgetClient == pCD) {
|
||
wmGD.gadgetClient = NULL;
|
||
wmGD.gadgetDepressed = 0;
|
||
}
|
||
|
||
if (wmGD.clickData.pCD == pCD) {
|
||
wmGD.clickData.pCD = NULL;
|
||
}
|
||
|
||
if (wmGD.nextKeyboardFocus == pCD)
|
||
wmGD.nextKeyboardFocus = NULL;
|
||
if (wmGD.keyboardFocus == pCD)
|
||
wmGD.keyboardFocus = NULL;
|
||
|
||
/*
|
||
* Fix for 5325 - Delete reference by dirty stack
|
||
*/
|
||
ClearDirtyStackEntry(pCD);
|
||
|
||
XtFree ((char *)pCD);
|
||
|
||
|
||
} /* END OF FUNCTION WithdrawWindow */
|
||
|
||
|
||
|
||
/*************************************<->*************************************
|
||
*
|
||
* DeleteClientContext (pCD)
|
||
*
|
||
*
|
||
* Description:
|
||
* -----------
|
||
* This function deletes the client from the X context manager
|
||
*
|
||
*
|
||
* Inputs:
|
||
* ------
|
||
* pCD - pointer to client data
|
||
*
|
||
* Outputs:
|
||
* -------
|
||
*
|
||
* Comments:
|
||
* --------
|
||
*
|
||
*************************************<->***********************************/
|
||
|
||
void DeleteClientContext (ClientData *pCD)
|
||
{
|
||
/*
|
||
* Free up window context associations. The context for the client
|
||
* window is always set if there is a client data structure.
|
||
*/
|
||
|
||
XDeleteContext (DISPLAY, pCD->client, wmGD.windowContextType);
|
||
if (pCD->clientFlags & CLIENT_CONTEXT_SAVED)
|
||
{
|
||
XDeleteContext (DISPLAY, pCD->clientFrameWin, wmGD.windowContextType);
|
||
XDeleteContext (DISPLAY, pCD->clientBaseWin, wmGD.windowContextType);
|
||
if (DECOUPLE_TITLE_APPEARANCE(pCD))
|
||
{
|
||
XDeleteContext (DISPLAY, pCD->clientTitleWin,
|
||
wmGD.windowContextType);
|
||
}
|
||
if (ICON_FRAME_WIN(pCD))
|
||
{
|
||
int k;
|
||
|
||
for (k=0; k < pCD->numInhabited; k++)
|
||
{
|
||
XDeleteContext (DISPLAY, pCD->pWsList[k].iconFrameWin,
|
||
wmGD.windowContextType);
|
||
}
|
||
}
|
||
pCD->clientFlags &= ~CLIENT_CONTEXT_SAVED;
|
||
}
|
||
|
||
} /* END OF FUNCTION DeleteClientContext */
|
||
|
||
|
||
|
||
/*************************************<->*************************************
|
||
*
|
||
* ResetWitdrawnFocii (pCD)
|
||
*
|
||
*
|
||
* Description:
|
||
* -----------
|
||
* This function resets the various types of focus if they are set to a
|
||
* window being withdrawn.
|
||
*
|
||
*
|
||
*
|
||
* Inputs:
|
||
* ------
|
||
* pCD - pointer to client data
|
||
*
|
||
* Outputs:
|
||
* -------
|
||
*
|
||
* Comments:
|
||
* --------
|
||
*
|
||
*************************************<->***********************************/
|
||
|
||
void ResetWithdrawnFocii (ClientData *pCD)
|
||
{
|
||
if ((wmGD.keyboardFocus == pCD) ||
|
||
/* BEGIN fix for CDExc21090 */
|
||
((wmGD.keyboardFocus == (ClientData *)NULL) &&
|
||
(wmGD.nextKeyboardFocus == pCD) &&
|
||
(wmGD.keyboardFocusPolicy == KEYBOARD_FOCUS_EXPLICIT)))
|
||
/* END fix for CDExc21090 */
|
||
{
|
||
if (wmGD.autoKeyFocus &&
|
||
(wmGD.keyboardFocusPolicy == KEYBOARD_FOCUS_EXPLICIT))
|
||
{
|
||
/* local hack: if we've already received a map for a new
|
||
** focus window, be sure to use wmGD.nextKeyboardFocus; otherwise
|
||
** AutoResetKeyFocus chooses an essentially arbitrary window to
|
||
** set focus to.
|
||
*/
|
||
if (wmGD.nextKeyboardFocus == pCD)
|
||
AutoResetKeyFocus (pCD, GetTimestamp());
|
||
else
|
||
Do_Focus_Key ((ClientData *)wmGD.nextKeyboardFocus,
|
||
GetTimestamp(), ALWAYS_SET_FOCUS);
|
||
}
|
||
else
|
||
{
|
||
/*
|
||
* Set the focus to the default state if the focus is not in
|
||
* the process of being set (i.e. a FocusIn event will be
|
||
* coming along shortly.
|
||
*/
|
||
|
||
if (wmGD.nextKeyboardFocus == wmGD.keyboardFocus)
|
||
{
|
||
Do_Focus_Key ((ClientData *)NULL, GetTimestamp(),
|
||
ALWAYS_SET_FOCUS | WORKSPACE_IF_NULL);
|
||
}
|
||
}
|
||
SetKeyboardFocus ((ClientData *)NULL, 0);
|
||
}
|
||
|
||
if (((pCD->inputMode == MWM_INPUT_PRIMARY_APPLICATION_MODAL) ||
|
||
(pCD->inputMode == MWM_INPUT_FULL_APPLICATION_MODAL)) &&
|
||
(wmGD.keyboardFocusPolicy == KEYBOARD_FOCUS_POINTER))
|
||
{
|
||
/*
|
||
* Repair the focus if an application modal dialog went
|
||
* away. We may not see an enter event and have the focus
|
||
* set to the wrong place.
|
||
*/
|
||
RepairFocus ();
|
||
}
|
||
|
||
if (wmGD.nextKeyboardFocus == pCD)
|
||
{
|
||
wmGD.nextKeyboardFocus = NULL;
|
||
}
|
||
|
||
if (ACTIVE_PSD->colormapFocus == pCD)
|
||
{
|
||
SetColormapFocus (ACTIVE_PSD, (ClientData *)NULL);
|
||
}
|
||
|
||
} /* END OF FUNCTION ResetWithdrawnFocii */
|
||
|
||
|
||
|
||
/*************************************<->*************************************
|
||
*
|
||
* FreeClientFrame (pCD)
|
||
*
|
||
*
|
||
* Description:
|
||
* -----------
|
||
* This function frees up frame windows and associated resources.
|
||
*
|
||
*
|
||
* Inputs:
|
||
* ------
|
||
* pCD - pointer to client data
|
||
*
|
||
*************************************<->***********************************/
|
||
|
||
void FreeClientFrame (ClientData *pCD)
|
||
{
|
||
if (pCD->pclientTopShadows) {
|
||
FreeRList (pCD->pclientTopShadows);
|
||
pCD->pclientTopShadows = NULL;
|
||
}
|
||
if (pCD->pclientBottomShadows) {
|
||
FreeRList (pCD->pclientBottomShadows);
|
||
pCD->pclientBottomShadows = NULL;
|
||
}
|
||
if (pCD->pclientTitleTopShadows) {
|
||
FreeRList (pCD->pclientTitleTopShadows);
|
||
pCD->pclientTitleTopShadows = NULL;
|
||
}
|
||
if (pCD->pclientTitleBottomShadows) {
|
||
FreeRList (pCD->pclientTitleBottomShadows);
|
||
pCD->pclientTitleBottomShadows = NULL;
|
||
}
|
||
if (pCD->pclientMatteTopShadows) {
|
||
FreeRList (pCD->pclientMatteTopShadows);
|
||
pCD->pclientMatteTopShadows = NULL;
|
||
}
|
||
if (pCD->pclientMatteBottomShadows) {
|
||
FreeRList (pCD->pclientMatteBottomShadows);
|
||
pCD->pclientMatteBottomShadows = NULL;
|
||
}
|
||
if (pCD->pTitleGadgets) {
|
||
XtFree ((char *)pCD->pTitleGadgets);
|
||
pCD->pTitleGadgets = NULL;
|
||
pCD->cTitleGadgets = 0;
|
||
}
|
||
if (pCD->pResizeGadgets) {
|
||
XtFree ((char *)pCD->pResizeGadgets);
|
||
pCD->pResizeGadgets = NULL;
|
||
}
|
||
|
||
/* destroy frame window & all children */
|
||
XDestroyWindow (DISPLAY, pCD->clientFrameWin);
|
||
|
||
} /* END OF FUNCTION FreeClientFrame */
|
||
|
||
|
||
|
||
/*************************************<->*************************************
|
||
*
|
||
* FreeIcon (pCD)
|
||
*
|
||
*
|
||
* Description:
|
||
* -----------
|
||
* This function frees up icon windows and associated resources.
|
||
*
|
||
*
|
||
* Inputs:
|
||
* ------
|
||
* pCD - pointer to client data
|
||
*
|
||
*************************************<->***********************************/
|
||
|
||
void FreeIcon (ClientData *pCD)
|
||
{
|
||
WmWorkspaceData *pWsTmp;
|
||
int i;
|
||
|
||
if (pCD->piconTopShadows) {
|
||
FreeRList (pCD->piconTopShadows);
|
||
pCD->piconTopShadows = NULL;
|
||
}
|
||
if (pCD->piconBottomShadows) {
|
||
FreeRList (pCD->piconBottomShadows);
|
||
pCD->piconBottomShadows = NULL;
|
||
}
|
||
|
||
/*
|
||
* destroy frame window & all children
|
||
*/
|
||
|
||
if ((pCD->pSD->useIconBox) && pCD->pWsList[0].pIconBox)
|
||
{
|
||
/*
|
||
* We're using icon boxes and it's in at least one ...
|
||
* Delete from all workspaces we live in
|
||
*/
|
||
for (i = 0; i< pCD->numInhabited; i++)
|
||
{
|
||
if ((pWsTmp = GetWorkspaceData(pCD->pSD, pCD->pWsList[i].wsID)))
|
||
{
|
||
DeleteIconFromBox (pWsTmp->pIconBox, pCD);
|
||
}
|
||
}
|
||
}
|
||
else
|
||
{
|
||
/* only one window, so destroying its first reference will
|
||
* clean it up adequately
|
||
*/
|
||
if (pCD->pWsList[0].iconFrameWin)
|
||
{
|
||
XDestroyWindow (DISPLAY, pCD->pWsList[0].iconFrameWin);
|
||
}
|
||
}
|
||
|
||
} /* END OF FUNCTION FreeIcon */
|
||
|
||
|
||
|
||
|
||
/*************************************<->*************************************
|
||
*
|
||
* WithdrawDialog (dialogboxW)
|
||
*
|
||
*
|
||
* Description:
|
||
* -----------
|
||
* This function removes a DialogBox widget "client" from the set of windows
|
||
* that are managed by the window manager.
|
||
*
|
||
*
|
||
* Inputs:
|
||
* ------
|
||
* dialogboxW = DialogBox widget to withdraw.
|
||
*
|
||
* Comments:
|
||
* --------
|
||
* Does not maintain the WM_STATE property on the dialog "client".
|
||
*
|
||
*************************************<->***********************************/
|
||
|
||
void WithdrawDialog (Widget dialogboxW)
|
||
{
|
||
int i;
|
||
ClientData *pCD = NULL;
|
||
|
||
/*
|
||
* Get the dialog shell window client data.
|
||
*/
|
||
|
||
if (XFindContext (DISPLAY, XtWindow (XtParent (dialogboxW)),
|
||
wmGD.windowContextType, (caddr_t *)&pCD))
|
||
return;
|
||
|
||
XtUnmanageChild (dialogboxW);
|
||
DeleteClientFromList (ACTIVE_WS, pCD);
|
||
/* TakeClientOutOfWorkspace (ACTIVE_WS, pCD); */
|
||
|
||
/*
|
||
* Count backward for efficiency --
|
||
* removes from end of list.
|
||
*/
|
||
for (i = pCD->numInhabited - 1; i >= 0; i--)
|
||
{
|
||
TakeClientOutOfWorkspace (
|
||
GetWorkspaceData(pCD->pSD, pCD->pWsList[i].wsID),
|
||
pCD);
|
||
}
|
||
|
||
ResetWithdrawnFocii (pCD);
|
||
XUnmapWindow (DISPLAY, pCD->clientFrameWin);
|
||
|
||
} /* END OF FUNCTION WithdrawDialog */
|
||
|
||
|
||
|
||
/*************************************<->*************************************
|
||
*
|
||
* ReManageDialog (pSD, dialogboxW)
|
||
*
|
||
*
|
||
* Description:
|
||
* -----------
|
||
* This function remanages a DialogBox "client" that was unmanaged via
|
||
* WithdrawDialog ().
|
||
*
|
||
*
|
||
* Inputs:
|
||
* ------
|
||
* pSD = pointer to screen data
|
||
* dialogboxW = DialogBox widget to remanage.
|
||
*
|
||
*
|
||
* Outputs:
|
||
* -------
|
||
* Does not maintain the WM_STATE property on the dialog "client".
|
||
*
|
||
*************************************<->***********************************/
|
||
|
||
void ReManageDialog (WmScreenData *pSD, Widget dialogboxW)
|
||
{
|
||
ClientData *pCD = NULL;
|
||
|
||
/*
|
||
* Get the dialog shell window client data.
|
||
*/
|
||
|
||
if (XFindContext (DISPLAY, XtWindow (XtParent (dialogboxW)),
|
||
wmGD.windowContextType, (caddr_t *)&pCD))
|
||
return;
|
||
|
||
/*
|
||
* The order is important here:
|
||
*/
|
||
|
||
/*
|
||
* Put system modal windows in all workspaces to
|
||
* avoid the race condition of the window coming up
|
||
* just as the user switches workspaces OR when
|
||
* the window is up and a user switces workspaces
|
||
* with a key binding. We may want to eventually short
|
||
* circuit F_Functions any time there is a modal
|
||
* window up, but for now, we will just make sure
|
||
* the modal window appears in all workspaces
|
||
*/
|
||
|
||
pCD->dtwmFunctions |= DtWM_FUNCTION_OCCUPY_WS;
|
||
F_AddToAllWorkspaces(0, pCD, 0);
|
||
pCD->dtwmFunctions &= ~DtWM_FUNCTION_OCCUPY_WS;
|
||
|
||
if (pSD->clientList)
|
||
{
|
||
StackWindow (pSD->pActiveWS, &pCD->clientEntry,
|
||
TRUE, (ClientListEntry *) NULL);
|
||
}
|
||
AddClientToList (pSD->pActiveWS, pCD, True /*on top*/);
|
||
XMapWindow (DISPLAY, pCD->clientFrameWin);
|
||
XtManageChild (dialogboxW);
|
||
|
||
if (wmGD.keyboardFocusPolicy == KEYBOARD_FOCUS_EXPLICIT)
|
||
{
|
||
Do_Focus_Key (pCD, GetTimestamp() , ALWAYS_SET_FOCUS);
|
||
}
|
||
|
||
|
||
} /* END OF FUNCTION ReManageDialog */
|
||
|
||
|
||
/*************************************<->*************************************
|
||
*
|
||
* RegisterEmbeddedClients (wPanelist, pECD, count)
|
||
*
|
||
*
|
||
* Description:
|
||
* -----------
|
||
* This function registers a list of clients to be embedded in the
|
||
* front panel subsystem.
|
||
*
|
||
*
|
||
* Inputs:
|
||
* ------
|
||
* wPanelist = front panel object (widget)
|
||
* pECD = pointer to list of data for clients to embed
|
||
* count = number of elements in the list
|
||
*
|
||
*
|
||
* Outputs:
|
||
* -------
|
||
*
|
||
*************************************<->***********************************/
|
||
|
||
void
|
||
RegisterEmbeddedClients (
|
||
Widget wPanelist,
|
||
WmFpEmbeddedClientList pECD,
|
||
int count)
|
||
{
|
||
WmScreenData *pSD;
|
||
int i;
|
||
|
||
for (i= 0; i < wmGD.numScreens; i++)
|
||
{
|
||
pSD = &(wmGD.Screens[i]);
|
||
|
||
if (pSD->managed)
|
||
{
|
||
if (pSD->wPanelist == wPanelist)
|
||
break;
|
||
}
|
||
}
|
||
|
||
if (i < wmGD.numScreens)
|
||
{
|
||
pSD->pECD = (struct _WmFpEmbeddedClientData *) pECD;
|
||
pSD->numEmbeddedClients = count;
|
||
}
|
||
#ifdef DEBUG
|
||
else
|
||
{
|
||
fprintf (stderr, "Couldn't match wPanelist to screen data\n");
|
||
}
|
||
#endif /* DEBUG */
|
||
|
||
} /* END OF FUNCTION RegisterEmbeddedClients */
|
||
|
||
|
||
#define LTT_INCREMENT 16
|
||
/*************************************<->*************************************
|
||
*
|
||
* ListTransientSubtree (pCD, ppWins, pSize, pCount)
|
||
*
|
||
*
|
||
* Description:
|
||
* -----------
|
||
* This function returns the list of windows in a transient window
|
||
* tree.
|
||
*
|
||
*
|
||
* Inputs:
|
||
* ------
|
||
* pCD - pointer to client data of a window.
|
||
* ppWins - address of a pointer to a list of windows
|
||
* (this must be in the heap -- XtMalloc).
|
||
* pSize - address of variable with size of list
|
||
* pCount - address of variable with number of windows in list
|
||
*
|
||
* Outputs:
|
||
* -------
|
||
* *ppWins - may point to a new area of memory if list grows
|
||
* *pSize - if list has to grow, this may be bigger
|
||
* *pCount - number of windows in the list
|
||
*
|
||
* Comments
|
||
* --------
|
||
* The list should be freed when done with XtFree().
|
||
*
|
||
*************************************<->***********************************/
|
||
|
||
static void
|
||
ListTransientSubtree (
|
||
ClientData *pCD,
|
||
Window **ppWins,
|
||
int *pSize,
|
||
int *pCount)
|
||
{
|
||
/*
|
||
* Check size
|
||
*/
|
||
if (*pCount == *pSize)
|
||
{
|
||
*pSize += LTT_INCREMENT;
|
||
*ppWins = (Window *)
|
||
XtRealloc ((char *) *ppWins, (*pSize * sizeof(Window)));
|
||
}
|
||
/*
|
||
* Add this window to the list
|
||
*/
|
||
(*ppWins)[*pCount] = pCD->client;
|
||
*pCount += 1;
|
||
|
||
/*
|
||
* Add siblings
|
||
*/
|
||
if (pCD->transientSiblings)
|
||
ListTransientSubtree (pCD->transientSiblings, ppWins, pSize, pCount);
|
||
|
||
/*
|
||
* Add children
|
||
*/
|
||
if (pCD->transientChildren)
|
||
ListTransientSubtree (pCD->transientChildren, ppWins, pSize, pCount);
|
||
|
||
|
||
} /* END OF FUNCTION ListTransientSubtree */
|
||
|
||
|
||
|
||
/*************************************<->*************************************
|
||
*
|
||
* ListTransientTree (pCD)
|
||
*
|
||
*
|
||
* Description:
|
||
* -----------
|
||
* This function returns the list of windows in a transient window
|
||
* tree.
|
||
*
|
||
*
|
||
* Inputs:
|
||
* ------
|
||
* pCD - pointer to client data of a primary window.
|
||
*
|
||
* Outputs:
|
||
* -------
|
||
* none
|
||
*
|
||
* Comments
|
||
* --------
|
||
* The list should be freed when done with XtFree().
|
||
*
|
||
*************************************<->***********************************/
|
||
|
||
static Window *
|
||
ListTransientTree (
|
||
ClientData *pCD)
|
||
{
|
||
Window *pWins;
|
||
int count;
|
||
int iSize;
|
||
|
||
/*
|
||
* Initial allocation
|
||
*/
|
||
iSize = LTT_INCREMENT;
|
||
pWins = (Window *) XtMalloc (iSize * sizeof(Window));
|
||
count = 0;
|
||
|
||
/*
|
||
* Add this window to the list
|
||
*/
|
||
ListTransientSubtree (pCD, &pWins, &iSize, &count);
|
||
|
||
/*
|
||
* Add terminator to end of window list
|
||
*/
|
||
if (count == iSize)
|
||
{
|
||
iSize += LTT_INCREMENT;
|
||
pWins = (Window *)
|
||
XtRealloc ((char *)pWins, (iSize * sizeof(Window)));
|
||
}
|
||
pWins[count++] = None;
|
||
|
||
/*
|
||
* Return the list of windows found
|
||
*/
|
||
return (pWins);
|
||
|
||
} /* END OF FUNCTION ListTransientTree */
|
||
|
||
|
||
/*************************************<->*************************************
|
||
*
|
||
* ReManageWindow (pCD)
|
||
*
|
||
*
|
||
* Description:
|
||
* -----------
|
||
* This function unmanages and remanages a window and it's associated
|
||
* transients.
|
||
*
|
||
*
|
||
* Inputs:
|
||
* ------
|
||
* pCD - pointer to client data of a primary window.
|
||
*
|
||
* Outputs:
|
||
* -------
|
||
* none
|
||
*
|
||
* Comments
|
||
* --------
|
||
* The pointer pCD is invalid after calling this function -- a
|
||
* side-effect of unmanaging the client before remanaging it.
|
||
*
|
||
*************************************<->***********************************/
|
||
|
||
static void
|
||
ReManageWindow (
|
||
ClientData *pCD)
|
||
{
|
||
long manageFlags = MANAGEW_NORMAL;
|
||
Window *pWins, *pW;
|
||
WmScreenData *pSD;
|
||
|
||
/*
|
||
* Get the list of windows in the transient window tree.
|
||
*/
|
||
pWins = ListTransientTree (pCD);
|
||
|
||
pSD = pCD->pSD;
|
||
|
||
/*
|
||
* Unmanage this window and associated transients
|
||
*/
|
||
UnManageWindow (pCD);
|
||
|
||
/*** pCD is no longer a valid pointer!!! ***/
|
||
|
||
/*
|
||
* Remanage this window and its secondaries
|
||
*/
|
||
pW = pWins;
|
||
while (*pW != None)
|
||
{
|
||
ManageWindow (pSD, *pW, manageFlags);
|
||
pW++;
|
||
}
|
||
|
||
XtFree ((char *) pWins);
|
||
|
||
} /* END OF FUNCTION ReManageWindow */
|
||
|
||
|
||
|
||
/*************************************<->*************************************
|
||
*
|
||
* ScanForEmbeddedClients (pSD)
|
||
*
|
||
*
|
||
* Description:
|
||
* -----------
|
||
* This function scans the managed windows and identifies those that
|
||
* should be embedded clients in the front panel
|
||
*
|
||
*
|
||
* Inputs:
|
||
* ------
|
||
* pSD - pointer to screen data.
|
||
*
|
||
* Outputs:
|
||
* -------
|
||
*
|
||
*************************************<->***********************************/
|
||
|
||
void
|
||
ScanForEmbeddedClients (
|
||
WmScreenData *pSD)
|
||
{
|
||
ClientData *pCD;
|
||
ClientListEntry *pCLE;
|
||
WmFpEmbeddedClientData *pECD;
|
||
Boolean bReset;
|
||
|
||
/*
|
||
* Search through all the windows we're managing right now to
|
||
* see if any should be embedded in a front/sub panel.
|
||
*/
|
||
pCLE = pSD->clientList;
|
||
bReset = False;
|
||
|
||
while (pCLE != NULL)
|
||
{
|
||
/*
|
||
* See if this is an previously unrecognized embedded client
|
||
*/
|
||
pCD = pCLE->pCD;
|
||
|
||
if ((pCD->pECD == NULL ) && IsEmbeddedClient (pCD, &pECD))
|
||
{
|
||
/*
|
||
* Remanage this window and associated transients
|
||
*/
|
||
ReManageWindow (pCD);
|
||
/*
|
||
* At this point pCD is no longer valid and the
|
||
* pSD->clientList has been changed.
|
||
*/
|
||
bReset = True;
|
||
}
|
||
|
||
/*
|
||
* Test for exit condition
|
||
*/
|
||
if (pCLE == pSD->lastClient)
|
||
{
|
||
/*
|
||
* Gone all the way through the list without finding
|
||
* anything -- time to quit
|
||
*/
|
||
break;
|
||
}
|
||
else if (bReset)
|
||
{
|
||
/*
|
||
* Remanaging a client restructures the client list.
|
||
* Start over at the beginning.
|
||
*/
|
||
bReset = False;
|
||
pCLE = pSD->clientList;
|
||
}
|
||
else
|
||
{
|
||
/*
|
||
* Move to next client.
|
||
*/
|
||
pCLE = pCLE->nextSibling;
|
||
}
|
||
}
|
||
|
||
} /* END OF FUNCTION ScanForEmbeddedClients */
|
||
|
||
|
||
/*************************************<->*************************************
|
||
*
|
||
* IsEmbeddedClient (pCD, ppECD)
|
||
*
|
||
*
|
||
* Description:
|
||
* -----------
|
||
* This function tests a a client to see if it should be embedded
|
||
* in the front panel.
|
||
*
|
||
*
|
||
* Inputs:
|
||
* ------
|
||
* pCD = ptr to Client Data
|
||
* ppECD = ptr to returned embedded client data ptr
|
||
*
|
||
*
|
||
* Outputs:
|
||
* -------
|
||
* *ppECD = ptr to embedded client data
|
||
*
|
||
*************************************<->***********************************/
|
||
|
||
Boolean
|
||
|
||
IsEmbeddedClient (ClientData *pCD, WmFpEmbeddedClientData **ppECD)
|
||
|
||
{
|
||
WmScreenData *pSD;
|
||
int i;
|
||
Boolean bFoundMatch = False;
|
||
WmFpEmbeddedClientData *pECD;
|
||
|
||
pSD = pCD->pSD;
|
||
pECD = (WmFpEmbeddedClientData *) pSD->pECD;
|
||
|
||
for (i = 0; i < pSD->numEmbeddedClients && !bFoundMatch; i++, pECD++)
|
||
{
|
||
/*
|
||
* It's an embedded client if
|
||
* the resource name matches a slot and
|
||
* it's not a subpanel and
|
||
* the slot isn't already filled.
|
||
*/
|
||
if ((!strcmp ((char *)pCD->clientName,
|
||
(char *)(pECD->pchResName))) &&
|
||
(!(pCD->dtwmBehaviors & DtWM_BEHAVIOR_SUBPANEL)) &&
|
||
(!pECD->pCD))
|
||
{
|
||
*ppECD = pECD;
|
||
bFoundMatch = True;
|
||
}
|
||
}
|
||
return (bFoundMatch);
|
||
|
||
} /* END OF FUNCTION IsEmbeddedClient */
|
||
|
||
|
||
/******************************<->*************************************
|
||
*
|
||
* ManageEmbeddedClient (pCD, pECD, manageFlags)
|
||
*
|
||
*
|
||
* Description:
|
||
* -----------
|
||
* This is the function that is used to setup a client window
|
||
* in the front panel.
|
||
*
|
||
* Inputs:
|
||
* ------
|
||
* pCD = initialized client data, including window of client that
|
||
* we want to manage.
|
||
* pECD = ptr to embedded client entry for this client
|
||
*
|
||
* manageFlags = additional control information
|
||
*
|
||
* Outputs:
|
||
* -------
|
||
* Returns False if normal client processing needs to be done.
|
||
*
|
||
* Returns True if this client has been embedded directly in the
|
||
* front panel and is NOT to be managed as a normal top level
|
||
* window--no further processing required.
|
||
******************************<->***********************************/
|
||
Boolean
|
||
|
||
ManageEmbeddedClient (
|
||
ClientData *pCD,
|
||
WmFpEmbeddedClientData *pECD,
|
||
long manageFlags)
|
||
|
||
{
|
||
XWindowChanges windowChanges;
|
||
unsigned int mask;
|
||
WmFpPushRecallClientData *pPRCD;
|
||
|
||
if (!pECD || !pCD)
|
||
{
|
||
return (False);
|
||
}
|
||
|
||
/*
|
||
* Add to all workspaces
|
||
*/
|
||
pCD->dtwmFunctions |= DtWM_FUNCTION_OCCUPY_WS;
|
||
|
||
F_AddToAllWorkspaces(0, pCD, 0);
|
||
|
||
pCD->dtwmFunctions &= ~DtWM_FUNCTION_OCCUPY_WS;
|
||
|
||
/*
|
||
* Set client list entries
|
||
* (in a list by itself)
|
||
*/
|
||
pCD->clientEntry.type = NORMAL_STATE;
|
||
pCD->clientEntry.pCD = pCD;
|
||
pCD->clientEntry.nextSibling = NULL;
|
||
pCD->clientEntry.prevSibling = NULL;
|
||
|
||
pCD->iconEntry.type = MINIMIZED_STATE;
|
||
pCD->iconEntry.pCD = pCD;
|
||
pCD->iconEntry.nextSibling = NULL;
|
||
pCD->iconEntry.prevSibling = NULL;
|
||
|
||
/*
|
||
* Save context for event processing.
|
||
*
|
||
*/
|
||
|
||
XSaveContext (DISPLAY, pCD->client, wmGD.windowContextType,
|
||
(caddr_t)pCD);
|
||
|
||
if (!(pCD->clientFlags & CLIENT_WM_CLIENTS))
|
||
{
|
||
XChangeSaveSet (DISPLAY, pCD->client, SetModeInsert);
|
||
XChangeSaveSet (DISPLAY1, pCD->client, SetModeInsert);
|
||
pCD->clientFlags |= CLIENT_IN_SAVE_SET;
|
||
}
|
||
if (!(manageFlags & MANAGEW_WM_CLIENTS))
|
||
{
|
||
XSync (DISPLAY1, False);
|
||
|
||
if (pCD->clientFlags & CLIENT_DESTROYED)
|
||
{
|
||
UnManageWindow (pCD);
|
||
return (True);
|
||
}
|
||
}
|
||
|
||
XtAddEventHandler(XtWindowToWidget (DISPLAY1, pECD->winParent),
|
||
(SubstructureRedirectMask | SubstructureNotifyMask),
|
||
False,
|
||
(XtEventHandler)HandleSubstructEvents,
|
||
(XtPointer)(pCD));
|
||
|
||
/*
|
||
* Fill in more client data
|
||
*/
|
||
pCD->clientX = pECD->x;
|
||
pCD->clientY = pECD->y;
|
||
pCD->clientWidth = pECD->width;
|
||
pCD->clientHeight = pECD->height;
|
||
|
||
pCD->clientFrameWin = 0;
|
||
pCD->clientBaseWin = pECD->winParent;
|
||
|
||
pECD->pCD = pCD;
|
||
pCD->pECD = (void *) pECD;
|
||
|
||
SetClientWsIndex(pCD);
|
||
SetClientWMState (pCD, NormalState, NORMAL_STATE);
|
||
|
||
/*
|
||
* Set state on subpanel in case it never gets mapped
|
||
* to prevent session manager from finding an embedded
|
||
* client on its own.
|
||
*/
|
||
ForceSubpanelWMState (pECD->winParent);
|
||
|
||
XReparentWindow (DISPLAY1, pCD->client,
|
||
pECD->winParent,
|
||
pECD->x, pECD->y);
|
||
pCD->clientFlags |= CLIENT_REPARENTED;
|
||
|
||
windowChanges.width = pECD->width;
|
||
windowChanges.height = pECD->height;
|
||
windowChanges.border_width = 0;
|
||
mask = (CWWidth | CWHeight | CWBorderWidth);
|
||
|
||
XConfigureWindow (DISPLAY1, pCD->client, mask, &windowChanges);
|
||
|
||
XMapWindow (DISPLAY1, pCD->client);
|
||
if (pCD->iconWindow)
|
||
{
|
||
XUnmapWindow (DISPLAY, pCD->iconWindow);
|
||
}
|
||
|
||
UpdateEmbeddedClientsProperty (pCD->pSD);
|
||
|
||
SendConfigureNotify (pCD);
|
||
|
||
if (IsPushRecallClient (pCD, &pPRCD))
|
||
{
|
||
/*
|
||
* There should only be one instance of this
|
||
* client started from a front panel button.
|
||
*/
|
||
pPRCD->pCD = pCD;
|
||
pCD->pPRCD = (void *) pPRCD;
|
||
}
|
||
|
||
WmStopWaiting();
|
||
|
||
return(True); /* successful embedation */
|
||
|
||
} /* END OF FUNCTION ManageEmbeddedClient */
|
||
|
||
|
||
/******************************<->*************************************
|
||
*
|
||
* ReparentEmbeddedClient (pECD, newControl, newWin, x, y, width, height)
|
||
*
|
||
*
|
||
* Description:
|
||
* -----------
|
||
*
|
||
* Inputs:
|
||
* ------
|
||
* pECD = ptr to embedded client entry for this client
|
||
* newControl = widget for new "parent" widget
|
||
* newWin = window ID of window that this embedded client will be
|
||
* a parent of. This is needed in case the control is a
|
||
* gadget.
|
||
* x = x-coord position within newWin where UL corner of the embedded
|
||
* client will go.
|
||
* y = y-coord as described above.
|
||
* width = desired width of embedded client in this new location
|
||
* height = desired height as above.
|
||
*
|
||
*
|
||
* Outputs:
|
||
* -------
|
||
* Returns False if embedded client was is not moved to the new
|
||
* location.
|
||
*
|
||
* Returns True if this client has been reparented to the new
|
||
* control.
|
||
******************************<->***********************************/
|
||
Boolean
|
||
|
||
ReparentEmbeddedClient (
|
||
WmFpEmbeddedClientData *pECD,
|
||
Widget newControl,
|
||
Window newWin,
|
||
int x,
|
||
int y,
|
||
unsigned int width,
|
||
unsigned int height
|
||
)
|
||
|
||
{
|
||
XWindowChanges windowChanges;
|
||
unsigned int mask;
|
||
ClientData *pCD;
|
||
|
||
/*
|
||
* If we have bogus data or if we're asked
|
||
* to reparent to our current parent, then just
|
||
* say no.
|
||
*/
|
||
if ((pECD == NULL) ||
|
||
(pECD->pCD == NULL) ||
|
||
(pECD->winParent == newWin))
|
||
{
|
||
return (False);
|
||
}
|
||
pCD=pECD->pCD;
|
||
|
||
/*
|
||
* Need event handler on new parent window?
|
||
*/
|
||
if (newWin != pECD->winParent)
|
||
{
|
||
XtRemoveEventHandler(XtWindowToWidget (DISPLAY1, pECD->winParent),
|
||
(SubstructureRedirectMask | SubstructureNotifyMask),
|
||
False,
|
||
(XtEventHandler)HandleSubstructEvents,
|
||
(XtPointer)(pCD));
|
||
|
||
XtAddEventHandler(XtWindowToWidget (DISPLAY1, newWin),
|
||
(SubstructureRedirectMask | SubstructureNotifyMask),
|
||
False,
|
||
(XtEventHandler)HandleSubstructEvents,
|
||
(XtPointer)(pCD));
|
||
}
|
||
|
||
/*
|
||
* Update embedding and client data
|
||
*/
|
||
pECD->wControl = newControl;
|
||
pECD->winParent = newWin;
|
||
pCD->clientX = pECD->x = x;
|
||
pCD->clientY = pECD->y = y;
|
||
pCD->clientWidth = pECD->width = width;
|
||
pCD->clientHeight = pECD->height = height;
|
||
pCD->clientBaseWin = pECD->winParent;
|
||
|
||
/*
|
||
* Set state on subpanel in case it never gets mapped
|
||
* to prevent session manager from finding an embedded
|
||
* client on its own.
|
||
*/
|
||
ForceSubpanelWMState (pECD->winParent);
|
||
|
||
/*
|
||
* Do the actual reparent
|
||
*/
|
||
XReparentWindow (DISPLAY1, pCD->client,
|
||
pECD->winParent,
|
||
pECD->x, pECD->y);
|
||
|
||
/*
|
||
* Configure the embedded client
|
||
*/
|
||
windowChanges.width = pECD->width;
|
||
windowChanges.height = pECD->height;
|
||
windowChanges.border_width = 0;
|
||
mask = (CWWidth | CWHeight | CWBorderWidth);
|
||
|
||
XConfigureWindow (DISPLAY1, pCD->client, mask, &windowChanges);
|
||
|
||
XMapWindow (DISPLAY1, pCD->client);
|
||
|
||
UpdateEmbeddedClientsProperty (pCD->pSD);
|
||
|
||
SendConfigureNotify (pCD);
|
||
|
||
return(True); /* successful reparent */
|
||
|
||
} /* END OF FUNCTION ReparentEmbeddedClient */
|
||
|
||
|
||
/*************************************<->*************************************
|
||
*
|
||
* ForceSubpanelWMState (win)
|
||
*
|
||
*
|
||
* Description:
|
||
* -----------
|
||
* This function forces a WM_STATE property on a subpanel window
|
||
* so that the session manager doesn't save multiple copies
|
||
* of embedded clients for subpanels that never get mapped.
|
||
*
|
||
*
|
||
* Inputs:
|
||
* ------
|
||
* win = window ID of a subpanel window (not necessarily the top level!)
|
||
*
|
||
*
|
||
* Outputs:
|
||
* -------
|
||
*
|
||
*************************************<->***********************************/
|
||
|
||
static void
|
||
ForceSubpanelWMState (Window win)
|
||
{
|
||
ClientData *pCD = NULL;
|
||
Window root, parent;
|
||
Window *children = NULL;
|
||
unsigned int numChildren;
|
||
PropWMState *wmStateProp;
|
||
Boolean bDone = False;
|
||
|
||
while (!bDone)
|
||
{
|
||
if (!XQueryTree (DISPLAY, win, &root, &parent,
|
||
&children, &numChildren))
|
||
{
|
||
break;
|
||
}
|
||
|
||
if (!XFindContext(DISPLAY, win, wmGD.windowContextType,
|
||
(caddr_t *)&pCD))
|
||
{
|
||
/*
|
||
* Only continue if we're not already managing this subpanel.
|
||
*/
|
||
bDone = True;
|
||
}
|
||
else if (parent == root)
|
||
{
|
||
if ((wmStateProp = GetWMState (win)))
|
||
{
|
||
/*
|
||
* Already has a WM_STATE.
|
||
*/
|
||
XFree ((char *)wmStateProp);
|
||
}
|
||
else
|
||
{
|
||
/*
|
||
* Add a dummy WM_STATE to foil the session manager
|
||
* search.
|
||
*/
|
||
SetWMState (win, WITHDRAWN_STATE, 0);
|
||
}
|
||
bDone = True;
|
||
}
|
||
else
|
||
{
|
||
/* continue ascent up to root */
|
||
win = parent;
|
||
}
|
||
|
||
XFree ((char *) children);
|
||
}
|
||
|
||
} /* END OF FUNCTION ForceSubpanelWMState */
|
||
|
||
|
||
/*************************************<->*************************************
|
||
*
|
||
* RegisterPushRecallClients (wPanelist, pPRCD, count)
|
||
*
|
||
*
|
||
* Description:
|
||
* -----------
|
||
* This function registers a list of push_recallclients for the
|
||
* front panel subsystem.
|
||
*
|
||
*
|
||
* Inputs:
|
||
* ------
|
||
* wPanelist = front panel object (widget)
|
||
* pPRCD = pointer to list of data for clients
|
||
* count = number of elements in the list
|
||
*
|
||
*
|
||
* Outputs:
|
||
* -------
|
||
*
|
||
*************************************<->***********************************/
|
||
|
||
void
|
||
RegisterPushRecallClients (
|
||
Widget wPanelist,
|
||
WmFpPushRecallClientList pPRCD,
|
||
int count)
|
||
{
|
||
WmScreenData *pSD = NULL;
|
||
int i;
|
||
|
||
for (i= 0; i < wmGD.numScreens; i++)
|
||
{
|
||
pSD = &(wmGD.Screens[i]);
|
||
|
||
if (pSD->managed)
|
||
{
|
||
if (pSD->wPanelist == wPanelist)
|
||
break;
|
||
}
|
||
}
|
||
|
||
if (i < wmGD.numScreens)
|
||
{
|
||
pSD->pPRCD = (struct _WmFpPushRecallClientData *) pPRCD;
|
||
pSD->numPushRecallClients = count;
|
||
}
|
||
#ifdef DEBUG
|
||
else
|
||
{
|
||
fprintf (stderr, "Couldn't match wPanelist to screen data\n");
|
||
}
|
||
#endif /* DEBUG */
|
||
|
||
for (i = 0; i < pSD->numPushRecallClients ; i++, pPRCD++)
|
||
{
|
||
/*
|
||
* Initialize data in each slot
|
||
*/
|
||
pPRCD->tvTimeout.tv_sec = 0;
|
||
}
|
||
|
||
} /* END OF FUNCTION RegisterPushRecallClients */
|
||
|
||
|
||
/*************************************<->*************************************
|
||
*
|
||
* IsPushRecallClient (pCD, ppPRCD)
|
||
*
|
||
*
|
||
* Description:
|
||
* -----------
|
||
* This function tests a a client to see if it should be embedded
|
||
* in the front panel.
|
||
*
|
||
*
|
||
* Inputs:
|
||
* ------
|
||
* pCD = ptr to Client Data
|
||
* ppPRCD = ptr to returned embedded client data ptr
|
||
*
|
||
*
|
||
* Outputs:
|
||
* -------
|
||
* *ppPRCD = ptr to embedded client data
|
||
*
|
||
*************************************<->***********************************/
|
||
|
||
Boolean
|
||
|
||
IsPushRecallClient (ClientData *pCD, WmFpPushRecallClientData **ppPRCD)
|
||
|
||
{
|
||
WmScreenData *pSD;
|
||
int i;
|
||
Boolean bFoundMatch = False;
|
||
WmFpPushRecallClientData *pPRCD;
|
||
|
||
pSD = pCD->pSD;
|
||
pPRCD = (WmFpPushRecallClientData *) pSD->pPRCD;
|
||
|
||
for (i = 0; i < pSD->numPushRecallClients && !bFoundMatch; i++, pPRCD++)
|
||
{
|
||
/*
|
||
* It's a push_recall client if the resource name matches
|
||
* a slot and the slot isn't already filled.
|
||
*/
|
||
if ((!strcmp ((char *)pCD->clientName,
|
||
(char *)(pPRCD->pchResName))) &&
|
||
(!pPRCD->pCD))
|
||
{
|
||
*ppPRCD = pPRCD;
|
||
bFoundMatch = True;
|
||
}
|
||
}
|
||
return (bFoundMatch);
|
||
|
||
} /* END OF FUNCTION IsPushRecallClient */
|
||
|
||
|
||
/*************************************<->*************************************
|
||
*
|
||
* ScanForPushRecallClients (pSD)
|
||
*
|
||
*
|
||
* Description:
|
||
* -----------
|
||
* This function scans the managed windows and identifies those that
|
||
* should be push recall clients of the front panel
|
||
*
|
||
*
|
||
* Inputs:
|
||
* ------
|
||
* pSD - pointer to screen data.
|
||
*
|
||
* Outputs:
|
||
* -------
|
||
*
|
||
*************************************<->***********************************/
|
||
|
||
void
|
||
ScanForPushRecallClients (
|
||
WmScreenData *pSD)
|
||
{
|
||
ClientData *pCD;
|
||
ClientListEntry *pCLE;
|
||
WmFpPushRecallClientData *pPRCD;
|
||
|
||
/*
|
||
* Search through all the windows we're managing right now
|
||
*/
|
||
pCLE = pSD->clientList;
|
||
|
||
while (pCLE != NULL)
|
||
{
|
||
/*
|
||
* See if this is an previously unrecognized push recall client
|
||
*/
|
||
pCD = pCLE->pCD;
|
||
|
||
if ((pCD->pPRCD == NULL ) && IsPushRecallClient (pCD, &pPRCD))
|
||
{
|
||
CheckPushRecallClient (pCD);
|
||
}
|
||
|
||
/*
|
||
* Test for exit condition
|
||
*/
|
||
if (pCLE == pSD->lastClient)
|
||
{
|
||
/*
|
||
* Gone all the way through the list without finding
|
||
* anything -- time to quit
|
||
*/
|
||
break;
|
||
}
|
||
else
|
||
{
|
||
/*
|
||
* Move to next client.
|
||
*/
|
||
pCLE = pCLE->nextSibling;
|
||
}
|
||
}
|
||
|
||
} /* END OF FUNCTION ScanForPushRecallClients */
|
||
|
||
|
||
/******************************<->*************************************
|
||
*
|
||
* static void CheckPushRecallClient (pCD)
|
||
*
|
||
*
|
||
* Description:
|
||
* -----------
|
||
* Checks a client against the list of push recall clients to see
|
||
* if there are any matches. All matches are marked.
|
||
*
|
||
* Inputs:
|
||
* ------
|
||
* pCD - pointer to the Client Data structure
|
||
*
|
||
* Outputs:
|
||
* -------
|
||
*
|
||
* Comments:
|
||
* --------
|
||
*
|
||
******************************<->***********************************/
|
||
static void
|
||
CheckPushRecallClient(
|
||
ClientData *pCD)
|
||
{
|
||
WmFpPushRecallClientData *pPRCD;
|
||
|
||
while (IsPushRecallClient (pCD, &pPRCD))
|
||
{
|
||
/*
|
||
* There should only be one instance of this
|
||
* client started from a front panel button.
|
||
*/
|
||
pPRCD->pCD = pCD;
|
||
pPRCD->tvTimeout.tv_sec = 0;
|
||
if (!pCD->pPRCD)
|
||
pCD->pPRCD = (void *) pPRCD;
|
||
}
|
||
}
|
||
|
||
|
||
/******************************<->*************************************
|
||
*
|
||
* static void HandleSubstructEvents (Widget w, caddr_t pCD, XEvent *event)
|
||
*
|
||
*
|
||
* Description:
|
||
* -----------
|
||
* Causes death of embedded clients to run through UnManageWindow()
|
||
* for proper cleanup.
|
||
*
|
||
* Note there is one of these event handlers instantiated for
|
||
* each live client window in the front panel. Hence, for each
|
||
* live client window death, each of the event handlers gets called
|
||
* once. We need to ensure that we've got the right pCD before
|
||
* calling UnManageWindow() on it.
|
||
*
|
||
*
|
||
* Inputs:
|
||
* ------
|
||
* w - not used
|
||
* pCD - pointer to the Client Data structure
|
||
* event - we only care about UnMapNotify
|
||
*
|
||
* Outputs:
|
||
* -------
|
||
*
|
||
* Comments:
|
||
* --------
|
||
* This routine is called LOTS of times, for all types of events.
|
||
*
|
||
******************************<->***********************************/
|
||
static void
|
||
HandleSubstructEvents(
|
||
Widget w,
|
||
caddr_t ptr,
|
||
XEvent *event )
|
||
{
|
||
struct _ClientData *pCD = (struct _ClientData *)ptr;
|
||
|
||
switch (event->type)
|
||
{
|
||
case UnmapNotify:
|
||
{
|
||
if (pCD->client == event->xunmap.window)
|
||
{
|
||
UnManageWindow (pCD);
|
||
}
|
||
break;
|
||
}
|
||
}
|
||
} /* END OF FUNCTION HandleSubstructEvents */
|
||
|
||
|
||
/*******************************<->*************************************
|
||
*
|
||
* UpdateEmbeddedClientsProperty (pSD)
|
||
*
|
||
* Description:
|
||
* -----------
|
||
*
|
||
* Inputs:
|
||
* ------
|
||
* pSD - pointer to the screen data
|
||
*
|
||
* Outputs:
|
||
* -------
|
||
* True if successful, False otherwise.
|
||
*
|
||
* Comments:
|
||
* --------
|
||
* The _DT_WORKSPACE_EMBEDDED_CLIENTS property on the
|
||
* root window for the screen will be updated to reflect the
|
||
* current contents of the front panel
|
||
*
|
||
*
|
||
******************************<->***********************************/
|
||
Boolean
|
||
UpdateEmbeddedClientsProperty(
|
||
WmScreenData *pSD )
|
||
{
|
||
unsigned int numClients = 0;
|
||
Window *pClients = NULL;
|
||
Boolean rval = True;
|
||
int i;
|
||
WmFpEmbeddedClientData *pECD;
|
||
|
||
pECD = (WmFpEmbeddedClientData *) pSD->pECD;
|
||
|
||
for (i = 0; i < pSD->numEmbeddedClients; i++, pECD++)
|
||
{
|
||
if (pECD->pCD)
|
||
{
|
||
numClients += 1;
|
||
|
||
if (!pClients)
|
||
{
|
||
pClients = (Window *) XtMalloc (sizeof(Window));
|
||
}
|
||
else
|
||
{
|
||
pClients = (Window *) XtRealloc ((char *)pClients,
|
||
(numClients * (sizeof(Window))));
|
||
}
|
||
|
||
if (!pClients)
|
||
{
|
||
Warning (
|
||
((char *)GETMESSAGE(4, 17, "Insufficient memory to write _DT_WORKSPACE_EMBEDDED_CLIENTS."))
|
||
);
|
||
rval = False;
|
||
break;
|
||
}
|
||
else
|
||
{
|
||
pClients[numClients-1] = pECD->pCD->client;
|
||
}
|
||
}
|
||
}
|
||
|
||
SetEmbeddedClientsProperty (pSD->rootWindow, pClients,
|
||
numClients);
|
||
|
||
if (pClients != NULL)
|
||
{
|
||
XtFree ((char *)pClients);
|
||
}
|
||
|
||
return (rval);
|
||
} /* END OF FUNCTION UpdateEmbeddedClientsProperty */
|
||
|
||
|
||
|
||
/*******************************<->*************************************
|
||
*
|
||
* UnParentControls (pSD, unmap)
|
||
*
|
||
* Description:
|
||
* -----------
|
||
*
|
||
* Inputs:
|
||
* ------
|
||
* pSD - pointer to the screen data
|
||
* unmap - if True, then unmap the windows after reparenting to root
|
||
*
|
||
* Outputs:
|
||
* -------
|
||
* none
|
||
*
|
||
* Comments:
|
||
* --------
|
||
* Reparents clients embedded in the front panel back to the
|
||
* root window
|
||
*
|
||
******************************<->***********************************/
|
||
void
|
||
UnParentControls(
|
||
WmScreenData *pSD,
|
||
Boolean unmap )
|
||
{
|
||
int i;
|
||
ClientData *pCD;
|
||
WmFpEmbeddedClientData *pECD;
|
||
|
||
if (pSD && pSD->managed)
|
||
{
|
||
pECD = (WmFpEmbeddedClientData *) pSD->pECD;
|
||
for (i = 0; i < pSD->numEmbeddedClients; i++, pECD++)
|
||
{
|
||
pCD = pECD->pCD;
|
||
|
||
if (pCD)
|
||
{
|
||
if ((pCD->clientFlags & CLIENT_IN_SAVE_SET) &&
|
||
!(pCD->clientFlags & CLIENT_DESTROYED))
|
||
{
|
||
XRemoveFromSaveSet (DISPLAY, pCD->client);
|
||
XRemoveFromSaveSet (DISPLAY1, pCD->client);
|
||
}
|
||
|
||
XReparentWindow (DISPLAY,
|
||
pCD->client,
|
||
pSD->rootWindow,
|
||
pCD->clientX,
|
||
pCD->clientY);
|
||
if (unmap)
|
||
{
|
||
XUnmapWindow (DISPLAY, pCD->client);
|
||
if (pCD->iconWindow)
|
||
{
|
||
if (pCD->clientFlags & ICON_IN_SAVE_SET)
|
||
{
|
||
XRemoveFromSaveSet (DISPLAY, pCD->iconWindow);
|
||
pCD->clientFlags &= ~ICON_IN_SAVE_SET;
|
||
}
|
||
XUnmapWindow (DISPLAY, pCD->iconWindow);
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
} /* END OF FUNCTION UnParentControl */
|
||
|
||
|
||
|
||
/*************************************<->*************************************
|
||
*
|
||
* RegisterIconBoxControl (wPanelist)
|
||
*
|
||
*
|
||
* Description:
|
||
* -----------
|
||
* This function registers the icon box control in a front panel
|
||
*
|
||
*
|
||
* Inputs:
|
||
* ------
|
||
* wPanelist = front panel object (widget)
|
||
*
|
||
* Outputs:
|
||
* -------
|
||
*
|
||
*************************************<->***********************************/
|
||
|
||
void
|
||
RegisterIconBoxControl (Widget wPanelist)
|
||
{
|
||
WmScreenData *pSD;
|
||
int i;
|
||
|
||
for (i= 0; i < wmGD.numScreens; i++)
|
||
{
|
||
pSD = &(wmGD.Screens[i]);
|
||
|
||
if (pSD->managed)
|
||
{
|
||
if (pSD->wPanelist == wPanelist)
|
||
break;
|
||
}
|
||
}
|
||
|
||
if (i < wmGD.numScreens)
|
||
{
|
||
pSD->iconBoxControl = True;
|
||
pSD->useIconBox = True;
|
||
}
|
||
#ifdef DEBUG
|
||
else
|
||
{
|
||
fprintf (stderr, "Couldn't match wPanelist to screen data\n");
|
||
}
|
||
#endif /* DEBUG */
|
||
|
||
} /* END OF FUNCTION RegisterIconBoxControl */
|
||
|
||
|