Update Serbian translation from master branch
[wmaker-crm.git] / src / window.c
blob1f16f4583b5d162efa4d8225db3249f4d30e5632
1 /* window.c - client window managing stuffs
3 * Window Maker window manager
5 * Copyright (c) 1997-2003 Alfredo K. Kojima
6 * Copyright (c) 2008-2023 Window Maker Team
8 * This program is free software; you can redistribute it and/or modify
9 * it under the terms of the GNU General Public License as published by
10 * the Free Software Foundation; either version 2 of the License, or
11 * (at your option) any later version.
13 * This program is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 * GNU General Public License for more details.
18 * You should have received a copy of the GNU General Public License along
19 * with this program; if not, write to the Free Software Foundation, Inc.,
20 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
23 #include "wconfig.h"
25 #include <X11/Xlib.h>
26 #include <X11/Xutil.h>
27 #ifdef USE_XSHAPE
28 #include <X11/extensions/shape.h>
29 #endif
30 #ifdef KEEP_XKB_LOCK_STATUS
31 #include <X11/XKBlib.h>
32 #endif /* KEEP_XKB_LOCK_STATUS */
33 #ifdef USE_XRES
34 #include <X11/extensions/XRes.h>
35 #endif
36 #include <stdlib.h>
37 #include <stdio.h>
38 #include <string.h>
39 #include <stdint.h>
40 #include <math.h>
42 /* For getting mouse wheel mappings from WINGs */
43 #include <WINGs/WINGs.h>
45 #include "WindowMaker.h"
46 #include "GNUstep.h"
47 #include "wcore.h"
48 #include "framewin.h"
49 #include "texture.h"
50 #include "window.h"
51 #include "winspector.h"
52 #include "icon.h"
53 #include "properties.h"
54 #include "actions.h"
55 #include "client.h"
56 #include "colormap.h"
57 #include "keybind.h"
58 #include "stacking.h"
59 #include "defaults.h"
60 #include "workspace.h"
61 #include "xinerama.h"
62 #include "appmenu.h"
63 #include "appicon.h"
64 #include "superfluous.h"
65 #include "rootmenu.h"
66 #include "placement.h"
67 #include "misc.h"
68 #include "startup.h"
69 #include "winmenu.h"
70 #include "osdep.h"
72 #ifdef USE_MWM_HINTS
73 # include "motif.h"
74 #endif
75 #include "wmspec.h"
77 #define MOD_MASK wPreferences.modifier_mask
80 /***** Local Stuff *****/
81 static WWindowState *windowState = NULL;
82 static FocusMode getFocusMode(WWindow *wwin);
83 static int getSavedState(Window window, WSavedState **state);
84 static void setupGNUstepHints(WWindow *wwin, GNUstepWMAttributes *gs_hints);
86 /* frame window (during window grabs) */
87 static void frameMouseDown(WObjDescriptor *desc, XEvent *event);
89 /* close button */
90 static void windowCloseClick(WCoreWindow *sender, void *data, XEvent *event);
91 static void windowCloseDblClick(WCoreWindow *sender, void *data, XEvent *event);
93 /* iconify button */
94 static void windowIconifyClick(WCoreWindow *sender, void *data, XEvent *event);
96 #ifdef XKB_BUTTON_HINT
97 static void windowLanguageClick(WCoreWindow *sender, void *data, XEvent *event);
98 #endif
100 static void titlebarMouseDown(WCoreWindow *sender, void *data, XEvent *event);
101 static void titlebarDblClick(WCoreWindow *sender, void *data, XEvent *event);
102 static void resizebarMouseDown(WCoreWindow *sender, void *data, XEvent *event);
104 static void release_wwindowstate(WWindowState *wstate);
106 /****** Notification Observers ******/
108 static void appearanceObserver(void *self, WMNotification * notif)
110 WWindow *wwin = (WWindow *) self;
111 uintptr_t flags = (uintptr_t)WMGetNotificationClientData(notif);
113 if (!wwin->frame || (!wwin->frame->titlebar && !wwin->frame->resizebar))
114 return;
116 if (flags & WFontSettings) {
117 wWindowConfigureBorders(wwin);
118 if (wwin->flags.shaded) {
119 wFrameWindowResize(wwin->frame, wwin->frame->core->width, wwin->frame->top_width - 1);
120 wwin->client.y = wwin->frame_y - wwin->client.height + wwin->frame->top_width;
121 wWindowSynthConfigureNotify(wwin);
124 if (flags & WTextureSettings)
125 wwin->frame->flags.need_texture_remake = 1;
127 if (flags & (WTextureSettings | WColorSettings)) {
128 if (wwin->frame->titlebar)
129 XClearWindow(dpy, wwin->frame->titlebar->window);
131 wFrameWindowPaint(wwin->frame);
136 /* Return the WWindow associated with a given (Xlib) Window. */
137 WWindow *wWindowFor(Window window)
139 WObjDescriptor *desc;
141 if (window == None)
142 return NULL;
144 if (XFindContext(dpy, window, w_global.context.client_win, (XPointer *) & desc) == XCNOENT)
145 return NULL;
147 if (desc->parent_type == WCLASS_WINDOW)
148 return desc->parent;
149 else if (desc->parent_type == WCLASS_FRAME) {
150 WFrameWindow *frame = (WFrameWindow *) desc->parent;
151 if (frame->flags.is_client_window_frame)
152 return frame->child;
155 return NULL;
158 WWindow *wWindowCreate(void)
160 WWindow *wwin;
162 wwin = wmalloc(sizeof(WWindow));
163 wretain(wwin);
165 wwin->client_descriptor.handle_mousedown = frameMouseDown;
166 wwin->client_descriptor.parent = wwin;
167 wwin->client_descriptor.self = wwin;
168 wwin->client_descriptor.parent_type = WCLASS_WINDOW;
170 return wwin;
173 void wWindowDestroy(WWindow *wwin)
175 int i;
177 if (wwin->screen_ptr->cmap_window == wwin)
178 wwin->screen_ptr->cmap_window = NULL;
180 WMRemoveNotificationObserver(wwin);
182 wwin->flags.destroyed = 1;
184 for (i = 0; i < MAX_WINDOW_SHORTCUTS; i++) {
185 if (!wwin->screen_ptr->shortcutWindows[i])
186 continue;
188 WMRemoveFromArray(wwin->screen_ptr->shortcutWindows[i], wwin);
190 if (!WMGetArrayItemCount(wwin->screen_ptr->shortcutWindows[i])) {
191 WMFreeArray(wwin->screen_ptr->shortcutWindows[i]);
192 wwin->screen_ptr->shortcutWindows[i] = NULL;
196 if (wwin->fake_group && wwin->fake_group->retainCount > 0) {
197 wwin->fake_group->retainCount--;
198 if (wwin->fake_group->retainCount == 0 && wwin->fake_group->leader != None) {
199 XDestroyWindow(dpy, wwin->fake_group->leader);
200 wwin->fake_group->leader = None;
201 wwin->fake_group->origLeader = None;
202 XFlush(dpy);
206 if (wwin->normal_hints)
207 XFree(wwin->normal_hints);
209 if (wwin->wm_hints)
210 XFree(wwin->wm_hints);
212 if (wwin->wm_instance)
213 XFree(wwin->wm_instance);
215 if (wwin->wm_class)
216 XFree(wwin->wm_class);
218 if (wwin->wm_gnustep_attr)
219 wfree(wwin->wm_gnustep_attr);
221 if (wwin->cmap_windows)
222 XFree(wwin->cmap_windows);
224 XDeleteContext(dpy, wwin->client_win, w_global.context.client_win);
226 if (wwin->frame)
227 wFrameWindowDestroy(wwin->frame);
229 if (wwin->icon) {
230 RemoveFromStackList(wwin->icon->core);
231 wIconDestroy(wwin->icon);
232 if (wPreferences.auto_arrange_icons)
233 wArrangeIcons(wwin->screen_ptr, True);
235 if (wwin->net_icon_image)
236 RReleaseImage(wwin->net_icon_image);
238 wrelease(wwin);
241 static void setupGNUstepHints(WWindow *wwin, GNUstepWMAttributes *gs_hints)
243 if (gs_hints->flags & GSWindowStyleAttr) {
244 if (gs_hints->window_style == WMBorderlessWindowMask) {
245 wwin->client_flags.no_border = 1;
246 wwin->client_flags.no_titlebar = 1;
247 wwin->client_flags.no_closable = 1;
248 wwin->client_flags.no_miniaturizable = 1;
249 wwin->client_flags.no_resizable = 1;
250 wwin->client_flags.no_close_button = 1;
251 wwin->client_flags.no_miniaturize_button = 1;
252 wwin->client_flags.no_resizebar = 1;
253 } else {
254 wwin->client_flags.no_close_button =
255 ((gs_hints->window_style & WMClosableWindowMask) ? 0 : 1);
257 wwin->client_flags.no_closable = ((gs_hints->window_style & WMClosableWindowMask) ? 0 : 1);
259 wwin->client_flags.no_miniaturize_button =
260 ((gs_hints->window_style & WMMiniaturizableWindowMask) ? 0 : 1);
262 wwin->client_flags.no_miniaturizable = wwin->client_flags.no_miniaturize_button;
264 wwin->client_flags.no_resizebar =
265 ((gs_hints->window_style & WMResizableWindowMask) ? 0 : 1);
267 wwin->client_flags.no_resizable = wwin->client_flags.no_resizebar;
269 /* these attributes supposedly imply in the existence
270 * of a titlebar */
271 if (gs_hints->window_style & (WMResizableWindowMask |
272 WMClosableWindowMask | WMMiniaturizableWindowMask)) {
273 wwin->client_flags.no_titlebar = 0;
274 } else {
275 wwin->client_flags.no_titlebar =
276 ((gs_hints->window_style & WMTitledWindowMask) ? 0 : 1);
280 } else {
281 /* setup the defaults */
282 wwin->client_flags.no_border = 0;
283 wwin->client_flags.no_titlebar = 0;
284 wwin->client_flags.no_closable = 0;
285 wwin->client_flags.no_miniaturizable = 0;
286 wwin->client_flags.no_resizable = 0;
287 wwin->client_flags.no_close_button = 0;
288 wwin->client_flags.no_miniaturize_button = 0;
289 wwin->client_flags.no_resizebar = 0;
291 if (gs_hints->extra_flags & GSNoApplicationIconFlag)
292 wwin->client_flags.no_appicon = 1;
295 static void discard_hints_from_gtk(WWindow *wwin)
297 Atom type;
298 int format;
299 unsigned long nb_item, nb_remain;
300 unsigned char *result;
301 int status;
303 status = XGetWindowProperty(dpy, wwin->client_win, w_global.atom.desktop.gtk_object_path, 0, 16, False,
304 AnyPropertyType, &type, &format, &nb_item, &nb_remain, &result);
305 if (status != Success)
306 return;
308 /* If we're here, that means the Property exists. We don't care what is inside, it means it is a GTK-based application */
310 if (result != NULL)
311 XFree(result);
313 /* GTK is asking to remove these decorations: */
314 wwin->client_flags.no_titlebar = 0;
315 wwin->client_flags.no_close_button = 0;
316 wwin->client_flags.no_miniaturize_button = 0;
317 wwin->client_flags.no_resizebar = 0;
320 void wWindowSetupInitialAttributes(WWindow *wwin, int *level, int *workspace)
322 WScreen *scr = wwin->screen_ptr;
324 /* sets global default stuff */
325 wDefaultFillAttributes(wwin->wm_instance, wwin->wm_class, &wwin->user_flags, NULL, True);
326 wwin->defined_user_flags = wwin->user_flags;
329 * Decoration setting is done in this precedence (lower to higher)
330 * - use global default in the resource database
331 * - guess some settings
332 * - use GNUstep/external window attributes
333 * - set hints specified for the app in the resource DB
336 wwin->client_flags.broken_close = 0;
338 if (wwin->protocols.DELETE_WINDOW)
339 wwin->client_flags.kill_close = 0;
340 else
341 wwin->client_flags.kill_close = 1;
343 /* transients can't be iconified or maximized */
344 if (wwin->transient_for != None && wwin->transient_for != scr->root_win) {
345 wwin->client_flags.no_miniaturizable = 1;
346 wwin->client_flags.no_miniaturize_button = 1;
349 /* if the window can't be resized, remove the resizebar */
350 if (wwin->normal_hints->flags & (PMinSize | PMaxSize)
351 && (wwin->normal_hints->min_width == wwin->normal_hints->max_width)
352 && (wwin->normal_hints->min_height == wwin->normal_hints->max_height)) {
353 wwin->client_flags.no_resizable = 1;
354 wwin->client_flags.no_resizebar = 1;
357 /* set GNUstep window attributes */
358 if (wwin->wm_gnustep_attr) {
359 setupGNUstepHints(wwin, wwin->wm_gnustep_attr);
361 if (wwin->wm_gnustep_attr->flags & GSWindowLevelAttr) {
363 *level = wwin->wm_gnustep_attr->window_level;
365 * INT_MIN is the only illegal window level.
367 if (*level == INT_MIN)
368 *level = INT_MIN + 1;
369 } else {
370 /* setup defaults */
371 *level = WMNormalLevel;
373 } else {
374 int tmp_workspace = -1;
375 int tmp_level = INT_MIN; /* INT_MIN is never used by the window levels */
377 #ifdef USE_MWM_HINTS
378 wMWMCheckClientHints(wwin);
379 #endif /* USE_MWM_HINTS */
381 wNETWMCheckClientHints(wwin, &tmp_level, &tmp_workspace);
383 if (wPreferences.ignore_gtk_decoration_hints)
384 discard_hints_from_gtk(wwin);
386 /* window levels are between INT_MIN+1 and INT_MAX, so if we still
387 * have INT_MIN that means that no window level was requested. -Dan
389 if (tmp_level == INT_MIN) {
390 if (WFLAGP(wwin, floating))
391 *level = WMFloatingLevel;
392 else if (WFLAGP(wwin, sunken))
393 *level = WMSunkenLevel;
394 else
395 *level = WMNormalLevel;
396 } else {
397 *level = tmp_level;
400 if (wwin->transient_for != None && wwin->transient_for != scr->root_win) {
401 WWindow *transientOwner = wWindowFor(wwin->transient_for);
402 if (transientOwner) {
403 int ownerLevel = transientOwner->frame->core->stacking->window_level;
404 if (ownerLevel > *level)
405 *level = ownerLevel;
409 if (tmp_workspace >= 0)
410 *workspace = tmp_workspace % scr->workspace_count;
414 * Set attributes specified only for that window/class.
415 * This might do duplicate work with the 1st wDefaultFillAttributes().
417 wDefaultFillAttributes(wwin->wm_instance, wwin->wm_class, &wwin->user_flags,
418 &wwin->defined_user_flags, False);
420 /* Restore decoration if the user has enabled the
421 * IgnoreDecorationChanges option */
422 if (wwin->user_flags.ignore_decoration_changes) {
423 WSETUFLAG(wwin, no_titlebar, 0);
424 WSETUFLAG(wwin, no_resizable, 0);
425 WSETUFLAG(wwin, no_miniaturizable, 0);
426 WSETUFLAG(wwin, no_resizebar, 0);
427 WSETUFLAG(wwin, no_close_button, 0);
428 WSETUFLAG(wwin, no_miniaturize_button, 0);
429 WSETUFLAG(wwin, no_border, 0);
430 WSETUFLAG(wwin, no_movable, 0);
434 * Sanity checks for attributes that depend on other attributes
436 if (wwin->user_flags.no_appicon && wwin->defined_user_flags.no_appicon)
437 wwin->user_flags.emulate_appicon = 0;
439 if (wwin->main_window != None) {
440 WApplication *wapp = wApplicationOf(wwin->main_window);
441 if (wapp && !wapp->flags.emulated)
442 wwin->user_flags.emulate_appicon = 0;
445 if (wwin->transient_for != None && wwin->transient_for != wwin->screen_ptr->root_win)
446 wwin->user_flags.emulate_appicon = 0;
448 if (wwin->user_flags.sunken && wwin->defined_user_flags.sunken
449 && wwin->user_flags.floating && wwin->defined_user_flags.floating)
450 wwin->user_flags.sunken = 0;
452 /* A window that does not have a title cannot be Shaded and we don't let user override this */
453 wwin->client_flags.no_shadeable = WFLAGP(wwin, no_titlebar);
454 wwin->defined_user_flags.no_shadeable = 0;
456 /* windows that have takefocus=False shouldn't take focus at all */
457 if (wwin->focus_mode == WFM_NO_INPUT)
458 wwin->client_flags.no_focusable = 1;
461 Bool wWindowObscuresWindow(WWindow *wwin, WWindow *obscured)
463 int w1, h1, w2, h2;
465 w1 = wwin->frame->core->width;
466 h1 = wwin->frame->core->height;
467 w2 = obscured->frame->core->width;
468 h2 = obscured->frame->core->height;
470 if (!IS_OMNIPRESENT(wwin) && !IS_OMNIPRESENT(obscured)
471 && wwin->frame->workspace != obscured->frame->workspace)
472 return False;
474 if (wwin->frame_x + w1 < obscured->frame_x
475 || wwin->frame_y + h1 < obscured->frame_y
476 || wwin->frame_x > obscured->frame_x + w2 || wwin->frame_y > obscured->frame_y + h2)
477 return False;
479 return True;
482 /* Get the corresponding Process Identification Number of the active window */
483 static pid_t getWindowPid(Window win)
485 pid_t pid = -1;
487 pid = wNETWMGetPidForWindow(win);
488 #ifdef USE_XRES
489 if (pid > 0)
490 return pid;
491 else {
492 XResClientIdSpec spec;
493 int status;
494 long i, num_ids = 0;
495 XResClientIdValue *client_ids = NULL;
497 spec.client = win;
498 spec.mask = XRES_CLIENT_ID_PID_MASK;
500 status = XResQueryClientIds(dpy, 1, &spec, &num_ids, &client_ids);
502 if (status != Success)
503 return -1;
505 for (i = 0; i < num_ids; i++) {
506 if (client_ids[i].spec.mask == XRES_CLIENT_ID_PID_MASK) {
507 pid = XResGetClientPid(&client_ids[i]);
508 break;
511 XResClientIdsDestroy(num_ids, client_ids);
513 #endif
514 return pid;
517 static void fixLeaderProperties(WWindow *wwin)
519 XClassHint *classHint;
520 XWMHints *hints, *clientHints;
521 XWindowAttributes attr;
522 Window leaders[2], window;
523 char **argv, *command;
524 int argc, i, pid;
525 Bool haveCommand;
527 classHint = XAllocClassHint();
528 clientHints = XGetWMHints(dpy, wwin->client_win);
529 pid = getWindowPid(wwin->client_win);
530 if (pid > 0)
531 haveCommand = GetCommandForPid(pid, &argv, &argc);
532 else
533 haveCommand = False;
535 leaders[0] = wwin->client_leader;
536 leaders[1] = wwin->group_id;
538 if (haveCommand) {
539 command = GetCommandForWindow(wwin->client_win);
540 if (command)
541 wfree(command); /* command already set. nothing to do. */
542 else
543 XSetCommand(dpy, wwin->client_win, argv, argc);
546 for (i = 0; i < 2; i++) {
547 window = leaders[i];
548 if (window) {
549 if (XGetClassHint(dpy, window, classHint) == 0) {
550 classHint->res_name = wwin->wm_instance;
551 classHint->res_class = wwin->wm_class;
552 XSetClassHint(dpy, window, classHint);
554 hints = XGetWMHints(dpy, window);
555 if (hints) {
556 XFree(hints);
557 } else if (clientHints) {
558 /* set window group leader to self */
559 clientHints->window_group = window;
560 clientHints->flags |= WindowGroupHint;
561 XSetWMHints(dpy, window, clientHints);
564 if (haveCommand) {
565 command = GetCommandForWindow(window);
566 if (command)
567 wfree(command); /* command already set. nothing to do. */
568 else
569 XSetCommand(dpy, window, argv, argc);
572 /* Make sure we get notification when this window is destroyed */
573 if (XGetWindowAttributes(dpy, window, &attr))
574 XSelectInput(dpy, window, attr.your_event_mask | StructureNotifyMask | PropertyChangeMask);
578 XFree(classHint);
579 if (clientHints)
580 XFree(clientHints);
581 if (haveCommand)
582 wfree(argv);
585 static Window createFakeWindowGroupLeader(WScreen *scr, Window win, char *instance, char *class)
587 XClassHint *classHint;
588 XWMHints *hints;
589 Window leader;
590 int argc;
591 char **argv;
593 leader = XCreateSimpleWindow(dpy, scr->root_win, 10, 10, 10, 10, 0, 0, 0);
594 /* set class hint */
595 classHint = XAllocClassHint();
596 classHint->res_name = instance;
597 classHint->res_class = class;
598 XSetClassHint(dpy, leader, classHint);
599 XFree(classHint);
601 /* inherit these from the original leader if available */
602 hints = XGetWMHints(dpy, win);
603 if (!hints) {
604 hints = XAllocWMHints();
605 hints->flags = 0;
607 /* set window group leader to self */
608 hints->window_group = leader;
609 hints->flags |= WindowGroupHint;
610 XSetWMHints(dpy, leader, hints);
611 XFree(hints);
613 if (XGetCommand(dpy, win, &argv, &argc) != 0 && argc > 0) {
614 XSetCommand(dpy, leader, argv, argc);
615 XFreeStringList(argv);
618 return leader;
621 static int matchIdentifier(const void *item, const void *cdata)
623 return (strcmp(((WFakeGroupLeader *) item)->identifier, (char *)cdata) == 0);
627 *----------------------------------------------------------------
628 * wManageWindow--
629 * reparents the window and allocates a descriptor for it.
630 * Window manager hints and other hints are fetched to configure
631 * the window decoration attributes and others. User preferences
632 * for the window are used if available, to configure window
633 * decorations and some behaviour.
634 * If in startup, windows that are override redirect,
635 * unmapped and never were managed and are Withdrawn are not
636 * managed.
638 * Returns:
639 * the new window descriptor
641 * Side effects:
642 * The window is reparented and appropriate notification
643 * is done to the client. Input mask for the window is setup.
644 * The window descriptor is also associated with various window
645 * contexts and inserted in the head of the window list.
646 * Event handler contexts are associated for some objects
647 * (buttons, titlebar and resizebar)
649 *----------------------------------------------------------------
651 WWindow *wManageWindow(WScreen *scr, Window window)
653 WWindow *wwin;
654 int x, y;
655 unsigned width, height;
656 XWindowAttributes wattribs;
657 XSetWindowAttributes attribs;
658 WWindowState *win_state;
659 WWindow *transientOwner = NULL;
660 int window_level;
661 int wm_state;
662 int foo;
663 int workspace = -1;
664 char *title;
665 Bool withdraw = False;
666 Bool raise = False;
668 /* mutex. */
669 XGrabServer(dpy);
670 XSync(dpy, False);
672 /* make sure the window is still there */
673 if (!XGetWindowAttributes(dpy, window, &wattribs)) {
674 XUngrabServer(dpy);
675 return NULL;
678 /* if it's an override-redirect, ignore it */
679 if (wattribs.override_redirect) {
680 XUngrabServer(dpy);
681 return NULL;
684 wm_state = PropGetWindowState(window);
686 /* if it's startup and the window is unmapped, don't manage it */
687 if (scr->flags.startup && wm_state < 0 && wattribs.map_state == IsUnmapped) {
688 XUngrabServer(dpy);
689 return NULL;
692 wwin = wWindowCreate();
694 title = wNETWMGetWindowName(window);
695 if (title)
696 wwin->flags.net_has_title = 1;
697 else if (!wFetchName(dpy, window, &title))
698 title = NULL;
700 XSaveContext(dpy, window, w_global.context.client_win, (XPointer) & wwin->client_descriptor);
702 #ifdef USE_XSHAPE
703 if (w_global.xext.shape.supported) {
704 int junk;
705 unsigned int ujunk;
706 int b_shaped;
708 XShapeSelectInput(dpy, window, ShapeNotifyMask);
709 XShapeQueryExtents(dpy, window, &b_shaped, &junk, &junk, &ujunk,
710 &ujunk, &junk, &junk, &junk, &ujunk, &ujunk);
711 wwin->flags.shaped = b_shaped;
713 #endif
715 /* Get hints and other information in properties */
716 PropGetWMClass(window, &wwin->wm_class, &wwin->wm_instance);
718 /* setup descriptor */
719 wwin->client_win = window;
720 wwin->screen_ptr = scr;
721 wwin->old_border_width = wattribs.border_width;
722 wwin->event_mask = CLIENT_EVENTS;
723 attribs.event_mask = CLIENT_EVENTS;
724 attribs.do_not_propagate_mask = ButtonPressMask | ButtonReleaseMask;
725 attribs.save_under = False;
727 XChangeWindowAttributes(dpy, window, CWEventMask | CWDontPropagate | CWSaveUnder, &attribs);
728 XSetWindowBorderWidth(dpy, window, 0);
730 if (wwin->wm_class != NULL && strcmp(wwin->wm_class, "DockApp") != 0)
731 wwin->flags.fullscreen_monitors[0] = -1;
733 /* get hints from GNUstep app */
734 if (wwin->wm_class != NULL && strcmp(wwin->wm_class, "GNUstep") == 0)
735 wwin->flags.is_gnustep = 1;
737 if (!PropGetGNUstepWMAttr(window, &wwin->wm_gnustep_attr))
738 wwin->wm_gnustep_attr = NULL;
740 if (wwin->wm_class != NULL && strcmp(wwin->wm_class, "DockApp") == 0) {
741 wwin->flags.is_dockapp = 1;
742 withdraw = True;
745 wwin->client_leader = PropGetClientLeader(window);
746 if (wwin->client_leader != None)
747 wwin->main_window = wwin->client_leader;
749 wwin->wm_hints = XGetWMHints(dpy, window);
751 if (wwin->wm_hints) {
752 if (wwin->wm_hints->flags & StateHint) {
753 if (wwin->wm_hints->initial_state == IconicState) {
754 wwin->flags.miniaturized = 1;
755 } else if (wwin->wm_hints->initial_state == WithdrawnState) {
756 wwin->flags.is_dockapp = 1;
757 withdraw = True;
761 if (wwin->wm_hints->flags & WindowGroupHint) {
762 wwin->group_id = wwin->wm_hints->window_group;
763 /* window_group has priority over CLIENT_LEADER */
764 wwin->main_window = wwin->group_id;
765 } else {
766 wwin->group_id = None;
769 if (wwin->wm_hints->flags & UrgencyHint) {
770 wwin->flags.urgent = 1;
771 wAppBounceWhileUrgent(wApplicationOf(wwin->main_window));
773 } else {
774 wwin->group_id = None;
777 PropGetProtocols(window, &wwin->protocols);
779 if (!XGetTransientForHint(dpy, window, &wwin->transient_for)) {
780 wwin->transient_for = None;
781 } else {
782 if (wwin->transient_for == None || wwin->transient_for == window) {
783 wwin->transient_for = scr->root_win;
784 } else {
785 transientOwner = wWindowFor(wwin->transient_for);
786 if (transientOwner && transientOwner->main_window != None)
787 wwin->main_window = transientOwner->main_window;
791 /* guess the focus mode */
792 wwin->focus_mode = getFocusMode(wwin);
794 /* get geometry stuff */
795 wClientGetNormalHints(wwin, &wattribs, True, &x, &y, &width, &height);
797 /* get colormap windows */
798 GetColormapWindows(wwin);
801 * Setup the decoration/window attributes and
802 * geometry
804 wWindowSetupInitialAttributes(wwin, &window_level, &workspace);
806 /* Make broken apps behave as a nice app. */
807 if (WFLAGP(wwin, emulate_appicon))
808 wwin->main_window = wwin->client_win;
810 fixLeaderProperties(wwin);
812 wwin->orig_main_window = wwin->main_window;
814 if (wwin->flags.is_gnustep)
815 wwin->client_flags.shared_appicon = 0;
817 if (wwin->main_window) {
818 XTextProperty text_prop;
820 if (XGetTextProperty(dpy, wwin->main_window, &text_prop, w_global.atom.wmaker.menu))
821 wwin->client_flags.shared_appicon = 0;
824 if (wwin->flags.is_dockapp)
825 wwin->client_flags.shared_appicon = 0;
827 if (wwin->main_window) {
828 WApplication *app = wApplicationOf(wwin->main_window);
829 if (app && app->app_icon)
830 wwin->client_flags.shared_appicon = 0;
833 if (!withdraw && wwin->main_window && WFLAGP(wwin, shared_appicon)) {
834 char *buffer, *instance, *class;
835 WFakeGroupLeader *fPtr;
836 int index;
838 #define ADEQUATE(x) ((x)!=None && (x)!=wwin->client_win && (x)!=fPtr->leader)
840 /* // only enter here if PropGetWMClass() succeeds */
841 PropGetWMClass(wwin->main_window, &class, &instance);
842 buffer = StrConcatDot(instance, class);
844 index = WMFindInArray(scr->fakeGroupLeaders, matchIdentifier, (void *)buffer);
845 if (index != WANotFound) {
846 fPtr = WMGetFromArray(scr->fakeGroupLeaders, index);
847 if (fPtr->retainCount == 0)
848 fPtr->leader = createFakeWindowGroupLeader(scr, wwin->main_window,
849 instance, class);
851 fPtr->retainCount++;
852 if (fPtr->origLeader == None) {
853 if (ADEQUATE(wwin->main_window)) {
854 fPtr->retainCount++;
855 fPtr->origLeader = wwin->main_window;
858 wwin->fake_group = fPtr;
859 wwin->main_window = fPtr->leader;
860 wfree(buffer);
861 } else {
862 fPtr = (WFakeGroupLeader *) wmalloc(sizeof(WFakeGroupLeader));
864 fPtr->identifier = buffer;
865 fPtr->leader = createFakeWindowGroupLeader(scr, wwin->main_window, instance, class);
866 fPtr->origLeader = None;
867 fPtr->retainCount = 1;
869 WMAddToArray(scr->fakeGroupLeaders, fPtr);
871 if (ADEQUATE(wwin->main_window)) {
872 fPtr->retainCount++;
873 fPtr->origLeader = wwin->main_window;
875 wwin->fake_group = fPtr;
876 wwin->main_window = fPtr->leader;
879 if (instance)
880 free(instance);
882 if (class)
883 free(class);
884 #undef ADEQUATE
888 * Setup the initial state of the window
890 if (WFLAGP(wwin, start_miniaturized) && !WFLAGP(wwin, no_miniaturizable))
891 wwin->flags.miniaturized = 1;
893 if (WFLAGP(wwin, start_maximized) && IS_RESIZABLE(wwin))
894 wwin->flags.maximized = MAX_VERTICAL | MAX_HORIZONTAL;
896 wNETWMCheckInitialClientState(wwin);
898 /* apply previous state if it exists and we're in startup */
899 if (scr->flags.startup && wm_state >= 0) {
900 if (wm_state == IconicState)
901 wwin->flags.miniaturized = 1;
902 else if (wm_state == WithdrawnState)
903 withdraw = True;
906 /* if there is a saved state (from file), restore it */
907 win_state = NULL;
908 if (wwin->main_window != None)
909 win_state = (WWindowState *) wWindowGetSavedState(wwin->main_window);
910 else
911 win_state = (WWindowState *) wWindowGetSavedState(window);
913 if (win_state && !withdraw) {
914 if (win_state->state->hidden > 0)
915 wwin->flags.hidden = win_state->state->hidden;
917 if (win_state->state->shaded > 0 && !WFLAGP(wwin, no_shadeable))
918 wwin->flags.shaded = win_state->state->shaded;
920 if (win_state->state->miniaturized > 0 && !WFLAGP(wwin, no_miniaturizable))
921 wwin->flags.miniaturized = win_state->state->miniaturized;
923 if (win_state->state->maximized > 0)
924 wwin->flags.maximized = win_state->state->maximized;
926 if (!IS_OMNIPRESENT(wwin)) {
927 int w = wDefaultGetStartWorkspace(scr, wwin->wm_instance,
928 wwin->wm_class);
929 if (w < 0 || w >= scr->workspace_count) {
930 workspace = win_state->state->workspace;
931 if (workspace >= scr->workspace_count)
932 workspace = scr->current_workspace;
933 } else {
934 workspace = w;
936 } else {
937 workspace = scr->current_workspace;
941 /* if we're restarting, restore saved state (from hints).
942 * This will overwrite previous */
944 WSavedState *wstate;
946 if (getSavedState(window, &wstate)) {
947 wwin->flags.shaded = wstate->shaded;
948 wwin->flags.hidden = wstate->hidden;
949 wwin->flags.miniaturized = wstate->miniaturized;
950 wwin->flags.maximized = wstate->maximized;
951 if (wwin->flags.maximized) {
952 wwin->old_geometry.x = wstate->x;
953 wwin->old_geometry.y = wstate->y;
954 wwin->old_geometry.width = wstate->w;
955 wwin->old_geometry.height = wstate->h;
958 workspace = wstate->workspace;
959 } else {
960 wstate = NULL;
963 /* restore window shortcut */
964 if (wstate != NULL || win_state != NULL) {
965 unsigned mask = 0;
967 if (win_state != NULL)
968 mask = win_state->state->window_shortcuts;
970 if (wstate != NULL && mask == 0)
971 mask = wstate->window_shortcuts;
973 if (mask > 0) {
974 int i;
976 for (i = 0; i < MAX_WINDOW_SHORTCUTS; i++) {
977 if (mask & (1 << i)) {
978 if (!scr->shortcutWindows[i])
979 scr->shortcutWindows[i] = WMCreateArray(4);
981 WMAddToArray(scr->shortcutWindows[i], wwin);
987 if (wstate != NULL)
988 wfree(wstate);
991 /* don't let transients start miniaturized if their owners are not */
992 if (transientOwner && !transientOwner->flags.miniaturized && wwin->flags.miniaturized && !withdraw) {
993 wwin->flags.miniaturized = 0;
994 if (wwin->wm_hints)
995 wwin->wm_hints->initial_state = NormalState;
998 /* set workspace on which the window starts */
999 if (workspace >= 0) {
1000 if (workspace > scr->workspace_count - 1)
1001 workspace = workspace % scr->workspace_count;
1002 } else {
1003 int w;
1005 w = wDefaultGetStartWorkspace(scr, wwin->wm_instance, wwin->wm_class);
1007 if (w >= 0 && w < scr->workspace_count && !(IS_OMNIPRESENT(wwin))) {
1008 workspace = w;
1009 } else {
1010 if (wPreferences.open_transients_with_parent && transientOwner)
1011 workspace = transientOwner->frame->workspace;
1012 else
1013 workspace = scr->current_workspace;
1017 /* setup window geometry */
1018 if (win_state && win_state->state->w > 0) {
1019 width = win_state->state->w;
1020 height = win_state->state->h;
1023 wWindowConstrainSize(wwin, &width, &height);
1025 /* do not ask for window placement if the window is
1026 * transient, during startup, or if the window wants
1027 * to start iconic. If geometry was saved, restore it. */
1029 Bool dontBring = False;
1031 if (win_state && win_state->state->w > 0) {
1032 x = win_state->state->x;
1033 y = win_state->state->y;
1034 } else if ((wwin->transient_for == None || wPreferences.window_placement != WPM_MANUAL)
1035 && !scr->flags.startup
1036 && !wwin->flags.miniaturized
1037 && !wwin->flags.maximized && !(wwin->normal_hints->flags & (USPosition | PPosition))) {
1039 if (transientOwner && transientOwner->flags.mapped) {
1040 int offs = WMAX(20, 2 * transientOwner->frame->top_width);
1041 WMRect rect;
1042 int head;
1044 x = transientOwner->frame_x +
1045 abs(((int)transientOwner->frame->core->width - (int)width) / 2) + offs;
1046 y = transientOwner->frame_y +
1047 abs(((int)transientOwner->frame->core->height - (int)height) / 3) + offs;
1049 /* limit transient windows to be inside their parent's head */
1050 rect.pos.x = transientOwner->frame_x;
1051 rect.pos.y = transientOwner->frame_y;
1052 rect.size.width = transientOwner->frame->core->width;
1053 rect.size.height = transientOwner->frame->core->height;
1055 head = wGetHeadForRect(scr, rect);
1056 rect = wGetRectForHead(scr, head);
1058 if (x < rect.pos.x)
1059 x = rect.pos.x;
1060 else if (x + width > rect.pos.x + rect.size.width)
1061 x = rect.pos.x + rect.size.width - width;
1063 if (y < rect.pos.y)
1064 y = rect.pos.y;
1065 else if (y + height > rect.pos.y + rect.size.height)
1066 y = rect.pos.y + rect.size.height - height;
1068 } else {
1069 PlaceWindow(wwin, &x, &y, width, height);
1072 if (wPreferences.window_placement == WPM_MANUAL)
1073 dontBring = True;
1075 } else if (scr->xine_info.count && (wwin->normal_hints->flags & PPosition)) {
1076 int head, flags;
1077 WMRect rect;
1078 int reposition = 0;
1080 /* Make spash screens come out in the center of a head
1081 * trouble is that most splashies never get here
1082 * they are managed trough atoms but god knows where.
1083 * Dan, do you know ? -peter
1085 * Most of them are not managed, they have set
1086 * OverrideRedirect, which means we can't do anything about
1087 * them. -alfredo */
1089 /* xinerama checks for: across head and dead space */
1090 rect.pos.x = x;
1091 rect.pos.y = y;
1092 rect.size.width = width;
1093 rect.size.height = height;
1095 head = wGetRectPlacementInfo(scr, rect, &flags);
1097 if (flags & XFLAG_DEAD)
1098 reposition = 1;
1100 if (flags & XFLAG_MULTIPLE)
1101 reposition = 2;
1104 switch (reposition) {
1105 case 1:
1106 head = wGetHeadForPointerLocation(scr);
1107 rect = wGetRectForHead(scr, head);
1109 x = rect.pos.x + (x * rect.size.width) / scr->scr_width;
1110 y = rect.pos.y + (y * rect.size.height) / scr->scr_height;
1111 break;
1113 case 2:
1114 rect = wGetRectForHead(scr, head);
1116 if (x < rect.pos.x)
1117 x = rect.pos.x;
1118 else if (x + width > rect.pos.x + rect.size.width)
1119 x = rect.pos.x + rect.size.width - width;
1121 if (y < rect.pos.y)
1122 y = rect.pos.y;
1123 else if (y + height > rect.pos.y + rect.size.height)
1124 y = rect.pos.y + rect.size.height - height;
1126 break;
1128 default:
1129 break;
1133 if (WFLAGP(wwin, dont_move_off) && dontBring)
1134 wScreenBringInside(scr, &x, &y, width, height);
1137 wNETWMPositionSplash(wwin, &x, &y, width, height);
1139 if (wwin->flags.urgent) {
1140 if (!IS_OMNIPRESENT(wwin))
1141 wwin->flags.omnipresent ^= 1;
1144 /* Create frame, borders and do reparenting */
1145 foo = WFF_LEFT_BUTTON | WFF_RIGHT_BUTTON;
1146 #ifdef XKB_BUTTON_HINT
1147 if (wPreferences.modelock)
1148 foo |= WFF_LANGUAGE_BUTTON;
1149 #endif
1150 if (HAS_TITLEBAR(wwin))
1151 foo |= WFF_TITLEBAR;
1153 if (HAS_RESIZEBAR(wwin))
1154 foo |= WFF_RESIZEBAR;
1156 if (HAS_BORDER(wwin))
1157 foo |= WFF_BORDER;
1159 wwin->frame = wFrameWindowCreate(scr, window_level,
1160 x, y, width, height,
1161 &wPreferences.window_title_clearance,
1162 &wPreferences.window_title_min_height,
1163 &wPreferences.window_title_max_height,
1164 foo,
1165 scr->window_title_texture,
1166 scr->resizebar_texture, scr->window_title_color, &scr->title_font,
1167 wattribs.depth, wattribs.visual, wattribs.colormap);
1169 wwin->frame->flags.is_client_window_frame = 1;
1170 wwin->frame->flags.justification = wPreferences.title_justification;
1172 wNETWMCheckInitialFrameState(wwin);
1174 /* setup button images */
1175 wWindowUpdateButtonImages(wwin);
1177 /* hide unused buttons */
1178 foo = 0;
1179 if (WFLAGP(wwin, no_close_button))
1180 foo |= WFF_RIGHT_BUTTON;
1182 if (WFLAGP(wwin, no_miniaturize_button))
1183 foo |= WFF_LEFT_BUTTON;
1185 #ifdef XKB_BUTTON_HINT
1186 if (WFLAGP(wwin, no_language_button) || WFLAGP(wwin, no_focusable))
1187 foo |= WFF_LANGUAGE_BUTTON;
1188 #endif
1189 if (foo != 0)
1190 wFrameWindowHideButton(wwin->frame, foo);
1192 wwin->frame->child = wwin;
1193 wwin->frame->workspace = workspace;
1194 wwin->frame->on_click_left = windowIconifyClick;
1196 #ifdef XKB_BUTTON_HINT
1197 if (wPreferences.modelock)
1198 wwin->frame->on_click_language = windowLanguageClick;
1199 #endif
1201 wwin->frame->on_click_right = windowCloseClick;
1202 wwin->frame->on_dblclick_right = windowCloseDblClick;
1203 wwin->frame->on_mousedown_titlebar = titlebarMouseDown;
1204 wwin->frame->on_dblclick_titlebar = titlebarDblClick;
1205 wwin->frame->on_mousedown_resizebar = resizebarMouseDown;
1207 XSelectInput(dpy, wwin->client_win, wwin->event_mask & ~StructureNotifyMask);
1208 XReparentWindow(dpy, wwin->client_win, wwin->frame->core->window, 0, wwin->frame->top_width);
1209 XSelectInput(dpy, wwin->client_win, wwin->event_mask);
1212 int gx, gy;
1214 wClientGetGravityOffsets(wwin, &gx, &gy);
1216 /* if gravity is to the south, account for the border sizes */
1217 if (gy > 0)
1218 y -= wwin->frame->top_width + wwin->frame->bottom_width;
1222 * wWindowConfigure() will init the client window's size
1223 * (wwin->client.{width,height}) and all other geometry
1224 * related variables (frame_x,frame_y) */
1225 wWindowConfigure(wwin, x, y, width, height);
1227 /* to make sure the window receives it's new position after reparenting */
1228 wWindowSynthConfigureNotify(wwin);
1230 /* Setup descriptors and save window to internal lists */
1231 if (wwin->main_window != None) {
1232 WApplication *app;
1233 WWindow *leader;
1235 /* Leader windows do not necessary set themselves as leaders.
1236 * If this is the case, point the leader of this window to
1237 * itself */
1238 leader = wWindowFor(wwin->main_window);
1239 if (leader && leader->main_window == None)
1240 leader->main_window = leader->client_win;
1242 app = wApplicationCreate(wwin);
1243 if (app) {
1244 app->last_workspace = workspace;
1246 /* Do application specific stuff, like setting application
1247 * wide attributes. */
1249 if (wwin->flags.hidden) {
1250 /* if the window was set to hidden because it was hidden
1251 * in a previous incarnation and that state was restored */
1252 app->flags.hidden = 1;
1253 } else if (app->flags.hidden) {
1254 if (WFLAGP(app->main_window_desc, start_hidden)) {
1255 wwin->flags.hidden = 1;
1256 } else {
1257 wUnhideApplication(app, False, False);
1258 raise = True;
1261 wAppBounce(app);
1265 /* setup the frame descriptor */
1266 wwin->frame->core->descriptor.handle_mousedown = frameMouseDown;
1267 wwin->frame->core->descriptor.parent = wwin;
1268 wwin->frame->core->descriptor.parent_type = WCLASS_WINDOW;
1270 /* don't let windows go away if we die */
1271 XAddToSaveSet(dpy, window);
1273 XLowerWindow(dpy, window);
1275 /* if window is in this workspace and should be mapped, then map it */
1276 if (!wwin->flags.miniaturized && (workspace == scr->current_workspace || IS_OMNIPRESENT(wwin))
1277 && !wwin->flags.hidden && !withdraw) {
1279 /* The following "if" is to avoid crashing of clients that expect
1280 * WM_STATE set before they get mapped. Else WM_STATE is set later,
1281 * after the return from this function. */
1282 if (wwin->wm_hints && (wwin->wm_hints->flags & StateHint))
1283 wClientSetState(wwin, wwin->wm_hints->initial_state, None);
1284 else
1285 wClientSetState(wwin, NormalState, None);
1287 if (wPreferences.superfluous && !wPreferences.no_animations && !scr->flags.startup &&
1288 (wwin->transient_for == None || wwin->transient_for == scr->root_win) &&
1290 * The brain damaged idiotic non-click to focus modes will
1291 * have trouble with this because:
1293 * 1. window is created and mapped by the client
1294 * 2. window is mapped by wmaker in small size
1295 * 3. window is animated to grow to normal size
1296 * 4. this function returns to normal event loop
1297 * 5. eventually, the EnterNotify event that would trigger
1298 * the window focusing (if the mouse is over that window)
1299 * will be processed by wmaker.
1300 * But since this event will be rather delayed
1301 * (step 3 has a large delay) the time when the event occurred
1302 * and when it is processed, the client that owns that window
1303 * will reject the XSetInputFocus() for it.
1305 (wPreferences.focus_mode == WKF_CLICK || wPreferences.auto_focus))
1306 DoWindowBirth(wwin);
1308 wWindowMap(wwin);
1311 /* setup stacking descriptor */
1312 if (transientOwner)
1313 wwin->frame->core->stacking->child_of = transientOwner->frame->core;
1314 else
1315 wwin->frame->core->stacking->child_of = NULL;
1317 if (!scr->focused_window) {
1318 /* first window on the list */
1319 wwin->next = NULL;
1320 wwin->prev = NULL;
1321 scr->focused_window = wwin;
1322 } else {
1323 WWindow *tmp;
1325 /* add window at beginning of focus window list */
1326 tmp = scr->focused_window;
1327 while (tmp->prev)
1328 tmp = tmp->prev;
1330 tmp->prev = wwin;
1331 wwin->next = tmp;
1332 wwin->prev = NULL;
1335 /* raise is set to true if we un-hid the app when this window was born.
1336 * we raise, else old windows of this app will be above this new one. */
1337 if (raise)
1338 wRaiseFrame(wwin->frame->core);
1340 /* Update name must come after WApplication stuff is done */
1341 wWindowUpdateName(wwin, title);
1342 if (title)
1343 XFree(title);
1345 XUngrabServer(dpy);
1347 /* Final preparations before window is ready to go */
1348 wFrameWindowChangeState(wwin->frame, WS_UNFOCUSED);
1350 if (!wwin->flags.miniaturized && workspace == scr->current_workspace && !wwin->flags.hidden) {
1351 if (((transientOwner && transientOwner->flags.focused)
1352 || wPreferences.auto_focus) && !WFLAGP(wwin, no_focusable)) {
1354 /* only auto_focus if on same screen as mouse
1355 * (and same head for xinerama mode)
1356 * TODO: make it an option */
1358 /*TODO add checking the head of the window, is it available? */
1359 short same_screen = 0, same_head = 1;
1361 int foo;
1362 unsigned int bar;
1363 Window dummy;
1365 if (XQueryPointer(dpy, scr->root_win, &dummy, &dummy,
1366 &foo, &foo, &foo, &foo, &bar) != False)
1367 same_screen = 1;
1369 if (same_screen == 1 && same_head == 1)
1370 wSetFocusTo(scr, wwin);
1373 wWindowResetMouseGrabs(wwin);
1375 if (!WFLAGP(wwin, no_bind_keys))
1376 wWindowSetKeyGrabs(wwin);
1378 WMPostNotificationName(WMNManaged, wwin, NULL);
1379 wColormapInstallForWindow(scr, scr->cmap_window);
1381 /* Setup Notification Observers */
1382 WMAddNotificationObserver(appearanceObserver, wwin, WNWindowAppearanceSettingsChanged, wwin);
1384 /* Cleanup temporary stuff */
1385 if (win_state)
1386 wWindowDeleteSavedState(win_state);
1388 /* If the window must be withdrawed, then do it now.
1389 * Must do some optimization, 'though */
1390 if (withdraw) {
1391 wwin->flags.mapped = 0;
1392 wClientSetState(wwin, WithdrawnState, None);
1393 wUnmanageWindow(wwin, True, False);
1394 wwin = NULL;
1397 return wwin;
1400 WWindow *wManageInternalWindow(WScreen *scr, Window window, Window owner,
1401 const char *title, int x, int y, int width, int height)
1403 WWindow *wwin;
1404 int foo;
1406 wwin = wWindowCreate();
1408 WMAddNotificationObserver(appearanceObserver, wwin, WNWindowAppearanceSettingsChanged, wwin);
1410 wwin->flags.internal_window = 1;
1412 wwin->client_flags.omnipresent = 1;
1413 wwin->client_flags.no_shadeable = 1;
1414 wwin->client_flags.no_resizable = 1;
1415 wwin->client_flags.no_miniaturizable = 1;
1417 wwin->focus_mode = WFM_PASSIVE;
1418 wwin->client_win = window;
1419 wwin->screen_ptr = scr;
1420 wwin->transient_for = owner;
1421 wwin->client.x = x;
1422 wwin->client.y = y;
1423 wwin->client.width = width;
1424 wwin->client.height = height;
1425 wwin->frame_x = wwin->client.x;
1426 wwin->frame_y = wwin->client.y;
1428 foo = WFF_RIGHT_BUTTON | WFF_BORDER;
1429 foo |= WFF_TITLEBAR;
1430 #ifdef XKB_BUTTON_HINT
1431 foo |= WFF_LANGUAGE_BUTTON;
1432 #endif
1434 wwin->frame = wFrameWindowCreate(scr, WMFloatingLevel,
1435 wwin->frame_x, wwin->frame_y,
1436 width, height,
1437 &wPreferences.window_title_clearance,
1438 &wPreferences.window_title_min_height,
1439 &wPreferences.window_title_max_height,
1440 foo,
1441 scr->window_title_texture,
1442 scr->resizebar_texture, scr->window_title_color, &scr->title_font,
1443 scr->w_depth, scr->w_visual, scr->w_colormap);
1445 XSaveContext(dpy, window, w_global.context.client_win, (XPointer) & wwin->client_descriptor);
1447 wwin->frame->flags.is_client_window_frame = 1;
1448 wwin->frame->flags.justification = wPreferences.title_justification;
1450 wFrameWindowChangeTitle(wwin->frame, title);
1452 /* setup button images */
1453 wWindowUpdateButtonImages(wwin);
1455 /* hide buttons */
1456 wFrameWindowHideButton(wwin->frame, WFF_RIGHT_BUTTON);
1458 wwin->frame->child = wwin;
1459 wwin->frame->workspace = wwin->screen_ptr->current_workspace;
1461 #ifdef XKB_BUTTON_HINT
1462 if (wPreferences.modelock)
1463 wwin->frame->on_click_language = windowLanguageClick;
1464 #endif
1466 wwin->frame->on_click_right = windowCloseClick;
1467 wwin->frame->on_mousedown_titlebar = titlebarMouseDown;
1468 wwin->frame->on_dblclick_titlebar = titlebarDblClick;
1469 wwin->frame->on_mousedown_resizebar = resizebarMouseDown;
1470 wwin->client.y += wwin->frame->top_width;
1472 XReparentWindow(dpy, wwin->client_win, wwin->frame->core->window, 0, wwin->frame->top_width);
1473 wWindowConfigure(wwin, wwin->frame_x, wwin->frame_y, wwin->client.width, wwin->client.height);
1475 /* setup the frame descriptor */
1476 wwin->frame->core->descriptor.handle_mousedown = frameMouseDown;
1477 wwin->frame->core->descriptor.parent = wwin;
1478 wwin->frame->core->descriptor.parent_type = WCLASS_WINDOW;
1480 XLowerWindow(dpy, window);
1481 XMapSubwindows(dpy, wwin->frame->core->window);
1483 /* setup stacking descriptor */
1484 if (wwin->transient_for != None && wwin->transient_for != scr->root_win) {
1485 WWindow *tmp;
1486 tmp = wWindowFor(wwin->transient_for);
1487 if (tmp)
1488 wwin->frame->core->stacking->child_of = tmp->frame->core;
1489 } else {
1490 wwin->frame->core->stacking->child_of = NULL;
1493 if (!scr->focused_window) {
1494 /* first window on the list */
1495 wwin->next = NULL;
1496 wwin->prev = NULL;
1497 scr->focused_window = wwin;
1498 } else {
1499 WWindow *tmp;
1501 /* add window at beginning of focus window list */
1502 tmp = scr->focused_window;
1503 while (tmp->prev)
1504 tmp = tmp->prev;
1505 tmp->prev = wwin;
1506 wwin->next = tmp;
1507 wwin->prev = NULL;
1510 if (wwin->flags.is_gnustep == 0)
1511 wFrameWindowChangeState(wwin->frame, WS_UNFOCUSED);
1513 /* if (wPreferences.auto_focus) */
1514 wSetFocusTo(scr, wwin);
1515 wWindowResetMouseGrabs(wwin);
1516 wWindowSetKeyGrabs(wwin);
1518 return wwin;
1522 *----------------------------------------------------------------------
1523 * wUnmanageWindow--
1524 * Removes the frame window from a window and destroys all data
1525 * related to it. The window will be reparented back to the root window
1526 * if restore is True.
1528 * Side effects:
1529 * Everything related to the window is destroyed and the window
1530 * is removed from the window lists. Focus is set to the previous on the
1531 * window list.
1532 *----------------------------------------------------------------------
1534 void wUnmanageWindow(WWindow *wwin, Bool restore, Bool destroyed)
1536 WCoreWindow *frame = wwin->frame->core;
1537 WWindow *owner = NULL;
1538 WWindow *newFocusedWindow = NULL;
1539 int wasFocused;
1540 WScreen *scr = wwin->screen_ptr;
1542 /* First close attribute editor window if open */
1543 if (wwin->flags.inspector_open)
1544 wCloseInspectorForWindow(wwin);
1546 /* Close window menu if it's open for this window */
1547 if (wwin->flags.menu_open_for_me)
1548 CloseWindowMenu(scr);
1550 /* Don't restore focus to this window after a window exits
1551 * fullscreen mode */
1552 if (scr->bfs_focused_window == wwin)
1553 scr->bfs_focused_window = NULL;
1555 if (!destroyed) {
1556 if (!wwin->flags.internal_window)
1557 XRemoveFromSaveSet(dpy, wwin->client_win);
1559 /* If this is a leader window, we still need to listen for
1560 * DestroyNotify and PropertyNotify. */
1561 if (wApplicationOf(wwin->client_win))
1562 XSelectInput(dpy, wwin->client_win, StructureNotifyMask | PropertyChangeMask);
1563 else
1564 XSelectInput(dpy, wwin->client_win, NoEventMask);
1566 XUngrabButton(dpy, AnyButton, AnyModifier, wwin->client_win);
1567 XUngrabKey(dpy, AnyKey, AnyModifier, wwin->client_win);
1570 XUnmapWindow(dpy, frame->window);
1572 XUnmapWindow(dpy, wwin->client_win);
1574 /* deselect window */
1575 wSelectWindow(wwin, False);
1577 /* remove all pending events on window */
1578 /* I think this only matters for autoraise */
1579 if (wPreferences.raise_delay)
1580 WMDeleteTimerWithClientData(wwin->frame->core);
1582 XFlush(dpy);
1584 /* reparent the window back to the root */
1585 if (restore)
1586 wClientRestore(wwin);
1588 if (wwin->transient_for != scr->root_win) {
1589 owner = wWindowFor(wwin->transient_for);
1590 if (owner) {
1591 if (!owner->flags.semi_focused)
1592 owner = NULL;
1593 else
1594 owner->flags.semi_focused = 0;
1598 wasFocused = wwin->flags.focused;
1600 /* remove from window focus list */
1601 if (!wwin->prev && !wwin->next) {
1602 /* was the only window */
1603 scr->focused_window = NULL;
1604 newFocusedWindow = NULL;
1605 } else {
1606 WWindow *tmp;
1608 if (wwin->prev)
1609 wwin->prev->next = wwin->next;
1611 if (wwin->next)
1612 wwin->next->prev = wwin->prev;
1613 else {
1614 scr->focused_window = wwin->prev;
1615 scr->focused_window->next = NULL;
1618 if (wPreferences.focus_mode == WKF_CLICK) {
1620 /* if in click to focus mode and the window
1621 * was a transient, focus the owner window
1623 tmp = NULL;
1624 if (wPreferences.focus_mode == WKF_CLICK) {
1625 tmp = wWindowFor(wwin->transient_for);
1626 if (tmp && (!tmp->flags.mapped || WFLAGP(tmp, no_focusable))) {
1627 tmp = NULL;
1630 /* otherwise, focus the next one in the focus list */
1631 if (!tmp) {
1632 tmp = scr->focused_window;
1633 while (tmp) { /* look for one in the window list first */
1634 if (!WFLAGP(tmp, no_focusable) && !WFLAGP(tmp, skip_window_list)
1635 && (tmp->flags.mapped || tmp->flags.shaded))
1636 break;
1637 tmp = tmp->prev;
1639 if (!tmp) { /* if unsuccessful, choose any focusable window */
1640 tmp = scr->focused_window;
1641 while (tmp) {
1642 if (!WFLAGP(tmp, no_focusable)
1643 && (tmp->flags.mapped || tmp->flags.shaded))
1644 break;
1645 tmp = tmp->prev;
1650 newFocusedWindow = tmp;
1652 } else if (wPreferences.focus_mode == WKF_SLOPPY) {
1653 unsigned int mask;
1654 int foo;
1655 Window bar, win;
1657 /* This is to let the root window get the keyboard input
1658 * if Sloppy focus mode and no other window get focus.
1659 * This way keybindings will not freeze.
1661 tmp = NULL;
1662 if (XQueryPointer(dpy, scr->root_win, &bar, &win, &foo, &foo, &foo, &foo, &mask))
1663 tmp = wWindowFor(win);
1664 if (tmp == wwin)
1665 tmp = NULL;
1666 newFocusedWindow = tmp;
1667 } else {
1668 newFocusedWindow = NULL;
1672 if (!wwin->flags.internal_window)
1673 WMPostNotificationName(WMNUnmanaged, wwin, NULL);
1674 if (wasFocused) {
1675 if (newFocusedWindow != owner && owner) {
1676 if (wwin->flags.is_gnustep == 0)
1677 wFrameWindowChangeState(owner->frame, WS_UNFOCUSED);
1679 wSetFocusTo(scr, newFocusedWindow);
1682 /* Close menu and unhighlight */
1683 WApplication *oapp = wApplicationOf(wwin->main_window);
1684 WApplication *napp = scr->focused_window ? wApplicationOf(scr->focused_window->main_window) : NULL;
1685 if (oapp && oapp != napp) {
1686 wAppMenuUnmap(oapp->menu);
1687 if (wPreferences.highlight_active_app)
1688 wApplicationDeactivate(oapp);
1691 wNETCleanupFrameExtents(wwin);
1693 wWindowDestroy(wwin);
1694 XFlush(dpy);
1697 void wWindowMap(WWindow *wwin)
1699 XMapWindow(dpy, wwin->frame->core->window);
1700 if (!wwin->flags.shaded) {
1701 /* window will be remapped when getting MapNotify */
1702 XSelectInput(dpy, wwin->client_win, wwin->event_mask & ~StructureNotifyMask);
1703 XMapWindow(dpy, wwin->client_win);
1704 XSelectInput(dpy, wwin->client_win, wwin->event_mask);
1706 wwin->flags.mapped = 1;
1710 void wWindowUnmap(WWindow *wwin)
1712 wwin->flags.mapped = 0;
1714 /* prevent window withdrawal when getting UnmapNotify */
1715 XSelectInput(dpy, wwin->client_win, wwin->event_mask & ~StructureNotifyMask);
1716 XUnmapWindow(dpy, wwin->client_win);
1717 XSelectInput(dpy, wwin->client_win, wwin->event_mask);
1719 XUnmapWindow(dpy, wwin->frame->core->window);
1722 void wWindowSingleFocus(WWindow *wwin)
1724 WScreen *scr;
1725 int x, y, move = 0;
1727 if (!wwin)
1728 return;
1730 scr = wwin->screen_ptr;
1731 wMakeWindowVisible(wwin);
1733 x = wwin->frame_x;
1734 y = wwin->frame_y;
1736 /* bring window back to visible area */
1737 move = wScreenBringInside(scr, &x, &y, wwin->frame->core->width, wwin->frame->core->height);
1739 if (move)
1740 wWindowConfigure(wwin, x, y, wwin->client.width, wwin->client.height);
1743 void wWindowFocusPrev(WWindow *wwin, Bool inSameWorkspace)
1745 WWindow *tmp;
1747 if (!wwin || !wwin->prev)
1748 return;
1750 tmp = wwin;
1751 while (tmp->prev)
1752 tmp = tmp->prev;
1754 if (inSameWorkspace)
1755 while (tmp && (tmp->frame->workspace != wwin->frame->workspace))
1756 tmp = tmp->next;
1758 wWindowSingleFocus(tmp);
1761 void wWindowFocusNext(WWindow *wwin, Bool inSameWorkspace)
1763 WWindow *tmp;
1765 if (!wwin || !wwin->prev)
1766 return;
1768 tmp = wwin->prev;
1769 if (inSameWorkspace)
1770 while (tmp && (tmp->frame->workspace != wwin->frame->workspace))
1771 tmp = tmp->prev;
1773 wWindowSingleFocus(tmp);
1776 void wWindowFocus(WWindow *wwin, WWindow *owin)
1778 WWindow *nowner;
1779 WWindow *oowner;
1781 #ifdef KEEP_XKB_LOCK_STATUS
1782 if (wPreferences.modelock)
1783 XkbLockGroup(dpy, XkbUseCoreKbd, wwin->frame->languagemode);
1784 #endif /* KEEP_XKB_LOCK_STATUS */
1786 wwin->flags.semi_focused = 0;
1788 if (wwin->flags.is_gnustep == 0)
1789 wFrameWindowChangeState(wwin->frame, WS_FOCUSED);
1791 wwin->flags.focused = 1;
1793 wWindowResetMouseGrabs(wwin);
1795 WMPostNotificationName(WMNChangedFocus, wwin, (void *)True);
1797 if (owin == wwin || !owin)
1798 return;
1800 nowner = wWindowFor(wwin->transient_for);
1802 /* new window is a transient for the old window */
1803 if (nowner == owin) {
1804 owin->flags.semi_focused = 1;
1805 wWindowUnfocus(nowner);
1806 return;
1809 oowner = wWindowFor(owin->transient_for);
1811 /* new window is owner of old window */
1812 if (wwin == oowner) {
1813 wWindowUnfocus(owin);
1814 return;
1817 if (!nowner) {
1818 wWindowUnfocus(owin);
1819 return;
1822 /* new window has same owner of old window */
1823 if (oowner == nowner) {
1824 /* prevent unfocusing of owner */
1825 oowner->flags.semi_focused = 0;
1826 wWindowUnfocus(owin);
1827 oowner->flags.semi_focused = 1;
1829 return;
1832 /* nowner != NULL && oowner != nowner */
1833 nowner->flags.semi_focused = 1;
1834 wWindowUnfocus(nowner);
1835 wWindowUnfocus(owin);
1838 void wWindowUnfocus(WWindow *wwin)
1840 CloseWindowMenu(wwin->screen_ptr);
1842 if (wwin->flags.is_gnustep == 0)
1843 wFrameWindowChangeState(wwin->frame, wwin->flags.semi_focused ? WS_PFOCUSED : WS_UNFOCUSED);
1845 if (wwin->transient_for != None && wwin->transient_for != wwin->screen_ptr->root_win) {
1846 WWindow *owner;
1847 owner = wWindowFor(wwin->transient_for);
1848 if (owner && owner->flags.semi_focused) {
1849 owner->flags.semi_focused = 0;
1850 if (owner->flags.mapped || owner->flags.shaded) {
1851 wWindowUnfocus(owner);
1852 wFrameWindowPaint(owner->frame);
1856 wwin->flags.focused = 0;
1857 wWindowResetMouseGrabs(wwin);
1858 WMPostNotificationName(WMNChangedFocus, wwin, (void *)False);
1861 void wWindowUpdateName(WWindow *wwin, const char *newTitle)
1863 const char *title;
1865 if (!wwin->frame)
1866 return;
1868 if (!newTitle)
1869 title = DEF_WINDOW_TITLE; /* the hint was removed */
1870 else
1871 title = newTitle;
1873 if (wFrameWindowChangeTitle(wwin->frame, title))
1874 WMPostNotificationName(WMNChangedName, wwin, NULL);
1878 *----------------------------------------------------------------------
1880 * wWindowConstrainSize--
1881 * Constrains size for the client window, taking the maximal size,
1882 * window resize increments and other size hints into account.
1884 * Returns:
1885 * The closest size to what was given that the client window can
1886 * have.
1888 *----------------------------------------------------------------------
1890 void wWindowConstrainSize(WWindow *wwin, unsigned int *nwidth, unsigned int *nheight)
1892 int width = (int)*nwidth;
1893 int height = (int)*nheight;
1894 int winc = 1;
1895 int hinc = 1;
1896 int minW = 1, minH = 1;
1897 int maxW = wwin->screen_ptr->scr_width * 2;
1898 int maxH = wwin->screen_ptr->scr_height * 2;
1899 int minAX = -1, minAY = -1;
1900 int maxAX = -1, maxAY = -1;
1901 int baseW = 0;
1902 int baseH = 0;
1904 if (wwin->normal_hints) {
1905 if (!wwin->flags.maximized) {
1906 winc = wwin->normal_hints->width_inc;
1907 hinc = wwin->normal_hints->height_inc;
1909 if (wwin->normal_hints->min_width > minW)
1910 minW = wwin->normal_hints->min_width;
1911 if (wwin->normal_hints->min_height > minH)
1912 minH = wwin->normal_hints->min_height;
1913 if (wwin->normal_hints->max_width < maxW)
1914 maxW = wwin->normal_hints->max_width;
1915 if (wwin->normal_hints->max_height < maxH)
1916 maxH = wwin->normal_hints->max_height;
1918 if (wwin->normal_hints->flags & PAspect) {
1919 minAX = wwin->normal_hints->min_aspect.x;
1920 minAY = wwin->normal_hints->min_aspect.y;
1921 maxAX = wwin->normal_hints->max_aspect.x;
1922 maxAY = wwin->normal_hints->max_aspect.y;
1925 baseW = wwin->normal_hints->base_width;
1926 baseH = wwin->normal_hints->base_height;
1929 /* trust the mins provided by the client but not the maxs */
1930 if (width < minW)
1931 width = minW;
1932 if (height < minH)
1933 height = minH;
1935 /* if only one dimension is over the top, set a default 4/3 ratio */
1936 if (width > maxW && height < maxH)
1937 width = height * 4 / 3;
1938 else if(height > maxH && width < maxW)
1939 height = width * 3 / 4;
1940 else if(width > maxW && height > maxH) {
1941 /* if both are over the top, set size to almost fullscreen */
1942 height = wwin->screen_ptr->scr_height - 2 * wPreferences.icon_size;
1943 width = wwin->screen_ptr->scr_width - 2 * wPreferences.icon_size;
1946 /* aspect ratio code borrowed from olwm */
1947 if (minAX > 0) {
1948 /* adjust max aspect ratio */
1949 if (!(maxAX == 1 && maxAY == 1) && width * maxAY > height * maxAX) {
1950 if (maxAX > maxAY) {
1951 height = (width * maxAY) / maxAX;
1952 if (height > maxH) {
1953 height = maxH;
1954 width = (height * maxAX) / maxAY;
1956 } else {
1957 width = (height * maxAX) / maxAY;
1958 if (width > maxW) {
1959 width = maxW;
1960 height = (width * maxAY) / maxAX;
1965 /* adjust min aspect ratio */
1966 if (!(minAX == 1 && minAY == 1) && width * minAY < height * minAX) {
1967 if (minAX > minAY) {
1968 height = (width * minAY) / minAX;
1969 if (height < minH) {
1970 height = minH;
1971 width = (height * minAX) / minAY;
1973 } else {
1974 width = (height * minAX) / minAY;
1975 if (width < minW) {
1976 width = minW;
1977 height = (width * minAY) / minAX;
1983 if (!wwin->flags.maximized) {
1984 if (baseW != 0)
1985 width = (((width - baseW) / winc) * winc) + baseW;
1986 else
1987 width = (((width - minW) / winc) * winc) + minW;
1989 if (baseH != 0)
1990 height = (((height - baseH) / hinc) * hinc) + baseH;
1991 else
1992 height = (((height - minH) / hinc) * hinc) + minH;
1995 /* broken stupid apps may cause preposterous values for these.. */
1996 if (width > 0)
1997 *nwidth = width;
1998 if (height > 0)
1999 *nheight = height;
2002 void wWindowCropSize(WWindow *wwin, unsigned int maxW, unsigned int maxH,
2003 unsigned int *width, unsigned int *height)
2005 int baseW = 0, baseH = 0;
2006 int winc = 1, hinc = 1;
2008 if (wwin->normal_hints) {
2009 baseW = wwin->normal_hints->base_width;
2010 baseH = wwin->normal_hints->base_height;
2012 winc = wwin->normal_hints->width_inc;
2013 hinc = wwin->normal_hints->height_inc;
2016 if (*width > maxW)
2017 *width = maxW - (maxW - baseW) % winc;
2019 if (*height > maxH)
2020 *height = maxH - (maxH - baseH) % hinc;
2023 void wWindowChangeWorkspace(WWindow *wwin, int workspace)
2025 WScreen *scr = wwin->screen_ptr;
2026 WApplication *wapp;
2027 int unmap = 0;
2029 if (workspace >= scr->workspace_count || workspace < 0 || workspace == wwin->frame->workspace)
2030 return;
2032 if (workspace != scr->current_workspace) {
2033 /* Sent to other workspace. Unmap window */
2034 if ((wwin->flags.mapped
2035 || wwin->flags.shaded || (wwin->flags.miniaturized && !wPreferences.sticky_icons))
2036 && !IS_OMNIPRESENT(wwin) && !wwin->flags.changing_workspace) {
2038 wapp = wApplicationOf(wwin->main_window);
2039 if (wapp)
2040 wapp->last_workspace = workspace;
2042 if (wwin->flags.miniaturized) {
2043 if (wwin->icon) {
2044 XUnmapWindow(dpy, wwin->icon->core->window);
2045 wwin->icon->mapped = 0;
2047 } else {
2048 unmap = 1;
2049 wSetFocusTo(scr, NULL);
2052 } else {
2053 /* brought to current workspace. Map window */
2054 if (wwin->flags.miniaturized && !wPreferences.sticky_icons) {
2055 if (wwin->icon) {
2056 XMapWindow(dpy, wwin->icon->core->window);
2057 wwin->icon->mapped = 1;
2059 } else if (!wwin->flags.mapped && !(wwin->flags.miniaturized || wwin->flags.hidden)) {
2060 wWindowMap(wwin);
2063 if (!IS_OMNIPRESENT(wwin)) {
2064 int oldWorkspace = wwin->frame->workspace;
2065 wwin->frame->workspace = workspace;
2066 WMPostNotificationName(WMNChangedWorkspace, wwin, (void *)(uintptr_t) oldWorkspace);
2069 if (unmap)
2070 wWindowUnmap(wwin);
2073 void wWindowChangeWorkspaceRelative(WWindow *wwin, int amount)
2075 WScreen *scr = wwin->screen_ptr;
2076 int w = scr->current_workspace + amount;
2078 if (amount < 0) {
2079 if (w >= 0) {
2080 wWindowChangeWorkspace(wwin, w);
2081 } else if (wPreferences.ws_cycle) {
2082 wWindowChangeWorkspace(wwin, scr->workspace_count + w);
2084 } else if (amount > 0) {
2085 if (w < scr->workspace_count) {
2086 wWindowChangeWorkspace(wwin, w);
2087 } else if (wPreferences.ws_advance) {
2088 int workspace = WMIN(w, MAX_WORKSPACES - 1);
2089 wWorkspaceMake(scr, workspace);
2090 wWindowChangeWorkspace(wwin, workspace);
2091 } else if (wPreferences.ws_cycle) {
2092 wWindowChangeWorkspace(wwin, w % scr->workspace_count);
2097 void wWindowSynthConfigureNotify(WWindow *wwin)
2099 XEvent sevent;
2101 sevent.type = ConfigureNotify;
2102 sevent.xconfigure.display = dpy;
2103 sevent.xconfigure.event = wwin->client_win;
2104 sevent.xconfigure.window = wwin->client_win;
2106 sevent.xconfigure.x = wwin->client.x;
2107 sevent.xconfigure.y = wwin->client.y;
2108 sevent.xconfigure.width = wwin->client.width;
2109 sevent.xconfigure.height = wwin->client.height;
2111 sevent.xconfigure.border_width = wwin->old_border_width;
2112 if (HAS_TITLEBAR(wwin) && wwin->frame->titlebar)
2113 sevent.xconfigure.above = wwin->frame->titlebar->window;
2114 else
2115 sevent.xconfigure.above = None;
2117 sevent.xconfigure.override_redirect = False;
2118 XSendEvent(dpy, wwin->client_win, False, StructureNotifyMask, &sevent);
2119 XFlush(dpy);
2123 *----------------------------------------------------------------------
2124 * wWindowConfigure()
2126 * req_x, req_y: new requested positions for the frame
2127 * req_width, req_height: new requested sizes for the client
2129 * Configures the frame, decorations and client window to the specified
2130 * geometry, whose validity is not checked -- wWindowConstrainSize()
2131 * must be used for that.
2132 * The size parameters are for the client window, but the position is
2133 * for the frame.
2134 * The client window receives a ConfigureNotify event, according
2135 * to what ICCCM says.
2137 * Returns:
2138 * None
2140 * Side effects:
2141 * Window size and position are changed and client window receives
2142 * a ConfigureNotify event.
2143 *----------------------------------------------------------------------
2145 void wWindowConfigure(WWindow *wwin, int req_x, int req_y, int req_width, int req_height)
2147 int synth_notify = False;
2148 int resize;
2150 resize = (req_width != wwin->client.width || req_height != wwin->client.height);
2152 if (WFLAGP(wwin, dont_move_off))
2153 wScreenBringInside(wwin->screen_ptr, &req_x, &req_y, req_width, req_height);
2155 if (resize) {
2156 if (req_width < MIN_WINDOW_SIZE)
2157 req_width = MIN_WINDOW_SIZE;
2158 if (req_height < MIN_WINDOW_SIZE)
2159 req_height = MIN_WINDOW_SIZE;
2161 /* If growing, resize inner part before frame,
2162 * if shrinking, resize frame before.
2163 * This will prevent the frame (that can have a different color)
2164 * to be exposed, causing flicker */
2165 if (req_height > wwin->frame->core->height || req_width > wwin->frame->core->width)
2166 XResizeWindow(dpy, wwin->client_win, req_width, req_height);
2168 if (wwin->flags.shaded) {
2169 wFrameWindowConfigure(wwin->frame, req_x, req_y, req_width, wwin->frame->core->height);
2170 wwin->old_geometry.height = req_height;
2171 } else {
2172 int h;
2174 h = req_height + wwin->frame->top_width + wwin->frame->bottom_width;
2176 wFrameWindowConfigure(wwin->frame, req_x, req_y, req_width, h);
2179 if (!(req_height > wwin->frame->core->height || req_width > wwin->frame->core->width))
2180 XResizeWindow(dpy, wwin->client_win, req_width, req_height);
2182 wwin->client.x = req_x;
2183 wwin->client.y = req_y + wwin->frame->top_width;
2184 wwin->client.width = req_width;
2185 wwin->client.height = req_height;
2186 } else {
2187 if (req_x != wwin->frame_x || req_y != wwin->frame_y) {
2188 wwin->client.x = req_x;
2189 wwin->client.y = req_y + wwin->frame->top_width;
2190 XMoveWindow(dpy, wwin->frame->core->window, req_x, req_y);
2193 * if the window is being moved but not resized
2194 * or if we change nothing then
2195 * send a synthetic ConfigureNotify
2197 synth_notify = True;
2199 wwin->frame_x = req_x;
2200 wwin->frame_y = req_y;
2201 if (HAS_BORDER(wwin)) {
2202 wwin->client.x += wwin->screen_ptr->frame_border_width;
2203 wwin->client.y += wwin->screen_ptr->frame_border_width;
2205 #ifdef USE_XSHAPE
2206 if (w_global.xext.shape.supported && wwin->flags.shaped && resize)
2207 wWindowSetShape(wwin);
2208 #endif
2210 if (synth_notify)
2211 wWindowSynthConfigureNotify(wwin);
2213 wNETFrameExtents(wwin);
2215 XFlush(dpy);
2218 /* req_x, req_y: new position of the frame */
2219 void wWindowMove(WWindow *wwin, int req_x, int req_y)
2221 #ifdef CONFIGURE_WINDOW_WHILE_MOVING
2222 int synth_notify = False;
2224 /* Send a synthetic ConfigureNotify event for every window movement. */
2225 if ((req_x != wwin->frame_x || req_y != wwin->frame_y))
2226 synth_notify = True;
2227 #else
2228 /* A single synthetic ConfigureNotify event is sent at the end of
2229 * a completed (opaque) movement in moveres.c */
2230 #endif
2232 if (WFLAGP(wwin, dont_move_off))
2233 wScreenBringInside(wwin->screen_ptr, &req_x, &req_y,
2234 wwin->frame->core->width, wwin->frame->core->height);
2236 wwin->client.x = req_x;
2237 wwin->client.y = req_y + wwin->frame->top_width;
2238 if (HAS_BORDER(wwin)) {
2239 wwin->client.x += wwin->screen_ptr->frame_border_width;
2240 wwin->client.y += wwin->screen_ptr->frame_border_width;
2243 XMoveWindow(dpy, wwin->frame->core->window, req_x, req_y);
2245 wwin->frame_x = req_x;
2246 wwin->frame_y = req_y;
2248 #ifdef CONFIGURE_WINDOW_WHILE_MOVING
2249 if (synth_notify)
2250 wWindowSynthConfigureNotify(wwin);
2251 #endif
2254 void wWindowUpdateButtonImages(WWindow *wwin)
2256 WScreen *scr = wwin->screen_ptr;
2257 Pixmap pixmap, mask;
2258 WFrameWindow *fwin = wwin->frame;
2260 if (!HAS_TITLEBAR(wwin))
2261 return;
2263 /* miniaturize button */
2264 if (!WFLAGP(wwin, no_miniaturize_button)) {
2265 if (wwin->wm_gnustep_attr && wwin->wm_gnustep_attr->flags & GSMiniaturizePixmapAttr) {
2266 pixmap = wwin->wm_gnustep_attr->miniaturize_pixmap;
2268 if (wwin->wm_gnustep_attr->flags & GSMiniaturizeMaskAttr)
2269 mask = wwin->wm_gnustep_attr->miniaturize_mask;
2270 else
2271 mask = None;
2273 if (fwin->lbutton_image
2274 && (fwin->lbutton_image->image != pixmap || fwin->lbutton_image->mask != mask)) {
2275 wPixmapDestroy(fwin->lbutton_image);
2276 fwin->lbutton_image = NULL;
2279 if (!fwin->lbutton_image) {
2280 fwin->lbutton_image = wPixmapCreate(pixmap, mask);
2281 fwin->lbutton_image->client_owned = 1;
2282 fwin->lbutton_image->client_owned_mask = 1;
2284 } else {
2285 if (fwin->lbutton_image && !fwin->lbutton_image->shared)
2286 wPixmapDestroy(fwin->lbutton_image);
2288 fwin->lbutton_image = scr->b_pixmaps[WBUT_ICONIFY];
2291 #ifdef XKB_BUTTON_HINT
2292 if (!WFLAGP(wwin, no_language_button)) {
2293 if (fwin->languagebutton_image && !fwin->languagebutton_image->shared)
2294 wPixmapDestroy(fwin->languagebutton_image);
2296 fwin->languagebutton_image = scr->b_pixmaps[WBUT_XKBGROUP1 + fwin->languagemode];
2298 #endif
2300 /* close button */
2302 /* redefine WFLAGP to MGFLAGP to allow broken close operation */
2303 #define MGFLAGP(wwin, FLAG) (wwin)->client_flags.FLAG
2305 if (!WFLAGP(wwin, no_close_button)) {
2306 if (wwin->wm_gnustep_attr && wwin->wm_gnustep_attr->flags & GSClosePixmapAttr) {
2307 pixmap = wwin->wm_gnustep_attr->close_pixmap;
2309 if (wwin->wm_gnustep_attr->flags & GSCloseMaskAttr)
2310 mask = wwin->wm_gnustep_attr->close_mask;
2311 else
2312 mask = None;
2314 if (fwin->rbutton_image && (fwin->rbutton_image->image != pixmap
2315 || fwin->rbutton_image->mask != mask)) {
2316 wPixmapDestroy(fwin->rbutton_image);
2317 fwin->rbutton_image = NULL;
2320 if (!fwin->rbutton_image) {
2321 fwin->rbutton_image = wPixmapCreate(pixmap, mask);
2322 fwin->rbutton_image->client_owned = 1;
2323 fwin->rbutton_image->client_owned_mask = 1;
2326 } else if (WFLAGP(wwin, kill_close)) {
2327 if (fwin->rbutton_image && !fwin->rbutton_image->shared)
2328 wPixmapDestroy(fwin->rbutton_image);
2330 fwin->rbutton_image = scr->b_pixmaps[WBUT_KILL];
2332 } else if (MGFLAGP(wwin, broken_close)) {
2333 if (fwin->rbutton_image && !fwin->rbutton_image->shared)
2334 wPixmapDestroy(fwin->rbutton_image);
2336 fwin->rbutton_image = scr->b_pixmaps[WBUT_BROKENCLOSE];
2338 } else {
2339 if (fwin->rbutton_image && !fwin->rbutton_image->shared)
2340 wPixmapDestroy(fwin->rbutton_image);
2342 fwin->rbutton_image = scr->b_pixmaps[WBUT_CLOSE];
2346 /* force buttons to be redrawn */
2347 fwin->flags.need_texture_change = 1;
2348 wFrameWindowPaint(fwin);
2352 *---------------------------------------------------------------------------
2353 * wWindowConfigureBorders--
2354 * Update window border configuration according to attribute flags.
2356 *---------------------------------------------------------------------------
2358 void wWindowConfigureBorders(WWindow *wwin)
2360 if (wwin->frame) {
2361 int flags;
2362 int newy, oldh;
2364 flags = WFF_LEFT_BUTTON | WFF_RIGHT_BUTTON;
2366 #ifdef XKB_BUTTON_HINT
2367 if (wPreferences.modelock)
2368 flags |= WFF_LANGUAGE_BUTTON;
2369 #endif
2371 if (HAS_TITLEBAR(wwin))
2372 flags |= WFF_TITLEBAR;
2373 if (HAS_RESIZEBAR(wwin) && IS_RESIZABLE(wwin))
2374 flags |= WFF_RESIZEBAR;
2375 if (HAS_BORDER(wwin))
2376 flags |= WFF_BORDER;
2377 if (wwin->flags.shaded)
2378 flags |= WFF_IS_SHADED;
2379 if (wwin->flags.selected)
2380 flags |= WFF_SELECTED;
2382 oldh = wwin->frame->top_width;
2383 wFrameWindowUpdateBorders(wwin->frame, flags);
2384 if (oldh != wwin->frame->top_width) {
2385 newy = wwin->frame_y + oldh - wwin->frame->top_width;
2387 XMoveWindow(dpy, wwin->client_win, 0, wwin->frame->top_width);
2388 wWindowConfigure(wwin, wwin->frame_x, newy, wwin->client.width, wwin->client.height);
2391 flags = 0;
2392 if (!WFLAGP(wwin, no_miniaturize_button)
2393 && wwin->frame->flags.hide_left_button)
2394 flags |= WFF_LEFT_BUTTON;
2396 #ifdef XKB_BUTTON_HINT
2397 if (!WFLAGP(wwin, no_language_button)
2398 && wwin->frame->flags.hide_language_button)
2399 flags |= WFF_LANGUAGE_BUTTON;
2400 #endif
2402 if (!WFLAGP(wwin, no_close_button)
2403 && wwin->frame->flags.hide_right_button)
2404 flags |= WFF_RIGHT_BUTTON;
2406 if (flags != 0) {
2407 wWindowUpdateButtonImages(wwin);
2408 wFrameWindowShowButton(wwin->frame, flags);
2411 flags = 0;
2412 if (WFLAGP(wwin, no_miniaturize_button)
2413 && !wwin->frame->flags.hide_left_button)
2414 flags |= WFF_LEFT_BUTTON;
2416 #ifdef XKB_BUTTON_HINT
2417 if (WFLAGP(wwin, no_language_button)
2418 && !wwin->frame->flags.hide_language_button)
2419 flags |= WFF_LANGUAGE_BUTTON;
2420 #endif
2422 if (WFLAGP(wwin, no_close_button)
2423 && !wwin->frame->flags.hide_right_button)
2424 flags |= WFF_RIGHT_BUTTON;
2426 if (flags != 0)
2427 wFrameWindowHideButton(wwin->frame, flags);
2429 #ifdef USE_XSHAPE
2430 if (w_global.xext.shape.supported && wwin->flags.shaped)
2431 wWindowSetShape(wwin);
2432 #endif
2436 void wWindowSaveState(WWindow *wwin)
2438 long data[10];
2439 int i;
2441 memset(data, 0, sizeof(long) * 10);
2442 data[0] = wwin->frame->workspace;
2443 data[1] = wwin->flags.miniaturized;
2444 data[2] = wwin->flags.shaded;
2445 data[3] = wwin->flags.hidden;
2446 data[4] = wwin->flags.maximized;
2447 if (wwin->flags.maximized == 0) {
2448 data[5] = wwin->frame_x;
2449 data[6] = wwin->frame_y;
2450 data[7] = wwin->frame->core->width;
2451 data[8] = wwin->frame->core->height;
2452 } else {
2453 data[5] = wwin->old_geometry.x;
2454 data[6] = wwin->old_geometry.y;
2455 data[7] = wwin->old_geometry.width;
2456 data[8] = wwin->old_geometry.height;
2459 for (i = 0; i < MAX_WINDOW_SHORTCUTS; i++) {
2460 if (wwin->screen_ptr->shortcutWindows[i] &&
2461 WMCountInArray(wwin->screen_ptr->shortcutWindows[i], wwin))
2462 data[9] |= 1 << i;
2465 XChangeProperty(dpy, wwin->client_win, w_global.atom.wmaker.state,
2466 w_global.atom.wmaker.state, 32, PropModeReplace, (unsigned char *)data, 10);
2469 static int getSavedState(Window window, WSavedState ** state)
2471 Atom type_ret;
2472 int fmt_ret;
2473 unsigned long nitems_ret;
2474 unsigned long bytes_after_ret;
2475 long *data;
2477 if (XGetWindowProperty(dpy, window, w_global.atom.wmaker.state, 0, 10,
2478 True, w_global.atom.wmaker.state,
2479 &type_ret, &fmt_ret, &nitems_ret, &bytes_after_ret,
2480 (unsigned char **)&data) != Success || !data || nitems_ret < 10)
2481 return 0;
2483 if (type_ret != w_global.atom.wmaker.state) {
2484 XFree(data);
2485 return 0;
2488 *state = wmalloc(sizeof(WSavedState));
2490 (*state)->workspace = data[0];
2491 (*state)->miniaturized = data[1];
2492 (*state)->shaded = data[2];
2493 (*state)->hidden = data[3];
2494 (*state)->maximized = data[4];
2495 (*state)->x = data[5];
2496 (*state)->y = data[6];
2497 (*state)->w = data[7];
2498 (*state)->h = data[8];
2499 (*state)->window_shortcuts = data[9];
2501 XFree(data);
2503 return 1;
2506 #ifdef USE_XSHAPE
2507 void wWindowClearShape(WWindow * wwin)
2509 XShapeCombineMask(dpy, wwin->frame->core->window, ShapeBounding,
2510 0, wwin->frame->top_width, None, ShapeSet);
2511 XFlush(dpy);
2514 void wWindowSetShape(WWindow * wwin)
2516 XRectangle rect[2];
2517 int count;
2518 #ifdef OPTIMIZE_SHAPE
2519 XRectangle *rects;
2520 XRectangle *urec;
2521 int ordering;
2523 /* only shape is the client's */
2524 if (!HAS_TITLEBAR(wwin) && !HAS_RESIZEBAR(wwin))
2525 goto alt_code;
2527 /* Get array of rectangles describing the shape mask */
2528 rects = XShapeGetRectangles(dpy, wwin->client_win, ShapeBounding, &count, &ordering);
2529 if (!rects)
2530 goto alt_code;
2532 urec = malloc(sizeof(XRectangle) * (count + 2));
2533 if (!urec) {
2534 XFree(rects);
2535 goto alt_code;
2538 /* insert our decoration rectangles in the rect list */
2539 memcpy(urec, rects, sizeof(XRectangle) * count);
2540 XFree(rects);
2542 if (HAS_TITLEBAR(wwin)) {
2543 urec[count].x = -1;
2544 urec[count].y = -1 - wwin->frame->top_width;
2545 urec[count].width = wwin->frame->core->width + 2;
2546 urec[count].height = wwin->frame->top_width + 1;
2547 count++;
2549 if (HAS_RESIZEBAR(wwin)) {
2550 urec[count].x = -1;
2551 urec[count].y = wwin->frame->core->height - wwin->frame->bottom_width - wwin->frame->top_width;
2552 urec[count].width = wwin->frame->core->width + 2;
2553 urec[count].height = wwin->frame->bottom_width + 1;
2554 count++;
2557 /* shape our frame window */
2558 XShapeCombineRectangles(dpy, wwin->frame->core->window, ShapeBounding,
2559 0, wwin->frame->top_width, urec, count, ShapeSet, Unsorted);
2560 XFlush(dpy);
2561 wfree(urec);
2562 return;
2564 alt_code:
2565 #endif /* OPTIMIZE_SHAPE */
2566 count = 0;
2567 if (HAS_TITLEBAR(wwin)) {
2568 rect[count].x = -1;
2569 rect[count].y = -1;
2570 rect[count].width = wwin->frame->core->width + 2;
2571 rect[count].height = wwin->frame->top_width + 1;
2572 count++;
2574 if (HAS_RESIZEBAR(wwin)) {
2575 rect[count].x = -1;
2576 rect[count].y = wwin->frame->core->height - wwin->frame->bottom_width;
2577 rect[count].width = wwin->frame->core->width + 2;
2578 rect[count].height = wwin->frame->bottom_width + 1;
2579 count++;
2581 if (count > 0) {
2582 XShapeCombineRectangles(dpy, wwin->frame->core->window, ShapeBounding,
2583 0, 0, rect, count, ShapeSet, Unsorted);
2585 XShapeCombineShape(dpy, wwin->frame->core->window, ShapeBounding,
2586 0, wwin->frame->top_width, wwin->client_win,
2587 ShapeBounding, (count > 0 ? ShapeUnion : ShapeSet));
2588 XFlush(dpy);
2590 #endif /* USE_XSHAPE */
2592 /* ====================================================================== */
2594 static FocusMode getFocusMode(WWindow * wwin)
2596 FocusMode mode;
2598 if ((wwin->wm_hints) && (wwin->wm_hints->flags & InputHint)) {
2599 if (wwin->wm_hints->input == True) {
2600 if (wwin->protocols.TAKE_FOCUS)
2601 mode = WFM_LOCALLY_ACTIVE;
2602 else
2603 mode = WFM_PASSIVE;
2604 } else {
2605 if (wwin->protocols.TAKE_FOCUS)
2606 mode = WFM_GLOBALLY_ACTIVE;
2607 else
2608 mode = WFM_NO_INPUT;
2610 } else {
2611 mode = WFM_PASSIVE;
2613 return mode;
2616 void wWindowSetKeyGrabs(WWindow * wwin)
2618 int i;
2619 WShortKey *key;
2621 for (i = 0; i < WKBD_LAST; i++) {
2622 key = &wKeyBindings[i];
2624 if (key->keycode == 0)
2625 continue;
2626 if (key->modifier != AnyModifier) {
2627 XGrabKey(dpy, key->keycode, key->modifier | LockMask,
2628 wwin->frame->core->window, True, GrabModeAsync, GrabModeAsync);
2629 #ifdef NUMLOCK_HACK
2630 /* Also grab all modifier combinations possible that include,
2631 * LockMask, ScrollLockMask and NumLockMask, so that keygrabs
2632 * work even if the NumLock/ScrollLock key is on.
2634 wHackedGrabKey(key->keycode, key->modifier,
2635 wwin->frame->core->window, True, GrabModeAsync, GrabModeAsync);
2636 #endif
2638 XGrabKey(dpy, key->keycode, key->modifier,
2639 wwin->frame->core->window, True, GrabModeAsync, GrabModeAsync);
2642 wRootMenuBindShortcuts(wwin->frame->core->window);
2645 void wWindowResetMouseGrabs(WWindow * wwin)
2647 /* Mouse grabs can't be done on the client window because of
2648 * ICCCM and because clients that try to do the same will crash.
2650 * But there is a problem which makes tbar buttons of unfocused
2651 * windows not usable as the click goes to the frame window instead
2652 * of the button itself. Must figure a way to fix that.
2655 XUngrabButton(dpy, AnyButton, AnyModifier, wwin->client_win);
2657 if (!WFLAGP(wwin, no_bind_mouse)) {
2658 /* grabs for Meta+drag */
2659 wHackedGrabButton(AnyButton, MOD_MASK, wwin->client_win,
2660 True, ButtonPressMask | ButtonReleaseMask,
2661 GrabModeSync, GrabModeAsync, None, None);
2663 /* for CTRL+Wheel to Scroll Horiz, we have to grab CTRL as well
2664 * but we only grab it for Button4 and Button 5 since a lot of apps
2665 * use CTRL+Button1-3 for app related functionality
2667 if (wPreferences.resize_increment > 0) {
2668 wHackedGrabButton(Button4, ControlMask, wwin->client_win,
2669 True, ButtonPressMask | ButtonReleaseMask,
2670 GrabModeSync, GrabModeAsync, None, None);
2671 wHackedGrabButton(Button5, ControlMask, wwin->client_win,
2672 True, ButtonPressMask | ButtonReleaseMask,
2673 GrabModeSync, GrabModeAsync, None, None);
2675 wHackedGrabButton(Button4, MOD_MASK | ControlMask, wwin->client_win,
2676 True, ButtonPressMask | ButtonReleaseMask,
2677 GrabModeSync, GrabModeAsync, None, None);
2678 wHackedGrabButton(Button5, MOD_MASK | ControlMask, wwin->client_win,
2679 True, ButtonPressMask | ButtonReleaseMask,
2680 GrabModeSync, GrabModeAsync, None, None);
2684 if (!wwin->flags.focused && !WFLAGP(wwin, no_focusable)
2685 && !wwin->flags.is_gnustep) {
2686 /* the passive grabs to focus the window */
2687 /* if (wPreferences.focus_mode == WKF_CLICK) */
2688 XGrabButton(dpy, AnyButton, AnyModifier, wwin->client_win,
2689 True, ButtonPressMask | ButtonReleaseMask, GrabModeSync, GrabModeAsync, None, None);
2691 XFlush(dpy);
2694 void wWindowUpdateGNUstepAttr(WWindow * wwin, GNUstepWMAttributes * attr)
2696 if (attr->flags & GSExtraFlagsAttr) {
2697 if (MGFLAGP(wwin, broken_close) != (attr->extra_flags & GSDocumentEditedFlag)) {
2698 wwin->client_flags.broken_close = !MGFLAGP(wwin, broken_close);
2699 wWindowUpdateButtonImages(wwin);
2704 WMagicNumber wWindowAddSavedState(const char *instance, const char *class,
2705 const char *command, pid_t pid, WSavedState *state)
2707 WWindowState *wstate;
2709 wstate = malloc(sizeof(WWindowState));
2710 if (!wstate)
2711 return NULL;
2713 memset(wstate, 0, sizeof(WWindowState));
2714 wstate->pid = pid;
2715 if (instance)
2716 wstate->instance = wstrdup(instance);
2717 if (class)
2718 wstate->class = wstrdup(class);
2719 if (command)
2720 wstate->command = wstrdup(command);
2721 wstate->state = state;
2723 wstate->next = windowState;
2724 windowState = wstate;
2726 return wstate;
2729 static inline int is_same(const char *x, const char *y)
2731 if ((x == NULL) && (y == NULL))
2732 return 1;
2734 if ((x == NULL) || (y == NULL))
2735 return 0;
2737 if (strcmp(x, y) == 0)
2738 return 1;
2739 else
2740 return 0;
2743 WMagicNumber wWindowGetSavedState(Window win)
2745 char *instance, *class, *command = NULL;
2746 WWindowState *wstate = windowState;
2748 if (!wstate)
2749 return NULL;
2751 command = GetCommandForWindow(win);
2752 if (!command)
2753 return NULL;
2755 if (PropGetWMClass(win, &class, &instance)) {
2756 while (wstate) {
2757 if (is_same(instance, wstate->instance) &&
2758 is_same(class, wstate->class) &&
2759 is_same(command, wstate->command)) {
2760 break;
2762 wstate = wstate->next;
2764 } else {
2765 wstate = NULL;
2768 if (command)
2769 wfree(command);
2770 if (instance)
2771 free(instance);
2772 if (class)
2773 free(class);
2775 return wstate;
2778 void wWindowDeleteSavedState(WMagicNumber id)
2780 WWindowState *tmp, *wstate = (WWindowState *) id;
2782 if (!wstate || !windowState)
2783 return;
2785 tmp = windowState;
2786 if (tmp == wstate) {
2787 windowState = wstate->next;
2788 release_wwindowstate(wstate);
2789 } else {
2790 while (tmp->next) {
2791 if (tmp->next == wstate) {
2792 tmp->next = wstate->next;
2793 release_wwindowstate(wstate);
2794 break;
2796 tmp = tmp->next;
2801 void wWindowDeleteSavedStatesForPID(pid_t pid)
2803 WWindowState *tmp, *wstate;
2805 if (!windowState)
2806 return;
2808 tmp = windowState;
2809 if (tmp->pid == pid) {
2810 wstate = windowState;
2811 windowState = tmp->next;
2813 release_wwindowstate(wstate);
2814 } else {
2815 while (tmp->next) {
2816 if (tmp->next->pid == pid) {
2817 wstate = tmp->next;
2818 tmp->next = wstate->next;
2819 release_wwindowstate(wstate);
2820 break;
2822 tmp = tmp->next;
2827 static void release_wwindowstate(WWindowState *wstate)
2829 if (wstate->instance)
2830 wfree(wstate->instance);
2832 if (wstate->class)
2833 wfree(wstate->class);
2835 if (wstate->command)
2836 wfree(wstate->command);
2838 wfree(wstate->state);
2839 wfree(wstate);
2842 void wWindowSetOmnipresent(WWindow *wwin, Bool flag)
2844 if (wwin->flags.omnipresent == flag)
2845 return;
2847 wwin->flags.omnipresent = flag;
2848 WMPostNotificationName(WMNChangedState, wwin, "omnipresent");
2851 static void resizebarMouseDown(WCoreWindow *sender, void *data, XEvent *event)
2853 WWindow *wwin = data;
2855 /* Parameter not used, but tell the compiler that it is ok */
2856 (void) sender;
2858 #ifndef NUMLOCK_HACK
2859 if ((event->xbutton.state & ValidModMask)
2860 != (event->xbutton.state & ~LockMask)) {
2861 wwarning(_("The NumLock, ScrollLock or similar key seems to be turned on. "
2862 "Turn it off or some mouse actions and keyboard shortcuts will not work."));
2864 #endif
2866 event->xbutton.state &= w_global.shortcut.modifiers_mask;
2868 CloseWindowMenu(wwin->screen_ptr);
2870 if (wPreferences.focus_mode == WKF_CLICK && !(event->xbutton.state & ControlMask)
2871 && !WFLAGP(wwin, no_focusable)) {
2872 wSetFocusTo(wwin->screen_ptr, wwin);
2875 if (event->xbutton.button == Button1)
2876 wRaiseFrame(wwin->frame->core);
2878 if (event->xbutton.window != wwin->frame->resizebar->window) {
2879 if (XGrabPointer(dpy, wwin->frame->resizebar->window, True,
2880 ButtonMotionMask | ButtonReleaseMask | ButtonPressMask,
2881 GrabModeAsync, GrabModeAsync, None, None, CurrentTime) != GrabSuccess) {
2882 return;
2886 if (event->xbutton.state & MOD_MASK) {
2887 /* move the window */
2888 wMouseMoveWindow(wwin, event);
2889 XUngrabPointer(dpy, CurrentTime);
2890 } else {
2891 wMouseResizeWindow(wwin, event);
2892 XUngrabPointer(dpy, CurrentTime);
2896 static void titlebarDblClick(WCoreWindow *sender, void *data, XEvent *event)
2898 WWindow *wwin = data;
2900 /* Parameter not used, but tell the compiler that it is ok */
2901 (void) sender;
2903 event->xbutton.state &= w_global.shortcut.modifiers_mask;
2905 if (event->xbutton.button == Button1 ) {
2906 if (event->xbutton.state == 0 ) {
2907 if (!WFLAGP(wwin, no_shadeable) & !wPreferences.double_click_fullscreen) {
2908 /* shade window */
2909 if (wwin->flags.shaded)
2910 wUnshadeWindow(wwin);
2911 else
2912 wShadeWindow(wwin);
2916 if (wPreferences.double_click_fullscreen) {
2917 int dir = 0;
2919 if (event->xbutton.state == 0) {
2920 /* maximize window full screen*/
2921 dir |= (MAX_VERTICAL|MAX_HORIZONTAL);
2922 int ndir = dir ^ wwin->flags.maximized;
2923 if (ndir != 0)
2924 wMaximizeWindow(wwin, ndir, wGetHeadForWindow(wwin));
2925 else
2926 wUnmaximizeWindow(wwin);
2928 } else {
2929 int dir = 0;
2931 if (event->xbutton.state & ControlMask)
2932 dir |= MAX_VERTICAL;
2934 if (event->xbutton.state & ShiftMask) {
2935 dir |= MAX_HORIZONTAL;
2936 if (!(event->xbutton.state & ControlMask))
2937 wSelectWindow(wwin, !wwin->flags.selected);
2940 /* maximize window */
2941 if (dir != 0 && IS_RESIZABLE(wwin)) {
2942 int ndir = dir ^ wwin->flags.maximized;
2944 if (ndir != 0)
2945 wMaximizeWindow(wwin, ndir, wGetHeadForWindow(wwin));
2946 else
2947 wUnmaximizeWindow(wwin);
2950 } else if (event->xbutton.button == Button3) {
2951 if (event->xbutton.state & MOD_MASK)
2952 wHideOtherApplications(wwin);
2953 } else if (event->xbutton.button == Button2) {
2954 wSelectWindow(wwin, !wwin->flags.selected);
2955 } else if (event->xbutton.button == W_getconf_mouseWheelUp()) {
2956 wShadeWindow(wwin);
2957 } else if (event->xbutton.button == W_getconf_mouseWheelDown()) {
2958 wUnshadeWindow(wwin);
2962 static void frameMouseDown(WObjDescriptor *desc, XEvent *event)
2964 WWindow *wwin = desc->parent;
2965 unsigned int new_width, w_scale;
2966 unsigned int new_height, h_scale;
2967 unsigned int resize_width_increment = 0;
2968 unsigned int resize_height_increment = 0;
2970 if (wwin->normal_hints) {
2971 w_scale = (wPreferences.resize_increment + wwin->normal_hints->width_inc - 1) / wwin->normal_hints->width_inc;
2972 h_scale = (wPreferences.resize_increment + wwin->normal_hints->height_inc - 1) / wwin->normal_hints->height_inc;
2973 resize_width_increment = wwin->normal_hints->width_inc * w_scale;
2974 resize_height_increment = wwin->normal_hints->height_inc * h_scale;
2976 if (resize_width_increment <= 1 && resize_height_increment <= 1) {
2977 resize_width_increment = wPreferences.resize_increment;
2978 resize_height_increment = wPreferences.resize_increment;
2981 event->xbutton.state &= w_global.shortcut.modifiers_mask;
2983 CloseWindowMenu(wwin->screen_ptr);
2985 if (!(event->xbutton.state & ControlMask) && !WFLAGP(wwin, no_focusable))
2986 wSetFocusTo(wwin->screen_ptr, wwin);
2988 if (event->xbutton.button == Button1)
2989 wRaiseFrame(wwin->frame->core);
2991 if (event->xbutton.state & ControlMask) {
2992 if (event->xbutton.button == Button4) {
2993 new_width = wwin->client.width - resize_width_increment;
2994 wWindowConstrainSize(wwin, &new_width, &wwin->client.height);
2995 wWindowConfigure(wwin, wwin->frame_x, wwin->frame_y, new_width, wwin->client.height);
2997 if (event->xbutton.button == Button5) {
2998 new_width = wwin->client.width + resize_width_increment;
2999 wWindowConstrainSize(wwin, &new_width, &wwin->client.height);
3000 wWindowConfigure(wwin, wwin->frame_x, wwin->frame_y, new_width, wwin->client.height);
3004 if (event->xbutton.state & MOD_MASK) {
3005 /* move the window */
3006 if (XGrabPointer(dpy, wwin->client_win, False,
3007 ButtonMotionMask | ButtonReleaseMask | ButtonPressMask,
3008 GrabModeAsync, GrabModeAsync, None, None, CurrentTime) != GrabSuccess) {
3009 return;
3011 if (event->xbutton.button == Button3) {
3012 wMouseResizeWindow(wwin, event);
3013 } else if (event->xbutton.button == Button4) {
3014 new_height = wwin->client.height - resize_height_increment;
3015 wWindowConstrainSize(wwin, &wwin->client.width, &new_height);
3016 wWindowConfigure(wwin, wwin->frame_x, wwin->frame_y, wwin->client.width, new_height);
3017 } else if (event->xbutton.button == Button5) {
3018 new_height = wwin->client.height + resize_height_increment;
3019 wWindowConstrainSize(wwin, &wwin->client.width, &new_height);
3020 wWindowConfigure(wwin, wwin->frame_x, wwin->frame_y, wwin->client.width, new_height);
3021 } else if (event->xbutton.button == Button1 || event->xbutton.button == Button2) {
3022 wMouseMoveWindow(wwin, event);
3024 XUngrabPointer(dpy, CurrentTime);
3028 static void titlebarMouseDown(WCoreWindow *sender, void *data, XEvent *event)
3030 WWindow *wwin = (WWindow *) data;
3032 /* Parameter not used, but tell the compiler that it is ok */
3033 (void) sender;
3035 #ifndef NUMLOCK_HACK
3036 if ((event->xbutton.state & ValidModMask) != (event->xbutton.state & ~LockMask))
3037 wwarning(_("The NumLock, ScrollLock or similar key seems to be turned on. "
3038 "Turn it off or some mouse actions and keyboard shortcuts will not work."));
3039 #endif
3040 event->xbutton.state &= w_global.shortcut.modifiers_mask;
3042 CloseWindowMenu(wwin->screen_ptr);
3044 if (wPreferences.focus_mode == WKF_CLICK && !(event->xbutton.state & ControlMask)
3045 && !WFLAGP(wwin, no_focusable))
3046 wSetFocusTo(wwin->screen_ptr, wwin);
3048 if (event->xbutton.button == Button1 || event->xbutton.button == Button2) {
3050 if (event->xbutton.button == Button1) {
3051 if (event->xbutton.state & MOD_MASK)
3052 wLowerFrame(wwin->frame->core);
3053 else
3054 wRaiseFrame(wwin->frame->core);
3056 if ((event->xbutton.state & ShiftMask)
3057 && !(event->xbutton.state & ControlMask)) {
3058 wSelectWindow(wwin, !wwin->flags.selected);
3059 return;
3061 if (event->xbutton.window != wwin->frame->titlebar->window
3062 && XGrabPointer(dpy, wwin->frame->titlebar->window, False,
3063 ButtonMotionMask | ButtonReleaseMask | ButtonPressMask,
3064 GrabModeAsync, GrabModeAsync, None, None, CurrentTime) != GrabSuccess) {
3065 return;
3068 /* move the window */
3069 wMouseMoveWindow(wwin, event);
3071 XUngrabPointer(dpy, CurrentTime);
3072 } else if (event->xbutton.button == Button3 && event->xbutton.state == 0
3073 && !wwin->flags.internal_window && !WCHECK_STATE(WSTATE_MODAL)) {
3074 WObjDescriptor *desc;
3076 if (event->xbutton.window != wwin->frame->titlebar->window
3077 && XGrabPointer(dpy, wwin->frame->titlebar->window, False,
3078 ButtonMotionMask | ButtonReleaseMask | ButtonPressMask,
3079 GrabModeAsync, GrabModeAsync, None, None, CurrentTime) != GrabSuccess) {
3080 return;
3083 OpenWindowMenu(wwin, event->xbutton.x_root, wwin->frame_y + wwin->frame->top_width, False);
3085 /* allow drag select */
3086 desc = &wwin->screen_ptr->window_menu->menu->descriptor;
3087 event->xany.send_event = True;
3088 (*desc->handle_mousedown) (desc, event);
3090 XUngrabPointer(dpy, CurrentTime);
3094 static void windowCloseClick(WCoreWindow *sender, void *data, XEvent *event)
3096 WWindow *wwin = data;
3098 /* Parameter not used, but tell the compiler that it is ok */
3099 (void) sender;
3101 event->xbutton.state &= w_global.shortcut.modifiers_mask;
3103 CloseWindowMenu(wwin->screen_ptr);
3105 if (event->xbutton.button < Button1 || event->xbutton.button > Button3)
3106 return;
3108 /* if control-click, kill the client */
3109 if (event->xbutton.state & ControlMask) {
3110 wClientKill(wwin);
3111 } else {
3112 if (wwin->protocols.DELETE_WINDOW && event->xbutton.state == 0) {
3113 /* send delete message */
3114 wClientSendProtocol(wwin, w_global.atom.wm.delete_window,
3115 w_global.timestamp.last_event);
3120 static void windowCloseDblClick(WCoreWindow *sender, void *data, XEvent *event)
3122 WWindow *wwin = data;
3124 /* Parameter not used, but tell the compiler that it is ok */
3125 (void) sender;
3127 CloseWindowMenu(wwin->screen_ptr);
3129 if (event->xbutton.button < Button1 || event->xbutton.button > Button3)
3130 return;
3132 /* send delete message */
3133 if (wwin->protocols.DELETE_WINDOW)
3134 wClientSendProtocol(wwin, w_global.atom.wm.delete_window,
3135 w_global.timestamp.last_event);
3136 else
3137 wClientKill(wwin);
3140 #ifdef XKB_BUTTON_HINT
3141 static void windowLanguageClick(WCoreWindow *sender, void *data, XEvent *event)
3143 WWindow *wwin = data;
3144 WFrameWindow *fwin = wwin->frame;
3145 WScreen *scr = fwin->screen_ptr;
3146 int tl;
3148 /* Parameter not used, but tell the compiler that it is ok */
3149 (void) sender;
3151 if (event->xbutton.button != Button1 && event->xbutton.button != Button3)
3152 return;
3153 tl = wwin->frame->languagemode;
3154 wwin->frame->languagemode = wwin->frame->last_languagemode;
3155 wwin->frame->last_languagemode = tl;
3156 wSetFocusTo(scr, wwin);
3157 wwin->frame->languagebutton_image =
3158 wwin->frame->screen_ptr->b_pixmaps[WBUT_XKBGROUP1 + wwin->frame->languagemode];
3159 wFrameWindowUpdateLanguageButton(wwin->frame);
3160 if (event->xbutton.button == Button3)
3161 return;
3162 wRaiseFrame(fwin->core);
3164 #endif
3166 static void windowIconifyClick(WCoreWindow *sender, void *data, XEvent *event)
3168 WWindow *wwin = data;
3170 /* Parameter not used, but tell the compiler that it is ok */
3171 (void) sender;
3173 event->xbutton.state &= w_global.shortcut.modifiers_mask;
3175 CloseWindowMenu(wwin->screen_ptr);
3177 if (event->xbutton.button < Button1 || event->xbutton.button > Button3)
3178 return;
3180 if (wwin->protocols.MINIATURIZE_WINDOW && event->xbutton.state == 0) {
3181 wClientSendProtocol(wwin, w_global.atom.gnustep.wm_miniaturize_window,
3182 w_global.timestamp.last_event);
3183 } else {
3184 WApplication *wapp;
3185 if ((event->xbutton.state & ControlMask) || (event->xbutton.button == Button3)) {
3187 wapp = wApplicationOf(wwin->main_window);
3188 if (wapp && !WFLAGP(wwin, no_appicon))
3189 wHideApplication(wapp);
3190 } else if (event->xbutton.state == 0) {
3191 wIconifyWindow(wwin);