X-Git-Url: http://pileus.org/git/?p=wmpus;a=blobdiff_plain;f=sys-x11.c;h=1a833da3b23a033e22d81ea767a8fb91a1f29dd0;hp=fa66a31d83a2fedffbb78efc65931008ad6c3dbe;hb=08a8a76218133e513d7804fcc3307c895f22956b;hpb=33194e81b7543a4019d71c9a8c183e3500e1e994 diff --git a/sys-x11.c b/sys-x11.c index fa66a31..1a833da 100644 --- a/sys-x11.c +++ b/sys-x11.c @@ -32,6 +32,7 @@ /* Configuration */ static int border = 2; static int no_capture = 0; +static int stack = 25; /* Internal structures */ struct win_sys { @@ -40,7 +41,6 @@ struct win_sys { struct { int left, right, top, bottom; } strut; - state_t state; }; typedef struct { @@ -49,7 +49,10 @@ typedef struct { } event_map_t; typedef enum { - WM_PROTO, WM_FOCUS, NET_STRUT, NATOMS + WM_PROTO, WM_FOCUS, WM_DELETE, + NET_STATE, NET_FULL, NET_STRUT, + NET_TYPE, NET_DIALOG, + NATOMS } atom_t; typedef enum { @@ -57,6 +60,8 @@ typedef enum { } color_t; /* Global data */ +static win_t *root; +static win_t *last; static int running; static void *cache; static Atom atoms[NATOMS]; @@ -65,6 +70,17 @@ static unsigned long colors[NCOLORS]; static list_t *screens; static list_t *struts; +/* Debug functions */ +static char *state_map[] = { + [ST_HIDE ] "hide ", + [ST_SHOW ] "show ", + [ST_FULL ] "full ", + [ST_MAX ] "max ", + [ST_SHADE] "shade", + [ST_ICON ] "icon ", + [ST_CLOSE] "close", +}; + /* Conversion functions */ static event_map_t ev2sym[] = { {EV_LEFT , XK_Left }, @@ -167,6 +183,10 @@ static int strut_copy(win_t *to, win_t *from, int scale) to->y += scale*(top ); to->w -= scale*(left+right); to->h -= scale*(top+bottom); + to->sys->strut.left += scale*left; + to->sys->strut.right += scale*right; + to->sys->strut.top += scale*top; + to->sys->strut.bottom += scale*bottom; return 1; } @@ -183,10 +203,10 @@ static int strut_add(win_t *root, win_t *win) if (status != Success || ret_size != 32 || ret_items != 4) return 0; - win->sys->strut.left = ((int*)xdata)[0]; - win->sys->strut.right = ((int*)xdata)[1]; - win->sys->strut.top = ((int*)xdata)[2]; - win->sys->strut.bottom = ((int*)xdata)[3]; + win->sys->strut.left = ((long*)xdata)[0]; + win->sys->strut.right = ((long*)xdata)[1]; + win->sys->strut.top = ((long*)xdata)[2]; + win->sys->strut.bottom = ((long*)xdata)[3]; struts = list_insert(struts, win); for (list_t *cur = screens; cur; cur = cur->next) strut_copy(cur->data, win, 1); @@ -204,12 +224,17 @@ static int strut_del(win_t *root, win_t *win) } /* Window functions */ +static Atom win_prop(win_t *win, atom_t prop); +static win_t *win_find(Display *dpy, Window xid, int create); + static win_t *win_new(Display *dpy, Window xid) { + Window trans; XWindowAttributes attr; if (XGetWindowAttributes(dpy, xid, &attr)) if (attr.override_redirect) return NULL; + win_t *win = new0(win_t); win->x = attr.x; win->y = attr.y; @@ -218,9 +243,33 @@ static win_t *win_new(Display *dpy, Window xid) win->sys = new0(win_sys_t); win->sys->dpy = dpy; win->sys->xid = xid; - printf("win_new: %p = %p, %d (%d,%d %dx%d)\n", - win, dpy, (int)xid, - win->x, win->y, win->w, win->h); + + if (root) { + if (strut_add(root, win)) + win->type = TYPE_TOOLBAR; + + if (win_prop(win, NET_TYPE) == atoms[NET_DIALOG]) + win->type = TYPE_DIALOG; + + if (win_prop(win, NET_STATE) == atoms[NET_FULL]) + win->state = ST_FULL; + + if (XGetTransientForHint(dpy, xid, &trans)) + win->parent = win_find(dpy, trans, 0); + + XSelectInput(dpy, xid, PropertyChangeMask); + } + + printf("win_new: win=%p x11=(%p,%d) state=%x pos=(%d,%d %dx%d) type=%s\n", + win, dpy, (int)xid, win->state, + win->x, win->y, win->w, win->h, + win->type == TYPE_NORMAL ? "normal" : + win->type == TYPE_DIALOG ? "dialog" : + win->type == TYPE_TOOLBAR ? "toolbar" : "unknown"); + + if (root) + wm_insert(win); + return win; } @@ -251,25 +300,69 @@ static win_t *win_find(Display *dpy, Window xid, int create) static void win_free(win_t *win) { + if (win == last) + last = NULL; free(win->sys); free(win); } static void win_remove(win_t *win) { + if (win != root) { + strut_del(root, win); + wm_remove(win); + } tdelete(win, &cache, win_cmp); win_free(win); } -static int win_viewable(win_t *win) +static int win_viewable(Display *dpy, Window xid) { XWindowAttributes attr; - if (XGetWindowAttributes(win->sys->dpy, win->sys->xid, &attr)) + if (XGetWindowAttributes(dpy, xid, &attr)) return attr.map_state == IsViewable; else return True; } +static int win_msg(win_t *win, atom_t msg) +{ + int n, found = 0; + Atom *protos; + if (!XGetWMProtocols(win->sys->dpy, win->sys->xid, &protos, &n)) + return 0; + + while (!found && n--) + found = protos[n] == atoms[msg]; + XFree(protos); + if (!found) + return 0; + + XSendEvent(win->sys->dpy, win->sys->xid, False, NoEventMask, &(XEvent){ + .xclient.type = ClientMessage, + .xclient.window = win->sys->xid, + .xclient.message_type = atoms[WM_PROTO], + .xclient.format = 32, + .xclient.data.l[0] = atoms[msg], + .xclient.data.l[1] = CurrentTime, + }); + return 1; +} + +static Atom win_prop(win_t *win, atom_t prop) +{ + int format; + unsigned long nitems, bytes; + unsigned char *buf = NULL; + Atom atom, type = XA_ATOM; + if (XGetWindowProperty(win->sys->dpy, win->sys->xid, atoms[prop], + 0L, sizeof(Atom), False, type, &type, &format, &nitems, &bytes, &buf) || !buf) + return 0; + atom = *(Atom *)buf; + XFree(buf); + return atom; +} + /* Drawing functions */ static unsigned long get_color(Display *dpy, const char *name) { @@ -295,6 +388,7 @@ static void process_event(int type, XEvent *xe, win_t *root) Window xid = getfocus(root, xe); if (!(win = win_find(dpy,xid,0))) return; + //printf("button-press %p\n", win); ptr = x2ptr(xe); mod = x2mod(xe->xkey.state, type==KeyRelease||type==ButtonRelease); } @@ -302,19 +396,22 @@ static void process_event(int type, XEvent *xe, win_t *root) /* Split based on event */ if (type == KeyPress) { while (XCheckTypedEvent(dpy, KeyPress, xe)); - KeySym sym = XKeycodeToKeysym(dpy, xe->xkey.keycode, 0); - printf("got xe %c %hhx\n", xk2ev(sym), mod2int(mod)); + KeySym sym = XLookupKeysym(&xe->xkey, 0); + //printf("got xe %c %hhx\n", xk2ev(sym), mod2int(mod)); wm_handle_event(win, xk2ev(sym), mod, ptr); } else if (type == KeyRelease) { - //printf("release: %d\n", type); + //printf("release: %lx\n", xe->xkey.window); } else if (type == ButtonPress) { - if (wm_handle_event(win, xb2ev(xe->xbutton.button), mod, ptr)) + if (wm_handle_event(win, xb2ev(xe->xbutton.button), mod, ptr)) { + //printf("grab pointer\n"); XGrabPointer(dpy, xe->xbutton.root, True, PointerMotionMask|ButtonReleaseMask, GrabModeAsync, GrabModeAsync, None, None, CurrentTime); - else - XAllowEvents(win->sys->dpy, ReplayPointer, CurrentTime); + } else { + //printf("allow events\n"); + XAllowEvents(win->sys->dpy, ReplayPointer, xe->xbutton.time); + } } else if (type == ButtonRelease) { XUngrabPointer(dpy, CurrentTime); @@ -325,44 +422,43 @@ static void process_event(int type, XEvent *xe, win_t *root) wm_handle_ptr(win, ptr); } else if (type == EnterNotify || type == LeaveNotify) { - printf("%s: %d\n", type==EnterNotify?"enter":"leave", type); - event_t ev = EnterNotify ? EV_ENTER : EV_LEAVE; + printf("%s: %lx\n", type==EnterNotify?"enter":"leave", + xe->xcrossing.window); + event_t ev = type == EnterNotify ? EV_ENTER : EV_LEAVE; if ((win = win_find(dpy,xe->xcrossing.window,0))) wm_handle_event(win, ev, MOD(), PTR()); } else if (type == FocusIn || type == FocusOut) { - //printf("focus: %d\n", type); - event_t ev = FocusIn ? EV_FOCUS : EV_UNFOCUS; + //printf("focus: %lx\n", xe->xfocus.window); + event_t ev = type == FocusIn ? EV_FOCUS : EV_UNFOCUS; if ((win = win_find(dpy,xe->xfocus.window,0))) wm_handle_event(win, ev, MOD(), PTR()); } else if (type == ConfigureNotify) { - printf("configure: %d\n", type); + //printf("configure: %lx\n", xe->xconfigure.window); } else if (type == MapNotify) { - printf("map: %d\n", type); + printf("map: %lx\n", xe->xmap.window); } else if (type == UnmapNotify) { if ((win = win_find(dpy,xe->xunmap.window,0)) && - win->sys->state != ST_HIDE) { - if (!strut_del(root, win)) - wm_remove(win); - else - wm_update(); - win->sys->state = ST_HIDE; + win->state != ST_HIDE) { + printf("unmap: %lx\n", xe->xunmap.window); + wm_handle_state(win, win->state, ST_HIDE); + win->state = ST_HIDE; } } else if (type == DestroyNotify) { - //printf("destroy: %d\n", type); + printf("destroy: %lx\n", xe->xdestroywindow.window); if ((win = win_find(dpy,xe->xdestroywindow.window,0))) win_remove(win); } else if (type == ConfigureRequest) { XConfigureRequestEvent *cre = &xe->xconfigurerequest; - printf("configure_req: %d - %x, (0x%lx) %dx%d @ %d,%d\n", - type, (int)cre->window, cre->value_mask, + printf("configure_req: %lx - (0x%lx) %dx%d @ %d,%d\n", + cre->window, cre->value_mask, cre->height, cre->width, cre->x, cre->y); - if ((win = win_find(dpy,xe->xmaprequest.window,1))) { + if ((win = win_find(dpy,cre->window,1))) { XSendEvent(dpy, cre->window, False, StructureNotifyMask, &(XEvent){ .xconfigure.type = ConfigureNotify, .xconfigure.display = win->sys->dpy, @@ -372,19 +468,40 @@ static void process_event(int type, XEvent *xe, win_t *root) .xconfigure.y = win->y, .xconfigure.width = win->w, .xconfigure.height = win->h, + .xconfigure.border_width = border, }); XSync(win->sys->dpy, False); } } else if (type == MapRequest) { - printf("map_req: %d\n", type); - if ((win = win_find(dpy,xe->xmaprequest.window,1))) { - if (!strut_add(root, win)) - wm_insert(win); - else - wm_update(); + printf("map_req: %lx\n", xe->xmaprequest.window); + win = win_find(dpy,xe->xmaprequest.window,1); + // fixme, for hide -> max, etc + if (win->state == ST_HIDE) { + wm_handle_state(win, win->state, ST_SHOW); + win->state = ST_SHOW; } - XMapWindow(dpy, xe->xmaprequest.window); + sys_show(win, win->state); + } + else if (type == ClientMessage) { + XClientMessageEvent *cme = &xe->xclient; + printf("client_msg: %lx - %ld %ld,%ld,%ld,%ld,%ld\n", + cme->window, cme->message_type, + cme->data.l[0], cme->data.l[1], cme->data.l[2], + cme->data.l[3], cme->data.l[4]); + if ((win = win_find(dpy,cme->window,0)) && + (cme->message_type == atoms[NET_STATE]) && + (cme->data.l[1] == atoms[NET_FULL] || + cme->data.l[2] == atoms[NET_FULL])) { + state_t next = (cme->data.l[0] == 1 || /* _NET_WM_STATE_ADD */ + (cme->data.l[0] == 2 && /* _NET_WM_STATE_TOGGLE */ + win->state != ST_FULL)) ? ST_FULL : ST_SHOW; + wm_handle_state(win, win->state, next); + sys_show(win, next); + } + } + else if (type == PropertyNotify) { + printf("prop: %lx - %d\n", xe->xproperty.window, xe->xproperty.state); } else { printf("unknown event: %d\n", type); @@ -408,6 +525,12 @@ static int xerror(Display *dpy, XErrorEvent *err) return xerrorxlib(dpy, err); } +static int xnoerror(Display *dpy, XErrorEvent *err) +{ + return 0; +} + + /******************** * System functions * ********************/ @@ -418,6 +541,8 @@ void sys_move(win_t *win, int x, int y, int w, int h) win->x = x; win->y = y; win->w = MAX(w,1+b); win->h = MAX(h,1+b); w = MAX(w-b,1); h = MAX(h-b,1); + XConfigureWindow(win->sys->dpy, win->sys->xid, CWX|CWY|CWWidth|CWHeight, + &(XWindowChanges) { .x=x, .y=y, .width=w, .height=h }); XMoveResizeWindow(win->sys->dpy, win->sys->xid, x, y, w, h); /* Flush events, so moving window doesn't cause re-focus @@ -441,56 +566,99 @@ void sys_focus(win_t *win) { //printf("sys_focus: %p\n", win); + /* Set actual focus */ + XSetInputFocus(win->sys->dpy, win->sys->xid, + RevertToPointerRoot, CurrentTime); + //win_msg(win, WM_FOCUS); + /* Set border on focused window */ - static win_t *last = NULL; if (last) XSetWindowBorder(last->sys->dpy, last->sys->xid, colors[CLR_UNFOCUS]); + XSync(win->sys->dpy, False); XSetWindowBorder(win->sys->dpy, win->sys->xid, colors[CLR_FOCUS]); - XSetWindowBorderWidth(win->sys->dpy, win->sys->xid, border); last = win; - - /* Set actual focus */ - XSetInputFocus(win->sys->dpy, win->sys->xid, - RevertToPointerRoot, CurrentTime); - XSendEvent(win->sys->dpy, win->sys->xid, False, NoEventMask, &(XEvent){ - .type = ClientMessage, - .xclient.window = win->sys->xid, - .xclient.message_type = atoms[WM_PROTO], - .xclient.format = 32, - .xclient.data.l[0] = atoms[WM_FOCUS], - .xclient.data.l[1] = CurrentTime, - }); } void sys_show(win_t *win, state_t state) { - win->sys->state = state; - switch (state) { - case ST_SHOW: - printf("sys_show: show\n"); - XMapWindow(win->sys->dpy, win->sys->xid); - XSync(win->sys->dpy, False); - return; - case ST_FULL: - printf("sys_show: full\n"); - XMapWindow(win->sys->dpy, win->sys->xid); - return; - case ST_SHADE: - printf("sys_show: shade\n"); + //if (win->state == state) + // return; + + /* Debug */ + printf("sys_show: %p: %s -> %s\n", win, + state_map[win->state], state_map[state]); + + /* Find screen */ + win_t *screen = NULL; + if (state == ST_FULL || state == ST_MAX) { + for (list_t *cur = screens; cur; cur = cur->next) { + screen = cur->data; + if (win->x >= screen->x && win->x <= screen->x+screen->w && + win->y >= screen->y && win->y <= screen->y+screen->h) + break; + } + } + + /* Update properties */ + if (state == ST_FULL) + XChangeProperty(win->sys->dpy, win->sys->xid, atoms[NET_STATE], XA_ATOM, 32, + PropModeReplace, (unsigned char*)&atoms[NET_FULL], 1); + else if (state != ST_FULL) + XChangeProperty(win->sys->dpy, win->sys->xid, atoms[NET_STATE], XA_ATOM, 32, + PropModeReplace, (unsigned char*)0, 0); + + /* Update border */ + if (state == ST_SHOW || state == ST_MAX || state == ST_SHADE) + XSetWindowBorderWidth(win->sys->dpy, win->sys->xid, border); + else if (state == ST_FULL) + XSetWindowBorderWidth(win->sys->dpy, win->sys->xid, 0); + + /* Map/Unmap window */ + if (state == ST_SHOW || state == ST_FULL || state == ST_MAX || state == ST_SHADE) XMapWindow(win->sys->dpy, win->sys->xid); - return; - case ST_ICON: - printf("sys_show: icon\n"); - return; - case ST_HIDE: - printf("sys_show: hide\n"); + else if (state == ST_HIDE) XUnmapWindow(win->sys->dpy, win->sys->xid); - return; - case ST_CLOSE: - printf("sys_show: close\n"); - XDestroyWindow(win->sys->dpy, win->sys->xid); - return; + + /* Resize windows */ + if (state == ST_SHOW) { + sys_move(win, win->x, win->y, win->w, win->h); + } else if (state == ST_MAX) { + sys_move(win, screen->x, screen->y, screen->w, screen->h); + } else if (state == ST_FULL) { + XWindowChanges wc = { + .x = screen->x - screen->sys->strut.left , + .y = screen->y - screen->sys->strut.top , + .width = screen->w + screen->sys->strut.left + screen->sys->strut.right, + .height = screen->h + screen->sys->strut.top + screen->sys->strut.bottom + }; + win->x = wc.x; win->y = wc.y; + win->w = wc.width; win->h = wc.height; + XConfigureWindow(win->sys->dpy, win->sys->xid, CWX|CWY|CWWidth|CWHeight, &wc); + XMoveResizeWindow(win->sys->dpy, win->sys->xid, wc.x, wc.y, wc.width, wc.height); + } else if (state == ST_SHADE) { + XConfigureWindow(win->sys->dpy, win->sys->xid, CWHeight, + &(XWindowChanges) { .height = stack }); + } + + /* Raise window */ + if (state == ST_FULL || state == ST_MAX) + XRaiseWindow(win->sys->dpy, win->sys->xid); + + /* Close windows */ + if (state == ST_CLOSE) { + if (!win_msg(win, WM_DELETE)) { + XGrabServer(win->sys->dpy); + XSetErrorHandler(xnoerror); + XSetCloseDownMode(win->sys->dpy, DestroyAll); + XKillClient(win->sys->dpy, win->sys->xid); + XSync(win->sys->dpy, False); + XSetErrorHandler(xerror); + XUngrabServer(win->sys->dpy); + } } + + /* Update state */ + win->state = state; } void sys_watch(win_t *win, event_t ev, mod_t mod) @@ -535,6 +703,7 @@ list_t *sys_info(win_t *win) screen->y = info[i].y_org; screen->w = info[i].width; screen->h = info[i].height; + screen->sys = new0(win_sys_t); screens = list_append(screens, screen); } } @@ -553,6 +722,7 @@ win_t *sys_init(void) Window xid; /* Load configuration */ + stack = conf_get_int("main.stack", stack); border = conf_get_int("main.border", border); no_capture = conf_get_int("main.no-capture", no_capture); @@ -563,9 +733,14 @@ win_t *sys_init(void) error("Unable to get root window"); /* Setup X11 data */ - atoms[WM_PROTO] = XInternAtom(dpy, "WM_PROTOCOLS", False); - atoms[WM_FOCUS] = XInternAtom(dpy, "WM_TAKE_FOCUS", False); - atoms[NET_STRUT] = XInternAtom(dpy, "_NET_WM_STRUT", False); + atoms[WM_PROTO] = XInternAtom(dpy, "WM_PROTOCOLS", False); + atoms[WM_FOCUS] = XInternAtom(dpy, "WM_TAKE_FOCUS", False); + atoms[WM_DELETE] = XInternAtom(dpy, "WM_DELETE_WINDOW", False); + atoms[NET_STATE] = XInternAtom(dpy, "_NET_WM_STATE", False); + atoms[NET_FULL] = XInternAtom(dpy, "_NET_WM_STATE_FULLSCREEN", False); + atoms[NET_STRUT] = XInternAtom(dpy, "_NET_WM_STRUT", False); + atoms[NET_TYPE] = XInternAtom(dpy, "_NET_WM_WINDOW_TYPE", False); + atoms[NET_DIALOG] = XInternAtom(dpy, "_NET_WM_WINDOW_TYPE_DIALOG", False); colors[CLR_FOCUS] = get_color(dpy, "#a0a0ff"); colors[CLR_UNFOCUS] = get_color(dpy, "#101066"); @@ -576,7 +751,7 @@ win_t *sys_init(void) XSelectInput(dpy, xid, SubstructureRedirectMask|SubstructureNotifyMask); xerrorxlib = XSetErrorHandler(xerror); - return win_find(dpy, xid, 1); + return root = win_find(dpy, xid, 1); } void sys_run(win_t *root) @@ -587,14 +762,11 @@ void sys_run(win_t *root) Window par, xid, *kids = NULL; if (XQueryTree(root->sys->dpy, root->sys->xid, &par, &xid, &kids, &nkids)) { - for(int i = 0; i < nkids; i++) { - win_t *win = win_find(root->sys->dpy, kids[i], 1); - if (win && win_viewable(win) && !strut_add(root,win)) - wm_insert(win); - } + for(int i = 0; i < nkids; i++) + if (win_viewable(root->sys->dpy, kids[i])) + win_find(root->sys->dpy, kids[i], 1); XFree(kids); } - wm_update(); // For struts } /* Main loop */ @@ -615,7 +787,9 @@ void sys_exit(void) void sys_free(win_t *root) { XCloseDisplay(root->sys->dpy); - while (screens) - screens = list_remove(screens, screens, 1); + while (screens) { + win_free(screens->data); + screens = list_remove(screens, screens, 0); + } tdestroy(cache, (void(*)(void*))win_free); }