Skip to content
Snippets Groups Projects
events.c 37.3 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"
#include "frame.h"
#include "client.h"
#include "menu.h"
#include "debug.h"

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(c, CLIENT_FLAG_HAS_BORDER) && CLIENT_FLAG_TEST(c, CLIENT_FLAG_HAS_MOVE) && !CLIENT_FLAG_TEST(c, CLIENT_FLAG_FULLSCREEN))
static inline void _resizeRequest(Client * c, int corner, XEvent * ev)
    if(CLIENT_FLAG_TEST(c, CLIENT_FLAG_HAS_RESIZE) && CLIENT_FLAG_TEST(c, CLIENT_FLAG_IS_RESIZABLE))
    else if(CLIENT_FLAG_TEST(c, CLIENT_FLAG_HAS_BORDER) && CLIENT_FLAG_TEST(c, CLIENT_FLAG_HAS_MOVE) && !CLIENT_FLAG_TEST(c, CLIENT_FLAG_FULLSCREEN))
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();
    state = ev->state & (ShiftMask | ControlMask | AltMask | MetaMask);
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(c, CLIENT_FLAG_HAS_RESIZE) && CLIENT_FLAG_TEST(c, 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_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);
Olivier Fourdan's avatar
Olivier Fourdan committed
                break;
            case KEY_DEL_WORKSPACE:
                workspaceSetCount(params.workspace_count - 1);
Olivier Fourdan's avatar
Olivier Fourdan committed
                break;
            case KEY_STICK_WINDOW:
                clientToggleSticky(c);
                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_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;
        }
    }
    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;
            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);
Olivier Fourdan's avatar
Olivier Fourdan committed
                break;
            case KEY_DEL_WORKSPACE:
                workspaceSetCount(params.workspace_count - 1);
Olivier Fourdan's avatar
Olivier Fourdan committed
                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;
        }
    }

    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)
    {
        state = ev->state & (ShiftMask | ControlMask | AltMask | MetaMask);
        win = ev->subwindow;
        if((win == c->buttons[HIDE_BUTTON]) || (win == c->buttons[CLOSE_BUTTON]) || (win == c->buttons[MAXIMIZE_BUTTON]) || (win == c->buttons[SHADE_BUTTON]) || (win == 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);
        }
        else if(((win == c->title) && (ev->button == Button3)) || ((win == c->buttons[MENU_BUTTON]) && (ev->button == Button1)))
        {
            /*
               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;
            tclick = typeOfClick(c->frame, &copy_event, win == 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
        }
        else if(((win == 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))
        else if((win == c->corners[CORNER_TOP_LEFT]) && (state == 0))
Olivier Fourdan's avatar
Olivier Fourdan committed
        {
Olivier Fourdan's avatar
Olivier Fourdan committed
        }
        else if((win == 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
        }
        else if((win == 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
        }
        else if((win == 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
        }
        else if((win == 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
        }
        else if((win == 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
        }
        else if((win == 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;

    DBG("entering handleConfigureRequest\n");

    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)
    {
        if(c->type == WINDOW_DESKTOP)
        {
            /* Ignore stacking request for DESKTOP windows */
            wc.stack_mode &= ~CWStackMode;
        }
        clientCoordGravitate(c, APPLY, &wc.x, &wc.y);
        if((ev->value_mask & (CWX | CWY | CWWidth | CWHeight)) && CLIENT_FLAG_TEST(c, CLIENT_FLAG_MAXIMIZED))
        {
            clientRemoveMaximizeFlag(c);
        }
Olivier Fourdan's avatar
Olivier Fourdan committed
        clientConfigure(c, &wc, ev->value_mask);
    }
    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);
        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 = (c->client_flag & CLIENT_FLAG_IS_RESIZABLE);
            if(!(c->size->flags & (PMinSize | PMaxSize)) || ((c->size->flags & (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)
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);
            }
Olivier Fourdan's avatar
Olivier Fourdan committed
            frameDraw(c);
        }
        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);
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);
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);
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);
	    }
	    else
	    {
                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);
	    if (ev->data.l[1] == WIN_STATE_STICKY)
	    {
                clientStick(c);
	    }
	    else
	    {
                clientUnstick(c);
	    }
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);
                clientSetWorkspace(c, ev->data.l[0], TRUE);
