RetroArch/menu/menu_input.c

568 lines
16 KiB
C
Raw Normal View History

/* RetroArch - A frontend for libretro.
* Copyright (C) 2010-2014 - Hans-Kristian Arntzen
2016-01-10 04:06:50 +01:00
* Copyright (C) 2011-2016 - Daniel De Matteis
*
* RetroArch is free software: you can redistribute it and/or modify it under the terms
* of the GNU General Public License as published by the Free Software Found-
* ation, either version 3 of the License, or (at your option) any later version.
*
* RetroArch is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
* without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
* PURPOSE. See the GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License along with RetroArch.
* If not, see <http://www.gnu.org/licenses/>.
*/
2014-03-02 11:45:41 +01:00
#include <stdint.h>
#include <stddef.h>
#include <stdlib.h>
#include <string.h>
2016-09-08 05:43:10 +02:00
#ifdef HAVE_CONFIG_H
#include "../config.h"
#endif
2016-09-15 20:35:39 +02:00
#include "widgets/menu_entry.h"
#include "widgets/menu_input_dialog.h"
#include "widgets/menu_input_bind_dialog.h"
2015-12-06 17:55:27 +01:00
#include "menu_driver.h"
#include "menu_input.h"
2015-09-27 01:10:15 +02:00
#include "menu_animation.h"
2015-06-13 16:47:51 +02:00
#include "menu_display.h"
#include "menu_navigation.h"
2016-10-30 11:48:35 +01:00
#include "menu_event.h"
2015-06-13 19:12:10 +02:00
2016-09-01 18:01:41 +02:00
#include "../configuration.h"
#include "../performance_counters.h"
2015-06-13 22:57:55 +02:00
2015-11-01 22:57:13 +01:00
enum menu_mouse_action
{
2016-02-26 21:43:20 +01:00
MENU_MOUSE_ACTION_NONE = 0,
MENU_MOUSE_ACTION_BUTTON_L,
MENU_MOUSE_ACTION_BUTTON_L_TOGGLE,
MENU_MOUSE_ACTION_BUTTON_L_SET_NAVIGATION,
MENU_MOUSE_ACTION_BUTTON_R,
MENU_MOUSE_ACTION_WHEEL_UP,
MENU_MOUSE_ACTION_WHEEL_DOWN,
MENU_MOUSE_ACTION_HORIZ_WHEEL_UP,
MENU_MOUSE_ACTION_HORIZ_WHEEL_DOWN
2015-11-01 22:57:13 +01:00
};
2016-11-20 14:46:11 +01:00
static int mouse_old_x = 0;
static int mouse_old_y = 0;
static rarch_timer_t mouse_activity_timer = {0};
2016-09-16 16:39:30 +02:00
menu_input_t *menu_input_get_ptr(void)
{
static menu_input_t menu_input_state;
return &menu_input_state;
}
bool menu_input_mouse_check_vector_inside_hitbox(menu_input_ctx_hitbox_t *hitbox)
{
int16_t mouse_x = menu_input_mouse_state(MENU_MOUSE_X_AXIS);
int16_t mouse_y = menu_input_mouse_state(MENU_MOUSE_Y_AXIS);
bool inside_hitbox =
(mouse_x >= hitbox->x1)
&& (mouse_x <= hitbox->x2)
&& (mouse_y >= hitbox->y1)
&& (mouse_y <= hitbox->y2)
;
return inside_hitbox;
}
2015-09-26 02:52:05 +02:00
bool menu_input_ctl(enum menu_input_ctl_state state, void *data)
2015-09-24 19:47:41 +02:00
{
static bool pointer_dragging = false;
menu_input_t *menu_input = menu_input_get_ptr();
2015-09-24 19:47:41 +02:00
if (!menu_input)
return false;
switch (state)
{
2015-12-11 11:56:31 +01:00
case MENU_INPUT_CTL_DEINIT:
memset(menu_input, 0, sizeof(menu_input_t));
pointer_dragging = false;
2015-12-11 11:56:31 +01:00
break;
2015-09-26 02:52:05 +02:00
case MENU_INPUT_CTL_MOUSE_PTR:
2016-11-20 14:46:11 +01:00
menu_input->mouse.ptr = (*(unsigned*)data);
2016-02-11 01:57:52 +01:00
break;
2015-09-26 02:52:05 +02:00
case MENU_INPUT_CTL_POINTER_PTR:
2016-11-20 14:46:11 +01:00
menu_input->pointer.ptr = (*(unsigned*)data);
2016-02-11 01:57:52 +01:00
break;
2015-09-26 02:52:05 +02:00
case MENU_INPUT_CTL_POINTER_ACCEL_READ:
{
2015-09-24 23:33:41 +02:00
float *ptr = (float*)data;
*ptr = menu_input->pointer.accel;
}
2016-02-11 01:57:52 +01:00
break;
2015-09-26 02:52:05 +02:00
case MENU_INPUT_CTL_POINTER_ACCEL_WRITE:
2016-11-20 14:46:11 +01:00
menu_input->pointer.accel = (*(float*)data);
2016-02-11 01:57:52 +01:00
break;
case MENU_INPUT_CTL_IS_POINTER_DRAGGED:
return pointer_dragging;
case MENU_INPUT_CTL_SET_POINTER_DRAGGED:
pointer_dragging = true;
break;
case MENU_INPUT_CTL_UNSET_POINTER_DRAGGED:
pointer_dragging = false;
2016-02-11 01:57:52 +01:00
break;
default:
case MENU_INPUT_CTL_NONE:
break;
2015-09-24 19:47:41 +02:00
}
2016-02-11 01:57:52 +01:00
return true;
2015-09-24 19:47:41 +02:00
}
static int menu_input_mouse_post_iterate(uint64_t *input_mouse,
menu_file_list_cbs_t *cbs, unsigned action, bool *mouse_activity)
{
settings_t *settings = config_get_ptr();
static bool mouse_oldleft = false;
static bool mouse_oldright = false;
2016-02-26 21:02:29 +01:00
if (
!settings->menu.mouse.enable
2015-04-14 15:35:14 +01:00
#ifdef HAVE_OVERLAY
|| (settings->input.overlay_enable && input_overlay_is_alive(overlay_ptr))
2015-04-14 15:35:14 +01:00
#endif
2016-02-26 21:02:29 +01:00
)
{
2016-09-15 21:05:39 +02:00
/* HACK: Need to lie to avoid false hits if mouse is held
* when entering the RetroArch window. */
/* This happens if, for example, someone double clicks the
* window border to maximize it.
*
* The proper fix is, of course, triggering on WM_LBUTTONDOWN
* rather than this state change. */
2016-07-16 17:01:56 +02:00
mouse_oldleft = true;
mouse_oldright = true;
return 0;
}
if (menu_input_mouse_state(MENU_MOUSE_LEFT_BUTTON))
{
if (!mouse_oldleft)
{
2016-02-26 21:02:29 +01:00
size_t selection;
menu_input_t *menu_input = menu_input_get_ptr();
menu_navigation_ctl(MENU_NAVIGATION_CTL_GET_SELECTION, &selection);
2015-09-25 21:52:04 +02:00
2016-02-26 21:43:20 +01:00
BIT64_SET(*input_mouse, MENU_MOUSE_ACTION_BUTTON_L);
2015-05-11 12:03:02 +02:00
mouse_oldleft = true;
if ((menu_input->mouse.ptr == selection) && cbs && cbs->action_select)
2015-05-11 12:03:02 +02:00
{
2016-02-26 21:43:20 +01:00
BIT64_SET(*input_mouse, MENU_MOUSE_ACTION_BUTTON_L_TOGGLE);
2015-05-11 12:03:02 +02:00
}
2015-10-17 19:21:18 +02:00
else if (menu_input->mouse.ptr <= (menu_entries_get_size() - 1))
2015-05-11 22:36:12 +02:00
{
2016-02-26 21:43:20 +01:00
BIT64_SET(*input_mouse, MENU_MOUSE_ACTION_BUTTON_L_SET_NAVIGATION);
2015-05-11 22:36:12 +02:00
}
*mouse_activity = true;
}
}
else
mouse_oldleft = false;
if (menu_input_mouse_state(MENU_MOUSE_RIGHT_BUTTON))
{
if (!mouse_oldright)
{
mouse_oldright = true;
2016-02-26 21:43:20 +01:00
BIT64_SET(*input_mouse, MENU_MOUSE_ACTION_BUTTON_R);
*mouse_activity = true;
}
}
else
mouse_oldright = false;
2016-02-26 20:31:02 +01:00
if (menu_input_mouse_state(MENU_MOUSE_WHEEL_DOWN))
{
2016-02-26 21:43:20 +01:00
BIT64_SET(*input_mouse, MENU_MOUSE_ACTION_WHEEL_DOWN);
*mouse_activity = true;
}
2016-02-26 20:31:02 +01:00
if (menu_input_mouse_state(MENU_MOUSE_WHEEL_UP))
{
2016-02-26 21:43:20 +01:00
BIT64_SET(*input_mouse, MENU_MOUSE_ACTION_WHEEL_UP);
*mouse_activity = true;
2016-02-26 21:43:20 +01:00
}
if (menu_input_mouse_state(MENU_MOUSE_HORIZ_WHEEL_DOWN))
{
BIT64_SET(*input_mouse, MENU_MOUSE_ACTION_HORIZ_WHEEL_DOWN);
*mouse_activity = true;
2016-02-26 21:43:20 +01:00
}
if (menu_input_mouse_state(MENU_MOUSE_HORIZ_WHEEL_UP))
{
BIT64_SET(*input_mouse, MENU_MOUSE_ACTION_HORIZ_WHEEL_UP);
*mouse_activity = true;
}
return 0;
}
2016-09-16 16:26:06 +02:00
static int menu_input_mouse_frame(
menu_file_list_cbs_t *cbs, menu_entry_t *entry,
unsigned action)
{
bool mouse_activity = false;
bool no_mouse_activity = false;
2016-09-16 16:26:06 +02:00
uint64_t mouse_state = MENU_MOUSE_ACTION_NONE;
int ret = 0;
settings_t *settings = config_get_ptr();
menu_input_t *menu_input = menu_input_get_ptr();
if (settings->menu.mouse.enable)
ret = menu_input_mouse_post_iterate(&mouse_state, cbs, action, &mouse_activity);
2016-09-16 16:26:06 +02:00
if (settings->menu.pointer.enable || settings->menu.mouse.enable)
2016-10-30 11:48:35 +01:00
{
menu_ctx_pointer_t point;
point.x = menu_input_mouse_state(MENU_MOUSE_X_AXIS);
point.y = menu_input_mouse_state(MENU_MOUSE_Y_AXIS);
menu_driver_ctl(RARCH_MENU_CTL_OSK_PTR_AT_POS, &point);
if (rarch_timer_is_running(&mouse_activity_timer))
rarch_timer_tick(&mouse_activity_timer);
if (mouse_old_x != point.x || mouse_old_y != point.y)
{
if (!rarch_timer_is_running(&mouse_activity_timer))
mouse_activity = true;
menu_event_set_osk_ptr(point.retcode);
}
else
{
if (rarch_timer_has_expired(&mouse_activity_timer))
no_mouse_activity = true;
}
mouse_old_x = point.x;
mouse_old_y = point.y;
2016-10-30 11:48:35 +01:00
}
2016-09-16 16:26:06 +02:00
if (BIT64_GET(mouse_state, MENU_MOUSE_ACTION_BUTTON_L))
{
menu_ctx_pointer_t point;
point.x = menu_input_mouse_state(MENU_MOUSE_X_AXIS);
point.y = menu_input_mouse_state(MENU_MOUSE_Y_AXIS);
point.ptr = menu_input->mouse.ptr;
point.cbs = cbs;
point.entry = entry;
point.action = action;
2016-10-30 11:48:35 +01:00
if (menu_input_dialog_get_display_kb())
{
menu_driver_ctl(RARCH_MENU_CTL_OSK_PTR_AT_POS, &point);
if (point.retcode > -1)
{
menu_event_set_osk_ptr(point.retcode);
2016-10-31 19:36:33 +01:00
menu_event_osk_append(point.retcode);
2016-10-30 11:48:35 +01:00
}
}
else
{
menu_driver_ctl(RARCH_MENU_CTL_POINTER_TAP, &point);
ret = point.retcode;
}
2016-09-16 16:26:06 +02:00
}
if (BIT64_GET(mouse_state, MENU_MOUSE_ACTION_BUTTON_R))
{
size_t selection;
menu_navigation_ctl(MENU_NAVIGATION_CTL_GET_SELECTION, &selection);
menu_entry_action(entry, selection, MENU_ACTION_CANCEL);
}
if (BIT64_GET(mouse_state, MENU_MOUSE_ACTION_WHEEL_DOWN))
{
unsigned increment_by = 1;
menu_navigation_ctl(MENU_NAVIGATION_CTL_INCREMENT, &increment_by);
}
if (BIT64_GET(mouse_state, MENU_MOUSE_ACTION_WHEEL_UP))
{
unsigned decrement_by = 1;
menu_navigation_ctl(MENU_NAVIGATION_CTL_DECREMENT, &decrement_by);
}
if (BIT64_GET(mouse_state, MENU_MOUSE_ACTION_HORIZ_WHEEL_UP))
{
/* stub */
}
if (BIT64_GET(mouse_state, MENU_MOUSE_ACTION_HORIZ_WHEEL_DOWN))
{
/* stub */
}
if (mouse_activity)
{
menu_ctx_environment_t menu_environ;
rarch_timer_begin(&mouse_activity_timer, 4);
menu_environ.type = MENU_ENVIRON_ENABLE_MOUSE_CURSOR;
menu_environ.data = NULL;
menu_driver_ctl(RARCH_MENU_CTL_ENVIRONMENT, &menu_environ);
}
if (no_mouse_activity)
{
menu_ctx_environment_t menu_environ;
rarch_timer_end(&mouse_activity_timer);
menu_environ.type = MENU_ENVIRON_DISABLE_MOUSE_CURSOR;
menu_environ.data = NULL;
menu_driver_ctl(RARCH_MENU_CTL_ENVIRONMENT, &menu_environ);
}
2016-09-16 16:26:06 +02:00
return ret;
}
int16_t menu_input_pointer_state(enum menu_input_pointer_state state)
{
menu_input_t *menu = menu_input_get_ptr();
if (!menu)
return 0;
switch (state)
{
case MENU_POINTER_X_AXIS:
return menu->pointer.x;
case MENU_POINTER_Y_AXIS:
return menu->pointer.y;
case MENU_POINTER_DELTA_X_AXIS:
return menu->pointer.dx;
case MENU_POINTER_DELTA_Y_AXIS:
return menu->pointer.dy;
2015-10-25 04:04:46 +01:00
case MENU_POINTER_PRESSED:
return menu->pointer.pressed[0];
}
return 0;
}
2015-09-24 17:29:46 +02:00
int16_t menu_input_mouse_state(enum menu_input_mouse_state state)
{
2016-03-02 22:02:35 +01:00
unsigned type = 0;
unsigned device = RETRO_DEVICE_MOUSE;
2015-09-24 17:29:46 +02:00
switch (state)
{
case MENU_MOUSE_X_AXIS:
device = RARCH_DEVICE_MOUSE_SCREEN;
type = RETRO_DEVICE_ID_MOUSE_X;
break;
2015-09-24 17:29:46 +02:00
case MENU_MOUSE_Y_AXIS:
device = RARCH_DEVICE_MOUSE_SCREEN;
type = RETRO_DEVICE_ID_MOUSE_Y;
break;
case MENU_MOUSE_LEFT_BUTTON:
2016-02-26 20:31:02 +01:00
type = RETRO_DEVICE_ID_MOUSE_LEFT;
break;
case MENU_MOUSE_RIGHT_BUTTON:
2016-02-26 20:31:02 +01:00
type = RETRO_DEVICE_ID_MOUSE_RIGHT;
break;
2015-10-24 03:13:40 +02:00
case MENU_MOUSE_WHEEL_UP:
2016-02-26 20:31:02 +01:00
type = RETRO_DEVICE_ID_MOUSE_WHEELUP;
break;
2015-10-24 03:13:40 +02:00
case MENU_MOUSE_WHEEL_DOWN:
2016-02-26 20:31:02 +01:00
type = RETRO_DEVICE_ID_MOUSE_WHEELDOWN;
break;
2016-02-26 21:43:20 +01:00
case MENU_MOUSE_HORIZ_WHEEL_UP:
type = RETRO_DEVICE_ID_MOUSE_HORIZ_WHEELUP;
break;
case MENU_MOUSE_HORIZ_WHEEL_DOWN:
type = RETRO_DEVICE_ID_MOUSE_HORIZ_WHEELDOWN;
break;
2016-02-26 20:31:02 +01:00
default:
return 0;
2015-09-24 17:29:46 +02:00
}
return current_input->input_state(current_input_data, NULL,
0, device, 0, type);
2015-09-24 17:29:46 +02:00
}
2016-02-25 20:53:52 +01:00
static int menu_input_pointer_post_iterate(
menu_file_list_cbs_t *cbs,
menu_entry_t *entry, unsigned action)
{
2016-02-26 21:04:56 +01:00
static bool pointer_oldpressed[2];
2016-02-26 20:43:12 +01:00
static bool pointer_oldback = false;
static int16_t start_x = 0;
static int16_t start_y = 0;
2016-02-26 20:43:12 +01:00
static int16_t pointer_old_x = 0;
static int16_t pointer_old_y = 0;
int ret = 0;
bool check_overlay = false;
2016-02-26 20:43:12 +01:00
menu_input_t *menu_input = menu_input_get_ptr();
settings_t *settings = config_get_ptr();
if (!menu_input || !settings)
return -1;
check_overlay = !settings->menu.pointer.enable;
#ifdef HAVE_OVERLAY
if (!check_overlay)
check_overlay = (settings->input.overlay_enable
&& input_overlay_is_alive(overlay_ptr));
#endif
2015-12-04 11:08:12 +01:00
if (check_overlay)
return 0;
2015-06-13 22:57:55 +02:00
if (menu_input->pointer.pressed[0])
{
2016-02-13 22:14:33 +01:00
gfx_ctx_metrics_t metrics;
float dpi;
static float accel0 = 0.0f;
static float accel1 = 0.0f;
int16_t pointer_x = menu_input_pointer_state(MENU_POINTER_X_AXIS);
int16_t pointer_y = menu_input_pointer_state(MENU_POINTER_Y_AXIS);
2016-01-21 16:38:45 -05:00
2016-02-13 22:14:33 +01:00
metrics.type = DISPLAY_METRIC_DPI;
metrics.value = &dpi;
2016-02-13 22:14:33 +01:00
2016-05-08 20:32:46 +02:00
video_context_driver_get_metrics(&metrics);
2016-02-26 21:04:56 +01:00
if (!pointer_oldpressed[0])
{
menu_input->pointer.accel = 0;
accel0 = 0;
accel1 = 0;
start_x = pointer_x;
start_y = pointer_y;
2016-02-26 20:43:12 +01:00
pointer_old_x = pointer_x;
pointer_old_y = pointer_y;
2016-02-26 21:04:56 +01:00
pointer_oldpressed[0] = true;
2015-04-04 21:26:11 +02:00
}
else if (abs(pointer_x - start_x) > (dpi / 10)
|| abs(pointer_y - start_y) > (dpi / 10))
2015-04-04 21:26:11 +02:00
{
2015-09-25 23:37:02 +02:00
float s, delta_time;
menu_input_ctl(MENU_INPUT_CTL_SET_POINTER_DRAGGED, NULL);
2016-02-26 20:43:12 +01:00
menu_input->pointer.dx = pointer_x - pointer_old_x;
menu_input->pointer.dy = pointer_y - pointer_old_y;
pointer_old_x = pointer_x;
pointer_old_y = pointer_y;
2015-07-15 03:04:44 +02:00
2015-09-25 23:37:02 +02:00
menu_animation_ctl(MENU_ANIMATION_CTL_DELTA_TIME, &delta_time);
s = (menu_input->pointer.dy * 550000000.0 ) /
( dpi * delta_time );
menu_input->pointer.accel = (accel0 + accel1 + s) / 3;
accel0 = accel1;
accel1 = menu_input->pointer.accel;
2015-04-04 21:26:11 +02:00
}
}
else
{
2016-02-26 21:04:56 +01:00
if (pointer_oldpressed[0])
{
if (!menu_input_ctl(MENU_INPUT_CTL_IS_POINTER_DRAGGED, NULL))
2016-02-11 00:47:00 +01:00
{
menu_ctx_pointer_t point;
point.x = start_x;
point.y = start_y;
2016-02-11 00:47:00 +01:00
point.ptr = menu_input->pointer.ptr;
point.cbs = cbs;
point.entry = entry;
point.action = action;
2016-10-30 11:48:35 +01:00
if (menu_input_dialog_get_display_kb())
{
menu_driver_ctl(RARCH_MENU_CTL_OSK_PTR_AT_POS, &point);
if (point.retcode > -1)
{
menu_event_set_osk_ptr(point.retcode);
2016-10-31 19:36:33 +01:00
menu_event_osk_append(point.retcode);
2016-10-30 11:48:35 +01:00
}
}
else
{
menu_driver_ctl(RARCH_MENU_CTL_POINTER_TAP, &point);
ret = point.retcode;
}
2016-02-11 00:47:00 +01:00
}
2015-06-13 22:57:55 +02:00
2016-02-26 21:04:56 +01:00
pointer_oldpressed[0] = false;
start_x = 0;
start_y = 0;
2016-02-26 20:43:12 +01:00
pointer_old_x = 0;
pointer_old_y = 0;
2015-06-13 22:57:55 +02:00
menu_input->pointer.dx = 0;
menu_input->pointer.dy = 0;
menu_input_ctl(MENU_INPUT_CTL_UNSET_POINTER_DRAGGED, NULL);
}
}
2015-06-13 22:57:55 +02:00
if (menu_input->pointer.back)
{
2016-02-26 20:43:12 +01:00
if (!pointer_oldback)
{
2016-06-22 06:55:11 +02:00
size_t selection;
2016-02-26 20:43:12 +01:00
pointer_oldback = true;
2016-06-22 06:55:11 +02:00
menu_navigation_ctl(MENU_NAVIGATION_CTL_GET_SELECTION, &selection);
menu_entry_action(entry, selection, MENU_ACTION_CANCEL);
}
}
2016-02-26 20:43:12 +01:00
pointer_oldback = menu_input->pointer.back;
2015-04-04 21:26:11 +02:00
return ret;
}
2015-05-11 11:48:43 +02:00
void menu_input_post_iterate(int *ret, unsigned action)
{
2015-09-25 14:57:37 +02:00
size_t selection;
menu_entry_t entry;
2015-10-17 17:44:57 +02:00
menu_file_list_cbs_t *cbs = NULL;
settings_t *settings = config_get_ptr();
file_list_t *selection_buf = menu_entries_get_selection_buf_ptr(0);
2015-05-11 11:48:43 +02:00
2015-09-25 14:57:37 +02:00
if (!menu_navigation_ctl(MENU_NAVIGATION_CTL_GET_SELECTION, &selection))
return;
2015-10-17 17:44:57 +02:00
if (selection_buf)
2015-10-19 16:32:51 +02:00
cbs = menu_entries_get_actiondata_at_offset(selection_buf, selection);
2015-09-25 14:57:37 +02:00
entry.path[0] = '\0';
entry.label[0] = '\0';
2016-10-20 15:47:02 +02:00
entry.sublabel[0] = '\0';
entry.value[0] = '\0';
entry.rich_label[0] = '\0';
entry.enum_idx = MSG_UNKNOWN;
entry.entry_idx = 0;
entry.idx = 0;
entry.type = 0;
entry.spacing = 0;
menu_entry_get(&entry, 0, selection, NULL, false);
2015-04-04 21:26:11 +02:00
2016-09-16 16:26:06 +02:00
*ret = menu_input_mouse_frame(cbs, &entry, action);
if (settings->menu.pointer.enable)
*ret |= menu_input_pointer_post_iterate(cbs, &entry, action);
}