271 lines
6.7 KiB
C
271 lines
6.7 KiB
C
/*
|
|
* wmfs2 by Martin Duquesnoy <xorg62@gmail.com> { for(i = 2011; i < 2111; ++i) ©(i); }
|
|
* For license, see COPYING.
|
|
*/
|
|
|
|
#include "event.h"
|
|
#include "ewmh.h"
|
|
#include "util.h"
|
|
#include "wmfs.h"
|
|
#include "client.h"
|
|
#include "barwin.h"
|
|
#include "screen.h"
|
|
|
|
#define EVDPY(e) (e)->xany.display
|
|
|
|
static void
|
|
event_buttonpress(XEvent *e)
|
|
{
|
|
XButtonEvent *ev = &e->xbutton;
|
|
struct mousebind *m;
|
|
struct barwin *b;
|
|
|
|
screen_update_sel();
|
|
|
|
SLIST_FOREACH(b, &W->h.barwin, next)
|
|
if(b->win == ev->window)
|
|
{
|
|
W->last_clicked_barwin = b;
|
|
|
|
SLIST_FOREACH(m, &b->mousebinds, next)
|
|
if(m->button == ev->button)
|
|
if(!m->use_area || (m->use_area && INAREA(ev->x, ev->y, m->area)))
|
|
if(m->func)
|
|
m->func(m->cmd);
|
|
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
event_enternotify(XEvent *e)
|
|
{
|
|
XCrossingEvent *ev = &e->xcrossing;
|
|
struct client *c;
|
|
|
|
if((ev->mode != NotifyNormal
|
|
|| ev->detail == NotifyInferior)
|
|
&& ev->window != W->root)
|
|
return;
|
|
|
|
if((c = client_gb_win(ev->window))
|
|
|| (c = client_gb_frame(ev->window)))
|
|
{
|
|
if(c->flags & CLIENT_IGNORE_ENTER)
|
|
c->flags ^= CLIENT_IGNORE_ENTER;
|
|
else if(c != W->client && !(c->flags & CLIENT_TABBED))
|
|
client_focus(c);
|
|
}
|
|
}
|
|
|
|
static void
|
|
event_clientmessageevent(XEvent *e)
|
|
{
|
|
XClientMessageEvent *ev = &e->xclient;
|
|
struct client *c;
|
|
int type = 0;
|
|
|
|
while(type < net_last && W->net_atom[type] != ev->message_type)
|
|
++type;
|
|
|
|
switch(type)
|
|
{
|
|
/* _NET_WM_STATE */
|
|
case net_wm_state:
|
|
if((c = client_gb_win(ev->window)))
|
|
ewmh_manage_state(ev->data.l, c);
|
|
break;
|
|
/* _NET_CLOSE_WINDOW */
|
|
case net_close_window:
|
|
if((c = client_gb_win(ev->window)))
|
|
client_close(c);
|
|
break;
|
|
/* _NET_WM_DESKTOP */
|
|
case net_wm_desktop:
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
event_configureevent(XEvent *e)
|
|
{
|
|
XConfigureRequestEvent *ev = &e->xconfigurerequest;
|
|
XWindowChanges wc;
|
|
struct client *c;
|
|
|
|
if((c = client_gb_win(ev->window)))
|
|
{
|
|
if(ev->value_mask & CWWidth)
|
|
_fac_resize(c, Right, ev->width - c->wgeo.w);
|
|
if(ev->value_mask & CWHeight)
|
|
_fac_resize(c, Bottom, ev->height - c->wgeo.h);
|
|
|
|
client_apply_tgeo(c->tag);
|
|
}
|
|
else
|
|
{
|
|
wc.x = ev->x;
|
|
wc.y = ev->y;
|
|
wc.width = ev->width;
|
|
wc.height = ev->height;
|
|
wc.border_width = ev->border_width;
|
|
wc.sibling = ev->above;
|
|
wc.stack_mode = ev->detail;
|
|
|
|
XConfigureWindow(EVDPY(e), ev->window, ev->value_mask, &wc);
|
|
}
|
|
}
|
|
|
|
static void
|
|
event_destroynotify(XEvent *e)
|
|
{
|
|
XDestroyWindowEvent *ev = &e->xdestroywindow;
|
|
struct client *c;
|
|
|
|
if((c = client_gb_win(ev->window)))
|
|
client_remove(c);
|
|
}
|
|
|
|
static void
|
|
event_focusin(XEvent *e)
|
|
{
|
|
if(W->client
|
|
&& e->xfocus.window != W->client->win
|
|
&& e->xfocus.window != W->client->frame)
|
|
client_focus(W->client);
|
|
}
|
|
|
|
static void
|
|
event_maprequest(XEvent *e)
|
|
{
|
|
XMapRequestEvent *ev = &e->xmaprequest;
|
|
XWindowAttributes at;
|
|
|
|
/* Which windows to manage */
|
|
if(!XGetWindowAttributes(EVDPY(e), ev->window, &at)
|
|
|| at.override_redirect)
|
|
return;
|
|
|
|
if(!client_gb_win(ev->window))
|
|
client_new(ev->window, &at, false);
|
|
}
|
|
|
|
static void
|
|
event_mappingnotify(XEvent *e)
|
|
{
|
|
XMappingEvent *ev = &e->xmapping;
|
|
XRefreshKeyboardMapping(ev);
|
|
|
|
if(ev->request == MappingKeyboard)
|
|
wmfs_grab_keys();
|
|
}
|
|
|
|
static void
|
|
event_propertynotify(XEvent *e)
|
|
{
|
|
XPropertyEvent *ev = &e->xproperty;
|
|
struct client *c;
|
|
|
|
if(ev->state == PropertyDelete)
|
|
return;
|
|
|
|
if((c = client_gb_win(ev->window)))
|
|
{
|
|
switch(ev->atom)
|
|
{
|
|
case XA_WM_TRANSIENT_FOR:
|
|
break;
|
|
case XA_WM_NORMAL_HINTS:
|
|
/* client_get_size_hints(c); */
|
|
break;
|
|
case XA_WM_HINTS:
|
|
/*
|
|
XWMHints *h;
|
|
|
|
if((h = XGetWMHints(EVDPY, c->win)) && (h->flags & XUrgencyHint) && c != sel)
|
|
{
|
|
client_urgent(c, True);
|
|
XFree(h);
|
|
}
|
|
*/
|
|
break;
|
|
default:
|
|
if(ev->atom == XA_WM_NAME || ev->atom == W->net_atom[net_wm_name])
|
|
client_get_name(c);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
static void
|
|
event_unmapnotify(XEvent *e)
|
|
{
|
|
XUnmapEvent *ev = &e->xunmap;
|
|
struct client *c;
|
|
|
|
if((c = client_gb_win(ev->window)) && ev->send_event)
|
|
client_remove(c);
|
|
}
|
|
|
|
static void
|
|
event_keypress(XEvent *e)
|
|
{
|
|
XKeyPressedEvent *ev = &e->xkey;
|
|
KeySym keysym = XKeycodeToKeysym(EVDPY(e), (KeyCode)ev->keycode, 0);
|
|
struct keybind *k;
|
|
|
|
screen_update_sel();
|
|
|
|
SLIST_FOREACH(k, &W->h.keybind, next)
|
|
if(k->keysym == keysym && KEYPRESS_MASK(k->mod) == KEYPRESS_MASK(ev->state))
|
|
if(k->func)
|
|
k->func(k->cmd);
|
|
}
|
|
|
|
static void
|
|
event_expose(XEvent *e)
|
|
{
|
|
XExposeEvent *ev = &e->xexpose;
|
|
struct barwin *b;
|
|
|
|
SLIST_FOREACH(b, &W->h.barwin, next)
|
|
if(b->win == ev->window)
|
|
{
|
|
barwin_refresh(b);
|
|
return;
|
|
}
|
|
}
|
|
|
|
static void
|
|
event_dummy(XEvent *e)
|
|
{
|
|
/* printf("%d\n", e->type);*/
|
|
(void)e;
|
|
}
|
|
|
|
void
|
|
event_init(void)
|
|
{
|
|
int i = MAX_EV;
|
|
|
|
while(i--)
|
|
event_handle[i] = event_dummy;
|
|
|
|
event_handle[ButtonPress] = event_buttonpress;
|
|
event_handle[ClientMessage] = event_clientmessageevent;
|
|
event_handle[ConfigureRequest] = event_configureevent;
|
|
event_handle[DestroyNotify] = event_destroynotify;
|
|
event_handle[EnterNotify] = event_enternotify;
|
|
event_handle[Expose] = event_expose;
|
|
event_handle[FocusIn] = event_focusin;
|
|
event_handle[KeyPress] = event_keypress;
|
|
/*event_handle[MapNotify] = event_mapnotify;*/
|
|
event_handle[MapRequest] = event_maprequest;
|
|
event_handle[MappingNotify] = event_mappingnotify;
|
|
event_handle[PropertyNotify] = event_propertynotify;
|
|
/*event_handle[ReparentNotify] = event_reparentnotify;*/
|
|
/*event_handle[SelectionClear] = event_selectionclearevent;*/
|
|
event_handle[UnmapNotify] = event_unmapnotify;
|
|
}
|
|
|