spectrwm/lib/swm_hack.c
Reginald Kennedy ebb72d68f2 Fix check_conn.
Resolve declaration shadows.
2012-07-28 04:48:27 +08:00

263 lines
8 KiB
C

/*
* Copyright (c) 2009 Marco Peereboom <marco@peereboom.us>
* Copyright (c) 2009 Ryan McBride <mcbride@countersiege.com>
*
* 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.
*/
/*
* Copyright (C) 2005-2007 Carsten Haitzler
* Copyright (C) 2006-2007 Kim Woelders
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to
* deal in the Software without restriction, including without limitation the
* rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
* sell copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies of the Software, its documentation and marketing & publicity
* materials, and acknowledgment shall be given in the documentation, materials
* and software packages that this Software was used.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
* THE AUTHORS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
* IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
*/
/*
* Basic hack mechanism (dlopen etc.) taken from e_hack.c in e17.
*/
#include <string.h>
#include <stdio.h>
#include <stdlib.h>
#include <dlfcn.h>
#include <X11/Xlib.h>
#include <X11/X.h>
#include <X11/Xatom.h>
#include <X11/Intrinsic.h>
/* dlopened libs so we can find the symbols in the real one to call them */
static void *lib_xlib = NULL;
static void *lib_xtlib = NULL;
static Window root = None;
static int xterm = 0;
static Display *display = NULL;
/* Find our root window */
static Window
MyRoot(Display * dpy)
{
char *s;
if (root != None)
return root;
root = DefaultRootWindow(dpy);
s = getenv("ENL_WM_ROOT");
if (!s)
return root;
sscanf(s, "%lx", &root);
return root;
}
typedef Atom (XIA) (Display *display, char *atom_name, Bool
only_if_exists);
typedef int (XCP) (Display *display, Window w, Atom property,
Atom type, int format, int mode, unsigned char *data,
int nelements);
#define SWM_PROPLEN (16)
void
set_property(Display *dpy, Window id, char *name, char *val)
{
Atom atom = 0;
char prop[SWM_PROPLEN];
static XIA *xia = NULL;
static XCP *xcp = NULL;
/* find the real Xlib and the real X function */
if (!lib_xlib)
lib_xlib = dlopen("libX11.so", RTLD_GLOBAL | RTLD_LAZY);
if (!xia)
xia = (XIA *) dlsym(lib_xlib, "XInternAtom");
if (!xcp)
xcp = (XCP *) dlsym(lib_xlib, "XChangeProperty");
/* Try to update the window's workspace property */
atom = (*xia)(dpy, name, False);
if (atom)
if (snprintf(prop, SWM_PROPLEN, "%s", val) < SWM_PROPLEN)
(*xcp)(dpy, id, atom, XA_STRING,
8, PropModeReplace, (unsigned char *)prop,
strlen((char *)prop));
}
typedef Window(CWF) (Display * _display, Window _parent, int _x,
int _y, unsigned int _width,
unsigned int _height,
unsigned int _border_width, int _depth,
unsigned int _class, Visual * _visual,
unsigned long _valuemask,
XSetWindowAttributes * _attributes);
/* XCreateWindow intercept hack */
Window
XCreateWindow(Display *dpy, Window parent, int x, int y,
unsigned int width, unsigned int height,
unsigned int border_width,
int depth, unsigned int clss, Visual * visual,
unsigned long valuemask, XSetWindowAttributes * attributes)
{
static CWF *func = NULL;
char *env;
Window id;
/* find the real Xlib and the real X function */
if (!lib_xlib)
lib_xlib = dlopen("libX11.so", RTLD_GLOBAL | RTLD_LAZY);
if (!func) {
func = (CWF *) dlsym(lib_xlib, "XCreateWindow");
display = dpy;
}
if (parent == DefaultRootWindow(display))
parent = MyRoot(display);
id = (*func) (display, parent, x, y, width, height, border_width,
depth, clss, visual, valuemask, attributes);
if (id) {
if ((env = getenv("_SWM_WS")) != NULL)
set_property(display, id, "_SWM_WS", env);
if ((env = getenv("_SWM_PID")) != NULL)
set_property(display, id, "_SWM_PID", env);
if ((env = getenv("_SWM_XTERM_FONTADJ")) != NULL) {
unsetenv("_SWM_XTERM_FONTADJ");
xterm = 1;
}
}
return (id);
}
typedef Window(CSWF) (Display * _display, Window _parent, int _x,
int _y, unsigned int _width,
unsigned int _height,
unsigned int _border_width,
unsigned long _border,
unsigned long _background);
/* XCreateSimpleWindow intercept hack */
Window
XCreateSimpleWindow(Display *dpy, Window parent, int x, int y,
unsigned int width, unsigned int height,
unsigned int border_width,
unsigned long border, unsigned long background)
{
static CSWF *func = NULL;
char *env;
Window id;
/* find the real Xlib and the real X function */
if (!lib_xlib)
lib_xlib = dlopen("libX11.so", RTLD_GLOBAL | RTLD_LAZY);
if (!func)
func = (CSWF *) dlsym(lib_xlib, "XCreateSimpleWindow");
if (parent == DefaultRootWindow(dpy))
parent = MyRoot(dpy);
id = (*func) (dpy, parent, x, y, width, height,
border_width, border, background);
if (id) {
if ((env = getenv("_SWM_WS")) != NULL)
set_property(dpy, id, "_SWM_WS", env);
if ((env = getenv("_SWM_PID")) != NULL)
set_property(dpy, id, "_SWM_PID", env);
if ((env = getenv("_SWM_XTERM_FONTADJ")) != NULL) {
unsetenv("_SWM_XTERM_FONTADJ");
xterm = 1;
}
}
return (id);
}
typedef int (RWF) (Display * _display, Window _window, Window _parent,
int x, int y);
/* XReparentWindow intercept hack */
int
XReparentWindow(Display *dpy, Window window, Window parent, int x, int y)
{
static RWF *func = NULL;
/* find the real Xlib and the real X function */
if (!lib_xlib)
lib_xlib = dlopen("libX11.so", RTLD_GLOBAL | RTLD_LAZY);
if (!func)
func = (RWF *) dlsym(lib_xlib, "XReparentWindow");
if (parent == DefaultRootWindow(dpy))
parent = MyRoot(dpy);
return (*func) (dpy, window, parent, x, y);
}
typedef void (ANEF) (XtAppContext app_context, XEvent *event_return);
int evcount = 0;
/*
* XtAppNextEvent Intercept Hack
* Normally xterm rejects "synthetic" (XSendEvent) events to prevent spoofing.
* We don't want to disable this completely, it's insecure. But hook here
* and allow these mostly harmless ones that we use to adjust fonts.
*/
void
XtAppNextEvent(XtAppContext app_context, XEvent *event_return)
{
static ANEF *func = NULL;
static KeyCode kp_add = 0, kp_subtract = 0;
/* find the real Xlib and the real X function */
if (!lib_xtlib)
lib_xtlib = dlopen("libXt.so", RTLD_GLOBAL | RTLD_LAZY);
if (!func) {
func = (ANEF *) dlsym(lib_xtlib, "XtAppNextEvent");
if (display != NULL) {
kp_add = XKeysymToKeycode(display, XK_KP_Add);
kp_subtract = XKeysymToKeycode(display, XK_KP_Subtract);
}
}
(*func) (app_context, event_return);
/* Return here if it's not an Xterm. */
if (!xterm)
return;
/* Allow spoofing of font change keystrokes. */
if ((event_return->type == KeyPress ||
event_return->type == KeyRelease) &&
event_return->xkey.state == ShiftMask &&
(event_return->xkey.keycode == kp_add ||
event_return->xkey.keycode == kp_subtract))
event_return->xkey.send_event = 0;
}