factored common X windows screen stuff into a common base class.

This commit is contained in:
crs 2001-11-11 21:15:30 +00:00
parent 05928f28f8
commit cf4e1fd9ca
7 changed files with 502 additions and 401 deletions

View File

@ -1,8 +1,6 @@
#include "CXWindowsPrimaryScreen.h" #include "CXWindowsPrimaryScreen.h"
#include "CServer.h" #include "CServer.h"
#include "CThread.h" #include "CThread.h"
#include "CLock.h"
#include "TMethodJob.h"
#include "CLog.h" #include "CLog.h"
#include <assert.h> #include <assert.h>
#include <X11/X.h> #include <X11/X.h>
@ -13,17 +11,15 @@
CXWindowsPrimaryScreen::CXWindowsPrimaryScreen() : CXWindowsPrimaryScreen::CXWindowsPrimaryScreen() :
m_server(NULL), m_server(NULL),
m_display(NULL), m_active(false),
m_w(0), m_h(0), m_window(None)
m_window(None),
m_active(false)
{ {
// do nothing // do nothing
} }
CXWindowsPrimaryScreen::~CXWindowsPrimaryScreen() CXWindowsPrimaryScreen::~CXWindowsPrimaryScreen()
{ {
assert(m_display == NULL); assert(m_window == None);
} }
void CXWindowsPrimaryScreen::open(CServer* server) void CXWindowsPrimaryScreen::open(CServer* server)
@ -35,90 +31,37 @@ void CXWindowsPrimaryScreen::open(CServer* server)
m_server = server; m_server = server;
// open the display // open the display
log((CLOG_DEBUG "XOpenDisplay(%s)", "NULL")); openDisplay();
m_display = XOpenDisplay(NULL); // FIXME -- allow non-default
if (m_display == NULL)
throw int(5); // FIXME -- make exception for this
// get default screen
m_screen = DefaultScreen(m_display);
Screen* screen = ScreenOfDisplay(m_display, m_screen);
// get screen size
m_w = WidthOfScreen(screen);
m_h = HeightOfScreen(screen);
log((CLOG_INFO "primary display size: %dx%d", m_w, m_h));
// get the root window
m_root = RootWindow(m_display, m_screen);
// create the grab window. this window is used to capture user
// input when the user is focussed on another client. don't let
// the window manager mess with it.
XSetWindowAttributes attr;
attr.event_mask = PointerMotionMask |// PointerMotionHintMask |
ButtonPressMask | ButtonReleaseMask |
KeyPressMask | KeyReleaseMask |
KeymapStateMask;
attr.do_not_propagate_mask = 0;
attr.override_redirect = True;
attr.cursor = createBlankCursor();
m_window = XCreateWindow(m_display, m_root, 0, 0, m_w, m_h, 0, 0,
InputOnly, CopyFromParent,
CWDontPropagate | CWEventMask |
CWOverrideRedirect | CWCursor,
&attr);
// start watching for events on other windows
selectEvents(m_root);
// start processing events
m_eventThread = new CThread(new TMethodJob<CXWindowsPrimaryScreen>(
this, &CXWindowsPrimaryScreen::eventThread));
} }
void CXWindowsPrimaryScreen::close() void CXWindowsPrimaryScreen::close()
{ {
assert(m_server != NULL); assert(m_server != NULL);
assert(m_window != None);
assert(m_eventThread != NULL);
// stop event thread
log((CLOG_DEBUG "stopping event thread"));
m_eventThread->cancel();
m_eventThread->wait();
delete m_eventThread;
m_eventThread = NULL;
log((CLOG_DEBUG "stopped event thread"));
// destroy window
XDestroyWindow(m_display, m_window);
m_window = None;
// close the display // close the display
XCloseDisplay(m_display); closeDisplay();
m_display = NULL;
log((CLOG_DEBUG "closed display")); // done with server
m_server = NULL;
} }
void CXWindowsPrimaryScreen::enter(SInt32 x, SInt32 y) void CXWindowsPrimaryScreen::enter(SInt32 x, SInt32 y)
{ {
log((CLOG_INFO "entering primary at %d,%d", x, y)); log((CLOG_INFO "entering primary at %d,%d", x, y));
assert(m_display != NULL); assert(m_active == true);
assert(m_window != None); assert(m_window != None);
assert(m_active == true);
CLock lock(&m_mutex); CDisplayLock display(this);
// warp to requested location // warp to requested location
XWarpPointer(m_display, None, m_window, 0, 0, 0, 0, x, y); XWarpPointer(display, None, m_window, 0, 0, 0, 0, x, y);
// unmap the grab window. this also ungrabs the mouse and keyboard. // unmap the grab window. this also ungrabs the mouse and keyboard.
XUnmapWindow(m_display, m_window); XUnmapWindow(display, m_window);
// remove all input events for grab window // remove all input events for grab window
XEvent event; XEvent event;
while (XCheckWindowEvent(m_display, m_window, while (XCheckWindowEvent(display, m_window,
PointerMotionMask | PointerMotionMask |
ButtonPressMask | ButtonReleaseMask | ButtonPressMask | ButtonReleaseMask |
KeyPressMask | KeyReleaseMask | KeyPressMask | KeyReleaseMask |
@ -134,14 +77,13 @@ void CXWindowsPrimaryScreen::enter(SInt32 x, SInt32 y)
void CXWindowsPrimaryScreen::leave() void CXWindowsPrimaryScreen::leave()
{ {
log((CLOG_INFO "leaving primary")); log((CLOG_INFO "leaving primary"));
assert(m_display != NULL); assert(m_active == false);
assert(m_window != None); assert(m_window != None);
assert(m_active == false);
CLock lock(&m_mutex); CDisplayLock display(this);
// raise and show the input window // raise and show the input window
XMapRaised(m_display, m_window); XMapRaised(display, m_window);
// grab the mouse and keyboard. keep trying until we get them. // grab the mouse and keyboard. keep trying until we get them.
// if we can't grab one after grabbing the other then ungrab // if we can't grab one after grabbing the other then ungrab
@ -150,7 +92,7 @@ void CXWindowsPrimaryScreen::leave()
do { do {
// mouse first // mouse first
do { do {
result = XGrabPointer(m_display, m_window, True, 0, result = XGrabPointer(display, m_window, True, 0,
GrabModeAsync, GrabModeAsync, GrabModeAsync, GrabModeAsync,
m_window, None, CurrentTime); m_window, None, CurrentTime);
assert(result != GrabNotViewable); assert(result != GrabNotViewable);
@ -162,11 +104,12 @@ void CXWindowsPrimaryScreen::leave()
log((CLOG_DEBUG "grabbed pointer")); log((CLOG_DEBUG "grabbed pointer"));
// now the keyboard // now the keyboard
result = XGrabKeyboard(m_display, m_window, True, result = XGrabKeyboard(display, m_window, True,
GrabModeAsync, GrabModeAsync, CurrentTime); GrabModeAsync, GrabModeAsync, CurrentTime);
assert(result != GrabNotViewable); assert(result != GrabNotViewable);
if (result != GrabSuccess) { if (result != GrabSuccess) {
XUngrabPointer(m_display, CurrentTime); // back off to avoid grab deadlock
XUngrabPointer(display, CurrentTime);
log((CLOG_DEBUG "ungrabbed pointer, waiting to grab keyboard")); log((CLOG_DEBUG "ungrabbed pointer, waiting to grab keyboard"));
CThread::sleep(0.25); CThread::sleep(0.25);
} }
@ -174,7 +117,9 @@ void CXWindowsPrimaryScreen::leave()
log((CLOG_DEBUG "grabbed keyboard")); log((CLOG_DEBUG "grabbed keyboard"));
// move the mouse to the center of grab window // move the mouse to the center of grab window
warpCursorNoLock(m_w >> 1, m_h >> 1); SInt32 w, h;
getScreenSize(&w, &h);
warpCursorNoLock(display, w >> 1, h >> 1);
// local client now active // local client now active
m_active = true; m_active = true;
@ -182,21 +127,24 @@ void CXWindowsPrimaryScreen::leave()
void CXWindowsPrimaryScreen::warpCursor(SInt32 x, SInt32 y) void CXWindowsPrimaryScreen::warpCursor(SInt32 x, SInt32 y)
{ {
CLock lock(&m_mutex); CDisplayLock display(this);
warpCursorNoLock(x, y); warpCursorNoLock(display, x, y);
} }
void CXWindowsPrimaryScreen::warpCursorNoLock( void CXWindowsPrimaryScreen::warpCursorNoLock(
SInt32 x, SInt32 y) Display* display, SInt32 x, SInt32 y)
{ {
assert(display != NULL);
assert(m_window != None);
// warp the mouse // warp the mouse
XWarpPointer(m_display, None, m_root, 0, 0, 0, 0, x, y); XWarpPointer(display, None, getRoot(), 0, 0, 0, 0, x, y);
XSync(m_display, False); XSync(display, False);
log((CLOG_DEBUG "warped to %d,%d", x, y)); log((CLOG_DEBUG "warped to %d,%d", x, y));
// discard mouse events since we just added one we don't want // discard mouse events since we just added one we don't want
XEvent xevent; XEvent xevent;
while (XCheckWindowEvent(m_display, m_window, while (XCheckWindowEvent(display, m_window,
PointerMotionMask, &xevent)) { PointerMotionMask, &xevent)) {
// do nothing // do nothing
} }
@ -205,21 +153,57 @@ void CXWindowsPrimaryScreen::warpCursorNoLock(
void CXWindowsPrimaryScreen::getSize( void CXWindowsPrimaryScreen::getSize(
SInt32* width, SInt32* height) const SInt32* width, SInt32* height) const
{ {
assert(m_display != NULL); getScreenSize(width, height);
assert(width != NULL && height != NULL);
*width = m_w;
*height = m_h;
} }
SInt32 CXWindowsPrimaryScreen::getJumpZoneSize() const SInt32 CXWindowsPrimaryScreen::getJumpZoneSize() const
{ {
assert(m_display != NULL);
return 1; return 1;
} }
void CXWindowsPrimaryScreen::selectEvents(Window w) const void CXWindowsPrimaryScreen::onOpenDisplay()
{
assert(m_window == None);
CDisplayLock display(this);
// get size of screen
SInt32 w, h;
getScreenSize(&w, &h);
// create the grab window. this window is used to capture user
// input when the user is focussed on another client. don't let
// the window manager mess with it.
XSetWindowAttributes attr;
attr.event_mask = PointerMotionMask |// PointerMotionHintMask |
ButtonPressMask | ButtonReleaseMask |
KeyPressMask | KeyReleaseMask |
KeymapStateMask;
attr.do_not_propagate_mask = 0;
attr.override_redirect = True;
attr.cursor = createBlankCursor();
m_window = XCreateWindow(display, getRoot(), 0, 0, w, h, 0, 0,
InputOnly, CopyFromParent,
CWDontPropagate | CWEventMask |
CWOverrideRedirect | CWCursor,
&attr);
// start watching for events on other windows
selectEvents(display, getRoot());
}
void CXWindowsPrimaryScreen::onCloseDisplay()
{
assert(m_window != None);
// destroy window
CDisplayLock display(this);
XDestroyWindow(display, m_window);
m_window = None;
}
void CXWindowsPrimaryScreen::selectEvents(
Display* display, Window w) const
{ {
// we want to track the mouse everywhere on the display. to achieve // we want to track the mouse everywhere on the display. to achieve
// that we select PointerMotionMask on every window. we also select // that we select PointerMotionMask on every window. we also select
@ -231,73 +215,31 @@ void CXWindowsPrimaryScreen::selectEvents(Window w) const
return; return;
// select events of interest // select events of interest
XSelectInput(m_display, w, PointerMotionMask | SubstructureNotifyMask); XSelectInput(display, w, PointerMotionMask | SubstructureNotifyMask);
// recurse on child windows // recurse on child windows
Window rw, pw, *cw; Window rw, pw, *cw;
unsigned int nc; unsigned int nc;
if (XQueryTree(m_display, w, &rw, &pw, &cw, &nc)) { if (XQueryTree(display, w, &rw, &pw, &cw, &nc)) {
for (unsigned int i = 0; i < nc; ++i) for (unsigned int i = 0; i < nc; ++i)
selectEvents(cw[i]); selectEvents(display, cw[i]);
XFree(cw); XFree(cw);
} }
} }
Cursor CXWindowsPrimaryScreen::createBlankCursor()
{
// this seems just a bit more complicated than really necessary
// get the closet cursor size to 1x1
unsigned int w, h;
XQueryBestCursor(m_display, m_root, 1, 1, &w, &h);
// make bitmap data for cursor of closet size. since the cursor
// is blank we can use the same bitmap for shape and mask: all
// zeros.
const int size = ((w + 7) >> 3) * h;
char* data = new char[size];
memset(data, 0, size);
// make bitmap
Pixmap bitmap = XCreateBitmapFromData(m_display, m_root, data, w, h);
// need an arbitrary color for the cursor
XColor color;
color.pixel = 0;
color.red = color.green = color.blue = 0;
color.flags = DoRed | DoGreen | DoBlue;
// make cursor from bitmap
Cursor cursor = XCreatePixmapCursor(m_display, bitmap, bitmap,
&color, &color, 0, 0);
// don't need bitmap or the data anymore
delete[] data;
XFreePixmap(m_display, bitmap);
return cursor;
}
void CXWindowsPrimaryScreen::eventThread(void*) void CXWindowsPrimaryScreen::eventThread(void*)
{ {
for (;;) { for (;;) {
// wait for and then get the next event // wait for and get the next event
m_mutex.lock();
while (XPending(m_display) == 0) {
m_mutex.unlock();
CThread::sleep(0.05);
m_mutex.lock();
}
XEvent xevent; XEvent xevent;
XNextEvent(m_display, &xevent); getEvent(&xevent);
m_mutex.unlock();
// handle event // handle event
switch (xevent.type) { switch (xevent.type) {
case CreateNotify: { case CreateNotify: {
// select events on new window // select events on new window
CLock lock(&m_mutex); CDisplayLock display(this);
selectEvents(xevent.xcreatewindow.window); selectEvents(display, xevent.xcreatewindow.window);
break; break;
} }
@ -355,19 +297,22 @@ void CXWindowsPrimaryScreen::eventThread(void*)
// get mouse deltas // get mouse deltas
{ {
CLock lock(&m_mutex); CDisplayLock display(this);
Window root, window; Window root, window;
int xRoot, yRoot, xWindow, yWindow; int xRoot, yRoot, xWindow, yWindow;
unsigned int mask; unsigned int mask;
if (!XQueryPointer(m_display, m_window, &root, &window, if (!XQueryPointer(display, m_window, &root, &window,
&xRoot, &yRoot, &xWindow, &yWindow, &mask)) &xRoot, &yRoot, &xWindow, &yWindow, &mask))
break; break;
x = xRoot - (m_w >> 1); // compute position of center of window
y = yRoot - (m_h >> 1); SInt32 w, h;
getScreenSize(&w, &h);
x = xRoot - (w >> 1);
y = yRoot - (h >> 1);
// warp mouse back to center // warp mouse back to center
warpCursorNoLock(m_w >> 1, m_h >> 1); warpCursorNoLock(display, w >> 1, h >> 1);
} }
m_server->onMouseMoveSecondary(x, y); m_server->onMouseMoveSecondary(x, y);
@ -422,7 +367,8 @@ KeyID CXWindowsPrimaryScreen::mapKey(
index = 1; index = 1;
else else
index = 0; index = 0;
return static_cast<KeyID>(XKeycodeToKeysym(m_display, keycode, index)); CDisplayLock display(this);
return static_cast<KeyID>(XKeycodeToKeysym(display, keycode, index));
} }
ButtonID CXWindowsPrimaryScreen::mapButton( ButtonID CXWindowsPrimaryScreen::mapButton(

View File

@ -1,15 +1,12 @@
#ifndef CXWINDOWSPRIMARYSCREEN_H #ifndef CXWINDOWSPRIMARYSCREEN_H
#define CXWINDOWSPRIMARYSCREEN_H #define CXWINDOWSPRIMARYSCREEN_H
#include "CMutex.h"
#include "KeyTypes.h" #include "KeyTypes.h"
#include "MouseTypes.h" #include "MouseTypes.h"
#include "CXWindowsScreen.h"
#include "IPrimaryScreen.h" #include "IPrimaryScreen.h"
#include <X11/Xlib.h>
class CThread; class CXWindowsPrimaryScreen : public CXWindowsScreen, public IPrimaryScreen {
class CXWindowsPrimaryScreen : public IPrimaryScreen {
public: public:
CXWindowsPrimaryScreen(); CXWindowsPrimaryScreen();
virtual ~CXWindowsPrimaryScreen(); virtual ~CXWindowsPrimaryScreen();
@ -23,28 +20,25 @@ class CXWindowsPrimaryScreen : public IPrimaryScreen {
virtual void getSize(SInt32* width, SInt32* height) const; virtual void getSize(SInt32* width, SInt32* height) const;
virtual SInt32 getJumpZoneSize() const; virtual SInt32 getJumpZoneSize() const;
private: protected:
void selectEvents(Window) const; // CXWindowsScreen overrides
Cursor createBlankCursor(); virtual void onOpenDisplay();
void warpCursorNoLock(SInt32 xAbsolute, SInt32 yAbsolute); virtual void onCloseDisplay();
virtual void eventThread(void*);
private:
void selectEvents(Display*, Window) const;
void warpCursorNoLock(Display*,
SInt32 xAbsolute, SInt32 yAbsolute);
void eventThread(void*);
KeyModifierMask mapModifier(unsigned int state) const; KeyModifierMask mapModifier(unsigned int state) const;
KeyID mapKey(KeyCode, KeyModifierMask) const; KeyID mapKey(KeyCode, KeyModifierMask) const;
ButtonID mapButton(unsigned int button) const; ButtonID mapButton(unsigned int button) const;
private: private:
CServer* m_server; CServer* m_server;
CThread* m_eventThread;
Display* m_display;
int m_screen;
Window m_root;
SInt32 m_w, m_h;
Window m_window;
bool m_active; bool m_active;
Window m_window;
// X is not thread safe
CMutex m_mutex;
}; };
#endif #endif

172
synergy/CXWindowsScreen.cpp Normal file
View File

@ -0,0 +1,172 @@
#include "CXWindowsScreen.h"
#include "CThread.h"
#include "CLock.h"
#include "TMethodJob.h"
#include "CLog.h"
#include <string.h>
#include <assert.h>
#include <X11/X.h>
//
// CXWindowsScreen
//
CXWindowsScreen::CXWindowsScreen() :
m_display(NULL),
m_root(None),
m_w(0), m_h(0)
{
// do nothing
}
CXWindowsScreen::~CXWindowsScreen()
{
assert(m_display == NULL);
}
void CXWindowsScreen::openDisplay()
{
assert(m_display == NULL);
// open the display
log((CLOG_DEBUG "XOpenDisplay(%s)", "NULL"));
m_display = XOpenDisplay(NULL); // FIXME -- allow non-default
if (m_display == NULL)
throw int(5); // FIXME -- make exception for this
// get default screen
m_screen = DefaultScreen(m_display);
Screen* screen = ScreenOfDisplay(m_display, m_screen);
// get screen size
m_w = WidthOfScreen(screen);
m_h = HeightOfScreen(screen);
log((CLOG_INFO "display size: %dx%d", m_w, m_h));
// get the root window
m_root = RootWindow(m_display, m_screen);
// let subclass prep display
onOpenDisplay();
// start processing events
m_eventThread = new CThread(new TMethodJob<CXWindowsScreen>(
this, &CXWindowsScreen::eventThread));
}
void CXWindowsScreen::closeDisplay()
{
assert(m_display != NULL);
assert(m_eventThread != NULL);
// stop event thread
log((CLOG_DEBUG "stopping event thread"));
m_eventThread->cancel();
m_eventThread->wait();
delete m_eventThread;
m_eventThread = NULL;
log((CLOG_DEBUG "stopped event thread"));
// let subclass close down display
onCloseDisplay();
// close the display
XCloseDisplay(m_display);
m_display = NULL;
log((CLOG_DEBUG "closed display"));
}
int CXWindowsScreen::getScreen() const
{
assert(m_display != NULL);
return m_screen;
}
Window CXWindowsScreen::getRoot() const
{
assert(m_display != NULL);
return m_root;
}
void CXWindowsScreen::getScreenSize(
SInt32* w, SInt32* h) const
{
assert(m_display != NULL);
assert(w != NULL && h != NULL);
*w = m_w;
*h = m_h;
}
Cursor CXWindowsScreen::createBlankCursor() const
{
// this seems just a bit more complicated than really necessary
// get the closet cursor size to 1x1
unsigned int w, h;
XQueryBestCursor(m_display, m_root, 1, 1, &w, &h);
// make bitmap data for cursor of closet size. since the cursor
// is blank we can use the same bitmap for shape and mask: all
// zeros.
const int size = ((w + 7) >> 3) * h;
char* data = new char[size];
memset(data, 0, size);
// make bitmap
Pixmap bitmap = XCreateBitmapFromData(m_display, m_root, data, w, h);
// need an arbitrary color for the cursor
XColor color;
color.pixel = 0;
color.red = color.green = color.blue = 0;
color.flags = DoRed | DoGreen | DoBlue;
// make cursor from bitmap
Cursor cursor = XCreatePixmapCursor(m_display, bitmap, bitmap,
&color, &color, 0, 0);
// don't need bitmap or the data anymore
delete[] data;
XFreePixmap(m_display, bitmap);
return cursor;
}
void CXWindowsScreen::getEvent(XEvent* xevent) const
{
// wait for an event in a cancellable way and don't lock the
// display while we're waiting.
m_mutex.lock();
while (XPending(m_display) == 0) {
m_mutex.unlock();
CThread::sleep(0.05);
m_mutex.lock();
}
XNextEvent(m_display, xevent);
m_mutex.unlock();
}
//
// CXWindowsScreen::CDisplayLock
//
CXWindowsScreen::CDisplayLock::CDisplayLock(const CXWindowsScreen* screen) :
m_mutex(&screen->m_mutex),
m_display(screen->m_display)
{
assert(m_display != NULL);
m_mutex->lock();
}
CXWindowsScreen::CDisplayLock::~CDisplayLock()
{
m_mutex->unlock();
}
CXWindowsScreen::CDisplayLock::operator Display*() const
{
return m_display;
}

73
synergy/CXWindowsScreen.h Normal file
View File

@ -0,0 +1,73 @@
#ifndef CXWINDOWSSCREEN_H
#define CXWINDOWSSCREEN_H
#include "CMutex.h"
#include "BasicTypes.h"
#include <X11/Xlib.h>
class CThread;
class CXWindowsScreen {
public:
CXWindowsScreen();
virtual ~CXWindowsScreen();
protected:
class CDisplayLock {
public:
CDisplayLock(const CXWindowsScreen*);
~CDisplayLock();
operator Display*() const;
private:
const CMutex* m_mutex;
Display* m_display;
};
friend class CDisplayLock;
// open the X display. calls onOpenDisplay() after opening the display,
// getting the screen, its size, and root window. then it starts the
// event thread.
void openDisplay();
// destroy the window and close the display. calls onCloseDisplay()
// after the event thread has been shut down but before the display
// is closed.
void closeDisplay();
// get the opened screen, its size, its root window. to get the
// display create a CDisplayLock object passing this. while the
// object exists no other threads may access the display. do not
// save the Display* beyond the lifetime of the CDisplayLock.
int getScreen() const;
void getScreenSize(SInt32* w, SInt32* h) const;
Window getRoot() const;
// create a cursor that is transparent everywhere
Cursor createBlankCursor() const;
// wait for and get the next X event. cancellable.
void getEvent(XEvent*) const;
// called by openDisplay() to allow subclasses to prepare the display
virtual void onOpenDisplay() = 0;
// called by closeDisplay() to
virtual void onCloseDisplay() = 0;
// override to process X events
virtual void eventThread(void*) = 0;
private:
CThread* m_eventThread;
Display* m_display;
int m_screen;
Window m_root;
SInt32 m_w, m_h;
// X is not thread safe
CMutex m_mutex;
};
#endif

View File

@ -1,8 +1,6 @@
#include "CXWindowsSecondaryScreen.h" #include "CXWindowsSecondaryScreen.h"
#include "CClient.h" #include "CClient.h"
#include "CThread.h" #include "CThread.h"
#include "CLock.h"
#include "TMethodJob.h"
#include "CLog.h" #include "CLog.h"
#include <assert.h> #include <assert.h>
#include <X11/X.h> #include <X11/X.h>
@ -14,16 +12,14 @@
CXWindowsSecondaryScreen::CXWindowsSecondaryScreen() : CXWindowsSecondaryScreen::CXWindowsSecondaryScreen() :
m_client(NULL), m_client(NULL),
m_display(NULL), m_window(None)
m_window(None),
m_w(0), m_h(0)
{ {
// do nothing // do nothing
} }
CXWindowsSecondaryScreen::~CXWindowsSecondaryScreen() CXWindowsSecondaryScreen::~CXWindowsSecondaryScreen()
{ {
assert(m_display == NULL); assert(m_window == None);
} }
void CXWindowsSecondaryScreen::open(CClient* client) void CXWindowsSecondaryScreen::open(CClient* client)
@ -35,28 +31,113 @@ void CXWindowsSecondaryScreen::open(CClient* client)
m_client = client; m_client = client;
// open the display // open the display
log((CLOG_DEBUG "XOpenDisplay(%s)", "NULL")); openDisplay();
m_display = XOpenDisplay(NULL); // FIXME -- allow non-default
if (m_display == NULL)
throw int(5); // FIXME -- make exception for this
// get default screen
m_screen = DefaultScreen(m_display);
Screen* screen = ScreenOfDisplay(m_display, m_screen);
// get screen size
m_w = WidthOfScreen(screen);
m_h = HeightOfScreen(screen);
log((CLOG_INFO "secondary display size: %dx%d", m_w, m_h));
// verify the availability of the XTest extension // verify the availability of the XTest extension
CDisplayLock display(this);
int majorOpcode, firstEvent, firstError; int majorOpcode, firstEvent, firstError;
if (!XQueryExtension(m_display, XTestExtensionName, if (!XQueryExtension(display, XTestExtensionName,
&majorOpcode, &firstEvent, &firstError)) &majorOpcode, &firstEvent, &firstError))
throw int(6); // FIXME -- make exception for this throw int(6); // FIXME -- make exception for this
}
// get the root window void CXWindowsSecondaryScreen::close()
m_root = RootWindow(m_display, m_screen); {
assert(m_client != NULL);
// close the display
closeDisplay();
// done with client
m_client = NULL;
}
void CXWindowsSecondaryScreen::enter(SInt32 x, SInt32 y)
{
assert(m_window != None);
CDisplayLock display(this);
// warp to requested location
XTestFakeMotionEvent(display, getScreen(), x, y, CurrentTime);
XSync(display, False);
// show cursor
XUnmapWindow(display, m_window);
}
void CXWindowsSecondaryScreen::leave()
{
CDisplayLock display(this);
leaveNoLock(display);
}
void CXWindowsSecondaryScreen::keyDown(
KeyID key, KeyModifierMask mask)
{
CDisplayLock display(this);
XTestFakeKeyEvent(display, mapKey(key, mask), True, CurrentTime);
XSync(display, False);
}
void CXWindowsSecondaryScreen::keyRepeat(
KeyID, KeyModifierMask, SInt32)
{
CDisplayLock display(this);
// FIXME
}
void CXWindowsSecondaryScreen::keyUp(
KeyID key, KeyModifierMask mask)
{
CDisplayLock display(this);
XTestFakeKeyEvent(display, mapKey(key, mask), False, CurrentTime);
XSync(display, False);
}
void CXWindowsSecondaryScreen::mouseDown(ButtonID button)
{
CDisplayLock display(this);
XTestFakeButtonEvent(display, mapButton(button), True, CurrentTime);
XSync(display, False);
}
void CXWindowsSecondaryScreen::mouseUp(ButtonID button)
{
CDisplayLock display(this);
XTestFakeButtonEvent(display, mapButton(button), False, CurrentTime);
XSync(display, False);
}
void CXWindowsSecondaryScreen::mouseMove(SInt32 x, SInt32 y)
{
CDisplayLock display(this);
XTestFakeMotionEvent(display, getScreen(), x, y, CurrentTime);
XSync(display, False);
}
void CXWindowsSecondaryScreen::mouseWheel(SInt32)
{
CDisplayLock display(this);
// FIXME
}
void CXWindowsSecondaryScreen::getSize(
SInt32* width, SInt32* height) const
{
getScreenSize(width, height);
}
SInt32 CXWindowsSecondaryScreen::getJumpZoneSize() const
{
return 0;
}
void CXWindowsSecondaryScreen::onOpenDisplay()
{
assert(m_window == None);
CDisplayLock display(this);
// create the cursor hiding window. this window is used to hide the // create the cursor hiding window. this window is used to hide the
// cursor when it's not on the screen. the window is hidden as soon // cursor when it's not on the screen. the window is hidden as soon
@ -67,230 +148,48 @@ void CXWindowsSecondaryScreen::open(CClient* client)
attr.do_not_propagate_mask = 0; attr.do_not_propagate_mask = 0;
attr.override_redirect = True; attr.override_redirect = True;
attr.cursor = createBlankCursor(); attr.cursor = createBlankCursor();
m_window = XCreateWindow(m_display, m_root, 0, 0, 1, 1, 0, 0, m_window = XCreateWindow(display, getRoot(), 0, 0, 1, 1, 0, 0,
InputOnly, CopyFromParent, InputOnly, CopyFromParent,
CWDontPropagate | CWEventMask | CWDontPropagate | CWEventMask |
CWOverrideRedirect | CWCursor, CWOverrideRedirect | CWCursor,
&attr); &attr);
// become impervious to server grabs // become impervious to server grabs
XTestGrabControl(m_display, True); XTestGrabControl(display, True);
// hide the cursor // hide the cursor
leave(); leaveNoLock(display);
// start processing events
m_eventThread = new CThread(new TMethodJob<CXWindowsSecondaryScreen>(
this, &CXWindowsSecondaryScreen::eventThread));
} }
void CXWindowsSecondaryScreen::close() void CXWindowsSecondaryScreen::onCloseDisplay()
{ {
assert(m_client != NULL); assert(m_window != None);
assert(m_eventThread != NULL);
// stop event thread
m_eventThread->cancel();
m_eventThread->wait();
delete m_eventThread;
m_eventThread = NULL;
// no longer impervious to server grabs // no longer impervious to server grabs
XTestGrabControl(m_display, False); CDisplayLock display(this);
XTestGrabControl(display, False);
// destroy window // destroy window
XDestroyWindow(m_display, m_window); XDestroyWindow(display, m_window);
m_window = None; m_window = None;
// close the display
XCloseDisplay(m_display);
m_display = NULL;
}
void CXWindowsSecondaryScreen::enter(SInt32 x, SInt32 y)
{
assert(m_display != NULL);
assert(m_window != None);
CLock lock(&m_mutex);
// warp to requested location
XTestFakeMotionEvent(m_display, m_screen, x, y, CurrentTime);
XSync(m_display, False);
// show cursor
XUnmapWindow(m_display, m_window);
}
void CXWindowsSecondaryScreen::leave()
{
assert(m_display != NULL);
assert(m_window != None);
CLock lock(&m_mutex);
// move hider window under the mouse (rather than moving the mouse
// somewhere else on the screen)
int x, y, dummy;
unsigned int dummyMask;
Window dummyWindow;
XQueryPointer(m_display, m_root, &dummyWindow, &dummyWindow,
&x, &y, &dummy, &dummy, &dummyMask);
XMoveWindow(m_display, m_window, x, y);
// raise and show the hider window
XMapRaised(m_display, m_window);
// hide cursor by moving it into the hider window
XWarpPointer(m_display, None, m_window, 0, 0, 0, 0, 0, 0);
}
void CXWindowsSecondaryScreen::keyDown(
KeyID key, KeyModifierMask mask)
{
assert(m_display != NULL);
CLock lock(&m_mutex);
XTestFakeKeyEvent(m_display, mapKey(key, mask), True, CurrentTime);
XSync(m_display, False);
}
void CXWindowsSecondaryScreen::keyRepeat(
KeyID, KeyModifierMask, SInt32)
{
assert(m_display != NULL);
CLock lock(&m_mutex);
// FIXME
}
void CXWindowsSecondaryScreen::keyUp(
KeyID key, KeyModifierMask mask)
{
assert(m_display != NULL);
CLock lock(&m_mutex);
XTestFakeKeyEvent(m_display, mapKey(key, mask), False, CurrentTime);
XSync(m_display, False);
}
void CXWindowsSecondaryScreen::mouseDown(ButtonID button)
{
assert(m_display != NULL);
CLock lock(&m_mutex);
XTestFakeButtonEvent(m_display, mapButton(button), True, CurrentTime);
XSync(m_display, False);
}
void CXWindowsSecondaryScreen::mouseUp(ButtonID button)
{
assert(m_display != NULL);
CLock lock(&m_mutex);
XTestFakeButtonEvent(m_display, mapButton(button), False, CurrentTime);
XSync(m_display, False);
}
void CXWindowsSecondaryScreen::mouseMove(SInt32 x, SInt32 y)
{
assert(m_display != NULL);
CLock lock(&m_mutex);
XTestFakeMotionEvent(m_display, m_screen, x, y, CurrentTime);
XSync(m_display, False);
}
void CXWindowsSecondaryScreen::mouseWheel(SInt32)
{
assert(m_display != NULL);
CLock lock(&m_mutex);
// FIXME
}
void CXWindowsSecondaryScreen::getSize(
SInt32* width, SInt32* height) const
{
assert(m_display != NULL);
assert(width != NULL && height != NULL);
*width = m_w;
*height = m_h;
}
SInt32 CXWindowsSecondaryScreen::getJumpZoneSize() const
{
assert(m_display != NULL);
return 0;
}
Cursor CXWindowsSecondaryScreen::createBlankCursor()
{
// this seems just a bit more complicated than really necessary
// get the closet cursor size to 1x1
unsigned int w, h;
XQueryBestCursor(m_display, m_root, 1, 1, &w, &h);
// make bitmap data for cursor of closet size. since the cursor
// is blank we can use the same bitmap for shape and mask: all
// zeros.
const int size = ((w + 7) >> 3) * h;
char* data = new char[size];
memset(data, 0, size);
// make bitmap
Pixmap bitmap = XCreateBitmapFromData(m_display, m_root, data, w, h);
// need an arbitrary color for the cursor
XColor color;
color.pixel = 0;
color.red = color.green = color.blue = 0;
color.flags = DoRed | DoGreen | DoBlue;
// make cursor from bitmap
Cursor cursor = XCreatePixmapCursor(m_display, bitmap, bitmap,
&color, &color, 0, 0);
// don't need bitmap or the data anymore
delete[] data;
XFreePixmap(m_display, bitmap);
return cursor;
} }
void CXWindowsSecondaryScreen::eventThread(void*) void CXWindowsSecondaryScreen::eventThread(void*)
{ {
assert(m_display != NULL);
assert(m_window != None); assert(m_window != None);
for (;;) { for (;;) {
// wait for and then get the next event // wait for and get the next event
m_mutex.lock();
while (XPending(m_display) == 0) {
m_mutex.unlock();
CThread::sleep(0.05);
m_mutex.lock();
}
XEvent xevent; XEvent xevent;
XNextEvent(m_display, &xevent); getEvent(&xevent);
m_mutex.unlock();
// handle event // handle event
switch (xevent.type) { switch (xevent.type) {
case LeaveNotify: { case LeaveNotify: {
// mouse moved out of hider window somehow. hide the window. // mouse moved out of hider window somehow. hide the window.
CLock lock(&m_mutex); assert(m_window != None);
XUnmapWindow(m_display, m_window); CDisplayLock display(this);
XUnmapWindow(display, m_window);
break; break;
} }
@ -313,11 +212,33 @@ void CXWindowsSecondaryScreen::eventThread(void*)
} }
} }
void CXWindowsSecondaryScreen::leaveNoLock(Display* display)
{
assert(display != NULL);
assert(m_window != None);
// move hider window under the mouse (rather than moving the mouse
// somewhere else on the screen)
int x, y, dummy;
unsigned int dummyMask;
Window dummyWindow;
XQueryPointer(display, getRoot(), &dummyWindow, &dummyWindow,
&x, &y, &dummy, &dummy, &dummyMask);
XMoveWindow(display, m_window, x, y);
// raise and show the hider window
XMapRaised(display, m_window);
// hide cursor by moving it into the hider window
XWarpPointer(display, None, m_window, 0, 0, 0, 0, 0, 0);
}
KeyCode CXWindowsSecondaryScreen::mapKey( KeyCode CXWindowsSecondaryScreen::mapKey(
KeyID id, KeyModifierMask /*mask*/) const KeyID id, KeyModifierMask /*mask*/) const
{ {
CDisplayLock display(this);
// FIXME -- use mask // FIXME -- use mask
return XKeysymToKeycode(m_display, static_cast<KeySym>(id)); return XKeysymToKeycode(display, static_cast<KeySym>(id));
} }
unsigned int CXWindowsSecondaryScreen::mapButton( unsigned int CXWindowsSecondaryScreen::mapButton(

View File

@ -1,13 +1,10 @@
#ifndef CXWINDOWSSECONDARYSCREEN_H #ifndef CXWINDOWSSECONDARYSCREEN_H
#define CXWINDOWSSECONDARYSCREEN_H #define CXWINDOWSSECONDARYSCREEN_H
#include "CMutex.h" #include "CXWindowsScreen.h"
#include "ISecondaryScreen.h" #include "ISecondaryScreen.h"
#include <X11/Xlib.h>
class CThread; class CXWindowsSecondaryScreen : public CXWindowsScreen, public ISecondaryScreen {
class CXWindowsSecondaryScreen : public ISecondaryScreen {
public: public:
CXWindowsSecondaryScreen(); CXWindowsSecondaryScreen();
virtual ~CXWindowsSecondaryScreen(); virtual ~CXWindowsSecondaryScreen();
@ -27,23 +24,20 @@ class CXWindowsSecondaryScreen : public ISecondaryScreen {
virtual void getSize(SInt32* width, SInt32* height) const; virtual void getSize(SInt32* width, SInt32* height) const;
virtual SInt32 getJumpZoneSize() const; virtual SInt32 getJumpZoneSize() const;
protected:
// CXWindowsScreen overrides
virtual void onOpenDisplay();
virtual void onCloseDisplay();
virtual void eventThread(void*);
private: private:
Cursor createBlankCursor(); void leaveNoLock(Display*);
void eventThread(void*);
KeyCode mapKey(KeyID, KeyModifierMask) const; KeyCode mapKey(KeyID, KeyModifierMask) const;
unsigned int mapButton(ButtonID button) const; unsigned int mapButton(ButtonID button) const;
private: private:
CClient* m_client; CClient* m_client;
CThread* m_eventThread;
Display* m_display;
int m_screen;
Window m_root;
Window m_window; Window m_window;
SInt32 m_w, m_h;
// X is not thread safe
CMutex m_mutex;
}; };
#endif #endif

View File

@ -20,6 +20,7 @@ CXXFILES = \
CServerProtocol1_0.cpp \ CServerProtocol1_0.cpp \
CScreenMap.cpp \ CScreenMap.cpp \
CServer.cpp \ CServer.cpp \
CXWindowsScreen.cpp \
CXWindowsPrimaryScreen.cpp \ CXWindowsPrimaryScreen.cpp \
CXWindowsSecondaryScreen.cpp \ CXWindowsSecondaryScreen.cpp \
XSynergy.cpp \ XSynergy.cpp \