forked from etc/pineapple-src
276 lines
7.3 KiB
C
276 lines
7.3 KiB
C
|
/*
|
||
|
Copyright (C) 1997-2021 Sam Lantinga <slouken@libsdl.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.
|
||
|
*/
|
||
|
|
||
|
/* Simple program: Loop, watching keystrokes
|
||
|
Note that you need to call SDL_PollEvent() or SDL_WaitEvent() to
|
||
|
pump the event loop and catch keystrokes.
|
||
|
*/
|
||
|
|
||
|
#include <stdio.h>
|
||
|
#include <stdlib.h>
|
||
|
#include <string.h>
|
||
|
|
||
|
#ifdef __EMSCRIPTEN__
|
||
|
#include <emscripten/emscripten.h>
|
||
|
#endif
|
||
|
|
||
|
#include "SDL.h"
|
||
|
|
||
|
int done;
|
||
|
|
||
|
/* Call this instead of exit(), so we can clean up SDL: atexit() is evil. */
|
||
|
static void
|
||
|
quit(int rc)
|
||
|
{
|
||
|
SDL_Quit();
|
||
|
exit(rc);
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
print_string(char **text, size_t *maxlen, const char *fmt, ...)
|
||
|
{
|
||
|
int len;
|
||
|
va_list ap;
|
||
|
|
||
|
va_start(ap, fmt);
|
||
|
len = SDL_vsnprintf(*text, *maxlen, fmt, ap);
|
||
|
if (len > 0) {
|
||
|
*text += len;
|
||
|
if ( ((size_t) len) < *maxlen ) {
|
||
|
*maxlen -= (size_t) len;
|
||
|
} else {
|
||
|
*maxlen = 0;
|
||
|
}
|
||
|
}
|
||
|
va_end(ap);
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
print_modifiers(char **text, size_t *maxlen)
|
||
|
{
|
||
|
int mod;
|
||
|
print_string(text, maxlen, " modifiers:");
|
||
|
mod = SDL_GetModState();
|
||
|
if (!mod) {
|
||
|
print_string(text, maxlen, " (none)");
|
||
|
return;
|
||
|
}
|
||
|
if (mod & KMOD_LSHIFT)
|
||
|
print_string(text, maxlen, " LSHIFT");
|
||
|
if (mod & KMOD_RSHIFT)
|
||
|
print_string(text, maxlen, " RSHIFT");
|
||
|
if (mod & KMOD_LCTRL)
|
||
|
print_string(text, maxlen, " LCTRL");
|
||
|
if (mod & KMOD_RCTRL)
|
||
|
print_string(text, maxlen, " RCTRL");
|
||
|
if (mod & KMOD_LALT)
|
||
|
print_string(text, maxlen, " LALT");
|
||
|
if (mod & KMOD_RALT)
|
||
|
print_string(text, maxlen, " RALT");
|
||
|
if (mod & KMOD_LGUI)
|
||
|
print_string(text, maxlen, " LGUI");
|
||
|
if (mod & KMOD_RGUI)
|
||
|
print_string(text, maxlen, " RGUI");
|
||
|
if (mod & KMOD_NUM)
|
||
|
print_string(text, maxlen, " NUM");
|
||
|
if (mod & KMOD_CAPS)
|
||
|
print_string(text, maxlen, " CAPS");
|
||
|
if (mod & KMOD_MODE)
|
||
|
print_string(text, maxlen, " MODE");
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
PrintModifierState()
|
||
|
{
|
||
|
char message[512];
|
||
|
char *spot;
|
||
|
size_t left;
|
||
|
|
||
|
spot = message;
|
||
|
left = sizeof(message);
|
||
|
|
||
|
print_modifiers(&spot, &left);
|
||
|
SDL_Log("Initial state:%s\n", message);
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
PrintKey(SDL_Keysym * sym, SDL_bool pressed, SDL_bool repeat)
|
||
|
{
|
||
|
char message[512];
|
||
|
char *spot;
|
||
|
size_t left;
|
||
|
|
||
|
spot = message;
|
||
|
left = sizeof(message);
|
||
|
|
||
|
/* Print the keycode, name and state */
|
||
|
if (sym->sym) {
|
||
|
print_string(&spot, &left,
|
||
|
"Key %s: scancode %d = %s, keycode 0x%08X = %s ",
|
||
|
pressed ? "pressed " : "released",
|
||
|
sym->scancode,
|
||
|
SDL_GetScancodeName(sym->scancode),
|
||
|
sym->sym, SDL_GetKeyName(sym->sym));
|
||
|
} else {
|
||
|
print_string(&spot, &left,
|
||
|
"Unknown Key (scancode %d = %s) %s ",
|
||
|
sym->scancode,
|
||
|
SDL_GetScancodeName(sym->scancode),
|
||
|
pressed ? "pressed " : "released");
|
||
|
}
|
||
|
print_modifiers(&spot, &left);
|
||
|
if (repeat) {
|
||
|
print_string(&spot, &left, " (repeat)");
|
||
|
}
|
||
|
SDL_Log("%s\n", message);
|
||
|
fflush(stderr);
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
PrintText(char *eventtype, char *text)
|
||
|
{
|
||
|
char *spot, expanded[1024];
|
||
|
|
||
|
expanded[0] = '\0';
|
||
|
for ( spot = text; *spot; ++spot )
|
||
|
{
|
||
|
size_t length = SDL_strlen(expanded);
|
||
|
SDL_snprintf(expanded + length, sizeof(expanded) - length, "\\x%.2x", (unsigned char)*spot);
|
||
|
}
|
||
|
SDL_Log("%s Text (%s): \"%s%s\"\n", eventtype, expanded, *text == '"' ? "\\" : "", text);
|
||
|
}
|
||
|
|
||
|
void
|
||
|
loop()
|
||
|
{
|
||
|
SDL_Event event;
|
||
|
/* Check for events */
|
||
|
/*SDL_WaitEvent(&event); emscripten does not like waiting*/
|
||
|
|
||
|
fprintf(stderr, "starting loop\n"); fflush(stderr);
|
||
|
// while (SDL_PollEvent(&event)) {
|
||
|
while (!done && SDL_WaitEvent(&event)) {
|
||
|
fprintf(stderr, "got event type: %d\n", event.type); fflush(stderr);
|
||
|
switch (event.type) {
|
||
|
case SDL_KEYDOWN:
|
||
|
case SDL_KEYUP:
|
||
|
PrintKey(&event.key.keysym, (event.key.state == SDL_PRESSED) ? SDL_TRUE : SDL_FALSE, (event.key.repeat) ? SDL_TRUE : SDL_FALSE);
|
||
|
break;
|
||
|
case SDL_TEXTEDITING:
|
||
|
PrintText("EDIT", event.text.text);
|
||
|
break;
|
||
|
case SDL_TEXTINPUT:
|
||
|
PrintText("INPUT", event.text.text);
|
||
|
break;
|
||
|
case SDL_MOUSEBUTTONDOWN:
|
||
|
/* Left button quits the app, other buttons toggles text input */
|
||
|
fprintf(stderr, "mouse button down button: %d (LEFT=%d)\n", event.button.button, SDL_BUTTON_LEFT); fflush(stderr);
|
||
|
if (event.button.button == SDL_BUTTON_LEFT) {
|
||
|
done = 1;
|
||
|
} else {
|
||
|
if (SDL_IsTextInputActive()) {
|
||
|
SDL_Log("Stopping text input\n");
|
||
|
SDL_StopTextInput();
|
||
|
} else {
|
||
|
SDL_Log("Starting text input\n");
|
||
|
SDL_StartTextInput();
|
||
|
}
|
||
|
}
|
||
|
break;
|
||
|
case SDL_QUIT:
|
||
|
done = 1;
|
||
|
break;
|
||
|
default:
|
||
|
break;
|
||
|
}
|
||
|
fprintf(stderr, "waiting new event\n"); fflush(stderr);
|
||
|
}
|
||
|
fprintf(stderr, "exiting event loop\n"); fflush(stderr);
|
||
|
#ifdef __EMSCRIPTEN__
|
||
|
if (done) {
|
||
|
emscripten_cancel_main_loop();
|
||
|
}
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
/* Very simple thread - counts 0 to 9 delaying 50ms between increments */
|
||
|
static int SDLCALL ping_thread(void *ptr)
|
||
|
{
|
||
|
int cnt;
|
||
|
SDL_Event sdlevent;
|
||
|
memset(&sdlevent, 0 , sizeof(SDL_Event));
|
||
|
for (cnt = 0; cnt < 10; ++cnt) {
|
||
|
fprintf(stderr, "sending event (%d/%d) from thread.\n", cnt + 1, 10); fflush(stderr);
|
||
|
sdlevent.type = SDL_KEYDOWN;
|
||
|
sdlevent.key.keysym.sym = SDLK_1;
|
||
|
SDL_PushEvent(&sdlevent);
|
||
|
SDL_Delay(1000 + rand() % 1000);
|
||
|
}
|
||
|
return cnt;
|
||
|
}
|
||
|
|
||
|
int
|
||
|
main(int argc, char *argv[])
|
||
|
{
|
||
|
SDL_Window *window;
|
||
|
SDL_Thread *thread;
|
||
|
|
||
|
/* Enable standard application logging */
|
||
|
SDL_LogSetPriority(SDL_LOG_CATEGORY_APPLICATION, SDL_LOG_PRIORITY_INFO);
|
||
|
|
||
|
/* Initialize SDL */
|
||
|
if (SDL_Init(SDL_INIT_VIDEO) < 0) {
|
||
|
SDL_LogError(SDL_LOG_CATEGORY_APPLICATION, "Couldn't initialize SDL: %s\n", SDL_GetError());
|
||
|
return (1);
|
||
|
}
|
||
|
|
||
|
/* Set 640x480 video mode */
|
||
|
window = SDL_CreateWindow("CheckKeys Test",
|
||
|
SDL_WINDOWPOS_CENTERED, SDL_WINDOWPOS_CENTERED,
|
||
|
640, 480, 0);
|
||
|
if (!window) {
|
||
|
SDL_LogError(SDL_LOG_CATEGORY_APPLICATION, "Couldn't create 640x480 window: %s\n",
|
||
|
SDL_GetError());
|
||
|
quit(2);
|
||
|
}
|
||
|
|
||
|
#if __IPHONEOS__
|
||
|
/* Creating the context creates the view, which we need to show keyboard */
|
||
|
SDL_GL_CreateContext(window);
|
||
|
#endif
|
||
|
|
||
|
SDL_StartTextInput();
|
||
|
|
||
|
/* Print initial modifier state */
|
||
|
SDL_PumpEvents();
|
||
|
PrintModifierState();
|
||
|
|
||
|
/* Watch keystrokes */
|
||
|
done = 0;
|
||
|
|
||
|
thread = SDL_CreateThread(ping_thread, "PingThread", (void *)NULL);
|
||
|
|
||
|
#ifdef __EMSCRIPTEN__
|
||
|
emscripten_set_main_loop(loop, 0, 1);
|
||
|
#else
|
||
|
while (!done) {
|
||
|
loop();
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
SDL_WaitThread(thread, NULL);
|
||
|
SDL_Quit();
|
||
|
return (0);
|
||
|
}
|
||
|
|
||
|
/* vi: set ts=4 sw=4 expandtab: */
|