|
@@ -0,0 +1,381 @@
|
|
|
|
|
+/*
|
|
|
|
|
+Copyright (c) 2012 Daniele Bartolini, Simone Boscaratto
|
|
|
|
|
+
|
|
|
|
|
+Permission is hereby granted, free of charge, to any person
|
|
|
|
|
+obtaining a copy of this software and associated documentation
|
|
|
|
|
+files (the "Software"), to deal in the Software without
|
|
|
|
|
+restriction, including without limitation the rights to use,
|
|
|
|
|
+copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
|
+copies of the Software, and to permit persons to whom the
|
|
|
|
|
+Software is furnished to do so, subject to the following
|
|
|
|
|
+conditions:
|
|
|
|
|
+
|
|
|
|
|
+The above copyright notice and this permission notice shall be
|
|
|
|
|
+included in all copies or substantial portions of the Software.
|
|
|
|
|
+
|
|
|
|
|
+THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
|
|
|
+EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES
|
|
|
|
|
+OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
|
|
|
|
+NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
|
|
|
|
|
+HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
|
|
|
|
|
+WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
|
|
|
|
+FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
|
|
|
|
|
+OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
|
+*/
|
|
|
|
|
+
|
|
|
|
|
+#include "OsWindow.h"
|
|
|
|
|
+#include "Assert.h"
|
|
|
|
|
+#include "Keyboard.h"
|
|
|
|
|
+#include "OS.h"
|
|
|
|
|
+#include "GLXContext.h"
|
|
|
|
|
+#include "String.h"
|
|
|
|
|
+
|
|
|
|
|
+namespace crown
|
|
|
|
|
+{
|
|
|
|
|
+
|
|
|
|
|
+//-----------------------------------------------------------------------------
|
|
|
|
|
+static Key x11_translate_key(int32_t x11_key)
|
|
|
|
|
+{
|
|
|
|
|
+ if ((x11_key > 0x40 && x11_key < 0x5B) || (x11_key > 0x60 && x11_key < 0x7B) || (x11_key > 0x2F && x11_key < 0x3A))
|
|
|
|
|
+ {
|
|
|
|
|
+ return (Key)x11_key;
|
|
|
|
|
+ }
|
|
|
|
|
+
|
|
|
|
|
+ switch (x11_key)
|
|
|
|
|
+ {
|
|
|
|
|
+ case XK_BackSpace: return KC_BACKSPACE;
|
|
|
|
|
+ case XK_Tab: return KC_TAB;
|
|
|
|
|
+ case XK_space: return KC_SPACE;
|
|
|
|
|
+ case XK_Escape: return KC_ESCAPE;
|
|
|
|
|
+ case XK_Return: return KC_ENTER;
|
|
|
|
|
+ case XK_F1: return KC_F1;
|
|
|
|
|
+ case XK_F2: return KC_F2;
|
|
|
|
|
+ case XK_F3: return KC_F3;
|
|
|
|
|
+ case XK_F4: return KC_F4;
|
|
|
|
|
+ case XK_F5: return KC_F5;
|
|
|
|
|
+ case XK_F6: return KC_F6;
|
|
|
|
|
+ case XK_F7: return KC_F7;
|
|
|
|
|
+ case XK_F8: return KC_F8;
|
|
|
|
|
+ case XK_F9: return KC_F9;
|
|
|
|
|
+ case XK_F10: return KC_F10;
|
|
|
|
|
+ case XK_F11: return KC_F11;
|
|
|
|
|
+ case XK_F12: return KC_F12;
|
|
|
|
|
+ case XK_Home: return KC_HOME;
|
|
|
|
|
+ case XK_Left: return KC_LEFT;
|
|
|
|
|
+ case XK_Up: return KC_UP;
|
|
|
|
|
+ case XK_Right: return KC_RIGHT;
|
|
|
|
|
+ case XK_Down: return KC_DOWN;
|
|
|
|
|
+ case XK_Page_Up: return KC_PAGE_UP;
|
|
|
|
|
+ case XK_Page_Down: return KC_PAGE_DOWN;
|
|
|
|
|
+ case XK_Shift_L: return KC_LSHIFT;
|
|
|
|
|
+ case XK_Shift_R: return KC_RSHIFT;
|
|
|
|
|
+ case XK_Control_L: return KC_LCONTROL;
|
|
|
|
|
+ case XK_Control_R: return KC_RCONTROL;
|
|
|
|
|
+ case XK_Caps_Lock: return KC_CAPS_LOCK;
|
|
|
|
|
+ case XK_Alt_L: return KC_LALT;
|
|
|
|
|
+ case XK_Alt_R: return KC_RALT;
|
|
|
|
|
+ case XK_Super_L: return KC_LSUPER;
|
|
|
|
|
+ case XK_Super_R: return KC_RSUPER;
|
|
|
|
|
+ case XK_KP_0: return KC_KP_0;
|
|
|
|
|
+ case XK_KP_1: return KC_KP_1;
|
|
|
|
|
+ case XK_KP_2: return KC_KP_2;
|
|
|
|
|
+ case XK_KP_3: return KC_KP_3;
|
|
|
|
|
+ case XK_KP_4: return KC_KP_4;
|
|
|
|
|
+ case XK_KP_5: return KC_KP_5;
|
|
|
|
|
+ case XK_KP_6: return KC_KP_6;
|
|
|
|
|
+ case XK_KP_7: return KC_KP_7;
|
|
|
|
|
+ case XK_KP_8: return KC_KP_8;
|
|
|
|
|
+ case XK_KP_9: return KC_KP_9;
|
|
|
|
|
+ default: return KC_NOKEY;
|
|
|
|
|
+ }
|
|
|
|
|
+}
|
|
|
|
|
+
|
|
|
|
|
+//-----------------------------------------------------------------------------
|
|
|
|
|
+OsWindow::OsWindow(uint32_t width, uint32_t height) :
|
|
|
|
|
+ m_x11_display(NULL),
|
|
|
|
|
+ m_x11_window(None),
|
|
|
|
|
+ m_x(0),
|
|
|
|
|
+ m_y(0),
|
|
|
|
|
+ m_width(width),
|
|
|
|
|
+ m_height(height),
|
|
|
|
|
+ m_x11_detectable_autorepeat(false),
|
|
|
|
|
+ m_x11_hidden_cursor(None)
|
|
|
|
|
+{
|
|
|
|
|
+ ce_assert(width != 0 || height != 0, "Width and height must differ from zero");
|
|
|
|
|
+
|
|
|
|
|
+ m_x11_display = XOpenDisplay(NULL);
|
|
|
|
|
+
|
|
|
|
|
+ ce_assert(m_x11_display != NULL, "Unable to open X11 display");
|
|
|
|
|
+
|
|
|
|
|
+ int screen = DefaultScreen(m_x11_display);
|
|
|
|
|
+ Window root_window = RootWindow(m_x11_display, screen);
|
|
|
|
|
+ int depth = DefaultDepth(m_x11_display, screen);
|
|
|
|
|
+ Visual* visual = DefaultVisual(m_x11_display, screen);
|
|
|
|
|
+
|
|
|
|
|
+ // We want to track keyboard and mouse events
|
|
|
|
|
+ XSetWindowAttributes win_attribs;
|
|
|
|
|
+ win_attribs.background_pixmap = 0;
|
|
|
|
|
+ win_attribs.border_pixel = 0;
|
|
|
|
|
+ win_attribs.event_mask = FocusChangeMask | StructureNotifyMask | KeyPressMask |
|
|
|
|
|
+ KeyReleaseMask | ButtonPressMask | ButtonReleaseMask | PointerMotionMask;
|
|
|
|
|
+
|
|
|
|
|
+ m_x11_window = XCreateWindow(
|
|
|
|
|
+ m_x11_display,
|
|
|
|
|
+ root_window,
|
|
|
|
|
+ 0, 0,
|
|
|
|
|
+ width, height,
|
|
|
|
|
+ 0,
|
|
|
|
|
+ depth,
|
|
|
|
|
+ InputOutput,
|
|
|
|
|
+ visual,
|
|
|
|
|
+ CWBorderPixel | CWEventMask,
|
|
|
|
|
+ &win_attribs
|
|
|
|
|
+ );
|
|
|
|
|
+
|
|
|
|
|
+ ce_assert(m_x11_window != None, "Unable to create X window");
|
|
|
|
|
+
|
|
|
|
|
+ // Check presence of detectable autorepeat
|
|
|
|
|
+ Bool detectable;
|
|
|
|
|
+ m_x11_detectable_autorepeat = (bool) XkbSetDetectableAutoRepeat(m_x11_display, true, &detectable);
|
|
|
|
|
+
|
|
|
|
|
+ // Build hidden cursor
|
|
|
|
|
+ Pixmap bm_no;
|
|
|
|
|
+ XColor black, dummy;
|
|
|
|
|
+ Colormap colormap;
|
|
|
|
|
+ static char no_data[] = { 0, 0, 0, 0, 0, 0, 0, 0 };
|
|
|
|
|
+
|
|
|
|
|
+ colormap = XDefaultColormap(m_x11_display, screen);
|
|
|
|
|
+ XAllocNamedColor(m_x11_display, colormap, "black", &black, &dummy);
|
|
|
|
|
+ bm_no = XCreateBitmapFromData(m_x11_display, m_x11_window, no_data, 8, 8);
|
|
|
|
|
+ m_x11_hidden_cursor = XCreatePixmapCursor(m_x11_display, bm_no, bm_no, &black, &black, 0, 0);
|
|
|
|
|
+
|
|
|
|
|
+ set_x11_display_and_window(m_x11_display, m_x11_window);
|
|
|
|
|
+}
|
|
|
|
|
+
|
|
|
|
|
+//-----------------------------------------------------------------------------
|
|
|
|
|
+OsWindow::~OsWindow()
|
|
|
|
|
+{
|
|
|
|
|
+ if (m_x11_display)
|
|
|
|
|
+ {
|
|
|
|
|
+ if (m_x11_window != None)
|
|
|
|
|
+ {
|
|
|
|
|
+ XDestroyWindow(m_x11_display, m_x11_window);
|
|
|
|
|
+ }
|
|
|
|
|
+
|
|
|
|
|
+ XCloseDisplay(m_x11_display);
|
|
|
|
|
+ }
|
|
|
|
|
+}
|
|
|
|
|
+
|
|
|
|
|
+//-----------------------------------------------------------------------------
|
|
|
|
|
+void OsWindow::show()
|
|
|
|
|
+{
|
|
|
|
|
+ XMapRaised(m_x11_display, m_x11_window);
|
|
|
|
|
+}
|
|
|
|
|
+
|
|
|
|
|
+//-----------------------------------------------------------------------------
|
|
|
|
|
+void OsWindow::hide()
|
|
|
|
|
+{
|
|
|
|
|
+ XUnmapWindow(m_x11_display, m_x11_window);
|
|
|
|
|
+}
|
|
|
|
|
+
|
|
|
|
|
+//-----------------------------------------------------------------------------
|
|
|
|
|
+void OsWindow::get_size(uint32_t& width, uint32_t& height)
|
|
|
|
|
+{
|
|
|
|
|
+ width = m_width;
|
|
|
|
|
+ height = m_height;
|
|
|
|
|
+}
|
|
|
|
|
+
|
|
|
|
|
+//-----------------------------------------------------------------------------
|
|
|
|
|
+void OsWindow::get_position(uint32_t& x, uint32_t& y)
|
|
|
|
|
+{
|
|
|
|
|
+ x = m_x;
|
|
|
|
|
+ y = m_y;
|
|
|
|
|
+}
|
|
|
|
|
+
|
|
|
|
|
+//-----------------------------------------------------------------------------
|
|
|
|
|
+void OsWindow::resize(uint32_t width, uint32_t height)
|
|
|
|
|
+{
|
|
|
|
|
+ XResizeWindow(m_x11_display, m_x11_window, width, height);
|
|
|
|
|
+}
|
|
|
|
|
+
|
|
|
|
|
+//-----------------------------------------------------------------------------
|
|
|
|
|
+void OsWindow::move(uint32_t x, uint32_t y)
|
|
|
|
|
+{
|
|
|
|
|
+ XMoveWindow(m_x11_display, m_x11_window, x, y);
|
|
|
|
|
+}
|
|
|
|
|
+
|
|
|
|
|
+//-----------------------------------------------------------------------------
|
|
|
|
|
+void OsWindow::show_cursor()
|
|
|
|
|
+{
|
|
|
|
|
+ XDefineCursor(m_x11_display, m_x11_window, None);
|
|
|
|
|
+}
|
|
|
|
|
+
|
|
|
|
|
+//-----------------------------------------------------------------------------
|
|
|
|
|
+void OsWindow::hide_cursor()
|
|
|
|
|
+{
|
|
|
|
|
+ XDefineCursor(m_x11_display, m_x11_window, m_x11_hidden_cursor);
|
|
|
|
|
+}
|
|
|
|
|
+
|
|
|
|
|
+//-----------------------------------------------------------------------------
|
|
|
|
|
+void OsWindow::get_cursor_xy(int32_t& x, int32_t& y)
|
|
|
|
|
+{
|
|
|
|
|
+ Window unused;
|
|
|
|
|
+ int32_t pointer_x, pointer_y, dummy;
|
|
|
|
|
+ uint32_t dummy2;
|
|
|
|
|
+
|
|
|
|
|
+ XQueryPointer(m_x11_display, m_x11_window, &unused, &unused, &dummy, &dummy, &pointer_x, &pointer_y, &dummy2);
|
|
|
|
|
+
|
|
|
|
|
+ x = pointer_x;
|
|
|
|
|
+ y = pointer_y;
|
|
|
|
|
+}
|
|
|
|
|
+
|
|
|
|
|
+//-----------------------------------------------------------------------------
|
|
|
|
|
+void OsWindow::set_cursor_xy(int32_t x, int32_t y)
|
|
|
|
|
+{
|
|
|
|
|
+ XWarpPointer(m_x11_display, None, m_x11_window, 0, 0, m_width, m_height, x, y);
|
|
|
|
|
+
|
|
|
|
|
+ XFlush(m_x11_display);
|
|
|
|
|
+}
|
|
|
|
|
+
|
|
|
|
|
+//-----------------------------------------------------------------------------
|
|
|
|
|
+char* OsWindow::title()
|
|
|
|
|
+{
|
|
|
|
|
+ static char title[1024];
|
|
|
|
|
+
|
|
|
|
|
+ char* tmp_title;
|
|
|
|
|
+ XFetchName(m_x11_display, m_x11_window, &tmp_title);
|
|
|
|
|
+
|
|
|
|
|
+ string::strncpy(title, tmp_title, 1024);
|
|
|
|
|
+ XFree(tmp_title);
|
|
|
|
|
+
|
|
|
|
|
+ return title;
|
|
|
|
|
+}
|
|
|
|
|
+
|
|
|
|
|
+//-----------------------------------------------------------------------------
|
|
|
|
|
+void OsWindow::set_title(const char* title)
|
|
|
|
|
+{
|
|
|
|
|
+ XStoreName(m_x11_display, m_x11_window, title);
|
|
|
|
|
+}
|
|
|
|
|
+
|
|
|
|
|
+//-----------------------------------------------------------------------------
|
|
|
|
|
+void OsWindow::frame()
|
|
|
|
|
+{
|
|
|
|
|
+ XEvent event;
|
|
|
|
|
+
|
|
|
|
|
+ OsEventParameter data_button[4] = {0, 0, 0, 0};
|
|
|
|
|
+ OsEventParameter data_key[4] = {0, 0, 0, 0};
|
|
|
|
|
+
|
|
|
|
|
+ while (XPending(m_x11_display))
|
|
|
|
|
+ {
|
|
|
|
|
+ XNextEvent(m_x11_display, &event);
|
|
|
|
|
+
|
|
|
|
|
+ switch (event.type)
|
|
|
|
|
+ {
|
|
|
|
|
+ case ConfigureNotify:
|
|
|
|
|
+ {
|
|
|
|
|
+ m_x = event.xconfigure.x;
|
|
|
|
|
+ m_y = event.xconfigure.y;
|
|
|
|
|
+ m_width = event.xconfigure.width;
|
|
|
|
|
+ m_height = event.xconfigure.height;
|
|
|
|
|
+ break;
|
|
|
|
|
+ }
|
|
|
|
|
+ case ButtonPress:
|
|
|
|
|
+ case ButtonRelease:
|
|
|
|
|
+ {
|
|
|
|
|
+ OsEventType oset_type = event.type == ButtonPress ? OSET_BUTTON_PRESS : OSET_BUTTON_RELEASE;
|
|
|
|
|
+
|
|
|
|
|
+ data_button[0].int_value = event.xbutton.x;
|
|
|
|
|
+ data_button[1].int_value = event.xbutton.y;
|
|
|
|
|
+
|
|
|
|
|
+ switch (event.xbutton.button)
|
|
|
|
|
+ {
|
|
|
|
|
+ case Button1:
|
|
|
|
|
+ {
|
|
|
|
|
+ data_button[2].int_value = 0;
|
|
|
|
|
+ push_event(oset_type, data_button[0], data_button[1], data_button[2], data_button[3]);
|
|
|
|
|
+ break;
|
|
|
|
|
+ }
|
|
|
|
|
+ case Button2:
|
|
|
|
|
+ {
|
|
|
|
|
+ data_button[2].int_value = 1;
|
|
|
|
|
+ push_event(oset_type, data_button[0], data_button[1], data_button[2], data_button[3]);
|
|
|
|
|
+ break;
|
|
|
|
|
+ }
|
|
|
|
|
+ case Button3:
|
|
|
|
|
+ {
|
|
|
|
|
+ data_button[2].int_value = 2;
|
|
|
|
|
+ push_event(oset_type, data_button[0], data_button[1], data_button[2], data_button[3]);
|
|
|
|
|
+ break;
|
|
|
|
|
+ }
|
|
|
|
|
+ }
|
|
|
|
|
+
|
|
|
|
|
+ break;
|
|
|
|
|
+ }
|
|
|
|
|
+ case MotionNotify:
|
|
|
|
|
+ {
|
|
|
|
|
+ push_event(OSET_MOTION_NOTIFY, data_button[0], data_button[1], data_button[2], data_button[3]);
|
|
|
|
|
+ break;
|
|
|
|
|
+ }
|
|
|
|
|
+ case KeyPress:
|
|
|
|
|
+ case KeyRelease:
|
|
|
|
|
+ {
|
|
|
|
|
+ char string[4] = {0, 0, 0, 0};
|
|
|
|
|
+ int32_t len = -1;
|
|
|
|
|
+ KeySym key;
|
|
|
|
|
+
|
|
|
|
|
+ len = XLookupString(&event.xkey, string, 4, &key, NULL);
|
|
|
|
|
+
|
|
|
|
|
+ Key kc = x11_translate_key(key);
|
|
|
|
|
+
|
|
|
|
|
+ // Check if any modifier key is pressed or released
|
|
|
|
|
+ int32_t modifier_mask = 0;
|
|
|
|
|
+
|
|
|
|
|
+ if (kc == KC_LSHIFT || kc == KC_RSHIFT)
|
|
|
|
|
+ {
|
|
|
|
|
+ (event.type == KeyPress) ? modifier_mask |= MK_SHIFT : modifier_mask &= ~MK_SHIFT;
|
|
|
|
|
+ }
|
|
|
|
|
+ else if (kc == KC_LCONTROL || kc == KC_RCONTROL)
|
|
|
|
|
+ {
|
|
|
|
|
+ (event.type == KeyPress) ? modifier_mask |= MK_CTRL : modifier_mask &= ~MK_CTRL;
|
|
|
|
|
+ }
|
|
|
|
|
+ else if (kc == KC_LALT || kc == KC_RALT)
|
|
|
|
|
+ {
|
|
|
|
|
+ (event.type == KeyPress) ? modifier_mask |= MK_ALT : modifier_mask &= ~MK_ALT;
|
|
|
|
|
+ }
|
|
|
|
|
+
|
|
|
|
|
+ OsEventType oset_type = event.type == KeyPress ? OSET_KEY_PRESS : OSET_KEY_RELEASE;
|
|
|
|
|
+
|
|
|
|
|
+ data_key[0].int_value = ((int32_t)kc);
|
|
|
|
|
+ data_key[1].int_value = modifier_mask;
|
|
|
|
|
+
|
|
|
|
|
+ push_event(oset_type, data_key[0], data_key[1], data_key[2], data_key[3]);
|
|
|
|
|
+
|
|
|
|
|
+// // Text input part
|
|
|
|
|
+// if (event.type == KeyPress && len > 0)
|
|
|
|
|
+// {
|
|
|
|
|
+// //crownEvent.event_type = ET_TEXT;
|
|
|
|
|
+// //crownEvent.text.type = TET_TEXT_INPUT;
|
|
|
|
|
+// strncpy(keyboardEvent.text, string, 4);
|
|
|
|
|
+
|
|
|
|
|
+// if (mListener)
|
|
|
|
|
+// {
|
|
|
|
|
+// mListener->TextInput(keyboardEvent);
|
|
|
|
|
+// }
|
|
|
|
|
+// }
|
|
|
|
|
+
|
|
|
|
|
+ break;
|
|
|
|
|
+ }
|
|
|
|
|
+ case KeymapNotify:
|
|
|
|
|
+ {
|
|
|
|
|
+ XRefreshKeyboardMapping(&event.xmapping);
|
|
|
|
|
+ break;
|
|
|
|
|
+ }
|
|
|
|
|
+ default:
|
|
|
|
|
+ {
|
|
|
|
|
+ break;
|
|
|
|
|
+ }
|
|
|
|
|
+ }
|
|
|
|
|
+ }
|
|
|
|
|
+}
|
|
|
|
|
+
|
|
|
|
|
+} // namespace crown
|