/* * calmwm - the calm window manager * * Copyright (c) 2004 Martin Murray * * Permission to use, copy, modify, and distribute this software for any * purpose with or without fee is hereby granted, provided that the above * copyright notice and this permission notice appear in all copies. * * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. * * $OpenBSD$ */ #include #include "queue.h" #include #include #include #include #include #include #include #include #include #include #include "calmwm.h" #define HASH_MARKER "|1|" extern sig_atomic_t cwm_status; static void kbfunc_amount(int, unsigned int *, unsigned int *); void kbfunc_client_lower(struct client_ctx *cc, union arg *arg) { client_ptrsave(cc); client_lower(cc); } void kbfunc_client_raise(struct client_ctx *cc, union arg *arg) { client_raise(cc); } static void kbfunc_amount(int flags, unsigned int *mx, unsigned int *my) { #define CWM_FACTOR 10 int amt; amt = Conf.mamount; if (flags & CWM_BIGAMOUNT) amt *= CWM_FACTOR; switch (flags & DIRECTIONMASK) { case CWM_UP: *my -= amt; break; case CWM_DOWN: *my += amt; break; case CWM_RIGHT: *mx += amt; break; case CWM_LEFT: *mx -= amt; break; } } void kbfunc_ptrmove(struct client_ctx *cc, union arg *arg) { struct screen_ctx *sc = cc->sc; int x, y; unsigned int mx = 0, my = 0; kbfunc_amount(arg->i, &mx, &my); xu_ptr_getpos(sc->rootwin, &x, &y); xu_ptr_setpos(sc->rootwin, x + mx, y + my); } void kbfunc_client_move(struct client_ctx *cc, union arg *arg) { struct screen_ctx *sc = cc->sc; struct geom area; int x, y; unsigned int mx = 0, my = 0; if (cc->flags & CLIENT_FREEZE) return; kbfunc_amount(arg->i, &mx, &my); cc->geom.x += mx; if (cc->geom.x + cc->geom.w < 0) cc->geom.x = -cc->geom.w; if (cc->geom.x > sc->view.w - 1) cc->geom.x = sc->view.w - 1; cc->geom.y += my; if (cc->geom.y + cc->geom.h < 0) cc->geom.y = -cc->geom.h; if (cc->geom.y > sc->view.h - 1) cc->geom.y = sc->view.h - 1; area = screen_area(sc, cc->geom.x + cc->geom.w / 2, cc->geom.y + cc->geom.h / 2, CWM_GAP); cc->geom.x += client_snapcalc(cc->geom.x, cc->geom.x + cc->geom.w + (cc->bwidth * 2), area.x, area.x + area.w, sc->snapdist); cc->geom.y += client_snapcalc(cc->geom.y, cc->geom.y + cc->geom.h + (cc->bwidth * 2), area.y, area.y + area.h, sc->snapdist); client_move(cc); xu_ptr_getpos(cc->win, &x, &y); cc->ptr.x = x + mx; cc->ptr.y = y + my; client_ptrwarp(cc); } void kbfunc_client_resize(struct client_ctx *cc, union arg *arg) { unsigned int mx = 0, my = 0; if (cc->flags & CLIENT_FREEZE) return; kbfunc_amount(arg->i, &mx, &my); if ((cc->geom.w += mx * cc->hint.incw) < cc->hint.minw) cc->geom.w = cc->hint.minw; if ((cc->geom.h += my * cc->hint.inch) < cc->hint.minh) cc->geom.h = cc->hint.minh; client_resize(cc, 1); /* Make sure the pointer stays within the window. */ xu_ptr_getpos(cc->win, &cc->ptr.x, &cc->ptr.y); if (cc->ptr.x > cc->geom.w) cc->ptr.x = cc->geom.w - cc->bwidth; if (cc->ptr.y > cc->geom.h) cc->ptr.y = cc->geom.h - cc->bwidth; client_ptrwarp(cc); } void kbfunc_menu_client(struct client_ctx *cc, union arg *arg) { struct screen_ctx *sc = cc->sc; struct client_ctx *old_cc; struct menu *mi; struct menu_q menuq; old_cc = client_current(); TAILQ_INIT(&menuq); TAILQ_FOREACH(cc, &sc->clientq, entry) menuq_add(&menuq, cc, NULL); if ((mi = menu_filter(sc, &menuq, "window", NULL, 0, search_match_client, search_print_client)) != NULL) { cc = (struct client_ctx *)mi->ctx; if (cc->flags & CLIENT_HIDDEN) client_unhide(cc); if (old_cc) client_ptrsave(old_cc); client_ptrwarp(cc); } menuq_clear(&menuq); } void kbfunc_menu_cmd(struct client_ctx *cc, union arg *arg) { struct screen_ctx *sc = cc->sc; struct cmd *cmd; struct menu *mi; struct menu_q menuq; TAILQ_INIT(&menuq); TAILQ_FOREACH(cmd, &Conf.cmdq, entry) { if ((strcmp(cmd->name, "lock") == 0) || (strcmp(cmd->name, "term") == 0)) continue; /* search_match_text() needs mi->text */ menuq_add(&menuq, cmd, "%s", cmd->name); } if ((mi = menu_filter(sc, &menuq, "application", NULL, 0, search_match_text, search_print_cmd)) != NULL) u_spawn(((struct cmd *)mi->ctx)->path); menuq_clear(&menuq); } void kbfunc_menu_group(struct client_ctx *cc, union arg *arg) { struct screen_ctx *sc = cc->sc; struct group_ctx *gc; struct menu *mi; struct menu_q menuq; TAILQ_INIT(&menuq); TAILQ_FOREACH(gc, &sc->groupq, entry) { if (group_holds_only_sticky(gc)) continue; menuq_add(&menuq, gc, "%d %s", gc->num, gc->name); } if ((mi = menu_filter(sc, &menuq, "group", NULL, CWM_MENU_LIST, search_match_text, search_print_group)) != NULL) { gc = (struct group_ctx *)mi->ctx; (group_holds_only_hidden(gc)) ? group_show(gc) : group_hide(gc); } menuq_clear(&menuq); } void kbfunc_client_cycle(struct client_ctx *cc, union arg *arg) { client_cycle(cc->sc, arg->i); } void kbfunc_client_hide(struct client_ctx *cc, union arg *arg) { client_hide(cc); } void kbfunc_exec(struct client_ctx *cc, union arg *arg) { u_spawn(arg->c); } void kbfunc_exec_term(struct client_ctx *cc, union arg *arg) { struct cmd *cmd; TAILQ_FOREACH(cmd, &Conf.cmdq, entry) { if (strcmp(cmd->name, "term") == 0) u_spawn(cmd->path); } } void kbfunc_exec_lock(struct client_ctx *cc, union arg *arg) { struct cmd *cmd; TAILQ_FOREACH(cmd, &Conf.cmdq, entry) { if (strcmp(cmd->name, "lock") == 0) u_spawn(cmd->path); } } void kbfunc_menu_exec(struct client_ctx *cc, union arg *arg) { #define NPATHS 256 struct screen_ctx *sc = cc->sc; char **ap, *paths[NPATHS], *path, *pathcpy; char tpath[PATH_MAX]; const char *label; DIR *dirp; struct dirent *dp; struct menu *mi; struct menu_q menuq; int l, i, cmd = arg->i; struct stat sb; switch (cmd) { case CWM_MENU_EXEC: label = "exec"; break; case CWM_MENU_EXEC_WM: label = "wm"; break; default: errx(1, "kbfunc_menu_exec: invalid cmd %d", cmd); /*NOTREACHED*/ } TAILQ_INIT(&menuq); if ((path = getenv("PATH")) == NULL) path = _PATH_DEFPATH; pathcpy = path = xstrdup(path); for (ap = paths; ap < &paths[NPATHS - 1] && (*ap = strsep(&pathcpy, ":")) != NULL;) { if (**ap != '\0') ap++; } *ap = NULL; for (i = 0; i < NPATHS && paths[i] != NULL; i++) { if ((dirp = opendir(paths[i])) == NULL) continue; while ((dp = readdir(dirp)) != NULL) { (void)memset(tpath, '\0', sizeof(tpath)); l = snprintf(tpath, sizeof(tpath), "%s/%s", paths[i], dp->d_name); if (l == -1 || l >= sizeof(tpath)) continue; /* skip everything but regular files and symlinks */ if (dp->d_type != DT_REG && dp->d_type != DT_LNK) { /* use an additional stat-based check in case d_type isn't supported */ if (lstat(tpath, &sb) < 0) continue; if (!S_ISREG(sb.st_mode) && !S_ISLNK(sb.st_mode)) continue; } if (access(tpath, X_OK) == 0) menuq_add(&menuq, NULL, "%s", dp->d_name); } (void)closedir(dirp); } free(path); if ((mi = menu_filter(sc, &menuq, label, NULL, CWM_MENU_DUMMY | CWM_MENU_FILE, search_match_exec_path, NULL)) != NULL) { if (mi->text[0] == '\0') goto out; switch (cmd) { case CWM_MENU_EXEC: u_spawn(mi->text); break; case CWM_MENU_EXEC_WM: cwm_status = CWM_EXEC_WM; free(wm_argv); wm_argv = xstrdup(mi->text); break; default: errx(1, "kb_func: egad, cmd changed value!"); break; } } out: if (mi != NULL && mi->dummy) free(mi); menuq_clear(&menuq); } void kbfunc_menu_ssh(struct client_ctx *cc, union arg *arg) { struct screen_ctx *sc = cc->sc; struct cmd *cmd; struct menu *mi; struct menu_q menuq; FILE *fp; char *buf, *lbuf, *p; char hostbuf[_POSIX_HOST_NAME_MAX+1]; char path[PATH_MAX]; int l; size_t len; TAILQ_FOREACH(cmd, &Conf.cmdq, entry) { if (strcmp(cmd->name, "term") == 0) break; } TAILQ_INIT(&menuq); if ((fp = fopen(Conf.known_hosts, "r")) == NULL) { warn("kbfunc_menu_ssh: %s", Conf.known_hosts); goto menu; } lbuf = NULL; while ((buf = fgetln(fp, &len))) { if (buf[len - 1] == '\n') buf[len - 1] = '\0'; else { /* EOF without EOL, copy and add the NUL */ lbuf = xmalloc(len + 1); (void)memcpy(lbuf, buf, len); lbuf[len] = '\0'; buf = lbuf; } /* skip hashed hosts */ if (strncmp(buf, HASH_MARKER, strlen(HASH_MARKER)) == 0) continue; for (p = buf; *p != ',' && *p != ' ' && p != buf + len; p++) { /* do nothing */ } /* ignore badness */ if (p - buf + 1 > sizeof(hostbuf)) continue; (void)strlcpy(hostbuf, buf, p - buf + 1); menuq_add(&menuq, NULL, hostbuf); } free(lbuf); (void)fclose(fp); menu: if ((mi = menu_filter(sc, &menuq, "ssh", NULL, CWM_MENU_DUMMY, search_match_exec, NULL)) != NULL) { if (mi->text[0] == '\0') goto out; l = snprintf(path, sizeof(path), "%s -T '[ssh] %s' -e ssh %s", cmd->path, mi->text, mi->text); if (l == -1 || l >= sizeof(path)) goto out; u_spawn(path); } out: if (mi != NULL && mi->dummy) free(mi); menuq_clear(&menuq); } void kbfunc_client_label(struct client_ctx *cc, union arg *arg) { struct menu *mi; struct menu_q menuq; TAILQ_INIT(&menuq); /* dummy is set, so this will always return */ mi = menu_filter(cc->sc, &menuq, "label", cc->label, CWM_MENU_DUMMY, search_match_text, NULL); if (!mi->abort) { free(cc->label); cc->label = xstrdup(mi->text); } free(mi); } void kbfunc_client_delete(struct client_ctx *cc, union arg *arg) { client_send_delete(cc); } void kbfunc_group_toggle(struct client_ctx *cc, union arg *arg) { group_hidetoggle(cc->sc, arg->i); } void kbfunc_group_only(struct client_ctx *cc, union arg *arg) { group_only(cc->sc, arg->i); } void kbfunc_group_cycle(struct client_ctx *cc, union arg *arg) { group_cycle(cc->sc, arg->i); } void kbfunc_group_alltoggle(struct client_ctx *cc, union arg *arg) { group_alltoggle(cc->sc); } void kbfunc_client_grouptoggle(struct client_ctx *cc, union arg *arg) { if (arg->i == CWM_KBD) { /* For X apps that steal events. */ XGrabKeyboard(X_Dpy, cc->win, True, GrabModeAsync, GrabModeAsync, CurrentTime); } group_toggle_membership_enter(cc); } void kbfunc_client_movetogroup(struct client_ctx *cc, union arg *arg) { group_movetogroup(cc, arg->i); } void kbfunc_client_toggle_sticky(struct client_ctx *cc, union arg *arg) { client_toggle_sticky(cc); } void kbfunc_client_toggle_fullscreen(struct client_ctx *cc, union arg *arg) { client_toggle_fullscreen(cc); } void kbfunc_client_toggle_maximize(struct client_ctx *cc, union arg *arg) { client_toggle_maximize(cc); } void kbfunc_client_toggle_vmaximize(struct client_ctx *cc, union arg *arg) { client_toggle_vmaximize(cc); } void kbfunc_client_toggle_hmaximize(struct client_ctx *cc, union arg *arg) { client_toggle_hmaximize(cc); } void kbfunc_client_toggle_freeze(struct client_ctx *cc, union arg *arg) { client_toggle_freeze(cc); } void kbfunc_cwm_status(struct client_ctx *cc, union arg *arg) { cwm_status = arg->i; } void kbfunc_client_tile(struct client_ctx *cc, union arg *arg) { switch (arg->i) { case CWM_CLIENT_TILE_HORIZ: client_htile(cc); break; case CWM_CLIENT_TILE_VERT: client_vtile(cc); break; } }