Skip to content
Snippets Groups Projects
events.c 40.8 KiB
Newer Older
Olivier Fourdan's avatar
Olivier Fourdan committed
/*
        This program is free software; you can redistribute it and/or modify
        it under the terms of the GNU General Public License as published by
        the Free Software Foundation; You may only use version 2 of the License,
        you have no option to use any other version.
Olivier Fourdan's avatar
Olivier Fourdan committed

        This program is distributed in the hope that it will be useful,
        but WITHOUT ANY WARRANTY; without even the implied warranty of
        MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
        GNU General Public License for more details.
Olivier Fourdan's avatar
Olivier Fourdan committed

        You should have received a copy of the GNU General Public License
        along with this program; if not, write to the Free Software
        Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
Olivier Fourdan's avatar
Olivier Fourdan committed

        oroborus - (c) 2001 Ken Lynch
        xfwm4    - (c) 2002 Olivier Fourdan

 */

#ifdef HAVE_CONFIG_H
#  include "config.h"
#endif

#include <X11/Xlib.h>
Olivier Fourdan's avatar
Olivier Fourdan committed
#include <gtk/gtk.h>
#include <gdk/gdk.h>
Olivier Fourdan's avatar
Olivier Fourdan committed
#include <gdk/gdkx.h>
#include <libxfcegui4/libxfcegui4.h>
Olivier Fourdan's avatar
Olivier Fourdan committed
#include "main.h"
#include "workspaces.h"
#include "settings.h"
Olivier Fourdan's avatar
Olivier Fourdan committed
#include "mywindow.h"
Olivier Fourdan's avatar
Olivier Fourdan committed
#include "frame.h"
#include "client.h"
#include "menu.h"
#include "startup_notification.h"
Olivier Fourdan's avatar
Olivier Fourdan committed
#include "debug.h"
Olivier Fourdan's avatar
Olivier Fourdan committed
#include "my_intl.h"
Olivier Fourdan's avatar
Olivier Fourdan committed

static guint raise_timeout = 0;
static gulong button_handler_id = 0;
static GdkAtom atom_rcfiles = GDK_NONE;
static Window menu_event_window = None;
Olivier Fourdan's avatar
Olivier Fourdan committed

static void menu_callback(Menu * menu, MenuOp op, Window client_xwindow, gpointer menu_data, gpointer item_data);
static gboolean show_popup_cb(GtkWidget * widget, GdkEventButton * ev, gpointer data);
static gboolean client_event_cb(GtkWidget * widget, GdkEventClient * ev);
Olivier Fourdan's avatar
Olivier Fourdan committed

typedef enum
{
    XFWM_BUTTON_UNDEFINED = 0,
    XFWM_BUTTON_DRAG = 1,
    XFWM_BUTTON_CLICK = 2,
    XFWM_BUTTON_CLICK_AND_DRAG = 3,
    XFWM_BUTTON_DOUBLE_CLICK = 4
}
XfwmButtonClickType;
static inline XfwmButtonClickType typeOfClick(Window w, XEvent * ev, gboolean allow_double_click)
    unsigned int button;
    g_return_val_if_fail(ev != NULL, XFWM_BUTTON_UNDEFINED);
    g_return_val_if_fail(w != None, XFWM_BUTTON_UNDEFINED);

    g = XGrabPointer(dpy, w, False, ButtonMotionMask | PointerMotionMask | PointerMotionHintMask | ButtonPressMask | ButtonReleaseMask, GrabModeAsync, GrabModeAsync, None, None, ev->xbutton.time);
    if(g != GrabSuccess)
    {
        DBG("grab failed in typeOfClick\n");
        gdk_beep();
        return XFWM_BUTTON_UNDEFINED;
    }
    button = ev->xbutton.button;
    x = xcurrent = ev->xbutton.x_root;
    y = ycurrent = ev->xbutton.y_root;
    while((ABS(x - xcurrent) < 1) && (ABS(y - ycurrent) < 1) && (total < params.dbl_click_time) && ((t1 - t0) < params.dbl_click_time))
        if(XCheckMaskEvent(dpy, ButtonReleaseMask | ButtonPressMask, ev))
            if(ev->xbutton.button == button)
            {
                clicks++;
            }
            t1 = ev->xbutton.time;
        if(XCheckMaskEvent(dpy, ButtonMotionMask | PointerMotionMask | PointerMotionHintMask, ev))
        {
            xcurrent = ev->xmotion.x_root;
            ycurrent = ev->xmotion.y_root;
        if((XfwmButtonClickType) clicks == XFWM_BUTTON_DOUBLE_CLICK || (!allow_double_click && (XfwmButtonClickType) clicks == XFWM_BUTTON_CLICK))
        {
            break;
        }
    }
    XUngrabPointer(dpy, ev->xbutton.time);
    return (XfwmButtonClickType) clicks;
}

