|
@@ -2,8 +2,21 @@
|
2
|
2
|
|
3
|
3
|
slop::XEngine* xengine = new slop::XEngine();
|
4
|
4
|
|
|
5
|
+static Bool isDestroyNotify( Display* dpy, XEvent* ev, XPointer win ) {
|
|
6
|
+ return ev->type == DestroyNotify && ev->xdestroywindow.window == *((Window*)win);
|
|
7
|
+}
|
|
8
|
+
|
|
9
|
+int slop::XEngineErrorHandler( Display* dpy, XErrorEvent* event ) {
|
|
10
|
+ // Ignore XGrabKeyboard BadAccess errors
|
|
11
|
+ // 31 = XGrabKeyboard's request code
|
|
12
|
+ if ( event->request_code == 31 && event->error_code == BadAccess ) {
|
|
13
|
+ return 0;
|
|
14
|
+ }
|
|
15
|
+ // Otherwise call the default error handler
|
|
16
|
+ return slop::OldXErrorHandler( dpy, event );
|
|
17
|
+}
|
|
18
|
+
|
5
|
19
|
slop::XEngine::XEngine() {
|
6
|
|
- m_keypressed = false;
|
7
|
20
|
m_display = NULL;
|
8
|
21
|
m_visual = NULL;
|
9
|
22
|
m_screen = NULL;
|
|
@@ -65,23 +78,33 @@ int slop::XEngine::init( std::string display ) {
|
65
|
78
|
m_root = DefaultRootWindow( m_display );
|
66
|
79
|
|
67
|
80
|
m_good = true;
|
|
81
|
+ slop::OldXErrorHandler = XSetErrorHandler( slop::XEngineErrorHandler );
|
68
|
82
|
return 0;
|
69
|
83
|
}
|
70
|
84
|
|
|
85
|
+bool slop::XEngine::anyKeyPressed() {
|
|
86
|
+ if ( !m_good ) {
|
|
87
|
+ return false;
|
|
88
|
+ }
|
|
89
|
+ // Thanks to SFML for some reliable key state grabbing.
|
|
90
|
+ // Get the whole keyboard state
|
|
91
|
+ char keys[ 32 ];
|
|
92
|
+ XQueryKeymap( m_display, keys );
|
|
93
|
+ // Each bit indicates a different key, 1 for pressed, 0 otherwise.
|
|
94
|
+ // Every bit should be 0 if nothing is pressed.
|
|
95
|
+ for ( unsigned int i = 0; i < 32; i++ ) {
|
|
96
|
+ if ( keys[ i ] != 0 ) {
|
|
97
|
+ return true;
|
|
98
|
+ }
|
|
99
|
+ }
|
|
100
|
+ return false;
|
|
101
|
+}
|
|
102
|
+
|
71
|
103
|
int slop::XEngine::grabKeyboard() {
|
72
|
104
|
if ( !m_good ) {
|
73
|
105
|
return 1;
|
74
|
106
|
}
|
75
|
|
- XGrabKey( m_display, AnyKey, AnyModifier, m_root, False, GrabModeAsync, GrabModeAsync );
|
76
|
|
- // For whatever we fail to grab the keyboard 100% of the time if slop is launched in the background.
|
77
|
|
- /*int err = XGrabKeyboard( m_display, m_root, False,
|
78
|
|
- GrabModeAsync, GrabModeAsync, CurrentTime );
|
79
|
|
- if ( err != GrabSuccess ) {
|
80
|
|
- fprintf( stderr, "Error: Failed to grab X keyboard.\n" );
|
81
|
|
- fprintf( stderr, "This can be caused by launching slop incorrectly.\n" );
|
82
|
|
- fprintf( stderr, "gnome-session launches it fine from keyboard binds.\n" );
|
83
|
|
- return 1;
|
84
|
|
- }*/
|
|
107
|
+ XGrabKeyboard( m_display, m_root, False, GrabModeAsync, GrabModeAsync, CurrentTime );
|
85
|
108
|
return 0;
|
86
|
109
|
}
|
87
|
110
|
|
|
@@ -164,15 +187,11 @@ void slop::XEngine::tick() {
|
164
|
187
|
}
|
165
|
188
|
break;
|
166
|
189
|
}
|
167
|
|
- // For this particular utility, we only care if a key is pressed.
|
168
|
|
- // I'm too lazy to implement an actual keyhandler for that.
|
|
190
|
+ // Due to X11 really hating applications grabbing the keyboard, we use XQueryKeymap to check for downed keys elsewhere.
|
169
|
191
|
case KeyPress: {
|
170
|
|
- m_keypressed = true;
|
171
|
192
|
break;
|
172
|
193
|
}
|
173
|
|
- // Since we also don't care if it's released, do nothing! yay
|
174
|
194
|
case KeyRelease: {
|
175
|
|
- //m_keypressed = false;
|
176
|
195
|
break;
|
177
|
196
|
}
|
178
|
197
|
default: break;
|
|
@@ -226,10 +245,13 @@ slop::Rectangle::~Rectangle() {
|
226
|
245
|
}
|
227
|
246
|
//XFreeColors( xengine->m_display, xengine->m_colormap, m_color.pixel, 1,
|
228
|
247
|
// Attempt to move window offscreen before trying to remove it.
|
229
|
|
- XResizeWindow( xengine->m_display, m_window, 1, 1 );
|
230
|
|
- XMoveWindow( xengine->m_display, m_window, 0, 0 );
|
231
|
|
- XUnmapWindow( xengine->m_display, m_window );
|
|
248
|
+ //XResizeWindow( xengine->m_display, m_window, 1, 1 );
|
|
249
|
+ //XMoveWindow( xengine->m_display, m_window, 0, 0 );
|
|
250
|
+ //XUnmapWindow( xengine->m_display, m_window );
|
232
|
251
|
XDestroyWindow( xengine->m_display, m_window );
|
|
252
|
+ XEvent event;
|
|
253
|
+ // Block until the window is actually completely removed.
|
|
254
|
+ XIfEvent( xengine->m_display, &event, &isDestroyNotify, (XPointer)&m_window );
|
233
|
255
|
}
|
234
|
256
|
|
235
|
257
|
slop::Rectangle::Rectangle( int x, int y, int width, int height, int border, int padding, float r, float g, float b ) {
|
|
@@ -264,7 +286,9 @@ slop::Rectangle::Rectangle( int x, int y, int width, int height, int border, int
|
264
|
286
|
attributes.override_redirect = True;
|
265
|
287
|
// We must use our color map, because that's where our color is allocated.
|
266
|
288
|
attributes.colormap = xengine->m_colormap;
|
267
|
|
- unsigned long valueMask = CWBackPixmap | CWBackPixel | CWOverrideRedirect | CWColormap;
|
|
289
|
+ // Make sure we know when we've been successfully destroyed later!
|
|
290
|
+ attributes.event_mask = StructureNotifyMask;
|
|
291
|
+ unsigned long valueMask = CWBackPixmap | CWBackPixel | CWOverrideRedirect | CWColormap | CWEventMask;
|
268
|
292
|
|
269
|
293
|
// Create the window offset by our generated offsets (see constrain( float, float ))
|
270
|
294
|
m_window = XCreateWindow( xengine->m_display, xengine->m_root, m_x+m_xoffset-m_border, m_y+m_yoffset-m_border, m_width+m_border*2, m_height+m_border*2,
|