diff options
Diffstat (limited to 'app/cwm/xevents.c')
-rw-r--r-- | app/cwm/xevents.c | 603 |
1 files changed, 603 insertions, 0 deletions
diff --git a/app/cwm/xevents.c b/app/cwm/xevents.c new file mode 100644 index 000000000..fdcdbcb55 --- /dev/null +++ b/app/cwm/xevents.c @@ -0,0 +1,603 @@ +/* + * calmwm - the calm window manager + * + * Copyright (c) 2004 Marius Aamodt Eriksen <marius@monkey.org> + * All rights reserved. + * + * $Id: xevents.c,v 1.1 2007/04/27 17:58:48 bernd Exp $ + */ + +/* + * NOTE: + * It is the responsibility of the caller to deal with memory + * management of the xevent's. + */ + +#include "headers.h" +#include "calmwm.h" + +void _sendxmsg(Window, Atom, long); + +/* + * NOTE: in reality, many of these should move to client.c now that + * we've got this nice event layer. + */ + +void +xev_handle_maprequest(struct xevent *xev, XEvent *ee) +{ + XMapRequestEvent *e = &ee->xmaprequest; + struct client_ctx *cc = NULL, *old_cc = client_current(); + XWindowAttributes xattr; + struct screen_ctx *sc; +#ifdef notyet + int state; +#endif + + if (old_cc != NULL) + client_ptrsave(old_cc); + + if ((cc = client_find(e->window)) == NULL) { + XGetWindowAttributes(G_dpy, e->window, &xattr); + cc = client_new(e->window, screen_fromroot(xattr.root), 1); + sc = CCTOSC(cc); + } else { + cc->beepbeep = 1; + } + +#ifdef notyet /* XXX - possibly, we shouldn't map if + * the window is withdrawn. */ + if (xu_getstate(cc, &state) == 0 && state == WithdrawnState) + warnx("WITHDRAWNSTATE for %s", cc->name); +#endif + + client_ptrwarp(cc); + xev_register(xev); +} + +void +xev_handle_unmapnotify(struct xevent *xev, XEvent *ee) +{ + XUnmapEvent *e = &ee->xunmap; + struct client_ctx *cc; + struct screen_ctx *sc; + int wascurrent; + + if ((cc = client_find(e->window)) != NULL) { + sc = CCTOSC(cc); + wascurrent = cc == client_current(); + client_delete(cc, e->send_event, 0); + +#ifdef notyet + /* XXX disable the ptrwarp until we handle it + * better. */ + if (!client_delete(cc, e->send_event, 0) && wascurrent) + ;/* client_ptrwarp(new_cc); */ +#endif + } + + xev_register(xev); +} + +void +xev_handle_destroynotify(struct xevent *xev, XEvent *ee) +{ + XDestroyWindowEvent *e = &ee->xdestroywindow; + struct client_ctx *cc; + + if ((cc = client_find(e->window)) != NULL) + client_delete(cc, 1, 1); + + xev_register(xev); +} + +void +xev_handle_configurerequest(struct xevent *xev, XEvent *ee) +{ + XConfigureRequestEvent *e = &ee->xconfigurerequest; + struct client_ctx *cc; + struct screen_ctx *sc; + XWindowChanges wc; + + if ((cc = client_find(e->window)) != NULL) { + sc = CCTOSC(cc); + + client_gravitate(cc, 0); + if (e->value_mask & CWWidth) + cc->geom.width = e->width; + if (e->value_mask & CWHeight) + cc->geom.height = e->height; + if (e->value_mask & CWX) + cc->geom.x = e->x; + if (e->value_mask & CWY) + cc->geom.y = e->y; + + if (cc->geom.x == 0 && + cc->geom.width >= DisplayWidth(G_dpy, sc->which)) + cc->geom.x -= cc->bwidth; + + if (cc->geom.y == 0 && + cc->geom.height >= DisplayHeight(G_dpy, sc->which)) + cc->geom.y -= cc->bwidth; + + client_gravitate(cc, 1); + + wc.x = cc->geom.x - cc->bwidth; + wc.y = cc->geom.y - cc->bwidth; + wc.width = cc->geom.width + cc->bwidth*2; + wc.height = cc->geom.height + cc->bwidth*2; + wc.border_width = 0; + + /* We need to move the parent window, too. */ + XConfigureWindow(G_dpy, cc->pwin, e->value_mask, &wc); + xev_reconfig(cc); + } + + wc.x = cc != NULL ? 0 : e->x; + wc.y = cc != NULL ? 0 : e->y; + wc.width = e->width; + wc.height = e->height; + wc.stack_mode = Above; + wc.border_width = 0; + e->value_mask &= ~CWStackMode; + e->value_mask |= CWBorderWidth; + + XConfigureWindow(G_dpy, e->window, e->value_mask, &wc); + + xev_register(xev); +} + +void +xev_handle_propertynotify(struct xevent *xev, XEvent *ee) +{ + XPropertyEvent *e = &ee->xproperty; + struct client_ctx *cc; + long tmp; + + if ((cc = client_find(e->window)) != NULL) { + switch(e->atom) { + case XA_WM_NORMAL_HINTS: + XGetWMNormalHints(G_dpy, cc->win, cc->size, &tmp); + break; + case XA_WM_NAME: + client_setname(cc); + break; + default: + /* do nothing */ + break; + } + } + + xev_register(xev); +} + +void +xev_reconfig(struct client_ctx *cc) +{ + XConfigureEvent ce; + + ce.type = ConfigureNotify; + ce.event = cc->win; + ce.window = cc->win; + ce.x = cc->geom.x; + ce.y = cc->geom.y; + ce.width = cc->geom.width; + ce.height = cc->geom.height; + ce.border_width = 0; + ce.above = None; + ce.override_redirect = 0; + + XSendEvent(G_dpy, cc->win, False, StructureNotifyMask, (XEvent *)&ce); +} + +void +xev_handle_enternotify(struct xevent *xev, XEvent *ee) +{ + XCrossingEvent *e = &ee->xcrossing; + struct client_ctx *cc; + + if ((cc = client_find(e->window)) == NULL) { + /* + * XXX - later. messes up unclutter. but may be + * needed when we introduce menu windows and such into + * the main event loop. + */ +#ifdef notyet + if (e->window != e->root) + client_nocurrent(); +#endif + } else + client_setactive(cc, 1); + + xev_register(xev); +} + +void +xev_handle_leavenotify(struct xevent *xev, XEvent *ee) +{ + client_leave(NULL); + + xev_register(xev); +} + +/* We can split this into two event handlers. */ +void +xev_handle_buttonpress(struct xevent *xev, XEvent *ee) +{ + XButtonEvent *e = &ee->xbutton; + struct client_ctx *cc, *old_cc = client_current(); + struct screen_ctx *sc = screen_fromroot(e->root); + char *wname; + int altcontrol = e->state == (ControlMask|Mod1Mask); + + cc = client_find(e->window); + + if (sc->rootwin == e->window && !altcontrol) { + struct menu_q menuq; + struct menu *mi; + + /* XXXSIGH!!!! */ + if (e->button == Button2) { + group_menu(e); + goto out; + } + + TAILQ_INIT(&menuq); + + switch (e->button) { + case Button1: + TAILQ_FOREACH(cc, &G_clientq, entry) { + if (cc->flags & CLIENT_HIDDEN) { + if (cc->label != NULL) + wname = cc->label; + else + wname = cc->name; + + if (wname == NULL) + continue; + + XCALLOC(mi, struct menu); + strlcpy(mi->text, + wname, sizeof(mi->text)); + mi->ctx = cc; + TAILQ_INSERT_TAIL(&menuq, mi, entry); + } + } + break; + case Button3: { + struct cmd *cmd; + if (conf_cmd_changed(G_conf.menu_path)) { + conf_cmd_clear(&G_conf); + conf_cmd_populate(&G_conf, G_conf.menu_path); + } + TAILQ_FOREACH(cmd, &G_conf.cmdq, entry) { + XCALLOC(mi, struct menu); + strlcpy(mi->text, cmd->label, sizeof(mi->text)); + mi->ctx = cmd; + TAILQ_INSERT_TAIL(&menuq, mi, entry); + } + break; + } + default: + break; + } + + if (TAILQ_EMPTY(&menuq)) + goto out; + + mi = (struct menu *)grab_menu(e, &menuq); + if (mi == NULL) + goto cleanup; + + switch (e->button) { + case Button1: + cc = (struct client_ctx *)mi->ctx; + client_unhide(cc); + + if (old_cc != NULL) + client_ptrsave(old_cc); + client_ptrwarp(cc); + break; + case Button3: + u_spawn(((struct cmd *)mi->ctx)->image); + break; + default: + break; + } + + cleanup: + while ((mi = TAILQ_FIRST(&menuq)) != NULL) { + TAILQ_REMOVE(&menuq, mi, entry); + xfree(mi); + } + + goto out; + } + + if (cc == NULL || e->state == 0) + goto out; + + sc = CCTOSC(cc); + + switch (e->button) { + case Button1: + if (altcontrol && !G_groupmode) + group_sticky_toggle_enter(cc); + else { + grab_drag(cc); + client_move(cc); + } + break; + case Button2: + /* XXXSIGH!!! */ + if (G_groupmode) + group_click(cc); + else { + grab_sweep(cc); + client_resize(cc); + } + break; + case Button3: + client_ptrsave(cc); + client_lower(cc); + break; + } + out: + xev_register(xev); +} + +void +xev_handle_buttonrelease(struct xevent *xev, XEvent *ee) +{ + struct client_ctx *cc = client_current(); + + if (cc != NULL && !G_groupmode) + group_sticky_toggle_exit(cc); + + xev_register(xev); + +} + +void +xev_handle_keypress(struct xevent *xev, XEvent *ee) +{ + XKeyEvent *e = &ee->xkey; + struct client_ctx *cc = NULL; /* Make gcc happy. */ + struct keybinding *kb; + KeySym keysym, skeysym; + int modshift; + + keysym = XKeycodeToKeysym(G_dpy, e->keycode, 0); + skeysym = XKeycodeToKeysym(G_dpy, e->keycode, 1); + + TAILQ_FOREACH(kb, &G_conf.keybindingq, entry) { + if (keysym != kb->keysym && skeysym == kb->keysym) + modshift = ShiftMask; + else + modshift = 0; + + if ((kb->modmask | modshift) != e->state) + continue; + + if ((kb->keycode != 0 && kb->keysym == NoSymbol && + kb->keycode == e->keycode) || kb->keysym == + (modshift == 0 ? keysym : skeysym)) + break; + } + + if (kb == NULL && e->window == screen_current()->groupwin) + group_display_keypress(e->keycode); + + if (kb == NULL) + goto out; + + if ((kb->flags & (KBFLAG_NEEDCLIENT|KBFLAG_FINDCLIENT)) && + (cc = client_find(e->window)) == NULL && + (cc = client_current()) == NULL) + if (kb->flags & KBFLAG_NEEDCLIENT) + goto out; + + (*kb->callback)(cc, kb->argument); + +out: + xev_register(xev); +} + +/* + * This is only used for the alt supression detection. + */ +void +xev_handle_keyrelease(struct xevent *xev, XEvent *ee) +{ + XKeyEvent *e = &ee->xkey; + struct screen_ctx *sc = screen_fromroot(e->root); + int keysym; + + keysym = XKeycodeToKeysym(G_dpy, e->keycode, 0); + if (keysym != XK_Alt_L && keysym != XK_Alt_R) + goto out; + + sc->altpersist = 0; + + /* + * XXX - better interface... xevents should not know about + * how/when to mtf. + */ + client_mtf(NULL); + client_altrelease(); + + out: + xev_register(xev); +} + +void +xev_handle_clientmessage(struct xevent *xev, XEvent *ee) +{ + XClientMessageEvent *e = &ee->xclient; + struct client_ctx *cc = client_find(e->window); + Atom xa_wm_change_state = XInternAtom(G_dpy, "WM_CHANGE_STATE", False); + + if (cc == NULL) + goto out; + + if (e->message_type == xa_wm_change_state && e->format == 32 && + e->data.l[0] == IconicState) + client_hide(cc); +out: + xev_register(xev); +} + +/* + * X Event handling + */ + +static struct xevent_q _xevq, _xevq_putaway; +static short _xev_q_lock = 0; + +void +xev_init(void) +{ + TAILQ_INIT(&_xevq); + TAILQ_INIT(&_xevq_putaway); +} + +struct xevent * +xev_new(Window *win, Window *root, + int type, void (*cb)(struct xevent *, XEvent *), void *arg) +{ + struct xevent *xev; + + XMALLOC(xev, struct xevent); + xev->xev_win = win; + xev->xev_root = root; + xev->xev_type = type; + xev->xev_cb = cb; + xev->xev_arg = arg; + + return (xev); +} + +void +xev_register(struct xevent *xev) +{ + struct xevent_q *xq; + + xq = _xev_q_lock ? &_xevq_putaway : &_xevq; + TAILQ_INSERT_TAIL(xq, xev, entry); +} + +void +_xev_reincorporate(void) +{ + struct xevent *xev; + + while ((xev = TAILQ_FIRST(&_xevq_putaway)) != NULL) { + TAILQ_REMOVE(&_xevq_putaway, xev, entry); + TAILQ_INSERT_TAIL(&_xevq, xev, entry); + } +} + +void +xev_handle_expose(struct xevent *xev, XEvent *ee) +{ + XExposeEvent *e = &ee->xexpose; + struct screen_ctx *sc = screen_current(); + struct client_ctx *cc; + + if ((cc = client_find(e->window)) != NULL) + client_draw_border(cc); + + if (sc->groupwin == e->window) + group_display_draw(sc); + + xev_register(xev); +} + +#define ASSIGN(xtype) do { \ + root = e. xtype .root; \ + win = e. xtype .window; \ +} while (0) + +#define ASSIGN1(xtype) do { \ + win = e. xtype .window; \ +} while (0) + +void +xev_loop(void) +{ + Window win, root; + int type; + XEvent e; + struct xevent *xev, *nextxev; + + for (;;) { +#ifdef DIAGNOSTIC + if (TAILQ_EMPTY(&_xevq)) + errx(1, "X event queue empty"); +#endif + + XNextEvent(G_dpy, &e); + type = e.type; + + win = root = 0; + + switch (type) { + case MapRequest: + ASSIGN1(xmaprequest); + break; + case UnmapNotify: + ASSIGN1(xunmap); + break; + case ConfigureRequest: + ASSIGN1(xconfigurerequest); + break; + case PropertyNotify: + ASSIGN1(xproperty); + break; + case EnterNotify: + case LeaveNotify: + ASSIGN(xcrossing); + break; + case ButtonPress: + ASSIGN(xbutton); + break; + case ButtonRelease: + ASSIGN(xbutton); + break; + case KeyPress: + case KeyRelease: + ASSIGN(xkey); + break; + case DestroyNotify: + ASSIGN1(xdestroywindow); + break; + case ClientMessage: + ASSIGN1(xclient); + break; + default: /* XXX - still need shape event support. */ + break; + } + + /* + * Now, search for matches, and call each of them. + */ + _xev_q_lock = 1; + for (xev = TAILQ_FIRST(&_xevq); xev != NULL; xev = nextxev) { + nextxev = TAILQ_NEXT(xev, entry); + + if ((type != xev->xev_type && xev->xev_type != 0) || + (xev->xev_win != NULL && win != *xev->xev_win) || + (xev->xev_root != NULL && root != *xev->xev_root)) + continue; + + TAILQ_REMOVE(&_xevq, xev, entry); + + (*xev->xev_cb)(xev, &e); + } + _xev_q_lock = 0; + _xev_reincorporate(); + } +} + +#undef ASSIGN +#undef ASSIGN1 |