static int action_toggle_scroll(unsigned type, const char *label, unsigned action) { unsigned scroll_speed = 0, fast_scroll_speed = 0; menu_handle_t *menu = menu_driver_resolve(); if (!menu) return -1; scroll_speed = (max(menu->navigation.scroll.acceleration, 2) - 2) / 4 + 1; fast_scroll_speed = 4 + 4 * scroll_speed; switch (action) { case MENU_ACTION_LEFT: if (menu->navigation.selection_ptr > fast_scroll_speed) menu_navigation_set(&menu->navigation, menu->navigation.selection_ptr - fast_scroll_speed, true); else menu_navigation_clear(&menu->navigation, false); break; case MENU_ACTION_RIGHT: if (menu->navigation.selection_ptr + fast_scroll_speed < (menu_list_get_size(menu->menu_list))) menu_navigation_set(&menu->navigation, menu->navigation.selection_ptr + fast_scroll_speed, true); else { if ((menu_list_get_size(menu->menu_list) > 0)) menu_navigation_set_last(&menu->navigation); } break; } return 0; }
/** * menu_navigation_increment: * * Increment the navigation pointer. **/ void menu_navigation_increment(menu_navigation_t *nav, unsigned scroll_speed) { settings_t *settings = config_get_ptr(); menu_list_t *menu_list = menu_list_get_ptr(); if (!nav) return; if (nav->selection_ptr + scroll_speed < (menu_list_get_size(menu_list))) { menu_navigation_set(nav, nav->selection_ptr + scroll_speed, true); menu_driver_navigation_increment(); } else { if (settings->menu.navigation.wraparound.vertical_enable) menu_navigation_clear(nav, false); else { if ((menu_list_get_size(menu_list) > 0)) { menu_navigation_set_last(nav); menu_driver_navigation_increment(); } } } }
static int menu_input_mouse_frame( menu_file_list_cbs_t *cbs, menu_entry_t *entry, uint64_t input_mouse) { menu_input_t *menu_input = menu_input_get_ptr(); menu_list_t *menu_list = menu_list_get_ptr(); menu_navigation_t *nav = menu_navigation_get_ptr(); if (BIT64_GET(input_mouse, MOUSE_ACTION_BUTTON_L)) { if (BIT64_GET(input_mouse, MOUSE_ACTION_BUTTON_L_TOGGLE)) return menu_entry_action(entry, nav->selection_ptr, MENU_ACTION_SELECT); if (BIT64_GET(input_mouse, MOUSE_ACTION_BUTTON_L_SET_NAVIGATION)) menu_navigation_set(nav, menu_input->mouse.ptr, false); } if (BIT64_GET(input_mouse, MOUSE_ACTION_BUTTON_R)) menu_list_pop_stack(menu_list, &nav->selection_ptr); if (BIT64_GET(input_mouse, MOUSE_ACTION_WHEEL_DOWN)) menu_navigation_increment(nav, 1); if (BIT64_GET(input_mouse, MOUSE_ACTION_WHEEL_UP)) menu_navigation_decrement(nav, 1); return 0; }
static int mouse_post_iterate(menu_file_list_cbs_t *cbs, const char *path, const char *label, unsigned type, unsigned action) { if (!driver.menu->mouse.enable) return 0; if (driver.menu->mouse.ptr <= menu_list_get_size(driver.menu->menu_list)-1) menu_navigation_set(driver.menu, driver.menu->mouse.ptr, false); if (driver.menu->mouse.left) { if (!driver.menu->mouse.oldleft) { driver.menu->mouse.oldleft = true; if (cbs && cbs->action_ok) return cbs->action_ok(path, label, type, driver.menu->selection_ptr); } } else driver.menu->mouse.oldleft = false; if (driver.menu->mouse.right) { if (!driver.menu->mouse.oldright) { driver.menu->mouse.oldright = true; menu_list_pop_stack(driver.menu->menu_list); } } else driver.menu->mouse.oldright = false; return 0; }
static int action_bind_up_or_down_generic(unsigned type, const char *label, unsigned action) { unsigned scroll_speed = 0; menu_handle_t *menu = menu_driver_resolve(); if (!menu) return -1; scroll_speed = (max(menu->navigation.scroll.acceleration, 2) - 2) / 4 + 1; if (menu_list_get_size(menu->menu_list) <= 0) return 0; switch (action) { case MENU_ACTION_UP: if (menu->navigation.selection_ptr >= scroll_speed) menu_navigation_set(&menu->navigation, menu->navigation.selection_ptr - scroll_speed, true); else { if (g_settings.menu.navigation.wraparound.vertical_enable) menu_navigation_set(&menu->navigation, menu_list_get_size(menu->menu_list) - 1, true); else menu_navigation_set(&menu->navigation, 0, true); } break; case MENU_ACTION_DOWN: if (menu->navigation.selection_ptr + scroll_speed < (menu_list_get_size(menu->menu_list))) menu_navigation_set(&menu->navigation, menu->navigation.selection_ptr + scroll_speed, true); else { if (g_settings.menu.navigation.wraparound.vertical_enable) menu_navigation_clear(&menu->navigation, false); else menu_navigation_set(&menu->navigation, menu_list_get_size(menu->menu_list) - 1, true); } break; } return 0; }
/** * menu_navigation_decrement: * * Decrement the navigation pointer. **/ void menu_navigation_decrement(menu_navigation_t *nav, unsigned scroll_speed) { menu_list_t *menu_list = menu_list_get_ptr(); settings_t *settings = config_get_ptr(); if (!nav) return; if (nav->selection_ptr >= scroll_speed) menu_navigation_set(nav, nav->selection_ptr - scroll_speed, true); else { if (settings->menu.navigation.wraparound.vertical_enable) menu_navigation_set(nav, menu_list_get_size(menu_list) - 1, true); else menu_navigation_set(nav, 0, true); } menu_driver_navigation_decrement(); }
static void menu_input_search_callback(void *userdata, const char *str) { size_t idx; menu_handle_t *menu = menu_driver_get_ptr(); if (!menu) return; if (str && *str && file_list_search(menu->menu_list->selection_buf, str, &idx)) menu_navigation_set(&menu->navigation, idx, true); menu_input_key_end_line(); }
static void menu_input_search_callback(void *userdata, const char *str) { size_t idx = 0; menu_list_t *menu_list = menu_list_get_ptr(); menu_navigation_t *nav = menu_navigation_get_ptr(); if (!menu_list || !nav) return; if (str && *str && file_list_search(menu_list->selection_buf, str, &idx)) menu_navigation_set(nav, idx, true); menu_input_key_end_line(); }
static int pointer_tap(menu_file_list_cbs_t *cbs, menu_entry_t *entry, unsigned action) { menu_input_t *menu_input = menu_input_get_ptr(); menu_navigation_t *nav = menu_navigation_get_ptr(); if (menu_input->pointer.ptr == nav->selection_ptr && cbs && cbs->action_select) return menu_entry_action(entry, nav->selection_ptr, MENU_ACTION_SELECT); else menu_navigation_set(nav, menu_input->pointer.ptr, false); return 0; }
/** * Before a refresh, we could have deleted a * file on disk, causing selection_ptr to * suddendly be out of range. * * Ensure it doesn't overflow. **/ static void menu_entries_refresh(file_list_t *list) { menu_handle_t *menu = menu_driver_get_ptr(); if (!menu) return; if (!list) return; if (menu->navigation.selection_ptr >= menu_list_get_size(menu->menu_list) && menu_list_get_size(menu->menu_list)) menu_navigation_set(&menu->navigation, menu_list_get_size(menu->menu_list) - 1, true); else if (!menu_list_get_size(menu->menu_list)) menu_navigation_clear(&menu->navigation, true); }
/** * Before a refresh, we could have deleted a * file on disk, causing selection_ptr to * suddendly be out of range. * * Ensure it doesn't overflow. **/ void menu_list_refresh(file_list_t *list) { menu_navigation_t *nav = menu_navigation_get_ptr(); menu_list_t *menu_list = menu_list_get_ptr(); if (!nav || !menu_list || !list) return; nav->scroll.indices.size = 0; menu_list_build_scroll_indices(list); if (nav->selection_ptr >= menu_list_get_size(menu_list) && menu_list_get_size(menu_list)) menu_navigation_set(nav, menu_list_get_size(menu_list) - 1, true); else if (!menu_list_get_size(menu_list)) menu_navigation_clear(nav, true); }
static int action_left_scroll(unsigned type, const char *label, bool wraparound) { unsigned scroll_speed = 0, fast_scroll_speed = 0; menu_navigation_t *nav = menu_navigation_get_ptr(); menu_list_t *menu_list = menu_list_get_ptr(); if (!nav || !menu_list) return -1; scroll_speed = (max(nav->scroll.acceleration, 2) - 2) / 4 + 1; fast_scroll_speed = 4 + 4 * scroll_speed; if (nav->selection_ptr > fast_scroll_speed) menu_navigation_set(nav, nav->selection_ptr - fast_scroll_speed, true); else menu_navigation_clear(nav, false); return 0; }
static int pointer_tap(menu_file_list_cbs_t *cbs, menu_entry_t *entry, unsigned action) { menu_handle_t *menu = menu_driver_get_ptr(); menu_list_t *menu_list = menu_list_get_ptr(); menu_navigation_t *nav = menu_navigation_get_ptr(); rarch_setting_t *setting = menu_setting_find( menu_list->selection_buf->list[nav->selection_ptr].label); if (menu->pointer.ptr == nav->selection_ptr && cbs && cbs->action_right && setting && (setting->type == ST_BOOL || setting->type == ST_UINT || setting->type == ST_FLOAT || setting->type == ST_STRING)) return menu_entry_action(entry, nav->selection_ptr, MENU_ACTION_RIGHT); else if (menu->pointer.ptr == nav->selection_ptr) return menu_entry_action(entry, nav->selection_ptr, MENU_ACTION_OK); else menu_navigation_set(nav, menu->pointer.ptr, false); return 0; }
static int action_right_scroll(unsigned type, const char *label, bool wraparound) { unsigned scroll_speed = 0, fast_scroll_speed = 0; menu_list_t *menu_list = menu_list_get_ptr(); menu_navigation_t *nav = menu_navigation_get_ptr(); if (!nav || !menu_list) return -1; scroll_speed = (max(nav->scroll.acceleration, 2) - 2) / 4 + 1; fast_scroll_speed = 4 + 4 * scroll_speed; if (nav->selection_ptr + fast_scroll_speed < (menu_list_get_size(menu_list))) menu_navigation_set(nav, nav->selection_ptr + fast_scroll_speed, true); else { if ((menu_list_get_size(menu_list) > 0)) menu_navigation_set_last(nav); } return 0; }
static int menu_input_pointer_post_iterate(menu_file_list_cbs_t *cbs, const char *path, const char *label, unsigned type, unsigned action) { menu_handle_t *menu = menu_driver_get_ptr(); settings_t *settings = config_get_ptr(); driver_t *driver = driver_get_ptr(); if (!menu) return -1; if (!settings->menu.pointer.enable #ifdef HAVE_OVERLAY || (settings->input.overlay_enable && driver && driver->overlay) #endif ) return 0; #if defined(HAVE_XMB) if (driver->menu_ctx == &menu_ctx_xmb) return 0; #endif if (menu->pointer.pressed[0]) { if (menu->pointer.oldpressed[0]) { if (menu->mouse.ptr <= menu_list_get_size(menu->menu_list)-1) menu_navigation_set(&menu->navigation, menu->mouse.ptr, false); } else menu->pointer.oldpressed[0] = true; } else { if (menu->pointer.oldpressed[0]) { rarch_setting_t *setting = NULL; menu->pointer.oldpressed[0] = false; setting = (rarch_setting_t*)setting_find_setting (menu->list_settings, menu->menu_list->selection_buf->list[menu->navigation.selection_ptr].label); if (menu->mouse.ptr == menu->navigation.selection_ptr && !menu->pointer.cancel && cbs && cbs->action_toggle && setting && (setting->type == ST_BOOL || setting->type == ST_UINT || setting->type == ST_FLOAT || setting->type == ST_STRING)) return cbs->action_toggle(type, label, MENU_ACTION_RIGHT, true); if (menu->mouse.ptr == menu->navigation.selection_ptr && !menu->pointer.cancel && cbs && cbs->action_ok) return cbs->action_ok(path, label, type, menu->navigation.selection_ptr); else if (menu->mouse.ptr <= menu_list_get_size(menu->menu_list) - 1) menu_navigation_set(&menu->navigation, menu->mouse.ptr, false); } } if (menu->pointer.back) { if (!menu->pointer.oldback) { menu->pointer.oldback = true; menu_list_pop_stack(menu->menu_list); } } menu->pointer.oldback = menu->pointer.back; return 0; }
static int menu_input_mouse_post_iterate(menu_file_list_cbs_t *cbs, const char *path, const char *label, unsigned type, unsigned action) { driver_t *driver = driver_get_ptr(); menu_handle_t *menu = menu_driver_get_ptr(); settings_t *settings = config_get_ptr(); if (!menu) return -1; if (!settings->menu.mouse.enable #ifdef HAVE_OVERLAY || (settings->input.overlay_enable && driver && driver->overlay) #endif ) { menu->mouse.wheeldown = false; menu->mouse.wheelup = false; menu->mouse.oldleft = false; menu->mouse.oldright = false; return 0; } if (menu->mouse.left) { if (!menu->mouse.oldleft) { rarch_setting_t *setting = (rarch_setting_t*)setting_find_setting (menu->list_settings, menu->menu_list->selection_buf->list[menu->navigation.selection_ptr].label); menu->mouse.oldleft = true; #if 0 RARCH_LOG("action OK: %d\n", cbs && cbs->action_ok); RARCH_LOG("action toggle: %d\n", cbs && cbs->action_toggle); if (setting && setting->type) RARCH_LOG("action type: %d\n", setting->type); #endif if (menu->mouse.ptr == menu->navigation.selection_ptr && cbs && cbs->action_toggle && setting && (setting->type == ST_BOOL || setting->type == ST_UINT || setting->type == ST_FLOAT || setting->type == ST_STRING)) return cbs->action_toggle(type, label, MENU_ACTION_RIGHT, true); if (menu->mouse.ptr == menu->navigation.selection_ptr && cbs && cbs->action_ok) return cbs->action_ok(path, label, type, menu->navigation.selection_ptr); else if (menu->mouse.ptr <= menu_list_get_size(menu->menu_list)-1) menu_navigation_set(&menu->navigation, menu->mouse.ptr, false); } } else menu->mouse.oldleft = false; if (menu->mouse.right) { if (!menu->mouse.oldright) { menu->mouse.oldright = true; menu_list_pop_stack(menu->menu_list); } } else menu->mouse.oldright = false; if (menu->mouse.wheeldown) menu_navigation_increment(&menu->navigation, 1); if (menu->mouse.wheelup) menu_navigation_decrement(&menu->navigation, 1); return 0; }
static int menu_common_iterate(unsigned action) { int ret = 0; unsigned type = 0; unsigned type_offset = 0; const char *label = NULL; const char *label_offset = NULL; const char *path_offset = NULL; unsigned scroll_speed = 0; menu_file_list_cbs_t *cbs = (menu_file_list_cbs_t*) menu_list_get_actiondata_at_offset(driver.menu->menu_list->selection_buf, driver.menu->selection_ptr); menu_list_get_last_stack(driver.menu->menu_list, NULL, &label, &type); mouse_iterate(action); if (driver.video_data && driver.menu_ctx && driver.menu_ctx->set_texture) driver.menu_ctx->set_texture(driver.menu); if (!strcmp(label, "help")) return menu_start_screen_iterate(action); else if (!strcmp(label, "message")) return menu_message_toggle(action); else if (!strcmp(label, "load_open_zip")) { switch (g_settings.archive.mode) { case 0: return menu_load_or_open_zip_iterate(action); case 1: return menu_archive_load(); case 2: return menu_archive_open(); default: break; } } else if (!strcmp(label, "info_screen")) return menu_info_screen_iterate(action); else if ( type == MENU_SETTINGS_CUSTOM_VIEWPORT || !strcmp(label, "custom_viewport_2") ) return menu_viewport_iterate(action); else if (type == MENU_SETTINGS_CUSTOM_BIND) { if (menu_input_bind_iterate(driver.menu)) menu_list_pop_stack(driver.menu->menu_list); return 0; } else if (type == MENU_SETTINGS_CUSTOM_BIND_KEYBOARD) { if (menu_input_bind_iterate_keyboard(driver.menu)) menu_list_pop_stack(driver.menu->menu_list); return 0; } menu_list_get_at_offset(driver.menu->menu_list->selection_buf, driver.menu->selection_ptr, &path_offset, &label_offset, &type_offset); if (driver.menu->need_refresh && action != MENU_ACTION_MESSAGE) action = MENU_ACTION_REFRESH; scroll_speed = (max(driver.menu->scroll_accel, 2) - 2) / 4 + 1; switch (action) { case MENU_ACTION_UP: if (driver.menu->selection_ptr >= scroll_speed) menu_navigation_set(driver.menu, driver.menu->selection_ptr - scroll_speed, true); else menu_navigation_set(driver.menu, menu_list_get_size(driver.menu->menu_list) - 1, true); break; case MENU_ACTION_DOWN: if (driver.menu->selection_ptr + scroll_speed < (menu_list_get_size(driver.menu->menu_list))) menu_navigation_set(driver.menu, driver.menu->selection_ptr + scroll_speed, true); else menu_navigation_clear(driver.menu, false); break; case MENU_ACTION_SCROLL_UP: menu_navigation_descend_alphabet(driver.menu, &driver.menu->selection_ptr); break; case MENU_ACTION_SCROLL_DOWN: menu_navigation_ascend_alphabet(driver.menu, &driver.menu->selection_ptr); break; case MENU_ACTION_CANCEL: apply_deferred_settings(); menu_list_pop_stack(driver.menu->menu_list); break; case MENU_ACTION_OK: if (cbs && cbs->action_ok) return cbs->action_ok(path_offset, label_offset, type_offset, driver.menu->selection_ptr); break; case MENU_ACTION_START: if (cbs && cbs->action_start) return cbs->action_start(type_offset, label_offset, action); break; case MENU_ACTION_LEFT: case MENU_ACTION_RIGHT: if (cbs && cbs->action_toggle) ret = cbs->action_toggle(type_offset, label_offset, action); break; case MENU_ACTION_SELECT: menu_list_push_stack(driver.menu->menu_list, "", "info_screen", 0, driver.menu->selection_ptr); break; case MENU_ACTION_REFRESH: menu_entries_deferred_push(driver.menu->menu_list->selection_buf, driver.menu->menu_list->menu_stack); driver.menu->need_refresh = false; break; case MENU_ACTION_MESSAGE: driver.menu->msg_force = true; break; default: break; } if (ret) return ret; ret = mouse_post_iterate(cbs, path_offset, label_offset, type_offset, action); if (driver.menu_ctx && driver.menu_ctx->iterate) driver.menu_ctx->iterate(driver.menu, action); if (driver.video_data && driver.menu_ctx && driver.menu_ctx->render) driver.menu_ctx->render(); /* Have to defer it so we let settings refresh. */ if (driver.menu->push_start_screen) { menu_list_push_stack(driver.menu->menu_list, "", "help", 0, 0); driver.menu->push_start_screen = false; } return ret; }