core, add mouse support for Pico
authorkub <derkub@gmail.com>
Tue, 25 Feb 2025 23:29:20 +0000 (00:29 +0100)
committerkub <derkub@gmail.com>
Wed, 26 Feb 2025 23:21:07 +0000 (00:21 +0100)
platform/common/emu.c
platform/common/emu.h
platform/common/plat_sdl.c
platform/linux/emu.c

index 86b5429..618271e 100644 (file)
@@ -66,6 +66,7 @@ struct vkbd *vkbd;
 static int pico_page;\r
 static int pico_w, pico_h;\r
 static u16 *pico_overlay;\r
+static int pico_pad;\r
 \r
 static short __attribute__((aligned(4))) sndBuffer[2*54000/50];\r
 \r
@@ -1148,6 +1149,9 @@ void emu_pico_overlay(u16 *pd, int w, int h, int pitch)
 \r
 void run_events_pico(unsigned int events)\r
 {\r
+       // treat pad ports equal to support pad in one and mouse in the other\r
+       PicoIn.pad[0] |= PicoIn.pad[1];\r
+\r
        if (events & PEV_PICO_PPREV) {\r
                PicoPicohw.page--;\r
                if (PicoPicohw.page < 0)\r
@@ -1200,12 +1204,23 @@ void run_events_pico(unsigned int events)
        if (pico_inp_mode == 0)\r
                return;\r
 \r
-       /* handle other input modes */\r
-       if (PicoIn.pad[0] & 1) pico_pen_y--;\r
-       if (PicoIn.pad[0] & 2) pico_pen_y++;\r
-       if (PicoIn.pad[0] & 4) pico_pen_x--;\r
-       if (PicoIn.pad[0] & 8) pico_pen_x++;\r
-       PicoIn.pad[0] &= ~0x0f; // release UDLR\r
+       /* handle other input modes using the pen */\r
+       if (currentConfig.input_dev0 == PICO_INPUT_MOUSE ||\r
+           currentConfig.input_dev1 == PICO_INPUT_MOUSE) {\r
+               pico_pen_x = PicoIn.mouse[0];\r
+               pico_pen_y = PicoIn.mouse[1];\r
+               if ((pico_pad ^ PicoIn.pad[0]) & (1<<GBTN_START) & ~pico_pad) {\r
+                       PicoPicohw.pen_pos[0] ^= 0x8000;\r
+                       PicoPicohw.pen_pos[1] ^= 0x8000;\r
+               }\r
+               pico_pad = PicoIn.pad[0];\r
+       } else {\r
+               if (PicoIn.pad[0] & 1) pico_pen_y--;\r
+               if (PicoIn.pad[0] & 2) pico_pen_y++;\r
+               if (PicoIn.pad[0] & 4) pico_pen_x--;\r
+               if (PicoIn.pad[0] & 8) pico_pen_x++;\r
+               PicoIn.pad[0] &= ~0x0f; // release UDLR\r
+       }\r
 \r
        /* cursor position, cursor drawing must not cross screen borders */\r
        if (pico_pen_y < PICO_PEN_ADJUST_Y)\r
@@ -1333,7 +1348,7 @@ void emu_update_input(void)
        int actions[IN_BINDTYPE_COUNT] = { 0, };\r
        int actions_kbd[IN_BIND_LAST] = { 0, };\r
        int pl_actions[4];\r
-       int count_kbd = 0;\r
+       int count_kbd = 0, buttons = 0;\r
        int events, i = 0;\r
 \r
        in_update(actions);\r
@@ -1346,21 +1361,23 @@ void emu_update_input(void)
        events = actions[IN_BINDTYPE_EMU] & PEV_MASK;\r
 \r
        // update mouse coordinates if there is an emulated mouse\r
-       in_update_analog(0, 0, &PicoIn.mouse[0]);\r
-       in_update_analog(0, 1, &PicoIn.mouse[1]);\r
-       // scale mouse coordinates according to window scale\r
-       PicoIn.mouse[0] = PicoIn.mouse[0] * g_screen_width  / (2*1024);\r
-       PicoIn.mouse[1] = PicoIn.mouse[1] * g_screen_height / (2*1024);\r
-\r
-       in_update_analog(0, -1, &i);\r
-       int buttons = 0;\r
-       if (i & 1) buttons |= 1<<GBTN_B;\r
-       if (i & 4) buttons |= 1<<GBTN_C;\r
-       if (i & 2) buttons |= 1<<GBTN_START;\r
-       if (currentConfig.input_dev0 == PICO_INPUT_MOUSE)\r
-               pl_actions[0] |= buttons;\r
-       if (currentConfig.input_dev1 == PICO_INPUT_MOUSE)\r
-               pl_actions[1] |= buttons;\r
+       if (currentConfig.input_dev0 == PICO_INPUT_MOUSE ||\r
+           currentConfig.input_dev1 == PICO_INPUT_MOUSE) {\r
+               in_update_analog(0, 0, &PicoIn.mouse[0]);\r
+               in_update_analog(0, 1, &PicoIn.mouse[1]);\r
+               // scale mouse coordinates from -1024..1024 to 0..screen_w/h\r
+               PicoIn.mouse[0] = (PicoIn.mouse[0]+1024) * g_screen_width /2048;\r
+               PicoIn.mouse[1] = (PicoIn.mouse[1]+1024) * g_screen_height/2048;\r
+\r
+               in_update_analog(0, -1, &i); // get mouse buttons, bit 2-0 = RML\r
+               if (i & 1) buttons |= 1<<GBTN_B;\r
+               if (i & 4) buttons |= 1<<GBTN_C;\r
+               if (i & 2) buttons |= 1<<GBTN_START;\r
+               if (currentConfig.input_dev0 == PICO_INPUT_MOUSE)\r
+                       pl_actions[0] |= buttons;\r
+               if (currentConfig.input_dev1 == PICO_INPUT_MOUSE)\r
+                       pl_actions[1] |= buttons;\r
+       }\r
 \r
        if (kbd_mode) {\r
                int mask = (PicoIn.AHW & PAHW_PICO ? 0xf : 0x0);\r
index 30ab20f..8c4f733 100644 (file)
@@ -193,6 +193,9 @@ void plat_early_init(void);
 void plat_init(void);
 void plat_finish(void);
 
+void plat_show_cursor(int on);
+void plat_grab_input(int on);
+
 /* used before things blocking for a while (these funcs redraw on return) */
 void plat_status_msg_busy_first(const char *msg);
 void plat_status_msg_busy_next(const char *msg);
index b3cd239..bdef051 100644 (file)
@@ -28,6 +28,8 @@ static struct area { int w, h; } area;
 
 static struct in_pdata in_sdl_platform_data;
 
+static int hide_cursor;
+
 static int sound_rates[] = { 8000, 11025, 16000, 22050, 32000, 44100, 53000, -1 };
 struct plat_target plat_target = { .sound_rates = sound_rates };
 
@@ -452,6 +454,16 @@ void plat_video_loop_prepare(void)
        plat_video_set_buffer(g_screen_ptr);
 }
 
+void plat_show_cursor(int on)
+{
+       SDL_ShowCursor(on && !hide_cursor);
+}
+
+void plat_grab_input(int on)
+{
+       SDL_WM_GrabInput(on ? SDL_GRAB_ON : SDL_GRAB_OFF);
+}
+
 static void plat_sdl_resize(int w, int h)
 {
        // take over new settings
@@ -506,7 +518,10 @@ void plat_init(void)
 #elif !defined(__MIYOO__) && !defined(__RETROFW__) && !defined(__DINGUX__)
        if (! plat_sdl_is_windowed())
 #endif
+       {
+               hide_cursor = 1;
                SDL_ShowCursor(0);
+       }
 
        plat_sdl_quit_cb = plat_sdl_quit;
        plat_sdl_resize_cb = plat_sdl_resize;
index f26689d..cb4ce0a 100644 (file)
@@ -486,6 +486,8 @@ void pemu_loop_prep(void)
 {\r
        apply_renderer();\r
        plat_video_clear_buffers();\r
+       plat_show_cursor(currentConfig.input_dev0 != PICO_INPUT_MOUSE &&\r
+                       currentConfig.input_dev1 != PICO_INPUT_MOUSE);\r
 }\r
 \r
 void pemu_loop_end(void)\r
@@ -497,6 +499,7 @@ void pemu_loop_end(void)
                free(ghost_buf);\r
                ghost_buf = NULL;\r
        }\r
+       plat_show_cursor(1);\r
 }\r
 \r
 void plat_wait_till_us(unsigned int us_to)\r