lwjgl/src/native/linux/org_lwjgl_input_Keyboard.cpp

280 lines
7.5 KiB
C++
Raw Normal View History

2002-11-15 05:40:55 -05:00
/*
* Copyright (c) 2002 Light Weight Java Game Library Project
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met:
*
* * Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
*
* * Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* * Neither the name of 'Light Weight Java Game Library' nor the names of
* its contributors may be used to endorse or promote products derived
* from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
* TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
* PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
/**
* $Id$
*
2002-11-15 06:10:32 -05:00
* Linux keyboard handling.
2002-11-15 05:40:55 -05:00
*
2002-11-15 06:10:32 -05:00
* @author elias_naur <elias_naur@users.sourceforge.net>
2002-11-15 05:40:55 -05:00
* @version $Revision$
*/
#include <X11/X.h>
#include <X11/Xlib.h>
#include <string.h>
#include <assert.h>
#include "org_lwjgl_input_Keyboard.h"
#define KEYBOARD_BUFFER_SIZE 50
#define KEYBOARD_SIZE 256
2002-11-26 15:07:49 -05:00
#define KEY_EVENT_BACKLOG 20
2002-11-21 10:05:51 -05:00
unsigned char readBuffer[KEYBOARD_BUFFER_SIZE * 2];
2002-11-15 05:40:55 -05:00
jfieldID fid_readBuffer;
jfieldID fid_readBufferAddress;
unsigned char key_buf[KEYBOARD_SIZE];
2002-11-25 09:57:13 -05:00
unsigned char key_map[KEYBOARD_SIZE];
2002-11-15 05:40:55 -05:00
typedef struct {
unsigned char keycode;
unsigned char state;
} input_event;
input_event saved_key_events[KEY_EVENT_BACKLOG];
int list_start = 0;
int list_end = 0;
2002-11-21 10:05:51 -05:00
bool keyboard_grabbed;
2002-11-15 05:40:55 -05:00
extern Display *disp;
extern Window win;
extern int current_fullscreen;
extern int isFocused(void);
2002-11-15 05:40:55 -05:00
/*
* Class: org_lwjgl_input_Keyboard
* Method: initIDs
* Signature: ()V
*/
JNIEXPORT void JNICALL Java_org_lwjgl_input_Keyboard_initIDs
(JNIEnv * env, jclass clazz)
{
// Get a global class instance, just to be sure
static jobject globalClassLock = NULL;
if (globalClassLock == NULL) {
2002-11-19 03:50:57 -05:00
globalClassLock = env->NewGlobalRef(clazz);
2002-11-15 05:40:55 -05:00
}
2002-11-19 03:50:57 -05:00
fid_readBuffer = env->GetStaticFieldID(clazz, "readBuffer", "Ljava/nio/ByteBuffer;");
fid_readBufferAddress = env->GetStaticFieldID(clazz, "readBufferAddress", "I");
2002-11-15 05:40:55 -05:00
}
int grabKeyboard(void) {
int result = XGrabKeyboard(disp, win, False, GrabModeAsync, GrabModeAsync, CurrentTime);
if (result == GrabSuccess)
2002-11-21 10:05:51 -05:00
keyboard_grabbed = true;
return result;
}
void ungrabKeyboard(void) {
2002-11-21 10:05:51 -05:00
keyboard_grabbed = false;
XUngrabKeyboard(disp, CurrentTime);
}
int updateKeyboardGrab(void) {
if (current_fullscreen) {
if (!keyboard_grabbed)
return grabKeyboard();
} else {
if (isFocused()) {
if (!keyboard_grabbed)
return grabKeyboard();
} else {
if (keyboard_grabbed)
ungrabKeyboard();
}
}
return GrabSuccess;
}
2002-11-15 05:40:55 -05:00
/*
* Class: org_lwjgl_input_Keyboard
* Method: nCreate
* Signature: ()Z
*/
JNIEXPORT jboolean JNICALL Java_org_lwjgl_input_Keyboard_nCreate
(JNIEnv * env, jclass clazz)
{
keyboard_grabbed = 0;
if (updateKeyboardGrab() != GrabSuccess) {
2002-11-15 05:40:55 -05:00
#ifdef _DEBUG
printf("Could not grab keyboard\n");
#endif
return JNI_FALSE;
}
2002-11-25 09:57:13 -05:00
for (int i = 0; i < KEYBOARD_SIZE; i++)
key_map[i] = i;
key_map[0x6b] = 0xdb; // Left doze key
key_map[0x6c] = 0xdc; // Right doze key
key_map[0x6d] = 0xdd; // Apps key
key_map[0x5a] = 0xc8; // Up arrow
key_map[0x5c] = 0xcb; // Left arrow
key_map[0x5e] = 0xcd; // Right arrow
key_map[0x60] = 0xd0; // Down arrow
key_map[0x59] = 0xc7; // Home
key_map[0x62] = 0xd2; // Insert
key_map[0x63] = 0xd3; // Delete
key_map[0x5f] = 0xcf; // End
key_map[0x5b] = 0xc9; // Page up
key_map[0x61] = 0xd1; // Page down
key_map[0x67] = 0xb7; // SysRQ
key_map[0x66] = 0xc5; // Pause
key_map[0x64] = 0x9c; // Numpad enter
key_map[0x68] = 0xb5; // Numpad divide
2002-11-15 05:40:55 -05:00
memset(key_buf, 0, KEYBOARD_SIZE*sizeof(unsigned char));
return JNI_TRUE;
}
/*
* Class: org_lwjgl_input_Keyboard
* Method: nDestroy
* Signature: ()V
*/
JNIEXPORT void JNICALL Java_org_lwjgl_input_Keyboard_nDestroy
(JNIEnv * env, jclass clazz)
{
2002-11-21 10:05:51 -05:00
if (keyboard_grabbed)
ungrabKeyboard();
2002-11-15 05:40:55 -05:00
}
input_event *nextEventElement(void) {
if (list_start == list_end)
return NULL;
input_event *result = &(saved_key_events[list_start]);
list_start = (list_start + 1)%KEY_EVENT_BACKLOG;
return result;
}
void putEventElement(unsigned char keycode, unsigned char state) {
int next_index = (list_end + 1)%KEY_EVENT_BACKLOG;
if (next_index == list_start)
return;
saved_key_events[list_end].keycode = keycode;
saved_key_events[list_end].state = state;
list_end = next_index;
}
unsigned char getKeycode(XEvent *event) {
unsigned char keycode = (unsigned char)((event->xkey.keycode - 8) & 0xff);
keycode = key_map[keycode];
return keycode;
}
/*
* Class: org_lwjgl_input_Keyboard
* Method: nPoll
* Signature: (I)V
*/
JNIEXPORT void JNICALL Java_org_lwjgl_input_Keyboard_nPoll
(JNIEnv * env, jclass clazz, jint buf)
{
2002-11-15 05:40:55 -05:00
XEvent event;
2002-11-21 10:05:51 -05:00
int state;
updateKeyboardGrab();
2002-11-15 05:40:55 -05:00
while (XCheckMaskEvent(disp, KeyPressMask | KeyReleaseMask, &event)) {
unsigned char keycode = getKeycode(&event);
2002-11-15 05:40:55 -05:00
if (event.type == KeyPress) {
2002-11-21 10:05:51 -05:00
state = 1;
2002-11-15 05:40:55 -05:00
} else if (event.type == KeyRelease) {
2002-11-21 10:05:51 -05:00
state = 0;
2002-11-15 05:40:55 -05:00
} else
assert(0);
2002-11-21 10:05:51 -05:00
key_buf[keycode] = state;
putEventElement(keycode, state);
2002-11-15 05:40:55 -05:00
}
memcpy((unsigned char*)buf, key_buf, KEYBOARD_SIZE*sizeof(unsigned char));
}
/*
* Class: org_lwjgl_input_Keyboard
* Method: nRead
* Signature: (I)V
*/
JNIEXPORT jint JNICALL Java_org_lwjgl_input_Keyboard_nRead
(JNIEnv * env, jclass clazz, jint keys)
{
XEvent event;
int count = 0;
int buf_count = 0;
int state;
input_event *input_ev;
unsigned char *result_buf = (unsigned char *)keys;
updateKeyboardGrab();
while ((input_ev = nextEventElement()) != NULL) {
count++;
// printf("Reading a key %d %d count %d\n", (int)input_ev->keycode, (int)input_ev->state, count);
result_buf[buf_count++] = input_ev->keycode;
result_buf[buf_count++] = input_ev->state;
if (buf_count >= KEYBOARD_BUFFER_SIZE * 2)
return count;
}
while (XCheckMaskEvent(disp, KeyPressMask | KeyReleaseMask, &event)) {
count++;
unsigned char keycode = getKeycode(&event);
if (event.type == KeyPress) {
state = 1;
} else if (event.type == KeyRelease) {
state = 0;
} else
assert(0);
key_buf[keycode] = state;
result_buf[buf_count++] = keycode;
result_buf[buf_count++] = state;
if (buf_count >= KEYBOARD_BUFFER_SIZE * 2)
return count;
}
return count;
2002-11-15 05:40:55 -05:00
}
/*
* Class: org_lwjgl_input_Keyboard
* Method: nEnableBuffer
* Signature: ()I
*/
JNIEXPORT jint JNICALL Java_org_lwjgl_input_Keyboard_nEnableBuffer
(JNIEnv * env, jclass clazz)
{
2002-11-21 10:05:51 -05:00
jobject newBuffer = env->NewDirectByteBuffer(&readBuffer, KEYBOARD_BUFFER_SIZE * 2);
2002-11-19 03:50:57 -05:00
env->SetStaticObjectField(clazz, fid_readBuffer, newBuffer);
env->SetStaticIntField(clazz, fid_readBufferAddress, (jint) (&readBuffer));
2002-11-15 05:40:55 -05:00
return KEYBOARD_BUFFER_SIZE;
}