/* * (C)opyright MMIV-MMV Anselm R. Garbe * See LICENSE file for license details. */ #include #include #include #include "wm.h" #include static Page zero_page = {0}; static void select_frame(void *obj, char *cmd); static void _toggle_frame(void *obj, char *cmd); static void handle_after_write_page(IXPServer * s, File * f); static void handle_before_read_page(IXPServer * s, File * f); /* action table for /page/?/ namespace */ Action page_acttbl[] = { {"select", select_frame}, {"toggle", _toggle_frame}, {0, 0} }; Page * alloc_page(char *autodestroy) { Page *p = emalloc(sizeof(Page)); char buf[MAX_BUF], buf2[MAX_BUF]; int id = count_items((void **) pages) + 1; *p = zero_page; p->managed_rect = rect; if (core_files[CORE_PAGE_MANAGED_SIZE]->content) blitz_strtorect(dpy, &rect, &p->managed_rect, core_files[CORE_PAGE_MANAGED_SIZE]->content); snprintf(buf, sizeof(buf), "/page/%d", id); p->files[P_PREFIX] = ixp_create(ixps, buf); snprintf(buf, sizeof(buf), "/page/%d/floating", id); p->files[P_FLOATING_PREFIX] = ixp_create(ixps, buf); snprintf(buf, sizeof(buf), "/page/%d/floating/sel", id); p->files[P_FLOATING_SELECTED] = ixp_create(ixps, buf); p->files[P_FLOATING_SELECTED]->bind = 1; /* mount point */ snprintf(buf, sizeof(buf), "/page/%d/floating/name", id); p->files[P_FLOATING_LAYOUT] = wmii_create_ixpfile(ixps, buf, "float"); snprintf(buf, sizeof(buf), "/page/%d/managed", id); p->files[P_MANAGED_PREFIX] = ixp_create(ixps, buf); snprintf(buf, sizeof(buf), "/page/%d/managed/sel", id); p->files[P_MANAGED_SELECTED] = ixp_create(ixps, buf); p->files[P_MANAGED_SELECTED]->bind = 1; /* mount point */ snprintf(buf, sizeof(buf), "/page/%d/ctl", id); p->files[P_CTL] = ixp_create(ixps, buf); p->files[P_CTL]->after_write = handle_after_write_page; snprintf(buf, sizeof(buf), "/page/%d/name", id); snprintf(buf2, sizeof(buf2), "%d", id); p->files[P_NAME] = wmii_create_ixpfile(ixps, buf, buf2); snprintf(buf, sizeof(buf), "/page/%d/managed/name", id); p->files[P_MANAGED_LAYOUT] = wmii_create_ixpfile(ixps, buf, core_files[CORE_PAGE_LAYOUT]->content); p->files[P_MANAGED_LAYOUT]->after_write = handle_after_write_page; snprintf(buf, sizeof(buf), "/page/%d/mode", id); p->files[P_MODE] = ixp_create(ixps, buf); p->files[P_MODE]->bind = 1; snprintf(buf, sizeof(buf), "/page/%d/managed/size", id); p->files[P_MANAGED_SIZE] = wmii_create_ixpfile(ixps, buf, core_files[CORE_PAGE_MANAGED_SIZE]->content); p->files[P_MANAGED_SIZE]->after_write = handle_after_write_page; p->files[P_MANAGED_SIZE]->before_read = handle_before_read_page; snprintf(buf, sizeof(buf), "/page/%d/auto-destroy", id); p->files[P_AUTO_DESTROY] = wmii_create_ixpfile(ixps, buf, autodestroy); p->layout = get_layout(p->files[P_MANAGED_LAYOUT]->content); if (p->layout) { p->layout->init(p); p->files[P_MODE]->content = p->files[P_MANAGED_PREFIX]->content; } else p->files[P_MODE]->content = p->files[P_FLOATING_PREFIX]->content; pages = (Page **) attach_item_end((void **) pages, p, sizeof(Page *)); sel_page = index_item((void **) pages, p); core_files[CORE_PAGE_SELECTED]->content = p->files[P_PREFIX]->content; invoke_core_event(core_files[CORE_EVENT_PAGE_UPDATE]); return p; } void free_page(Page * p) { pages = (Page **) detach_item((void **) pages, p, sizeof(Page *)); if (pages) { int i; char buf[8]; if (sel_page - 1 >= 0) sel_page--; else sel_page = 0; for (i = 0; pages[i]; i++) { snprintf(buf, sizeof(buf), "%d", i + 1); free(pages[i]->files[P_PREFIX]->name); pages[i]->files[P_PREFIX]->name = estrdup(buf); free(pages[i]->files[P_NAME]->content); pages[i]->files[P_NAME]->content = estrdup(buf); pages[i]->files[P_NAME]->size = strlen(buf); } } if (p->layout) p->layout->deinit(p); core_files[CORE_PAGE_SELECTED]->content = 0; ixp_remove_file(ixps, p->files[P_PREFIX]); if (ixps->errstr) fprintf(stderr, "wmiiwm: free_page(): %s\n", ixps->errstr); free(p); } void draw_page(Page * p) { int i; if (!p) return; for (i = 0; p->floating && p->floating[i]; i++) draw_frame(p->floating[i]); for (i = 0; p->managed && p->managed[i]; i++) draw_frame(p->managed[i]); } XRectangle * rectangles(unsigned int *num) { XRectangle *result = 0; int i, j = 0; Window d1, d2; Window *wins; XWindowAttributes wa; XRectangle r; if (XQueryTree(dpy, root, &d1, &d2, &wins, num)) { result = emalloc(*num * sizeof(XRectangle)); for (i = 0; i < *num; i++) { if (!XGetWindowAttributes(dpy, wins[i], &wa)) continue; if (wa.override_redirect && (wa.map_state == IsViewable)) { r.x = wa.x; r.y = wa.y; r.width = wa.width; r.height = wa.height; result[j++] = r; } } } if (wins) { XFree(wins); } *num = j; return result; } static void _toggle_frame(void *obj, char *cmd) { Page *p = obj; Frame *f; if (!p || !p->layout) return; f = get_selected(p); if (!f) return; toggle_frame(f); } static Frame * select_toggled(Page * p, Frame * f) { if (is_managed_frame(f)) return p->floating_stack ? p->floating_stack[0] : 0; else return f = p->managed_stack ? p->managed_stack[0] : 0; return 0; } Frame * select_floating(Page * p, Frame * f, char *what) { int idx; if (!strncmp(what, "prev", 5)) { idx = index_prev_item((void **) p->floating, f); if (idx >= 0) return p->floating[idx]; } else if (!strncmp(what, "next", 5)) { idx = index_next_item((void **) p->floating, f); if (idx >= 0) return p->floating[idx]; } return 0; } static void center_pointer(Frame * f) { Window dummy; int wex, wey, ex, ey, i; unsigned int dmask; XRectangle *r; if (!f) return; r = rect_of_frame(f); XQueryPointer(dpy, f->win, &dummy, &dummy, &i, &i, &wex, &wey, &dmask); XTranslateCoordinates(dpy, f->win, root, wex, wey, &ex, &ey, &dummy); if (blitz_ispointinrect(ex, ey, r)) return; /* suppress EnterNotify's while mouse warping */ XSelectInput(dpy, root, ROOT_MASK & ~StructureNotifyMask); XWarpPointer(dpy, None, f->win, 0, 0, 0, 0, r->width / 2, r->height / 2); XSync(dpy, False); XSelectInput(dpy, root, ROOT_MASK); } static void select_frame(void *obj, char *cmd) { Page *p = (Page *) obj; Frame *f, *old, *old2; if (!p || !cmd) return; old2 = old = f = get_selected(p); if (!f) return; if (is_managed_frame(f)) old2 = p->managed[0]; if (!strncmp(cmd, "toggled", 8)) f = select_toggled(p, f); else if (is_managed_frame(f)) f = p->layout->select(f, cmd); else f = select_floating(p, f, cmd); if (!f) return; focus_frame(f, 1, 1, 1); center_pointer(f); if (old) draw_frame(old); if (old2 != old) draw_frame(old2); /* on zoom */ draw_frame(f); } void hide_page(Page * p) { int i; for (i = 0; p->floating && p->floating[i]; i++) XUnmapWindow(dpy, p->floating[i]->win); for (i = 0; p->managed && p->managed[i]; i++) XUnmapWindow(dpy, p->managed[i]->win); XSync(dpy, False); } void show_page(Page * p) { int i; for (i = 0; p->floating && p->floating[i]; i++) XMapWindow(dpy, p->floating[i]->win); for (i = 0; p->managed && p->managed[i]; i++) XMapWindow(dpy, p->managed[i]->win); XSync(dpy, False); } static void handle_before_read_page(IXPServer * s, File * f) { int i; XRectangle *rct = 0; for (i = 0; pages && pages[i]; i++) { if (pages[i]->files[P_MANAGED_SIZE] == f) { rct = &pages[i]->managed_rect; break; } } if (rct) { char buf[64]; snprintf(buf, 64, "%d,%d,%d,%d", rct->x, rct->y, rct->width, rct->height); if (f->content) free(f->content); f->content = estrdup(buf); f->size = strlen(buf); } } Layout * get_layout(char *name) { int i = 0; size_t len; if (!name) return 0; len = strlen(name); for (i = 0; layouts[i]; i++) { if (!strncmp(name, layouts[i]->name, len)) return layouts[i]; } return 0; } static void handle_after_write_page(IXPServer * s, File * f) { int i; for (i = 0; pages && pages[i]; i++) { Page *p = pages[i]; if (p->files[P_CTL] == f) { run_action(f, p, page_acttbl); return; } else if (p->files[P_MANAGED_SIZE] == f) { /* resize stuff */ blitz_strtorect(dpy, &rect, &p->managed_rect, p->files[P_MANAGED_SIZE]->content); if (!p->managed_rect.width) p->managed_rect.width = 10; if (!p->managed_rect.height) p->managed_rect.height = 10; if (p->layout) p->layout->arrange(p); draw_page(p); return; } else if (p->files[P_MANAGED_LAYOUT] == f) { int had_valid_layout = p->layout ? 1 : 0; if (p->layout) p->layout->deinit(p); p->layout = get_layout(p->files[P_MANAGED_LAYOUT]->content); if (p->layout) { p->layout->init(p); p->layout->arrange(p); if (!had_valid_layout) { int j; Frame **tmp = 0; for (j = 0; p->floating && p->floating[j]; j++) { if (!p->floating[j]->floating) tmp = (Frame **) attach_item_begin((void **) tmp, p-> floating[j], sizeof(Frame *)); } for (j = 0; tmp && tmp[j]; j++) toggle_frame(tmp[j]); free(tmp); } } if (!p->layout) { /* make all managed clients floating */ int j; Frame **tmp = 0; while (p->managed) { tmp = (Frame **) attach_item_begin((void **) tmp, p->managed[0], sizeof(Frame *)); detach_frame_from_page(p->managed[0], 1); } for (j = 0; tmp && tmp[j]; j++) { attach_Frameo_page(p, tmp[j], 0); resize_frame(tmp[j], rect_of_frame(tmp[j]), 0, 1); } free(tmp); } draw_page(p); invoke_core_event(core_files[CORE_EVENT_PAGE_UPDATE]); return; } } } void attach_Frameo_page(Page * p, Frame * f, int managed) { Frame *old = get_selected(p); XSelectInput(dpy, root, ROOT_MASK & ~StructureNotifyMask); XMapRaised(dpy, f->win); if (!f->floating && managed && p->layout) { int i; p->managed = (Frame **) attach_item_end((void **) p->managed, f, sizeof(Frame *)); p->managed_stack = (Frame **) attach_item_begin((void **) p->managed_stack, f, sizeof(Frame *)); wmii_move_ixpfile(f->files[F_PREFIX], p->files[P_MANAGED_PREFIX]); p->files[P_MANAGED_SELECTED]->content = f->files[F_PREFIX]->content; if (p == pages[sel_page]) for (i = 0; p->floating && p->floating[i]; i++) XRaiseWindow(dpy, p->floating[i]->win); } else { p->floating = (Frame **) attach_item_end((void **) p->floating, f, sizeof(Frame *)); p->floating_stack = (Frame **) attach_item_begin((void **) p->floating_stack, f, sizeof(Frame *)); wmii_move_ixpfile(f->files[F_PREFIX], p->files[P_FLOATING_PREFIX]); p->files[P_FLOATING_SELECTED]->content = f->files[F_PREFIX]->content; p->files[P_MODE]->content = p->files[P_FLOATING_PREFIX]->content; } f->page = p; focus_frame(f, 1, 0, 1); if (is_managed_frame(f) && p->layout) p->layout->manage(f); center_pointer(f); if (old) draw_frame(old); draw_frame(f); } void detach_frame_from_page(Frame * f, int ignore_focus_and_destroy) { Page *p = f->page; wmii_move_ixpfile(f->files[F_PREFIX], core_files[CORE_DETACHED_FRAME]); if (is_managed_frame(f)) { p->managed = (Frame **) detach_item((void **) p->managed, f, sizeof(Frame *)); p->managed_stack = (Frame **) detach_item((void **) p->managed_stack, f, sizeof(Frame *)); p->files[P_MANAGED_SELECTED]->content = 0; } else { p->floating = (Frame **) detach_item((void **) p->floating, f, sizeof(Frame *)); p->floating_stack = (Frame **) detach_item((void **) p->floating_stack, f, sizeof(Frame *)); p->files[P_FLOATING_SELECTED]->content = 0; } XUnmapWindow(dpy, f->win); if (is_managed_mode(p) && p->layout) p->layout->unmanage(f); f->page = 0; if (!ignore_focus_and_destroy) { Frame *fr; if (!p->managed && !p->floating && _strtonum(p->files[P_AUTO_DESTROY]->content, 0, 1)) { destroy_page(p); return; } focus_page(p, 0, 1); fr = get_selected(p); if (fr) { center_pointer(fr); draw_frame(fr); } } }