2014-11-06 08:26:16 +00:00
|
|
|
//========================================================================
|
|
|
|
// GLFW 3.1 Mir - www.glfw.org
|
|
|
|
//------------------------------------------------------------------------
|
|
|
|
// Copyright (c) 2014 Brandon Schaefer <brandon.schaefer@canonical.com>
|
|
|
|
//
|
|
|
|
// This software is provided 'as-is', without any express or implied
|
|
|
|
// warranty. In no event will the authors be held liable for any damages
|
|
|
|
// arising from the use of this software.
|
|
|
|
//
|
|
|
|
// Permission is granted to anyone to use this software for any purpose,
|
|
|
|
// including commercial applications, and to alter it and redistribute it
|
|
|
|
// freely, subject to the following restrictions:
|
|
|
|
//
|
|
|
|
// 1. The origin of this software must not be misrepresented; you must not
|
|
|
|
// claim that you wrote the original software. If you use this software
|
|
|
|
// in a product, an acknowledgment in the product documentation would
|
|
|
|
// be appreciated but is not required.
|
|
|
|
//
|
|
|
|
// 2. Altered source versions must be plainly marked as such, and must not
|
|
|
|
// be misrepresented as being the original software.
|
|
|
|
//
|
|
|
|
// 3. This notice may not be removed or altered from any source
|
|
|
|
// distribution.
|
|
|
|
//
|
|
|
|
//========================================================================
|
|
|
|
|
2014-11-06 08:21:12 +00:00
|
|
|
#include "internal.h"
|
2014-11-06 08:24:49 +00:00
|
|
|
#include "xkb_unicode.h"
|
|
|
|
|
|
|
|
#include <linux/input.h>
|
2014-11-10 17:42:41 +00:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
2014-11-06 08:24:49 +00:00
|
|
|
|
2014-11-06 22:52:38 +00:00
|
|
|
|
2014-11-10 17:42:41 +00:00
|
|
|
typedef struct EventNode
|
|
|
|
{
|
|
|
|
TAILQ_ENTRY(EventNode) entries;
|
|
|
|
MirEvent* event;
|
|
|
|
_GLFWwindow* window;
|
|
|
|
} EventNode;
|
|
|
|
|
|
|
|
static void deleteNode(EventQueue* queue, EventNode* node)
|
|
|
|
{
|
|
|
|
free(node->event);
|
|
|
|
free(node);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int emptyEventQueue(EventQueue* queue)
|
|
|
|
{
|
|
|
|
return queue->head.tqh_first == NULL ? GL_TRUE : GL_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static EventNode* newEventNode(MirEvent const* event, _GLFWwindow* context)
|
|
|
|
{
|
|
|
|
EventNode* new_node = calloc(1, sizeof(EventNode));
|
|
|
|
new_node->event = calloc(1, sizeof(MirEvent));
|
|
|
|
new_node->window = context;
|
|
|
|
|
|
|
|
memcpy(new_node->event, event, sizeof(MirEvent));
|
|
|
|
return new_node;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void enqueueEvent(MirEvent const* event, _GLFWwindow* context)
|
|
|
|
{
|
|
|
|
pthread_mutex_lock(&_glfw.mir.event_mutex);
|
|
|
|
|
|
|
|
EventNode* new_node = newEventNode(event, context);
|
|
|
|
TAILQ_INSERT_TAIL(&_glfw.mir.event_queue->head, new_node, entries);
|
|
|
|
|
|
|
|
pthread_cond_signal(&_glfw.mir.event_cond);
|
|
|
|
|
|
|
|
pthread_mutex_unlock(&_glfw.mir.event_mutex);
|
|
|
|
}
|
|
|
|
|
|
|
|
static EventNode* dequeueEvent(EventQueue* queue)
|
|
|
|
{
|
|
|
|
EventNode* node = NULL;
|
|
|
|
|
|
|
|
pthread_mutex_lock(&_glfw.mir.event_mutex);
|
|
|
|
|
|
|
|
node = queue->head.tqh_first;
|
|
|
|
|
|
|
|
if (node)
|
|
|
|
TAILQ_REMOVE(&queue->head, node, entries);
|
|
|
|
|
|
|
|
pthread_mutex_unlock(&_glfw.mir.event_mutex);
|
|
|
|
|
|
|
|
return node;
|
|
|
|
}
|
|
|
|
|
2014-11-06 22:52:38 +00:00
|
|
|
static MirPixelFormat findValidPixelFormat(void)
|
2014-11-06 08:23:02 +00:00
|
|
|
{
|
2014-11-06 22:52:38 +00:00
|
|
|
unsigned int i, validFormats, size = 32;
|
|
|
|
MirPixelFormat formats[size];
|
2014-11-06 08:23:02 +00:00
|
|
|
|
|
|
|
mir_connection_get_available_surface_formats(_glfw.mir.connection, formats,
|
2014-11-06 22:52:38 +00:00
|
|
|
size, &validFormats);
|
2014-11-06 08:23:02 +00:00
|
|
|
|
2014-11-06 22:52:38 +00:00
|
|
|
for (i = 0; i < validFormats; i++)
|
2014-11-06 08:23:02 +00:00
|
|
|
{
|
2014-11-06 22:52:38 +00:00
|
|
|
if (formats[i] == mir_pixel_format_abgr_8888 ||
|
|
|
|
formats[i] == mir_pixel_format_xbgr_8888 ||
|
|
|
|
formats[i] == mir_pixel_format_argb_8888 ||
|
|
|
|
formats[i] == mir_pixel_format_xrgb_8888)
|
2014-11-06 08:23:02 +00:00
|
|
|
{
|
2014-11-06 22:52:38 +00:00
|
|
|
return formats[i];
|
2014-11-06 08:23:02 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return mir_pixel_format_invalid;
|
|
|
|
}
|
|
|
|
|
2014-11-06 22:52:38 +00:00
|
|
|
static int mirModToGLFWMod(uint32_t mods)
|
2014-11-06 08:27:54 +00:00
|
|
|
{
|
2014-11-06 22:52:38 +00:00
|
|
|
int publicMods = 0x0;
|
2014-11-06 08:27:54 +00:00
|
|
|
|
2014-11-06 22:52:38 +00:00
|
|
|
if (mods & mir_key_modifier_alt)
|
|
|
|
publicMods |= GLFW_MOD_ALT;
|
|
|
|
else if (mods & mir_key_modifier_shift)
|
|
|
|
publicMods |= GLFW_MOD_SHIFT;
|
|
|
|
else if (mods & mir_key_modifier_ctrl)
|
|
|
|
publicMods |= GLFW_MOD_CONTROL;
|
|
|
|
else if (mods & mir_key_modifier_meta)
|
|
|
|
publicMods |= GLFW_MOD_SUPER;
|
2014-11-06 08:27:54 +00:00
|
|
|
|
2014-11-06 22:52:38 +00:00
|
|
|
return publicMods;
|
2014-11-06 08:27:54 +00:00
|
|
|
}
|
|
|
|
|
2014-11-06 08:24:49 +00:00
|
|
|
// Taken from wl_init.c
|
|
|
|
static int toGLFWKeyCode(uint32_t key)
|
|
|
|
{
|
|
|
|
switch (key)
|
|
|
|
{
|
|
|
|
case KEY_GRAVE: return GLFW_KEY_GRAVE_ACCENT;
|
|
|
|
case KEY_1: return GLFW_KEY_1;
|
|
|
|
case KEY_2: return GLFW_KEY_2;
|
|
|
|
case KEY_3: return GLFW_KEY_3;
|
|
|
|
case KEY_4: return GLFW_KEY_4;
|
|
|
|
case KEY_5: return GLFW_KEY_5;
|
|
|
|
case KEY_6: return GLFW_KEY_6;
|
|
|
|
case KEY_7: return GLFW_KEY_7;
|
|
|
|
case KEY_8: return GLFW_KEY_8;
|
|
|
|
case KEY_9: return GLFW_KEY_9;
|
|
|
|
case KEY_0: return GLFW_KEY_0;
|
|
|
|
case KEY_MINUS: return GLFW_KEY_MINUS;
|
|
|
|
case KEY_EQUAL: return GLFW_KEY_EQUAL;
|
|
|
|
case KEY_Q: return GLFW_KEY_Q;
|
|
|
|
case KEY_W: return GLFW_KEY_W;
|
|
|
|
case KEY_E: return GLFW_KEY_E;
|
|
|
|
case KEY_R: return GLFW_KEY_R;
|
|
|
|
case KEY_T: return GLFW_KEY_T;
|
|
|
|
case KEY_Y: return GLFW_KEY_Y;
|
|
|
|
case KEY_U: return GLFW_KEY_U;
|
|
|
|
case KEY_I: return GLFW_KEY_I;
|
|
|
|
case KEY_O: return GLFW_KEY_O;
|
|
|
|
case KEY_P: return GLFW_KEY_P;
|
|
|
|
case KEY_LEFTBRACE: return GLFW_KEY_LEFT_BRACKET;
|
|
|
|
case KEY_RIGHTBRACE: return GLFW_KEY_RIGHT_BRACKET;
|
|
|
|
case KEY_A: return GLFW_KEY_A;
|
|
|
|
case KEY_S: return GLFW_KEY_S;
|
|
|
|
case KEY_D: return GLFW_KEY_D;
|
|
|
|
case KEY_F: return GLFW_KEY_F;
|
|
|
|
case KEY_G: return GLFW_KEY_G;
|
|
|
|
case KEY_H: return GLFW_KEY_H;
|
|
|
|
case KEY_J: return GLFW_KEY_J;
|
|
|
|
case KEY_K: return GLFW_KEY_K;
|
|
|
|
case KEY_L: return GLFW_KEY_L;
|
|
|
|
case KEY_SEMICOLON: return GLFW_KEY_SEMICOLON;
|
|
|
|
case KEY_APOSTROPHE: return GLFW_KEY_APOSTROPHE;
|
|
|
|
case KEY_Z: return GLFW_KEY_Z;
|
|
|
|
case KEY_X: return GLFW_KEY_X;
|
|
|
|
case KEY_C: return GLFW_KEY_C;
|
|
|
|
case KEY_V: return GLFW_KEY_V;
|
|
|
|
case KEY_B: return GLFW_KEY_B;
|
|
|
|
case KEY_N: return GLFW_KEY_N;
|
|
|
|
case KEY_M: return GLFW_KEY_M;
|
|
|
|
case KEY_COMMA: return GLFW_KEY_COMMA;
|
|
|
|
case KEY_DOT: return GLFW_KEY_PERIOD;
|
|
|
|
case KEY_SLASH: return GLFW_KEY_SLASH;
|
|
|
|
case KEY_BACKSLASH: return GLFW_KEY_BACKSLASH;
|
|
|
|
case KEY_ESC: return GLFW_KEY_ESCAPE;
|
|
|
|
case KEY_TAB: return GLFW_KEY_TAB;
|
|
|
|
case KEY_LEFTSHIFT: return GLFW_KEY_LEFT_SHIFT;
|
|
|
|
case KEY_RIGHTSHIFT: return GLFW_KEY_RIGHT_SHIFT;
|
|
|
|
case KEY_LEFTCTRL: return GLFW_KEY_LEFT_CONTROL;
|
|
|
|
case KEY_RIGHTCTRL: return GLFW_KEY_RIGHT_CONTROL;
|
|
|
|
case KEY_LEFTALT: return GLFW_KEY_LEFT_ALT;
|
|
|
|
case KEY_RIGHTALT: return GLFW_KEY_RIGHT_ALT;
|
|
|
|
case KEY_LEFTMETA: return GLFW_KEY_LEFT_SUPER;
|
|
|
|
case KEY_RIGHTMETA: return GLFW_KEY_RIGHT_SUPER;
|
|
|
|
case KEY_MENU: return GLFW_KEY_MENU;
|
|
|
|
case KEY_NUMLOCK: return GLFW_KEY_NUM_LOCK;
|
|
|
|
case KEY_CAPSLOCK: return GLFW_KEY_CAPS_LOCK;
|
|
|
|
case KEY_PRINT: return GLFW_KEY_PRINT_SCREEN;
|
|
|
|
case KEY_SCROLLLOCK: return GLFW_KEY_SCROLL_LOCK;
|
|
|
|
case KEY_PAUSE: return GLFW_KEY_PAUSE;
|
|
|
|
case KEY_DELETE: return GLFW_KEY_DELETE;
|
|
|
|
case KEY_BACKSPACE: return GLFW_KEY_BACKSPACE;
|
|
|
|
case KEY_ENTER: return GLFW_KEY_ENTER;
|
|
|
|
case KEY_HOME: return GLFW_KEY_HOME;
|
|
|
|
case KEY_END: return GLFW_KEY_END;
|
|
|
|
case KEY_PAGEUP: return GLFW_KEY_PAGE_UP;
|
|
|
|
case KEY_PAGEDOWN: return GLFW_KEY_PAGE_DOWN;
|
|
|
|
case KEY_INSERT: return GLFW_KEY_INSERT;
|
|
|
|
case KEY_LEFT: return GLFW_KEY_LEFT;
|
|
|
|
case KEY_RIGHT: return GLFW_KEY_RIGHT;
|
|
|
|
case KEY_DOWN: return GLFW_KEY_DOWN;
|
|
|
|
case KEY_UP: return GLFW_KEY_UP;
|
|
|
|
case KEY_F1: return GLFW_KEY_F1;
|
|
|
|
case KEY_F2: return GLFW_KEY_F2;
|
|
|
|
case KEY_F3: return GLFW_KEY_F3;
|
|
|
|
case KEY_F4: return GLFW_KEY_F4;
|
|
|
|
case KEY_F5: return GLFW_KEY_F5;
|
|
|
|
case KEY_F6: return GLFW_KEY_F6;
|
|
|
|
case KEY_F7: return GLFW_KEY_F7;
|
|
|
|
case KEY_F8: return GLFW_KEY_F8;
|
|
|
|
case KEY_F9: return GLFW_KEY_F9;
|
|
|
|
case KEY_F10: return GLFW_KEY_F10;
|
|
|
|
case KEY_F11: return GLFW_KEY_F11;
|
|
|
|
case KEY_F12: return GLFW_KEY_F12;
|
|
|
|
case KEY_F13: return GLFW_KEY_F13;
|
|
|
|
case KEY_F14: return GLFW_KEY_F14;
|
|
|
|
case KEY_F15: return GLFW_KEY_F15;
|
|
|
|
case KEY_F16: return GLFW_KEY_F16;
|
|
|
|
case KEY_F17: return GLFW_KEY_F17;
|
|
|
|
case KEY_F18: return GLFW_KEY_F18;
|
|
|
|
case KEY_F19: return GLFW_KEY_F19;
|
|
|
|
case KEY_F20: return GLFW_KEY_F20;
|
|
|
|
case KEY_F21: return GLFW_KEY_F21;
|
|
|
|
case KEY_F22: return GLFW_KEY_F22;
|
|
|
|
case KEY_F23: return GLFW_KEY_F23;
|
|
|
|
case KEY_F24: return GLFW_KEY_F24;
|
|
|
|
case KEY_KPSLASH: return GLFW_KEY_KP_DIVIDE;
|
|
|
|
case KEY_KPDOT: return GLFW_KEY_KP_MULTIPLY;
|
|
|
|
case KEY_KPMINUS: return GLFW_KEY_KP_SUBTRACT;
|
|
|
|
case KEY_KPPLUS: return GLFW_KEY_KP_ADD;
|
|
|
|
case KEY_KP0: return GLFW_KEY_KP_0;
|
|
|
|
case KEY_KP1: return GLFW_KEY_KP_1;
|
|
|
|
case KEY_KP2: return GLFW_KEY_KP_2;
|
|
|
|
case KEY_KP3: return GLFW_KEY_KP_3;
|
|
|
|
case KEY_KP4: return GLFW_KEY_KP_4;
|
|
|
|
case KEY_KP5: return GLFW_KEY_KP_5;
|
|
|
|
case KEY_KP6: return GLFW_KEY_KP_6;
|
|
|
|
case KEY_KP7: return GLFW_KEY_KP_7;
|
|
|
|
case KEY_KP8: return GLFW_KEY_KP_8;
|
|
|
|
case KEY_KP9: return GLFW_KEY_KP_9;
|
|
|
|
case KEY_KPCOMMA: return GLFW_KEY_KP_DECIMAL;
|
|
|
|
case KEY_KPEQUAL: return GLFW_KEY_KP_EQUAL;
|
|
|
|
case KEY_KPENTER: return GLFW_KEY_KP_ENTER;
|
|
|
|
default: return GLFW_KEY_UNKNOWN;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-11-06 22:52:38 +00:00
|
|
|
static void handleKeyEvent(const MirKeyEvent key, _GLFWwindow* window)
|
2014-11-06 08:24:49 +00:00
|
|
|
{
|
2014-11-06 22:52:38 +00:00
|
|
|
const int pressed = key.action == mir_key_action_up ? GLFW_RELEASE : GLFW_PRESS;
|
|
|
|
const int mods = mirModToGLFWMod(key.modifiers);
|
|
|
|
const long text = _glfwKeySym2Unicode(key.key_code);
|
|
|
|
const int plain = !(mods & (GLFW_MOD_CONTROL | GLFW_MOD_ALT));
|
2014-11-06 08:24:49 +00:00
|
|
|
|
2014-11-06 08:27:54 +00:00
|
|
|
_glfwInputKey(window, toGLFWKeyCode(key.scan_code), key.scan_code, pressed, mods);
|
2014-11-06 08:24:49 +00:00
|
|
|
|
2014-11-06 08:27:54 +00:00
|
|
|
if (text != -1)
|
|
|
|
_glfwInputChar(window, text, mods, plain);
|
2014-11-06 08:24:49 +00:00
|
|
|
}
|
|
|
|
|
2014-11-06 22:52:38 +00:00
|
|
|
static void handleMouseButton(_GLFWwindow* window,
|
|
|
|
int pressed, int mods, MirMotionButton button)
|
2014-11-06 08:24:49 +00:00
|
|
|
{
|
2014-11-06 22:52:38 +00:00
|
|
|
static int lastButton;
|
|
|
|
int publicButton;
|
|
|
|
const int publicMods = mirModToGLFWMod(mods);
|
2014-11-06 08:24:49 +00:00
|
|
|
|
|
|
|
switch (button)
|
|
|
|
{
|
|
|
|
case mir_motion_button_primary:
|
2014-11-06 22:52:38 +00:00
|
|
|
publicButton = GLFW_MOUSE_BUTTON_LEFT;
|
2014-11-06 08:24:49 +00:00
|
|
|
break;
|
|
|
|
case mir_motion_button_secondary:
|
2014-11-06 22:52:38 +00:00
|
|
|
publicButton = GLFW_MOUSE_BUTTON_RIGHT;
|
2014-11-06 08:24:49 +00:00
|
|
|
break;
|
|
|
|
case mir_motion_button_tertiary:
|
2014-11-06 22:52:38 +00:00
|
|
|
publicButton = GLFW_MOUSE_BUTTON_MIDDLE;
|
2014-11-06 08:24:49 +00:00
|
|
|
break;
|
|
|
|
case mir_motion_button_forward:
|
|
|
|
// FIXME What is the forward button?
|
2014-11-06 22:52:38 +00:00
|
|
|
publicButton = GLFW_MOUSE_BUTTON_4;
|
2014-11-06 08:24:49 +00:00
|
|
|
break;
|
|
|
|
case mir_motion_button_back:
|
|
|
|
// FIXME What is the back button?
|
2014-11-06 22:52:38 +00:00
|
|
|
publicButton = GLFW_MOUSE_BUTTON_5;
|
2014-11-06 08:24:49 +00:00
|
|
|
break;
|
|
|
|
default:
|
2014-11-06 22:52:38 +00:00
|
|
|
publicButton = lastButton;
|
2014-11-06 08:24:49 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2014-11-06 22:52:38 +00:00
|
|
|
lastButton = publicButton;
|
2014-11-06 08:24:49 +00:00
|
|
|
|
2014-11-06 22:52:38 +00:00
|
|
|
_glfwInputMouseClick(window, publicButton, pressed, publicMods);
|
2014-11-06 08:24:49 +00:00
|
|
|
}
|
|
|
|
|
2014-11-06 08:27:54 +00:00
|
|
|
static void handleMouseMotion(_GLFWwindow* window, int x, int y)
|
2014-11-06 08:24:49 +00:00
|
|
|
{
|
|
|
|
_glfwInputCursorMotion(window, x, y);
|
|
|
|
}
|
|
|
|
|
2014-11-06 08:27:54 +00:00
|
|
|
static void handleMouseScroll(_GLFWwindow* window, int dx, int dy)
|
2014-11-06 08:24:49 +00:00
|
|
|
{
|
|
|
|
_glfwInputScroll(window, dx, dy);
|
|
|
|
}
|
|
|
|
|
2014-11-06 22:52:38 +00:00
|
|
|
static void handleMouseEvent(const MirMotionEvent motion,
|
|
|
|
int cord_index,
|
|
|
|
_GLFWwindow* window)
|
2014-11-06 08:24:49 +00:00
|
|
|
{
|
|
|
|
switch (motion.action)
|
|
|
|
{
|
|
|
|
case mir_motion_action_down:
|
|
|
|
case mir_motion_action_pointer_down:
|
2014-11-06 22:52:38 +00:00
|
|
|
handleMouseButton(window, GLFW_PRESS,
|
|
|
|
motion.modifiers, motion.button_state);
|
2014-11-06 08:24:49 +00:00
|
|
|
break;
|
|
|
|
case mir_motion_action_up:
|
|
|
|
case mir_motion_action_pointer_up:
|
2014-11-06 22:52:38 +00:00
|
|
|
handleMouseButton(window, GLFW_RELEASE,
|
|
|
|
motion.modifiers, motion.button_state);
|
2014-11-06 08:24:49 +00:00
|
|
|
break;
|
|
|
|
case mir_motion_action_hover_move:
|
|
|
|
case mir_motion_action_move:
|
2014-11-06 08:25:52 +00:00
|
|
|
handleMouseMotion(window,
|
2014-11-06 08:24:49 +00:00
|
|
|
motion.pointer_coordinates[cord_index].x,
|
|
|
|
motion.pointer_coordinates[cord_index].y);
|
|
|
|
break;
|
|
|
|
case mir_motion_action_outside:
|
|
|
|
break;
|
|
|
|
case mir_motion_action_scroll:
|
2014-11-06 08:25:52 +00:00
|
|
|
handleMouseScroll(window,
|
2014-11-06 08:24:49 +00:00
|
|
|
motion.pointer_coordinates[cord_index].hscroll,
|
|
|
|
motion.pointer_coordinates[cord_index].vscroll);
|
|
|
|
break;
|
|
|
|
case mir_motion_action_cancel:
|
|
|
|
case mir_motion_action_hover_enter:
|
|
|
|
case mir_motion_action_hover_exit:
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-11-06 22:52:38 +00:00
|
|
|
static void handleMotionEvent(const MirMotionEvent motion, _GLFWwindow* window)
|
2014-11-06 08:24:49 +00:00
|
|
|
{
|
2014-11-06 22:52:38 +00:00
|
|
|
int i;
|
|
|
|
for (i = 0; i < motion.pointer_count; i++)
|
|
|
|
handleMouseEvent(motion, i, window);
|
2014-11-06 08:24:49 +00:00
|
|
|
}
|
|
|
|
|
2014-11-10 17:42:41 +00:00
|
|
|
static void handleInput(MirEvent const* event, _GLFWwindow* window)
|
2014-11-06 08:24:49 +00:00
|
|
|
{
|
2014-11-06 22:52:38 +00:00
|
|
|
switch (event->type)
|
|
|
|
{
|
|
|
|
case mir_event_type_key:
|
2014-11-10 17:42:41 +00:00
|
|
|
handleKeyEvent(event->key, window);
|
2014-11-06 22:52:38 +00:00
|
|
|
break;
|
|
|
|
case mir_event_type_motion:
|
2014-11-10 17:42:41 +00:00
|
|
|
handleMotionEvent(event->motion, window);
|
2014-11-06 22:52:38 +00:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2014-11-06 08:24:49 +00:00
|
|
|
}
|
|
|
|
|
2014-11-10 17:42:41 +00:00
|
|
|
static void addNewEvent(MirSurface* surface, MirEvent const* event, void* context)
|
|
|
|
{
|
|
|
|
enqueueEvent(event, context);
|
|
|
|
}
|
|
|
|
|
2014-11-06 08:27:54 +00:00
|
|
|
static int createSurface(_GLFWwindow* window)
|
2014-11-06 08:21:12 +00:00
|
|
|
{
|
2014-11-06 22:52:38 +00:00
|
|
|
MirSurfaceParameters params =
|
2014-11-06 08:23:02 +00:00
|
|
|
{
|
|
|
|
.name = "MirSurface",
|
2014-11-06 08:25:52 +00:00
|
|
|
.width = window->mir.width,
|
|
|
|
.height = window->mir.height,
|
2014-11-06 08:23:02 +00:00
|
|
|
.pixel_format = mir_pixel_format_invalid,
|
|
|
|
.buffer_usage = mir_buffer_usage_hardware,
|
|
|
|
.output_id = mir_display_output_id_invalid
|
|
|
|
};
|
|
|
|
|
|
|
|
MirEventDelegate delegate =
|
|
|
|
{
|
2014-11-10 17:42:41 +00:00
|
|
|
addNewEvent,
|
2014-11-06 08:24:49 +00:00
|
|
|
window
|
2014-11-06 08:23:02 +00:00
|
|
|
};
|
|
|
|
|
2014-11-06 08:25:52 +00:00
|
|
|
params.pixel_format = findValidPixelFormat();
|
2014-11-06 08:23:02 +00:00
|
|
|
if (params.pixel_format == mir_pixel_format_invalid)
|
|
|
|
{
|
|
|
|
_glfwInputError(GLFW_PLATFORM_ERROR,
|
2014-11-06 22:52:38 +00:00
|
|
|
"Mir: Unable to find a correct pixel format");
|
2014-11-06 08:23:02 +00:00
|
|
|
return GL_FALSE;
|
|
|
|
}
|
|
|
|
|
2014-11-06 22:52:38 +00:00
|
|
|
window->mir.surface =
|
|
|
|
mir_connection_create_surface_sync(_glfw.mir.connection, ¶ms);
|
2014-11-06 08:23:02 +00:00
|
|
|
if (!mir_surface_is_valid(window->mir.surface))
|
|
|
|
{
|
|
|
|
_glfwInputError(GLFW_PLATFORM_ERROR,
|
2014-11-06 22:52:38 +00:00
|
|
|
"Mir: Unable to create surface");
|
2014-11-06 08:23:02 +00:00
|
|
|
return GL_FALSE;
|
|
|
|
}
|
|
|
|
|
2014-11-06 08:24:49 +00:00
|
|
|
mir_surface_set_event_handler(window->mir.surface, &delegate);
|
|
|
|
|
2014-11-06 08:25:52 +00:00
|
|
|
return GL_TRUE;
|
|
|
|
}
|
|
|
|
|
2014-11-10 17:42:41 +00:00
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
////// GLFW internal API //////
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
void _glfwInitEventQueue(EventQueue* queue)
|
|
|
|
{
|
|
|
|
TAILQ_INIT(&queue->head);
|
|
|
|
}
|
|
|
|
|
|
|
|
void _glfwDeleteEventQueue(EventQueue* queue)
|
|
|
|
{
|
2014-11-10 20:17:21 +00:00
|
|
|
if (queue)
|
2014-11-10 17:42:41 +00:00
|
|
|
{
|
2014-11-10 20:17:21 +00:00
|
|
|
EventNode* node, *node_next;
|
|
|
|
node = queue->head.tqh_first;
|
2014-11-10 17:42:41 +00:00
|
|
|
|
2014-11-10 20:17:21 +00:00
|
|
|
while (node != NULL)
|
|
|
|
{
|
|
|
|
node_next = node->entries.tqe_next;
|
2014-11-10 17:42:41 +00:00
|
|
|
|
2014-11-10 20:17:21 +00:00
|
|
|
TAILQ_REMOVE(&queue->head, node, entries);
|
|
|
|
deleteNode(queue, node);
|
|
|
|
|
|
|
|
node = node_next;
|
|
|
|
}
|
|
|
|
|
2014-11-12 17:41:12 +00:00
|
|
|
free(queue);
|
|
|
|
}
|
2014-11-10 17:42:41 +00:00
|
|
|
}
|
2014-11-06 22:52:38 +00:00
|
|
|
|
2014-11-06 08:25:52 +00:00
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
////// GLFW platform API //////
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
int _glfwPlatformCreateWindow(_GLFWwindow* window,
|
|
|
|
const _GLFWwndconfig* wndconfig,
|
|
|
|
const _GLFWctxconfig* ctxconfig,
|
|
|
|
const _GLFWfbconfig* fbconfig)
|
|
|
|
{
|
|
|
|
if (!_glfwCreateContext(window, ctxconfig, fbconfig))
|
|
|
|
return GL_FALSE;
|
|
|
|
|
2014-11-06 20:07:20 +00:00
|
|
|
if (wndconfig->monitor)
|
|
|
|
{
|
2014-11-06 22:45:56 +00:00
|
|
|
GLFWvidmode mode;
|
|
|
|
_glfwPlatformGetVideoMode(wndconfig->monitor, &mode);
|
|
|
|
|
2014-11-06 20:07:20 +00:00
|
|
|
mir_surface_set_type(window->mir.surface, mir_surface_state_fullscreen);
|
|
|
|
|
2014-11-06 22:45:56 +00:00
|
|
|
if (wndconfig->width > mode.width || wndconfig->height > mode.height)
|
2014-11-06 20:07:20 +00:00
|
|
|
{
|
|
|
|
_glfwInputError(GLFW_PLATFORM_ERROR,
|
2014-11-06 22:52:38 +00:00
|
|
|
"Mir: Requested surface size is to large (%i %i)",
|
2014-11-06 20:07:20 +00:00
|
|
|
wndconfig->width, wndconfig->height);
|
|
|
|
|
|
|
|
return GL_FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-11-06 08:25:52 +00:00
|
|
|
window->mir.width = wndconfig->width;
|
|
|
|
window->mir.height = wndconfig->height;
|
|
|
|
|
|
|
|
if (!createSurface(window))
|
|
|
|
return GL_FALSE;
|
|
|
|
|
2014-11-06 22:52:38 +00:00
|
|
|
window->mir.window = mir_surface_get_egl_native_window(window->mir.surface);
|
2014-11-06 08:25:52 +00:00
|
|
|
|
2014-11-06 08:23:02 +00:00
|
|
|
return GL_TRUE;
|
2014-11-06 08:21:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void _glfwPlatformDestroyWindow(_GLFWwindow* window)
|
|
|
|
{
|
2014-11-06 22:52:38 +00:00
|
|
|
if (mir_surface_is_valid(window->mir.surface))
|
|
|
|
{
|
|
|
|
mir_surface_release_sync(window->mir.surface);
|
|
|
|
window->mir.surface = NULL;
|
|
|
|
}
|
2014-11-06 20:07:20 +00:00
|
|
|
|
2014-11-06 22:52:38 +00:00
|
|
|
_glfwDestroyContext(window);
|
2014-11-06 08:21:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void _glfwPlatformSetWindowTitle(_GLFWwindow* window, const char* title)
|
|
|
|
{
|
2014-11-06 08:24:08 +00:00
|
|
|
_glfwInputError(GLFW_PLATFORM_ERROR,
|
2014-11-06 22:52:38 +00:00
|
|
|
"Mir: Unsupported Function %s!", __PRETTY_FUNCTION__);
|
2014-11-06 08:21:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void _glfwPlatformSetWindowPos(_GLFWwindow* window, int xpos, int ypos)
|
|
|
|
{
|
2014-11-06 08:24:08 +00:00
|
|
|
_glfwInputError(GLFW_PLATFORM_ERROR,
|
2014-11-06 22:52:38 +00:00
|
|
|
"Mir: Unsupported Function %s!", __PRETTY_FUNCTION__);
|
2014-11-06 08:21:12 +00:00
|
|
|
}
|
|
|
|
|
2014-11-06 22:52:38 +00:00
|
|
|
void _glfwPlatformGetWindowFrameSize(_GLFWwindow* window,
|
|
|
|
int* left, int* top,
|
|
|
|
int* right, int* bottom)
|
2014-11-06 08:26:39 +00:00
|
|
|
{
|
|
|
|
_glfwInputError(GLFW_PLATFORM_ERROR,
|
2014-11-06 22:52:38 +00:00
|
|
|
"Mir: Unsupported Function %s!", __PRETTY_FUNCTION__);
|
2014-11-06 08:26:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void _glfwPlatformGetWindowPos(_GLFWwindow* window, int* xpos, int* ypos)
|
|
|
|
{
|
|
|
|
_glfwInputError(GLFW_PLATFORM_ERROR,
|
2014-11-06 22:52:38 +00:00
|
|
|
"Mir: Unsupported Function %s!", __PRETTY_FUNCTION__);
|
2014-11-06 08:26:39 +00:00
|
|
|
}
|
|
|
|
|
2014-11-06 08:21:12 +00:00
|
|
|
void _glfwPlatformSetWindowSize(_GLFWwindow* window, int width, int height)
|
|
|
|
{
|
2014-11-06 08:24:08 +00:00
|
|
|
_glfwInputError(GLFW_PLATFORM_ERROR,
|
2014-11-06 22:52:38 +00:00
|
|
|
"Mir: Unsupported Function %s!", __PRETTY_FUNCTION__);
|
2014-11-06 08:21:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void _glfwPlatformGetWindowSize(_GLFWwindow* window, int* width, int* height)
|
|
|
|
{
|
2014-11-06 22:52:38 +00:00
|
|
|
if (width)
|
|
|
|
*width = window->mir.width;
|
|
|
|
if (height)
|
|
|
|
*height = window->mir.height;
|
2014-11-06 08:21:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void _glfwPlatformIconifyWindow(_GLFWwindow* window)
|
|
|
|
{
|
2014-11-06 20:07:20 +00:00
|
|
|
mir_surface_set_type(window->mir.surface, mir_surface_state_minimized);
|
2014-11-06 08:21:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void _glfwPlatformRestoreWindow(_GLFWwindow* window)
|
|
|
|
{
|
2014-11-06 08:26:39 +00:00
|
|
|
mir_surface_set_type(window->mir.surface, mir_surface_state_restored);
|
2014-11-06 08:21:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void _glfwPlatformHideWindow(_GLFWwindow* window)
|
2014-11-06 08:26:39 +00:00
|
|
|
{
|
2014-11-06 20:07:20 +00:00
|
|
|
_glfwInputError(GLFW_PLATFORM_ERROR,
|
2014-11-06 22:52:38 +00:00
|
|
|
"Mir: Unsupported Function %s!", __PRETTY_FUNCTION__);
|
2014-11-06 08:26:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void _glfwPlatformShowWindow(_GLFWwindow* window)
|
|
|
|
{
|
2014-11-06 20:07:20 +00:00
|
|
|
_glfwInputError(GLFW_PLATFORM_ERROR,
|
2014-11-06 22:52:38 +00:00
|
|
|
"Mir: Unsupported Function %s!", __PRETTY_FUNCTION__);
|
2014-11-06 08:26:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void _glfwPlatformUnhideWindow(_GLFWwindow* window)
|
2014-11-06 08:21:12 +00:00
|
|
|
{
|
2014-11-06 20:07:20 +00:00
|
|
|
_glfwInputError(GLFW_PLATFORM_ERROR,
|
2014-11-06 22:52:38 +00:00
|
|
|
"Mir: Unsupported Function %s!", __PRETTY_FUNCTION__);
|
2014-11-06 08:21:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void _glfwPlatformPollEvents(void)
|
|
|
|
{
|
2014-11-10 17:42:41 +00:00
|
|
|
EventNode* node = NULL;
|
|
|
|
|
|
|
|
while ((node = dequeueEvent(_glfw.mir.event_queue)))
|
|
|
|
{
|
|
|
|
handleInput(node->event, node->window);
|
|
|
|
deleteNode(_glfw.mir.event_queue, node);
|
|
|
|
}
|
2014-11-06 08:21:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void _glfwPlatformWaitEvents(void)
|
|
|
|
{
|
2014-11-10 17:42:41 +00:00
|
|
|
pthread_mutex_lock(&_glfw.mir.event_mutex);
|
|
|
|
|
|
|
|
if (emptyEventQueue(_glfw.mir.event_queue))
|
|
|
|
pthread_cond_wait(&_glfw.mir.event_cond, &_glfw.mir.event_mutex);
|
|
|
|
|
|
|
|
pthread_mutex_unlock(&_glfw.mir.event_mutex);
|
|
|
|
|
|
|
|
_glfwPlatformPollEvents();
|
2014-11-06 08:21:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void _glfwPlatformPostEmptyEvent(void)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void _glfwPlatformGetFramebufferSize(_GLFWwindow* window, int* width, int* height)
|
|
|
|
{
|
2014-11-06 22:52:38 +00:00
|
|
|
if (width)
|
|
|
|
*width = window->mir.width;
|
|
|
|
if (height)
|
|
|
|
*height = window->mir.height;
|
2014-11-06 08:21:12 +00:00
|
|
|
}
|
|
|
|
|
2014-11-06 22:52:38 +00:00
|
|
|
int _glfwPlatformCreateCursor(_GLFWcursor* cursor,
|
|
|
|
const GLFWimage* image,
|
|
|
|
int xhot, int yhot)
|
2014-11-06 08:21:12 +00:00
|
|
|
{
|
2014-11-06 08:24:08 +00:00
|
|
|
_glfwInputError(GLFW_PLATFORM_ERROR,
|
2014-11-06 22:52:38 +00:00
|
|
|
"Mir: Unsupported Function %s!", __PRETTY_FUNCTION__);
|
2014-11-06 08:24:08 +00:00
|
|
|
|
2014-11-06 22:52:38 +00:00
|
|
|
return GL_FALSE;
|
2014-11-06 08:21:12 +00:00
|
|
|
}
|
|
|
|
|
2014-09-02 14:52:16 +00:00
|
|
|
int _glfwPlatformCreateStandardCursor(_GLFWcursor* cursor, int shape)
|
|
|
|
{
|
|
|
|
_glfwInputError(GLFW_PLATFORM_ERROR,
|
|
|
|
"Mir: Unsupported Function %s!", __PRETTY_FUNCTION__);
|
|
|
|
|
|
|
|
return GL_FALSE;
|
|
|
|
}
|
|
|
|
|
2014-11-06 08:21:12 +00:00
|
|
|
void _glfwPlatformDestroyCursor(_GLFWcursor* cursor)
|
|
|
|
{
|
2014-11-06 20:07:20 +00:00
|
|
|
_glfwInputError(GLFW_PLATFORM_ERROR,
|
2014-11-06 22:52:38 +00:00
|
|
|
"Mir: Unsupported Function %s!", __PRETTY_FUNCTION__);
|
2014-11-06 08:21:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void _glfwPlatformSetCursor(_GLFWwindow* window, _GLFWcursor* cursor)
|
|
|
|
{
|
2014-11-06 08:26:39 +00:00
|
|
|
_glfwInputError(GLFW_PLATFORM_ERROR,
|
2014-11-06 22:52:38 +00:00
|
|
|
"Mir: Unsupported Function %s!", __PRETTY_FUNCTION__);
|
2014-11-06 08:21:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void _glfwPlatformSetCursorPos(_GLFWwindow* window, double xpos, double ypos)
|
|
|
|
{
|
2014-11-06 08:24:08 +00:00
|
|
|
_glfwInputError(GLFW_PLATFORM_ERROR,
|
2014-11-06 22:52:38 +00:00
|
|
|
"Mir: Unsupported Function %s!", __PRETTY_FUNCTION__);
|
2014-11-06 08:21:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void _glfwPlatformApplyCursorMode(_GLFWwindow* window)
|
|
|
|
{
|
2014-11-06 20:07:20 +00:00
|
|
|
_glfwInputError(GLFW_PLATFORM_ERROR,
|
2014-11-06 22:52:38 +00:00
|
|
|
"Mir: Unsupported Function %s!", __PRETTY_FUNCTION__);
|
2014-11-06 08:21:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void _glfwPlatformSetClipboardString(_GLFWwindow* window, const char* string)
|
|
|
|
{
|
2014-11-06 08:24:08 +00:00
|
|
|
_glfwInputError(GLFW_PLATFORM_ERROR,
|
2014-11-06 22:52:38 +00:00
|
|
|
"Mir: Unsupported Function %s!", __PRETTY_FUNCTION__);
|
2014-11-06 08:21:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
const char* _glfwPlatformGetClipboardString(_GLFWwindow* window)
|
|
|
|
{
|
2014-11-06 08:24:08 +00:00
|
|
|
_glfwInputError(GLFW_PLATFORM_ERROR,
|
2014-11-06 22:52:38 +00:00
|
|
|
"Mir: Unsupported Function %s!", __PRETTY_FUNCTION__);
|
2014-11-06 08:24:08 +00:00
|
|
|
|
|
|
|
return NULL;
|
2014-11-06 08:21:12 +00:00
|
|
|
}
|
2014-11-06 22:52:38 +00:00
|
|
|
|