Olivier Fourdan's avatar
Olivier Fourdan committed
        else if((ev->message_type == net_close_window) && (ev->format == 32))
Olivier Fourdan's avatar
Olivier Fourdan committed
            DBG("client \"%s\" (%#lx) has received a net_close_window event\n", c->name, c->window);
Olivier Fourdan's avatar
Olivier Fourdan committed
        else if((ev->message_type == net_wm_state) && (ev->format == 32))
        {
            DBG("client \"%s\" (%#lx) has received a net_wm_state event\n", c->name, c->window);
Olivier Fourdan's avatar
Olivier Fourdan committed
        else if((ev->message_type == net_wm_moveresize) && (ev->format == 32))
        {
            DBG("client \"%s\" (%#lx) has received a net_wm_moveresize event\n", c->name, c->window);
            g_message("Operation not supported (yet)\n");
            /* TBD */
        }
Olivier Fourdan's avatar
Olivier Fourdan committed
        else if((ev->message_type == net_active_window) && (ev->format == 32))
        {
            DBG("client \"%s\" (%#lx) has received a net_active_window event\n", c->name, c->window);
            workspaceSwitch(c->win_workspace, NULL);
            clientShow(c, True);
            clientRaise(c);
            clientSetFocus(c, True);
        }
Olivier Fourdan's avatar
Olivier Fourdan committed
    }
    else
    {
        if(((ev->message_type == win_workspace) || (ev->message_type == net_current_desktop)) && (ev->format == 32))
Olivier Fourdan's avatar
Olivier Fourdan committed
            DBG("root has received a win_workspace or a net_current_desktop event\n");
            workspaceSwitch(ev->data.l[0], NULL);
        }
        else if(((ev->message_type == win_workspace_count) || (ev->message_type == net_number_of_desktops)) && (ev->format == 32))
        {
Olivier Fourdan's avatar
Olivier Fourdan committed
            DBG("root has received a win_workspace_count event\n");
            workspaceSetCount(ev->data.l[0]);
static inline void handleShape(XShapeEvent * ev)
Olivier Fourdan's avatar
Olivier Fourdan committed
{
    Client *c;

    DBG("entering handleShape\n");

    c = clientGetFromWindow(ev->window, WINDOW);
    if(c)
    {
        frameDraw(c);
    }
}

static inline void handleColormapNotify(XColormapEvent * ev)
    DBG("entering handleColormapNotify\n");
    c = clientGetFromWindow(ev->window, WINDOW);
    if((c) && (ev->window == c->window) && (ev->new))
Olivier Fourdan's avatar
Olivier Fourdan committed
void handleEvent(XEvent * ev)
{
    DBG("entering handleEvent\n");

    switch (ev->type)
    {
        case KeyPress:
            handleKeyPress((XKeyEvent *) ev);
            break;
        case ButtonPress:
            handleButtonPress((XButtonEvent *) ev);
            break;
        case ButtonRelease:
            handleButtonRelease((XButtonEvent *) ev);
            break;
        case DestroyNotify:
            handleDestroyNotify((XDestroyWindowEvent *) ev);
            break;
        case UnmapNotify:
            handleUnmapNotify((XUnmapEvent *) ev);
            break;
        case MapRequest:
            handleMapRequest((XMapRequestEvent *) ev);
            break;
        case ConfigureRequest:
            handleConfigureRequest((XConfigureRequestEvent *) ev);
            break;
        case EnterNotify:
            handleEnterNotify((XCrossingEvent *) ev);
            break;
        case FocusIn:
            handleFocusIn((XFocusChangeEvent *) ev);
            break;
        case FocusOut:
            handleFocusOut((XFocusChangeEvent *) ev);
            break;
        case PropertyNotify:
            handlePropertyNotify((XPropertyEvent *) ev);
            break;
        case ClientMessage: