openbsd-ports/x11/x2x/patches/patch-x2x_c

811 lines
28 KiB
Plaintext
Raw Normal View History

$OpenBSD: patch-x2x_c,v 1.1 2011/11/22 09:33:09 jasper Exp $
- Various fixes including -north/-south support.
From Debian's x2x_1.27-8 patch.
--- x2x.c.orig Wed Aug 20 18:14:52 1997
+++ x2x.c Mon Nov 21 14:57:27 2011
@@ -1,7 +1,7 @@
/*
- * x2x: Uses the XTEST extension to forward keystrokes from a window on
- * one display to another display. Useful for desks
- * with multiple keyboards.
+ * x2x: Uses the XTEST extension to forward mouse movements and keystrokes
+ * from a window on one display to another display. Useful for
+ * desks with multiple keyboards.
*
* Copyright (c) 1997
* Digital Equipment Corporation. All rights reserved.
@@ -37,26 +37,42 @@
* OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
* ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
+
+/*
+ * Modified on 3 Oct 1998 by Charles Briscoe-Smith:
+ * added options -north and -south
+ */
+
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
+#include <unistd.h>
#include <X11/Xlib.h>
#include <X11/Xresource.h>
#include <X11/Xutil.h>
#include <X11/cursorfont.h>
#include <X11/Xatom.h> /* for selection */
+#include <X11/extensions/XTest.h>
#include <sys/types.h> /* for select */
#include <sys/time.h> /* for select */
-#include "format.h"
/*#define DEBUG*/
+#ifndef MIN
+#define MIN(x,y) (((x) < (y)) ? (x) : (y))
+#endif
+#ifndef MAX
+#define MAX(x,y) (((x) > (y)) ? (x) : (y))
+#endif
+
/**********
* definitions for edge
**********/
#define EDGE_NONE 0 /* don't transfer between edges of screens */
-#define EDGE_EAST 1 /* from display is on the east side of to display */
-#define EDGE_WEST 2 /* from display is on the west side of to display */
+#define EDGE_NORTH 1 /* from display is on the north side of to display */
+#define EDGE_SOUTH 2 /* from display is on the south side of to display */
+#define EDGE_EAST 3 /* from display is on the east side of to display */
+#define EDGE_WEST 4 /* from display is on the west side of to display */
/**********
* functions
@@ -92,25 +108,6 @@ static void RefreshPointerMapping();
static void Usage();
/**********
- * text formatting instructions
- **********/
-#define toDpyFormatLength (sizeof(toDpyFormat) / sizeof(Format))
-static Format toDpyFormat[] = {
- FormatMeasureText,
- FormatSetLeft, 0,
- FormatSetTop, 0,
- FormatAddHalfTextX, 1,
- FormatAddHalfTextY, 3,
- FormatString, (Format)"unknown",
- FormatAddHalfTextX, 1,
- FormatAddHalfTextY, 1
- };
-/* indexes of values to be filled in at runtime */
-#define toDpyLeftIndex 2
-#define toDpyTopIndex 4
-#define toDpyStringIndex 10
-
-/**********
* stuff for selection forwarding
**********/
typedef struct _dpyxtra {
@@ -135,6 +132,8 @@ typedef struct _fakestr {
#define N_BUTTONS 5
+#define MAX_BUTTONMAPEVENTS 20
+
#define GETDPYXTRA(DPY,PDPYINFO)\
(((DPY) == (PDPYINFO)->fromDpy) ?\
&((PDPYINFO)->fromDpyXtra) : &((PDPYINFO)->toDpyXtra))
@@ -161,9 +160,10 @@ typedef struct {
GC textGC;
Atom wmpAtom, wmdwAtom;
Cursor grabCursor;
- XFS *font;
- int twidth, theight;
- int lastFromX;
+ Font fid;
+ int width, height, twidth, theight, tascent;
+ Bool vertical;
+ int lastFromCoord;
int unreasonableDelta;
/* stuff on "to" display */
@@ -180,8 +180,10 @@ typedef struct {
int nScreens;
short **xTables; /* precalculated conversion tables */
short **yTables;
- int fromXConn, fromXDisc; /* location of cursor after conn/disc ops */
- int fromXIncr, fromXDecr; /* location of cursor after incr/decr ops */
+ int fromConnCoord; /* location of cursor after conn/disc ops */
+ int fromDiscCoord;
+ int fromIncrCoord; /* location of cursor after incr/decr ops */
+ int fromDecrCoord;
/* selection forwarding info */
DPYXTRA fromDpyXtra;
@@ -218,6 +220,7 @@ static char *fromDpyName = NULL;
static char *toDpyName = NULL;
static char *defaultFN = "-*-times-bold-r-*-*-*-180-*-*-*-*-*-*";
static char *fontName = "-*-times-bold-r-*-*-*-180-*-*-*-*-*-*";
+static char *label = NULL;
static char *pingStr = "PING"; /* atom for ping request */
static char *geomStr = NULL;
static Bool waitDpy = False;
@@ -232,13 +235,13 @@ static int triggerw = 2;
static Bool doPointerMap = True;
static PSTICKY stickies = NULL;
static Bool doBtnBlock = False;
+static int nButtons = 0;
+static KeySym buttonmap[N_BUTTONS + 1][MAX_BUTTONMAPEVENTS + 1];
/**********
* main
**********/
-main(argc, argv)
-int argc;
-char **argv;
+int main(int argc, char **argv)
{
Display *fromDpy;
PSHADOW pShadow;
@@ -253,7 +256,7 @@ char **argv;
exit(1);
}
- /* no OS independent wat to stop Xlib from complaining via stderr,
+ /* no OS independent way to stop Xlib from complaining via stderr,
but can always pipe stdout/stderr to /dev/null */
/* convert to real name: */
while ((fromDpy = XOpenDisplay(fromDpyName)) == NULL) {
@@ -326,11 +329,18 @@ char **argv;
extern char *lawyerese;
PSTICKY pNewSticky;
KeySym keysym;
+ int button;
+ int eventno;
+ char *keyname, *argptr;
#ifdef DEBUG
printf ("programStr = %s\n", programStr);
#endif
+ /* Clear button map */
+ for (button = 0; button <= N_BUTTONS; button++)
+ buttonmap[button][0] = NoSymbol;
+
for (arg = 1; arg < argc; ++arg) {
if (!strcasecmp(argv[arg], "-from")) {
if (++arg >= argc) Usage();
@@ -353,6 +363,13 @@ char **argv;
#ifdef DEBUG
printf ("fontName = %s\n", fontName);
#endif
+ } else if (!strcasecmp(argv[arg], "-label")) {
+ if (++arg >= argc) Usage();
+ label = argv[arg];
+
+#ifdef DEBUG
+ printf ("label = %s\n", label);
+#endif
} else if (!strcasecmp(argv[arg], "-geometry")) {
if (++arg >= argc) Usage();
geomStr = argv[arg];
@@ -384,6 +401,16 @@ char **argv;
#ifdef DEBUG
printf("will not do pointer mapping\n");
#endif
+ } else if (!strcasecmp(argv[arg], "-north")) {
+ doEdge = EDGE_NORTH;
+#ifdef DEBUG
+ printf("\"from\" is on the north side of \"to\"\n");
+#endif
+ } else if (!strcasecmp(argv[arg], "-south")) {
+ doEdge = EDGE_SOUTH;
+#ifdef DEBUG
+ printf("\"from\" is on the south side of \"to\"\n");
+#endif
} else if (!strcasecmp(argv[arg], "-east")) {
doEdge = EDGE_EAST;
#ifdef DEBUG
@@ -422,6 +449,34 @@ char **argv;
} else {
printf("x2x: warning: can't translate %s\n", argv[arg]);
}
+ } else if (!strcasecmp(argv[arg], "-buttonmap")) {
+ if (++arg >= argc) Usage();
+ button = atoi(argv[arg]);
+
+ if ((button < 1) || (button > N_BUTTONS))
+ printf("x2x: warning: invalid button %d\n", button);
+ else if (++arg >= argc)
+ Usage();
+ else
+ {
+#ifdef DEBUG
+ printf("will map button %d to keysyms '%s'\n", button, argv[arg]);
+#endif
+ argptr = argv[arg];
+ eventno = 0;
+ while ((keyname = strtok(argptr, " \t\n\r")) != NULL)
+ {
+ if ((keysym = XStringToKeysym(keyname)) == NoSymbol)
+ printf("x2x: warning: can't translate %s\n", keyname);
+ else if (eventno + 1 >= MAX_BUTTONMAPEVENTS)
+ printf("x2x: warning: too many keys mapped to button %d\n",
+ button);
+ else
+ buttonmap[button][eventno++] = keysym;
+ argptr = NULL;
+ }
+ buttonmap[button][eventno] = NoSymbol;
+ }
} else if (!strcasecmp(argv[arg], "-resurface")) {
doResurface = True;
#ifdef DEBUG
@@ -458,6 +513,9 @@ static void Usage()
printf(" -big\n");
printf(" -buttonblock\n");
printf(" -nomouse\n");
+ printf(" -nopointermap\n");
+ printf(" -north\n");
+ printf(" -south\n");
printf(" -east\n");
printf(" -west\n");
printf(" -nosel\n");
@@ -465,6 +523,8 @@ static void Usage()
printf(" -resurface\n");
printf(" -shadow <DISPLAY>\n");
printf(" -sticky <sticky key>\n");
+ printf(" -label <LABEL>\n");
+ printf(" -buttonmap <button#> \"<keysym> ...\"\n");
exit(4);
} /* END Usage */
@@ -510,7 +570,7 @@ Display *toDpy;
toConn = XConnectionNumber(toDpy);
while (True) { /* FOREVER */
- if (fromPending = XPending(fromDpy))
+ if ((fromPending = XPending(fromDpy)))
if (ProcessEvent(fromDpy, &dpyInfo)) /* done! */
break;
@@ -534,7 +594,7 @@ static void InitDpyInfo(pDpyInfo)
PDPYINFO pDpyInfo;
{
Display *fromDpy, *toDpy;
- Screen *fromScreen, *toScreen;
+ Screen *fromScreen;
long black, white;
int fromHeight, fromWidth, toHeight, toWidth;
Pixmap nullPixmap;
@@ -544,7 +604,7 @@ PDPYINFO pDpyInfo;
int *heights, *widths;
int counter;
int nScreens, screenNum;
- int twidth, theight; /* text dimensions */
+ int twidth, theight, tascent; /* text dimensions */
int xoff, yoff; /* window offsets */
unsigned int width, height; /* window width, height */
int geomMask; /* mask returned by parse */
@@ -556,9 +616,10 @@ PDPYINFO pDpyInfo;
int eventMask;
GC textGC;
char *windowName;
- XFS *font;
+ Font fid;
PSHADOW pShadow;
int triggerLoc;
+ Bool vertical;
/* cache commonly used variables */
fromDpy = pDpyInfo->fromDpy;
@@ -574,10 +635,12 @@ PDPYINFO pDpyInfo;
/* values also in dpyinfo */
root = pDpyInfo->root = XDefaultRootWindow(fromDpy);
nScreens = pDpyInfo->nScreens = XScreenCount(toDpy);
+ vertical = pDpyInfo->vertical = (doEdge == EDGE_NORTH
+ || doEdge == EDGE_SOUTH);
/* other dpyinfo values */
pDpyInfo->mode = X2X_DISCONNECTED;
- pDpyInfo->unreasonableDelta = fromWidth / 2;
+ pDpyInfo->unreasonableDelta = (vertical ? fromHeight : fromWidth) / 2;
pDpyInfo->pFakeThings = NULL;
/* window init structures */
@@ -586,54 +649,76 @@ PDPYINFO pDpyInfo;
eventMask = KeyPressMask | KeyReleaseMask;
/* cursor locations for moving between screens */
- pDpyInfo->fromXIncr = triggerw;
- pDpyInfo->fromXDecr = fromWidth - triggerw - 1;
+ pDpyInfo->fromIncrCoord = triggerw;
+ pDpyInfo->fromDecrCoord = (vertical ? fromHeight : fromWidth) - triggerw - 1;
if (doEdge) { /* edge triggers x2x */
nullPixmap = XCreatePixmap(fromDpy, root, 1, 1, 1);
eventMask |= EnterWindowMask;
pDpyInfo->grabCursor =
XCreatePixmapCursor(fromDpy, nullPixmap, nullPixmap,
&dummyColor, &dummyColor, 0, 0);
- if (doEdge == EDGE_EAST) {
- /* trigger window location */
+ /* trigger window location */
+ if (doEdge == EDGE_NORTH) {
+ triggerLoc = 0;
+ pDpyInfo->fromConnCoord = fromHeight - triggerw - 1;
+ pDpyInfo->fromDiscCoord = triggerw;
+ } else if (doEdge == EDGE_SOUTH) {
+ triggerLoc = fromHeight - triggerw;
+ pDpyInfo->fromConnCoord = 1;
+ pDpyInfo->fromDiscCoord = triggerLoc - 1;
+ } else if (doEdge == EDGE_EAST) {
triggerLoc = fromWidth - triggerw;
- toHeight = XHeightOfScreen(XScreenOfDisplay(toDpy, 0));
- pDpyInfo->fromXConn = 1;
- pDpyInfo->fromXDisc = fromWidth - triggerw - 1;
- } else {
- /* trigger window location */
+ pDpyInfo->fromConnCoord = 1;
+ pDpyInfo->fromDiscCoord = triggerLoc - 1;
+ } else /* doEdge == EDGE_WEST */ {
triggerLoc = 0;
- toHeight = XHeightOfScreen(XScreenOfDisplay(toDpy, nScreens - 1));
- toWidth = XWidthOfScreen(XScreenOfDisplay(toDpy, nScreens - 1));
- pDpyInfo->fromXConn = fromWidth - triggerw - 1;
- pDpyInfo->fromXDisc = triggerw;
+ pDpyInfo->fromConnCoord = fromWidth - triggerw - 1;
+ pDpyInfo->fromDiscCoord = triggerw;
} /* END if doEdge == ... */
xswa.background_pixel = black;
/* fromWidth - 1 doesn't seem to work for some reason */
+ /* Use triggerw offsets so that if an x2x is running
+ along the left edge and along the north edge, both with
+ -resurface, we don't get a feedback loop of them each
+ fighting to be on top.
+ --09/27/99 Greg J. Badros <gjb@cs.washington.edu> */
+ /* also, make it an InputOnly window so I don't lose
+ screen real estate --09/29/99 gjb */
trigger = pDpyInfo->trigger =
- XCreateWindow(fromDpy, root, triggerLoc, 0, triggerw, fromHeight,
- 0, 0, InputOutput, 0,
- CWBackPixel | CWOverrideRedirect, &xswa);
- font = NULL;
+ XCreateWindow(fromDpy, root,
+ vertical ? triggerw : triggerLoc,
+ vertical ? triggerLoc : triggerw,
+ vertical ? fromWidth - (2*triggerw) : triggerw,
+ vertical ? triggerw : fromHeight - (2*triggerw),
+ 0, 0, InputOnly, 0,
+ CWOverrideRedirect, &xswa);
+ fid = 0;
} else { /* normal window for text: do size grovelling */
pDpyInfo->grabCursor = XCreateFontCursor(fromDpy, XC_exchange);
eventMask |= StructureNotifyMask | ExposureMask;
if (doMouse) eventMask |= ButtonPressMask | ButtonReleaseMask;
+ if (label == NULL)
+ label = toDpyName;
/* determine size of text */
- if (((font = XLoadQueryFont(fromDpy, fontName)) != NULL) ||
- ((font = XLoadQueryFont(fromDpy, defaultFN)) != NULL) ||
- ((font = XLoadQueryFont(fromDpy, "fixed")) != NULL)) {
+ if (((fid = XLoadFont(fromDpy, fontName)) != 0) ||
+ ((fid = XLoadFont(fromDpy, defaultFN)) != 0) ||
+ ((fid = XLoadFont(fromDpy, "fixed")) != 0)) {
/* have a font */
- toDpyFormat[toDpyStringIndex] = (Format)toDpyName;
- formatText(NULL, NULL, NULL, font,
- toDpyFormatLength, toDpyFormat, &twidth, &theight);
+ int ascent, descent, direction;
+ XCharStruct overall;
+ XQueryTextExtents(fromDpy, fid, label, strlen(label),
+ &direction, &ascent, &descent, &overall);
+ twidth = - overall.lbearing + overall.rbearing;
+ theight = ascent + descent;
+ tascent = ascent;
+
textGC = pDpyInfo->textGC = XCreateGC(fromDpy, root, 0, NULL);
XSetState(fromDpy, textGC, black, white, GXcopy, AllPlanes);
- XSetFont(fromDpy, textGC, font->fid);
+ XSetFont(fromDpy, textGC, fid);
} else { /* should not have to execute this clause: */
twidth = theight = 100; /* default window size */
@@ -641,8 +726,8 @@ PDPYINFO pDpyInfo;
/* determine size of window */
xoff = yoff = 0;
- width = twidth;
- height = theight;
+ width = twidth + 4; /* XXX gap around text -- should be configurable */
+ height = theight + 4;
geomMask = XParseGeometry(geomStr, &xoff, &yoff, &width, &height);
switch (gravMask = (geomMask & (XNegative | YNegative))) {
case (XNegative | YNegative): gravity = SouthEastGravity; break;
@@ -711,7 +796,8 @@ PDPYINFO pDpyInfo;
free(windowName);
/* conversion stuff */
- pDpyInfo->toScreen = (doEdge == EDGE_WEST) ? (nScreens - 1) : 0;
+ pDpyInfo->toScreen = (doEdge == EDGE_WEST || doEdge == EDGE_NORTH)
+ ? (nScreens - 1) : 0;
/* construct table lookup for screen coordinate conversion */
pDpyInfo->xTables = (short **)malloc(sizeof(short *) * nScreens);
@@ -739,13 +825,25 @@ PDPYINFO pDpyInfo;
xTable[counter] = (counter * toWidth) / fromWidth;
/* adjustment for boundaries */
- if ((screenNum != 0) || (doEdge == EDGE_EAST))
- xTable[0] = COORD_DECR;
- if (((screenNum + 1) < nScreens) || (doEdge == EDGE_WEST)) {
- xTable[fromWidth - 1] = COORD_INCR;
- /* work-around for bug: on at least one tested screen, cursor
- never moved past fromWidth - 2 */
- xTable[fromWidth - 2] = COORD_INCR;
+ if (vertical) {
+ if ((screenNum != 0) || (doEdge == EDGE_SOUTH))
+ yTable[0] = COORD_DECR;
+ if (((screenNum + 1) < nScreens) || (doEdge == EDGE_NORTH)) {
+ yTable[fromHeight - 1] = COORD_INCR;
+ /* work-around for bug: on at least one tested screen, cursor
+ never moved past fromWidth - 2 (I'll assume this might apply
+ in the vertical case, too. --cpbs) */
+ yTable[fromHeight - 2] = COORD_INCR;
+ }
+ } else {
+ if ((screenNum != 0) || (doEdge == EDGE_EAST))
+ xTable[0] = COORD_DECR;
+ if (((screenNum + 1) < nScreens) || (doEdge == EDGE_WEST)) {
+ xTable[fromWidth - 1] = COORD_INCR;
+ /* work-around for bug: on at least one tested screen, cursor
+ never moved past fromWidth - 2 */
+ xTable[fromWidth - 2] = COORD_INCR;
+ }
}
} /* END for screenNum */
@@ -787,15 +885,18 @@ PDPYINFO pDpyInfo;
pDpyInfo->eventMask = eventMask; /* save for future munging */
if (doSel) XSetSelectionOwner(fromDpy, XA_PRIMARY, trigger, CurrentTime);
XMapRaised(fromDpy, trigger);
- if (pDpyInfo->font = font) { /* paint text */
+ if ((pDpyInfo->fid = fid)) { /* paint text */
/* position text */
pDpyInfo->twidth = twidth;
pDpyInfo->theight = theight;
- toDpyFormat[toDpyLeftIndex] = MAX(0,((width - twidth) / 2));
- toDpyFormat[toDpyTopIndex] = MAX(0,((height - theight) / 2));
+ pDpyInfo->tascent = tascent;
+ pDpyInfo->width = width;
+ pDpyInfo->height = height;
- formatText(fromDpy, trigger, &(textGC), font,
- toDpyFormatLength, toDpyFormat, NULL, NULL);
+ XDrawImageString(fromDpy, trigger, textGC,
+ MAX(0,((width - twidth) / 2)),
+ MAX(0,((height - theight) / 2)) + tascent, label,
+ strlen(label));
} /* END if font */
for (pShadow = shadows; pShadow; pShadow = pShadow->pNext)
@@ -937,33 +1038,36 @@ XMotionEvent *pEv; /* caution: might be pseudo-event!!
int toScreenNum;
PSHADOW pShadow;
- int toX, fromX, delta;
+ int toCoord, fromCoord, delta;
Display *fromDpy;
Bool bAbortedDisconnect;
+ Bool vert;
+ vert = pDpyInfo->vertical;
+
/* find the screen */
toScreenNum = pDpyInfo->toScreen;
- fromX = pEv->x_root;
+ fromCoord = vert ? pEv->y_root : pEv->x_root;
/* check to make sure the cursor is still on the from screen */
if (!(pEv->same_screen)) {
- toX = (pDpyInfo->lastFromX < fromX) ? COORD_DECR : COORD_INCR;
+ toCoord = (pDpyInfo->lastFromCoord < fromCoord) ? COORD_DECR : COORD_INCR;
} else {
- toX = pDpyInfo->xTables[toScreenNum][fromX];
+ toCoord = (vert?pDpyInfo->yTables:pDpyInfo->xTables)[toScreenNum][fromCoord];
/* sanity check motion: necessary for nondeterminism surrounding warps */
- delta = pDpyInfo->lastFromX - fromX;
+ delta = pDpyInfo->lastFromCoord - fromCoord;
if (delta < 0) delta = -delta;
if (delta > pDpyInfo->unreasonableDelta) return False;
}
- if (SPECIAL_COORD(toX) != 0) { /* special coordinate */
+ if (SPECIAL_COORD(toCoord) != 0) { /* special coordinate */
bAbortedDisconnect = False;
- if (toX == COORD_INCR) {
+ if (toCoord == COORD_INCR) {
if (toScreenNum != (pDpyInfo->nScreens - 1)) { /* next screen */
toScreenNum = ++(pDpyInfo->toScreen);
- fromX = pDpyInfo->fromXIncr;
- toX = pDpyInfo->xTables[toScreenNum][fromX];
+ fromCoord = pDpyInfo->fromIncrCoord;
+ toCoord = (vert?pDpyInfo->yTables:pDpyInfo->xTables)[toScreenNum][fromCoord];
} else { /* disconnect! */
if (doBtnBlock &&
(pEv->state & (Button1Mask | Button2Mask | Button3Mask |
@@ -971,15 +1075,15 @@ XMotionEvent *pEv; /* caution: might be pseudo-event!!
bAbortedDisconnect = True;
else {
DoDisconnect(pDpyInfo);
- fromX = pDpyInfo->fromXDisc;
+ fromCoord = pDpyInfo->fromDiscCoord;
}
- toX = pDpyInfo->xTables[toScreenNum][pDpyInfo->fromXConn];
+ toCoord = (vert?pDpyInfo->yTables:pDpyInfo->xTables)[toScreenNum][pDpyInfo->fromConnCoord];
}
} else { /* DECR */
if (toScreenNum != 0) { /* previous screen */
toScreenNum = --(pDpyInfo->toScreen);
- fromX = pDpyInfo->fromXDecr;
- toX = pDpyInfo->xTables[toScreenNum][fromX];
+ fromCoord = pDpyInfo->fromDecrCoord;
+ toCoord = (vert?pDpyInfo->yTables:pDpyInfo->xTables)[toScreenNum][fromCoord];
} else { /* disconnect! */
if (doBtnBlock &&
(pEv->state & (Button1Mask | Button2Mask | Button3Mask |
@@ -987,23 +1091,26 @@ XMotionEvent *pEv; /* caution: might be pseudo-event!!
bAbortedDisconnect = True;
else {
DoDisconnect(pDpyInfo);
- fromX = pDpyInfo->fromXDisc;
+ fromCoord = pDpyInfo->fromDiscCoord;
}
- toX = pDpyInfo->xTables[toScreenNum][pDpyInfo->fromXConn];
+ toCoord = (vert?pDpyInfo->yTables:pDpyInfo->xTables)[toScreenNum][pDpyInfo->fromConnCoord];
}
- } /* END if toX */
+ } /* END if toCoord */
if (!bAbortedDisconnect) {
fromDpy = pDpyInfo->fromDpy;
XWarpPointer(fromDpy, None, pDpyInfo->root, 0, 0, 0, 0,
- fromX, pEv->y_root);
+ vert ? pEv->x_root : fromCoord,
+ vert ? fromCoord : pEv->y_root);
XFlush(fromDpy);
}
} /* END if SPECIAL_COORD */
- pDpyInfo->lastFromX = fromX;
+ pDpyInfo->lastFromCoord = fromCoord;
for (pShadow = shadows; pShadow; pShadow = pShadow->pNext) {
- XTestFakeMotionEvent(pShadow->dpy, toScreenNum, toX,
- pDpyInfo->yTables[toScreenNum][pEv->y_root], 0);
+ XTestFakeMotionEvent(pShadow->dpy, toScreenNum,
+ vert?pDpyInfo->xTables[toScreenNum][pEv->x_root]:toCoord,
+ vert?toCoord:pDpyInfo->yTables[toScreenNum][pEv->y_root],
+ 0);
XFlush(pShadow->dpy);
} /* END for */
@@ -1017,10 +1124,12 @@ PDPYINFO pDpyInfo;
XExposeEvent *pEv;
{
XClearWindow(pDpyInfo->fromDpy, pDpyInfo->trigger);
- if (pDpyInfo->font)
- formatText(pDpyInfo->fromDpy, pDpyInfo->trigger,
- &(pDpyInfo->textGC), pDpyInfo->font,
- toDpyFormatLength, toDpyFormat, NULL, NULL);
+ if (pDpyInfo->fid)
+ XDrawImageString(pDpyInfo->fromDpy, pDpyInfo->trigger, pDpyInfo->textGC,
+ MAX(0,((pDpyInfo->width - pDpyInfo->twidth) / 2)),
+ MAX(0,((pDpyInfo->height - pDpyInfo->theight) / 2)) +
+ pDpyInfo->tascent, label, strlen(label));
+
return False;
} /* END ProcessExpose */
@@ -1036,10 +1145,17 @@ XCrossingEvent *pEv;
if ((pEv->mode == NotifyNormal) &&
(pDpyInfo->mode == X2X_DISCONNECTED) && (dpy == pDpyInfo->fromDpy)) {
DoConnect(pDpyInfo);
- XWarpPointer(fromDpy, None, pDpyInfo->root, 0, 0, 0, 0,
- pDpyInfo->fromXConn, pEv->y_root);
- xmev.x_root = pDpyInfo->lastFromX = pDpyInfo->fromXConn;
- xmev.y_root = pEv->y_root;
+ if (pDpyInfo->vertical) {
+ XWarpPointer(fromDpy, None, pDpyInfo->root, 0, 0, 0, 0,
+ pEv->x_root, pDpyInfo->fromConnCoord);
+ xmev.x_root = pEv->x_root;
+ xmev.y_root = pDpyInfo->lastFromCoord = pDpyInfo->fromConnCoord;
+ } else {
+ XWarpPointer(fromDpy, None, pDpyInfo->root, 0, 0, 0, 0,
+ pDpyInfo->fromConnCoord, pEv->y_root);
+ xmev.x_root = pDpyInfo->lastFromCoord = pDpyInfo->fromConnCoord;
+ xmev.y_root = pEv->y_root;
+ }
xmev.same_screen = True;
ProcessMotionNotify(NULL, pDpyInfo, &xmev);
} /* END if NotifyNormal... */
@@ -1055,7 +1171,10 @@ XButtonEvent *pEv;
int state;
PSHADOW pShadow;
unsigned int toButton;
-
+ KeySym keysym;
+ KeyCode keycode;
+ int eventno;
+
switch (pDpyInfo->mode) {
case X2X_DISCONNECTED:
pDpyInfo->mode = X2X_AWAIT_RELEASE;
@@ -1064,17 +1183,48 @@ XButtonEvent *pEv;
#endif
break;
case X2X_CONNECTED:
- if (pEv->button <= N_BUTTONS) toButton = pDpyInfo->inverseMap[pEv->button];
- for (pShadow = shadows; pShadow; pShadow = pShadow->pNext) {
- XTestFakeButtonEvent(pShadow->dpy, toButton, True, 0);
+ if ((pEv->button <= N_BUTTONS) &&
+ (buttonmap[pEv->button][0] != NoSymbol))
+ {
+ for (pShadow = shadows; pShadow; pShadow = pShadow->pNext)
+ {
#ifdef DEBUG
- printf("from button %d down, to button %d down\n", pEv->button,toButton);
+ printf("Button %d is mapped, sending keys: ", pEv->button);
#endif
- XFlush(pShadow->dpy);
- } /* END for */
- if (doAutoUp)
- FakeAction(pDpyInfo, FAKE_BUTTON, toButton, True);
- if (doEdge) break;
+ for (eventno = 0;
+ (keysym = buttonmap[pEv->button][eventno]) != NoSymbol;
+ eventno++)
+ {
+ if ((keycode = XKeysymToKeycode(pShadow->dpy, keysym))) {
+ XTestFakeKeyEvent(pShadow->dpy, keycode, True, 0);
+ XTestFakeKeyEvent(pShadow->dpy, keycode, False, 0);
+ XFlush(pShadow->dpy);
+#ifdef DEBUG
+ printf(" (0x%04X)", keycode);
+#endif
+ }
+#ifdef DEBUG
+ else
+ printf(" (no code)");
+#endif
+ }
+#ifdef DEBUG
+ printf("\n");
+#endif
+ }
+ } else if (pEv->button <= nButtons) {
+ toButton = pDpyInfo->inverseMap[pEv->button];
+ for (pShadow = shadows; pShadow; pShadow = pShadow->pNext) {
+ XTestFakeButtonEvent(pShadow->dpy, toButton, True, 0);
+#ifdef DEBUG
+ printf("from button %d down, to button %d down\n", pEv->button,toButton);
+#endif
+ XFlush(pShadow->dpy);
+ } /* END for */
+ if (doAutoUp)
+ FakeAction(pDpyInfo, FAKE_BUTTON, toButton, True);
+ }
+ if (doEdge) break;
/* check if more than one button pressed */
state = pEv->state;
@@ -1113,16 +1263,21 @@ XButtonEvent *pEv;
if ((pDpyInfo->mode == X2X_CONNECTED) ||
(pDpyInfo->mode == X2X_CONN_RELEASE)) {
- if (pEv->button <= N_BUTTONS) toButton = pDpyInfo->inverseMap[pEv->button];
- for (pShadow = shadows; pShadow; pShadow = pShadow->pNext) {
- XTestFakeButtonEvent(pShadow->dpy, toButton, False, 0);
+ if ((pEv->button <= nButtons) &&
+ (buttonmap[pEv->button][0] == NoSymbol))
+ // Do not process button release if it was mapped to keys
+ {
+ toButton = pDpyInfo->inverseMap[pEv->button];
+ for (pShadow = shadows; pShadow; pShadow = pShadow->pNext) {
+ XTestFakeButtonEvent(pShadow->dpy, toButton, False, 0);
#ifdef DEBUG
- printf("from button %d up, to button %d up\n", pEv->button, toButton);
+ printf("from button %d up, to button %d up\n", pEv->button, toButton);
#endif
- XFlush(pShadow->dpy);
- } /* END for */
- if (doAutoUp)
- FakeAction(pDpyInfo, FAKE_BUTTON, toButton, False);
+ XFlush(pShadow->dpy);
+ } /* END for */
+ if (doAutoUp)
+ FakeAction(pDpyInfo, FAKE_BUTTON, toButton, False);
+ }
} /* END if */
if (doEdge) return False;
@@ -1145,8 +1300,13 @@ XButtonEvent *pEv;
if (!state) { /* all buttons up: time to (dis)connect */
if (pDpyInfo->mode == X2X_AWAIT_RELEASE) { /* connect */
DoConnect(pDpyInfo);
- xmev.x_root = pDpyInfo->lastFromX = pEv->x_root;
- xmev.y_root = pEv->y_root;
+ if (pDpyInfo->vertical) {
+ xmev.x_root = pEv->x_root;
+ xmev.y_root = pDpyInfo->lastFromCoord = pEv->y_root;
+ } else {
+ xmev.x_root = pDpyInfo->lastFromCoord = pEv->x_root;
+ xmev.y_root = pEv->y_root;
+ }
xmev.same_screen = True;
ProcessMotionNotify(NULL, pDpyInfo, &xmev);
} else { /* disconnect */
@@ -1178,7 +1338,7 @@ XKeyEvent *pEv;
if (pSticky) {
for (pShadow = shadows; pShadow; pShadow = pShadow->pNext) {
- if (keycode = XKeysymToKeycode(pShadow->dpy, keysym)) {
+ if ((keycode = XKeysymToKeycode(pShadow->dpy, keysym))) {
XTestFakeKeyEvent(pShadow->dpy, keycode, True, 0);
XTestFakeKeyEvent(pShadow->dpy, keycode, False, 0);
XFlush(pShadow->dpy);
@@ -1186,7 +1346,7 @@ XKeyEvent *pEv;
} /* END for */
} else {
for (pShadow = shadows; pShadow; pShadow = pShadow->pNext) {
- if (keycode = XKeysymToKeycode(pShadow->dpy, keysym)) {
+ if ((keycode = XKeysymToKeycode(pShadow->dpy, keysym))) {
XTestFakeKeyEvent(pShadow->dpy, keycode, bPress, 0);
XFlush(pShadow->dpy);
} /* END if */
@@ -1204,12 +1364,10 @@ Display *dpy;
PDPYINFO pDpyInfo;
XConfigureEvent *pEv;
{
- if (pDpyInfo->font) {
+ if (pDpyInfo->fid) {
/* reposition text */
- toDpyFormat[toDpyLeftIndex] =
- MAX(0,((pEv->width - pDpyInfo->twidth) / 2));
- toDpyFormat[toDpyTopIndex] =
- MAX(0,((pEv->height - pDpyInfo->theight) / 2));
+ pDpyInfo->width = pEv->width;
+ pDpyInfo->height = pEv->height;
} /* END if font */
return False;
@@ -1276,7 +1434,6 @@ Display *dpy;
PDPYINFO pDpyInfo;
XPropertyEvent *pEv;
{
- XSelectionRequestEvent *pSelReq;
PDPYXTRA pDpyXtra = GETDPYXTRA(dpy, pDpyInfo);
#ifdef DEBUG
@@ -1501,7 +1658,7 @@ PDPYINFO pDpyInfo;
/* send up to all shadows */
for (pShadow = shadows; pShadow; pShadow = pShadow->pNext) {
if (type == FAKE_KEY) { /* key goes up */
- if (keycode = XKeysymToKeycode(pShadow->dpy, pFake->thing)) {
+ if ((keycode = XKeysymToKeycode(pShadow->dpy, pFake->thing))) {
XTestFakeKeyEvent(pShadow->dpy, keycode, False, 0);
#ifdef DEBUG
printf("key 0x%x up\n", pFake->thing);
@@ -1535,7 +1692,6 @@ PDPYINFO pDpyInfo;
{
unsigned int buttCtr;
unsigned char buttonMap[N_BUTTONS];
- int nButtons;
if (dpy == pDpyInfo->toDpy) { /* only care about toDpy */
/* straightforward mapping */