|
- /* ----
- * ---- file : lglw_windows.c
- * ---- author : bsp
- * ---- legal : Distributed under terms of the MIT LICENSE (MIT).
- * ----
- * ---- Permission is hereby granted, free of charge, to any person obtaining a copy
- * ---- of this software and associated documentation files (the "Software"), to deal
- * ---- in the Software without restriction, including without limitation the rights
- * ---- to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
- * ---- copies of the Software, and to permit persons to whom the Software is
- * ---- furnished to do so, subject to the following conditions:
- * ----
- * ---- The above copyright notice and this permission notice shall be included in
- * ---- all copies or substantial portions of the Software.
- * ----
- * ---- THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
- * ---- IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
- * ---- FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
- * ---- AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
- * ---- LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
- * ---- OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
- * ---- THE SOFTWARE.
- * ----
- * ---- info : This is part of the "lglw" package.
- * ----
- * ---- created: 04Aug2018
- * ---- changed: 05Aug2018, 06Aug2018, 07Aug2018, 08Aug2018, 09Aug2018, 18Aug2018
- * ----
- * ----
- */
-
- #include "lglw.h"
-
- #include <stdlib.h>
- #include <stdio.h>
- #include <string.h>
-
- #include <windows.h>
- #include <windowsx.h>
-
- // #define Dprintf if(0);else printf
- #define Dprintf if(1);else printf
-
- // ---------------------------------------------------------------------------- macros and defines
- #define LGLW(a) lglw_int_t *lglw = ((lglw_int_t*)(a))
-
- #define LGLW_HIDDEN_BASE_CLASS_NAME "hidden_LGLW"
- #define LGLW_WIN_BASE_CLASS_NAME "LGLW_"
-
- #define LGLW_DEFAULT_HIDDEN_W (800)
- #define LGLW_DEFAULT_HIDDEN_H (600)
-
- #define LGLW_MOUSE_TOUCH_LMB_TIMEOUT (250u)
- #define LGLW_MOUSE_TOUCH_RMB_TIMEOUT (500u)
- #define LGLW_MOUSE_TOUCH_RMB_STATE_IDLE (0u)
- #define LGLW_MOUSE_TOUCH_RMB_STATE_LMB (1u)
- #define LGLW_MOUSE_TOUCH_RMB_STATE_WAIT (2u)
- #define LGLW_MOUSE_TOUCH_RMB_STATE_RMB (3u)
- #define LGLW_MOUSE_TOUCH_RMB_MOVE_THRESHOLD (7u)
-
- #define sABS(x) (((x)>0)?(x):-(x))
-
-
- // ---------------------------------------------------------------------------- structs and typedefs
- typedef struct lglw_int_s {
- void *user_data; // arbitrary user data
- HWND parent_hwnd; // created by host
-
- struct {
- char class_name[128];
- HWND hwnd;
- HDC hdc;
- lglw_vec2i_t size;
- } hidden;
-
- struct {
- char class_name[128];
- HWND hwnd;
- HDC hdc;
- HHOOK keyboard_ll_hhook;
- lglw_vec2i_t size;
- int32_t swap_interval;
- } win;
-
- struct {
- HDC hdc;
- HGLRC hglrc;
- } prev;
-
- HGLRC hglrc;
-
- struct {
- uint32_t kmod_state; // See LGLW_KMOD_xxx
- lglw_keyboard_fxn_t cbk;
- } keyboard;
-
- struct {
- lglw_vec2i_t p; // last seen mouse position
- uint32_t button_state;
- lglw_mouse_fxn_t cbk;
- struct {
- uint32_t mode;
- lglw_vec2i_t p; // grab-start mouse position
- lglw_bool_t b_queue_warp;
- lglw_vec2i_t last_p;
- } grab;
- struct {
- lglw_bool_t b_enable;
- lglw_bool_t b_update_queued;
- lglw_bool_t b_syn_rmb;
- uint32_t syn_rmb_hold_state; // see LGLW_MOUSE_TOUCH_RMB_STATE_xxx
- uint32_t hold_start_ms;
- lglw_vec2i_t hold_start_p;
- } touch;
- } mouse;
-
- struct {
- uint32_t state;
- lglw_focus_fxn_t cbk;
- } focus;
-
- struct {
- lglw_bool_t b_running;
- lglw_timer_fxn_t cbk;
- } timer;
-
- struct {
- lglw_dropfiles_fxn_t cbk;
- } dropfiles;
-
- struct {
- lglw_redraw_fxn_t cbk;
- } redraw;
-
- } lglw_int_t;
-
-
- // ---------------------------------------------------------------------------- module fxn fwd decls
- static lglw_bool_t loc_create_hidden_window (lglw_int_t *lglw, int32_t _w, int32_t _h);
- static void loc_destroy_hidden_window(lglw_int_t *lglw);
-
- static LRESULT CALLBACK loc_WndProc (HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam);
-
- static void loc_key_hook(lglw_int_t *lglw);
- static void loc_key_unhook(lglw_int_t *lglw);
- static LRESULT CALLBACK loc_LowLevelKeyboardProc(int nCode, WPARAM wParam, LPARAM lParam);
- static lglw_bool_t loc_handle_key (lglw_int_t *lglw, lglw_bool_t _bPressed, uint32_t _vkey);
- // static lglw_bool_t loc_touchkeyboard_get_rect (RECT *rect);
- // static lglw_bool_t loc_touchkeyboard_is_visible (void);
- extern lglw_bool_t lglw_int_touchkeyboard_toggle (void);
-
- static void loc_handle_mouseleave (lglw_int_t *lglw);
- static void loc_handle_mouseenter (lglw_int_t *lglw);
- static void loc_handle_mousebutton (lglw_int_t *lglw, lglw_bool_t _bPressed, uint32_t _button);
- static void loc_handle_mousemotion (lglw_int_t *lglw);
- static void loc_handle_queued_mouse_warp (lglw_int_t *lglw);
- static void loc_touchinput_update (lglw_int_t *lglw);
-
- static void loc_enable_dropfiles (lglw_int_t *lglw, lglw_bool_t _bEnable);
-
-
- // ---------------------------------------------------------------------------- module vars
- static lglw_int_t *khook_lglw = NULL; // currently key-hooked lglw instance (one at a time)
-
-
- // ---------------------------------------------------------------------------- lglw_init
- lglw_t lglw_init(int32_t _w, int32_t _h) {
- lglw_int_t *lglw = malloc(sizeof(lglw_int_t));
-
- if(NULL != lglw)
- {
- memset(lglw, 0, sizeof(lglw_int_t));
-
- if(_w <= 16)
- _w = LGLW_DEFAULT_HIDDEN_W;
-
- if(_h <= 16)
- _h = LGLW_DEFAULT_HIDDEN_H;
-
- if(!loc_create_hidden_window(lglw, _w, _h))
- {
- free(lglw);
- lglw = NULL;
- }
- }
-
- return lglw;
- }
-
-
- // ---------------------------------------------------------------------------- lglw_exit
- void lglw_exit(lglw_t _lglw) {
- LGLW(_lglw);
-
- if(NULL != lglw)
- {
- loc_destroy_hidden_window(lglw);
-
- free(lglw);
- }
- }
-
-
- // ---------------------------------------------------------------------------- lglw_userdata_set
- void lglw_userdata_set(lglw_t _lglw, void *_userData) {
- LGLW(_lglw);
-
- if(NULL != lglw)
- {
- lglw->user_data = _userData;
- }
- }
-
- // ---------------------------------------------------------------------------- lglw_userdata_get
- void *lglw_userdata_get(lglw_t _lglw) {
- LGLW(_lglw);
-
- if(NULL != lglw)
- {
- return lglw->user_data;
- }
-
- return NULL;
- }
-
-
- // ---------------------------------------------------------------------------- loc_create_hidden_window
- static lglw_bool_t loc_create_hidden_window(lglw_int_t *lglw, int32_t _w, int32_t _h) {
-
- sprintf(lglw->hidden.class_name, LGLW_HIDDEN_BASE_CLASS_NAME "%p", lglw);
-
- WNDCLASS wc;
- ZeroMemory(&wc, sizeof(wc));
- wc.style = CS_OWNDC;
- wc.lpfnWndProc = (WNDPROC) &loc_WndProc;
- wc.hInstance = GetModuleHandle(NULL);
- wc.hCursor = LoadCursor(NULL, IDC_ARROW);
- wc.lpszClassName = lglw->hidden.class_name;
-
- if(!RegisterClass(&wc))
- {
- // something went terribly wrong
- printf("[---] lglw: failed to register hidden window class\n");
- return LGLW_FALSE;
- }
-
- DWORD dwExStyle = 0;
- DWORD dwStyle = 0;
-
- lglw->hidden.hwnd = CreateWindowEx(dwExStyle,
- lglw->hidden.class_name,
- "LGLW_hidden",
- dwStyle,
- 0/*xpos*/, 0/*ypos*/,
- _w, _h,
- NULL/*parentHWND*/,
- NULL/*window menu*/,
- GetModuleHandle(NULL),
- NULL
- );
-
- lglw->hidden.size.x = _w;
- lglw->hidden.size.y = _h;
-
- PIXELFORMATDESCRIPTOR pfd = {
- sizeof(PIXELFORMATDESCRIPTOR),
- 1,
- PFD_DRAW_TO_WINDOW | PFD_SUPPORT_OPENGL | PFD_DOUBLEBUFFER, //Flags
- PFD_TYPE_RGBA, // The kind of framebuffer. RGBA or palette.
- 32, // Colordepth of the framebuffer.
- 0, 0, 0, 0, 0, 0,
- 0,
- 0,
- 0,
- 0, 0, 0, 0,
- 24, // Number of bits for the depthbuffer
- 8, // Number of bits for the stencilbuffer
- 0, // Number of Aux buffers in the framebuffer.
- PFD_MAIN_PLANE,
- 0,
- 0, 0, 0
- };
-
- lglw->hidden.hdc = GetDC(lglw->hidden.hwnd);
- Dprintf("xxx lglw: hidden hdc=%p\n", lglw->hidden.hdc);
-
- int pfmt = ChoosePixelFormat(lglw->hidden.hdc, &pfd);
- Dprintf("xxx lglw: hidden pfmt=%d\n", pfmt);
- SetPixelFormat(lglw->hidden.hdc, pfmt, &pfd);
-
- lglw->hglrc = wglCreateContext(lglw->hidden.hdc);
- Dprintf("xxx lglw: hidden hglrc=%p\n", lglw->hglrc);
-
- return LGLW_TRUE;
- }
-
-
- // ---------------------------------------------------------------------------- loc_destroy_hidden_window
- static void loc_destroy_hidden_window(lglw_int_t *lglw) {
- if(NULL != lglw->hidden.hwnd)
- {
- wglDeleteContext(lglw->hglrc);
- lglw->hglrc = NULL;
-
- DestroyWindow(lglw->hidden.hwnd);
- lglw->hidden.hwnd = NULL;
-
- UnregisterClass(lglw->hidden.class_name, GetModuleHandle(NULL));
- }
- }
-
-
- // ---------------------------------------------------------------------------- lglw_window_open
- lglw_bool_t lglw_window_open (lglw_t _lglw, void *_parentHWNDOrNull, int32_t _x, int32_t _y, int32_t _w, int32_t _h) {
- lglw_bool_t r = LGLW_FALSE;
- LGLW(_lglw);
-
- if(NULL != lglw)
- {
- lglw->parent_hwnd = (HWND)_parentHWNDOrNull;
-
- sprintf(lglw->win.class_name, LGLW_WIN_BASE_CLASS_NAME "%p", lglw);
-
- WNDCLASS wc;
- ZeroMemory(&wc, sizeof(wc));
- wc.style = CS_HREDRAW | CS_VREDRAW | CS_OWNDC;
- wc.lpfnWndProc = (WNDPROC) &loc_WndProc;
- wc.hInstance = GetModuleHandle(NULL);
- wc.hCursor = LoadCursor(NULL, IDC_ARROW);
- wc.lpszClassName = lglw->win.class_name;
-
- if(!RegisterClass(&wc))
- {
- // something went terribly wrong
- printf("[---] lglw: failed to register window class\n");
- return LGLW_FALSE;
- }
-
- if(_w <= 16)
- _w = lglw->hidden.size.x;
-
- if(_h <= 16)
- _h = lglw->hidden.size.y;
-
- DWORD dwExStyle = 0;
- DWORD dwStyle = ((NULL != _parentHWNDOrNull) ? WS_CHILD : 0) | WS_VISIBLE;
-
- lglw->win.hwnd = CreateWindowEx(dwExStyle,
- lglw->win.class_name,
- "LGLW",
- dwStyle,
- (NULL == _parentHWNDOrNull) ? _x : 0,
- (NULL == _parentHWNDOrNull) ? _y : 0,
- _w, _h,
- _parentHWNDOrNull,
- NULL/*window menu*/,
- GetModuleHandle(NULL),
- NULL
- );
-
- lglw->win.size.x = _w;
- lglw->win.size.y = _h;
-
- // window_to_wrapper = this;
-
- PIXELFORMATDESCRIPTOR pfd = {
- sizeof(PIXELFORMATDESCRIPTOR),
- 1,
- PFD_DRAW_TO_WINDOW | PFD_SUPPORT_OPENGL | PFD_DOUBLEBUFFER, //Flags
- PFD_TYPE_RGBA, // The kind of framebuffer. RGBA or palette.
- 32, // Colordepth of the framebuffer.
- 0, 0, 0, 0, 0, 0,
- 0,
- 0,
- 0,
- 0, 0, 0, 0,
- 24, // Number of bits for the depthbuffer
- 8, // Number of bits for the stencilbuffer
- 0, // Number of Aux buffers in the framebuffer.
- PFD_MAIN_PLANE,
- 0,
- 0, 0, 0
- };
-
- lglw->win.hdc = GetDC(lglw->win.hwnd);
-
- int pfmt = ChoosePixelFormat(lglw->win.hdc, &pfd);
- Dprintf("xxx lglw: win pfmt=%d\n", pfmt);
- SetPixelFormat(lglw->win.hdc, pfmt, &pfd);
-
- (void)SetWindowLongPtr(lglw->win.hwnd, GWLP_USERDATA, (LONG_PTR)lglw);
-
- loc_enable_dropfiles(lglw, (NULL != lglw->dropfiles.cbk));
- }
- return r;
- }
-
-
- // ---------------------------------------------------------------------------- lglw_window_resize
- lglw_bool_t lglw_window_resize (lglw_t _lglw, int32_t _w, int32_t _h) {
- lglw_bool_t r = LGLW_FALSE;
- LGLW(_lglw);
-
- if(NULL != lglw)
- {
- if(NULL != lglw->win.hwnd)
- {
- r = LGLW_TRUE;
-
- SetWindowPos(lglw->win.hwnd,
- HWND_TOP,
- 0, 0, // x/y
- _w, _h,
- SWP_NOMOVE | SWP_SHOWWINDOW
- );
-
- int deltaW = _w - lglw->win.size.x;
- int deltaH = _h - lglw->win.size.y;
-
- lglw->win.size.x = _w;
- lglw->win.size.y = _h;
-
- // Resize parent window (if any)
- {
- HWND hwnd;
- hwnd = GetAncestor(lglw->win.hwnd, GA_ROOT);
- Dprintf("xxx lglw_window_resize: ancestor=%p\n", hwnd);
- if(NULL != hwnd)
- {
- RECT rect;
- if(GetWindowRect(hwnd, &rect))
- {
- rect.right += deltaW;
- rect.bottom += deltaH;
-
- SetWindowPos(hwnd,
- HWND_TOP,
- 0, 0, // x/y
- rect.right - rect.left, rect.bottom - rect.top,
- SWP_NOMOVE | SWP_FRAMECHANGED | SWP_SHOWWINDOW
- );
- }
- }
- }
- }
- }
-
- return r;
- }
-
-
- // ---------------------------------------------------------------------------- lglw_window_close
- void lglw_window_close (lglw_t _lglw) {
- LGLW(_lglw);
-
- if(NULL != lglw)
- {
- if(NULL != lglw->win.hwnd)
- {
- lglw_timer_stop(_lglw);
-
- loc_key_unhook(lglw);
-
- DestroyWindow(lglw->win.hwnd);
- lglw->win.hwnd = NULL;
- lglw->win.hdc = NULL;
-
- UnregisterClass(lglw->win.class_name, GetModuleHandle(NULL));
-
- // window_to_wrapper = NULL;
- }
- }
- }
-
-
- // ---------------------------------------------------------------------------- lglw_window_show
- void lglw_window_show(lglw_t _lglw) {
- LGLW(_lglw);
-
- if(NULL != lglw)
- {
- if(NULL != lglw->win.hwnd)
- {
- ShowWindow(lglw->win.hwnd, SW_SHOWNORMAL);
- }
- }
- }
-
-
- // ---------------------------------------------------------------------------- lglw_window_hide
- void lglw_window_hide(lglw_t _lglw) {
- LGLW(_lglw);
-
- if(NULL != lglw)
- {
- if(NULL != lglw->win.hwnd)
- {
- ShowWindow(lglw->win.hwnd, SW_HIDE);
- }
- }
- }
-
-
- // ---------------------------------------------------------------------------- lglw_window_is_visible
- lglw_bool_t lglw_window_is_visible(lglw_t _lglw) {
- lglw_bool_t r = LGLW_FALSE;
- LGLW(_lglw);
-
- if(NULL != lglw)
- {
- r = (lglw_bool_t)IsWindowVisible(lglw->win.hwnd);
- }
-
- return r;
- }
-
-
- // ---------------------------------------------------------------------------- lglw_window_size_get
- void lglw_window_size_get(lglw_t _lglw, int32_t *_retX, int32_t *_retY) {
- LGLW(_lglw);
-
- if(NULL != lglw)
- {
- if(NULL != lglw->win.hwnd)
- {
- if(NULL != _retX)
- *_retX = lglw->win.size.x;
-
- if(NULL != _retY)
- *_retY = lglw->win.size.y;
- }
- }
- }
-
-
- // ---------------------------------------------------------------------------- lglw_redraw
- void lglw_redraw(lglw_t _lglw) {
- LGLW(_lglw);
-
- if(NULL != lglw)
- {
- if(NULL != lglw->win.hwnd)
- {
- RedrawWindow(lglw->win.hwnd, NULL/*lprcUpdate*/, NULL/*hrgnUpdate*/, RDW_INTERNALPAINT | RDW_UPDATENOW);
- }
- }
- }
-
-
- // ---------------------------------------------------------------------------- lglw_redraw_callback_set
- void lglw_redraw_callback_set(lglw_t _lglw, lglw_redraw_fxn_t _cbk) {
- LGLW(_lglw);
-
- if(NULL != lglw)
- {
- lglw->redraw.cbk = _cbk;
- }
- }
-
-
- // ---------------------------------------------------------------------------- lglw_glcontext_push
- void lglw_glcontext_push(lglw_t _lglw) {
- LGLW(_lglw);
-
- if(NULL != lglw)
- {
- lglw->prev.hdc = wglGetCurrentDC();
- lglw->prev.hglrc = wglGetCurrentContext();
- // Dprintf("xxx lglw_glcontext_push: prev.hdc=%p prev.hglrc=%p\n", lglw->prev.hdc, lglw->prev.hglrc);
-
- // Dprintf("xxx lglw_glcontext_push: win.hdc=%p lglw->hglrc=%p\n", lglw->win.hdc, lglw->hglrc);
- if(!wglMakeCurrent((NULL == lglw->win.hdc) ? lglw->hidden.hdc : lglw->win.hdc, lglw->hglrc))
- {
- printf("[---] lglw_glcontext_push: wglMakeCurrent() failed. win.hdc=%p hidden.hdc=%p hglrc=%p GetLastError()=%d\n", lglw->win.hdc, lglw->hidden.hdc, lglw->hglrc, GetLastError());
- }
- }
- }
-
-
- // ---------------------------------------------------------------------------- lglw_glcontext_pop
- void lglw_glcontext_pop(lglw_t _lglw) {
- LGLW(_lglw);
-
- if(NULL != lglw)
- {
- // Dprintf("xxx lglw_glcontext_pop: prev.hdc=%p prev.hglrc=%p\n", lglw->prev.hdc, lglw->prev.hglrc);
- if(!wglMakeCurrent(lglw->prev.hdc, lglw->prev.hglrc))
- {
- printf("[---] lglw_glcontext_pop: wglMakeCurrent() failed. prev.hdc=%p hglrc=%p GetLastError()=%d\n", lglw->prev.hdc, lglw->prev.hglrc, GetLastError());
- }
- }
- }
-
-
- // ---------------------------------------------------------------------------- lglw_swap_buffers
- void lglw_swap_buffers(lglw_t _lglw) {
- LGLW(_lglw);
-
- if(NULL != lglw)
- {
- wglSwapLayerBuffers(lglw->win.hdc, WGL_SWAP_MAIN_PLANE);
- }
- }
-
-
- // ---------------------------------------------------------------------------- lglw_swap_interval_set
- typedef void (APIENTRY *PFNWGLEXTSWAPINTERVALPROC) (int);
- void lglw_swap_interval_set(lglw_t _lglw, int32_t _ival) {
- LGLW(_lglw);
-
- if(NULL != lglw)
- {
- PFNWGLEXTSWAPINTERVALPROC wglSwapIntervalEXT;
- wglSwapIntervalEXT = (PFNWGLEXTSWAPINTERVALPROC) wglGetProcAddress("wglSwapIntervalEXT");
- if(NULL != wglSwapIntervalEXT)
- {
- wglSwapIntervalEXT(_ival);
- lglw->win.swap_interval = _ival;
- }
- }
- }
-
-
- // ---------------------------------------------------------------------------- lglw_swap_interval_get
- int32_t lglw_swap_interval_get(lglw_t _lglw) {
- LGLW(_lglw);
- int32_t r = 0;
-
- if(NULL != lglw)
- {
- r = lglw->win.swap_interval;
- }
-
- return r;
- }
-
-
- // ---------------------------------------------------------------------------- loc_key_hook
- static void loc_key_hook(lglw_int_t *lglw) {
- loc_key_unhook(lglw);
-
- // https://msdn.microsoft.com/en-us/library/windows/desktop/ms644990(v=vs.85).aspx
- lglw->win.keyboard_ll_hhook = SetWindowsHookEx(WH_KEYBOARD_LL, &loc_LowLevelKeyboardProc, GetModuleHandle(NULL), 0/*dwThreadId*/);
-
- khook_lglw = lglw;
- }
-
-
- // ---------------------------------------------------------------------------- loc_key_unhook
- static void loc_key_unhook(lglw_int_t *lglw) {
- if(NULL != lglw->win.keyboard_ll_hhook)
- {
- UnhookWindowsHookEx(lglw->win.keyboard_ll_hhook);
- lglw->win.keyboard_ll_hhook = NULL;
-
- if(khook_lglw == lglw)
- khook_lglw = NULL;
- }
- }
-
-
- // ---------------------------------------------------------------------------- loc_handle_mouseleave
- static void loc_handle_mouseleave(lglw_int_t *lglw) {
- loc_key_unhook(lglw);
-
- lglw->focus.state &= ~LGLW_FOCUS_MOUSE;
-
- if(NULL != lglw->focus.cbk)
- {
- lglw->focus.cbk(lglw, lglw->focus.state, LGLW_FOCUS_MOUSE);
- }
-
- Dprintf("xxx lglw:loc_handle_mouseleave: LEAVE\n");
- }
-
-
- // ---------------------------------------------------------------------------- loc_handle_mouseenter
- static void loc_handle_mouseenter(lglw_int_t *lglw) {
- TRACKMOUSEEVENT tme;
- ZeroMemory(&tme, sizeof(tme));
- tme.cbSize = sizeof(TRACKMOUSEEVENT);
- tme.dwFlags = TME_LEAVE;
- tme.hwndTrack = lglw->win.hwnd;
- tme.dwHoverTime = HOVER_DEFAULT;
- (void)TrackMouseEvent(&tme);
-
- loc_key_hook(lglw);
-
- lglw->focus.state |= LGLW_FOCUS_MOUSE;
-
- if(NULL != lglw->focus.cbk)
- {
- lglw->focus.cbk(lglw, lglw->focus.state, LGLW_FOCUS_MOUSE);
- }
-
- Dprintf("xxx lglw:loc_handle_mouseenter: LEAVE\n");
- }
-
-
- // ---------------------------------------------------------------------------- loc_handle_mousebutton
- static void loc_handle_mousebutton(lglw_int_t *lglw, lglw_bool_t _bPressed, uint32_t _button) {
- if(_bPressed)
- lglw->mouse.button_state |= _button;
- else
- lglw->mouse.button_state &= ~_button;
-
- if(NULL != lglw->mouse.cbk)
- {
- lglw->mouse.cbk(lglw, lglw->mouse.p.x, lglw->mouse.p.y, lglw->mouse.button_state, _button);
- }
- }
-
-
- // ---------------------------------------------------------------------------- loc_handle_mousemotion
- static void loc_handle_mousemotion(lglw_int_t *lglw) {
-
- if(NULL != lglw->mouse.cbk)
- {
- lglw->mouse.cbk(lglw, lglw->mouse.p.x, lglw->mouse.p.y, lglw->mouse.button_state, 0u/*changedbuttonstate*/);
- }
- }
-
-
- // ---------------------------------------------------------------------------- lglw_mouse_callback_set
- void lglw_mouse_callback_set(lglw_t _lglw, lglw_mouse_fxn_t _cbk) {
- LGLW(_lglw);
-
- if(NULL != lglw)
- {
- lglw->mouse.cbk = _cbk;
- }
- }
-
-
- // ---------------------------------------------------------------------------- lglw_mouse_callback_set
- void lglw_focus_callback_set(lglw_t _lglw, lglw_focus_fxn_t _cbk) {
- LGLW(_lglw);
-
- if(NULL != lglw)
- {
- lglw->focus.cbk = _cbk;
- }
- }
-
-
- // ---------------------------------------------------------------------------- loc_handle_key
- static lglw_bool_t loc_handle_key(lglw_int_t *lglw, lglw_bool_t _bPressed, uint32_t _vkey) {
- lglw_bool_t r = LGLW_FALSE;
-
- if(NULL != lglw->keyboard.cbk)
- {
- r = lglw->keyboard.cbk(lglw, _vkey, lglw->keyboard.kmod_state, _bPressed);
- }
-
- return r;
- }
-
-
- // ---------------------------------------------------------------------------- lglw_keyboard_callback_set
- void lglw_keyboard_callback_set(lglw_t _lglw, lglw_keyboard_fxn_t _cbk) {
- LGLW(_lglw);
-
- if(NULL != lglw)
- {
- lglw->keyboard.cbk = _cbk;
- }
- }
-
-
- // ---------------------------------------------------------------------------- lglw_keyboard_get_modifiers
- uint32_t lglw_keyboard_get_modifiers(lglw_t _lglw) {
- uint32_t r = 0u;
- LGLW(_lglw);
-
- if(NULL != lglw)
- {
- r = lglw->keyboard.kmod_state;
- }
-
- return r;
- }
-
-
- // ---------------------------------------------------------------------------- loc_touchkeyboard_get_rect
- #if 0
- static lglw_bool_t loc_touchkeyboard_get_rect(RECT *rect) {
- lglw_bool_t r = LGLW_FALSE;
- //*******************************************************************
- //
- // RETURNS KEYBOARD RECTANGLE OR EMPTY ONE IF KEYBOARD IS NOT VISIBLE
- //
- //*******************************************************************
- // by "Sevast", <https://stackoverflow.com/questions/38774139/show-touch-keyboard-tabtip-exe-in-windows-10-anniversary-edition>
- IFrameworkInputPane *inputPane = NULL;
- HRESULT hr = CoInitializeEx(NULL, COINIT_APARTMENTTHREADED);
- rect->left = rect->right = rect->top = rect->bottom = 0; // screen location
- if (SUCCEEDED(hr))
- {
- hr = CoCreateInstance(CLSID_FrameworkInputPane, NULL, CLSCTX_INPROC_SERVER, IID_IFrameworkInputPane, (LPVOID*)&inputPane);
- if (SUCCEEDED(hr))
- {
- hr=inputPane->Location(rect);
- if(SUCCEEDED(hr))
- {
- r = LGLW_TRUE;
- }
- inputPane->Release();
- }
- }
- CoUninitialize();
- return r;
- }
- #endif
-
-
-
- // ---------------------------------------------------------------------------- loc_touchkeyboard_is_visible
- #if 0
- static lglw_bool_t loc_touchkeyboard_is_visible(void) {
- lglw_bool_t r = LGLW_FALSE;
- HWND hwnd = FindWindow("IPTIP_Main_Window", NULL);
- printf("xxx lglw: IPTIP hwnd=%p\n", hwnd);
- if(NULL != hwnd)
- {
- #if 0
- // (note) does not work, style flags are always the same and rect is always (0; 0; 0; 0)
- LONG style = GetWindowLong(hwnd, GWL_STYLE);
- RECT rect;
- r = (0u != (style & WS_VISIBLE)) && (0u == (style & WS_DISABLED));
- GetWindowRect(hwnd, &rect);
- printf("xxx lglw: touchkeyboard_is_visible: style=0x%08x rect=(%d; %d; %d; %d) r=%d\n", style, rect.top, rect.left, rect.bottom, rect.right, r);
- #endif
- // (note) does not work, either
- static const char *WindowClass1709 = "Windows.UI.Core.CoreWindow";
- static const char *WindowCaption1709 = "Microsoft Text Input Application";
- static const char *WindowParentClass1709 = "ApplicationFrameWindow";
- HWND hwnd1709 = FindWindowEx(NULL, NULL, WindowClass1709, WindowCaption1709);
- printf("xxx lglw: touchkeyboard_is_visible: hwnd1709=%p\n", hwnd1709);
- // if there is a top-level window - the keyboard is closed
- if(NULL != hwnd1709)
- {
- r = LGLW_FALSE;
- }
- else
- {
- HWND parent = NULL;
- for(;;)
- {
- HWND childHWND;
-
- parent = FindWindowEx(NULL, parent, WindowParentClass1709, NULL);
- if(NULL == parent)
- break; // no more windows, keyboard state is unknown
-
- // if it's a child of a WindowParentClass1709 window - the keyboard is open
- childHWND = FindWindowEx(parent, NULL, WindowClass1709, WindowCaption1709);
-
- if (NULL != childHWND)
- {
- r = LGLW_TRUE;
- break;
- }
- }
- }
- }
- return r;
- }
- #endif
-
-
- // ---------------------------------------------------------------------------- lglw_touchkeyboard_show
- void lglw_touchkeyboard_show(lglw_t _lglw, lglw_bool_t _bEnable) {
- LGLW(_lglw);
-
- if(NULL != lglw)
- {
- #if 0
- RECT keyRect;
- if(loc_touchkeyboard_get_rect(&keyRect))
- {
- lglw_bool_t bToggle = _bEnable ^ ((keyRect.right - keyRect.left) > 0);
-
- if(bToggle)
- {
- loc_touchkeyboard_toggle();
- }
- }
- #endif
-
- #if 0
- lglw_bool_t bToggle = _bEnable ^ loc_touchkeyboard_is_visible();
-
- if(bToggle)
- {
- lglw_int_touchkeyboard_toggle();
- }
- #endif
-
- // (note) there seems to be no (sane) way to determine whether the virtual keyboard is visible.
- // the solutions suggested on stack overflow don't work (anymore).
- // => just toggle the kbd
- lglw_int_touchkeyboard_toggle();
- }
- }
-
-
- // ---------------------------------------------------------------------------- lglw_mouse_get_buttons
- uint32_t lglw_mouse_get_buttons(lglw_t _lglw) {
- uint32_t r = 0u;
- LGLW(_lglw);
-
- if(NULL != lglw)
- {
- r = lglw->mouse.button_state;
- }
-
- return r;
- }
-
-
- // ---------------------------------------------------------------------------- lglw_mouse_grab
- void lglw_mouse_grab(lglw_t _lglw, uint32_t _grabMode) {
- LGLW(_lglw);
-
- if(NULL != lglw)
- {
- if(NULL != lglw->win.hwnd)
- {
- if(!lglw->mouse.touch.b_enable)
- {
- if(lglw->mouse.grab.mode != _grabMode)
- {
- lglw_mouse_ungrab(_lglw);
- }
-
- switch(_grabMode)
- {
- default:
- case LGLW_MOUSE_GRAB_NONE:
- break;
-
- case LGLW_MOUSE_GRAB_CAPTURE:
- (void)SetCapture(lglw->win.hwnd);
- lglw->mouse.grab.mode = _grabMode;
- break;
-
- case LGLW_MOUSE_GRAB_WARP:
- (void)SetCapture(lglw->win.hwnd);
- lglw_mouse_cursor_show(_lglw, LGLW_FALSE);
- lglw->mouse.grab.p = lglw->mouse.p;
- lglw->mouse.grab.last_p = lglw->mouse.p;
- lglw->mouse.grab.mode = _grabMode;
- break;
- }
- }
- }
- }
- }
-
-
- // ---------------------------------------------------------------------------- lglw_mouse_ungrab
- void lglw_mouse_ungrab(lglw_t _lglw) {
- LGLW(_lglw);
-
- if(NULL != lglw)
- {
- if(NULL != lglw->win.hwnd)
- {
- if(!lglw->mouse.touch.b_enable)
- {
- switch(lglw->mouse.grab.mode)
- {
- default:
- case LGLW_MOUSE_GRAB_NONE:
- break;
-
- case LGLW_MOUSE_GRAB_CAPTURE:
- (void)ReleaseCapture();
- lglw->mouse.grab.mode = LGLW_MOUSE_GRAB_NONE;
- break;
-
- case LGLW_MOUSE_GRAB_WARP:
- (void)ReleaseCapture();
- lglw->mouse.grab.mode = LGLW_MOUSE_GRAB_NONE;
- lglw->mouse.grab.b_queue_warp = LGLW_TRUE;
- lglw_mouse_cursor_show(_lglw, LGLW_TRUE);
- break;
- }
- }
- }
- }
- }
-
-
- // ---------------------------------------------------------------------------- lglw_mouse_warp
- void lglw_mouse_warp(lglw_t _lglw, int32_t _x, int32_t _y) {
- LGLW(_lglw);
-
- if(NULL != lglw)
- {
- if(NULL != lglw->win.hwnd)
- {
- POINT p;
- p.x = _x;
- p.y = _y;
-
- if(ClientToScreen(lglw->win.hwnd, &p))
- {
- SetCursorPos(p.x, p.y);
-
- if(LGLW_MOUSE_GRAB_WARP != lglw->mouse.grab.mode)
- {
- lglw->mouse.p.x = p.x;
- lglw->mouse.p.y = p.y;
- }
- }
- }
- }
- }
-
-
- // ---------------------------------------------------------------------------- loc_handle_queued_mouse_warp
- static void loc_handle_queued_mouse_warp(lglw_int_t *lglw) {
- if(lglw->mouse.grab.b_queue_warp)
- {
- lglw->mouse.grab.b_queue_warp = LGLW_FALSE;
- lglw_mouse_warp(lglw, lglw->mouse.grab.p.x, lglw->mouse.grab.p.y);
- lglw->mouse.grab.last_p = lglw->mouse.grab.p;
- }
- }
-
-
- // ---------------------------------------------------------------------------- lglw_mouse_cursor_show
- void lglw_mouse_cursor_show (lglw_t _lglw, lglw_bool_t _bShow) {
- LGLW(_lglw);
-
- if(NULL != lglw)
- {
- if(NULL != lglw->win.hwnd)
- {
- (void)ShowCursor(_bShow);
- }
- }
- }
-
-
- // ---------------------------------------------------------------------------- lglw_timer_start
- void lglw_timer_start(lglw_t _lglw, uint32_t _millisec) {
- LGLW(_lglw);
-
- if(NULL != lglw)
- {
- if(NULL != lglw->win.hwnd)
- {
- (void)SetTimer(lglw->win.hwnd, 1/*nIDEvent*/, _millisec, NULL/*lpTimerFunc*/);
- lglw->timer.b_running = LGLW_TRUE;
- }
- }
- }
-
-
- // ---------------------------------------------------------------------------- lglw_timer_stop
- void lglw_timer_stop(lglw_t _lglw) {
- LGLW(_lglw);
-
- if(NULL != lglw)
- {
- if(NULL != lglw->win.hwnd)
- {
- if(lglw->timer.b_running)
- {
- (void)KillTimer(lglw->win.hwnd, 1/*nIDEvent*/);
- lglw->timer.b_running = LGLW_FALSE;
- }
- }
- }
- }
-
-
- // ---------------------------------------------------------------------------- lglw_timer_callback_set
- void lglw_timer_callback_set(lglw_t _lglw, lglw_timer_fxn_t _cbk) {
- LGLW(_lglw);
-
- if(NULL != lglw)
- {
- lglw->timer.cbk = _cbk;
- }
- }
-
-
- // ---------------------------------------------------------------------------- loc_enable_dropfiles
- static void loc_enable_dropfiles(lglw_int_t *lglw, lglw_bool_t _bEnable) {
- if(NULL != lglw->win.hwnd)
- {
- DragAcceptFiles(lglw->win.hwnd, _bEnable ? TRUE : FALSE);
- }
- }
-
-
- // ---------------------------------------------------------------------------- lglw_dropfiles_callback_set
- void lglw_dropfiles_callback_set(lglw_t _lglw, lglw_dropfiles_fxn_t _cbk) {
- LGLW(_lglw);
-
- if(NULL != _lglw)
- {
- lglw->dropfiles.cbk = _cbk;
-
- loc_enable_dropfiles(lglw, (NULL != _cbk));
- }
- }
-
-
- // ---------------------------------------------------------------------------- loc_touchinput_update
- typedef void (*EnableMouseInPointer_fxn_t)(BOOL fEnable);
- static void loc_touchinput_update(lglw_int_t *lglw) {
- HMODULE hDLL = LoadLibrary("User32.dll");
-
- if(NULL != hDLL)
- {
- EnableMouseInPointer_fxn_t enableMouseInPointer = (EnableMouseInPointer_fxn_t)
- GetProcAddress(hDLL, "EnableMouseInPointer");
-
- if(NULL != enableMouseInPointer)
- {
- Dprintf("[dbg] lglw: EnableMouseInPointer() detected (win8+)\n");
- (void)enableMouseInPointer(lglw->mouse.touch.b_enable);
- }
-
- FreeLibrary(hDLL);
- }
- }
-
-
- // ---------------------------------------------------------------------------- lglw_touchinput_set
- void lglw_touchinput_set(lglw_t _lglw, lglw_bool_t _bEnable) {
- LGLW(_lglw);
-
- if(NULL != _lglw)
- {
- lglw->mouse.touch.b_enable = _bEnable;
- lglw->mouse.touch.b_update_queued = LGLW_TRUE;
- }
- }
-
-
- // ---------------------------------------------------------------------------- lglw_touchinput_get
- lglw_bool_t lglw_touchinput_get(lglw_t _lglw) {
- lglw_bool_t r = LGLW_FALSE;
- LGLW(_lglw);
-
- if(NULL != _lglw)
- {
- r = lglw->mouse.touch.b_enable;
- }
-
- return r;
- }
-
-
- // ---------------------------------------------------------------------------- lglw_clipboard_text_set
- void lglw_clipboard_text_set(lglw_t _lglw, const uint32_t _numChars, const char *_text) {
- LGLW(_lglw);
- (void)_numChars;
-
- if(NULL != _text)
- {
- if(NULL != _lglw)
- {
- if(NULL != lglw->win.hwnd)
- {
- uint32_t numChars = (0u == _numChars) ? ((uint32_t)strlen(_text)+1u) : _numChars;
-
- if(numChars > 0u)
- {
- if(OpenClipboard(lglw->win.hwnd))
- {
- HANDLE hGlobal = GlobalAlloc(GMEM_MOVEABLE, numChars * sizeof(TCHAR));
-
- if(NULL != hGlobal)
- {
- char *d = (char*) GlobalLock(hGlobal);
-
- if(NULL != d)
- {
- uint32_t i;
- for(i = 0u; i < numChars; i++)
- {
- d[i] = _text[i];
- }
- d[numChars - 1] = 0;
-
- Dprintf("xxx lglw_clipboard_text_set: SetClipboardData(\"%s\")\n", d);
-
- GlobalUnlock(hGlobal);
- (void)SetClipboardData(CF_TEXT, hGlobal);
- }
- }
-
- CloseClipboard();
- }
- }
- }
- }
- }
- }
-
-
- // ---------------------------------------------------------------------------- lglw_clipboard_text_get
- void lglw_clipboard_text_get(lglw_t _lglw, uint32_t _maxChars, uint32_t *_retNumChars, char *_retText) {
- LGLW(_lglw);
-
- if(NULL != _retNumChars)
- *_retNumChars = 0u;
-
- if(NULL != _retText)
- *_retText = 0;
-
- if(_maxChars > 0u)
- {
- if(NULL != _lglw)
- {
- if(NULL != lglw->win.hwnd)
- {
- if(OpenClipboard(lglw->win.hwnd))
- {
- const char *data = (const char*)GetClipboardData(CF_TEXT);
-
- Dprintf("xxx lglw_clipboard_text_get: data=\"%s\"\n", data);
-
- if(NULL != data)
- {
- uint32_t i = 0u;
-
- for(; i < _maxChars; i++)
- {
- _retText[i] = data[i];
-
- if(0 == _retText[i])
- break;
- }
- _retText[_maxChars - 1u] = 0; // ASCIIZ
-
- if(NULL != _retNumChars)
- {
- *_retNumChars = i;
- }
- }
-
- CloseClipboard();
- }
- }
- }
- }
- }
-
-
- // ---------------------------------------------------------------------------- loc_LowLevelKeyboardProc
- static LRESULT CALLBACK loc_LowLevelKeyboardProc(int nCode, WPARAM wParam, LPARAM lParam) {
-
- Dprintf("xxx lglw:loc_LowLevelKeyboardProc: wParam=0x%08x lParam=0x%08x\n", (uint32_t)wParam, (uint32_t)lParam);
-
- if(HC_ACTION == nCode)
- {
- if(NULL != khook_lglw)
- {
- KBDLLHOOKSTRUCT *hs = (KBDLLHOOKSTRUCT*)lParam;
- lglw_bool_t bHandled = LGLW_FALSE;
-
- switch(wParam)
- {
- default:
- break;
-
- case WM_KEYDOWN:
- // Dprintf("xxx lglw:loc_LowLevelKeyboardProc: WM_KEYDOWN vkCode=0x%08x scanCode=0x%08x bExt=%d\n", hs->vkCode, hs->scanCode, (0 != (hs->flags & LLKHF_EXTENDED)));
-
- switch(hs->vkCode)
- {
- default:
- if(0u == (hs->flags & LLKHF_EXTENDED))
- {
- uint16_t ucBuf[16];
- BYTE keyState[256];
-
- // Dprintf("xxx lglw call toUnicode\n");
- GetKeyboardState(keyState);
-
- // don't let e.g. ctrl-<a..z> produce symbol chars
- keyState[VK_CONTROL] = 0;
- keyState[VK_LCONTROL] = 0;
- keyState[VK_RCONTROL] = 0;
-
- if(ToUnicode(hs->vkCode, hs->scanCode, keyState, (LPWSTR)ucBuf, 8/*cchBuff*/, 0/*wFlags*/) >= 1)
- {
- // Dprintf("xxx lglw toUnicode: ucBuf[0]=0x%04x\n", ucBuf[0]);
- bHandled = loc_handle_key(khook_lglw, LGLW_TRUE/*bPressed*/, ucBuf[0]);
- }
- else
- {
- Dprintf("xxx lglw:loc_LowLevelKeyboardProc<down>: ToUnicode failed\n");
- }
- }
- else
- {
- bHandled = loc_handle_key(khook_lglw, LGLW_TRUE/*bPressed*/, hs->vkCode | LGLW_VKEY_EXT);
- bHandled = LGLW_FALSE;
- }
- break;
-
- case VK_F1:
- case VK_F2:
- case VK_F3:
- case VK_F4:
- case VK_F5:
- case VK_F6:
- case VK_F7:
- case VK_F8:
- case VK_F9:
- case VK_F10:
- case VK_F11:
- case VK_F12:
- case VK_BACK:
- case VK_TAB:
- case VK_RETURN:
- case VK_ESCAPE:
- bHandled = loc_handle_key(khook_lglw, LGLW_TRUE/*bPressed*/, hs->vkCode | LGLW_VKEY_EXT);
- break;
-
- case VK_LSHIFT:
- // Dprintf("xxx lglw:loc_LowLevelKeyboardProc<down>: VK_LSHIFT\n");
- khook_lglw->keyboard.kmod_state |= LGLW_KMOD_LSHIFT;
- bHandled = loc_handle_key(khook_lglw, LGLW_TRUE/*bPressed*/, LGLW_VKEY_LSHIFT);
- bHandled = LGLW_FALSE;
- break;
-
- case VK_RSHIFT:
- // Dprintf("xxx lglw:loc_LowLevelKeyboardProc<down>: VK_RSHIFT\n");
- khook_lglw->keyboard.kmod_state |= LGLW_KMOD_RSHIFT;
- bHandled = loc_handle_key(khook_lglw, LGLW_TRUE/*bPressed*/, LGLW_VKEY_RSHIFT);
- bHandled = LGLW_FALSE;
- break;
-
- case VK_LCONTROL:
- // Dprintf("xxx lglw:loc_LowLevelKeyboardProc<down>: VK_LCONTROL\n");
- khook_lglw->keyboard.kmod_state |= LGLW_KMOD_LCTRL;
- bHandled = loc_handle_key(khook_lglw, LGLW_TRUE/*bPressed*/, LGLW_VKEY_LCTRL);
- bHandled = LGLW_FALSE;
- break;
-
- case VK_RCONTROL:
- // Dprintf("xxx lglw:loc_LowLevelKeyboardProc<down>: VK_RCONTROL\n");
- khook_lglw->keyboard.kmod_state |= LGLW_KMOD_RCTRL;
- bHandled = loc_handle_key(khook_lglw, LGLW_TRUE/*bPressed*/, LGLW_VKEY_RCTRL);
- bHandled = LGLW_FALSE;
- break;
-
- //case VK_RWIN:
- //case VK_F1:
- case VK_LMENU: // alt
- // not received
- // Dprintf("xxx lglw:loc_LowLevelKeyboardProc<down>: VK_LMENU\n");
- break;
-
- case VK_RMENU:
- // Dprintf("xxx lglw:loc_LowLevelKeyboardProc<down>: VK_RMENU\n");
- break;
- }
-
- break;
-
- case WM_KEYUP:
- // Dprintf("xxx lglw:loc_LowLevelKeyboardProc: WM_KEYUP vkCode=0x%08x scanCode=0x%08x bExt=%d\n", hs->vkCode, hs->scanCode, (0 != (hs->flags & LLKHF_EXTENDED)));
- switch(hs->vkCode)
- {
- default:
- if(0u == (hs->flags & LLKHF_EXTENDED))
- {
- uint16_t ucBuf[16];
- BYTE keyState[256];
-
- GetKeyboardState(keyState);
-
- if(ToUnicode(hs->vkCode, hs->scanCode, keyState, (LPWSTR)ucBuf, 8/*cchBuff*/, 0/*wFlags*/) >= 1)
- {
- bHandled = loc_handle_key(khook_lglw, LGLW_FALSE/*bPressed*/, ucBuf[0]);
- }
- }
- else
- {
- bHandled = loc_handle_key(khook_lglw, LGLW_FALSE/*bPressed*/, hs->vkCode | LGLW_VKEY_EXT);
- bHandled = LGLW_FALSE;
- }
- break;
-
- case VK_F1:
- case VK_F2:
- case VK_F3:
- case VK_F4:
- case VK_F5:
- case VK_F6:
- case VK_F7:
- case VK_F8:
- case VK_F9:
- case VK_F10:
- case VK_F11:
- case VK_F12:
- case VK_BACK:
- case VK_TAB:
- case VK_RETURN:
- case VK_ESCAPE:
- bHandled = loc_handle_key(khook_lglw, LGLW_FALSE/*bPressed*/, hs->vkCode | LGLW_VKEY_EXT);
- break;
-
- case VK_LSHIFT:
- khook_lglw->keyboard.kmod_state &= ~LGLW_KMOD_LSHIFT;
- bHandled = loc_handle_key(khook_lglw, LGLW_FALSE/*bPressed*/, LGLW_VKEY_LSHIFT);
- bHandled = LGLW_FALSE;
- break;
-
- case VK_RSHIFT:
- khook_lglw->keyboard.kmod_state &= ~LGLW_KMOD_RSHIFT;
- bHandled = loc_handle_key(khook_lglw, LGLW_FALSE/*bPressed*/, LGLW_VKEY_RSHIFT);
- bHandled = LGLW_FALSE;
- break;
-
- case VK_LCONTROL:
- khook_lglw->keyboard.kmod_state &= ~LGLW_KMOD_LCTRL;
- bHandled = loc_handle_key(khook_lglw, LGLW_FALSE/*bPressed*/, LGLW_VKEY_LCTRL);
- bHandled = LGLW_FALSE;
- break;
-
- case VK_RCONTROL:
- khook_lglw->keyboard.kmod_state &= ~LGLW_KMOD_RCTRL;
- bHandled = loc_handle_key(khook_lglw, LGLW_FALSE/*bPressed*/, LGLW_VKEY_RCTRL);
- bHandled = LGLW_FALSE;
- break;
- }
- break;
- }
-
- if(!bHandled)
- {
- Dprintf("xxx lglw: CallNextHookEx<kbd>\n");
- return CallNextHookEx(NULL, nCode, wParam, lParam);
- }
- else
- {
- Dprintf("xxx lglw: CallNextHookEx<kbd>: was handled\n");
- return 1;
- }
-
- } // if khook_lglw
- }
- return CallNextHookEx(NULL, nCode, wParam, lParam);
- }
-
-
- // ---------------------------------------------------------------------------- loc_WndProc
- LRESULT CALLBACK loc_WndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam) {
-
- LGLW(GetWindowLongPtr(hWnd, GWLP_USERDATA));
-
- if(NULL != lglw)
- {
- // Dprintf("xxx lglw:loc_WndProc: message=%08x wParam=0x%08x lParam=0x%08x\n", message, (uint32_t)wParam, (uint32_t)lParam);
-
- switch(message)
- {
- case WM_CREATE:
- Dprintf("xxx lglw: WM_CREATE\n");
- break;
-
- case WM_DESTROY:
- Dprintf("xxx lglw: WM_DESTROY\n");
- break;
-
- case WM_ACTIVATEAPP: // 0x1c wParam=0/1
- Dprintf("xxx lglw: WM_ACTIVATEAPP wParam=%d\n", (int32_t)wParam);
- break;
-
- case WM_MOUSELEAVE:
- // (note) only received after TrackMouseEvent() has been called
- Dprintf("xxx lglw: WM_MOUSELEAVE\n");
- loc_handle_mouseleave(lglw);
- break;
-
- case WM_MOUSEMOVE:
- // Dprintf("xxx lglw: WM_MOUSEMOVE:\n");
- if(lglw->mouse.touch.b_update_queued)
- {
- loc_touchinput_update(lglw);
- lglw->mouse.touch.b_update_queued = LGLW_FALSE;
- }
- if(!lglw->mouse.touch.b_enable)
- {
- int32_t x = GET_X_LPARAM(lParam); // lo
- int32_t y = GET_Y_LPARAM(lParam); // hi
-
- if(LGLW_MOUSE_GRAB_WARP == lglw->mouse.grab.mode)
- {
- lglw->mouse.grab.b_queue_warp = LGLW_TRUE;
-
- lglw->mouse.p.x += (x - lglw->mouse.grab.last_p.x);
- lglw->mouse.p.y += (y - lglw->mouse.grab.last_p.y);
-
- lglw->mouse.grab.last_p.x = x;
- lglw->mouse.grab.last_p.y = y;
- }
- else
- {
- lglw->mouse.p.x = x;
- lglw->mouse.p.y = y;
- }
-
- if(0u == (lglw->focus.state & LGLW_FOCUS_MOUSE))
- {
- loc_handle_mouseenter(lglw);
- }
-
- loc_handle_mousemotion(lglw);
- }
- break;
-
- case WM_LBUTTONDOWN:
- if(!lglw->mouse.touch.b_enable)
- {
- // Dprintf("xxx lglw: WM_LBUTTONDOWN\n");
- loc_handle_mousebutton(lglw, LGLW_TRUE/*bPressed*/, LGLW_MOUSE_LBUTTON);
- }
- break;
-
- case WM_LBUTTONUP:
- if(!lglw->mouse.touch.b_enable)
- {
- // Dprintf("xxx lglw: WM_LBUTTONUP\n");
- loc_handle_mousebutton(lglw, LGLW_FALSE/*bPressed*/, LGLW_MOUSE_LBUTTON);
- }
- break;
-
- case WM_RBUTTONDOWN:
- if(!lglw->mouse.touch.b_enable)
- {
- // Dprintf("xxx lglw: WM_RBUTTONDOWN\n");
- loc_handle_mousebutton(lglw, LGLW_TRUE/*bPressed*/, LGLW_MOUSE_RBUTTON);
- }
- break;
-
- case WM_RBUTTONUP:
- if(!lglw->mouse.touch.b_enable)
- {
- // Dprintf("xxx lglw: WM_RBUTTONUP\n");
- loc_handle_mousebutton(lglw, LGLW_FALSE/*bPressed*/, LGLW_MOUSE_RBUTTON);
- }
- break;
-
- case WM_MBUTTONDOWN:
- if(!lglw->mouse.touch.b_enable)
- {
- // Dprintf("xxx lglw: WM_MBUTTONDOWN\n");
- loc_handle_mousebutton(lglw, LGLW_TRUE/*bPressed*/, LGLW_MOUSE_MBUTTON);
- }
- break;
-
- case WM_MBUTTONUP:
- if(!lglw->mouse.touch.b_enable)
- {
- // Dprintf("xxx lglw: WM_MBUTTONUP\n");
- loc_handle_mousebutton(lglw, LGLW_FALSE/*bPressed*/, LGLW_MOUSE_MBUTTON);
- }
- break;
-
- case WM_MOUSEWHEEL:
- // Dprintf("xxx lglw: WM_MOUSEWHEEL\n");
- {
- uint32_t bt = (((int16_t)(((uint32_t)wParam)>>16)) > 0) ? LGLW_MOUSE_WHEELUP : LGLW_MOUSE_WHEELDOWN;
- loc_handle_mousebutton(lglw, LGLW_TRUE/*bPressed*/, bt);
- loc_handle_mousebutton(lglw, LGLW_FALSE/*bPressed*/, bt);
- }
- break;
-
- case WM_CAPTURECHANGED:
- // e.g. after alt-tab
- Dprintf("xxx lglw: WM_CAPTURECHANGED\n");
- lglw->mouse.grab.mode = LGLW_MOUSE_GRAB_NONE;
- break;
-
- #if 1
- // (note) VST windows usually don't receive key/char messages (they are consumed by the DAW)
-
- case WM_GETDLGCODE:
- // never received
- Dprintf("xxx lglw: WM_GETDLGCODE\n");
- return DLGC_WANTALLKEYS;
-
- case WM_KEYDOWN:
- // never received
- Dprintf("xxx lglw: WM_KEYDOWN nVirtKey=0x%08x lKeydata=0x%08x\n", (uint32_t)wParam, (uint32_t)lParam);
- break;
-
- case WM_KEYUP:
- // never received
- Dprintf("xxx lglw: WM_KEYUP nVirtKey=0x%08x lKeydata=0x%08x\n", (uint32_t)wParam, (uint32_t)lParam);
- break;
-
- case WM_CHAR:
- // never received
- Dprintf("xxx lglw: WM_CHAR charCode=0x%08x bPressed=%u\n", (uint32_t)wParam, (((uint32_t)lParam)>>31)&1u);
- break;
- #endif
-
- case WM_PAINT:
- // https://docs.microsoft.com/en-us/windows/desktop/api/Winuser/nf-winuser-redrawwindow
- // Dprintf("xxx lglw: WM_PAINT\n");
- loc_handle_queued_mouse_warp(lglw);
- if(NULL != lglw->redraw.cbk)
- {
- lglw->redraw.cbk(lglw);
- }
- break;
-
- case WM_TIMER:
- // Dprintf("xxx lglw: WM_TIMER cbk=%p\n", lglw->timer.cbk);
- if(lglw->mouse.touch.b_enable)
- {
- // Dprintf("xxx lglw: WM_TIMER hold_state=%u\n", lglw->mouse.touch.syn_rmb_hold_state);
- if(LGLW_MOUSE_TOUCH_RMB_STATE_IDLE != lglw->mouse.touch.syn_rmb_hold_state)
- {
- uint32_t ms = (uint32_t)GetTickCount();
- uint32_t delta;
- if(ms < lglw->mouse.touch.hold_start_ms)
- {
- // Overflow after 49.7 days
- delta = ~lglw->mouse.touch.hold_start_ms + 1u + ms;
- }
- else
- {
- delta = (ms - lglw->mouse.touch.hold_start_ms);
- }
-
- if(LGLW_MOUSE_TOUCH_RMB_STATE_LMB == lglw->mouse.touch.syn_rmb_hold_state)
- {
- if(delta >= LGLW_MOUSE_TOUCH_LMB_TIMEOUT)
- {
- loc_handle_mousebutton(lglw, LGLW_FALSE/*bPressed*/, LGLW_MOUSE_LBUTTON);
- lglw->mouse.touch.syn_rmb_hold_state = LGLW_MOUSE_TOUCH_RMB_STATE_WAIT;
- }
- }
-
- if(LGLW_MOUSE_TOUCH_RMB_STATE_WAIT == lglw->mouse.touch.syn_rmb_hold_state)
- {
- if(delta >= LGLW_MOUSE_TOUCH_RMB_TIMEOUT)
- {
- lglw->mouse.touch.syn_rmb_hold_state = LGLW_MOUSE_TOUCH_RMB_STATE_RMB;
- loc_handle_mousebutton(lglw, LGLW_TRUE/*bPressed*/, LGLW_MOUSE_RBUTTON);
- }
- }
- }
- }
- else
- {
- loc_handle_queued_mouse_warp(lglw);
- }
-
- if(NULL != lglw->timer.cbk)
- {
- lglw->timer.cbk(lglw);
- }
- break;
-
- case WM_DROPFILES:
- if(NULL != lglw->dropfiles.cbk)
- {
- HDROP hDrop = (HDROP) wParam;
- POINT p;
-
- if(TRUE == DragQueryPoint(hDrop, &p))
- {
- Dprintf("xxx lglw: WM_DROPFILES: p=(%d; %d)\n", p.x, p.y);
-
- UINT numFiles = DragQueryFile(hDrop, 0xffffffff, NULL, 0);
- Dprintf("xxx lglw: WM_DROPFILES: dropped %u file(s)\n", numFiles);
-
- if(numFiles > 0u)
- {
- UINT i;
- char **pathnames = malloc((sizeof(char*) * numFiles) + (MAX_PATH * sizeof(char) * numFiles));
- char *pathname = (char*)&pathnames[numFiles];
-
- for(i=0; i<numFiles; i++)
- {
- DragQueryFile(hDrop, i, pathname, MAX_PATH);
- Dprintf("xxx lglw: WM_DROPFILES: file[%d] = \"%s\"\n", i, pathname);
- pathnames[i] = pathname;
- pathname += MAX_PATH;
- }
-
- lglw->dropfiles.cbk(lglw, p.x, p.y, numFiles, pathnames);
-
- free(pathnames);
- }
- }
- }
- break;
-
- #ifdef BUILD_64
- // Touch messages:
- #ifndef GET_X_LPARAM
- #define GET_X_LPARAM(lp) ((int)(short)LOWORD(lp))
- #define GET_Y_LPARAM(lp) ((int)(short)HIWORD(lp))
- #define GET_POINTERID_WPARAM(wParam) (LOWORD(wParam))
- #define IS_POINTER_FLAG_SET_WPARAM(wParam, flag) (((DWORD)HIWORD(wParam) & (flag)) == (flag))
- #define IS_POINTER_NEW_WPARAM(wParam) IS_POINTER_FLAG_SET_WPARAM(wParam, POINTER_MESSAGE_FLAG_NEW)
- #define IS_POINTER_INRANGE_WPARAM(wParam) IS_POINTER_FLAG_SET_WPARAM(wParam, POINTER_MESSAGE_FLAG_INRANGE)
- #define IS_POINTER_INCONTACT_WPARAM(wParam) IS_POINTER_FLAG_SET_WPARAM(wParam, POINTER_MESSAGE_FLAG_INCONTACT)
- #define IS_POINTER_FIRSTBUTTON_WPARAM(wParam) IS_POINTER_FLAG_SET_WPARAM(wParam, POINTER_MESSAGE_FLAG_FIRSTBUTTON)
- #define IS_POINTER_SECONDBUTTON_WPARAM(wParam) IS_POINTER_FLAG_SET_WPARAM(wParam, POINTER_MESSAGE_FLAG_SECONDBUTTON)
- #define IS_POINTER_THIRDBUTTON_WPARAM(wParam) IS_POINTER_FLAG_SET_WPARAM(wParam, POINTER_MESSAGE_FLAG_THIRDBUTTON)
- #define IS_POINTER_FOURTHBUTTON_WPARAM(wParam) IS_POINTER_FLAG_SET_WPARAM(wParam, POINTER_MESSAGE_FLAG_FOURTHBUTTON)
- #define IS_POINTER_FIFTHBUTTON_WPARAM(wParam) IS_POINTER_FLAG_SET_WPARAM(wParam, POINTER_MESSAGE_FLAG_FIFTHBUTTON)
- #define IS_POINTER_PRIMARY_WPARAM(wParam) IS_POINTER_FLAG_SET_WPARAM(wParam, POINTER_MESSAGE_FLAG_PRIMARY)
- #define HAS_POINTER_CONFIDENCE_WPARAM(wParam) IS_POINTER_FLAG_SET_WPARAM(wParam, POINTER_MESSAGE_FLAG_CONFIDENCE)
- #define IS_POINTER_CANCELED_WPARAM(wParam) IS_POINTER_FLAG_SET_WPARAM(wParam, POINTER_MESSAGE_FLAG_CANCELED)
- #endif
- case WM_POINTERDOWN:
- {
- POINT p;
-
- lglw->mouse.touch.b_enable = LGLW_TRUE;
- lglw->mouse.touch.b_update_queued = LGLW_FALSE;
-
- if(0u == (lglw->focus.state & LGLW_FOCUS_MOUSE))
- {
- loc_handle_mouseenter(lglw);
- }
-
- // Dprintf("xxx lglw: WM_POINTERDOWN\n");
-
- // Dprintf("xxx lglw: WM_POINTERDOWN: IS_POINTER_FIRSTBUTTON_WPARAM(wParam)=%d\n", IS_POINTER_FIRSTBUTTON_WPARAM(wParam));
- // Dprintf("xxx lglw: WM_POINTERDOWN: IS_POINTER_SECONDBUTTON_WPARAM(wParam)=%d\n", IS_POINTER_SECONDBUTTON_WPARAM(wParam));
- // Dprintf("xxx lglw: WM_POINTERDOWN: IS_POINTER_THIRDBUTTON_WPARAM(wParam)=%d\n", IS_POINTER_THIRDBUTTON_WPARAM(wParam));
-
- p.x = GET_X_LPARAM(lParam);
- p.y = GET_Y_LPARAM(lParam);
- (void)ScreenToClient(lglw->win.hwnd, &p);
- lglw->mouse.p.x = p.x;
- lglw->mouse.p.y = p.y;
- loc_handle_mousemotion(lglw);
-
- if(IS_POINTER_FIRSTBUTTON_WPARAM(wParam))
- {
- lglw->mouse.touch.b_syn_rmb = (0u != (lglw->keyboard.kmod_state & LGLW_KMOD_CTRL));
- loc_handle_mousebutton(lglw, LGLW_TRUE/*bPressed*/,
- lglw->mouse.touch.b_syn_rmb ? LGLW_MOUSE_RBUTTON : LGLW_MOUSE_LBUTTON
- );
- if(lglw->mouse.touch.b_syn_rmb)
- {
- lglw->mouse.touch.syn_rmb_hold_state = LGLW_MOUSE_TOUCH_RMB_STATE_IDLE;
- }
- else
- {
- lglw->mouse.touch.syn_rmb_hold_state = LGLW_MOUSE_TOUCH_RMB_STATE_LMB;
- lglw->mouse.touch.hold_start_ms = GetTickCount();
- lglw->mouse.touch.hold_start_p = lglw->mouse.p;
- }
- }
- else if(IS_POINTER_SECONDBUTTON_WPARAM(wParam))
- {
- loc_handle_mousebutton(lglw, LGLW_TRUE/*bPressed*/, LGLW_MOUSE_RBUTTON);
- }
- else if(IS_POINTER_THIRDBUTTON_WPARAM(wParam))
- {
- loc_handle_mousebutton(lglw, LGLW_TRUE/*bPressed*/, LGLW_MOUSE_MBUTTON);
- }
- }
- break;
-
- case WM_POINTERUP:
- {
- POINT p;
-
- // Dprintf("xxx lglw: WM_POINTERUP\n");
-
- lglw->mouse.touch.b_enable = LGLW_TRUE;
- lglw->mouse.touch.b_update_queued = LGLW_FALSE;
-
- p.x = GET_X_LPARAM(lParam);
- p.y = GET_Y_LPARAM(lParam);
- (void)ScreenToClient(lglw->win.hwnd, &p);
- lglw->mouse.p.x = p.x;
- lglw->mouse.p.y = p.y;
- loc_handle_mousemotion(lglw);
-
- // Dprintf("xxx lglw: WM_POINTERUP: IS_POINTER_FIRSTBUTTON_WPARAM(wParam)=%d\n", IS_POINTER_FIRSTBUTTON_WPARAM(wParam));
- // Dprintf("xxx lglw: WM_POINTERUP: IS_POINTER_SECONDBUTTON_WPARAM(wParam)=%d\n", IS_POINTER_SECONDBUTTON_WPARAM(wParam));
- // Dprintf("xxx lglw: WM_POINTERUP: IS_POINTER_THIRDBUTTON_WPARAM(wParam)=%d\n", IS_POINTER_THIRDBUTTON_WPARAM(wParam));
-
- if( (0u != (lglw->mouse.button_state & LGLW_MOUSE_LBUTTON)) && !IS_POINTER_FIRSTBUTTON_WPARAM(wParam) )
- {
- if(LGLW_MOUSE_TOUCH_RMB_STATE_RMB == lglw->mouse.touch.syn_rmb_hold_state)
- {
- loc_handle_mousebutton(lglw, LGLW_FALSE/*bPressed*/, LGLW_MOUSE_RBUTTON);
- }
- else
- {
- loc_handle_mousebutton(lglw, LGLW_FALSE/*bPressed*/,
- lglw->mouse.touch.b_syn_rmb ? LGLW_MOUSE_RBUTTON : LGLW_MOUSE_LBUTTON
- );
- }
- lglw->mouse.touch.b_syn_rmb = LGLW_FALSE;
- lglw->mouse.touch.syn_rmb_hold_state = LGLW_MOUSE_TOUCH_RMB_STATE_IDLE;
- }
- else if( (0u != (lglw->mouse.button_state & LGLW_MOUSE_RBUTTON)) && !IS_POINTER_SECONDBUTTON_WPARAM(wParam))
- {
- loc_handle_mousebutton(lglw, LGLW_FALSE/*bPressed*/, LGLW_MOUSE_RBUTTON);
- }
- else if( (0u != (lglw->mouse.button_state & LGLW_MOUSE_MBUTTON)) && !IS_POINTER_THIRDBUTTON_WPARAM(wParam))
- {
- loc_handle_mousebutton(lglw, LGLW_FALSE/*bPressed*/, LGLW_MOUSE_MBUTTON);
- }
- }
- break;
-
- case WM_POINTERENTER:
- {
- POINT p;
-
- Dprintf("xxx lglw: WM_POINTERENTER\n");
-
- lglw->mouse.touch.b_enable = LGLW_TRUE;
- lglw->mouse.touch.b_update_queued = LGLW_FALSE;
-
- if(0u == (lglw->focus.state & LGLW_FOCUS_MOUSE))
- {
- loc_handle_mouseenter(lglw);
- }
-
- p.x = GET_X_LPARAM(lParam);
- p.y = GET_Y_LPARAM(lParam);
- (void)ScreenToClient(lglw->win.hwnd, &p);
- lglw->mouse.p.x = p.x;
- lglw->mouse.p.y = p.y;
- loc_handle_mousemotion(lglw);
- loc_handle_mouseenter(lglw);
- }
- break;
-
- case WM_POINTERLEAVE:
- {
- POINT p;
-
- Dprintf("xxx lglw: WM_POINTERLEAVE\n");
-
- lglw->mouse.touch.b_enable = LGLW_TRUE;
- lglw->mouse.touch.b_update_queued = LGLW_FALSE;
-
- p.x = GET_X_LPARAM(lParam);
- p.y = GET_Y_LPARAM(lParam);
- (void)ScreenToClient(lglw->win.hwnd, &p);
- loc_handle_mouseleave(lglw);
- }
- break;
-
- case WM_POINTERUPDATE:
- {
- POINT p;
-
- lglw->mouse.touch.b_enable = LGLW_TRUE;
- lglw->mouse.touch.b_update_queued = LGLW_FALSE;
-
- p.x = GET_X_LPARAM(lParam);
- p.y = GET_Y_LPARAM(lParam);
- (void)ScreenToClient(lglw->win.hwnd, &p);
- lglw->mouse.p.x = p.x;
- lglw->mouse.p.y = p.y;
- loc_handle_mousemotion(lglw);
-
- if(LGLW_MOUSE_TOUCH_RMB_STATE_LMB == lglw->mouse.touch.syn_rmb_hold_state)
- {
- if( (sABS(p.x - lglw->mouse.touch.hold_start_p.x) >= LGLW_MOUSE_TOUCH_RMB_MOVE_THRESHOLD) ||
- (sABS(p.y - lglw->mouse.touch.hold_start_p.y) >= LGLW_MOUSE_TOUCH_RMB_MOVE_THRESHOLD)
- )
- {
- lglw->mouse.touch.syn_rmb_hold_state = LGLW_MOUSE_TOUCH_RMB_STATE_IDLE;
- }
- }
-
- }
- break;
- #endif // BUILD_64
-
- } // switch message
- } // if lglw
-
- return DefWindowProc(hWnd, message, wParam, lParam);
- }
|