2024-03-25 05:34:59 +00:00
|
|
|
#include <iso646.h>
|
|
|
|
#include <stdbool.h>
|
|
|
|
#include <stdint.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
2024-04-11 19:57:22 +00:00
|
|
|
#include <windows.h>
|
2024-03-25 05:34:59 +00:00
|
|
|
|
2024-04-11 19:57:22 +00:00
|
|
|
#include "fumotris.h"
|
|
|
|
#include "ctrl.h"
|
2024-04-16 22:14:53 +00:00
|
|
|
#include "gametime.h"
|
2024-03-25 05:34:59 +00:00
|
|
|
|
2024-04-18 05:04:44 +00:00
|
|
|
static struct windows {
|
2024-04-11 19:57:22 +00:00
|
|
|
HANDLE timer;
|
2024-04-16 22:14:53 +00:00
|
|
|
HANDLE input_handle;
|
2024-04-15 19:29:51 +00:00
|
|
|
} win;
|
2024-04-11 19:57:22 +00:00
|
|
|
|
2024-04-15 19:29:51 +00:00
|
|
|
bool init_handles()
|
2024-03-25 05:34:59 +00:00
|
|
|
{
|
2024-04-16 22:14:53 +00:00
|
|
|
win.input_handle = GetStdHandle(STD_INPUT_HANDLE);
|
|
|
|
if (win.input_handle == INVALID_HANDLE_VALUE)
|
2024-03-25 05:34:59 +00:00
|
|
|
return false;
|
|
|
|
|
2024-04-18 05:04:44 +00:00
|
|
|
win.timer = CreateWaitableTimerW(NULL, TRUE, NULL);
|
2024-04-15 19:29:51 +00:00
|
|
|
if (win.timer == NULL)
|
2024-04-11 19:57:22 +00:00
|
|
|
return false;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2024-04-15 19:29:51 +00:00
|
|
|
bool init_console()
|
2024-04-11 19:57:22 +00:00
|
|
|
{
|
|
|
|
DWORD mode = ENABLE_EXTENDED_FLAGS
|
|
|
|
| ENABLE_PROCESSED_INPUT
|
|
|
|
| ENABLE_PROCESSED_OUTPUT
|
|
|
|
| ENABLE_MOUSE_INPUT
|
|
|
|
| ENABLE_WINDOW_INPUT;
|
2024-04-18 05:04:44 +00:00
|
|
|
|
2024-04-16 22:14:53 +00:00
|
|
|
return SetConsoleMode(win.input_handle, mode) != 0;
|
2024-04-15 19:29:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool PlatformInit()
|
|
|
|
{
|
|
|
|
if (!init_handles())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (!init_console())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return true;
|
2024-04-11 19:57:22 +00:00
|
|
|
}
|
|
|
|
|
2024-04-15 19:29:51 +00:00
|
|
|
bool PlatformGetRefreshRate(u16f *out)
|
2024-04-11 19:57:22 +00:00
|
|
|
{
|
|
|
|
DEVMODE mode;
|
|
|
|
mode.dmSize = sizeof(DEVMODE);
|
|
|
|
mode.dmDriverExtra = 0;
|
|
|
|
|
2024-04-18 05:04:44 +00:00
|
|
|
if(!EnumDisplaySettingsA(NULL, ENUM_CURRENT_SETTINGS, &mode))
|
2024-04-11 19:57:22 +00:00
|
|
|
return false;
|
|
|
|
|
|
|
|
*out = mode.dmDisplayFrequency;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2024-04-18 05:04:44 +00:00
|
|
|
void copy_key_event(struct InputRecord *rec, KEY_EVENT_RECORD *win_key)
|
2024-03-25 05:34:59 +00:00
|
|
|
{
|
2024-04-11 19:57:22 +00:00
|
|
|
rec->type = BUTTON;
|
2024-04-18 05:04:44 +00:00
|
|
|
rec->bind = win_key->wVirtualKeyCode;
|
2024-04-11 19:57:22 +00:00
|
|
|
|
2024-04-18 05:04:44 +00:00
|
|
|
rec->is_down = win_key->bKeyDown;
|
|
|
|
rec->is_up = !win_key->bKeyDown;
|
2024-03-25 05:34:59 +00:00
|
|
|
}
|
|
|
|
|
2024-04-18 05:04:44 +00:00
|
|
|
bool copy_mouse_event(struct InputRecord *rec, MOUSE_EVENT_RECORD *win_mouse)
|
2024-03-25 05:34:59 +00:00
|
|
|
{
|
2024-04-18 05:04:44 +00:00
|
|
|
if (win_mouse->dwEventFlags == MOUSE_MOVED) {
|
2024-04-11 19:57:22 +00:00
|
|
|
rec->type = JOYSTICK;
|
|
|
|
rec->bind = 0;
|
2024-04-18 05:04:44 +00:00
|
|
|
rec->js.x = win_mouse->dwMousePosition.X;
|
|
|
|
rec->js.y = win_mouse->dwMousePosition.Y;
|
2024-04-11 19:57:22 +00:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2024-04-18 05:04:44 +00:00
|
|
|
if (win_mouse->dwEventFlags & (MOUSE_WHEELED | MOUSE_HWHEELED) != 0) {
|
|
|
|
rec->type = AXIS;
|
|
|
|
rec->bind = win_mouse->dwEventFlags == MOUSE_WHEELED;
|
|
|
|
rec->axis.value = win_mouse->dwButtonState;
|
2024-04-11 19:57:22 +00:00
|
|
|
|
2024-04-18 05:04:44 +00:00
|
|
|
return true;
|
|
|
|
}
|
2024-04-03 23:31:47 +00:00
|
|
|
|
2024-04-18 05:04:44 +00:00
|
|
|
return false;
|
2024-04-11 19:57:22 +00:00
|
|
|
}
|
|
|
|
|
2024-04-18 05:04:44 +00:00
|
|
|
bool copy_rec(struct InputRecord *rec, INPUT_RECORD *win_rec)
|
2024-04-11 19:57:22 +00:00
|
|
|
{
|
2024-04-18 05:04:44 +00:00
|
|
|
switch (win_rec->EventType) {
|
2024-04-11 19:57:22 +00:00
|
|
|
case KEY_EVENT:
|
2024-04-18 05:04:44 +00:00
|
|
|
copy_key_event(rec, &win_rec->Event.KeyEvent);
|
2024-04-11 19:57:22 +00:00
|
|
|
return true;
|
|
|
|
|
|
|
|
case MOUSE_EVENT:
|
2024-04-18 05:04:44 +00:00
|
|
|
return copy_mouse_event(rec, &win_rec->Event.MouseEvent);
|
2024-04-11 19:57:22 +00:00
|
|
|
|
|
|
|
case WINDOW_BUFFER_SIZE_EVENT:
|
|
|
|
return false;
|
|
|
|
// TODO: Handle window resizing
|
|
|
|
}
|
|
|
|
|
|
|
|
rec->type = ESCAPE;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2024-04-18 05:04:44 +00:00
|
|
|
bool PlatformReadInput(struct InputBuffer *buf)
|
2024-04-11 19:57:22 +00:00
|
|
|
{
|
2024-04-18 05:04:44 +00:00
|
|
|
DWORD max_records = IO_BUF_SIZE - buf->len;
|
|
|
|
INPUT_RECORD win_buf[max_records];
|
|
|
|
|
|
|
|
DWORD filled;
|
|
|
|
if (!ReadConsoleInputW(win.input_handle, win_buf, max_records, &filled))
|
2024-04-11 19:57:22 +00:00
|
|
|
return false;
|
|
|
|
|
2024-04-16 22:14:53 +00:00
|
|
|
struct InputRecord rec = { .timestamp = TimeNow() };
|
2024-04-11 19:57:22 +00:00
|
|
|
|
2024-04-18 05:04:44 +00:00
|
|
|
for (size_t i = 0; i < filled; i++) {
|
|
|
|
if (!copy_rec(&rec, win_buf + i))
|
2024-04-11 19:57:22 +00:00
|
|
|
continue;
|
|
|
|
|
2024-04-18 05:04:44 +00:00
|
|
|
InputBufferCopy(buf, &rec);
|
2024-04-11 19:57:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2024-04-15 19:29:51 +00:00
|
|
|
bool PlatformWait(struct timespec relative)
|
2024-04-11 19:57:22 +00:00
|
|
|
{
|
|
|
|
LARGE_INTEGER duration;
|
|
|
|
duration.QuadPart = -10000000 * relative.tv_sec - relative.tv_nsec / 100;
|
|
|
|
|
2024-04-18 05:04:44 +00:00
|
|
|
if (!SetWaitableTimer(win.timer, &duration, 0, NULL,NULL, FALSE))
|
2024-04-11 19:57:22 +00:00
|
|
|
return false;
|
|
|
|
|
2024-04-15 19:29:51 +00:00
|
|
|
DWORD result = WaitForSingleObject(win.timer, INFINITE);
|
2024-04-11 19:57:22 +00:00
|
|
|
if (result != WAIT_OBJECT_0)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return true;
|
2024-03-25 05:34:59 +00:00
|
|
|
}
|