mirror of
https://github.com/vale981/spectrwm
synced 2025-03-06 10:11:43 -05:00
262 lines
8 KiB
C
262 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 *dpy = 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, SWM_PROPLEN);
|
|
}
|
|
|
|
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 * display, 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");
|
|
dpy = display;
|
|
}
|
|
|
|
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 * display, 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(display))
|
|
parent = MyRoot(display);
|
|
|
|
id = (*func) (display, parent, x, y, width, height,
|
|
border_width, border, background);
|
|
|
|
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 int (RWF) (Display * _display, Window _window, Window _parent,
|
|
int x, int y);
|
|
|
|
/* XReparentWindow intercept hack */
|
|
int
|
|
XReparentWindow(Display * display, 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(display))
|
|
parent = MyRoot(display);
|
|
|
|
return (*func) (display, 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 int 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 (dpy != NULL) {
|
|
kp_add = XKeysymToKeycode(dpy, XK_KP_Add);
|
|
kp_subtract = XKeysymToKeycode(dpy, 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;
|
|
}
|