Olivier Fourdan's avatar
Olivier Fourdan committed
{
Olivier Fourdan's avatar
Olivier Fourdan committed
    {
        gtk_timeout_remove(raise_timeout);
        raise_timeout = 0;
Olivier Fourdan's avatar
Olivier Fourdan committed
{
    Client *c;
    DBG("entering raise_cb\n");

Olivier Fourdan's avatar
Olivier Fourdan committed
    c = clientGetFocus();
    if(c)
    {
        clientRaise(c);
    }
    return (TRUE);
}

Olivier Fourdan's avatar
Olivier Fourdan committed
{
Olivier Fourdan's avatar
Olivier Fourdan committed
    {
Olivier Fourdan's avatar
Olivier Fourdan committed
    }
    raise_timeout = gtk_timeout_add(params.raise_delay, (GtkFunction) raise_cb, NULL);
static inline void _moveRequest(Client * c, XEvent * ev)
    if(CLIENT_FLAG_TEST_AND_NOT(c, CLIENT_FLAG_HAS_BORDER | CLIENT_FLAG_HAS_MOVE, CLIENT_FLAG_FULLSCREEN))
static inline void _resizeRequest(Client * c, int corner, XEvent * ev)
    if(CLIENT_FLAG_TEST_ALL(c, CLIENT_FLAG_HAS_RESIZE | CLIENT_FLAG_IS_RESIZABLE))
    else if(CLIENT_FLAG_TEST_AND_NOT(c, CLIENT_FLAG_HAS_BORDER | CLIENT_FLAG_HAS_MOVE, CLIENT_FLAG_FULLSCREEN))
static inline void spawn_shortcut(int i)
{
    GError *error = NULL;
    if((i >= NB_KEY_SHORTCUTS) || (!params.shortcut_exec[i]) || !strlen(params.shortcut_exec[i]))
    if(!g_spawn_command_line_async(params.shortcut_exec[i], &error))
        if(error)
        {
            g_warning("%s: %s", g_get_prgname(), error->message);
            g_error_free(error);
        }
static inline void handleKeyPress(XKeyEvent * ev)
Olivier Fourdan's avatar
Olivier Fourdan committed
{
    Client *c;
    int state, key;
    XEvent e;

    DBG("entering handleKeyEvent\n");

    c = clientGetFocus();
Olivier Fourdan's avatar
Olivier Fourdan committed
    state = ev->state & (ShiftMask | ControlMask | AltMask | MetaMask | SuperMask | HyperMask);
Olivier Fourdan's avatar
Olivier Fourdan committed
    for(key = 0; key < KEY_COUNT; key++)
    {
        if((params.keys[key].keycode == ev->keycode) && (params.keys[key].modifier == state))
Olivier Fourdan's avatar
Olivier Fourdan committed
            break;
Olivier Fourdan's avatar
Olivier Fourdan committed
    }
Olivier Fourdan's avatar
Olivier Fourdan committed
    if(c)
    {
        switch (key)
        {
            case KEY_MOVE_UP:
            case KEY_MOVE_DOWN:
            case KEY_MOVE_LEFT:
            case KEY_MOVE_RIGHT:
                _moveRequest(c, (XEvent *) ev);
Olivier Fourdan's avatar
Olivier Fourdan committed
                break;
            case KEY_RESIZE_UP:
            case KEY_RESIZE_DOWN:
            case KEY_RESIZE_LEFT:
            case KEY_RESIZE_RIGHT:
                if(CLIENT_FLAG_TEST_ALL(c, CLIENT_FLAG_HAS_RESIZE | CLIENT_FLAG_IS_RESIZABLE))
                    clientResize(c, CORNER_BOTTOM_RIGHT, (XEvent *) ev);
Olivier Fourdan's avatar
Olivier Fourdan committed
            case KEY_CYCLE_WINDOWS:
                clientCycle(c);
                break;
            case KEY_CLOSE_WINDOW:
                clientClose(c);
                break;
            case KEY_HIDE_WINDOW:
                if(CLIENT_CAN_HIDE_WINDOW(c))
Olivier Fourdan's avatar
Olivier Fourdan committed
            case KEY_MAXIMIZE_WINDOW:
                if(CLIENT_CAN_MAXIMIZE_WINDOW(c))
Olivier Fourdan's avatar
Olivier Fourdan committed
            case KEY_MAXIMIZE_VERT:
                if(CLIENT_CAN_MAXIMIZE_WINDOW(c))
Olivier Fourdan's avatar
Olivier Fourdan committed
            case KEY_MAXIMIZE_HORIZ:
                if(CLIENT_CAN_MAXIMIZE_WINDOW(c))
Olivier Fourdan's avatar
Olivier Fourdan committed
            case KEY_SHADE_WINDOW:
                clientToggleShaded(c);
                break;
            case KEY_STICK_WINDOW:
Olivier Fourdan's avatar
Olivier Fourdan committed
                if(CLIENT_FLAG_TEST(c, CLIENT_FLAG_HAS_STICK))
                {
                    clientToggleSticky(c, TRUE);
                }
Olivier Fourdan's avatar
Olivier Fourdan committed
                break;
Olivier Fourdan's avatar
Olivier Fourdan committed
            case KEY_MOVE_NEXT_WORKSPACE:
                workspaceSwitch(workspace + 1, c);
                break;
            case KEY_MOVE_PREV_WORKSPACE:
                workspaceSwitch(workspace - 1, c);
                break;
            case KEY_MOVE_WORKSPACE_1:
                workspaceSwitch(0, c);
                break;
            case KEY_MOVE_WORKSPACE_2:
                workspaceSwitch(1, c);
                break;
            case KEY_MOVE_WORKSPACE_3:
                workspaceSwitch(2, c);
                break;
            case KEY_MOVE_WORKSPACE_4:
                workspaceSwitch(3, c);
                break;
            case KEY_MOVE_WORKSPACE_5:
                workspaceSwitch(4, c);
                break;
            case KEY_MOVE_WORKSPACE_6:
                workspaceSwitch(5, c);
                break;
            case KEY_MOVE_WORKSPACE_7:
                workspaceSwitch(6, c);
                break;
            case KEY_MOVE_WORKSPACE_8:
                workspaceSwitch(7, c);
                break;
            case KEY_MOVE_WORKSPACE_9:
                workspaceSwitch(8, c);
                break;
Olivier Fourdan's avatar
Olivier Fourdan committed
        }
    }
    else
    {
        switch (key)
        {
            case KEY_CYCLE_WINDOWS:
Olivier Fourdan's avatar
Olivier Fourdan committed
                    clientCycle(clients->prev);
Olivier Fourdan's avatar
Olivier Fourdan committed
                break;
Olivier Fourdan's avatar
Olivier Fourdan committed
        }
    }
    switch (key)
    {
        case KEY_NEXT_WORKSPACE:
            workspaceSwitch(workspace + 1, NULL);
            break;
        case KEY_PREV_WORKSPACE:
            workspaceSwitch(workspace - 1, NULL);
            break;
        case KEY_ADD_WORKSPACE:
            workspaceSetCount(params.workspace_count + 1);
            break;
        case KEY_DEL_WORKSPACE:
            workspaceSetCount(params.workspace_count - 1);
            break;
        case KEY_WORKSPACE_1:
            workspaceSwitch(0, NULL);
            break;
        case KEY_WORKSPACE_2:
            workspaceSwitch(1, NULL);
            break;
        case KEY_WORKSPACE_3:
            workspaceSwitch(2, NULL);
            break;
        case KEY_WORKSPACE_4:
            workspaceSwitch(3, NULL);
            break;
        case KEY_WORKSPACE_5:
            workspaceSwitch(4, NULL);
            break;
        case KEY_WORKSPACE_6:
            workspaceSwitch(5, NULL);
            break;
        case KEY_WORKSPACE_7:
            workspaceSwitch(6, NULL);
            break;
        case KEY_WORKSPACE_8:
            workspaceSwitch(7, NULL);
            break;
        case KEY_WORKSPACE_9:
            workspaceSwitch(8, NULL);
            break;
        case KEY_SHORTCUT_1:
            break;
        case KEY_SHORTCUT_10:
Olivier Fourdan's avatar
Olivier Fourdan committed
    while(XCheckTypedEvent(dpy, EnterNotify, &e));
}

/* User has clicked on an edge or corner.
 * Button 1 : Raise and resize
 * Button 2 : Move
 * Button 3 : Resize
 */
static inline void _edgeButton(Client * c, int part, XButtonEvent * ev)
{
    if(ev->button == Button2)
    {
        XfwmButtonClickType tclick;

        tclick = typeOfClick(c->frame, (XEvent *) ev, FALSE);

        if(tclick == XFWM_BUTTON_CLICK)
        {
            clientLower(c);
        }
        else
        {
            _moveRequest(c, (XEvent *) ev);
        }
        if(ev->button == Button1)
            clientRaise(c);
        if(ev->button == Button1 || ev->button == Button3)
            _resizeRequest(c, part, (XEvent *) ev);
static inline void handleButtonPress(XButtonEvent * ev)
Olivier Fourdan's avatar
Olivier Fourdan committed
{
    Client *c;
    Window win;
    int state, replay = False;

    DBG("entering handleButtonPress\n");

    /* Clear timeout */
Olivier Fourdan's avatar
Olivier Fourdan committed

Olivier Fourdan's avatar
Olivier Fourdan committed
    if(c)
    {
Olivier Fourdan's avatar
Olivier Fourdan committed
        state = ev->state & (ShiftMask | ControlMask | AltMask | MetaMask | SuperMask | HyperMask);
        win = ev->subwindow;
Olivier Fourdan's avatar
Olivier Fourdan committed
        if((win == MYWINDOW_XWINDOW(c->buttons[HIDE_BUTTON]) || (win == MYWINDOW_XWINDOW(c->buttons[CLOSE_BUTTON])) || (win == MYWINDOW_XWINDOW(c->buttons[MAXIMIZE_BUTTON])) || (win == MYWINDOW_XWINDOW(c->buttons[SHADE_BUTTON])) || (win == MYWINDOW_XWINDOW(c->buttons[STICK_BUTTON]))))
Olivier Fourdan's avatar
Olivier Fourdan committed
        {
            if(params.raise_on_click)
            {
Olivier Fourdan's avatar
Olivier Fourdan committed
            clientButtonPress(c, win, ev);
        }
Olivier Fourdan's avatar
Olivier Fourdan committed
        else if(((win == MYWINDOW_XWINDOW(c->title)) && (ev->button == Button3)) || ((win == MYWINDOW_XWINDOW(c->buttons[MENU_BUTTON])) && (ev->button == Button1)))
Olivier Fourdan's avatar
Olivier Fourdan committed
        {
            /*
               We need to copy the event to keep the original event untouched
               for gtk to handle it (in case we open up the menu)
             */
            XfwmButtonClickType tclick;
Olivier Fourdan's avatar
Olivier Fourdan committed
            tclick = typeOfClick(c->frame, &copy_event, win == MYWINDOW_XWINDOW(c->buttons[MENU_BUTTON]));

            if(tclick == XFWM_BUTTON_DOUBLE_CLICK)
            {
                clientClose(c);
            }
            else if(ev->button == Button3 && (tclick == XFWM_BUTTON_DRAG))
                _moveRequest(c, (XEvent *) ev);
            else
            {
                clientSetFocus(c, True);
                if(params.raise_on_click)
                {
                ev->window = ev->root;
                if(button_handler_id)
                {
                    g_signal_handler_disconnect(GTK_OBJECT(getDefaultGtkWidget()), button_handler_id);
                }
                button_handler_id = g_signal_connect(GTK_OBJECT(getDefaultGtkWidget()), "button_press_event", GTK_SIGNAL_FUNC(show_popup_cb), (gpointer) c);
                /* Let GTK handle this for us. */
            }
Olivier Fourdan's avatar
Olivier Fourdan committed
        }
Olivier Fourdan's avatar
Olivier Fourdan committed
        else if(((win == MYWINDOW_XWINDOW(c->title)) && ((ev->button == Button1) && (state == 0))) || ((ev->button == Button1) && (state == AltMask)))
Olivier Fourdan's avatar
Olivier Fourdan committed
        {
            XfwmButtonClickType tclick;
            clientSetFocus(c, True);
            clientRaise(c);
            tclick = typeOfClick(c->frame, &copy_event, TRUE);
            if((tclick == XFWM_BUTTON_DRAG) || (tclick == XFWM_BUTTON_CLICK_AND_DRAG))
                _moveRequest(c, (XEvent *) ev);
            else if(tclick == XFWM_BUTTON_DOUBLE_CLICK)
Olivier Fourdan's avatar
Olivier Fourdan committed
            {
                switch (params.double_click_action)
Olivier Fourdan's avatar
Olivier Fourdan committed
                {
                    case ACTION_MAXIMIZE:
                        if(CLIENT_CAN_MAXIMIZE_WINDOW(c))
Olivier Fourdan's avatar
Olivier Fourdan committed
                    case ACTION_SHADE:
                        clientToggleShaded(c);
                        break;
                    case ACTION_HIDE:
                        if(CLIENT_CAN_HIDE_WINDOW(c))
Olivier Fourdan's avatar
Olivier Fourdan committed
        else if((win == MYWINDOW_XWINDOW(c->corners[CORNER_TOP_LEFT])) && (state == 0))
Olivier Fourdan's avatar
Olivier Fourdan committed
        {
Olivier Fourdan's avatar
Olivier Fourdan committed
        }
Olivier Fourdan's avatar
Olivier Fourdan committed
        else if((win == MYWINDOW_XWINDOW(c->corners[CORNER_TOP_RIGHT])) && (state == 0))
Olivier Fourdan's avatar
Olivier Fourdan committed
        {
            _edgeButton(c, CORNER_TOP_RIGHT, ev);
Olivier Fourdan's avatar
Olivier Fourdan committed
        }
Olivier Fourdan's avatar
Olivier Fourdan committed
        else if((win == MYWINDOW_XWINDOW(c->corners[CORNER_BOTTOM_LEFT])) && (state == 0))
Olivier Fourdan's avatar
Olivier Fourdan committed
        {
            _edgeButton(c, CORNER_BOTTOM_LEFT, ev);
Olivier Fourdan's avatar
Olivier Fourdan committed
        }
Olivier Fourdan's avatar
Olivier Fourdan committed
        else if((win == MYWINDOW_XWINDOW(c->corners[CORNER_BOTTOM_RIGHT])) && (state == 0))
Olivier Fourdan's avatar
Olivier Fourdan committed
        {
            _edgeButton(c, CORNER_BOTTOM_RIGHT, ev);
Olivier Fourdan's avatar
Olivier Fourdan committed
        }
Olivier Fourdan's avatar
Olivier Fourdan committed
        else if((win == MYWINDOW_XWINDOW(c->sides[SIDE_BOTTOM])) && (state == 0))
Olivier Fourdan's avatar
Olivier Fourdan committed
        {
            _edgeButton(c, 4 + SIDE_BOTTOM, ev);
Olivier Fourdan's avatar
Olivier Fourdan committed
        }
Olivier Fourdan's avatar
Olivier Fourdan committed
        else if((win == MYWINDOW_XWINDOW(c->sides[SIDE_LEFT])) && (state == 0))
Olivier Fourdan's avatar
Olivier Fourdan committed
        {
            _edgeButton(c, 4 + SIDE_LEFT, ev);
Olivier Fourdan's avatar
Olivier Fourdan committed
        }
Olivier Fourdan's avatar
Olivier Fourdan committed
        else if((win == MYWINDOW_XWINDOW(c->sides[SIDE_RIGHT])) && (state == 0))
Olivier Fourdan's avatar
Olivier Fourdan committed
        {
            _edgeButton(c, 4 + SIDE_RIGHT, ev);
Olivier Fourdan's avatar
Olivier Fourdan committed
        }
        else if(((ev->window != c->window) && (ev->button == Button2) && (state == 0)) || ((ev->button == Button2) && (state == (AltMask | ControlMask))))
Olivier Fourdan's avatar
Olivier Fourdan committed
        {
Olivier Fourdan's avatar
Olivier Fourdan committed
        }
Olivier Fourdan's avatar
Olivier Fourdan committed
        {
            {
                clientSetFocus(c, True);
                if(params.raise_on_click)
            if(ev->window == c->window)
Olivier Fourdan's avatar
Olivier Fourdan committed
                replay = True;
            }
Olivier Fourdan's avatar
Olivier Fourdan committed

        if(replay)
            XAllowEvents(dpy, ReplayPointer, ev->time);
Olivier Fourdan's avatar
Olivier Fourdan committed
        }
Olivier Fourdan's avatar
Olivier Fourdan committed
        {
            XAllowEvents(dpy, SyncPointer, ev->time);
Olivier Fourdan's avatar
Olivier Fourdan committed
        }
    }
    else
    {
        XUngrabPointer(dpy, CurrentTime);
        XSendEvent(dpy, gnome_win, False, SubstructureNotifyMask, (XEvent *) ev);
    }
}

static inline void handleButtonRelease(XButtonEvent * ev)
Olivier Fourdan's avatar
Olivier Fourdan committed
{
    DBG("entering handleButtonRelease\n");

    XSendEvent(dpy, gnome_win, False, SubstructureNotifyMask, (XEvent *) ev);
}

static inline void handleDestroyNotify(XDestroyWindowEvent * ev)
Olivier Fourdan's avatar
Olivier Fourdan committed
{
    Client *c;

    DBG("entering handleDestroyNotify\n");

    c = clientGetFromWindow(ev->window, WINDOW);
    if(c)
    {
        clientUnframe(c, False);
Olivier Fourdan's avatar
Olivier Fourdan committed
        if(clients)
Olivier Fourdan's avatar
Olivier Fourdan committed
            clientSetFocus(clientGetNext(clients->prev, 0), True);
        }
Olivier Fourdan's avatar
Olivier Fourdan committed
        {
static inline void handleUnmapNotify(XUnmapEvent * ev)
Olivier Fourdan's avatar
Olivier Fourdan committed
{
    Client *c;

    DBG("entering handleUnmapNotify\n");

    c = clientGetFromWindow(ev->window, WINDOW);
    if(c)
    {
        if(c->ignore_unmap)
Olivier Fourdan's avatar
Olivier Fourdan committed
            c->ignore_unmap--;
        }
Olivier Fourdan's avatar
Olivier Fourdan committed
        {
            clientUnframe(c, False);
            if(clients)
Olivier Fourdan's avatar
Olivier Fourdan committed
                clientSetFocus(clientGetNext(clients->prev, 0), True);
            }
Olivier Fourdan's avatar
Olivier Fourdan committed
                clientSetFocus(NULL, True);
static inline void handleMapRequest(XMapRequestEvent * ev)
Olivier Fourdan's avatar
Olivier Fourdan committed
{
    Client *c;

    DBG("entering handleMapRequest\n");

Olivier Fourdan's avatar
Olivier Fourdan committed
    {
        DBG("Mapping None ???\n");
        return;
    }
    c = clientGetFromWindow(ev->window, WINDOW);
    if(c)
    {
        clientShow(c, True);
    }
    else
    {
        clientFrame(ev->window);
    }
}

static inline void handleConfigureRequest(XConfigureRequestEvent * ev)
Olivier Fourdan's avatar
Olivier Fourdan committed
{
    Client *c;
    XWindowChanges wc;
Olivier Fourdan's avatar
Olivier Fourdan committed
    XEvent otherEvent;
Olivier Fourdan's avatar
Olivier Fourdan committed

    DBG("entering handleConfigureRequest\n");

Olivier Fourdan's avatar
Olivier Fourdan committed
    /* Compress events - logic taken from kwin */
    while(XCheckTypedWindowEvent(dpy, ev->window, ConfigureRequest, &otherEvent))
Olivier Fourdan's avatar
Olivier Fourdan committed
    {
        if(otherEvent.xconfigurerequest.value_mask == ev->value_mask)
        {
Olivier Fourdan's avatar
Olivier Fourdan committed
            ev = &otherEvent.xconfigurerequest;
        }
Olivier Fourdan's avatar
Olivier Fourdan committed
            XPutBackEvent(dpy, &otherEvent);
            break;
        }
    }

Olivier Fourdan's avatar
Olivier Fourdan committed
    wc.x = ev->x;
    wc.y = ev->y;
    wc.width = ev->width;
    wc.height = ev->height;
    wc.sibling = ev->above;
    wc.stack_mode = ev->detail;
    wc.border_width = ev->border_width;

    c = clientGetFromWindow(ev->window, WINDOW);
    if (!c)
    {
        /* Some app tend or try to manipulate the wm frame to achieve fullscreen mode */
	c = clientGetFromWindow(ev->window, FRAME);
	if (c)
	{
            if(ev->value_mask & CWX)
            {
		wc.x += frameLeft(c);
            }
            if(ev->value_mask & CWY)
            {
		wc.y += frameTop(c);
            }
            if(ev->value_mask & CWWidth)
            {
		wc.width -= frameLeft(c) + frameRight(c);
            }
            if(ev->value_mask & CWHeight)
            {
		wc.height -= frameTop(c) + frameBottom(c);
            }
	}
    }
Olivier Fourdan's avatar
Olivier Fourdan committed
    if(c)
    {
        gboolean constrained = FALSE;
        
        DBG("handleConfigureRequest managed window \"%s\" (%#lx)\n", c->name, c->window);
	if(CLIENT_FLAG_TEST(c, CLIENT_FLAG_MOVING | CLIENT_FLAG_RESIZING))
        {
            /* Sorry, but it's not the right time for configure request */
            return;
        }
        if(c->type == WINDOW_DESKTOP)
        {
            /* Ignore stacking request for DESKTOP windows */
            ev->value_mask &= ~CWStackMode;
        clientCoordGravitate(c, APPLY, &wc.x, &wc.y);
        if(ev->value_mask & (CWX | CWY | CWWidth | CWHeight))
            if(CLIENT_FLAG_TEST(c, CLIENT_FLAG_MAXIMIZED))
	    {
	        clientRemoveMaximizeFlag(c);
	    }
	    constrained = TRUE;
        /* Let's say that if the client performs a XRaiseWindow, we show the window if hidden */
        if((ev->value_mask & CWStackMode) && (wc.stack_mode == Above) && (CLIENT_FLAG_TEST(c, CLIENT_FLAG_HIDDEN)))
        {
            clientShow(c, True);
            if(params.focus_new && clientAcceptFocus(c))
            {
        clientConfigure(c, &wc, ev->value_mask, constrained);
Olivier Fourdan's avatar
Olivier Fourdan committed
    }
    else
    {
        XConfigureWindow(dpy, ev->window, ev->value_mask, &wc);
    }
}

static inline void handleEnterNotify(XCrossingEvent * ev)
Olivier Fourdan's avatar
Olivier Fourdan committed
{
    Client *c;

    DBG("entering handleEnterNotify\n");

    while(XCheckTypedEvent(dpy, EnterNotify, (XEvent *) ev));

    DBG("EnterNotify window is (%#lx)\n", ev->window);

    c = clientGetFromWindow(ev->window, FRAME);
    if(c && !(params.click_to_focus) && (clientAcceptFocus(c)))
Olivier Fourdan's avatar
Olivier Fourdan committed
    {
        DBG("EnterNotify window is \"%s\"\n", c->name);
        if((c->type != WINDOW_DOCK) && (c->type != WINDOW_DESKTOP))
        {
            clientSetFocus(c, True);
static inline void handleFocusIn(XFocusChangeEvent * ev)
Olivier Fourdan's avatar
Olivier Fourdan committed
{
    Client *c;

    DBG("entering handleFocusIn\n");
    DBG("FocusIn window is (%#lx)\n", ev->window);

Olivier Fourdan's avatar
Olivier Fourdan committed
    {
        /* Don't get fooled by our own gtk window ! */
        return;
    }
Olivier Fourdan's avatar
Olivier Fourdan committed
    c = clientGetFromWindow(ev->window, WINDOW);
    if(c)
    {
        DBG("focus set to \"%s\" (%#lx)\n", c->name, c->window);
        clientUpdateFocus(c);
        frameDraw(c, FALSE, FALSE);
        if(params.raise_on_focus && !params.click_to_focus)
Olivier Fourdan's avatar
Olivier Fourdan committed
    }
    else if(clients)
    {
        DBG("focus set to top window in list\n");
        clientSetFocus(clientGetNext(clients->prev, 0), True);
    }
    else
    {
        DBG("focus set to default fallback window\n");
        clientSetFocus(NULL, True);
    }
}

static inline void handleFocusOut(XFocusChangeEvent * ev)
Olivier Fourdan's avatar
Olivier Fourdan committed
{
    DBG("entering handleFocusOut\n");
}

static inline void handlePropertyNotify(XPropertyEvent * ev)
Olivier Fourdan's avatar
Olivier Fourdan committed
{
    Client *c;
    long dummy;

    DBG("entering handlePropertyNotify\n");

    c = clientGetFromWindow(ev->window, WINDOW);
    if(c)
    {
        if(ev->atom == XA_WM_NORMAL_HINTS)
        {
            unsigned long previous_value;
Olivier Fourdan's avatar
Olivier Fourdan committed
            DBG("client \"%s\" (%#lx) has received a XA_WM_NORMAL_HINTS notify\n", c->name, c->window);
            XGetWMNormalHints(dpy, c->window, c->size, &dummy);
            previous_value = CLIENT_FLAG_TEST(c, CLIENT_FLAG_IS_RESIZABLE);
            CLIENT_FLAG_UNSET(c, CLIENT_FLAG_IS_RESIZABLE);
            if(((c->size->flags & (PMinSize | PMaxSize)) != (PMinSize | PMaxSize)) || (((c->size->flags & (PMinSize | PMaxSize)) == (PMinSize | PMaxSize)) && ((c->size->min_width < c->size->max_width) || (c->size->min_height < c->size->max_height))))
            {
                CLIENT_FLAG_SET(c, CLIENT_FLAG_IS_RESIZABLE);
            }
            if(CLIENT_FLAG_TEST(c, CLIENT_FLAG_IS_RESIZABLE) != previous_value)
                frameDraw(c, TRUE, FALSE);
Olivier Fourdan's avatar
Olivier Fourdan committed
        }
        else if((ev->atom == XA_WM_NAME) || (ev->atom == net_wm_name))
Olivier Fourdan's avatar
Olivier Fourdan committed
        {
            DBG("client \"%s\" (%#lx) has received a XA_WM_NAME notify\n", c->name, c->window);
            if(c->name)
Olivier Fourdan's avatar
Olivier Fourdan committed
                free(c->name);
            }
            CLIENT_FLAG_SET(c, CLIENT_FLAG_NAME_CHANGED);
            frameDraw(c, TRUE, FALSE);
Olivier Fourdan's avatar
Olivier Fourdan committed
        }
        else if(ev->atom == motif_wm_hints)
        {
            DBG("client \"%s\" (%#lx) has received a motif_wm_hints notify\n", c->name, c->window);
            clientUpdateMWMHints(c);
            wc.x = c->x;
            wc.y = c->y;
            wc.width = c->width;
            wc.height = c->height;
            clientConfigure(c, &wc, CWX | CWY | CWWidth | CWHeight, FALSE);
        else if(ev->atom == XA_WM_HINTS)
        {
            DBG("client \"%s\" (%#lx) has received a XA_WM_HINTS notify\n", c->name, c->window);
            c->wmhints = XGetWMHints(dpy, c->window);
            if(c->wmhints)
            {
                c->group_leader = c->wmhints->window_group;
            }
        }
Olivier Fourdan's avatar
Olivier Fourdan committed
        else if(ev->atom == win_hints)
        {
            DBG("client \"%s\" (%#lx) has received a win_hints notify\n", c->name, c->window);
            getGnomeHint(dpy, c->window, win_hints, &c->win_hints);
Olivier Fourdan's avatar
Olivier Fourdan committed
        }
Olivier Fourdan's avatar
Olivier Fourdan committed
        {
            DBG("client \"%s\" (%#lx) has received a win_layer notify\n", c->name, c->window);
            getGnomeHint(dpy, c->window, win_layer, &dummy);
            clientSetLayer(c, dummy);
Olivier Fourdan's avatar
Olivier Fourdan committed
        }
Olivier Fourdan's avatar
Olivier Fourdan committed
        {
            DBG("client \"%s\" (%#lx) has received a net_wm_window_type notify\n", c->name, c->window);
            frameDraw(c, TRUE, FALSE);
Olivier Fourdan's avatar
Olivier Fourdan committed
        }
        else if((ev->atom == win_workspace) && !(c->transient_for))
Olivier Fourdan's avatar
Olivier Fourdan committed
        {
            DBG("client \"%s\" (%#lx) has received a win_workspace notify\n", c->name, c->window);
            getGnomeHint(dpy, c->window, win_workspace, &dummy);
            clientSetWorkspace(c, dummy, TRUE);
Olivier Fourdan's avatar
Olivier Fourdan committed
        }
        {
            DBG("client \"%s\" (%#lx) has received a net_wm_strut notify\n", c->name, c->window);
        else if(ev->atom == wm_colormap_windows)
        {
            clientUpdateColormaps(c);
            if(c == clientGetFocus())
            {
                clientInstallColormaps(c);
#ifdef HAVE_STARTUP_NOTIFICATION
        else if(ev->atom == net_startup_id)
        {
            if(c->startup_id)
            {
                free(c->startup_id);
                c->startup_id = NULL;
            }
Olivier Fourdan's avatar
Olivier Fourdan committed
            getWindowStartupId(dpy, c->window, &c->startup_id);
Olivier Fourdan's avatar
Olivier Fourdan committed
    }
    else
    {
        if(ev->atom == win_workspace_count)
        {
            DBG("root has received a win_workspace_count notify\n");
            getGnomeHint(dpy, root, win_workspace_count, &dummy);
            workspaceSetCount(dummy);
        }
        else if(ev->atom == gnome_panel_desktop_area)
        {
            DBG("root has received a gnome_panel_desktop_area notify\n");
            getGnomeDesktopMargins(dpy, screen, gnome_margins);
            workspaceUpdateArea(margins, gnome_margins);
static inline void handleClientMessage(XClientMessageEvent * ev)
Olivier Fourdan's avatar
Olivier Fourdan committed
{
    Client *c;

    DBG("entering handleClientMessage\n");

    c = clientGetFromWindow(ev->window, WINDOW);
    if(c)
    {
        if((ev->message_type == wm_change_state) && (ev->format == 32) && (ev->data.l[0] == IconicState))
Olivier Fourdan's avatar
Olivier Fourdan committed
            DBG("client \"%s\" (%#lx) has received a wm_change_state event\n", c->name, c->window);
            if(CLIENT_CAN_HIDE_WINDOW(c))
Olivier Fourdan's avatar
Olivier Fourdan committed
        else if((ev->message_type == win_state) && (ev->format == 32) && (ev->data.l[0] & WIN_STATE_SHADED))
Olivier Fourdan's avatar
Olivier Fourdan committed
            DBG("client \"%s\" (%#lx) has received a win_state/shaded event\n", c->name, c->window);
            if(ev->data.l[1] == WIN_STATE_SHADED)
            {
                clientShade(c);
                clientUnshade(c);
Olivier Fourdan's avatar
Olivier Fourdan committed
            clientToggleShaded(c);
        }
        else if((ev->message_type == win_state) && (ev->format == 32) && (ev->data.l[0] & WIN_STATE_STICKY))
Olivier Fourdan's avatar
Olivier Fourdan committed
        {
            DBG("client \"%s\" (%#lx) has received a win_state/stick event\n", c->name, c->window);
Olivier Fourdan's avatar
Olivier Fourdan committed
            if(CLIENT_FLAG_TEST(c, CLIENT_FLAG_HAS_STICK))
            {
                if(ev->data.l[1] == WIN_STATE_STICKY)
                {
                    clientUnstick(c, TRUE);
Olivier Fourdan's avatar
Olivier Fourdan committed
        }
        else if((ev->message_type == win_layer) && (ev->format == 32))
        {
Olivier Fourdan's avatar
Olivier Fourdan committed
            DBG("client \"%s\" (%#lx) has received a win_layer event\n", c->name, c->window);
            clientSetLayer(c, ev->data.l[0]);
        }
        else if((ev->message_type == win_workspace) && (ev->format == 32) && !(c->transient_for))
Olivier Fourdan's avatar
Olivier Fourdan committed
            DBG("client \"%s\" (%#lx) has received a win_workspace event\n", c->name, c->window);
            clientSetWorkspace(c, ev->data.l[0], TRUE);
        }
        else if((ev->message_type == net_wm_desktop) && (ev->format == 32))
        {
Olivier Fourdan's avatar
Olivier Fourdan committed
            DBG("client \"%s\" (%#lx) has received a net_wm_desktop event\n", c->name, c->window);
Olivier Fourdan's avatar
Olivier Fourdan committed
            if((ev->data.l[0] == (int)0xFFFFFFFF) && CLIENT_FLAG_TEST(c, CLIENT_FLAG_HAS_STICK))
Olivier Fourdan's avatar
Olivier Fourdan committed
                clientStick(c, TRUE);