diff options
author | Mitya Selivanov <automainint@guattari.tech> | 2024-08-12 14:37:58 +0200 |
---|---|---|
committer | Mitya Selivanov <automainint@guattari.tech> | 2024-08-12 14:37:58 +0200 |
commit | b204f057653f13ed2a578fec70a23ae333641513 (patch) | |
tree | 1f5ba4a08431a49d5342d16cbd1f7eb2216d7abb | |
parent | da9d431435211378205817199aaf24428697e3b3 (diff) | |
download | reduced_system_layer-b204f057653f13ed2a578fec70a23ae333641513.zip |
Update code
-rwxr-xr-x | examples/gravity.c | 890 | ||||
-rwxr-xr-x | examples/landscape.c | 406 | ||||
-rwxr-xr-x | reduced_system_layer.c | 431 |
3 files changed, 1395 insertions, 332 deletions
diff --git a/examples/gravity.c b/examples/gravity.c index 824558b..7c6faf0 100755 --- a/examples/gravity.c +++ b/examples/gravity.c @@ -3,6 +3,31 @@ #/ #/ gravity.c #/ +#/ This is a reduced system layer. +#/ It allows you to create a window, draw graphics in it, handle +#/ input events. +#/ +#/ ---------------------------------------------------------------- +#/ +#/ DESIGN PRINCIPLES +#/ +#/ - Minimalistic feature set. For graphics, you have access to the +#/ pixel buffer, and that's it. +#/ +#/ - No implicit control flow. No callbacks. You write your own +#/ main and call everything explicitly. But the number of things +#/ you have to call to do something is as little as possible. +#/ +#/ - Optimized to use in a single source file. +#/ Installation process? Ctrl+C, Ctrl+V, done. +#/ +#/ If you have an idea how to reduce the feature set further, +#/ let me know! +#/ +#/ ---------------------------------------------------------------- +#/ +#/ (C) 2024 Mitya Selivanov <guattari.tech>, MIT License +#/ #/ ================================================================ #/ #/ Self-compilation shell script @@ -16,6 +41,7 @@ gcc \ -Wno-unused-variable \ -Wno-unused-but-set-variable \ -Wno-unused-parameter \ + -Wno-overlength-strings \ -O3 \ -fsanitize=undefined,address,leak -mshstk \ -lX11 -lm \ @@ -40,6 +66,7 @@ typedef unsigned short u16; typedef unsigned u32; typedef unsigned long long u64; typedef char c8; +typedef int c32; typedef signed char b8; typedef float f32; typedef double f64; @@ -51,28 +78,117 @@ typedef double f64; // ================================================================ enum { - KEY_NONE, BUTTON_LEFT, BUTTON_MIDDLE, BUTTON_RIGHT, BUTTON_X1, BUTTON_X2, KEY_LEFT, KEY_RIGHT, KEY_UP, KEY_DOWN, KEY_1, KEY_2, KEY_3, KEY_4, KEY_5, KEY_6, KEY_7, KEY_8, KEY_9, KEY_0, KEY_A, KEY_B, KEY_C, KEY_D, KEY_E, KEY_F, KEY_G, KEY_H, KEY_I, KEY_J, KEY_K, KEY_L, KEY_M, KEY_N, KEY_O, KEY_P, KEY_Q, KEY_R, KEY_S, KEY_T, KEY_U, KEY_V, KEY_W, KEY_X, KEY_Y, KEY_Z, KEY_SPACE, KEY_BRACE_LEFT, KEY_BRACE_RIGHT, KEY_SEMICOLON, KEY_APOSTROPHE, KEY_TILDE, KEY_BACKSLASH, KEY_COMMA, KEY_PERIOD, KEY_SLASH, KEY_F1, KEY_F2, KEY_F3, KEY_F4, KEY_F5, KEY_F6, KEY_F7, KEY_F8, KEY_F9, KEY_F10, KEY_F11, KEY_F12, KEY_LCTRL, KEY_RCTRL, KEY_LSHIFT, KEY_RSHIFT, KEY_LALT, KEY_RALT, KEY_ESCAPE, KEY_BACKSPACE, KEY_TAB, KEY_ENTER, KEY_PRINTSCREEN, KEY_DELETE, KEY_PAUSE, KEY_INSERT, KEY_HOME, KEY_END, KEY_PAGEUP, KEY_PAGEDOWN, KEY_KP_0, KEY_KP_1, KEY_KP_2, KEY_KP_3, KEY_KP_4, KEY_KP_5, KEY_KP_6, KEY_KP_7, KEY_KP_8, KEY_KP_9, KEY_KP_ENTER, KEY_KP_DIVIDE, KEY_KP_MULTIPLY, KEY_KP_PLUS, KEY_KP_MINUS, KEY_KP_DECIMAL, KEY_KP_SEPARATOR, + MAX_NUM_PIXELS = 10 * 1024 * 1024, + MAX_INPUT_SIZE = 256, + MAX_CLIPBOARD_SIZE = 10 * 1024 * 1024, + MAX_NUM_AUDIO_SAMPLES = 0, + MAX_NUM_SOCKETS = 0, + + AUDIO_NUM_CHANNELS = 2, + AUDIO_SAMPLE_RATE = 44100, + + IPv4 = 1, + IPv6 = 2, + + KEY_LEFT = 128, + KEY_RIGHT, + KEY_UP, + KEY_DOWN, + KEY_LCTRL, + KEY_RCTRL, + KEY_LSHIFT, + KEY_RSHIFT, + KEY_LALT, + KEY_RALT, + KEY_ESCAPE, + KEY_PRINTSCREEN, + KEY_DELETE, + KEY_PAUSE, + KEY_INSERT, + KEY_HOME, + KEY_END, + KEY_PAGEUP, + KEY_PAGEDOWN, + BUTTON_LEFT, + BUTTON_MIDDLE, + BUTTON_RIGHT, + MOD_CTRL, + MOD_SHIFT, + MOD_ALT, + MOD_CAPS, + MOD_NUM, + MOD_SCROLL, + KEY_F_, + KEY_KP_ = KEY_F_ + 64, }; typedef struct { - c8 *title; - i32 frame_width; - i32 frame_height; - u32 *pixels; - b8 done; - b8 has_focus; - b8 has_cursor; - i32 cursor_x; - i32 cursor_y; - b8 key_down[512]; - b8 key_pressed[512]; + b8 ctrl : 1; + b8 shift : 1; + b8 alt : 1; + b8 caps : 1; + b8 num : 1; + b8 scroll : 1; + u16 key; + c32 c; +} Input_Key; + +typedef struct { + c8 * title; + i32 frame_width; + i32 frame_height; + u32 * pixels; + i64 input_size; + Input_Key *input; + i64 clipboard_size; + c8 * clipboard; + b8 done; + b8 has_focus; + b8 has_cursor; + i32 cursor_x; + i32 cursor_y; + i32 cursor_dx; + i32 cursor_dy; + i64 wheel_dy; + b8 key_down[512]; + b8 key_pressed[512]; } Platform; -Platform platform = {0}; +typedef struct { + u16 type; + u16 port; + union { + u8 v4_address[4]; + u8 v6_address[16]; + }; +} IP_Address; + +typedef i64 (*Thread_Proc)(void *user_data); +// UTF-8 +i32 utf8_size(c32 c); +c32 utf8_read(i64 len, c8 *s); +i32 utf8_write(c32 c, c8 *buffer); + +// Window void p_init(void); void p_cleanup(void); i32 p_handle_events(void); +i32 p_wait_events(void); +void p_render_frame(void); + +// Clipboard +void p_clipboard_write(i64 size, c8 *data); + +// Sound +void p_handle_audio(i64 time_elapsed); +void p_queue_sound(i64 delay, i64 num_samples, f32 *samples); + +// UDP sockets +i64 p_recv(u16 slot, IP_Address address, i64 size, u8 *data, IP_Address *remote_address); +i64 p_send(u16 slot, IP_Address address, i64 size, u8 *data); + +Platform platform = {0}; // ================================================================ // @@ -123,6 +239,9 @@ u32 u32_from_rgb(f32 red, f32 green, f32 blue) { } i32 main(i32 argc, c8 **argv) { + (void) argc; + (void) argv; + u32 background = u32_from_rgb(.0f, .0f, .08f); platform = (Platform) { @@ -139,7 +258,7 @@ i32 main(i32 argc, c8 **argv) { i64 time_0 = time_milliseconds(); while (!platform.done) { - i32 num_events = p_handle_events(); + p_handle_events(); i64 time_elapsed = time_milliseconds() - time_0; time_0 += time_elapsed; @@ -225,6 +344,8 @@ i32 main(i32 argc, c8 **argv) { } } } + + p_render_frame(); } p_cleanup(); @@ -233,197 +354,331 @@ i32 main(i32 argc, c8 **argv) { // ================================================================ // -// LINUX X11 PLATFORM IMPLEMENTATION +// PLATFORM IMPLEMENTATION +// +// ---------------------------------------------------------------- +// +// TODO +// - Sound +// - Sockets +// - Clipboard daemon +// +// ALSA +// https://www.alsa-project.org/alsa-doc/alsa-lib/_2test_2pcm_min_8c-example.html +// +// ================================================================ + +#include <assert.h> +#include <string.h> + +c32 utf8_read(i64 len, c8 *s) { + if (len >= 1 && + (s[0] & 0x80) == 0) + return s[0]; + if (len >= 2 && + (s[0] & 0xe0) == 0xc0 && + (s[1] & 0xc0) == 0x80) + return (s[1] & 0x3f) + | ((s[0] & 0x1f) << 6); + if (len >= 3 && + (s[0] & 0xf0) == 0xe0 && + (s[1] & 0xc0) == 0x80 && + (s[2] & 0xc0) == 0x80) + return (s[2] & 0x3f) + | ((s[1] & 0x3f) << 6) + | ((s[0] & 0x0f) << 12); + if (len >= 4 && + (s[0] & 0xf8) == 0xf0 && + (s[1] & 0xc0) == 0x80 && + (s[2] & 0xc0) == 0x80 && + (s[3] & 0xc0) == 0x80) + return (s[3] & 0x3f) + | ((s[2] & 0x3f) << 6) + | ((s[1] & 0x3f) << 12) + | ((s[0] & 0x07) << 18); + return 0; +} + +i32 utf8_size(c32 c) { + if ((c & 0x00007f) == c) return 1; + if ((c & 0x0007ff) == c) return 2; + if ((c & 0x00ffff) == c) return 3; + if ((c & 0x1fffff) == c) return 4; + return 0; +} + +i32 utf8_write(c32 c, c8 *buffer) { + if ((c & 0x7f) == c) { + buffer[0] = (c8) c; + return 1; + } + + if ((c & 0x7ff) == c) { + buffer[0] = 0xc0 | ((c >> 6) & 0x1f); + buffer[1] = 0x80 | ( c & 0x3f); + return 2; + } + + if ((c & 0xffff) == c) { + buffer[0] = 0xc0 | ((c >> 12) & 0x0f); + buffer[1] = 0x80 | ((c >> 6) & 0x3f); + buffer[2] = 0x80 | ( c & 0x3f); + return 3; + } + + if ((c & 0x1fffff) == c) { + buffer[0] = 0xc0 | ((c >> 18) & 0x07); + buffer[1] = 0x80 | ((c >> 12) & 0x3f); + buffer[2] = 0x80 | ((c >> 6) & 0x3f); + buffer[3] = 0x80 | ( c & 0x3f); + return 4; + } + + return 0; +} + +// ================================================================ +// +// UDP sockets // // ================================================================ +i64 p_recv(u16 slot, IP_Address address, i64 size, u8 *data, IP_Address *remote_address) { + (void) slot; + (void) address; + (void) size; + (void) data; + (void) remote_address; + assert(0); + return 0; +} + +i64 p_send(u16 slot, IP_Address address, i64 size, u8 *data) { + (void) slot; + (void) address; + (void) size; + (void) data; + assert(0); + return 0; +} + +// ================================================================ +// +// X11 +// +// ================================================================ + +#ifdef __linux__ + #include <X11/Xlib.h> #include <X11/Xutil.h> +#include <X11/Xatom.h> +#include <sched.h> +#include <time.h> -static i16 key_table[512]; -static u32 buffer[10 * 1024 * 1024]; - -static XImage image = { - .xoffset = 0, - .format = ZPixmap, - .data = (c8 *) buffer, - .byte_order = LSBFirst, - .bitmap_unit = 32, - .bitmap_bit_order = LSBFirst, - .bitmap_pad = 32, - .bits_per_pixel = 32, - .red_mask = 0xff0000, - .green_mask = 0x00ff00, - .blue_mask = 0x0000ff, -}; - -static Display * display; -static i32 screen; -static i32 depth; -static Visual * visual; -static GC gc; -static Window window; -static Pixmap pixmap; -static Atom wm_delete_window; -static XEvent ev; -static XWindowAttributes attrs; +static i16 _key_table[512] = {0}; +static u32 _buffer[MAX_NUM_PIXELS] = {0}; +static Input_Key _input[MAX_INPUT_SIZE] = {0}; +static c8 _clipboard_buffer[MAX_CLIPBOARD_SIZE] = {0}; +static XImage _image = {0}; +static Display * _display = NULL; +static GC _gc = NULL; +static XIM _im = NULL; +static XIC _ic = NULL; +static Window _window = 0; +static Atom _wm_delete_window = 0; +static Atom _clipboard = 0; +static Atom _targets = 0; +static Atom _utf8_string = 0; +static Atom _target = None; void p_init(void) { - display = XOpenDisplay(NULL); - assert(display != NULL); - - key_table[XKeysymToKeycode(display, XK_Left)] = KEY_LEFT; - key_table[XKeysymToKeycode(display, XK_Right)] = KEY_RIGHT; - key_table[XKeysymToKeycode(display, XK_Up)] = KEY_UP; - key_table[XKeysymToKeycode(display, XK_Down)] = KEY_DOWN; - key_table[XKeysymToKeycode(display, XK_1)] = KEY_1; - key_table[XKeysymToKeycode(display, XK_2)] = KEY_2; - key_table[XKeysymToKeycode(display, XK_3)] = KEY_3; - key_table[XKeysymToKeycode(display, XK_4)] = KEY_4; - key_table[XKeysymToKeycode(display, XK_5)] = KEY_5; - key_table[XKeysymToKeycode(display, XK_6)] = KEY_6; - key_table[XKeysymToKeycode(display, XK_7)] = KEY_7; - key_table[XKeysymToKeycode(display, XK_8)] = KEY_8; - key_table[XKeysymToKeycode(display, XK_9)] = KEY_9; - key_table[XKeysymToKeycode(display, XK_0)] = KEY_0; - key_table[XKeysymToKeycode(display, XK_A)] = KEY_A; - key_table[XKeysymToKeycode(display, XK_B)] = KEY_B; - key_table[XKeysymToKeycode(display, XK_C)] = KEY_C; - key_table[XKeysymToKeycode(display, XK_D)] = KEY_D; - key_table[XKeysymToKeycode(display, XK_E)] = KEY_E; - key_table[XKeysymToKeycode(display, XK_F)] = KEY_F; - key_table[XKeysymToKeycode(display, XK_G)] = KEY_G; - key_table[XKeysymToKeycode(display, XK_H)] = KEY_H; - key_table[XKeysymToKeycode(display, XK_I)] = KEY_I; - key_table[XKeysymToKeycode(display, XK_J)] = KEY_J; - key_table[XKeysymToKeycode(display, XK_K)] = KEY_K; - key_table[XKeysymToKeycode(display, XK_L)] = KEY_L; - key_table[XKeysymToKeycode(display, XK_M)] = KEY_M; - key_table[XKeysymToKeycode(display, XK_N)] = KEY_N; - key_table[XKeysymToKeycode(display, XK_O)] = KEY_O; - key_table[XKeysymToKeycode(display, XK_P)] = KEY_P; - key_table[XKeysymToKeycode(display, XK_Q)] = KEY_Q; - key_table[XKeysymToKeycode(display, XK_R)] = KEY_R; - key_table[XKeysymToKeycode(display, XK_S)] = KEY_S; - key_table[XKeysymToKeycode(display, XK_T)] = KEY_T; - key_table[XKeysymToKeycode(display, XK_V)] = KEY_V; - key_table[XKeysymToKeycode(display, XK_W)] = KEY_W; - key_table[XKeysymToKeycode(display, XK_X)] = KEY_X; - key_table[XKeysymToKeycode(display, XK_Y)] = KEY_Y; - key_table[XKeysymToKeycode(display, XK_Z)] = KEY_Z; - key_table[XKeysymToKeycode(display, XK_space)] = KEY_SPACE; - key_table[XKeysymToKeycode(display, XK_braceleft)] = KEY_BRACE_LEFT; - key_table[XKeysymToKeycode(display, XK_braceright)] = KEY_BRACE_RIGHT; - key_table[XKeysymToKeycode(display, XK_colon)] = KEY_SEMICOLON; - key_table[XKeysymToKeycode(display, XK_quotedbl)] = KEY_APOSTROPHE; - key_table[XKeysymToKeycode(display, XK_asciitilde)] = KEY_TILDE; - key_table[XKeysymToKeycode(display, XK_backslash)] = KEY_BACKSLASH; - key_table[XKeysymToKeycode(display, XK_comma)] = KEY_COMMA; - key_table[XKeysymToKeycode(display, XK_greater)] = KEY_PERIOD; - key_table[XKeysymToKeycode(display, XK_question)] = KEY_SLASH; - key_table[XKeysymToKeycode(display, XK_F1)] = KEY_F1; - key_table[XKeysymToKeycode(display, XK_F2)] = KEY_F2; - key_table[XKeysymToKeycode(display, XK_F3)] = KEY_F3; - key_table[XKeysymToKeycode(display, XK_F4)] = KEY_F4; - key_table[XKeysymToKeycode(display, XK_F5)] = KEY_F5; - key_table[XKeysymToKeycode(display, XK_F6)] = KEY_F6; - key_table[XKeysymToKeycode(display, XK_F7)] = KEY_F7; - key_table[XKeysymToKeycode(display, XK_F8)] = KEY_F8; - key_table[XKeysymToKeycode(display, XK_F9)] = KEY_F9; - key_table[XKeysymToKeycode(display, XK_F10)] = KEY_F10; - key_table[XKeysymToKeycode(display, XK_F11)] = KEY_F11; - key_table[XKeysymToKeycode(display, XK_F12)] = KEY_F12; - key_table[XKeysymToKeycode(display, XK_Control_L)] = KEY_LCTRL; - key_table[XKeysymToKeycode(display, XK_Control_R)] = KEY_RCTRL; - key_table[XKeysymToKeycode(display, XK_Shift_L)] = KEY_LSHIFT; - key_table[XKeysymToKeycode(display, XK_Shift_R)] = KEY_RSHIFT; - key_table[XKeysymToKeycode(display, XK_Alt_L)] = KEY_LALT; - key_table[XKeysymToKeycode(display, XK_Alt_R)] = KEY_RALT; - key_table[XKeysymToKeycode(display, XK_Escape)] = KEY_ESCAPE; - key_table[XKeysymToKeycode(display, XK_BackSpace)] = KEY_BACKSPACE; - key_table[XKeysymToKeycode(display, XK_Tab)] = KEY_TAB; - key_table[XKeysymToKeycode(display, XK_Return)] = KEY_ENTER; - key_table[XKeysymToKeycode(display, XK_Print)] = KEY_PRINTSCREEN; - key_table[XKeysymToKeycode(display, XK_Delete)] = KEY_DELETE; - key_table[XKeysymToKeycode(display, XK_Pause)] = KEY_PAUSE; - key_table[XKeysymToKeycode(display, XK_Insert)] = KEY_INSERT; - key_table[XKeysymToKeycode(display, XK_Home)] = KEY_HOME; - key_table[XKeysymToKeycode(display, XK_End)] = KEY_END; - key_table[XKeysymToKeycode(display, XK_Page_Up)] = KEY_PAGEUP; - key_table[XKeysymToKeycode(display, XK_Page_Down)] = KEY_PAGEDOWN; - key_table[XKeysymToKeycode(display, XK_KP_0)] = KEY_KP_0; - key_table[XKeysymToKeycode(display, XK_KP_1)] = KEY_KP_1; - key_table[XKeysymToKeycode(display, XK_KP_2)] = KEY_KP_2; - key_table[XKeysymToKeycode(display, XK_KP_3)] = KEY_KP_3; - key_table[XKeysymToKeycode(display, XK_KP_4)] = KEY_KP_4; - key_table[XKeysymToKeycode(display, XK_KP_5)] = KEY_KP_5; - key_table[XKeysymToKeycode(display, XK_KP_6)] = KEY_KP_6; - key_table[XKeysymToKeycode(display, XK_KP_7)] = KEY_KP_7; - key_table[XKeysymToKeycode(display, XK_KP_8)] = KEY_KP_8; - key_table[XKeysymToKeycode(display, XK_KP_9)] = KEY_KP_9; - key_table[XKeysymToKeycode(display, XK_KP_Enter)] = KEY_KP_ENTER; - key_table[XKeysymToKeycode(display, XK_KP_Divide)] = KEY_KP_DIVIDE; - key_table[XKeysymToKeycode(display, XK_KP_Multiply)] = KEY_KP_MULTIPLY; - key_table[XKeysymToKeycode(display, XK_KP_Add)] = KEY_KP_PLUS; - key_table[XKeysymToKeycode(display, XK_KP_Subtract)] = KEY_KP_MINUS; - key_table[XKeysymToKeycode(display, XK_KP_Decimal)] = KEY_KP_DECIMAL; - key_table[XKeysymToKeycode(display, XK_KP_Separator)] = KEY_KP_SEPARATOR; - - screen = DefaultScreen(display); - depth = DefaultDepth (display, screen); - visual = DefaultVisual(display, screen); - gc = DefaultGC (display, screen); - - XSetGraphicsExposures(display, gc, False); - - i32 display_width = DisplayWidth (display, screen); - i32 display_height = DisplayHeight(display, screen); - - i32 x = (display_width - platform.frame_width) / 2; - i32 y = (display_height - platform.frame_height) / 2; - - window = XCreateWindow(display, XDefaultRootWindow(display), x, y, platform.frame_width, platform.frame_height, 0, depth, InputOutput, visual, CWEventMask, &(XSetWindowAttributes) { .event_mask = KeyPressMask | KeyReleaseMask | ButtonPressMask | ButtonReleaseMask | EnterWindowMask | LeaveWindowMask | PointerMotionMask | VisibilityChangeMask | FocusChangeMask | StructureNotifyMask | SubstructureNotifyMask, }); - - platform.pixels = buffer; - - image.width = platform.frame_width; - image.height = platform.frame_height; - image.depth = depth; - image.bytes_per_line = 4 * platform.frame_width; - - XInitImage(&image); - - pixmap = XCreatePixmap(display, window, platform.frame_width, platform.frame_height, depth); - wm_delete_window = XInternAtom(display, "WM_DELETE_WINDOW", False); - - XSetWMProtocols(display, window, &wm_delete_window, 1); - - XStoreName(display, window, platform.title); - XMapWindow(display, window); + _display = XOpenDisplay(NULL); + assert(_display != NULL); + + _key_table[XKeysymToKeycode(_display, XK_Left)] = KEY_LEFT; + _key_table[XKeysymToKeycode(_display, XK_Right)] = KEY_RIGHT; + _key_table[XKeysymToKeycode(_display, XK_Up)] = KEY_UP; + _key_table[XKeysymToKeycode(_display, XK_Down)] = KEY_DOWN; + _key_table[XKeysymToKeycode(_display, XK_1)] = '1'; + _key_table[XKeysymToKeycode(_display, XK_2)] = '2'; + _key_table[XKeysymToKeycode(_display, XK_3)] = '3'; + _key_table[XKeysymToKeycode(_display, XK_4)] = '4'; + _key_table[XKeysymToKeycode(_display, XK_5)] = '5'; + _key_table[XKeysymToKeycode(_display, XK_6)] = '6'; + _key_table[XKeysymToKeycode(_display, XK_7)] = '7'; + _key_table[XKeysymToKeycode(_display, XK_8)] = '8'; + _key_table[XKeysymToKeycode(_display, XK_9)] = '9'; + _key_table[XKeysymToKeycode(_display, XK_0)] = '0'; + _key_table[XKeysymToKeycode(_display, XK_A)] = 'a'; + _key_table[XKeysymToKeycode(_display, XK_B)] = 'b'; + _key_table[XKeysymToKeycode(_display, XK_C)] = 'c'; + _key_table[XKeysymToKeycode(_display, XK_D)] = 'd'; + _key_table[XKeysymToKeycode(_display, XK_E)] = 'e'; + _key_table[XKeysymToKeycode(_display, XK_F)] = 'f'; + _key_table[XKeysymToKeycode(_display, XK_G)] = 'g'; + _key_table[XKeysymToKeycode(_display, XK_H)] = 'h'; + _key_table[XKeysymToKeycode(_display, XK_I)] = 'i'; + _key_table[XKeysymToKeycode(_display, XK_J)] = 'j'; + _key_table[XKeysymToKeycode(_display, XK_K)] = 'k'; + _key_table[XKeysymToKeycode(_display, XK_L)] = 'l'; + _key_table[XKeysymToKeycode(_display, XK_M)] = 'm'; + _key_table[XKeysymToKeycode(_display, XK_N)] = 'n'; + _key_table[XKeysymToKeycode(_display, XK_O)] = 'o'; + _key_table[XKeysymToKeycode(_display, XK_P)] = 'p'; + _key_table[XKeysymToKeycode(_display, XK_Q)] = 'q'; + _key_table[XKeysymToKeycode(_display, XK_R)] = 'r'; + _key_table[XKeysymToKeycode(_display, XK_S)] = 's'; + _key_table[XKeysymToKeycode(_display, XK_T)] = 't'; + _key_table[XKeysymToKeycode(_display, XK_U)] = 'u'; + _key_table[XKeysymToKeycode(_display, XK_V)] = 'v'; + _key_table[XKeysymToKeycode(_display, XK_W)] = 'w'; + _key_table[XKeysymToKeycode(_display, XK_X)] = 'x'; + _key_table[XKeysymToKeycode(_display, XK_Y)] = 'y'; + _key_table[XKeysymToKeycode(_display, XK_Z)] = 'z'; + _key_table[XKeysymToKeycode(_display, XK_space)] = ' '; + _key_table[XKeysymToKeycode(_display, XK_braceleft)] = '['; + _key_table[XKeysymToKeycode(_display, XK_braceright)] = ']'; + _key_table[XKeysymToKeycode(_display, XK_colon)] = ';'; + _key_table[XKeysymToKeycode(_display, XK_quotedbl)] = '\''; + _key_table[XKeysymToKeycode(_display, XK_asciitilde)] = '`'; + _key_table[XKeysymToKeycode(_display, XK_backslash)] = '\\'; + _key_table[XKeysymToKeycode(_display, XK_comma)] = ','; + _key_table[XKeysymToKeycode(_display, XK_greater)] = '.'; + _key_table[XKeysymToKeycode(_display, XK_question)] = '/'; + _key_table[XKeysymToKeycode(_display, XK_minus)] = '-'; + _key_table[XKeysymToKeycode(_display, XK_equal)] = '='; + _key_table[XKeysymToKeycode(_display, XK_F1)] = KEY_F_ + 1; + _key_table[XKeysymToKeycode(_display, XK_F2)] = KEY_F_ + 2; + _key_table[XKeysymToKeycode(_display, XK_F3)] = KEY_F_ + 3; + _key_table[XKeysymToKeycode(_display, XK_F4)] = KEY_F_ + 4; + _key_table[XKeysymToKeycode(_display, XK_F5)] = KEY_F_ + 5; + _key_table[XKeysymToKeycode(_display, XK_F6)] = KEY_F_ + 6; + _key_table[XKeysymToKeycode(_display, XK_F7)] = KEY_F_ + 7; + _key_table[XKeysymToKeycode(_display, XK_F8)] = KEY_F_ + 8; + _key_table[XKeysymToKeycode(_display, XK_F9)] = KEY_F_ + 9; + _key_table[XKeysymToKeycode(_display, XK_F10)] = KEY_F_ + 10; + _key_table[XKeysymToKeycode(_display, XK_F11)] = KEY_F_ + 11; + _key_table[XKeysymToKeycode(_display, XK_F12)] = KEY_F_ + 12; + _key_table[XKeysymToKeycode(_display, XK_Control_L)] = KEY_LCTRL; + _key_table[XKeysymToKeycode(_display, XK_Control_R)] = KEY_RCTRL; + _key_table[XKeysymToKeycode(_display, XK_Shift_L)] = KEY_LSHIFT; + _key_table[XKeysymToKeycode(_display, XK_Shift_R)] = KEY_RSHIFT; + _key_table[XKeysymToKeycode(_display, XK_Alt_L)] = KEY_LALT; + _key_table[XKeysymToKeycode(_display, XK_Alt_R)] = KEY_RALT; + _key_table[XKeysymToKeycode(_display, XK_Escape)] = KEY_ESCAPE; + _key_table[XKeysymToKeycode(_display, XK_BackSpace)] = '\b'; + _key_table[XKeysymToKeycode(_display, XK_Tab)] = '\t'; + _key_table[XKeysymToKeycode(_display, XK_Return)] = '\n'; + _key_table[XKeysymToKeycode(_display, XK_Print)] = KEY_PRINTSCREEN; + _key_table[XKeysymToKeycode(_display, XK_Delete)] = KEY_DELETE; + _key_table[XKeysymToKeycode(_display, XK_Pause)] = KEY_PAUSE; + _key_table[XKeysymToKeycode(_display, XK_Insert)] = KEY_INSERT; + _key_table[XKeysymToKeycode(_display, XK_Home)] = KEY_HOME; + _key_table[XKeysymToKeycode(_display, XK_End)] = KEY_END; + _key_table[XKeysymToKeycode(_display, XK_Page_Up)] = KEY_PAGEUP; + _key_table[XKeysymToKeycode(_display, XK_Page_Down)] = KEY_PAGEDOWN; + _key_table[XKeysymToKeycode(_display, XK_KP_0)] = KEY_KP_ + '0'; + _key_table[XKeysymToKeycode(_display, XK_KP_1)] = KEY_KP_ + '1'; + _key_table[XKeysymToKeycode(_display, XK_KP_2)] = KEY_KP_ + '2'; + _key_table[XKeysymToKeycode(_display, XK_KP_3)] = KEY_KP_ + '3'; + _key_table[XKeysymToKeycode(_display, XK_KP_4)] = KEY_KP_ + '4'; + _key_table[XKeysymToKeycode(_display, XK_KP_5)] = KEY_KP_ + '5'; + _key_table[XKeysymToKeycode(_display, XK_KP_6)] = KEY_KP_ + '6'; + _key_table[XKeysymToKeycode(_display, XK_KP_7)] = KEY_KP_ + '7'; + _key_table[XKeysymToKeycode(_display, XK_KP_8)] = KEY_KP_ + '8'; + _key_table[XKeysymToKeycode(_display, XK_KP_9)] = KEY_KP_ + '9'; + _key_table[XKeysymToKeycode(_display, XK_KP_Enter)] = KEY_KP_ + '\n'; + _key_table[XKeysymToKeycode(_display, XK_KP_Divide)] = KEY_KP_ + '/'; + _key_table[XKeysymToKeycode(_display, XK_KP_Multiply)] = KEY_KP_ + '*'; + _key_table[XKeysymToKeycode(_display, XK_KP_Add)] = KEY_KP_ + '+'; + _key_table[XKeysymToKeycode(_display, XK_KP_Subtract)] = KEY_KP_ + '-'; + _key_table[XKeysymToKeycode(_display, XK_KP_Decimal)] = KEY_KP_ + '.'; + _key_table[XKeysymToKeycode(_display, XK_KP_Separator)] = KEY_KP_ + ','; + + i32 screen = DefaultScreen(_display); + i32 depth = DefaultDepth (_display, screen); + Visual *visual = DefaultVisual(_display, screen); + + _gc = DefaultGC(_display, screen); + assert(_gc != NULL); + + XSetGraphicsExposures(_display, _gc, False); + + i32 _display_width = DisplayWidth (_display, screen); + i32 _display_height = DisplayHeight(_display, screen); + + i32 x = (_display_width - platform.frame_width) / 2; + i32 y = (_display_height - platform.frame_height) / 2; + + _window = XCreateWindow(_display, XDefaultRootWindow(_display), x, y, platform.frame_width, platform.frame_height, 0, depth, InputOutput, visual, CWEventMask, &(XSetWindowAttributes) { .event_mask = KeyPressMask | KeyReleaseMask | ButtonPressMask | ButtonReleaseMask | EnterWindowMask | LeaveWindowMask | PointerMotionMask | VisibilityChangeMask | FocusChangeMask | StructureNotifyMask | SubstructureNotifyMask, }); + + _im = XOpenIM(_display, NULL, NULL, NULL); + assert(_im != NULL); + + _ic = XCreateIC(_im, XNInputStyle, XIMPreeditNothing | XIMStatusNothing, XNClientWindow, _window, NULL); + assert(_ic != NULL); + + platform.pixels = _buffer; + platform.input = _input; + platform.clipboard = _clipboard_buffer; + + _image = (XImage) { + .width = platform.frame_width, + .height = platform.frame_height, + .depth = depth, + .xoffset = 0, + .format = ZPixmap, + .data = (c8 *) _buffer, + .byte_order = LSBFirst, + .bitmap_unit = 32, + .bitmap_bit_order = LSBFirst, + .bitmap_pad = 32, + .bits_per_pixel = 32, + .bytes_per_line = 4 * platform.frame_width, + .red_mask = 0xff0000, + .green_mask = 0x00ff00, + .blue_mask = 0x0000ff, + }; + + XInitImage(&_image); + + _wm_delete_window = XInternAtom(_display, "WM_DELETE_WINDOW", False); + _clipboard = XInternAtom(_display, "CLIPBOARD", False); + _targets = XInternAtom(_display, "TARGETS", False); + _utf8_string = XInternAtom(_display, "UTF8_STRING", False); + + XSetICFocus(_ic); + XSetWMProtocols(_display, _window, &_wm_delete_window, 1); + + XStoreName(_display, _window, platform.title); + XMapWindow(_display, _window); } void p_cleanup(void) { - XFreePixmap (display, pixmap); - XDestroyWindow(display, window); - XCloseDisplay (display); + XDestroyWindow(_display, _window); + XCloseDisplay (_display); } i32 p_handle_events(void) { - XGetWindowAttributes(display, window, &attrs); + i32 num_events = 0; - if (attrs.width == platform.frame_width && attrs.height == platform.frame_height) { - XPutImage(display, pixmap, gc, &image, 0, 0, 0, 0, attrs.width, attrs.height); - XCopyArea(display, pixmap, window, gc, 0, 0, attrs.width, attrs.height, 0, 0); - } + memset(platform.key_pressed, 0, sizeof platform.key_pressed); - XFlush(display); + platform.input_size = 0; + platform.cursor_dx = 0; + platform.cursor_dy = 0; + platform.wheel_dy = 0; - i32 num_events = XEventsQueued(display, QueuedAlready); + XEvent ev; - memset(platform.key_pressed, 0, sizeof platform.key_pressed); + b8 requested_clipboard = 0; + + while (XEventsQueued(_display, QueuedAlready) > 0) { + ++num_events; - for (i32 i = 0; i < num_events; ++i) { - XNextEvent(display, &ev); + XNextEvent(_display, &ev); + XFilterEvent(&ev, _window); switch (ev.type) { case DestroyNotify: @@ -431,8 +686,10 @@ i32 p_handle_events(void) { break; case MotionNotify: - platform.cursor_x = ev.xmotion.x; - platform.cursor_y = ev.xmotion.y; + platform.cursor_dx += ev.xmotion.x - platform.cursor_x; + platform.cursor_dy += ev.xmotion.y - platform.cursor_y; + platform.cursor_x = ev.xmotion.x; + platform.cursor_y = ev.xmotion.y; break; case ButtonPress: @@ -441,26 +698,24 @@ i32 p_handle_events(void) { switch (ev.xbutton.button) { case Button1: platform.key_down[BUTTON_LEFT] = 1; - platform.key_pressed[BUTTON_LEFT]++; + ++platform.key_pressed[BUTTON_LEFT]; break; case Button2: platform.key_down[BUTTON_MIDDLE] = 1; - platform.key_pressed[BUTTON_MIDDLE]++; + ++platform.key_pressed[BUTTON_MIDDLE]; break; case Button3: platform.key_down[BUTTON_RIGHT] = 1; - platform.key_pressed[BUTTON_RIGHT]++; - break; - case Button4: - platform.key_down[BUTTON_X1] = 1; - platform.key_pressed[BUTTON_X1]++; - break; - case Button5: - platform.key_down[BUTTON_X2] = 1; - platform.key_pressed[BUTTON_X2]++; + ++platform.key_pressed[BUTTON_RIGHT]; break; + case Button4: ++platform.wheel_dy; break; + case Button5: --platform.wheel_dy; break; default:; } + if (!requested_clipboard) { + XConvertSelection(_display, _clipboard, _targets, _clipboard, _window, CurrentTime); + requested_clipboard = 1; + } break; case ButtonRelease: @@ -470,33 +725,178 @@ i32 p_handle_events(void) { case Button1: platform.key_down[BUTTON_LEFT] = 0; break; case Button2: platform.key_down[BUTTON_MIDDLE] = 0; break; case Button3: platform.key_down[BUTTON_RIGHT] = 0; break; - case Button4: platform.key_down[BUTTON_X1] = 0; break; - case Button5: platform.key_down[BUTTON_X2] = 0; break; default:; } break; - case KeyPress: + case KeyPress: { + i16 k = _key_table[ev.xkey.keycode]; platform.cursor_x = ev.xkey.x; platform.cursor_y = ev.xkey.y; - platform.key_down [key_table[ev.xkey.keycode]] = 1; - platform.key_pressed[key_table[ev.xkey.keycode]]++; - break; + platform.key_down [k] = 1; + platform.key_pressed[k]++; + platform.key_down[MOD_CTRL] = !!(ev.xkey.state & ControlMask); + platform.key_down[MOD_SHIFT] = !!(ev.xkey.state & ShiftMask); + platform.key_down[MOD_ALT] = !!(ev.xkey.state & Mod1Mask); + platform.key_down[MOD_CAPS] = !!(ev.xkey.state & LockMask); + platform.key_down[MOD_NUM] = !!(ev.xkey.state & Mod2Mask); + platform.key_down[MOD_SCROLL] = !!(ev.xkey.state & Mod3Mask); + if (platform.input_size < MAX_INPUT_SIZE) { + if (k < 32 || k >= 128) + platform.input[platform.input_size++] = (Input_Key) { + .ctrl = !!(ev.xkey.state & ControlMask), + .shift = !!(ev.xkey.state & ShiftMask), + .alt = !!(ev.xkey.state & Mod1Mask), + .caps = !!(ev.xkey.state & LockMask), + .num = !!(ev.xkey.state & Mod2Mask), + .scroll = !!(ev.xkey.state & Mod3Mask), + .key = k, + .c = 0, + }; + else { + c8 buf[16]; + i32 len = Xutf8LookupString(_ic, &ev.xkey, buf, sizeof buf - 1, NULL, NULL); + if (len > 0) + platform.input[platform.input_size++] = (Input_Key) { + .ctrl = !!(ev.xkey.state & ControlMask), + .shift = !!(ev.xkey.state & ShiftMask), + .alt = !!(ev.xkey.state & Mod1Mask), + .caps = !!(ev.xkey.state & LockMask), + .num = !!(ev.xkey.state & Mod2Mask), + .scroll = !!(ev.xkey.state & Mod3Mask), + .key = k, + .c = utf8_read(len, buf), + }; + } + } + if (!requested_clipboard) { + XConvertSelection(_display, _clipboard, _targets, _clipboard, _window, CurrentTime); + requested_clipboard = 1; + } + } break; - case KeyRelease: + case KeyRelease: { + i16 k = _key_table[ev.xkey.keycode]; platform.cursor_x = ev.xkey.x; platform.cursor_y = ev.xkey.y; - platform.key_down [key_table[ev.xkey.keycode]] = 0; - platform.key_pressed[key_table[ev.xkey.keycode]]--; + platform.key_down [k] = 0; + platform.key_pressed[k]--; + platform.key_down[MOD_CTRL] = !!(ev.xkey.state & ControlMask); + platform.key_down[MOD_SHIFT] = !!(ev.xkey.state & ShiftMask); + platform.key_down[MOD_ALT] = !!(ev.xkey.state & Mod1Mask); + platform.key_down[MOD_CAPS] = !!(ev.xkey.state & LockMask); + platform.key_down[MOD_NUM] = !!(ev.xkey.state & Mod2Mask); + platform.key_down[MOD_SCROLL] = !!(ev.xkey.state & Mod3Mask); + } break; + + case SelectionRequest: + if (ev.xselectionrequest.requestor != _window && + ev.xselectionrequest.selection == _clipboard && + XGetSelectionOwner(_display, _clipboard) == _window) { + if (ev.xselectionrequest.property != None) { + if (ev.xselectionrequest.target == _targets) + XChangeProperty( + ev.xselectionrequest.display, + ev.xselectionrequest.requestor, + ev.xselectionrequest.property, + XA_ATOM, + 32, + PropModeReplace, + (u8 *) &_utf8_string, + 1 + ); + else if (ev.xselectionrequest.target == _utf8_string) + XChangeProperty( + ev.xselectionrequest.display, + ev.xselectionrequest.requestor, + ev.xselectionrequest.property, + ev.xselectionrequest.target, + 8, + PropModeReplace, + (u8 *) platform.clipboard, + platform.clipboard_size + ); + } + + XSendEvent(_display, ev.xselectionrequest.requestor, 0, 0, (XEvent *) &(XSelectionEvent) { + .type = SelectionNotify, + .serial = ev.xselectionrequest.serial, + .send_event = ev.xselectionrequest.send_event, + .display = ev.xselectionrequest.display, + .requestor = ev.xselectionrequest.requestor, + .selection = ev.xselectionrequest.selection, + .target = ev.xselectionrequest.target, + .property = ev.xselectionrequest.property, + .time = ev.xselectionrequest.time, + }); + } + break; + + case SelectionNotify: + if (ev.xselection.property != None) { + i64 len; + u8 *data; + + XGetWindowProperty( + _display, + _window, + _clipboard, + 0, + MAX_CLIPBOARD_SIZE, + False, + AnyPropertyType, + &(Atom) {0}, + &(int) {0}, + (unsigned long *) &len, + &(unsigned long) {0}, + &data + ); + + if (ev.xselection.target == _targets) { + _target = None; + Atom *list = (Atom *) data; + + for (i64 i = 0; i < len; i++) + if (list[i] == XA_STRING) + _target = XA_STRING; + else if (list[i] == _utf8_string) { + _target = _utf8_string; + break; + } + + if (_target != None) + XConvertSelection(_display, _clipboard, _target, _clipboard, _window, CurrentTime); + } else if (ev.xselection.target == _target) { + if (len > MAX_CLIPBOARD_SIZE) + len = MAX_CLIPBOARD_SIZE; + platform.clipboard_size = len; + if (len > 0) + memcpy(platform.clipboard, data, len); + } + + if (data) + XFree(data); + } break; case EnterNotify: platform.has_cursor = 1; break; case LeaveNotify: platform.has_cursor = 0; break; - case FocusIn: platform.has_focus = 1; break; case FocusOut: platform.has_focus = 0; break; + case FocusIn: + platform.has_focus = 1; + if (!requested_clipboard) { + XConvertSelection(_display, _clipboard, _targets, _clipboard, _window, CurrentTime); + requested_clipboard = 1; + } + break; + + case MappingNotify: + XRefreshKeyboardMapping(&ev.xmapping); + break; + case ClientMessage: - if ((Atom) ev.xclient.data.l[0] == wm_delete_window) + if ((Atom) ev.xclient.data.l[0] == _wm_delete_window) platform.done = 1; break; @@ -504,14 +904,17 @@ i32 p_handle_events(void) { } } - if ((platform.frame_width != attrs.width || platform.frame_height != attrs.height) && attrs.width * attrs.height * 4 <= (i32) sizeof buffer) { - if (attrs.width > 0 && attrs.height > 0) { - image.width = attrs.width; - image.height = attrs.height; - image.bytes_per_line = 4 * attrs.width; + for (i64 k = 0; k < (i64) (sizeof platform.key_pressed / sizeof *platform.key_pressed); ++k) + platform.key_pressed[k] = platform.key_pressed[k] > 0 ? 1 : 0; + + XWindowAttributes attrs; + XGetWindowAttributes(_display, _window, &attrs); - XFreePixmap(display, pixmap); - pixmap = XCreatePixmap(display, window, attrs.width, attrs.height, depth); + if ((platform.frame_width != attrs.width || platform.frame_height != attrs.height) && attrs.width * attrs.height * 4 <= (i32) sizeof _buffer) { + if (attrs.width > 0 && attrs.height > 0) { + _image.width = attrs.width; + _image.height = attrs.height; + _image.bytes_per_line = 4 * attrs.width; } platform.frame_width = attrs.width; @@ -520,3 +923,56 @@ i32 p_handle_events(void) { return num_events; } + +i32 p_wait_events(void) { + i32 num_events = 0; + + do { + num_events = p_handle_events(); + sched_yield(); + } while (num_events == 0); + + return num_events; +} + +void p_render_frame(void) { + if (platform.done) + return; + + XPutImage(_display, _window, _gc, &_image, 0, 0, 0, 0, platform.frame_width, platform.frame_height); + XFlush(_display); +} + +void p_clipboard_write(i64 size, c8 *data) { + assert(size <= MAX_CLIPBOARD_SIZE); + + XSetSelectionOwner(_display, _clipboard, _window, CurrentTime); + + platform.clipboard_size = size < MAX_CLIPBOARD_SIZE ? size : MAX_CLIPBOARD_SIZE; + if (platform.clipboard_size > 0) + memcpy(platform.clipboard, data, platform.clipboard_size); +} + +#endif + +// ================================================================ +// +// ALSA +// +// ================================================================ + +#ifdef __linux__ + +void p_handle_audio(i64 time_elapsed) { + (void) time_elapsed; + assert(0); +} + +void p_queue_sound(i64 delay, i64 num_samples, f32 *samples) { + (void) delay; + (void) num_samples; + (void) samples; + assert(0); +} + +#endif diff --git a/examples/landscape.c b/examples/landscape.c index a3de2cf..813ad21 100755 --- a/examples/landscape.c +++ b/examples/landscape.c @@ -41,6 +41,7 @@ gcc \ -Wno-unused-variable \ -Wno-unused-but-set-variable \ -Wno-unused-parameter \ + -Wno-overlength-strings \ -O3 \ -fsanitize=undefined,address,leak -mshstk \ -lX11 -lm \ @@ -65,6 +66,7 @@ typedef unsigned short u16; typedef unsigned u32; typedef unsigned long long u64; typedef char c8; +typedef int c32; typedef signed char b8; typedef float f32; typedef double f64; @@ -76,8 +78,9 @@ typedef double f64; // ================================================================ enum { - MAX_NUM_PIXELS = 4 * 1024 * 1024, - MAX_CLIPBOARD_SIZE = 0, + MAX_NUM_PIXELS = 10 * 1024 * 1024, + MAX_INPUT_SIZE = 256, + MAX_CLIPBOARD_SIZE = 10 * 1024 * 1024, MAX_NUM_AUDIO_SAMPLES = 0, MAX_NUM_SOCKETS = 0, @@ -87,7 +90,7 @@ enum { IPv4 = 1, IPv6 = 2, - KEY_LEFT, + KEY_LEFT = 128, KEY_RIGHT, KEY_UP, KEY_DOWN, @@ -106,30 +109,49 @@ enum { KEY_END, KEY_PAGEUP, KEY_PAGEDOWN, - KEY_KP_ = 128, - KEY_F_ = 256, - BUTTON_LEFT = KEY_F_ + 64, + BUTTON_LEFT, BUTTON_MIDDLE, BUTTON_RIGHT, + MOD_CTRL, + MOD_SHIFT, + MOD_ALT, + MOD_CAPS, + MOD_NUM, + MOD_SCROLL, + KEY_F_, + KEY_KP_ = KEY_F_ + 64, }; typedef struct { - c8 *title; - i32 frame_width; - i32 frame_height; - u32 *pixels; - i64 clipboard_size; - c8 *clipboard; // TODO - b8 done; - b8 has_focus; - b8 has_cursor; - i32 cursor_x; - i32 cursor_y; - i32 cursor_dx; - i32 cursor_dy; - i64 wheel_dy; - b8 key_down[512]; - b8 key_pressed[512]; + b8 ctrl : 1; + b8 shift : 1; + b8 alt : 1; + b8 caps : 1; + b8 num : 1; + b8 scroll : 1; + u16 key; + c32 c; +} Input_Key; + +typedef struct { + c8 * title; + i32 frame_width; + i32 frame_height; + u32 * pixels; + i64 input_size; + Input_Key *input; + i64 clipboard_size; + c8 * clipboard; + b8 done; + b8 has_focus; + b8 has_cursor; + i32 cursor_x; + i32 cursor_y; + i32 cursor_dx; + i32 cursor_dy; + i64 wheel_dy; + b8 key_down[512]; + b8 key_pressed[512]; } Platform; typedef struct { @@ -143,12 +165,21 @@ typedef struct { typedef i64 (*Thread_Proc)(void *user_data); +// UTF-8 +i32 utf8_size(c32 c); +c32 utf8_read(i64 len, c8 *s); +i32 utf8_write(c32 c, c8 *buffer); + // Window void p_init(void); void p_cleanup(void); i32 p_handle_events(void); +i32 p_wait_events(void); void p_render_frame(void); +// Clipboard +void p_clipboard_write(i64 size, c8 *data); + // Sound void p_handle_audio(i64 time_elapsed); void p_queue_sound(i64 delay, i64 num_samples, f32 *samples); @@ -164,11 +195,6 @@ Platform platform = {0}; // GAME // // ================================================================ -// -// TODO -// - Font -// -// ================================================================ #include <stdio.h> #include <stdlib.h> @@ -451,17 +477,85 @@ i32 main(i32 argc, c8 **argv) { // ---------------------------------------------------------------- // // TODO -// - Clipboard // - Sound // - Sockets -// -// X11 clipboard -// https://handmade.network/forums/articles/t/8544-implementing_copy_paste_in_x11 +// - Clipboard daemon // // ALSA // https://www.alsa-project.org/alsa-doc/alsa-lib/_2test_2pcm_min_8c-example.html // // ================================================================ + +#include <assert.h> +#include <string.h> + +c32 utf8_read(i64 len, c8 *s) { + if (len >= 1 && + (s[0] & 0x80) == 0) + return s[0]; + if (len >= 2 && + (s[0] & 0xe0) == 0xc0 && + (s[1] & 0xc0) == 0x80) + return (s[1] & 0x3f) + | ((s[0] & 0x1f) << 6); + if (len >= 3 && + (s[0] & 0xf0) == 0xe0 && + (s[1] & 0xc0) == 0x80 && + (s[2] & 0xc0) == 0x80) + return (s[2] & 0x3f) + | ((s[1] & 0x3f) << 6) + | ((s[0] & 0x0f) << 12); + if (len >= 4 && + (s[0] & 0xf8) == 0xf0 && + (s[1] & 0xc0) == 0x80 && + (s[2] & 0xc0) == 0x80 && + (s[3] & 0xc0) == 0x80) + return (s[3] & 0x3f) + | ((s[2] & 0x3f) << 6) + | ((s[1] & 0x3f) << 12) + | ((s[0] & 0x07) << 18); + return 0; +} + +i32 utf8_size(c32 c) { + if ((c & 0x00007f) == c) return 1; + if ((c & 0x0007ff) == c) return 2; + if ((c & 0x00ffff) == c) return 3; + if ((c & 0x1fffff) == c) return 4; + return 0; +} + +i32 utf8_write(c32 c, c8 *buffer) { + if ((c & 0x7f) == c) { + buffer[0] = (c8) c; + return 1; + } + + if ((c & 0x7ff) == c) { + buffer[0] = 0xc0 | ((c >> 6) & 0x1f); + buffer[1] = 0x80 | ( c & 0x3f); + return 2; + } + + if ((c & 0xffff) == c) { + buffer[0] = 0xc0 | ((c >> 12) & 0x0f); + buffer[1] = 0x80 | ((c >> 6) & 0x3f); + buffer[2] = 0x80 | ( c & 0x3f); + return 3; + } + + if ((c & 0x1fffff) == c) { + buffer[0] = 0xc0 | ((c >> 18) & 0x07); + buffer[1] = 0x80 | ((c >> 12) & 0x3f); + buffer[2] = 0x80 | ((c >> 6) & 0x3f); + buffer[3] = 0x80 | ( c & 0x3f); + return 4; + } + + return 0; +} + +// ================================================================ // // UDP sockets // @@ -496,14 +590,25 @@ i64 p_send(u16 slot, IP_Address address, i64 size, u8 *data) { #include <X11/Xlib.h> #include <X11/Xutil.h> +#include <X11/Xatom.h> +#include <sched.h> +#include <time.h> -static i16 _key_table[512]; -static u32 _buffer[MAX_NUM_PIXELS]; -static XImage _image; -static Display *_display; -static GC _gc; -static Window _window; -static Atom _wm_delete_window; +static i16 _key_table[512] = {0}; +static u32 _buffer[MAX_NUM_PIXELS] = {0}; +static Input_Key _input[MAX_INPUT_SIZE] = {0}; +static c8 _clipboard_buffer[MAX_CLIPBOARD_SIZE] = {0}; +static XImage _image = {0}; +static Display * _display = NULL; +static GC _gc = NULL; +static XIM _im = NULL; +static XIC _ic = NULL; +static Window _window = 0; +static Atom _wm_delete_window = 0; +static Atom _clipboard = 0; +static Atom _targets = 0; +static Atom _utf8_string = 0; +static Atom _target = None; void p_init(void) { _display = XOpenDisplay(NULL); @@ -559,6 +664,8 @@ void p_init(void) { _key_table[XKeysymToKeycode(_display, XK_comma)] = ','; _key_table[XKeysymToKeycode(_display, XK_greater)] = '.'; _key_table[XKeysymToKeycode(_display, XK_question)] = '/'; + _key_table[XKeysymToKeycode(_display, XK_minus)] = '-'; + _key_table[XKeysymToKeycode(_display, XK_equal)] = '='; _key_table[XKeysymToKeycode(_display, XK_F1)] = KEY_F_ + 1; _key_table[XKeysymToKeycode(_display, XK_F2)] = KEY_F_ + 2; _key_table[XKeysymToKeycode(_display, XK_F3)] = KEY_F_ + 3; @@ -612,6 +719,7 @@ void p_init(void) { Visual *visual = DefaultVisual(_display, screen); _gc = DefaultGC(_display, screen); + assert(_gc != NULL); XSetGraphicsExposures(_display, _gc, False); @@ -623,7 +731,15 @@ void p_init(void) { _window = XCreateWindow(_display, XDefaultRootWindow(_display), x, y, platform.frame_width, platform.frame_height, 0, depth, InputOutput, visual, CWEventMask, &(XSetWindowAttributes) { .event_mask = KeyPressMask | KeyReleaseMask | ButtonPressMask | ButtonReleaseMask | EnterWindowMask | LeaveWindowMask | PointerMotionMask | VisibilityChangeMask | FocusChangeMask | StructureNotifyMask | SubstructureNotifyMask, }); - platform.pixels = _buffer; + _im = XOpenIM(_display, NULL, NULL, NULL); + assert(_im != NULL); + + _ic = XCreateIC(_im, XNInputStyle, XIMPreeditNothing | XIMStatusNothing, XNClientWindow, _window, NULL); + assert(_ic != NULL); + + platform.pixels = _buffer; + platform.input = _input; + platform.clipboard = _clipboard_buffer; _image = (XImage) { .width = platform.frame_width, @@ -643,11 +759,14 @@ void p_init(void) { .blue_mask = 0x0000ff, }; - XInitImage(&_image); _wm_delete_window = XInternAtom(_display, "WM_DELETE_WINDOW", False); + _clipboard = XInternAtom(_display, "CLIPBOARD", False); + _targets = XInternAtom(_display, "TARGETS", False); + _utf8_string = XInternAtom(_display, "UTF8_STRING", False); + XSetICFocus(_ic); XSetWMProtocols(_display, _window, &_wm_delete_window, 1); XStoreName(_display, _window, platform.title); @@ -660,18 +779,24 @@ void p_cleanup(void) { } i32 p_handle_events(void) { - i32 num_events = XEventsQueued(_display, QueuedAlready); + i32 num_events = 0; memset(platform.key_pressed, 0, sizeof platform.key_pressed); - platform.cursor_dx = 0; - platform.cursor_dy = 0; - platform.wheel_dy = 0; + platform.input_size = 0; + platform.cursor_dx = 0; + platform.cursor_dy = 0; + platform.wheel_dy = 0; XEvent ev; - for (i32 i = 0; i < num_events; ++i) { + b8 requested_clipboard = 0; + + while (XEventsQueued(_display, QueuedAlready) > 0) { + ++num_events; + XNextEvent(_display, &ev); + XFilterEvent(&ev, _window); switch (ev.type) { case DestroyNotify: @@ -705,6 +830,10 @@ i32 p_handle_events(void) { case Button5: --platform.wheel_dy; break; default:; } + if (!requested_clipboard) { + XConvertSelection(_display, _clipboard, _targets, _clipboard, _window, CurrentTime); + requested_clipboard = 1; + } break; case ButtonRelease: @@ -718,25 +847,172 @@ i32 p_handle_events(void) { } break; - case KeyPress: + case KeyPress: { + i16 k = _key_table[ev.xkey.keycode]; platform.cursor_x = ev.xkey.x; platform.cursor_y = ev.xkey.y; - platform.key_down [_key_table[ev.xkey.keycode]] = 1; - platform.key_pressed[_key_table[ev.xkey.keycode]]++; - break; + platform.key_down [k] = 1; + platform.key_pressed[k]++; + platform.key_down[MOD_CTRL] = !!(ev.xkey.state & ControlMask); + platform.key_down[MOD_SHIFT] = !!(ev.xkey.state & ShiftMask); + platform.key_down[MOD_ALT] = !!(ev.xkey.state & Mod1Mask); + platform.key_down[MOD_CAPS] = !!(ev.xkey.state & LockMask); + platform.key_down[MOD_NUM] = !!(ev.xkey.state & Mod2Mask); + platform.key_down[MOD_SCROLL] = !!(ev.xkey.state & Mod3Mask); + if (platform.input_size < MAX_INPUT_SIZE) { + if (k < 32 || k >= 128) + platform.input[platform.input_size++] = (Input_Key) { + .ctrl = !!(ev.xkey.state & ControlMask), + .shift = !!(ev.xkey.state & ShiftMask), + .alt = !!(ev.xkey.state & Mod1Mask), + .caps = !!(ev.xkey.state & LockMask), + .num = !!(ev.xkey.state & Mod2Mask), + .scroll = !!(ev.xkey.state & Mod3Mask), + .key = k, + .c = 0, + }; + else { + c8 buf[16]; + i32 len = Xutf8LookupString(_ic, &ev.xkey, buf, sizeof buf - 1, NULL, NULL); + if (len > 0) + platform.input[platform.input_size++] = (Input_Key) { + .ctrl = !!(ev.xkey.state & ControlMask), + .shift = !!(ev.xkey.state & ShiftMask), + .alt = !!(ev.xkey.state & Mod1Mask), + .caps = !!(ev.xkey.state & LockMask), + .num = !!(ev.xkey.state & Mod2Mask), + .scroll = !!(ev.xkey.state & Mod3Mask), + .key = k, + .c = utf8_read(len, buf), + }; + } + } + if (!requested_clipboard) { + XConvertSelection(_display, _clipboard, _targets, _clipboard, _window, CurrentTime); + requested_clipboard = 1; + } + } break; - case KeyRelease: + case KeyRelease: { + i16 k = _key_table[ev.xkey.keycode]; platform.cursor_x = ev.xkey.x; platform.cursor_y = ev.xkey.y; - platform.key_down [_key_table[ev.xkey.keycode]] = 0; - platform.key_pressed[_key_table[ev.xkey.keycode]]--; + platform.key_down [k] = 0; + platform.key_pressed[k]--; + platform.key_down[MOD_CTRL] = !!(ev.xkey.state & ControlMask); + platform.key_down[MOD_SHIFT] = !!(ev.xkey.state & ShiftMask); + platform.key_down[MOD_ALT] = !!(ev.xkey.state & Mod1Mask); + platform.key_down[MOD_CAPS] = !!(ev.xkey.state & LockMask); + platform.key_down[MOD_NUM] = !!(ev.xkey.state & Mod2Mask); + platform.key_down[MOD_SCROLL] = !!(ev.xkey.state & Mod3Mask); + } break; + + case SelectionRequest: + if (ev.xselectionrequest.requestor != _window && + ev.xselectionrequest.selection == _clipboard && + XGetSelectionOwner(_display, _clipboard) == _window) { + if (ev.xselectionrequest.property != None) { + if (ev.xselectionrequest.target == _targets) + XChangeProperty( + ev.xselectionrequest.display, + ev.xselectionrequest.requestor, + ev.xselectionrequest.property, + XA_ATOM, + 32, + PropModeReplace, + (u8 *) &_utf8_string, + 1 + ); + else if (ev.xselectionrequest.target == _utf8_string) + XChangeProperty( + ev.xselectionrequest.display, + ev.xselectionrequest.requestor, + ev.xselectionrequest.property, + ev.xselectionrequest.target, + 8, + PropModeReplace, + (u8 *) platform.clipboard, + platform.clipboard_size + ); + } + + XSendEvent(_display, ev.xselectionrequest.requestor, 0, 0, (XEvent *) &(XSelectionEvent) { + .type = SelectionNotify, + .serial = ev.xselectionrequest.serial, + .send_event = ev.xselectionrequest.send_event, + .display = ev.xselectionrequest.display, + .requestor = ev.xselectionrequest.requestor, + .selection = ev.xselectionrequest.selection, + .target = ev.xselectionrequest.target, + .property = ev.xselectionrequest.property, + .time = ev.xselectionrequest.time, + }); + } + break; + + case SelectionNotify: + if (ev.xselection.property != None) { + i64 len; + u8 *data; + + XGetWindowProperty( + _display, + _window, + _clipboard, + 0, + MAX_CLIPBOARD_SIZE, + False, + AnyPropertyType, + &(Atom) {0}, + &(int) {0}, + (unsigned long *) &len, + &(unsigned long) {0}, + &data + ); + + if (ev.xselection.target == _targets) { + _target = None; + Atom *list = (Atom *) data; + + for (i64 i = 0; i < len; i++) + if (list[i] == XA_STRING) + _target = XA_STRING; + else if (list[i] == _utf8_string) { + _target = _utf8_string; + break; + } + + if (_target != None) + XConvertSelection(_display, _clipboard, _target, _clipboard, _window, CurrentTime); + } else if (ev.xselection.target == _target) { + if (len > MAX_CLIPBOARD_SIZE) + len = MAX_CLIPBOARD_SIZE; + platform.clipboard_size = len; + if (len > 0) + memcpy(platform.clipboard, data, len); + } + + if (data) + XFree(data); + } break; case EnterNotify: platform.has_cursor = 1; break; case LeaveNotify: platform.has_cursor = 0; break; - case FocusIn: platform.has_focus = 1; break; case FocusOut: platform.has_focus = 0; break; + case FocusIn: + platform.has_focus = 1; + if (!requested_clipboard) { + XConvertSelection(_display, _clipboard, _targets, _clipboard, _window, CurrentTime); + requested_clipboard = 1; + } + break; + + case MappingNotify: + XRefreshKeyboardMapping(&ev.xmapping); + break; + case ClientMessage: if ((Atom) ev.xclient.data.l[0] == _wm_delete_window) platform.done = 1; @@ -746,6 +1022,9 @@ i32 p_handle_events(void) { } } + for (i64 k = 0; k < (i64) (sizeof platform.key_pressed / sizeof *platform.key_pressed); ++k) + platform.key_pressed[k] = platform.key_pressed[k] > 0 ? 1 : 0; + XWindowAttributes attrs; XGetWindowAttributes(_display, _window, &attrs); @@ -763,6 +1042,17 @@ i32 p_handle_events(void) { return num_events; } +i32 p_wait_events(void) { + i32 num_events = 0; + + do { + num_events = p_handle_events(); + sched_yield(); + } while (num_events == 0); + + return num_events; +} + void p_render_frame(void) { if (platform.done) return; @@ -771,6 +1061,16 @@ void p_render_frame(void) { XFlush(_display); } +void p_clipboard_write(i64 size, c8 *data) { + assert(size <= MAX_CLIPBOARD_SIZE); + + XSetSelectionOwner(_display, _clipboard, _window, CurrentTime); + + platform.clipboard_size = size < MAX_CLIPBOARD_SIZE ? size : MAX_CLIPBOARD_SIZE; + if (platform.clipboard_size > 0) + memcpy(platform.clipboard, data, platform.clipboard_size); +} + #endif // ================================================================ diff --git a/reduced_system_layer.c b/reduced_system_layer.c index 6eb5144..379cc97 100755 --- a/reduced_system_layer.c +++ b/reduced_system_layer.c @@ -41,6 +41,7 @@ gcc \ -Wno-unused-variable \ -Wno-unused-but-set-variable \ -Wno-unused-parameter \ + -Wno-overlength-strings \ -O3 \ -fsanitize=undefined,address,leak -mshstk \ -lX11 -lm \ @@ -65,6 +66,7 @@ typedef unsigned short u16; typedef unsigned u32; typedef unsigned long long u64; typedef char c8; +typedef int c32; typedef signed char b8; typedef float f32; typedef double f64; @@ -76,8 +78,9 @@ typedef double f64; // ================================================================ enum { - MAX_NUM_PIXELS = 4 * 1024 * 1024, - MAX_CLIPBOARD_SIZE = 0, + MAX_NUM_PIXELS = 10 * 1024 * 1024, + MAX_INPUT_SIZE = 256, + MAX_CLIPBOARD_SIZE = 10 * 1024 * 1024, MAX_NUM_AUDIO_SAMPLES = 0, MAX_NUM_SOCKETS = 0, @@ -86,8 +89,8 @@ enum { IPv4 = 1, IPv6 = 2, - - KEY_LEFT, + + KEY_LEFT = 128, KEY_RIGHT, KEY_UP, KEY_DOWN, @@ -106,30 +109,49 @@ enum { KEY_END, KEY_PAGEUP, KEY_PAGEDOWN, - KEY_KP_ = 128, - KEY_F_ = 256, - BUTTON_LEFT = KEY_F_ + 64, + BUTTON_LEFT, BUTTON_MIDDLE, BUTTON_RIGHT, + MOD_CTRL, + MOD_SHIFT, + MOD_ALT, + MOD_CAPS, + MOD_NUM, + MOD_SCROLL, + KEY_F_, + KEY_KP_ = KEY_F_ + 64, }; typedef struct { - c8 *title; - i32 frame_width; - i32 frame_height; - u32 *pixels; - i64 clipboard_size; - c8 *clipboard; // TODO - b8 done; - b8 has_focus; - b8 has_cursor; - i32 cursor_x; - i32 cursor_y; - i32 cursor_dx; - i32 cursor_dy; - i64 wheel_dy; - b8 key_down[512]; - b8 key_pressed[512]; + b8 ctrl : 1; + b8 shift : 1; + b8 alt : 1; + b8 caps : 1; + b8 num : 1; + b8 scroll : 1; + u16 key; + c32 c; +} Input_Key; + +typedef struct { + c8 * title; + i32 frame_width; + i32 frame_height; + u32 * pixels; + i64 input_size; + Input_Key *input; + i64 clipboard_size; + c8 * clipboard; + b8 done; + b8 has_focus; + b8 has_cursor; + i32 cursor_x; + i32 cursor_y; + i32 cursor_dx; + i32 cursor_dy; + i64 wheel_dy; + b8 key_down[512]; + b8 key_pressed[512]; } Platform; typedef struct { @@ -143,10 +165,20 @@ typedef struct { typedef i64 (*Thread_Proc)(void *user_data); +// UTF-8 +i32 utf8_size(c32 c); +c32 utf8_read(i64 len, c8 *s); +i32 utf8_write(c32 c, c8 *buffer); + // Window void p_init(void); void p_cleanup(void); i32 p_handle_events(void); +i32 p_wait_events(void); +void p_render_frame(void); + +// Clipboard +void p_clipboard_write(i64 size, c8 *data); // Sound void p_handle_audio(i64 time_elapsed); @@ -178,6 +210,7 @@ i32 main(i32 argc, c8 **argv) { while (!platform.done) { p_handle_events(); + p_render_frame(); } p_cleanup(); @@ -191,25 +224,90 @@ i32 main(i32 argc, c8 **argv) { // ---------------------------------------------------------------- // // TODO -// - Clipboard // - Sound // - Sockets -// -// X11 clipboard -// https://handmade.network/forums/articles/t/8544-implementing_copy_paste_in_x11 +// - Clipboard daemon // // ALSA // https://www.alsa-project.org/alsa-doc/alsa-lib/_2test_2pcm_min_8c-example.html // // ================================================================ + +#include <assert.h> +#include <string.h> + +c32 utf8_read(i64 len, c8 *s) { + if (len >= 1 && + (s[0] & 0x80) == 0) + return s[0]; + if (len >= 2 && + (s[0] & 0xe0) == 0xc0 && + (s[1] & 0xc0) == 0x80) + return (s[1] & 0x3f) + | ((s[0] & 0x1f) << 6); + if (len >= 3 && + (s[0] & 0xf0) == 0xe0 && + (s[1] & 0xc0) == 0x80 && + (s[2] & 0xc0) == 0x80) + return (s[2] & 0x3f) + | ((s[1] & 0x3f) << 6) + | ((s[0] & 0x0f) << 12); + if (len >= 4 && + (s[0] & 0xf8) == 0xf0 && + (s[1] & 0xc0) == 0x80 && + (s[2] & 0xc0) == 0x80 && + (s[3] & 0xc0) == 0x80) + return (s[3] & 0x3f) + | ((s[2] & 0x3f) << 6) + | ((s[1] & 0x3f) << 12) + | ((s[0] & 0x07) << 18); + return 0; +} + +i32 utf8_size(c32 c) { + if ((c & 0x00007f) == c) return 1; + if ((c & 0x0007ff) == c) return 2; + if ((c & 0x00ffff) == c) return 3; + if ((c & 0x1fffff) == c) return 4; + return 0; +} + +i32 utf8_write(c32 c, c8 *buffer) { + if ((c & 0x7f) == c) { + buffer[0] = (c8) c; + return 1; + } + + if ((c & 0x7ff) == c) { + buffer[0] = 0xc0 | ((c >> 6) & 0x1f); + buffer[1] = 0x80 | ( c & 0x3f); + return 2; + } + + if ((c & 0xffff) == c) { + buffer[0] = 0xc0 | ((c >> 12) & 0x0f); + buffer[1] = 0x80 | ((c >> 6) & 0x3f); + buffer[2] = 0x80 | ( c & 0x3f); + return 3; + } + + if ((c & 0x1fffff) == c) { + buffer[0] = 0xc0 | ((c >> 18) & 0x07); + buffer[1] = 0x80 | ((c >> 12) & 0x3f); + buffer[2] = 0x80 | ((c >> 6) & 0x3f); + buffer[3] = 0x80 | ( c & 0x3f); + return 4; + } + + return 0; +} + +// ================================================================ // // UDP sockets // // ================================================================ -#include <assert.h> -#include <string.h> - i64 p_recv(u16 slot, IP_Address address, i64 size, u8 *data, IP_Address *remote_address) { (void) slot; (void) address; @@ -239,14 +337,25 @@ i64 p_send(u16 slot, IP_Address address, i64 size, u8 *data) { #include <X11/Xlib.h> #include <X11/Xutil.h> - -static i16 _key_table[512]; -static u32 _buffer[MAX_NUM_PIXELS]; -static XImage _image; -static Display *_display; -static GC _gc; -static Window _window; -static Atom _wm_delete_window; +#include <X11/Xatom.h> +#include <sched.h> +#include <time.h> + +static i16 _key_table[512] = {0}; +static u32 _buffer[MAX_NUM_PIXELS] = {0}; +static Input_Key _input[MAX_INPUT_SIZE] = {0}; +static c8 _clipboard_buffer[MAX_CLIPBOARD_SIZE] = {0}; +static XImage _image = {0}; +static Display * _display = NULL; +static GC _gc = NULL; +static XIM _im = NULL; +static XIC _ic = NULL; +static Window _window = 0; +static Atom _wm_delete_window = 0; +static Atom _clipboard = 0; +static Atom _targets = 0; +static Atom _utf8_string = 0; +static Atom _target = None; void p_init(void) { _display = XOpenDisplay(NULL); @@ -302,6 +411,8 @@ void p_init(void) { _key_table[XKeysymToKeycode(_display, XK_comma)] = ','; _key_table[XKeysymToKeycode(_display, XK_greater)] = '.'; _key_table[XKeysymToKeycode(_display, XK_question)] = '/'; + _key_table[XKeysymToKeycode(_display, XK_minus)] = '-'; + _key_table[XKeysymToKeycode(_display, XK_equal)] = '='; _key_table[XKeysymToKeycode(_display, XK_F1)] = KEY_F_ + 1; _key_table[XKeysymToKeycode(_display, XK_F2)] = KEY_F_ + 2; _key_table[XKeysymToKeycode(_display, XK_F3)] = KEY_F_ + 3; @@ -355,6 +466,7 @@ void p_init(void) { Visual *visual = DefaultVisual(_display, screen); _gc = DefaultGC(_display, screen); + assert(_gc != NULL); XSetGraphicsExposures(_display, _gc, False); @@ -366,7 +478,15 @@ void p_init(void) { _window = XCreateWindow(_display, XDefaultRootWindow(_display), x, y, platform.frame_width, platform.frame_height, 0, depth, InputOutput, visual, CWEventMask, &(XSetWindowAttributes) { .event_mask = KeyPressMask | KeyReleaseMask | ButtonPressMask | ButtonReleaseMask | EnterWindowMask | LeaveWindowMask | PointerMotionMask | VisibilityChangeMask | FocusChangeMask | StructureNotifyMask | SubstructureNotifyMask, }); - platform.pixels = _buffer; + _im = XOpenIM(_display, NULL, NULL, NULL); + assert(_im != NULL); + + _ic = XCreateIC(_im, XNInputStyle, XIMPreeditNothing | XIMStatusNothing, XNClientWindow, _window, NULL); + assert(_ic != NULL); + + platform.pixels = _buffer; + platform.input = _input; + platform.clipboard = _clipboard_buffer; _image = (XImage) { .width = platform.frame_width, @@ -386,11 +506,14 @@ void p_init(void) { .blue_mask = 0x0000ff, }; - XInitImage(&_image); - _wm_delete_window = XInternAtom(_display, "WM_DELETE__WINDOW", False); + _wm_delete_window = XInternAtom(_display, "WM_DELETE_WINDOW", False); + _clipboard = XInternAtom(_display, "CLIPBOARD", False); + _targets = XInternAtom(_display, "TARGETS", False); + _utf8_string = XInternAtom(_display, "UTF8_STRING", False); + XSetICFocus(_ic); XSetWMProtocols(_display, _window, &_wm_delete_window, 1); XStoreName(_display, _window, platform.title); @@ -403,26 +526,24 @@ void p_cleanup(void) { } i32 p_handle_events(void) { - XEvent ev; - XWindowAttributes attrs; + i32 num_events = 0; - XGetWindowAttributes(_display, _window, &attrs); + memset(platform.key_pressed, 0, sizeof platform.key_pressed); - if (attrs.width == platform.frame_width && attrs.height == platform.frame_height) - XPutImage(_display, _window, _gc, &_image, 0, 0, 0, 0, attrs.width, attrs.height); + platform.input_size = 0; + platform.cursor_dx = 0; + platform.cursor_dy = 0; + platform.wheel_dy = 0; - XFlush(_display); + XEvent ev; - i32 num_events = XEventsQueued(_display, QueuedAlready); - - memset(platform.key_pressed, 0, sizeof platform.key_pressed); + b8 requested_clipboard = 0; - platform.cursor_dx = 0; - platform.cursor_dy = 0; - platform.wheel_dy = 0; + while (XEventsQueued(_display, QueuedAlready) > 0) { + ++num_events; - for (i32 i = 0; i < num_events; ++i) { XNextEvent(_display, &ev); + XFilterEvent(&ev, _window); switch (ev.type) { case DestroyNotify: @@ -456,6 +577,10 @@ i32 p_handle_events(void) { case Button5: --platform.wheel_dy; break; default:; } + if (!requested_clipboard) { + XConvertSelection(_display, _clipboard, _targets, _clipboard, _window, CurrentTime); + requested_clipboard = 1; + } break; case ButtonRelease: @@ -469,25 +594,172 @@ i32 p_handle_events(void) { } break; - case KeyPress: + case KeyPress: { + i16 k = _key_table[ev.xkey.keycode]; platform.cursor_x = ev.xkey.x; platform.cursor_y = ev.xkey.y; - platform.key_down [_key_table[ev.xkey.keycode]] = 1; - platform.key_pressed[_key_table[ev.xkey.keycode]]++; - break; + platform.key_down [k] = 1; + platform.key_pressed[k]++; + platform.key_down[MOD_CTRL] = !!(ev.xkey.state & ControlMask); + platform.key_down[MOD_SHIFT] = !!(ev.xkey.state & ShiftMask); + platform.key_down[MOD_ALT] = !!(ev.xkey.state & Mod1Mask); + platform.key_down[MOD_CAPS] = !!(ev.xkey.state & LockMask); + platform.key_down[MOD_NUM] = !!(ev.xkey.state & Mod2Mask); + platform.key_down[MOD_SCROLL] = !!(ev.xkey.state & Mod3Mask); + if (platform.input_size < MAX_INPUT_SIZE) { + if (k < 32 || k >= 128) + platform.input[platform.input_size++] = (Input_Key) { + .ctrl = !!(ev.xkey.state & ControlMask), + .shift = !!(ev.xkey.state & ShiftMask), + .alt = !!(ev.xkey.state & Mod1Mask), + .caps = !!(ev.xkey.state & LockMask), + .num = !!(ev.xkey.state & Mod2Mask), + .scroll = !!(ev.xkey.state & Mod3Mask), + .key = k, + .c = 0, + }; + else { + c8 buf[16]; + i32 len = Xutf8LookupString(_ic, &ev.xkey, buf, sizeof buf - 1, NULL, NULL); + if (len > 0) + platform.input[platform.input_size++] = (Input_Key) { + .ctrl = !!(ev.xkey.state & ControlMask), + .shift = !!(ev.xkey.state & ShiftMask), + .alt = !!(ev.xkey.state & Mod1Mask), + .caps = !!(ev.xkey.state & LockMask), + .num = !!(ev.xkey.state & Mod2Mask), + .scroll = !!(ev.xkey.state & Mod3Mask), + .key = k, + .c = utf8_read(len, buf), + }; + } + } + if (!requested_clipboard) { + XConvertSelection(_display, _clipboard, _targets, _clipboard, _window, CurrentTime); + requested_clipboard = 1; + } + } break; - case KeyRelease: + case KeyRelease: { + i16 k = _key_table[ev.xkey.keycode]; platform.cursor_x = ev.xkey.x; platform.cursor_y = ev.xkey.y; - platform.key_down [_key_table[ev.xkey.keycode]] = 0; - platform.key_pressed[_key_table[ev.xkey.keycode]]--; + platform.key_down [k] = 0; + platform.key_pressed[k]--; + platform.key_down[MOD_CTRL] = !!(ev.xkey.state & ControlMask); + platform.key_down[MOD_SHIFT] = !!(ev.xkey.state & ShiftMask); + platform.key_down[MOD_ALT] = !!(ev.xkey.state & Mod1Mask); + platform.key_down[MOD_CAPS] = !!(ev.xkey.state & LockMask); + platform.key_down[MOD_NUM] = !!(ev.xkey.state & Mod2Mask); + platform.key_down[MOD_SCROLL] = !!(ev.xkey.state & Mod3Mask); + } break; + + case SelectionRequest: + if (ev.xselectionrequest.requestor != _window && + ev.xselectionrequest.selection == _clipboard && + XGetSelectionOwner(_display, _clipboard) == _window) { + if (ev.xselectionrequest.property != None) { + if (ev.xselectionrequest.target == _targets) + XChangeProperty( + ev.xselectionrequest.display, + ev.xselectionrequest.requestor, + ev.xselectionrequest.property, + XA_ATOM, + 32, + PropModeReplace, + (u8 *) &_utf8_string, + 1 + ); + else if (ev.xselectionrequest.target == _utf8_string) + XChangeProperty( + ev.xselectionrequest.display, + ev.xselectionrequest.requestor, + ev.xselectionrequest.property, + ev.xselectionrequest.target, + 8, + PropModeReplace, + (u8 *) platform.clipboard, + platform.clipboard_size + ); + } + + XSendEvent(_display, ev.xselectionrequest.requestor, 0, 0, (XEvent *) &(XSelectionEvent) { + .type = SelectionNotify, + .serial = ev.xselectionrequest.serial, + .send_event = ev.xselectionrequest.send_event, + .display = ev.xselectionrequest.display, + .requestor = ev.xselectionrequest.requestor, + .selection = ev.xselectionrequest.selection, + .target = ev.xselectionrequest.target, + .property = ev.xselectionrequest.property, + .time = ev.xselectionrequest.time, + }); + } + break; + + case SelectionNotify: + if (ev.xselection.property != None) { + i64 len; + u8 *data; + + XGetWindowProperty( + _display, + _window, + _clipboard, + 0, + MAX_CLIPBOARD_SIZE, + False, + AnyPropertyType, + &(Atom) {0}, + &(int) {0}, + (unsigned long *) &len, + &(unsigned long) {0}, + &data + ); + + if (ev.xselection.target == _targets) { + _target = None; + Atom *list = (Atom *) data; + + for (i64 i = 0; i < len; i++) + if (list[i] == XA_STRING) + _target = XA_STRING; + else if (list[i] == _utf8_string) { + _target = _utf8_string; + break; + } + + if (_target != None) + XConvertSelection(_display, _clipboard, _target, _clipboard, _window, CurrentTime); + } else if (ev.xselection.target == _target) { + if (len > MAX_CLIPBOARD_SIZE) + len = MAX_CLIPBOARD_SIZE; + platform.clipboard_size = len; + if (len > 0) + memcpy(platform.clipboard, data, len); + } + + if (data) + XFree(data); + } break; case EnterNotify: platform.has_cursor = 1; break; case LeaveNotify: platform.has_cursor = 0; break; - case FocusIn: platform.has_focus = 1; break; case FocusOut: platform.has_focus = 0; break; + case FocusIn: + platform.has_focus = 1; + if (!requested_clipboard) { + XConvertSelection(_display, _clipboard, _targets, _clipboard, _window, CurrentTime); + requested_clipboard = 1; + } + break; + + case MappingNotify: + XRefreshKeyboardMapping(&ev.xmapping); + break; + case ClientMessage: if ((Atom) ev.xclient.data.l[0] == _wm_delete_window) platform.done = 1; @@ -497,6 +769,12 @@ i32 p_handle_events(void) { } } + for (i64 k = 0; k < (i64) (sizeof platform.key_pressed / sizeof *platform.key_pressed); ++k) + platform.key_pressed[k] = platform.key_pressed[k] > 0 ? 1 : 0; + + XWindowAttributes attrs; + XGetWindowAttributes(_display, _window, &attrs); + if ((platform.frame_width != attrs.width || platform.frame_height != attrs.height) && attrs.width * attrs.height * 4 <= (i32) sizeof _buffer) { if (attrs.width > 0 && attrs.height > 0) { _image.width = attrs.width; @@ -511,6 +789,35 @@ i32 p_handle_events(void) { return num_events; } +i32 p_wait_events(void) { + i32 num_events = 0; + + do { + num_events = p_handle_events(); + sched_yield(); + } while (num_events == 0); + + return num_events; +} + +void p_render_frame(void) { + if (platform.done) + return; + + XPutImage(_display, _window, _gc, &_image, 0, 0, 0, 0, platform.frame_width, platform.frame_height); + XFlush(_display); +} + +void p_clipboard_write(i64 size, c8 *data) { + assert(size <= MAX_CLIPBOARD_SIZE); + + XSetSelectionOwner(_display, _clipboard, _window, CurrentTime); + + platform.clipboard_size = size < MAX_CLIPBOARD_SIZE ? size : MAX_CLIPBOARD_SIZE; + if (platform.clipboard_size > 0) + memcpy(platform.clipboard, data, platform.clipboard_size); +} + #endif // ================================================================ |