void SetClientStateWithEventMask (ClientData *pCD, int newState, Time setTime, unsigned int event_mask) { ClientData *pcdLeader; int currentState; WmScreenData *pSD = PSD_FOR_CLIENT(pCD); #ifdef WSM Boolean notShowing = (newState & UNSEEN_STATE); #endif /* WSM */ currentState = pCD->clientState; if (currentState == newState) { /* no change in state */ return; } /* * Undo the old state and setup the new state. If this is a transient * window then insure that it is put in a state that is compatible * with its transient leader (e.g., it cannot be minimized separately). */ pcdLeader = (pCD->transientLeader) ? FindTransientTreeLeader (pCD) : pCD; #ifdef WSM SetClientWsIndex (pCD); #endif /* WSM */ if (pCD->transientLeader) { if ((pcdLeader->clientState == MINIMIZED_STATE) && (newState != WITHDRAWN_STATE)) { newState = MINIMIZED_STATE; #ifdef WSM if (notShowing) { newState |= UNSEEN_STATE; } #endif /* WSM */ } else if ((newState == MINIMIZED_STATE) && (pcdLeader->clientState != MINIMIZED_STATE)) { if (currentState == WITHDRAWN_STATE) { newState = NORMAL_STATE; #ifdef WSM if (notShowing) { newState |= UNSEEN_STATE; } #endif /* WSM */ } else { newState = currentState; #ifdef WSM if (notShowing) { newState |= UNSEEN_STATE; } #endif /* WSM */ } } if (newState == currentState) { return; } } switch (newState) { #ifdef WSM case UNSEEN_STATE | WITHDRAWN_STATE: #else case WITHDRAWN_STATE: #endif /* WSM */ { /* * Free window manager resources (frame and icon). The * WM_STATE property is set in WithdrawWindow. */ UnManageWindow (pCD); break; } case NORMAL_STATE: case MAXIMIZED_STATE: { SetupWindowStateWithEventMask (pCD, newState, setTime, event_mask); #ifdef WSM XMapWindow (DISPLAY, pCD->client); XMapWindow (DISPLAY, pCD->clientFrameWin); #if defined(PANELIST) WmStopWaiting(); /* in WmIPC.c */ #endif /* PANELIST */ #endif /* WSM */ break; } case MINIMIZED_STATE: { Boolean clientHasFocus; /* * Transient windows are minimized with the rest of the transient * tree, including the transient leader. */ if ((pCD->clientState == NORMAL_STATE) || (pCD->clientState == MAXIMIZED_STATE)) { if ((wmGD.keyboardFocus == pCD) || (pCD->transientChildren && wmGD.keyboardFocus && (pCD == FindTransientTreeLeader (wmGD.keyboardFocus)))) { clientHasFocus = True; } else { clientHasFocus = False; } if (clientHasFocus || ((wmGD.nextKeyboardFocus == pCD) || (pCD->transientChildren && wmGD.keyboardFocus && (pCD == FindTransientTreeLeader (wmGD.nextKeyboardFocus))))) { /* * Give up the keyboard focus when minimized (including * the case in which an associated transient window has * the focus). Immediately remove the focus indication * from the window being minimized. */ if (wmGD.autoKeyFocus && (wmGD.keyboardFocusPolicy == KEYBOARD_FOCUS_EXPLICIT)) { AutoResetKeyFocus (pcdLeader, setTime); } else { Do_Focus_Key (NULL, setTime, ALWAYS_SET_FOCUS | WORKSPACE_IF_NULL); } if (clientHasFocus) { SetKeyboardFocus (NULL, 0); } } /* unmap main client and all transients */ UnmapClients (pCD, event_mask); } /* * Display the icon for the minimized client. */ if (ICON_FRAME_WIN(pCD)) { #ifdef WSM if (pCD->clientState & UNSEEN_STATE) { if (pCD->iconWindow) { XMapWindow (DISPLAY, pCD->iconWindow); } XMapWindow (DISPLAY, ICON_FRAME_WIN(pCD)); } ShowAllIconsForMinimizedClient (pCD); #else /* WSM */ ShowIconForMinimizedClient (pSD->pActiveWS, pCD); #endif /* WSM */ } SetClientWMState (pCD, IconicState, MINIMIZED_STATE); if ((pSD->useIconBox) && P_ICON_BOX(pCD)) { if ((pCD->clientFlags & ICON_BOX) && ACTIVE_ICON_TEXT_WIN) { /* * Hide active icon text window and reparent it to * root */ HideActiveIconText((WmScreenData *)NULL); pSD->activeLabelParent = ACTIVE_ROOT; XReparentWindow(DISPLAY, ACTIVE_ICON_TEXT_WIN , ACTIVE_ROOT, 0, 0 ); } if (ICON_FRAME_WIN(pCD)) { /* * force icon appearance in icon box to change */ IconExposureProc (pCD, True); } } break; } #ifdef WSM case UNSEEN_STATE | NORMAL_STATE: case UNSEEN_STATE | MAXIMIZED_STATE: case UNSEEN_STATE | MINIMIZED_STATE: { if (wmGD.keyboardFocus == pCD) { /* * Give up the keyboard focus */ Do_Focus_Key ((ClientData *)NULL, CurrentTime, ALWAYS_SET_FOCUS); SetKeyboardFocus (NULL, 0); } if (!(pCD->clientState & UNSEEN_STATE) && (((pCD->clientState & ~UNSEEN_STATE) == NORMAL_STATE) || ((pCD->clientState & ~UNSEEN_STATE) == MAXIMIZED_STATE))) { /* unmap main client and all transients */ UnmapClients (pcdLeader, event_mask); } if (pCD->clientFrameWin) { if (!P_ICON_BOX(pCD)) { if (ICON_FRAME_WIN(pCD)) { XUnmapWindow (DISPLAY, ICON_FRAME_WIN(pCD)); } if (pCD->iconWindow) XUnmapWindow (DISPLAY, pCD->iconWindow); } } switch (newState & ~UNSEEN_STATE) { case MINIMIZED_STATE: SetClientWMState (pCD, IconicState, newState); break; case NORMAL_STATE: case MAXIMIZED_STATE: default: SetClientWMState (pCD, NormalState, newState); break; } } break; #endif /* WSM */ } } /* END OF FUNCTION SetClientStateWithEventMask */
static void SetupWindowStateWithEventMask (ClientData *pCD, int newState, Time setTime, unsigned int event_mask) { int currentState; #ifdef WSM int wsI, iplace; WmWorkspaceData *pWS_i; #else /* WSM */ WmWorkspaceData *pWS = PSD_FOR_CLIENT(pCD)->pActiveWS; #endif /* WSM */ WmScreenData *pSD = PSD_FOR_CLIENT(pCD); currentState = pCD->clientState; /* * A transient window is not restored or maximized if the transient leader * is minimized. */ if (newState == NORMAL_STATE) { if (pCD->maxConfig == True) { /* * The configuration function uses maxConfig to determine * what the current configuration is (and then resets * maxConfig) and uses the state paramenter to determine * what the new configuration is. */ ConfigureNewState (pCD); } } else /* MAXIMIZED_STATE */ { if (pCD->maxConfig == False) { ConfigureNewState (pCD); } } if (currentState == MINIMIZED_STATE) { Boolean clearIconFocus; /* * give up keyboard focus */ if ((wmGD.keyboardFocus == pCD) || (wmGD.nextKeyboardFocus == pCD)) { Do_Focus_Key (NULL, setTime, ALWAYS_SET_FOCUS | WORKSPACE_IF_NULL); } if (wmGD.keyboardFocus == pCD) { clearIconFocus = True; } else { clearIconFocus = False; } /* * The wm icon frame window and the client icon window * (if it is being used) are mapped and the client window and * client frame are unmapped. */ if (ICON_FRAME_WIN(pCD)) { if (pSD->useIconBox && P_ICON_BOX(pCD) && !(pCD->clientFlags & ICON_BOX)) { ShowClientIconState(pCD, newState); } else { Boolean doGrab = False; if (event_mask) doGrab = (Success == XGrabPointer (DISPLAY, DefaultRootWindow(DISPLAY), False, event_mask, GrabModeAsync, GrabModeAsync, None, None, CurrentTime)); XUnmapWindow (DISPLAY, ICON_FRAME_WIN(pCD)); if (pCD->iconWindow) { XUnmapWindow (DISPLAY, pCD->iconWindow); } if (event_mask && doGrab) { XEvent event; XMaskEvent(DISPLAY, event_mask, &event); XUngrabPointer(DISPLAY,CurrentTime); } #ifdef WSM if (wmGD.iconAutoPlace) { for (wsI = 0; wsI < pCD->numInhabited; wsI++) { iplace = pCD->pWsList[wsI].iconPlace; if (iplace != NO_ICON_PLACE) { pWS_i = GetWorkspaceData (pCD->pSD, pCD->pWsList[wsI].wsID); pWS_i->IPData.placeList[iplace].pCD = NULL; } } } #else /* WSM */ if ((wmGD.iconAutoPlace) && (ICON_PLACE(pCD) != NO_ICON_PLACE)) { pWS->IPData.placeList[ICON_PLACE(pCD)].pCD = NULL; } #endif /* WSM */ } if (clearIconFocus) { ClearFocusIndication (pCD, False /*no refresh*/); wmGD.keyboardFocus = NULL; } } } if ((currentState != NORMAL_STATE) && (currentState != MAXIMIZED_STATE)) { /* * Note that maximized state is considered a NormalState in * the ICCC. SetClientWMState also sets the state in the * client data. */ if (currentState == MINIMIZED_STATE) { /* * Raise the window(s) when they are deiconified. */ pCD->clientState = newState; #ifdef WSM wmGD.bSuspendSecondaryRestack = True; #endif /* WSM */ F_Raise (NULL, pCD, NULL); #ifdef WSM wmGD.bSuspendSecondaryRestack = False; #endif /* WSM */ } if ( (!(pCD->clientFlags & ICON_BOX)) || ((pCD->clientFlags & ICON_BOX) && (!(firstTime))) ) { #ifdef PANELIST if ((currentState == WITHDRAWN_STATE) && (pCD->dtwmBehaviors & DtWM_BEHAVIOR_SUBPANEL) && !(pCD->transientChildren)) { if (pCD->dtwmBehaviors & DtWM_BEHAVIOR_SUB_RESTORED) { pCD->dtwmBehaviors &= ~DtWM_BEHAVIOR_SUB_RESTORED; pCD->dtwmBehaviors &= ~DtWM_BEHAVIOR_SUBPANEL; XMapWindow (DISPLAY, pCD->client); XMapWindow (DISPLAY, pCD->clientFrameWin); } else { SlideWindowOut (pCD); } } else #endif /* PANELIST */ MapClientWindows (pCD); } /* * Set the WM_STATE property of the window and any associated * transients, along with the clientState value. The call * is made with an indication of NORMAL_STATE to insure * that transient window clientState values are setup * correctly. The top-level window clientState is set later. */ SetClientWMState (pCD, NormalState, NORMAL_STATE); } pCD->clientState = newState; if ((wmGD.keyboardFocusPolicy == KEYBOARD_FOCUS_EXPLICIT) && (currentState == MINIMIZED_STATE) && wmGD.deiconifyKeyFocus) { ClientData *pcdFocus; pcdFocus = FindTransientFocus (pCD); if (pcdFocus) { Do_Focus_Key (pcdFocus, setTime, ALWAYS_SET_FOCUS); } } if ( pSD->useIconBox && P_ICON_BOX(pCD) && (!(pCD->clientFlags & ICON_BOX)) && (ICON_FRAME_WIN(pCD))) { /* * force icon appearance in icon box to change */ IconExposureProc (pCD, True); } } /* END OF FUNCTION SetupWindowStateWithEventMask */
void InitKeyboardFocus (void) { ClientData *pCD; Boolean sameScreen; Boolean focusSet = False; int scr; int junk; Window junk_win, root_returned; int currentX, currentY; /* * Set the keyboard focus based on the keyboard focus policy. */ wmGD.keyboardFocus = NULL; wmGD.nextKeyboardFocus = NULL; for (scr = 0; scr < wmGD.numScreens; scr++) { if (wmGD.Screens[scr].managed) { wmGD.Screens[scr].focusPriority = 0; if (wmGD.keyboardFocusPolicy == KEYBOARD_FOCUS_POINTER) { /* * Set the keyboard focus to the window that * currently contains the pointer. */ pCD = GetClientUnderPointer (&sameScreen); if (wmGD.colormapFocusPolicy == CMAP_FOCUS_POINTER) { /* * Do some colormap installation that has been * deferred from the InitColormapFocus routine. */ SetColormapFocus (ACTIVE_PSD, pCD); } if (pCD) { Do_Focus_Key (pCD, GetTimestamp (), ALWAYS_SET_FOCUS); focusSet = True; } } else { ButtonSpec *buttonSpec; /* * Prepare to do explicit selection button grabs. */ buttonSpec = wmGD.Screens[scr].buttonSpecs; while (buttonSpec) { if ((buttonSpec->button == FOCUS_SELECT_BUTTON) && (buttonSpec->context & F_CONTEXT_WINDOW) && (buttonSpec->subContext & F_SUBCONTEXT_W_CLIENT)) { if (buttonSpec->state == 0) { removeSelectGrab = False; } } buttonSpec = buttonSpec->nextButtonSpec; } } } } if (!focusSet) { /* * This is keyboard focus policy is either "explicit" or it it * "pointer" * and there is no window under the pointer. No window currently has * the keyboard input focus. Set the keyboard focus to the window * manager default (non-client) OR to the last client with focus. * * In Mwm 1.1.4 and later, calling Do_Focus_Key with NULL will try * to find a 'reasonable' window to put focus. This means that on * startup and restarts, a Mwm window will have focus! Yeah! */ /* * Set Active Screen First */ if (XQueryPointer(DISPLAY, DefaultRootWindow(DISPLAY), &root_returned, &junk_win, ¤tX, ¤tY, &junk, &junk, (unsigned int *)&junk)) { for (scr = 0; scr < wmGD.numScreens; scr++) { if (wmGD.Screens[scr].managed && wmGD.Screens[scr].rootWindow == root_returned) { SetActiveScreen(&(wmGD.Screens[scr])); break; } } } Do_Focus_Key ((ClientData *)NULL, CurrentTime, ALWAYS_SET_FOCUS); } } /* END OF FUNCTION InitKeyboardFocus */