mirror of
https://github.com/leahneukirchen/cwm.git
synced 2023-08-10 21:13:12 +03:00

* refs/heads/master: more client vs screen context differences If a client sets hints, honor them for kb resize requests, just like we do for mouse based resize requests. Move kb pointer movement out of the kbfunc_client_moveresize since it's got nothing to do with clients, thus doing flags work causes lots of waste and almost useless jumpy pointer movements; while here, split out move and resize since they share almost no code, just like mouse client move/resize; factor out amount and factor. Still wonder why this is here, but it works now. pledge "stdio rpath proc exec" cwm before main event loop, after init/setup - mostly for menu building. Partial revert of replacing screen_area() with region_find(); until a fix for a regression is found; this bug has been around for a long time it seems, but this change exposed it. Likely need to track clients in to and out of regions. Use position on root to figure out region. Start cleaning up name vs function differences; replace magic numbers. Clean up unused defines. Extend region to include both view and work areas; switch to region_find() which no longer needs to recalculate gap each time a client (or menu) is created or altered. If no RandR, fall back to display dimensions while building regions instead of during execution.
230 lines
5.2 KiB
C
230 lines
5.2 KiB
C
/*
|
|
* calmwm - the calm window manager
|
|
*
|
|
* Copyright (c) 2004 Marius Aamodt Eriksen <marius@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.
|
|
*
|
|
* $OpenBSD$
|
|
*/
|
|
|
|
#include <sys/types.h>
|
|
#include "queue.h"
|
|
#include <sys/wait.h>
|
|
|
|
#include <err.h>
|
|
#include <errno.h>
|
|
#include <getopt.h>
|
|
#include <limits.h>
|
|
#include <locale.h>
|
|
#include <pwd.h>
|
|
#include <signal.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <unistd.h>
|
|
|
|
#include "calmwm.h"
|
|
|
|
Display *X_Dpy;
|
|
Time Last_Event_Time = CurrentTime;
|
|
Atom cwmh[CWMH_NITEMS];
|
|
Atom ewmh[EWMH_NITEMS];
|
|
|
|
struct screen_ctx_q Screenq = TAILQ_HEAD_INITIALIZER(Screenq);
|
|
|
|
int HasRandr, Randr_ev;
|
|
struct conf Conf;
|
|
const char *homedir;
|
|
char *wm_argv;
|
|
volatile sig_atomic_t cwm_status;
|
|
|
|
static void sighdlr(int);
|
|
static int x_errorhandler(Display *, XErrorEvent *);
|
|
static void x_init(const char *);
|
|
static void x_teardown(void);
|
|
static int x_wmerrorhandler(Display *, XErrorEvent *);
|
|
|
|
int
|
|
main(int argc, char **argv)
|
|
{
|
|
const char *conf_file = NULL;
|
|
char *conf_path, *display_name = NULL;
|
|
int ch;
|
|
struct passwd *pw;
|
|
|
|
if (!setlocale(LC_CTYPE, "") || !XSupportsLocale())
|
|
warnx("no locale support");
|
|
mbtowc(NULL, NULL, MB_CUR_MAX);
|
|
|
|
wm_argv = u_argv(argv);
|
|
while ((ch = getopt(argc, argv, "c:d:")) != -1) {
|
|
switch (ch) {
|
|
case 'c':
|
|
conf_file = optarg;
|
|
break;
|
|
case 'd':
|
|
display_name = optarg;
|
|
break;
|
|
default:
|
|
usage();
|
|
}
|
|
}
|
|
argc -= optind;
|
|
argv += optind;
|
|
|
|
if (signal(SIGCHLD, sighdlr) == SIG_ERR)
|
|
err(1, "signal");
|
|
|
|
if ((homedir = getenv("HOME")) == NULL || *homedir == '\0') {
|
|
pw = getpwuid(getuid());
|
|
if (pw != NULL && pw->pw_dir != NULL && *pw->pw_dir != '\0')
|
|
homedir = pw->pw_dir;
|
|
else
|
|
homedir = "/";
|
|
}
|
|
|
|
if (conf_file == NULL)
|
|
xasprintf(&conf_path, "%s/%s", homedir, CONFFILE);
|
|
else
|
|
conf_path = xstrdup(conf_file);
|
|
|
|
if (access(conf_path, R_OK) != 0) {
|
|
if (conf_file != NULL)
|
|
warn("%s", conf_file);
|
|
free(conf_path);
|
|
conf_path = NULL;
|
|
}
|
|
|
|
conf_init(&Conf);
|
|
if (conf_path && (parse_config(conf_path, &Conf) == -1))
|
|
warnx("config file %s has errors", conf_path);
|
|
free(conf_path);
|
|
|
|
x_init(display_name);
|
|
cwm_status = CWM_RUNNING;
|
|
|
|
if (pledge("stdio rpath proc exec", NULL) == -1)
|
|
err(1, "pledge");
|
|
|
|
while (cwm_status == CWM_RUNNING)
|
|
xev_process();
|
|
x_teardown();
|
|
if (cwm_status == CWM_EXEC_WM)
|
|
u_exec(wm_argv);
|
|
|
|
return(0);
|
|
}
|
|
|
|
static void
|
|
x_init(const char *dpyname)
|
|
{
|
|
int i;
|
|
|
|
if ((X_Dpy = XOpenDisplay(dpyname)) == NULL)
|
|
errx(1, "unable to open display \"%s\"", XDisplayName(dpyname));
|
|
|
|
XSetErrorHandler(x_wmerrorhandler);
|
|
XSelectInput(X_Dpy, DefaultRootWindow(X_Dpy), SubstructureRedirectMask);
|
|
XSync(X_Dpy, False);
|
|
XSetErrorHandler(x_errorhandler);
|
|
|
|
HasRandr = XRRQueryExtension(X_Dpy, &Randr_ev, &i);
|
|
|
|
conf_atoms();
|
|
conf_cursor(&Conf);
|
|
|
|
for (i = 0; i < ScreenCount(X_Dpy); i++)
|
|
screen_init(i);
|
|
}
|
|
|
|
static void
|
|
x_teardown(void)
|
|
{
|
|
struct screen_ctx *sc;
|
|
unsigned int i;
|
|
|
|
conf_clear(&Conf);
|
|
|
|
TAILQ_FOREACH(sc, &Screenq, entry) {
|
|
for (i = 0; i < CWM_COLOR_NITEMS; i++)
|
|
XftColorFree(X_Dpy, DefaultVisual(X_Dpy, sc->which),
|
|
DefaultColormap(X_Dpy, sc->which),
|
|
&sc->xftcolor[i]);
|
|
XftDrawDestroy(sc->xftdraw);
|
|
XftFontClose(X_Dpy, sc->xftfont);
|
|
XUnmapWindow(X_Dpy, sc->menuwin);
|
|
XDestroyWindow(X_Dpy, sc->menuwin);
|
|
XUngrabKey(X_Dpy, AnyKey, AnyModifier, sc->rootwin);
|
|
}
|
|
XUngrabPointer(X_Dpy, CurrentTime);
|
|
XUngrabKeyboard(X_Dpy, CurrentTime);
|
|
for (i = 0; i < CF_NITEMS; i++)
|
|
XFreeCursor(X_Dpy, Conf.cursor[i]);
|
|
XSync(X_Dpy, False);
|
|
XSetInputFocus(X_Dpy, PointerRoot, RevertToPointerRoot, CurrentTime);
|
|
XCloseDisplay(X_Dpy);
|
|
}
|
|
|
|
static int
|
|
x_wmerrorhandler(Display *dpy, XErrorEvent *e)
|
|
{
|
|
errx(1, "root window unavailable - perhaps another wm is running?");
|
|
|
|
return(0);
|
|
}
|
|
|
|
static int
|
|
x_errorhandler(Display *dpy, XErrorEvent *e)
|
|
{
|
|
#ifdef DEBUG
|
|
char msg[80], number[80], req[80];
|
|
|
|
XGetErrorText(X_Dpy, e->error_code, msg, sizeof(msg));
|
|
(void)snprintf(number, sizeof(number), "%d", e->request_code);
|
|
XGetErrorDatabaseText(X_Dpy, "XRequest", number,
|
|
"<unknown>", req, sizeof(req));
|
|
|
|
warnx("%s(0x%x): %s", req, (unsigned int)e->resourceid, msg);
|
|
#endif
|
|
return(0);
|
|
}
|
|
|
|
static void
|
|
sighdlr(int sig)
|
|
{
|
|
pid_t pid;
|
|
int save_errno = errno, status;
|
|
|
|
switch (sig) {
|
|
case SIGCHLD:
|
|
/* Collect dead children. */
|
|
while ((pid = waitpid(WAIT_ANY, &status, WNOHANG)) > 0 ||
|
|
(pid < 0 && errno == EINTR))
|
|
;
|
|
break;
|
|
}
|
|
|
|
errno = save_errno;
|
|
}
|
|
|
|
void
|
|
usage(void)
|
|
{
|
|
extern char *__progname;
|
|
|
|
(void)fprintf(stderr, "usage: %s [-c file] [-d display]\n",
|
|
__progname);
|
|
exit(1);
|
|
}
|