cwm/kbfunc.c

571 lines
12 KiB
C
Raw Normal View History

2007-04-27 21:58:48 +04:00
/*
2011-05-11 17:53:51 +04:00
* calmwm - the calm window manager
2008-04-16 00:24:41 +04:00
*
2011-05-11 17:53:51 +04:00
* Copyright (c) 2004 Martin Murray <mmurray@monkey.org>
*
* 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.
2007-04-27 21:58:48 +04:00
*
2011-05-11 17:53:51 +04:00
* $OpenBSD$
2007-04-27 21:58:48 +04:00
*/
#include <sys/param.h>
2012-11-26 19:27:22 +04:00
#include "queue.h"
#include <dirent.h>
#include <err.h>
#include <errno.h>
#include <paths.h>
2011-08-04 22:53:37 +04:00
#include <signal.h>
2012-11-09 07:52:02 +04:00
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
2007-04-27 21:58:48 +04:00
#include "calmwm.h"
2008-04-16 00:24:41 +04:00
#define HASH_MARKER "|1|"
extern char **cwm_argv;
extern sig_atomic_t xev_quit;
2007-04-27 21:58:48 +04:00
void
kbfunc_client_lower(struct client_ctx *cc, union arg *arg)
2007-04-27 21:58:48 +04:00
{
client_lower(cc);
}
void
kbfunc_client_raise(struct client_ctx *cc, union arg *arg)
2007-04-27 21:58:48 +04:00
{
client_raise(cc);
}
2012-06-28 01:52:10 +04:00
#define TYPEMASK (CWM_MOVE | CWM_RESIZE | CWM_PTRMOVE | CWM_SNAP)
#define MOVEMASK (CWM_UP | CWM_DOWN | CWM_LEFT | CWM_RIGHT)
void
kbfunc_moveresize(struct client_ctx *cc, union arg *arg)
{
struct screen_ctx *sc = cc->sc;
2008-05-19 22:53:09 +04:00
int x, y, flags, amt;
u_int mx, my;
2012-06-28 01:52:10 +04:00
int ox, oy, ow, oh;
if (cc->flags & CLIENT_FREEZE)
return;
mx = my = 0;
flags = arg->i;
amt = Conf.mamount;
if (flags & CWM_BIGMOVE) {
flags -= CWM_BIGMOVE;
2008-05-19 22:53:09 +04:00
amt = amt * 10;
}
switch (flags & MOVEMASK) {
case CWM_UP:
my -= amt;
break;
2008-04-16 00:24:41 +04:00
case CWM_DOWN:
my += amt;
break;
case CWM_RIGHT:
mx += amt;
break;
case CWM_LEFT:
mx -= amt;
break;
}
switch (flags & TYPEMASK) {
case CWM_MOVE:
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;
cc->geom.x += client_snapcalc(cc->geom.x,
cc->geom.x + cc->geom.w + (cc->bwidth * 2),
sc->work.x, sc->work.w, Conf.snapdist);
cc->geom.y += client_snapcalc(cc->geom.y,
cc->geom.y + cc->geom.h + (cc->bwidth * 2),
sc->work.y, sc->work.h, Conf.snapdist);
client_move(cc);
xu_ptr_getpos(cc->win, &x, &y);
cc->ptr.x = x + mx;
cc->ptr.y = y + my;
client_ptrwarp(cc);
break;
case CWM_RESIZE:
if ((cc->geom.w += mx) < 1)
cc->geom.w = 1;
if ((cc->geom.h += my) < 1)
cc->geom.h = 1;
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);
break;
case CWM_PTRMOVE:
xu_ptr_getpos(sc->rootwin, &x, &y);
xu_ptr_setpos(sc->rootwin, x + mx, y + my);
break;
2012-06-28 01:52:10 +04:00
case CWM_SNAP:
ox = cc->geom.x; ow = cc->geom.w;
oy = cc->geom.y; oh = cc->geom.h;
2013-05-08 13:15:30 +04:00
#define sw sc->work.w
#define sh sc->work.h
2012-06-28 01:52:10 +04:00
#define bw 2 * cc->bwidth
#define nw cc->geom.w
#define nh cc->geom.h
2012-06-28 01:52:10 +04:00
#define nx cc->geom.x
#define ny cc->geom.y
if (flags & CWM_UP) {
if (oy > sh - oh - bw) ny = sh - oh - bw;
else if ( oy > (sh - oh - bw) / 2 ) ny = (sh - oh - bw) / 2;
else ny = 0;
} else if (flags & CWM_DOWN) {
if (oy < 0) ny = 0;
else if (oy < (sh - oh - bw) / 2) ny = (sh - oh - bw)/2;
else ny = sh - oh - bw;
} else if (flags & CWM_LEFT) {
if (ox + bw > sw - ow) nx = sw - ow - bw;
else if (ox > (sw - ow - bw) / 2) nx = (sw - ow - bw) / 2;
else cc->geom.x = 0;
} else if (flags & CWM_RIGHT) {
if (ox < 0) nx = 0;
else if (ox < (sw - ow - bw) / 2) nx = (sw - ow - bw) / 2;
else nx = sw - ow - bw;
2012-07-06 02:49:56 +04:00
} else if (flags & ( CWM_GROW | CWM_SHRINK)) {
if (flags & CWM_GROW ) {
if ((cc->flags & CLIENT_MAXFLAGS) == CLIENT_MAXIMIZED) {
} else if (ow + bw < sw / 3) {
nw = sw / 3 - bw;
nh = sh / 3 - bw;
} else if (ow + bw < sw / 2) {
nw = sw / 2 - bw;
nh = sh / 2 - bw;
} else if (ow + bw < sw * 2 / 3) {
nw = sw * 2 / 3 - bw;
nh = sh * 2 / 3 - bw;
} else {
client_maximize(cc);
goto end;
}
2012-06-28 01:52:10 +04:00
} else {
2012-07-06 02:49:56 +04:00
if ((cc->flags & CLIENT_MAXFLAGS) == CLIENT_MAXIMIZED) {
client_maximize(cc);
goto end;
} else if (ow + bw > sw * 2 / 3) {
nw = sw * 2 / 3 - bw;
nh = sh * 2 / 3 - bw;
} else if (ow + bw> sw / 2) {
nw = sw / 2 - bw;
nh = sh / 2 - bw;
} else if (ow + bw > sw / 3) {
nw = sw / 3 - bw;
nh = sh / 3 - bw;
}
2012-06-28 01:52:10 +04:00
}
2012-07-06 02:49:56 +04:00
nx += (ow - nw)/2;
ny += (oh - nh)/2;
if (abs(ox) < Conf.snapdist) nx = 0;
else if (abs(sw - bw - ox - ow) < Conf.snapdist) nx = sw - bw - nw;
if (abs(oy) < Conf.snapdist) ny = 0;
else if (abs(sh - bw - oy - oh) < Conf.snapdist) ny = sh - nh - bw;
if (nx < 0) nx = 0;
else if (nx + nw + bw > sw) nx = sw - nw - bw;
if (ny < 0) ny = 0;
else if (ny + nh + bw > sh) ny = sh - nh - bw;
2012-06-28 01:52:10 +04:00
}
client_ptrsave(cc);
client_resize(cc, 0);
2012-06-28 01:52:10 +04:00
cc->ptr.x += (nw - ow)/2;
cc->ptr.y += (nh - oh)/2;
client_ptrwarp(cc);
end:
break;
default:
warnx("invalid flags passed to kbfunc_client_moveresize");
}
}
2007-04-27 21:58:48 +04:00
void
2009-12-07 22:44:31 +03:00
kbfunc_client_search(struct client_ctx *cc, union arg *arg)
2007-04-27 21:58:48 +04:00
{
struct screen_ctx *sc = cc->sc;
2009-12-07 22:44:31 +03:00
struct client_ctx *old_cc;
2008-05-19 22:53:09 +04:00
struct menu *mi;
struct menu_q menuq;
old_cc = client_current();
2008-04-16 00:24:41 +04:00
2007-04-27 21:58:48 +04:00
TAILQ_INIT(&menuq);
2008-04-16 00:24:41 +04:00
TAILQ_FOREACH(cc, &Clientq, entry) {
mi = xcalloc(1, sizeof(*mi));
(void)strlcpy(mi->text, cc->name, sizeof(mi->text));
2007-04-27 21:58:48 +04:00
mi->ctx = cc;
TAILQ_INSERT_TAIL(&menuq, mi, entry);
}
if ((mi = menu_filter(sc, &menuq, "window", NULL, 0,
search_match_client, search_print_client)) != NULL) {
2007-04-27 21:58:48 +04:00
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);
2007-04-27 21:58:48 +04:00
}
void
2009-12-07 22:44:31 +03:00
kbfunc_menu_search(struct client_ctx *cc, union arg *arg)
2007-04-27 21:58:48 +04:00
{
struct screen_ctx *sc = cc->sc;
struct cmd *cmd;
struct menu *mi;
struct menu_q menuq;
2007-04-27 21:58:48 +04:00
TAILQ_INIT(&menuq);
TAILQ_FOREACH(cmd, &Conf.cmdq, entry) {
mi = xcalloc(1, sizeof(*mi));
(void)strlcpy(mi->text, cmd->label, sizeof(mi->text));
2007-04-27 21:58:48 +04:00
mi->ctx = cmd;
TAILQ_INSERT_TAIL(&menuq, mi, entry);
}
if ((mi = menu_filter(sc, &menuq, "application", NULL, 0,
search_match_text, NULL)) != NULL)
2007-04-27 21:58:48 +04:00
u_spawn(((struct cmd *)mi->ctx)->image);
menuq_clear(&menuq);
2007-04-27 21:58:48 +04:00
}
void
2009-12-07 22:44:31 +03:00
kbfunc_client_cycle(struct client_ctx *cc, union arg *arg)
2007-04-27 21:58:48 +04:00
{
struct screen_ctx *sc = cc->sc;
2007-04-27 21:58:48 +04:00
/* XXX for X apps that ignore events */
XGrabKeyboard(X_Dpy, sc->rootwin, True,
GrabModeAsync, GrabModeAsync, CurrentTime);
client_cycle(sc, arg->i);
2007-04-27 21:58:48 +04:00
}
void
kbfunc_client_hide(struct client_ctx *cc, union arg *arg)
2007-04-27 21:58:48 +04:00
{
client_hide(cc);
}
void
kbfunc_cmdexec(struct client_ctx *cc, union arg *arg)
2007-04-27 21:58:48 +04:00
{
u_spawn(arg->c);
2007-04-27 21:58:48 +04:00
}
void
kbfunc_term(struct client_ctx *cc, union arg *arg)
2007-04-27 21:58:48 +04:00
{
u_spawn(Conf.termpath);
2007-04-27 21:58:48 +04:00
}
void
kbfunc_lock(struct client_ctx *cc, union arg *arg)
2007-04-27 21:58:48 +04:00
{
u_spawn(Conf.lockpath);
2007-04-27 21:58:48 +04:00
}
void
2009-12-07 22:44:31 +03:00
kbfunc_exec(struct client_ctx *cc, union arg *arg)
{
#define NPATHS 256
struct screen_ctx *sc = cc->sc;
char **ap, *paths[NPATHS], *path, *pathcpy, *label;
char tpath[MAXPATHLEN];
DIR *dirp;
struct dirent *dp;
struct menu *mi;
struct menu_q menuq;
int l, i, cmd = arg->i;
2007-11-28 19:35:52 +03:00
2008-04-16 00:24:41 +04:00
switch (cmd) {
2007-11-28 19:35:52 +03:00
case CWM_EXEC_PROGRAM:
label = "exec";
break;
case CWM_EXEC_WM:
label = "wm";
break;
default:
err(1, "kbfunc_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) {
/* skip everything but regular files and symlinks */
if (dp->d_type != DT_REG && dp->d_type != DT_LNK)
continue;
(void)memset(tpath, '\0', sizeof(tpath));
l = snprintf(tpath, sizeof(tpath), "%s/%s", paths[i],
dp->d_name);
/* check for truncation etc */
if (l == -1 || l >= (int)sizeof(tpath))
continue;
if (access(tpath, X_OK) == 0) {
mi = xcalloc(1, sizeof(*mi));
(void)strlcpy(mi->text,
dp->d_name, sizeof(mi->text));
TAILQ_INSERT_TAIL(&menuq, mi, entry);
}
}
(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;
2007-11-28 19:35:52 +03:00
switch (cmd) {
case CWM_EXEC_PROGRAM:
u_spawn(mi->text);
break;
case CWM_EXEC_WM:
u_exec(mi->text);
warn("%s", mi->text);
2007-11-28 19:35:52 +03:00
break;
default:
err(1, "kb_func: egad, cmd changed value!");
break;
}
}
out:
if (mi != NULL && mi->dummy)
free(mi);
menuq_clear(&menuq);
}
void
2009-12-07 22:44:31 +03:00
kbfunc_ssh(struct client_ctx *cc, union arg *arg)
{
struct screen_ctx *sc = cc->sc;
struct menu *mi;
struct menu_q menuq;
FILE *fp;
char *buf, *lbuf, *p;
2012-12-18 03:54:57 +04:00
char hostbuf[MAXHOSTNAMELEN];
char cmd[256];
int l;
size_t len;
2012-12-18 03:54:57 +04:00
if ((fp = fopen(Conf.known_hosts, "r")) == NULL) {
warn("kbfunc_ssh: %s", Conf.known_hosts);
return;
2012-12-18 03:54:57 +04:00
}
TAILQ_INIT(&menuq);
2013-04-08 17:02:31 +04:00
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);
mi = xcalloc(1, sizeof(*mi));
(void)strlcpy(mi->text, hostbuf, sizeof(mi->text));
TAILQ_INSERT_TAIL(&menuq, mi, entry);
}
free(lbuf);
(void)fclose(fp);
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(cmd, sizeof(cmd), "%s -e ssh %s", Conf.termpath,
mi->text);
if (l != -1 && l < sizeof(cmd))
u_spawn(cmd);
}
out:
if (mi != NULL && mi->dummy)
free(mi);
menuq_clear(&menuq);
}
2007-04-27 21:58:48 +04:00
void
kbfunc_client_label(struct client_ctx *cc, union arg *arg)
2007-04-27 21:58:48 +04:00
{
struct menu *mi;
struct menu_q menuq;
TAILQ_INIT(&menuq);
2009-11-28 20:52:12 +03:00
/* 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);
2007-04-27 21:58:48 +04:00
}
void
kbfunc_client_delete(struct client_ctx *cc, union arg *arg)
2007-04-27 21:58:48 +04:00
{
client_send_delete(cc);
}
void
kbfunc_client_group(struct client_ctx *cc, union arg *arg)
2007-04-27 21:58:48 +04:00
{
group_hidetoggle(cc->sc, arg->i);
2007-04-27 21:58:48 +04:00
}
void
kbfunc_client_grouponly(struct client_ctx *cc, union arg *arg)
{
group_only(cc->sc, arg->i);
}
2007-04-27 21:58:48 +04:00
void
kbfunc_client_cyclegroup(struct client_ctx *cc, union arg *arg)
2007-04-27 21:58:48 +04:00
{
group_cycle(cc->sc, arg->i);
2007-04-27 21:58:48 +04:00
}
void
kbfunc_client_nogroup(struct client_ctx *cc, union arg *arg)
2007-04-27 21:58:48 +04:00
{
group_alltoggle(cc->sc);
2007-04-27 21:58:48 +04:00
}
void
kbfunc_client_grouptoggle(struct client_ctx *cc, union arg *arg)
{
/* XXX for stupid X apps like xpdf and gvim */
XGrabKeyboard(X_Dpy, cc->win, True,
GrabModeAsync, GrabModeAsync, CurrentTime);
group_sticky_toggle_enter(cc);
}
void
kbfunc_client_movetogroup(struct client_ctx *cc, union arg *arg)
{
group_movetogroup(cc, arg->i);
}
2007-04-27 21:58:48 +04:00
void
kbfunc_client_maximize(struct client_ctx *cc, union arg *arg)
2007-04-27 21:58:48 +04:00
{
client_maximize(cc);
}
void
kbfunc_client_vmaximize(struct client_ctx *cc, union arg *arg)
2007-04-27 21:58:48 +04:00
{
client_vmaximize(cc);
2007-04-27 21:58:48 +04:00
}
void
kbfunc_client_hmaximize(struct client_ctx *cc, union arg *arg)
{
client_hmaximize(cc);
}
void
kbfunc_client_freeze(struct client_ctx *cc, union arg *arg)
{
client_freeze(cc);
}
void
kbfunc_quit_wm(struct client_ctx *cc, union arg *arg)
{
xev_quit = 1;
}
void
kbfunc_restart(struct client_ctx *cc, union arg *arg)
{
(void)setsid();
(void)execvp(cwm_argv[0], cwm_argv);
}
void
kbfunc_tile(struct client_ctx *cc, union arg *arg)
{
switch (arg->i) {
case CWM_TILE_HORIZ:
client_htile(cc);
break;
case CWM_TILE_VERT:
client_vtile(cc);
break;
}
}