Mercurial > projects > dynamin
annotate dynamin/gui/x_window.d @ 24:43a88caead16
Implement keyDown and keyUp events with X.
author | Jordan Miner <jminer7@gmail.com> |
---|---|
date | Sat, 25 Jul 2009 15:01:46 -0500 |
parents | d55b5b998412 |
children | 0577e8738dc8 |
rev | line source |
---|---|
0 | 1 // Written in the D programming language |
2 // www.digitalmars.com/d/ | |
3 | |
4 /* | |
5 * The contents of this file are subject to the Mozilla Public License Version | |
6 * 1.1 (the "License"); you may not use this file except in compliance with | |
7 * the License. You may obtain a copy of the License at | |
8 * http://www.mozilla.org/MPL/ | |
9 * | |
10 * Software distributed under the License is distributed on an "AS IS" basis, | |
11 * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License | |
12 * for the specific language governing rights and limitations under the | |
13 * License. | |
14 * | |
15 * The Original Code is the Dynamin library. | |
16 * | |
17 * The Initial Developer of the Original Code is Jordan Miner. | |
18 * Portions created by the Initial Developer are Copyright (C) 2007-2009 | |
19 * the Initial Developer. All Rights Reserved. | |
20 * | |
21 * Contributor(s): | |
22 * Jordan Miner <jminer7@gmail.com> | |
23 * | |
24 */ | |
25 | |
26 module dynamin.gui.x_window; | |
27 | |
28 public import dynamin.core.string; | |
29 public import dynamin.core.global; | |
30 public import dynamin.core.math; | |
31 public import dynamin.gui.window; | |
24
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
32 public import dynamin.gui.key; |
0 | 33 public import dynamin.c.xlib; |
34 public import dynamin.c.xlib : XWindow = Window; | |
35 public import dynamin.c.xmu; | |
36 public import dynamin.c.cairo; | |
37 public import dynamin.c.cairo_xlib; | |
38 public import tango.stdc.string; | |
39 public import tango.io.Stdout; | |
40 | |
41 /* | |
42 ** Window property: | |
43 ** _NET_FRAME_EXTENTS(CARDINAL) = 4, 4, 23, 4 | |
44 ** left, right, top and bottom border sizes | |
45 */ | |
46 | |
47 Window[XWindow] windows; | |
48 void setControl(XWindow handle, Window win) { | |
49 if(win is null) | |
50 windows.remove(handle); | |
51 else | |
52 windows[handle] = win; | |
53 } | |
54 | |
55 Window getControl(XWindow handle) { | |
56 auto tmp = handle in windows; | |
57 return tmp is null ? null : *tmp; | |
58 } | |
59 | |
60 /** | |
61 * A simpler method that returns all the data in a property. | |
62 * NOTE: the returned data still has to be freed with XFree() | |
63 */ | |
64 void* getXWindowProperty(XDisplay* d, XWindow w, XAtom prop, int* numRet = null) { | |
65 XAtom actualType; | |
66 int actualFormat; | |
67 uint nitems, bytesAfter; | |
68 void* ptr; | |
69 XGetWindowProperty(d, w, prop, | |
70 0, 0xFFFFFFFF, false, AnyPropertyType, | |
71 &actualType, &actualFormat, &nitems, &bytesAfter, | |
72 &ptr); | |
73 if(numRet) *numRet = nitems; | |
74 return ptr; | |
75 } | |
76 bool isWMPropertySupported(XAtom prop) { | |
77 int count; | |
78 XAtom* atoms = cast(XAtom*)getXWindowProperty(display, | |
79 root, XA._NET_SUPPORTED, &count); | |
80 scope(exit) XFree(atoms); | |
81 for(int i = 0; i < count; ++i) | |
82 if(atoms[i] == prop) | |
83 return true; | |
84 return false; | |
85 } | |
86 bool isTopLevelReparented(XWindow w) { | |
87 XWindow root, parent; | |
88 XWindow* children; | |
89 uint numChildren; | |
90 XQueryTree(display, w, | |
91 &root, &parent, &children, &numChildren); | |
92 return parent != root; | |
93 } | |
94 | |
95 XDisplay* display; | |
96 XWindow root; | |
97 XWindow msgWin; | |
98 abstract class XA { // X atoms | |
99 static: | |
100 XAtom _NET_SUPPORTED, _NET_WM_NAME, _NET_WORKAREA, _NET_FRAME_EXTENTS; | |
101 XAtom _NET_REQUEST_FRAME_EXTENTS; | |
102 XAtom _NET_MOVERESIZE_WINDOW; | |
103 XAtom _NET_WM_WINDOW_TYPE; | |
104 XAtom _NET_WM_WINDOW_TYPE_MENU, _NET_WM_WINDOW_TYPE_UTILITY; | |
105 XAtom _NET_WM_WINDOW_TYPE_SPLASH; | |
106 XAtom _NET_WM_WINDOW_TYPE_DIALOG, _NET_WM_WINDOW_TYPE_NORMAL; | |
107 XAtom WM_PROTOCOLS, WM_DELETE_WINDOW, _NET_WM_SYNC_REQUEST; | |
108 XAtom UTF8_STRING, ATOM; | |
109 XAtom _MOTIF_WM_HINTS; | |
110 XAtom CLIPBOARD, PRIMARY, TARGETS, CLIPBOARD_MANAGER, SAVE_TARGETS; | |
111 XAtom DYNAMIN_SELECTION; | |
112 } | |
113 static this() { | |
114 display = XOpenDisplay(null); | |
115 if(!display) | |
116 Stdout("XOpenDisplay() failed").newline; | |
117 root = XRootWindow(display, XDefaultScreen(display)); | |
118 | |
119 msgWin = XCreateSimpleWindow(display, root, 0, 0, 1, 1, 0, 0, 0); | |
120 | |
121 XA._NET_SUPPORTED = XInternAtom(display, "_NET_SUPPORTED", false); | |
122 XA._NET_WM_NAME = XInternAtom(display, "_NET_WM_NAME", false); | |
123 XA._NET_WORKAREA = XInternAtom(display, "_NET_WORKAREA", false); | |
124 XA._NET_FRAME_EXTENTS = XInternAtom(display, "_NET_FRAME_EXTENTS", false); | |
125 XA._NET_REQUEST_FRAME_EXTENTS = | |
126 XInternAtom(display, "_NET_REQUEST_FRAME_EXTENTS", false); | |
127 XA._NET_MOVERESIZE_WINDOW = | |
128 XInternAtom(display, "_NET_MOVERESIZE_WINDOW", false); | |
129 XA._NET_WM_WINDOW_TYPE = | |
130 XInternAtom(display, "_NET_WM_WINDOW_TYPE", false); | |
131 XA._NET_WM_WINDOW_TYPE_MENU = | |
132 XInternAtom(display, "_NET_WM_WINDOW_TYPE_MENU", false); | |
133 XA._NET_WM_WINDOW_TYPE_UTILITY = | |
134 XInternAtom(display, "_NET_WM_WINDOW_TYPE_UTILITY", false); | |
135 XA._NET_WM_WINDOW_TYPE_SPLASH = | |
136 XInternAtom(display, "_NET_WM_WINDOW_TYPE_SPLASH", false); | |
137 XA._NET_WM_WINDOW_TYPE_DIALOG = | |
138 XInternAtom(display, "_NET_WM_WINDOW_TYPE_DIALOG", false); | |
139 XA._NET_WM_WINDOW_TYPE_NORMAL = | |
140 XInternAtom(display, "_NET_WM_WINDOW_TYPE_NORMAL", false); | |
141 XA.WM_PROTOCOLS = XInternAtom(display, "WM_PROTOCOLS", false); | |
142 XA.WM_DELETE_WINDOW = XInternAtom(display, "WM_DELETE_WINDOW", false); | |
143 XA._NET_WM_SYNC_REQUEST = | |
144 XInternAtom(display, "_NET_WM_SYNC_REQUEST", false); | |
145 XA.UTF8_STRING = XInternAtom(display, "UTF8_STRING", false); | |
146 XA.ATOM = XInternAtom(display, "ATOM", false); | |
147 XA._MOTIF_WM_HINTS = XInternAtom(display, "_MOTIF_WM_HINTS", false); | |
148 XA.CLIPBOARD = XInternAtom(display, "CLIPBOARD", false); | |
149 XA.PRIMARY = XInternAtom(display, "PRIMARY", false); | |
150 XA.TARGETS = XInternAtom(display, "TARGETS", false); | |
151 XA.CLIPBOARD_MANAGER = XInternAtom(display, "CLIPBOARD_MANAGER", false); | |
152 XA.SAVE_TARGETS = XInternAtom(display, "SAVE_TARGETS", false); | |
153 | |
154 XA.DYNAMIN_SELECTION = XInternAtom(display, "DYNAMIN_SELECTION", false); | |
155 if(!isWMPropertySupported(XA._NET_WM_NAME)) | |
156 Stdout("warning: WM does not support _NET_WM_NAME").newline; | |
157 if(!isWMPropertySupported(XA._NET_WORKAREA)) | |
158 Stdout("warning: WM does not support _NET_WORKAREA").newline; | |
159 if(!isWMPropertySupported(XA._NET_FRAME_EXTENTS)) | |
160 Stdout("warning: WM does not support _NET_FRAME_EXTENTS").newline; | |
161 if(!isWMPropertySupported(XA._NET_WORKAREA)) | |
162 Stdout("warning: WM does not support _NET_WORKAREA").newline; | |
163 } | |
164 | |
165 struct _InvalidRect { | |
166 XWindow window; | |
167 int x, y, width, height; | |
168 _InvalidRect getUnion(_InvalidRect rect) { | |
169 auto x2 = min(x, rect.x); | |
170 auto y2 = min(y, rect.y); | |
171 _InvalidRect rect2; | |
172 rect2.window = window; | |
173 rect2.width = max(x+width, rect.x+rect.width)-x2; | |
174 rect2.height = max(y+height, rect.y+rect.height)-y2; | |
175 rect2.x = x2; | |
176 rect2.y = y2; | |
177 return rect2; | |
178 } | |
179 } | |
180 /+struct InvalidRect { | |
181 XWindow window; | |
182 int x, y, width, height; | |
183 } | |
184 | |
185 static class PaintQueue { | |
186 static: | |
187 //LinkedList!(InvalidRect) rects; | |
188 InvalidRect[] rects; | |
189 static this() { | |
190 //rects = new LinkedList!(InvalidRect)(20); | |
191 rects.length = 20; | |
192 rects.length = 0; | |
193 } | |
194 void add(XWindow win, int x, int y, int width, int height) { | |
195 rects.length = rects.length + 1; | |
196 rects[$-1].window = win; | |
197 rects[$-1].x = x; | |
198 rects[$-1].y = y; | |
199 rects[$-1].width = width; | |
200 rects[$-1].height = height; | |
201 } | |
202 bool shouldMerge(int x1, int y1, int width1, int height1, | |
203 int x2, int y2, int width2, int height2) { | |
204 return x2 <= x1 + width1 && y2 <= y1 + height1 && | |
205 x2 + width2 >= x1 && y2 + height2 >= y1; | |
206 } | |
207 void Union(inout int x, inout int y, inout int width, inout int height, | |
208 int x2, int y2, int width2, int height2) { | |
209 } | |
210 void compress() { | |
211 } | |
212 }+/ | |
213 | |
24
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
214 Key prevKey = Key.None; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
215 |
0 | 216 //{{{ ApplicationBackend |
217 template ApplicationBackend() { | |
218 void backend_run(Window w) { | |
219 bool isWindowVisible() { | |
220 if(w is null) return true; | |
221 return w.visible; | |
222 } | |
223 XEvent ev; | |
224 while(isWindowVisible()) { | |
225 if(XEventsQueued(display, QueuedAlready) == 0) { | |
226 _InvalidRect[] rects = Window.invalidRects; | |
227 Window.invalidRects.length = 0; | |
228 while(rects.length > 0) { | |
229 auto rect = rects[0]; | |
230 // TODO: fix this...right now it gens one Expose with | |
231 // the union of invalid rects | |
232 for(int i = rects.length-1; i >= 0; --i) { | |
233 if(rect.window == rects[i].window) { | |
234 rect = rect.getUnion(rects[i]); | |
235 | |
236 arrayCopy!(_InvalidRect)(rects, i+1, rects, i, rects.length-i-1); | |
237 rects.length = rects.length-1; | |
238 } | |
239 } | |
240 ev.xexpose.type = Expose; | |
241 ev.xexpose.display = display; | |
242 ev.xexpose.window = rect.window; | |
243 ev.xexpose.x = rect.x; | |
244 ev.xexpose.y = rect.y; | |
245 ev.xexpose.width = rect.width; | |
246 ev.xexpose.height = rect.height; | |
247 ev.xexpose.count = -2; // came from here | |
248 XPutBackEvent(display, &ev); | |
249 } | |
250 } | |
251 XNextEvent(display, &ev); | |
252 auto evDisplay = ev.xany.display; | |
253 auto evWindow = ev.xany.window; | |
254 Window c = getControl(evWindow); | |
255 // c will be null for SelectionRequest events | |
256 //if(c is null) | |
257 // continue; | |
258 //{{{ helper functions | |
259 void createMouseEvent(void delegate(MouseEventArgs args) func) { | |
260 MouseButton button; | |
261 auto buttonEv = ev.xbutton; | |
262 switch(buttonEv.button) { | |
263 case 1: button = MouseButton.Left; break; | |
264 case 2: button = MouseButton.Middle; break; | |
265 case 3: button = MouseButton.Right; break; | |
266 default: return; | |
267 } | |
268 scope args = new MouseEventArgs(buttonEv.x+c._borderSize.left, buttonEv.y+c._borderSize.top, button); | |
269 func(args); | |
270 } | |
24
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
271 bool isKeyDown(uint keycode) { |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
272 ubyte[32] keys; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
273 XQueryKeymap(display, keys.ptr); |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
274 return cast(bool)( (keys[keycode / 8] >> (keycode % 8)) & 1 ); |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
275 } |
0 | 276 //}}} |
277 switch(ev.type) { | |
278 case MapNotify: | |
279 c.mapped = true; | |
280 break; | |
281 case UnmapNotify: | |
282 _InvalidRect[] rects = Window.invalidRects; | |
283 for(int i = rects.length-1; i >= 0; --i) { | |
284 if(rects[i].window == evWindow) { | |
285 arrayCopy!(_InvalidRect)( | |
286 rects, i+1, rects, i, rects.length-i-1); | |
287 rects.length = rects.length-1; | |
288 } | |
289 } | |
290 Window.invalidRects = rects; | |
291 c.mapped = false; | |
292 break; | |
293 case DestroyNotify: | |
294 setControl(evWindow, null); | |
295 break; | |
296 case ClientMessage: | |
297 auto clientEv = ev.xclient; | |
298 if(clientEv.message_type != XA.WM_PROTOCOLS) | |
299 break; | |
300 if(clientEv.data.l[0] == XA.WM_DELETE_WINDOW) { | |
301 XDestroyWindow(evDisplay, evWindow); | |
302 return; // TODO: check event, and figure out what to do | |
303 } | |
304 break; | |
305 case KeyPress: | |
24
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
306 auto sym = XLookupKeysym(&ev.xkey, 0); |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
307 if(sym == NoSymbol) |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
308 break; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
309 // Since X gives no way to tell if a KeyPress is generated by |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
310 // auto-repeat, prevKey is used here to tell. |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
311 auto k = KeysymToKey(sym); |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
312 scope args = new KeyEventArgs(k, k == prevKey); |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
313 prevKey = k; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
314 Control focused = c.focusedControl ? c.focusedControl : c; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
315 focused.keyDown(args); |
0 | 316 break; |
317 case KeyRelease: | |
24
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
318 // When X does auto-repeat for a held down key, it sends |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
319 // a KeyPress and KeyRelease every time, even though the key is |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
320 // down constantly. Here we check if the key is down, and if so, |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
321 // not send a keyUp event. |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
322 if(isKeyDown(ev.xkey.keycode)) |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
323 break; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
324 auto sym = XLookupKeysym(&ev.xkey, 0); |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
325 if(sym == NoSymbol) |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
326 break; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
327 auto k = KeysymToKey(sym); |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
328 if(k == prevKey) |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
329 prevKey = Key.None; // can't repeat after released |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
330 scope args = new KeyEventArgs(k, false); |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
331 Control focused = c.focusedControl ? c.focusedControl : c; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
332 focused.keyUp(args); |
0 | 333 break; |
334 case ButtonPress: | |
335 //Button4 is wheel scroll up | |
336 //Button5 is wheel scroll down | |
337 createMouseEvent((MouseEventArgs args) { c.mouseDown(args); }); | |
338 break; | |
339 case ButtonRelease: | |
340 createMouseEvent((MouseEventArgs args) { c.mouseUp(args); }); | |
341 break; | |
342 case MotionNotify: | |
343 auto motionEv = ev.xmotion; | |
344 Control captor = getCaptorControl(); | |
345 Point pt = Point(motionEv.x+c.borderSize.left, motionEv.y+c.borderSize.top); | |
346 if(captor) | |
347 pt = c.contentToContent(pt, captor); | |
348 else | |
349 captor = c; | |
350 scope args = new MouseEventArgs(pt.x, pt.y, MouseButton.None); | |
351 if(motionEv.state & | |
352 (Button1Mask | Button2Mask | Button3Mask)) { | |
353 captor.mouseDragged(args); | |
354 } else | |
355 captor.mouseMoved(args); | |
356 break; | |
357 case EnterNotify: | |
358 break; | |
359 case LeaveNotify: | |
360 break; | |
361 case FocusIn: | |
362 break; | |
363 case FocusOut: | |
364 break; | |
365 case Expose: | |
366 // TODO: move the painting code out of here and: | |
367 // make a PaintQueue class and put this here: | |
368 // PaintQueue.add(c, exposeEv.x, exposeEv.y, exposeEv.width, exposeEv.height); | |
369 // then, in Window.repaint(), have this: | |
370 // PaintQueue.add(this, cast(int)x, cast(int)exposeEv.y, cast(int)exposeEv.width, cast(int)exposeEv.height); | |
371 // Have a PaintQueue.Compress method that merges | |
372 // all invalidated rects that touch or overlap. | |
373 // In the if(!XPending(..)) above, just loop over all the | |
374 // rects in the PaintQueue, painting them. | |
375 auto exposeEv = ev.xexpose; | |
376 if(exposeEv.count != -2) { | |
377 c.repaint(exposeEv.x, exposeEv.y, exposeEv.width, exposeEv.height); | |
378 break; | |
379 } | |
380 //printf("repainting x=%d, y=%d, width=%d, height=%d\n", | |
381 // exposeEv.x, exposeEv.y, exposeEv.width, exposeEv.height); | |
382 | |
383 auto surfaceWin = cairo_xlib_surface_create( | |
384 evDisplay, evWindow, | |
385 XDefaultVisual(evDisplay, XDefaultScreen(evDisplay)), | |
386 cast(int)c.width, cast(int)c.height); | |
387 // TODO: ^ should be contentWidth/height or got from evWindow | |
388 auto crWin = cairo_create(surfaceWin); | |
389 cairo_surface_destroy(surfaceWin); | |
390 | |
391 auto surfaceBuff = cairo_surface_create_similar(surfaceWin, CAIRO_CONTENT_COLOR, exposeEv.width, exposeEv.height); | |
392 // TODO: use cairo_translate instead, I guess, as | |
393 // I had to change the Windows backend to it... | |
394 cairo_surface_set_device_offset(surfaceBuff, -exposeEv.x-c._borderSize.left, -exposeEv.y-c._borderSize.top); | |
395 auto crBuff = cairo_create(surfaceBuff); | |
396 cairo_surface_destroy(surfaceBuff); | |
397 | |
398 cairo_set_source_rgb(crBuff, w.content.backColor.R/255.0, w.content.backColor.G/255.0, w.content.backColor.B/255.0); | |
399 cairo_paint(crBuff); | |
400 | |
401 cairo_set_source_rgb(crBuff, 0, 0, 0); | |
402 cairo_set_line_width(crBuff, 1.0); | |
403 | |
404 auto g = new Graphics(crBuff); | |
405 scope args = new PaintingEventArgs(g); | |
406 c.painting(args); | |
407 delete g; | |
408 | |
409 cairo_surface_set_device_offset(surfaceBuff, -exposeEv.x, -exposeEv.y); | |
410 cairo_set_source_surface(crWin, surfaceBuff, 0, 0); | |
411 cairo_rectangle(crWin, exposeEv.x, exposeEv.y, exposeEv.width, exposeEv.height); | |
412 cairo_fill(crWin); | |
413 | |
414 cairo_destroy(crBuff); | |
415 cairo_destroy(crWin); | |
416 break; | |
417 case PropertyNotify: | |
418 auto propertyEv = ev.xproperty; | |
419 if(propertyEv.atom == XA._NET_FRAME_EXTENTS && | |
420 propertyEv.state != PropertyDelete) | |
421 c.backend_nativeToBorderSize(); | |
422 break; | |
423 case ConfigureNotify: | |
424 auto configureEv = ev.xconfigure; | |
425 c.repaint(); | |
426 c.backend_nativeToLocationSize(); | |
427 break; | |
428 case SelectionRequest: | |
429 auto selReqEv = ev.xselectionrequest; | |
430 XEvent fullEv; | |
431 auto selEv = &fullEv.xselection; | |
432 selEv.type = SelectionNotify; | |
433 selEv.requestor = selReqEv.requestor; | |
434 selEv.selection = selReqEv.selection; | |
435 selEv.target = selReqEv.target; | |
436 if(selReqEv.property != None) | |
437 selEv.property = selReqEv.property; | |
438 else | |
439 selEv.property = XA.DYNAMIN_SELECTION; | |
440 selEv.time = selReqEv.time; | |
441 Stdout.format("requestor: {}", selReqEv.requestor).newline; | |
442 Stdout.format("target: {}", selReqEv.target).newline; | |
443 ClipboardData* data; // change to array when supporting multiple | |
444 if(selReqEv.selection == XA.CLIPBOARD) | |
445 data = &Clipboard.data; | |
446 else if(selReqEv.selection == XA.PRIMARY) | |
447 data = &Selection.data; | |
448 else { | |
449 selEv.property = None; | |
450 XSendEvent(display, selEv.requestor, false, 0, &fullEv); | |
451 break; | |
452 } | |
453 if(selReqEv.target == XA.TARGETS) { | |
454 XChangeProperty(display, selEv.requestor, selEv.property, | |
455 selEv.target, 32, PropModeReplace, &data.target, 1); | |
456 XSendEvent(display, selEv.requestor, false, 0, &fullEv); | |
457 break; | |
458 } else if(selReqEv.target != data.target) { | |
459 selEv.property = None; | |
460 XSendEvent(display, selEv.requestor, false, 0, &fullEv); | |
461 break; | |
462 } | |
463 XChangeProperty(display, selEv.requestor, selEv.property, | |
464 data.target, 8, PropModeReplace, data.data, data.length); | |
465 XSendEvent(display, selEv.requestor, false, 0, &fullEv); | |
466 break; | |
24
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
467 case MappingNotify: |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
468 XRefreshKeyboardMapping(&ev.xmapping); |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
469 break; |
0 | 470 default: |
471 break; | |
472 } | |
473 } | |
474 } | |
18
836a064828e8
Implement FileDialog/DirectoryDialog with GTK
Jordan Miner <jminer7@gmail.com>
parents:
0
diff
changeset
|
475 void backend_invoke(void delegate() dg) { |
836a064828e8
Implement FileDialog/DirectoryDialog with GTK
Jordan Miner <jminer7@gmail.com>
parents:
0
diff
changeset
|
476 // TODO: |
836a064828e8
Implement FileDialog/DirectoryDialog with GTK
Jordan Miner <jminer7@gmail.com>
parents:
0
diff
changeset
|
477 } |
836a064828e8
Implement FileDialog/DirectoryDialog with GTK
Jordan Miner <jminer7@gmail.com>
parents:
0
diff
changeset
|
478 void backend_invokeNow(void delegate() dg) { |
836a064828e8
Implement FileDialog/DirectoryDialog with GTK
Jordan Miner <jminer7@gmail.com>
parents:
0
diff
changeset
|
479 // TODO: |
836a064828e8
Implement FileDialog/DirectoryDialog with GTK
Jordan Miner <jminer7@gmail.com>
parents:
0
diff
changeset
|
480 } |
836a064828e8
Implement FileDialog/DirectoryDialog with GTK
Jordan Miner <jminer7@gmail.com>
parents:
0
diff
changeset
|
481 |
0 | 482 } |
483 //}}} | |
484 | |
24
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
485 //{{{ KeysymToKey() |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
486 Key KeysymToKey(int sym) { |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
487 switch(sym) { |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
488 case XK_parenright: |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
489 case XK_0: return Key.D0; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
490 case XK_exclam: |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
491 case XK_1: return Key.D1; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
492 case XK_at: |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
493 case XK_2: return Key.D2; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
494 case XK_numbersign: |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
495 case XK_3: return Key.D3; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
496 case XK_dollar: |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
497 case XK_4: return Key.D4; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
498 case XK_percent: |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
499 case XK_5: return Key.D5; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
500 case XK_asciicircum: |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
501 case XK_6: return Key.D6; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
502 case XK_ampersand: |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
503 case XK_7: return Key.D7; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
504 case XK_asterisk: |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
505 case XK_8: return Key.D8; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
506 case XK_parenleft: |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
507 case XK_9: return Key.D9; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
508 |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
509 case XK_F1: return Key.F1; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
510 case XK_F2: return Key.F2; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
511 case XK_F3: return Key.F3; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
512 case XK_F4: return Key.F4; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
513 case XK_F5: return Key.F5; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
514 case XK_F6: return Key.F6; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
515 case XK_F7: return Key.F7; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
516 case XK_F8: return Key.F8; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
517 case XK_F9: return Key.F9; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
518 case XK_F10: return Key.F10; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
519 case XK_F11: return Key.F11; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
520 case XK_F12: return Key.F12; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
521 |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
522 case XK_Escape: return Key.Escape; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
523 case XK_Tab: return Key.Tab; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
524 case XK_BackSpace: return Key.Backspace; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
525 case XK_Return: return Key.Enter; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
526 case XK_KP_Enter: return Key.Enter; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
527 case XK_space: return Key.Space; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
528 |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
529 case XK_KP_Left: |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
530 case XK_Left: return Key.Left; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
531 case XK_KP_Right: |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
532 case XK_Right: return Key.Right; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
533 case XK_KP_Up: |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
534 case XK_Up: return Key.Up; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
535 case XK_KP_Down: |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
536 case XK_Down: return Key.Down; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
537 |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
538 case XK_KP_Insert: |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
539 case XK_Insert: return Key.Insert; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
540 case XK_KP_Delete: |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
541 case XK_Delete: return Key.Delete; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
542 case XK_KP_Home: |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
543 case XK_Home: return Key.Home; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
544 case XK_KP_End: |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
545 case XK_End: return Key.End; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
546 case XK_KP_Prior: |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
547 case XK_Prior: return Key.PageUp; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
548 case XK_KP_Next: |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
549 case XK_Next: return Key.PageDown; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
550 |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
551 case XK_Print: |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
552 case XK_Sys_Req: return Key.PrintScreen; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
553 case XK_Pause: |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
554 case XK_Break: return Key.Pause; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
555 |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
556 case XK_Caps_Lock: return Key.CapsLock; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
557 case XK_Num_Lock: return Key.NumLock; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
558 case XK_Scroll_Lock: return Key.ScrollLock; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
559 |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
560 case XK_KP_0: return Key.NumPad0; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
561 case XK_KP_1: return Key.NumPad1; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
562 case XK_KP_2: return Key.NumPad2; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
563 case XK_KP_3: return Key.NumPad3; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
564 case XK_KP_4: return Key.NumPad4; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
565 case XK_KP_5: return Key.NumPad5; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
566 case XK_KP_6: return Key.NumPad6; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
567 case XK_KP_7: return Key.NumPad7; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
568 case XK_KP_8: return Key.NumPad8; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
569 case XK_KP_9: return Key.NumPad9; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
570 case XK_KP_Divide: return Key.NumPadDivide; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
571 case XK_KP_Multiply: return Key.NumPadMultiply; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
572 case XK_KP_Subtract: return Key.NumPadSubtract; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
573 case XK_KP_Add: return Key.NumPadAdd; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
574 case XK_KP_Decimal: return Key.NumPadDecimal; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
575 |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
576 case XK_grave: |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
577 case XK_asciitilde: return Key.Backquote; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
578 case XK_minus: |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
579 case XK_underscore: return Key.Minus; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
580 case XK_equal: |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
581 case XK_plus: return Key.Equals; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
582 case XK_bracketleft: |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
583 case XK_braceleft: return Key.OpenBracket; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
584 case XK_bracketright: |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
585 case XK_braceright: return Key.CloseBracket; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
586 case XK_backslash: |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
587 case XK_bar: return Key.Backslash; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
588 case XK_semicolon: |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
589 case XK_colon: return Key.Semicolon; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
590 case XK_apostrophe: |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
591 case XK_quotedbl: return Key.Quote; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
592 case XK_comma: |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
593 case XK_less: return Key.Comma; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
594 case XK_period: |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
595 case XK_greater: return Key.Period; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
596 case XK_slash: |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
597 case XK_question: return Key.Slash; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
598 |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
599 //case XK_Menu: return Key.Menu; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
600 |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
601 case XK_Shift_L: |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
602 case XK_Shift_R: return Key.Shift; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
603 case XK_Control_L: |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
604 case XK_Control_R: return Key.Control; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
605 case XK_Alt_L: |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
606 case XK_Alt_R: return Key.Alt; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
607 |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
608 //case XK_: return Key.; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
609 default: |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
610 if(sym >= 0x41 && sym <= 0x5A) // Key.A - Key.Z |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
611 return cast(Key)sym; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
612 if(sym >= 0x61 && sym <= 0x7A) // Key.A - Key.Z |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
613 return cast(Key)(sym-32); |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
614 return cast(Key)0; |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
615 } |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
616 } |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
617 //}}} |
43a88caead16
Implement keyDown and keyUp events with X.
Jordan Miner <jminer7@gmail.com>
parents:
23
diff
changeset
|
618 |
0 | 619 public import tango.stdc.time; |
620 template WindowBackend() { | |
621 invariant { | |
622 //if(_handle == 0) | |
623 // return; | |
624 //XWindow root, parent; | |
625 //XWindow* children; | |
626 //uint numChildren; | |
627 //XQueryTree(display, _handle, | |
628 // &root, &parent, &children, &numChildren); | |
629 //XFree(children); | |
630 //int x, y; | |
631 //XWindow child; | |
632 //XTranslateCoordinates(display, _handle, root, 0, 0, &x, &y, &child); | |
633 //assert(_location.X == x-_borderSize.Left); | |
634 //assert(_location.Y == y-_borderSize.Top); | |
635 //XWindowAttributes attribs; | |
636 //XGetWindowAttributes(display, _handle, &attribs); | |
637 } | |
638 XWindow _handle; | |
639 bool mapped = false; | |
640 bool backend_handleCreated() { return _handle > 0; } | |
641 void backend_recreateHandle() { | |
642 auto primaryScreenNum = XDefaultScreen(display); | |
643 //XColor color; | |
644 //color.red = 65535*backColor.R/255; | |
645 //color.green = 65535*backColor.G/255; | |
646 //color.blue = 65535*backColor.B/255; | |
647 //if(XAllocColor(display, XDefaultColormap(display, primaryScreenNum), &color)) | |
648 // printf("XAllocColor() failed\n"); | |
649 | |
650 XSetWindowAttributes attribs; | |
651 attribs.bit_gravity = NorthWestGravity; | |
652 // TODO: should be backColor, and should change when backColor changes | |
653 // call XSetWindowBackground() for this | |
654 attribs.background_pixel = XWhitePixel(display, primaryScreenNum); | |
655 attribs.event_mask = | |
656 KeyPressMask | | |
657 KeyReleaseMask | | |
658 ButtonPressMask | | |
659 ButtonReleaseMask | | |
660 EnterWindowMask | | |
661 LeaveWindowMask | | |
662 PointerMotionMask | | |
663 ButtonMotionMask | | |
664 ExposureMask | | |
665 FocusChangeMask | | |
666 StructureNotifyMask | | |
667 PropertyChangeMask; | |
668 XWindow newHandle = XCreateWindow( | |
669 display, root, | |
670 cast(int)x, cast(int)y, | |
671 backend_insideWidth, backend_insideHeight, | |
672 0, CopyFromParent, InputOutput, null, | |
673 CWBitGravity | CWBackPixel | CWEventMask, &attribs); | |
674 | |
675 setControl(newHandle, this); | |
676 auto protocols = [XA.WM_DELETE_WINDOW]; | |
677 XSetWMProtocols(display, newHandle, protocols.ptr, protocols.length); | |
678 if(handleCreated) { | |
679 XDestroyWindow(display, _handle); | |
680 XSync(display, false); | |
681 } | |
682 _handle = newHandle; | |
683 text = _text; // move the text over to the new window | |
684 visible = _visible; | |
685 borderStyle = _borderStyle; | |
686 //backend_nativeToBorderSize(); | |
687 } | |
688 Graphics backend_quickCreateGraphics() { | |
689 auto surface = cairo_xlib_surface_create(display, _handle, | |
690 XDefaultVisual(display, XDefaultScreen(display)), | |
691 cast(int)width, cast(int)height); | |
692 auto cr = cairo_create(surface); | |
693 cairo_surface_destroy(surface); | |
694 cairo_translate(cr, -borderSize.left, -borderSize.top); | |
695 auto g = new Graphics(cr); | |
696 cairo_destroy(cr); | |
697 return g; | |
698 } | |
699 void backend_visible(bool b) { | |
700 if(b) | |
701 // if not created, create the handle by calling handle() | |
702 XMapWindow(display, handle); | |
703 else | |
704 XUnmapWindow(display, _handle); | |
705 } | |
706 void backend_borderStyle(WindowBorderStyle border) { | |
707 backend_update_NET_WM_WINDOW_TYPE(); | |
708 backend_update_MOTIF_WM_HINTS(); | |
709 backend_nativeToBorderSize(); | |
710 } | |
23
d55b5b998412
Implement built-in mouse cursors with X.
Jordan Miner <jminer7@gmail.com>
parents:
18
diff
changeset
|
711 void backend_setCurrentCursor(Cursor cur) { |
d55b5b998412
Implement built-in mouse cursors with X.
Jordan Miner <jminer7@gmail.com>
parents:
18
diff
changeset
|
712 XDefineCursor(display, _handle, cur.handle); |
d55b5b998412
Implement built-in mouse cursors with X.
Jordan Miner <jminer7@gmail.com>
parents:
18
diff
changeset
|
713 } |
d55b5b998412
Implement built-in mouse cursors with X.
Jordan Miner <jminer7@gmail.com>
parents:
18
diff
changeset
|
714 |
0 | 715 static _InvalidRect[] invalidRects; |
716 void backend_repaint(Rect rect) { | |
717 invalidRects.length = invalidRects.length+1; | |
718 invalidRects[$-1].window = _handle; | |
719 invalidRects[$-1].x = cast(int)(rect.x-borderSize.left); | |
720 invalidRects[$-1].y = cast(int)(rect.y-borderSize.top); | |
721 invalidRects[$-1].width = cast(int)rect.width+1; | |
722 invalidRects[$-1].height = cast(int)rect.height+1; | |
723 //printf("invalidating x=%.1f, y=%.1f, width=%.1f, height=%.1f\n", rect.X, rect.Y, rect.width, rect.height); | |
724 } | |
725 void backend_resizable(bool b) { | |
726 backend_updateWM_NORMAL_HINTS(); | |
727 } | |
728 void backend_contentMinSizeChanged() { | |
729 backend_updateWM_NORMAL_HINTS(); | |
730 } | |
731 void backend_contentMaxSizeChanged() { | |
732 backend_updateWM_NORMAL_HINTS(); | |
733 } | |
734 void backend_location(Point pt) { | |
735 backend_updateWM_NORMAL_HINTS(); | |
736 backend_locationSizeToNative(); | |
737 } | |
738 void backend_size(Size size) { | |
739 backend_updateWM_NORMAL_HINTS(); | |
740 backend_locationSizeToNative(); | |
741 } | |
742 void backend_text(string str) { | |
743 //auto tmp = str.ToCharPtr(); | |
744 //XTextProperty strProperty; | |
745 //if(!XStringListToTextProperty(&tmp, 1, &strProperty)) | |
746 //printf("XStringListToTextProperty() failed - out of memory\n"); | |
747 //XSetWMName(display, _handle, &strProperty); | |
748 XChangeProperty(display, _handle, XA._NET_WM_NAME, XA.UTF8_STRING, 8, PropModeReplace, str.ptr, str.length); | |
749 } | |
750 //{{{ backend specific | |
751 void backend_updateWM_NORMAL_HINTS() { | |
752 XSizeHints* sizeHints = XAllocSizeHints(); | |
753 scope(exit) XFree(sizeHints); | |
754 sizeHints.flags = PMinSize | PMaxSize | PPosition | PSize; | |
755 if(resizable) { | |
756 sizeHints.min_width = cast(int)content.minWidth; | |
757 sizeHints.min_height = cast(int)content.minHeight; | |
758 sizeHints.max_width = | |
759 content.maxWidth > 0 ? cast(int)content.maxWidth : 30_000; | |
760 sizeHints.max_height = | |
761 content.maxHeight > 0 ? cast(int)content.maxHeight : 30_000; | |
762 } else { | |
763 sizeHints.min_width = sizeHints.max_width = backend_insideWidth; | |
764 sizeHints.min_height = sizeHints.max_height = backend_insideHeight; | |
765 } | |
766 sizeHints.x = cast(int)x; | |
767 sizeHints.y = cast(int)y; | |
768 sizeHints.width = backend_insideWidth; | |
769 sizeHints.height = backend_insideHeight; | |
770 XSetWMNormalHints(display, _handle, sizeHints); | |
771 } | |
772 void backend_update_MOTIF_WM_HINTS() { | |
773 int[4] mwmHints; | |
774 mwmHints[0] = 1 << 1; // flags | |
775 mwmHints[2] = borderStyle == WindowBorderStyle.None ? 0 : 1; // decor | |
776 XChangeProperty(display, _handle, | |
777 XA._MOTIF_WM_HINTS, XA._MOTIF_WM_HINTS, 32, PropModeReplace, mwmHints.ptr, mwmHints.length); | |
778 } | |
779 void backend_update_NET_WM_WINDOW_TYPE() { | |
780 XAtom[] atoms; | |
781 // with Metacity, the decor is not being repainted from normal>dialog | |
782 // because they are the same size | |
783 if(borderStyle == WindowBorderStyle.Dialog) | |
784 atoms = [XA._NET_WM_WINDOW_TYPE_DIALOG]; | |
785 else if(borderStyle == WindowBorderStyle.Tool) | |
786 atoms = [XA._NET_WM_WINDOW_TYPE_UTILITY]; | |
787 else | |
788 atoms = [XA._NET_WM_WINDOW_TYPE_NORMAL]; | |
789 XChangeProperty(display, _handle, | |
790 XA._NET_WM_WINDOW_TYPE, XA.ATOM, 32, PropModeReplace, atoms.ptr, atoms.length); | |
791 } | |
792 // returns what width the x window should be...not including borders | |
793 int backend_insideWidth() { | |
794 return cast(int)(size.width-borderSize.left-borderSize.right); | |
795 } | |
796 // returns what height the x window should be...not including borders | |
797 int backend_insideHeight() { | |
798 return cast(int)(size.height-borderSize.top-borderSize.bottom); | |
799 } | |
800 // applies the currently set location and size to the native X Window | |
801 void backend_locationSizeToNative() { | |
802 Point pt = _location; | |
803 if(!isTopLevelReparented(_handle)) { | |
804 pt.x = pt.x + _borderSize.left; | |
805 pt.y = pt.y + _borderSize.top; | |
806 } | |
807 XMoveResizeWindow(display, _handle, cast(int)pt.x, cast(int)pt.y, | |
808 backend_insideWidth, backend_insideHeight); | |
809 // XMoveWindow: | |
810 // Under Metacity, sets the location of the WM's frame. | |
811 // Under Compiz, sets the location of the window. | |
812 // XResizeWindow: | |
813 // Under Metacity and Compiz, sets the size of the window not | |
814 // including the WM's frame. | |
815 } | |
816 // sets location and size based on where the native X Window is | |
817 void backend_nativeToLocationSize() { | |
818 XWindow root, parent; | |
819 XWindow* children; | |
820 uint numChildren; | |
821 XQueryTree(display, _handle, | |
822 &root, &parent, &children, &numChildren); | |
823 XFree(children); | |
824 | |
825 int x, y; | |
826 XWindow child; | |
827 XTranslateCoordinates(display, _handle, root, 0, 0, &x, &y, &child); | |
828 _location.x = x - _borderSize.left; | |
829 _location.y = y - _borderSize.top; | |
830 scope args = new EventArgs; | |
831 moved(args); | |
832 XWindowAttributes attribs; | |
833 XGetWindowAttributes(display, _handle, &attribs); | |
834 _size.width = attribs.width+_borderSize.left+_borderSize.right; | |
835 _size.height = attribs.height+_borderSize.top+_borderSize.bottom; | |
836 resized(args); | |
837 | |
838 //content._location = Point(_borderSize.Left, _borderSize.Top); | |
839 //content._size = Size(attribs.width, attribs.height); | |
840 //Stdout.format("location updated to {}", _location).newline; | |
841 //Stdout.format("size updated to {}", _size).newline; | |
842 } | |
843 void backend_nativeToBorderSize() { | |
844 _borderSize = backend_getBorderSize(); | |
845 //Stdout("borderSize updated to ", _borderSize).newline; | |
846 backend_nativeToLocationSize(); | |
847 } | |
848 BorderSize backend_getBorderSize() { | |
849 if(!isWMPropertySupported(XA._NET_FRAME_EXTENTS) || | |
850 borderStyle == WindowBorderStyle.None) | |
851 return BorderSize(); | |
852 // create handle if necessary | |
853 auto reqHandle = handle; | |
854 bool requested = false; | |
855 | |
856 //{{{ requestExtents() | |
857 void requestExtents() { | |
858 if(isWMPropertySupported(XA._NET_REQUEST_FRAME_EXTENTS)) { | |
859 XEvent ev; | |
860 ev.xclient.type = ClientMessage; | |
861 ev.xclient.window = handle; | |
862 ev.xclient.message_type = XA._NET_REQUEST_FRAME_EXTENTS; | |
863 ev.xclient.format = 8; | |
864 XSendEvent(display, root, false, | |
865 SubstructureNotifyMask | SubstructureRedirectMask, &ev); | |
866 } else { // compiz and beryl do not yet support requesting | |
867 XSetWindowAttributes attribs; | |
868 reqHandle = XCreateWindow(display, | |
869 root, 0, 0, 1, 1, | |
870 0, CopyFromParent, InputOnly, null, 0, &attribs); | |
871 | |
872 XWMHints* hints = XAllocWMHints(); | |
873 scope(exit) XFree(hints); | |
874 hints.flags = InputHint; | |
875 hints.input = false; | |
876 XSetWMHints(display, reqHandle, hints); | |
877 | |
878 auto mainHandle = _handle; | |
879 _handle = reqHandle; | |
880 backend_updateWM_NORMAL_HINTS(); | |
881 backend_update_NET_WM_WINDOW_TYPE(); | |
882 backend_update_MOTIF_WM_HINTS(); | |
883 backend_visible = true; | |
884 backend_visible = false; | |
885 _handle = mainHandle; | |
886 } | |
887 requested = true; | |
888 } | |
889 //}}} | |
890 | |
891 if(!mapped) | |
892 requestExtents(); | |
893 int* extents; | |
894 while(true) { | |
895 XSync(display, false); | |
896 extents = cast(int*)getXWindowProperty(display, reqHandle, | |
897 XA._NET_FRAME_EXTENTS); | |
898 if(extents !is null) | |
899 break; | |
900 if(!requested) | |
901 requestExtents(); | |
902 } | |
903 scope(exit) XFree(extents); | |
904 if(reqHandle != _handle) | |
905 XDestroyWindow(display, reqHandle); | |
906 return BorderSize(extents[0], extents[2], extents[1], extents[3]); | |
907 } | |
908 //}}} | |
909 } | |
910 |