Avoid icon change to default on winspector save
[wmaker-crm.git] / src / cycling.c
blobb2c77268de49b502a7c76a123d9ba8d7bf8a4cf5
1 /* cycling.c- window cycling
3 * Window Maker window manager
5 * Copyright (c) 2000-2003 Alfredo K. Kojima
7 * This program is free software; you can redistribute it and/or modify
8 * it under the terms of the GNU General Public License as published by
9 * the Free Software Foundation; either version 2 of the License, or
10 * (at your option) any later version.
12 * This program is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 * GNU General Public License for more details.
17 * You should have received a copy of the GNU General Public License along
18 * with this program; if not, write to the Free Software Foundation, Inc.,
19 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
22 #include "wconfig.h"
24 #include <stdlib.h>
25 #include <X11/Xlib.h>
26 #include <X11/Xutil.h>
27 #include <X11/keysym.h>
29 #include "WindowMaker.h"
30 #include "GNUstep.h"
31 #include "screen.h"
32 #include "window.h"
33 #include "framewin.h"
34 #include "keybind.h"
35 #include "actions.h"
36 #include "stacking.h"
37 #include "funcs.h"
38 #include "xinerama.h"
39 #include "switchpanel.h"
41 /* Globals */
42 extern WPreferences wPreferences;
44 extern WShortKey wKeyBindings[WKBD_LAST];
46 static void raiseWindow(WSwitchPanel * swpanel, WWindow * wwin)
48 Window swwin = wSwitchPanelGetWindow(swpanel);
50 if (wwin->flags.mapped) {
51 if (swwin != None) {
52 Window win[2];
54 win[0] = swwin;
55 win[1] = wwin->frame->core->window;
57 XRestackWindows(dpy, win, 2);
58 } else
59 XRaiseWindow(dpy, wwin->frame->core->window);
63 static WWindow *change_focus_and_raise(WWindow *newFocused, WWindow *oldFocused,
64 WSwitchPanel *swpanel, WScreen *scr, Bool esc_cancel)
66 if (!newFocused)
67 return oldFocused;
69 wWindowFocus(newFocused, oldFocused);
70 oldFocused = newFocused;
72 if (wPreferences.circ_raise) {
73 CommitStacking(scr);
75 if (!esc_cancel)
76 raiseWindow(swpanel, newFocused);
79 return oldFocused;
82 void StartWindozeCycle(WWindow * wwin, XEvent * event, Bool next, Bool class_only)
85 XModifierKeymap *keymap = NULL;
86 WSwitchPanel *swpanel = NULL;
87 WScreen *scr = wScreenForRootWindow(event->xkey.root);
88 KeyCode leftKey = XKeysymToKeycode(dpy, XK_Left);
89 KeyCode rightKey = XKeysymToKeycode(dpy, XK_Right);
90 KeyCode homeKey = XKeysymToKeycode(dpy, XK_Home);
91 KeyCode endKey = XKeysymToKeycode(dpy, XK_End);
92 KeyCode shiftLKey = XKeysymToKeycode(dpy, XK_Shift_L);
93 KeyCode shiftRKey = XKeysymToKeycode(dpy, XK_Shift_R);
94 KeyCode escapeKey = XKeysymToKeycode(dpy, XK_Escape);
95 Bool esc_cancel = False;
96 Bool somethingElse = False;
97 Bool done = False;
98 Bool hasModifier;
99 int modifiers;
100 WWindow *newFocused;
101 WWindow *oldFocused;
102 XEvent ev;
105 if (!wwin)
106 return;
108 if (next)
109 hasModifier = (wKeyBindings[WKBD_FOCUSNEXT].modifier != 0);
110 else
111 hasModifier = (wKeyBindings[WKBD_FOCUSPREV].modifier != 0);
113 if (hasModifier) {
114 keymap = XGetModifierMapping(dpy);
115 XGrabKeyboard(dpy, scr->root_win, False, GrabModeAsync, GrabModeAsync, CurrentTime);
118 scr->flags.doing_alt_tab = 1;
120 swpanel = wInitSwitchPanel(scr, wwin, class_only);
121 oldFocused = wwin;
123 if (swpanel) {
125 if (wwin->flags.mapped)
126 newFocused = wSwitchPanelSelectNext(swpanel, !next, True);
127 else
128 newFocused = wSwitchPanelSelectFirst(swpanel, False);
130 oldFocused = change_focus_and_raise(newFocused, oldFocused, swpanel, scr, False);
131 } else {
132 if (wwin->frame->workspace == scr->current_workspace)
133 newFocused = wwin;
134 else
135 newFocused = NULL;
138 while (hasModifier && !done) {
139 int i;
141 WMMaskEvent(dpy, KeyPressMask | KeyReleaseMask | ExposureMask
142 | PointerMotionMask | ButtonReleaseMask | EnterWindowMask, &ev);
144 /* ignore CapsLock */
145 modifiers = ev.xkey.state & ValidModMask;
147 if (!swpanel)
148 done = True;
150 switch (ev.type) {
152 case KeyPress:
154 if ((wKeyBindings[WKBD_FOCUSNEXT].keycode == ev.xkey.keycode
155 && wKeyBindings[WKBD_FOCUSNEXT].modifier == modifiers)
156 || (wKeyBindings[WKBD_GROUPNEXT].keycode == ev.xkey.keycode
157 && wKeyBindings[WKBD_GROUPNEXT].modifier == modifiers)
158 || ev.xkey.keycode == rightKey) {
160 newFocused = wSwitchPanelSelectNext(swpanel, False, ev.xkey.keycode != rightKey);
161 oldFocused = change_focus_and_raise(newFocused, oldFocused, swpanel, scr, False);
163 } else if ((wKeyBindings[WKBD_FOCUSPREV].keycode == ev.xkey.keycode
164 && wKeyBindings[WKBD_FOCUSPREV].modifier == modifiers)
165 || (wKeyBindings[WKBD_GROUPPREV].keycode == ev.xkey.keycode
166 && wKeyBindings[WKBD_GROUPPREV].modifier == modifiers)
167 || ev.xkey.keycode == leftKey) {
169 newFocused = wSwitchPanelSelectNext(swpanel, True, ev.xkey.keycode != leftKey);
170 oldFocused = change_focus_and_raise(newFocused, oldFocused, swpanel, scr, False);
172 } else if (ev.xkey.keycode == homeKey || ev.xkey.keycode == endKey) {
174 newFocused = wSwitchPanelSelectFirst(swpanel, ev.xkey.keycode != homeKey);
175 oldFocused = change_focus_and_raise(newFocused, oldFocused, swpanel, scr, False);
177 } else if (ev.xkey.keycode == escapeKey) {
179 /* Focus the first window of the swpanel, despite the 'False' */
180 newFocused = wSwitchPanelSelectFirst(swpanel, False);
181 oldFocused = change_focus_and_raise(newFocused, oldFocused, swpanel, scr, True);
182 esc_cancel = True;
183 done = True;
185 } else if (ev.xkey.keycode != shiftLKey && ev.xkey.keycode != shiftRKey) {
187 somethingElse = True;
188 done = True;
190 break;
192 case KeyRelease:
194 for (i = 0; i < 8 * keymap->max_keypermod; i++) {
196 int mask = 1 << (i / keymap->max_keypermod);
198 if (keymap->modifiermap[i] == ev.xkey.keycode &&
199 ((wKeyBindings[WKBD_FOCUSNEXT].modifier & mask)
200 || (wKeyBindings[WKBD_FOCUSPREV].modifier & mask)
201 || (wKeyBindings[WKBD_GROUPNEXT].modifier & mask)
202 || (wKeyBindings[WKBD_GROUPPREV].modifier & mask))) {
203 done = True;
204 break;
207 break;
209 case EnterNotify:
211 /* ignore unwanted EnterNotify's */
212 break;
214 case LeaveNotify:
215 case MotionNotify:
217 case ButtonRelease:
219 WWindow *tmp;
220 tmp = wSwitchPanelHandleEvent(swpanel, &ev);
221 if (tmp) {
222 newFocused = tmp;
223 oldFocused = change_focus_and_raise(newFocused, oldFocused, swpanel, scr, False);
225 if (ev.type == ButtonRelease)
226 done = True;
229 break;
231 default:
232 WMHandleEvent(&ev);
233 break;
236 if (keymap)
237 XFreeModifiermap(keymap);
239 if (hasModifier) {
241 XUngrabKeyboard(dpy, CurrentTime);
244 if (swpanel)
245 wSwitchPanelDestroy(swpanel);
247 if (newFocused && !esc_cancel) {
248 wRaiseFrame(newFocused->frame->core);
249 CommitStacking(scr);
250 if (!newFocused->flags.mapped)
251 wMakeWindowVisible(newFocused);
252 wSetFocusTo(scr, newFocused);
255 scr->flags.doing_alt_tab = 0;
257 if (somethingElse)
258 WMHandleEvent(&ev);