glfw/src/input.c

416 lines
11 KiB
C
Raw Normal View History

2010-09-07 15:34:51 +00:00
//========================================================================
// GLFW - An OpenGL library
2010-09-07 15:34:51 +00:00
// Platform: Any
2010-09-07 15:41:26 +00:00
// API version: 3.0
2010-09-07 15:34:51 +00:00
// WWW: http://www.glfw.org/
//------------------------------------------------------------------------
// Copyright (c) 2002-2006 Marcus Geelnard
// Copyright (c) 2006-2010 Camilla Berglund <elmindreda@elmindreda.org>
//
// 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.
//
//========================================================================
#include "internal.h"
2013-03-01 14:18:53 +00:00
// Internal key state used for sticky keys
#define _GLFW_STICK 3
2010-09-07 15:34:51 +00:00
// Sets the cursor mode for the specified window
2013-02-04 12:22:10 +00:00
//
static void setCursorMode(_GLFWwindow* window, int newMode)
{
int oldMode;
if (newMode != GLFW_CURSOR_NORMAL &&
newMode != GLFW_CURSOR_HIDDEN &&
newMode != GLFW_CURSOR_CAPTURED)
{
_glfwInputError(GLFW_INVALID_ENUM, NULL);
return;
}
oldMode = window->cursorMode;
if (oldMode == newMode)
return;
if (oldMode == GLFW_CURSOR_CAPTURED)
_glfwPlatformSetCursorPos(window, _glfw.cursorPosX, _glfw.cursorPosY);
else if (newMode == GLFW_CURSOR_CAPTURED)
{
int width, height;
_glfw.cursorPosX = window->cursorPosX;
_glfw.cursorPosY = window->cursorPosY;
_glfwPlatformGetWindowSize(window, &width, &height);
_glfwPlatformSetCursorPos(window, width / 2.0, height / 2.0);
}
_glfwPlatformSetCursorMode(window, newMode);
window->cursorMode = newMode;
}
// Set sticky keys mode for the specified window
2013-02-04 12:22:10 +00:00
//
static void setStickyKeys(_GLFWwindow* window, int enabled)
{
if (window->stickyKeys == enabled)
return;
if (!enabled)
{
int i;
// Release all sticky keys
for (i = 0; i <= GLFW_KEY_LAST; i++)
{
if (window->key[i] == _GLFW_STICK)
window->key[i] = GLFW_RELEASE;
}
}
window->stickyKeys = enabled;
}
// Set sticky mouse buttons mode for the specified window
2013-02-04 12:22:10 +00:00
//
static void setStickyMouseButtons(_GLFWwindow* window, int enabled)
{
if (window->stickyMouseButtons == enabled)
return;
if (!enabled)
{
int i;
// Release all sticky mouse buttons
for (i = 0; i <= GLFW_MOUSE_BUTTON_LAST; i++)
{
if (window->mouseButton[i] == _GLFW_STICK)
window->mouseButton[i] = GLFW_RELEASE;
}
}
window->stickyMouseButtons = enabled;
}
//////////////////////////////////////////////////////////////////////////
////// GLFW event API //////
//////////////////////////////////////////////////////////////////////////
void _glfwInputKey(_GLFWwindow* window, int key, int action)
{
2012-02-03 23:52:13 +00:00
GLboolean repeated = GL_FALSE;
if (key < 0 || key > GLFW_KEY_LAST)
return;
2013-01-12 16:06:35 +00:00
if (action == GLFW_PRESS && window->key[key] == GLFW_PRESS)
repeated = GL_TRUE;
2012-03-10 15:23:09 +00:00
if (action == GLFW_RELEASE && window->stickyKeys)
window->key[key] = _GLFW_STICK;
else
window->key[key] = (char) action;
2013-01-12 16:06:35 +00:00
if (repeated)
action = GLFW_REPEAT;
if (window->callbacks.key)
2013-01-15 20:34:26 +00:00
window->callbacks.key((GLFWwindow*) window, key, action);
}
2013-02-25 16:02:28 +00:00
void _glfwInputChar(_GLFWwindow* window, unsigned int character)
{
if (character == -1)
return;
if (character < 32 || (character > 126 && character < 160))
return;
2013-01-15 20:34:26 +00:00
if (window->callbacks.character)
window->callbacks.character((GLFWwindow*) window, character);
}
2012-03-28 19:54:09 +00:00
void _glfwInputScroll(_GLFWwindow* window, double xoffset, double yoffset)
{
2013-01-15 20:34:26 +00:00
if (window->callbacks.scroll)
window->callbacks.scroll((GLFWwindow*) window, xoffset, yoffset);
}
void _glfwInputMouseClick(_GLFWwindow* window, int button, int action)
{
if (button < 0 || button > GLFW_MOUSE_BUTTON_LAST)
return;
// Register mouse button action
if (action == GLFW_RELEASE && window->stickyMouseButtons)
window->mouseButton[button] = _GLFW_STICK;
else
window->mouseButton[button] = (char) action;
2013-01-15 20:34:26 +00:00
if (window->callbacks.mouseButton)
window->callbacks.mouseButton((GLFWwindow*) window, button, action);
}
2013-04-04 14:16:21 +00:00
void _glfwInputCursorMotion(_GLFWwindow* window, double x, double y)
{
if (window->cursorMode == GLFW_CURSOR_CAPTURED)
{
2013-04-04 14:16:21 +00:00
if (x == 0.0 && y == 0.0)
2011-10-13 12:07:52 +00:00
return;
window->cursorPosX += x;
window->cursorPosY += y;
}
else
{
if (window->cursorPosX == x && window->cursorPosY == y)
2011-10-13 12:07:52 +00:00
return;
window->cursorPosX = x;
window->cursorPosY = y;
}
2013-01-15 20:34:26 +00:00
if (window->callbacks.cursorPos)
2012-01-24 17:28:37 +00:00
{
2013-01-15 20:34:26 +00:00
window->callbacks.cursorPos((GLFWwindow*) window,
window->cursorPosX,
window->cursorPosY);
2012-01-24 17:28:37 +00:00
}
}
2012-01-30 21:59:38 +00:00
void _glfwInputCursorEnter(_GLFWwindow* window, int entered)
{
2013-01-15 20:34:26 +00:00
if (window->callbacks.cursorEnter)
window->callbacks.cursorEnter((GLFWwindow*) window, entered);
}
2010-09-09 18:59:50 +00:00
//////////////////////////////////////////////////////////////////////////
////// GLFW public API //////
//////////////////////////////////////////////////////////////////////////
GLFWAPI int glfwGetInputMode(GLFWwindow* handle, int mode)
2012-02-04 00:34:12 +00:00
{
_GLFWwindow* window = (_GLFWwindow*) handle;
_GLFW_REQUIRE_INIT_OR_RETURN(0);
2012-02-04 00:34:12 +00:00
switch (mode)
{
case GLFW_CURSOR_MODE:
return window->cursorMode;
case GLFW_STICKY_KEYS:
return window->stickyKeys;
case GLFW_STICKY_MOUSE_BUTTONS:
return window->stickyMouseButtons;
default:
_glfwInputError(GLFW_INVALID_ENUM, NULL);
2012-02-04 00:34:12 +00:00
return 0;
}
}
GLFWAPI void glfwSetInputMode(GLFWwindow* handle, int mode, int value)
2012-02-04 00:34:12 +00:00
{
_GLFWwindow* window = (_GLFWwindow*) handle;
_GLFW_REQUIRE_INIT();
2012-02-04 00:34:12 +00:00
switch (mode)
{
case GLFW_CURSOR_MODE:
setCursorMode(window, value);
break;
case GLFW_STICKY_KEYS:
setStickyKeys(window, value ? GL_TRUE : GL_FALSE);
break;
case GLFW_STICKY_MOUSE_BUTTONS:
setStickyMouseButtons(window, value ? GL_TRUE : GL_FALSE);
break;
default:
_glfwInputError(GLFW_INVALID_ENUM, NULL);
2012-02-04 00:34:12 +00:00
break;
}
}
GLFWAPI int glfwGetKey(GLFWwindow* handle, int key)
2010-09-07 15:34:51 +00:00
{
2011-04-06 18:38:55 +00:00
_GLFWwindow* window = (_GLFWwindow*) handle;
_GLFW_REQUIRE_INIT_OR_RETURN(GLFW_RELEASE);
2010-09-07 15:34:51 +00:00
2010-09-08 12:45:52 +00:00
if (key < 0 || key > GLFW_KEY_LAST)
{
_glfwInputError(GLFW_INVALID_ENUM, "The specified key is invalid");
2010-09-07 15:34:51 +00:00
return GLFW_RELEASE;
}
2010-09-07 15:34:51 +00:00
if (window->key[key] == _GLFW_STICK)
2010-09-07 15:34:51 +00:00
{
// Sticky mode: release key now
2010-09-09 16:15:32 +00:00
window->key[key] = GLFW_RELEASE;
2010-09-07 15:34:51 +00:00
return GLFW_PRESS;
}
2010-09-09 16:15:32 +00:00
return (int) window->key[key];
2010-09-07 15:34:51 +00:00
}
GLFWAPI int glfwGetMouseButton(GLFWwindow* handle, int button)
2010-09-07 15:34:51 +00:00
{
2011-04-06 18:38:55 +00:00
_GLFWwindow* window = (_GLFWwindow*) handle;
_GLFW_REQUIRE_INIT_OR_RETURN(GLFW_RELEASE);
2010-09-07 15:34:51 +00:00
2010-09-08 12:45:52 +00:00
if (button < 0 || button > GLFW_MOUSE_BUTTON_LAST)
{
_glfwInputError(GLFW_INVALID_ENUM,
"The specified mouse button is invalid");
2010-09-07 15:34:51 +00:00
return GLFW_RELEASE;
}
2010-09-07 15:34:51 +00:00
if (window->mouseButton[button] == _GLFW_STICK)
2010-09-07 15:34:51 +00:00
{
// Sticky mode: release mouse button now
2010-09-09 16:15:32 +00:00
window->mouseButton[button] = GLFW_RELEASE;
2010-09-07 15:34:51 +00:00
return GLFW_PRESS;
}
2010-09-09 16:15:32 +00:00
return (int) window->mouseButton[button];
2010-09-07 15:34:51 +00:00
}
2013-04-04 14:16:21 +00:00
GLFWAPI void glfwGetCursorPos(GLFWwindow* handle, double* xpos, double* ypos)
2010-09-07 15:34:51 +00:00
{
2011-04-06 18:38:55 +00:00
_GLFWwindow* window = (_GLFWwindow*) handle;
_GLFW_REQUIRE_INIT();
2010-09-07 15:34:51 +00:00
if (xpos)
*xpos = window->cursorPosX;
if (ypos)
*ypos = window->cursorPosY;
2010-09-07 15:34:51 +00:00
}
2013-04-04 14:16:21 +00:00
GLFWAPI void glfwSetCursorPos(GLFWwindow* handle, double xpos, double ypos)
2010-09-07 15:34:51 +00:00
{
2011-04-06 18:38:55 +00:00
_GLFWwindow* window = (_GLFWwindow*) handle;
_GLFW_REQUIRE_INIT();
2010-09-07 15:34:51 +00:00
if (_glfw.focusedWindow != window)
return;
// Don't do anything if the cursor position did not change
if (xpos == window->cursorPosX && ypos == window->cursorPosY)
2010-09-07 15:34:51 +00:00
return;
// Set GLFW cursor position
window->cursorPosX = xpos;
window->cursorPosY = ypos;
2010-09-07 15:34:51 +00:00
// Do not move physical cursor in locked cursor mode
if (window->cursorMode == GLFW_CURSOR_CAPTURED)
2010-09-07 15:34:51 +00:00
return;
// Update physical cursor position
_glfwPlatformSetCursorPos(window, xpos, ypos);
2010-09-07 15:34:51 +00:00
}
GLFWAPI GLFWkeyfun glfwSetKeyCallback(GLFWwindow* handle, GLFWkeyfun cbfun)
2010-09-07 15:34:51 +00:00
{
_GLFWwindow* window = (_GLFWwindow*) handle;
GLFWkeyfun previous;
_GLFW_REQUIRE_INIT_OR_RETURN(NULL);
previous = window->callbacks.key;
2013-01-15 20:34:26 +00:00
window->callbacks.key = cbfun;
return previous;
2010-09-07 15:34:51 +00:00
}
GLFWAPI GLFWcharfun glfwSetCharCallback(GLFWwindow* handle, GLFWcharfun cbfun)
2010-09-07 15:34:51 +00:00
{
_GLFWwindow* window = (_GLFWwindow*) handle;
GLFWcharfun previous;
_GLFW_REQUIRE_INIT_OR_RETURN(NULL);
previous = window->callbacks.character;
2013-01-15 20:34:26 +00:00
window->callbacks.character = cbfun;
return previous;
2010-09-07 15:34:51 +00:00
}
GLFWAPI GLFWmousebuttonfun glfwSetMouseButtonCallback(GLFWwindow* handle,
GLFWmousebuttonfun cbfun)
2010-09-07 15:34:51 +00:00
{
_GLFWwindow* window = (_GLFWwindow*) handle;
GLFWmousebuttonfun previous;
_GLFW_REQUIRE_INIT_OR_RETURN(NULL);
previous = window->callbacks.mouseButton;
2013-01-15 20:34:26 +00:00
window->callbacks.mouseButton = cbfun;
return previous;
2010-09-07 15:34:51 +00:00
}
GLFWAPI GLFWcursorposfun glfwSetCursorPosCallback(GLFWwindow* handle,
GLFWcursorposfun cbfun)
2010-09-07 15:34:51 +00:00
{
_GLFWwindow* window = (_GLFWwindow*) handle;
GLFWcursorposfun previous;
_GLFW_REQUIRE_INIT_OR_RETURN(NULL);
previous = window->callbacks.cursorPos;
2013-01-15 20:34:26 +00:00
window->callbacks.cursorPos = cbfun;
return previous;
2010-09-07 15:34:51 +00:00
}
GLFWAPI GLFWcursorenterfun glfwSetCursorEnterCallback(GLFWwindow* handle,
GLFWcursorenterfun cbfun)
2010-09-07 15:34:51 +00:00
{
_GLFWwindow* window = (_GLFWwindow*) handle;
GLFWcursorenterfun previous;
_GLFW_REQUIRE_INIT_OR_RETURN(NULL);
previous = window->callbacks.cursorEnter;
2013-01-15 20:34:26 +00:00
window->callbacks.cursorEnter = cbfun;
return previous;
2010-09-07 15:34:51 +00:00
}
GLFWAPI GLFWscrollfun glfwSetScrollCallback(GLFWwindow* handle,
GLFWscrollfun cbfun)
{
_GLFWwindow* window = (_GLFWwindow*) handle;
GLFWscrollfun previous;
_GLFW_REQUIRE_INIT_OR_RETURN(NULL);
previous = window->callbacks.scroll;
2013-01-15 20:34:26 +00:00
window->callbacks.scroll = cbfun;
return previous;
}