/* RetroArch - A frontend for libretro. * Copyright (C) 2011-2015 - 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 . */ #include #include "menu.h" #include "menu_entries_cbs.h" #include "menu_action.h" #include "menu_input.h" #include "menu_entries.h" #include "menu_shader.h" #include "../file_ext.h" #include "../file_extract.h" #include "../config.def.h" #include "../cheats.h" #include "../retroarch.h" #include "../performance.h" #ifdef HAVE_NETPLAY #include "../net_http.h" #endif #ifdef HAVE_LIBRETRODB #include "../libretrodb/libretrodb.h" #endif #include "../input/input_remapping.h" #ifdef GEKKO enum { GX_RESOLUTIONS_512_192 = 0, GX_RESOLUTIONS_598_200, GX_RESOLUTIONS_640_200, GX_RESOLUTIONS_384_224, GX_RESOLUTIONS_448_224, GX_RESOLUTIONS_480_224, GX_RESOLUTIONS_512_224, GX_RESOLUTIONS_576_224, GX_RESOLUTIONS_608_224, GX_RESOLUTIONS_640_224, GX_RESOLUTIONS_340_232, GX_RESOLUTIONS_512_232, GX_RESOLUTIONS_512_236, GX_RESOLUTIONS_336_240, GX_RESOLUTIONS_352_240, GX_RESOLUTIONS_384_240, GX_RESOLUTIONS_512_240, GX_RESOLUTIONS_530_240, GX_RESOLUTIONS_640_240, GX_RESOLUTIONS_512_384, GX_RESOLUTIONS_598_400, GX_RESOLUTIONS_640_400, GX_RESOLUTIONS_384_448, GX_RESOLUTIONS_448_448, GX_RESOLUTIONS_480_448, GX_RESOLUTIONS_512_448, GX_RESOLUTIONS_576_448, GX_RESOLUTIONS_608_448, GX_RESOLUTIONS_640_448, GX_RESOLUTIONS_340_464, GX_RESOLUTIONS_512_464, GX_RESOLUTIONS_512_472, GX_RESOLUTIONS_352_480, GX_RESOLUTIONS_384_480, GX_RESOLUTIONS_512_480, GX_RESOLUTIONS_530_480, GX_RESOLUTIONS_608_480, GX_RESOLUTIONS_640_480, GX_RESOLUTIONS_LAST, }; unsigned menu_gx_resolutions[GX_RESOLUTIONS_LAST][2] = { { 512, 192 }, { 598, 200 }, { 640, 200 }, { 384, 224 }, { 448, 224 }, { 480, 224 }, { 512, 224 }, { 576, 224 }, { 608, 224 }, { 640, 224 }, { 340, 232 }, { 512, 232 }, { 512, 236 }, { 336, 240 }, { 352, 240 }, { 384, 240 }, { 512, 240 }, { 530, 240 }, { 640, 240 }, { 512, 384 }, { 598, 400 }, { 640, 400 }, { 384, 448 }, { 448, 448 }, { 480, 448 }, { 512, 448 }, { 576, 448 }, { 608, 448 }, { 640, 448 }, { 340, 464 }, { 512, 464 }, { 512, 472 }, { 352, 480 }, { 384, 480 }, { 512, 480 }, { 530, 480 }, { 608, 480 }, { 640, 480 }, }; unsigned menu_current_gx_resolution = GX_RESOLUTIONS_640_480; #endif static int menu_action_generic_setting(rarch_setting_t *setting) { if (setting->change_handler) setting->change_handler(setting); if (setting->flags & SD_FLAG_EXIT && setting->cmd_trigger.triggered) { setting->cmd_trigger.triggered = false; return -1; } return 0; } int menu_action_setting_set_current_string( rarch_setting_t *setting, const char *str) { strlcpy(setting->value.string, str, setting->size); return menu_action_generic_setting(setting); } static void common_load_content(bool persist) { rarch_main_command(persist ? RARCH_CMD_LOAD_CONTENT_PERSIST : RARCH_CMD_LOAD_CONTENT); menu_list_flush_stack(driver.menu->menu_list, MENU_SETTINGS); driver.menu->msg_force = true; } static int action_ok_playlist_entry(const char *path, const char *label, unsigned type, size_t idx) { if (!driver.menu) return -1; rarch_playlist_load_content(g_defaults.history, driver.menu->selection_ptr); menu_list_flush_stack(driver.menu->menu_list, MENU_SETTINGS); return -1; } static int action_ok_cheat_apply_changes(const char *path, const char *label, unsigned type, size_t idx) { cheat_manager_t *cheat = g_extern.cheat; if (!cheat) return -1; cheat_manager_apply_cheats(cheat); return 0; } /* FIXME: Ugly hack, need to be refactored badly. */ size_t hack_shader_pass = 0; static int action_ok_shader_pass_load(const char *path, const char *label, unsigned type, size_t idx) { const char *menu_path = NULL; if (!driver.menu) return -1; (void)menu_path; #ifdef HAVE_SHADER_MANAGER menu_list_get_last_stack(driver.menu->menu_list, &menu_path, NULL, NULL); fill_pathname_join(driver.menu->shader->pass[hack_shader_pass].source.path, menu_path, path, sizeof(driver.menu->shader->pass[hack_shader_pass].source.path)); /* This will reset any changed parameters. */ video_shader_resolve_parameters(NULL, driver.menu->shader); menu_list_flush_stack_by_needle(driver.menu->menu_list, "Shader Options"); return 0; #else return -1; #endif } static int action_ok_file_generic(const char *path, const char *label, unsigned type, size_t idx) { if (!driver.menu) return -1; menu_list_push_stack_refresh( driver.menu->menu_list, path, label, type, driver.menu->selection_ptr); return 0; } #ifdef HAVE_SHADER_MANAGER extern size_t hack_shader_pass; #endif static int action_ok_shader_pass(const char *path, const char *label, unsigned type, size_t idx) { hack_shader_pass = type - MENU_SETTINGS_SHADER_PASS_0; return action_ok_file_generic( g_settings.video.shader_dir, label, type, idx); } static int action_ok_shader_parameters(const char *path, const char *label, unsigned type, size_t idx) { return action_ok_file_generic( "", label, MENU_SETTING_ACTION, idx); } static int action_ok_push_generic_list(const char *path, const char *label, unsigned type, size_t idx) { return action_ok_file_generic( "", label, type, idx); } static int action_ok_push_default(const char *path, const char *label, unsigned type, size_t idx) { return action_ok_file_generic( label, label, type, idx); } static int action_ok_shader_preset(const char *path, const char *label, unsigned type, size_t idx) { return action_ok_file_generic( g_settings.video.shader_dir, label, type, idx); } static int action_ok_push_content_list(const char *path, const char *label, unsigned type, size_t idx) { return action_ok_file_generic( g_settings.menu_content_directory, label, MENU_FILE_DIRECTORY, idx); } static int action_ok_disk_image_append_list(const char *path, const char *label, unsigned type, size_t idx) { return action_ok_file_generic( g_settings.menu_content_directory, label, type, idx); } static int action_ok_configurations_list(const char *path, const char *label, unsigned type, size_t idx) { const char *dir = g_settings.menu_config_directory; return action_ok_file_generic( dir ? dir : label, label, type, idx); } static int action_ok_cheat_file(const char *path, const char *label, unsigned type, size_t idx) { return action_ok_file_generic( g_settings.cheat_database, label, type, idx); } static int action_ok_remap_file(const char *path, const char *label, unsigned type, size_t idx) { return action_ok_file_generic( g_settings.input_remapping_directory, label, type, idx); } static int action_ok_core_list(const char *path, const char *label, unsigned type, size_t idx) { return action_ok_file_generic( g_settings.libretro_directory, label, type, idx); } static int action_ok_remap_file_load(const char *path, const char *label, unsigned type, size_t idx) { const char *menu_path = NULL; char remap_path[PATH_MAX_LENGTH]; if (!driver.menu) return -1; (void)remap_path; (void)menu_path; menu_list_get_last_stack(driver.menu->menu_list, &menu_path, NULL, NULL); fill_pathname_join(remap_path, menu_path, path, sizeof(remap_path)); input_remapping_load_file(remap_path); menu_list_flush_stack_by_needle(driver.menu->menu_list, "core_input_remapping_options"); return 0; } static int action_ok_cheat_file_load(const char *path, const char *label, unsigned type, size_t idx) { const char *menu_path = NULL; char cheat_path[PATH_MAX_LENGTH]; if (!driver.menu) return -1; (void)cheat_path; (void)menu_path; menu_list_get_last_stack(driver.menu->menu_list, &menu_path, NULL, NULL); fill_pathname_join(cheat_path, menu_path, path, sizeof(cheat_path)); if (g_extern.cheat) cheat_manager_free(g_extern.cheat); g_extern.cheat = cheat_manager_load(cheat_path); if (!g_extern.cheat) return -1; menu_list_flush_stack_by_needle(driver.menu->menu_list, "core_cheat_options"); return 0; } static int action_ok_menu_wallpaper_load(const char *path, const char *label, unsigned type, size_t idx) { const char *menu_label = NULL; const char *menu_path = NULL; rarch_setting_t *setting = NULL; char wallpaper_path[PATH_MAX_LENGTH]; if (!driver.menu) return -1; menu_list_get_last_stack(driver.menu->menu_list, &menu_path, &menu_label, NULL); setting = (rarch_setting_t*) setting_data_find_setting(driver.menu->list_settings, menu_label); if (!setting) return -1; fill_pathname_join(wallpaper_path, menu_path, path, sizeof(wallpaper_path)); strlcpy(g_settings.menu.wallpaper, wallpaper_path, sizeof(g_settings.menu.wallpaper)); menu_list_pop_stack_by_needle(driver.menu->menu_list, setting->name); return 0; } static int action_ok_shader_preset_load(const char *path, const char *label, unsigned type, size_t idx) { const char *menu_path = NULL; char shader_path[PATH_MAX_LENGTH]; if (!driver.menu) return -1; (void)shader_path; (void)menu_path; #ifdef HAVE_SHADER_MANAGER menu_list_get_last_stack(driver.menu->menu_list, &menu_path, NULL, NULL); fill_pathname_join(shader_path, menu_path, path, sizeof(shader_path)); menu_shader_manager_set_preset(driver.menu->shader, video_shader_parse_type(shader_path, RARCH_SHADER_NONE), shader_path); menu_list_flush_stack_by_needle(driver.menu->menu_list, "Shader Options"); return 0; #else return -1; #endif } static int action_ok_save_as_generic(const char *desc, const char *label, unsigned type, size_t idx, input_keyboard_line_complete_t cb) { if (!driver.menu) return -1; menu_input_key_start_line(driver.menu, desc, label, type, idx, cb); return 0; } static int action_ok_cheat(const char *path, const char *label, unsigned type, size_t idx) { return action_ok_save_as_generic("Input Cheat", label, type, idx, menu_input_st_cheat_callback); } static int action_ok_shader_preset_save_as(const char *path, const char *label, unsigned type, size_t idx) { return action_ok_save_as_generic("Preset Filename", label, type, idx, menu_input_st_string_callback); } static int action_ok_cheat_file_save_as(const char *path, const char *label, unsigned type, size_t idx) { return action_ok_save_as_generic("Cheat Filename", label, type, idx, menu_input_st_string_callback); } static int action_ok_remap_file_save_as(const char *path, const char *label, unsigned type, size_t idx) { return action_ok_save_as_generic("Remapping Filename", label, type, idx, menu_input_st_string_callback); } static int action_ok_path_use_directory(const char *path, const char *label, unsigned type, size_t idx) { const char *menu_label = NULL; const char *menu_path = NULL; rarch_setting_t *setting = NULL; if (!driver.menu) return -1; menu_list_get_last_stack(driver.menu->menu_list, &menu_path, &menu_label, NULL); setting = (rarch_setting_t*) setting_data_find_setting(driver.menu->list_settings, menu_label); if (!setting) return -1; if (setting->type == ST_DIR) { menu_action_setting_set_current_string(setting, menu_path); menu_list_pop_stack_by_needle(driver.menu->menu_list, setting->name); } return 0; } static int action_ok_core_load_deferred(const char *path, const char *label, unsigned type, size_t idx) { if (!driver.menu) return -1; strlcpy(g_settings.libretro, path, sizeof(g_settings.libretro)); strlcpy(g_extern.fullpath, driver.menu->deferred_path, sizeof(g_extern.fullpath)); common_load_content(false); return -1; } static int action_ok_database_manager_list_deferred(const char *path, const char *label, unsigned type, size_t idx) { return 0; } static int action_ok_core_load(const char *path, const char *label, unsigned type, size_t idx) { const char *menu_path = NULL; if (!driver.menu) return -1; menu_list_get_last_stack(driver.menu->menu_list, &menu_path, NULL, NULL); fill_pathname_join(g_settings.libretro, menu_path, path, sizeof(g_settings.libretro)); rarch_main_command(RARCH_CMD_LOAD_CORE); menu_list_flush_stack(driver.menu->menu_list, MENU_SETTINGS); #if defined(HAVE_DYNAMIC) /* No content needed for this core, load core immediately. */ if (driver.menu->load_no_content) { *g_extern.fullpath = '\0'; common_load_content(false); return -1; } return 0; /* Core selection on non-console just updates directory listing. * Will take effect on new content load. */ #elif defined(RARCH_CONSOLE) rarch_main_command(RARCH_CMD_RESTART_RETROARCH); return -1; #endif } static int action_ok_core_download(const char *path, const char *label, unsigned type, size_t idx) { const char *menu_path = NULL; if (!driver.menu) return -1; menu_list_get_last_stack(driver.menu->menu_list, &menu_path, NULL, NULL); return 0; } static int action_ok_compressed_archive_push(const char *path, const char *label, unsigned type, size_t idx) { if (!driver.menu) return -1; menu_list_push_stack( driver.menu->menu_list, path, "load_open_zip", 0, driver.menu->selection_ptr); return 0; } static int action_ok_directory_push(const char *path, const char *label, unsigned type, size_t idx) { const char *menu_path = NULL; const char *menu_label = NULL; char cat_path[PATH_MAX_LENGTH]; if (!driver.menu) return -1; if (!path) return -1; menu_list_get_last_stack(driver.menu->menu_list, &menu_path, &menu_label, NULL); fill_pathname_join(cat_path, menu_path, path, sizeof(cat_path)); return action_ok_file_generic(cat_path, menu_label, type, idx); } static int action_ok_database_manager_list(const char *path, const char *label, unsigned type, size_t idx) { char rdb_path[PATH_MAX_LENGTH]; fill_pathname_join(rdb_path, g_settings.content_database, path, sizeof(rdb_path)); return action_ok_file_generic( rdb_path, "deferred_database_manager_list", 0, driver.menu->selection_ptr); } static int action_ok_config_load(const char *path, const char *label, unsigned type, size_t idx) { const char *menu_path = NULL; char config[PATH_MAX_LENGTH]; if (!driver.menu) return -1; menu_list_get_last_stack(driver.menu->menu_list, &menu_path, NULL, NULL); fill_pathname_join(config, menu_path, path, sizeof(config)); menu_list_flush_stack(driver.menu->menu_list, MENU_SETTINGS); driver.menu->msg_force = true; if (rarch_replace_config(config)) { menu_navigation_clear(driver.menu, false); return -1; } return 0; } static int action_ok_disk_image_append(const char *path, const char *label, unsigned type, size_t idx) { const char *menu_path = NULL; char image[PATH_MAX_LENGTH]; if (!driver.menu) return -1; menu_list_get_last_stack(driver.menu->menu_list, &menu_path, NULL, NULL); fill_pathname_join(image, menu_path, path, sizeof(image)); rarch_disk_control_append_image(image); rarch_main_command(RARCH_CMD_RESUME); menu_list_flush_stack(driver.menu->menu_list, MENU_SETTINGS); return -1; } static int action_ok_file_load_with_detect_core(const char *path, const char *label, unsigned type, size_t idx) { const char *menu_path = NULL; int ret; if (!driver.menu) return -1; menu_list_get_last_stack(driver.menu->menu_list, &menu_path, NULL, NULL); ret = rarch_defer_core(g_extern.core_info, menu_path, path, label, driver.menu->deferred_path, sizeof(driver.menu->deferred_path)); if (ret == -1) { rarch_main_command(RARCH_CMD_LOAD_CORE); common_load_content(false); return -1; } else if (ret == 0) menu_list_push_stack_refresh( driver.menu->menu_list, g_settings.libretro_directory, "deferred_core_list", 0, driver.menu->selection_ptr); return ret; } static int menu_action_setting_set_current_string_path( rarch_setting_t *setting, const char *dir, const char *path) { fill_pathname_join(setting->value.string, dir, path, setting->size); return menu_action_generic_setting(setting); } static int action_ok_file_load(const char *path, const char *label, unsigned type, size_t idx) { const char *menu_label = NULL; const char *menu_path = NULL; rarch_setting_t *setting = NULL; if (!driver.menu) return -1; menu_list_get_last(driver.menu->menu_list->menu_stack, &menu_path, &menu_label, NULL); setting = (rarch_setting_t*) setting_data_find_setting(driver.menu->list_settings, menu_label); if (setting && setting->type == ST_PATH) { menu_action_setting_set_current_string_path(setting, menu_path, path); menu_list_pop_stack_by_needle(driver.menu->menu_list, setting->name); } else { if (type == MENU_FILE_IN_CARCHIVE) fill_pathname_join_delim(g_extern.fullpath, menu_path, path, '#',sizeof(g_extern.fullpath)); else fill_pathname_join(g_extern.fullpath, menu_path, path, sizeof(g_extern.fullpath)); common_load_content(true); return -1; } return 0; } static int action_ok_set_path(const char *path, const char *label, unsigned type, size_t idx) { const char *menu_path = NULL; const char *menu_label = NULL; rarch_setting_t *setting = NULL; if (!driver.menu) return -1; menu_list_get_last_stack(driver.menu->menu_list, &menu_path, &menu_label, NULL); setting = (rarch_setting_t*) setting_data_find_setting(driver.menu->list_settings, menu_label); if (!setting) return -1; menu_action_setting_set_current_string_path(setting, menu_path, path); menu_list_pop_stack_by_needle(driver.menu->menu_list, setting->name); return 0; } static int action_ok_custom_viewport(const char *path, const char *label, unsigned type, size_t idx) { menu_list_push_stack( driver.menu->menu_list, "", "", MENU_SETTINGS_CUSTOM_VIEWPORT, driver.menu->selection_ptr); /* Start with something sane. */ rarch_viewport_t *custom = (rarch_viewport_t*) &g_extern.console.screen.viewports.custom_vp; if (driver.video_data && driver.video && driver.video->viewport_info) driver.video->viewport_info(driver.video_data, custom); aspectratio_lut[ASPECT_RATIO_CUSTOM].value = (float)custom->width / custom->height; g_settings.video.aspect_ratio_idx = ASPECT_RATIO_CUSTOM; rarch_main_command(RARCH_CMD_VIDEO_SET_ASPECT_RATIO); return 0; } static int generic_action_ok_command(unsigned cmd) { if (!driver.menu) return -1; if (!rarch_main_command(cmd)) return -1; return 0; } static int action_ok_load_state(const char *path, const char *label, unsigned type, size_t idx) { if (generic_action_ok_command(RARCH_CMD_LOAD_STATE) == -1) return -1; return generic_action_ok_command(RARCH_CMD_RESUME); } static int action_ok_save_state(const char *path, const char *label, unsigned type, size_t idx) { if (generic_action_ok_command(RARCH_CMD_SAVE_STATE) == -1) return -1; return generic_action_ok_command(RARCH_CMD_RESUME); } /* HACK - we have to find some way to pass state inbetween * function pointer callback functions that don't necessarily * call each other. */ static char core_manager_path[PATH_MAX_LENGTH]; static int cb_core_manager_download(void *data_, size_t len) { FILE *f; char output_path[PATH_MAX_LENGTH]; char *data = (char*)data_; if (!data) return -1; fill_pathname_join(output_path, g_settings.libretro_directory, core_manager_path, sizeof(output_path)); f = fopen(output_path, "wb"); if (!f) return -1; fwrite(data, 1, len, f); fclose(f); return 0; } static int action_ok_core_manager_list(const char *path, const char *label, unsigned type, size_t idx) { #ifdef HAVE_NETPLAY char core_path[PATH_MAX_LENGTH]; fill_pathname_join(core_path, g_settings.network.buildbot_url, path, sizeof(core_path)); strlcpy(core_manager_path, path, sizeof(core_manager_path)); msg_queue_clear(g_extern.http_msg_queue); msg_queue_push(g_extern.http_msg_queue, core_path, 0, 180); net_http_set_pending_cb(cb_core_manager_download); #endif return 0; } static int action_ok_disk_cycle_tray_status(const char *path, const char *label, unsigned type, size_t idx) { return generic_action_ok_command(RARCH_CMD_DISK_EJECT_TOGGLE); } static int action_ok_quit(const char *path, const char *label, unsigned type, size_t idx) { return generic_action_ok_command(RARCH_CMD_QUIT); } static int action_ok_save_new_config(const char *path, const char *label, unsigned type, size_t idx) { return generic_action_ok_command(RARCH_CMD_MENU_SAVE_CONFIG); } static int action_ok_resume_content(const char *path, const char *label, unsigned type, size_t idx) { return generic_action_ok_command(RARCH_CMD_RESUME); } static int action_ok_restart_content(const char *path, const char *label, unsigned type, size_t idx) { return generic_action_ok_command(RARCH_CMD_RESET); } static int action_ok_screenshot(const char *path, const char *label, unsigned type, size_t idx) { return generic_action_ok_command(RARCH_CMD_TAKE_SCREENSHOT); } static int action_ok_shader_apply_changes(const char *path, const char *label, unsigned type, size_t idx) { return generic_action_ok_command(RARCH_CMD_SHADERS_APPLY_CHANGES); } static int action_ok_lookup_setting(const char *path, const char *label, unsigned type, size_t idx) { return menu_action_setting_set(type, label, MENU_ACTION_OK); } static int action_cancel_lookup_setting(const char *path, const char *label, unsigned type, size_t idx) { return menu_action_setting_set(type, label, MENU_ACTION_CANCEL); } static int action_cancel_pop_default(const char *path, const char *label, unsigned type, size_t idx) { menu_apply_deferred_settings(); menu_list_pop_stack(driver.menu->menu_list); return 0; } static int action_ok_help(const char *path, const char *label, unsigned type, size_t idx) { if (!driver.menu) return -1; menu_list_push_stack( driver.menu->menu_list, "", "help", 0, 0); driver.menu->push_start_screen = false; return 0; } static int action_start_performance_counters_core(unsigned type, const char *label, unsigned action) { struct retro_perf_counter **counters = (struct retro_perf_counter**) perf_counters_libretro; unsigned offset = type - MENU_SETTINGS_LIBRETRO_PERF_COUNTERS_BEGIN; (void)label; (void)action; if (counters[offset]) { counters[offset]->total = 0; counters[offset]->call_cnt = 0; } return 0; } static int action_start_input_desc(unsigned type, const char *label, unsigned action) { unsigned inp_desc_index_offset = type - MENU_SETTINGS_INPUT_DESC_BEGIN; unsigned inp_desc_user = inp_desc_index_offset / RARCH_FIRST_CUSTOM_BIND; unsigned inp_desc_button_index_offset = inp_desc_index_offset - (inp_desc_user * RARCH_FIRST_CUSTOM_BIND); (void)label; (void)action; g_settings.input.remap_ids[inp_desc_user][inp_desc_button_index_offset] = g_settings.input.binds[inp_desc_user][inp_desc_button_index_offset].id; return 0; } static int action_start_shader_action_parameter(unsigned type, const char *label, unsigned action) { #ifdef HAVE_SHADER_MANAGER struct video_shader *shader = NULL; struct video_shader_parameter *param = NULL; if (driver.video_poke && driver.video_data && driver.video_poke->get_current_shader) shader = driver.video_poke->get_current_shader(driver.video_data); if (!shader) return 0; param = &shader->parameters[type - MENU_SETTINGS_SHADER_PARAMETER_0]; param->current = param->initial; param->current = min(max(param->minimum, param->current), param->maximum); #endif return 0; } static int action_start_shader_action_preset_parameter(unsigned type, const char *label, unsigned action) { #ifdef HAVE_SHADER_MANAGER struct video_shader *shader = NULL; struct video_shader_parameter *param = NULL; if (!(shader = driver.menu->shader)) return 0; param = &shader->parameters[type - MENU_SETTINGS_SHADER_PRESET_PARAMETER_0]; param->current = param->initial; param->current = min(max(param->minimum, param->current), param->maximum); #endif return 0; } static int shader_action_parameter_toggle(unsigned type, const char *label, unsigned action) { #ifdef HAVE_SHADER_MANAGER struct video_shader *shader = NULL; struct video_shader_parameter *param = NULL; if (driver.video_poke && driver.video_data && driver.video_poke->get_current_shader) shader = driver.video_poke->get_current_shader(driver.video_data); if (!shader) return 0; param = &shader->parameters[type - MENU_SETTINGS_SHADER_PARAMETER_0]; switch (action) { case MENU_ACTION_LEFT: param->current -= param->step; break; case MENU_ACTION_RIGHT: param->current += param->step; break; default: break; } param->current = min(max(param->minimum, param->current), param->maximum); #endif return 0; } static int shader_action_parameter_preset_toggle(unsigned type, const char *label, unsigned action) { #ifdef HAVE_SHADER_MANAGER struct video_shader *shader = NULL; struct video_shader_parameter *param = NULL; if (!(shader = driver.menu->shader)) return 0; param = &shader->parameters[type - MENU_SETTINGS_SHADER_PRESET_PARAMETER_0]; switch (action) { case MENU_ACTION_LEFT: param->current -= param->step; break; case MENU_ACTION_RIGHT: param->current += param->step; break; default: break; } param->current = min(max(param->minimum, param->current), param->maximum); #endif return 0; } static int action_toggle_cheat(unsigned type, const char *label, unsigned action) { cheat_manager_t *cheat = g_extern.cheat; size_t idx = type - MENU_SETTINGS_CHEAT_BEGIN; if (!cheat) return -1; switch (action) { case MENU_ACTION_LEFT: case MENU_ACTION_RIGHT: cheat->cheats[idx].state = !cheat->cheats[idx].state; cheat_manager_update(cheat, idx); break; } return 0; } static int action_toggle_input_desc(unsigned type, const char *label, unsigned action) { unsigned inp_desc_index_offset = type - MENU_SETTINGS_INPUT_DESC_BEGIN; unsigned inp_desc_user = inp_desc_index_offset / RARCH_FIRST_CUSTOM_BIND; unsigned inp_desc_button_index_offset = inp_desc_index_offset - (inp_desc_user * RARCH_FIRST_CUSTOM_BIND); switch (action) { case MENU_ACTION_LEFT: if (g_settings.input.remap_ids[inp_desc_user][inp_desc_button_index_offset] > 0) g_settings.input.remap_ids[inp_desc_user][inp_desc_button_index_offset]--; break; case MENU_ACTION_RIGHT: if (g_settings.input.remap_ids[inp_desc_user][inp_desc_button_index_offset] < RARCH_FIRST_CUSTOM_BIND) g_settings.input.remap_ids[inp_desc_user][inp_desc_button_index_offset]++; break; } return 0; } static int action_start_shader_pass(unsigned type, const char *label, unsigned action) { #ifdef HAVE_SHADER_MANAGER hack_shader_pass = type - MENU_SETTINGS_SHADER_PASS_0; struct video_shader *shader = driver.menu->shader; struct video_shader_pass *shader_pass = NULL; if (shader) shader_pass = &shader->pass[hack_shader_pass]; if (shader_pass) *shader_pass->source.path = '\0'; #endif return 0; } static int action_start_shader_scale_pass(unsigned type, const char *label, unsigned action) { #ifdef HAVE_SHADER_MANAGER unsigned pass = type - MENU_SETTINGS_SHADER_PASS_SCALE_0; struct video_shader *shader = driver.menu->shader; struct video_shader_pass *shader_pass = &shader->pass[pass]; if (shader) { shader_pass->fbo.scale_x = shader_pass->fbo.scale_y = 0; shader_pass->fbo.valid = false; } #endif return 0; } static int action_toggle_save_state(unsigned type, const char *label, unsigned action) { switch (action) { case MENU_ACTION_LEFT: /* Slot -1 is (auto) slot. */ if (g_settings.state_slot >= 0) g_settings.state_slot--; break; case MENU_ACTION_RIGHT: g_settings.state_slot++; break; } return 0; } static int action_toggle_scroll(unsigned type, const char *label, unsigned action) { unsigned scroll_speed = 0, fast_scroll_speed = 0; if (!driver.menu) return -1; scroll_speed = (max(driver.menu->scroll_accel, 2) - 2) / 4 + 1; fast_scroll_speed = 4 + 4 * scroll_speed; (void)scroll_speed; switch (action) { case MENU_ACTION_LEFT: if (driver.menu->selection_ptr > fast_scroll_speed) menu_navigation_set(driver.menu, driver.menu->selection_ptr - fast_scroll_speed, true); else menu_navigation_clear(driver.menu, false); break; case MENU_ACTION_RIGHT: if (driver.menu->selection_ptr + fast_scroll_speed < (menu_list_get_size(driver.menu->menu_list))) menu_navigation_set(driver.menu, driver.menu->selection_ptr + fast_scroll_speed, true); else menu_navigation_set_last(driver.menu); break; } return 0; } static int action_toggle_mainmenu(unsigned type, const char *label, unsigned action) { menu_file_list_cbs_t *cbs = NULL; unsigned push_list = 0; if (!driver.menu) return -1; if (file_list_get_size(driver.menu->menu_list->menu_stack) == 1) { if (!strcmp(driver.menu_ctx->ident, "xmb")) { switch (action) { case MENU_ACTION_LEFT: if (driver.menu->cat_selection_ptr == 0) break; push_list = 1; break; case MENU_ACTION_RIGHT: if (driver.menu->cat_selection_ptr == g_extern.core_info->count) break; push_list = 1; break; } } } else push_list = 2; cbs = (menu_file_list_cbs_t*) menu_list_get_actiondata_at_offset(driver.menu->menu_list->selection_buf, driver.menu->selection_ptr); switch (push_list) { case 1: if (driver.menu_ctx->list_cache) driver.menu_ctx->list_cache(true, action); if (cbs && cbs->action_content_list_switch) return cbs->action_content_list_switch( driver.menu->menu_list->selection_buf, driver.menu->menu_list->menu_stack, "", "", 0); break; case 2: action_toggle_scroll(0, "", action); break; case 0: default: break; } return 0; } static int action_toggle_shader_scale_pass(unsigned type, const char *label, unsigned action) { #ifdef HAVE_SHADER_MANAGER unsigned pass = type - MENU_SETTINGS_SHADER_PASS_SCALE_0; struct video_shader *shader = driver.menu->shader; struct video_shader_pass *shader_pass = &shader->pass[pass]; switch (action) { case MENU_ACTION_LEFT: case MENU_ACTION_RIGHT: { unsigned current_scale = shader_pass->fbo.scale_x; unsigned delta = action == MENU_ACTION_LEFT ? 5 : 1; current_scale = (current_scale + delta) % 6; if (shader_pass) { shader_pass->fbo.valid = current_scale; shader_pass->fbo.scale_x = shader_pass->fbo.scale_y = current_scale; } } break; } #endif return 0; } static int action_start_shader_filter_pass(unsigned type, const char *label, unsigned action) { #ifdef HAVE_SHADER_MANAGER unsigned pass = type - MENU_SETTINGS_SHADER_PASS_FILTER_0; struct video_shader *shader = driver.menu->shader; struct video_shader_pass *shader_pass = &shader->pass[pass]; if (shader && shader_pass) shader_pass->filter = RARCH_FILTER_UNSPEC; #endif return 0; } static int action_toggle_shader_filter_pass(unsigned type, const char *label, unsigned action) { #ifdef HAVE_SHADER_MANAGER unsigned pass = type - MENU_SETTINGS_SHADER_PASS_FILTER_0; struct video_shader *shader = driver.menu->shader; struct video_shader_pass *shader_pass = &shader->pass[pass]; switch (action) { case MENU_ACTION_LEFT: case MENU_ACTION_RIGHT: { unsigned delta = (action == MENU_ACTION_LEFT) ? 2 : 1; if (shader_pass) shader_pass->filter = ((shader_pass->filter + delta) % 3); } break; } #endif return 0; } static int action_toggle_shader_filter_default(unsigned type, const char *label, unsigned action) { #ifdef HAVE_SHADER_MANAGER rarch_setting_t *current_setting = NULL; if ((current_setting = setting_data_find_setting( driver.menu->list_settings, "video_smooth"))) menu_action_setting_handler(current_setting, action); #endif return 0; } static int action_start_shader_num_passes(unsigned type, const char *label, unsigned action) { #ifdef HAVE_SHADER_MANAGER struct video_shader *shader = driver.menu->shader; if (!shader) return -1; if (shader->passes) shader->passes = 0; driver.menu->need_refresh = true; video_shader_resolve_parameters(NULL, driver.menu->shader); #endif return 0; } static int action_start_cheat_num_passes(unsigned type, const char *label, unsigned action) { cheat_manager_t *cheat = g_extern.cheat; if (!cheat) return -1; if (cheat->size) { cheat_manager_realloc(cheat, 0); driver.menu->need_refresh = true; } return 0; } static int action_toggle_cheat_num_passes(unsigned type, const char *label, unsigned action) { unsigned new_size = 0; cheat_manager_t *cheat = g_extern.cheat; if (!cheat) return -1; switch (action) { case MENU_ACTION_LEFT: if (cheat->size) new_size = cheat->size - 1; driver.menu->need_refresh = true; break; case MENU_ACTION_RIGHT: new_size = cheat->size + 1; driver.menu->need_refresh = true; break; } if (driver.menu->need_refresh) cheat_manager_realloc(cheat, new_size); return 0; } static int action_toggle_shader_num_passes(unsigned type, const char *label, unsigned action) { #ifdef HAVE_SHADER_MANAGER struct video_shader *shader = driver.menu->shader; if (!shader) return -1; switch (action) { case MENU_ACTION_LEFT: if (shader->passes) shader->passes--; driver.menu->need_refresh = true; break; case MENU_ACTION_RIGHT: if ((shader->passes < GFX_MAX_SHADERS)) shader->passes++; driver.menu->need_refresh = true; break; } if (driver.menu->need_refresh) video_shader_resolve_parameters(NULL, driver.menu->shader); #endif return 0; } static int action_ok_video_resolution(const char *path, const char *label, unsigned type, size_t idx) { #ifdef GEKKO if (driver.video_data) gx_set_video_mode(driver.video_data, menu_gx_resolutions [menu_current_gx_resolution][0], menu_gx_resolutions[menu_current_gx_resolution][1]); #elif defined(__CELLOS_LV2__) if (g_extern.console.screen.resolutions.list[ g_extern.console.screen.resolutions.current.idx] == CELL_VIDEO_OUT_RESOLUTION_576) { if (g_extern.console.screen.pal_enable) g_extern.console.screen.pal60_enable = true; } else { g_extern.console.screen.pal_enable = false; g_extern.console.screen.pal60_enable = false; } rarch_main_command(RARCH_CMD_REINIT); #endif return 0; } static int action_toggle_video_resolution(unsigned type, const char *label, unsigned action) { #ifdef GEKKO switch (action) { case MENU_ACTION_LEFT: if (menu_current_gx_resolution > 0) menu_current_gx_resolution--; break; case MENU_ACTION_RIGHT: if (menu_current_gx_resolution < GX_RESOLUTIONS_LAST - 1) { #ifdef HW_RVL if ((menu_current_gx_resolution + 1) > GX_RESOLUTIONS_640_480) if (CONF_GetVideo() != CONF_VIDEO_PAL) return 0; #endif menu_current_gx_resolution++; } break; } #elif defined(__CELLOS_LV2__) switch (action) { case MENU_ACTION_LEFT: if (g_extern.console.screen.resolutions.current.idx) { g_extern.console.screen.resolutions.current.idx--; g_extern.console.screen.resolutions.current.id = g_extern.console.screen.resolutions.list [g_extern.console.screen.resolutions.current.idx]; } break; case MENU_ACTION_RIGHT: if (g_extern.console.screen.resolutions.current.idx + 1 < g_extern.console.screen.resolutions.count) { g_extern.console.screen.resolutions.current.idx++; g_extern.console.screen.resolutions.current.id = g_extern.console.screen.resolutions.list [g_extern.console.screen.resolutions.current.idx]; } break; } #endif return 0; } static int action_start_performance_counters_frontend(unsigned type, const char *label, unsigned action) { struct retro_perf_counter **counters = (struct retro_perf_counter**) perf_counters_rarch; unsigned offset = type - MENU_SETTINGS_PERF_COUNTERS_BEGIN; (void)label; if (counters[offset]) { counters[offset]->total = 0; counters[offset]->call_cnt = 0; } return 0; } static int action_start_core_setting(unsigned type, const char *label, unsigned action) { unsigned idx = type - MENU_SETTINGS_CORE_OPTION_START; (void)label; core_option_set_default(g_extern.system.core_options, idx); return 0; } static int core_setting_toggle(unsigned type, const char *label, unsigned action) { unsigned idx = type - MENU_SETTINGS_CORE_OPTION_START; (void)label; switch (action) { case MENU_ACTION_LEFT: core_option_prev(g_extern.system.core_options, idx); break; case MENU_ACTION_RIGHT: core_option_next(g_extern.system.core_options, idx); break; } return 0; } static int disk_options_disk_idx_toggle(unsigned type, const char *label, unsigned action) { switch (action) { case MENU_ACTION_LEFT: rarch_main_command(RARCH_CMD_DISK_PREV); break; case MENU_ACTION_RIGHT: rarch_main_command(RARCH_CMD_DISK_NEXT); break; } return 0; } static int deferred_push_core_list_deferred(void *data, void *userdata, const char *path, const char *label, unsigned type) { unsigned i; size_t list_size = 0; const core_info_t *info = NULL; file_list_t *list = (file_list_t*)data; file_list_t *menu_list = (file_list_t*)userdata; if (!list || !menu_list) return -1; menu_list_clear(list); core_info_list_get_supported_cores(g_extern.core_info, driver.menu->deferred_path, &info, &list_size); for (i = 0; i < list_size; i++) { menu_list_push(list, info[i].path, "", MENU_FILE_CORE, 0); menu_list_set_alt_at_offset(list, i, info[i].display_name); } menu_list_sort_on_alt(list); driver.menu->scroll_indices_size = 0; menu_entries_build_scroll_indices(list); menu_entries_refresh(list); if (driver.menu_ctx && driver.menu_ctx->populate_entries) driver.menu_ctx->populate_entries(driver.menu, path, label, type); return 0; } static int deferred_push_database_manager_list_deferred(void *data, void *userdata, const char *path, const char *label, unsigned type) { unsigned i; #ifdef HAVE_LIBRETRODB int rv = 1; libretrodb_t db; libretrodb_cursor_t cur; struct rmsgpack_dom_value item; const core_info_t *info = NULL; #endif file_list_t *list = (file_list_t*)data; file_list_t *menu_list = (file_list_t*)userdata; if (!list || !menu_list) return -1; (void)info; (void)rv; menu_list_clear(list); #ifdef HAVE_LIBRETRODB if ((rv = libretrodb_open(path, &db)) != 0) return -1; if ((rv = libretrodb_cursor_open(&db, &cur, NULL)) != 0) return -1; while (libretrodb_cursor_read_item(&cur, &item) == 0) { if (item.type != RDT_MAP) continue; for (i = 0; i < item.map.len; i++) { struct rmsgpack_dom_value *key = &item.map.items[i].key; struct rmsgpack_dom_value *val = &item.map.items[i].value; if (!strcmp(key->string.buff, "description")) { menu_list_push(list, val->string.buff, "", MENU_FILE_RDB_ENTRY, 0); break; } } } libretrodb_cursor_close(&cur); libretrodb_close(&db); #endif menu_list_sort_on_alt(list); driver.menu->scroll_indices_size = 0; menu_entries_build_scroll_indices(list); menu_entries_refresh(list); if (driver.menu_ctx && driver.menu_ctx->populate_entries) driver.menu_ctx->populate_entries(driver.menu, path, label, type); return 0; } static int deferred_push_core_information(void *data, void *userdata, const char *path, const char *label, unsigned type) { unsigned i; core_info_t *info = NULL; file_list_t *list = (file_list_t*)data; file_list_t *menu_list = (file_list_t*)userdata; if (!list || !menu_list) return -1; info = (core_info_t*)g_extern.core_info_current; menu_list_clear(list); if (info->data) { char tmp[PATH_MAX_LENGTH]; snprintf(tmp, sizeof(tmp), "Core name: %s", info->core_name ? info->core_name : ""); menu_list_push(list, tmp, "", MENU_SETTINGS_CORE_INFO_NONE, 0); snprintf(tmp, sizeof(tmp), "Core label: %s", info->display_name ? info->display_name : ""); menu_list_push(list, tmp, "", MENU_SETTINGS_CORE_INFO_NONE, 0); if (info->systemname) { snprintf(tmp, sizeof(tmp), "System name: %s", info->systemname); menu_list_push(list, tmp, "", MENU_SETTINGS_CORE_INFO_NONE, 0); } if (info->system_manufacturer) { snprintf(tmp, sizeof(tmp), "System manufacturer: %s", info->system_manufacturer); menu_list_push(list, tmp, "", MENU_SETTINGS_CORE_INFO_NONE, 0); } if (info->categories_list) { strlcpy(tmp, "Categories: ", sizeof(tmp)); string_list_join_concat(tmp, sizeof(tmp), info->categories_list, ", "); menu_list_push(list, tmp, "", MENU_SETTINGS_CORE_INFO_NONE, 0); } if (info->authors_list) { strlcpy(tmp, "Authors: ", sizeof(tmp)); string_list_join_concat(tmp, sizeof(tmp), info->authors_list, ", "); menu_list_push(list, tmp, "", MENU_SETTINGS_CORE_INFO_NONE, 0); } if (info->permissions_list) { strlcpy(tmp, "Permissions: ", sizeof(tmp)); string_list_join_concat(tmp, sizeof(tmp), info->permissions_list, ", "); menu_list_push(list, tmp, "", MENU_SETTINGS_CORE_INFO_NONE, 0); } if (info->licenses_list) { strlcpy(tmp, "License(s): ", sizeof(tmp)); string_list_join_concat(tmp, sizeof(tmp), info->licenses_list, ", "); menu_list_push(list, tmp, "", MENU_SETTINGS_CORE_INFO_NONE, 0); } if (info->supported_extensions_list) { strlcpy(tmp, "Supported extensions: ", sizeof(tmp)); string_list_join_concat(tmp, sizeof(tmp), info->supported_extensions_list, ", "); menu_list_push(list, tmp, "", MENU_SETTINGS_CORE_INFO_NONE, 0); } if (info->firmware_count > 0) { core_info_list_update_missing_firmware( g_extern.core_info, info->path, g_settings.system_directory); menu_list_push(list, "Firmware: ", "", MENU_SETTINGS_CORE_INFO_NONE, 0); for (i = 0; i < info->firmware_count; i++) { if (info->firmware[i].desc) { snprintf(tmp, sizeof(tmp), " name: %s", info->firmware[i].desc ? info->firmware[i].desc : ""); menu_list_push(list, tmp, "", MENU_SETTINGS_CORE_INFO_NONE, 0); snprintf(tmp, sizeof(tmp), " status: %s, %s", info->firmware[i].missing ? "missing" : "present", info->firmware[i].optional ? "optional" : "required"); menu_list_push(list, tmp, "", MENU_SETTINGS_CORE_INFO_NONE, 0); } } } if (info->notes) { snprintf(tmp, sizeof(tmp), "Core notes: "); menu_list_push(list, tmp, "", MENU_SETTINGS_CORE_INFO_NONE, 0); for (i = 0; i < info->note_list->size; i++) { snprintf(tmp, sizeof(tmp), " %s", info->note_list->elems[i].data); menu_list_push(list, tmp, "", MENU_SETTINGS_CORE_INFO_NONE, 0); } } } else menu_list_push(list, "No information available.", "", MENU_SETTINGS_CORE_OPTION_NONE, 0); if (driver.menu_ctx && driver.menu_ctx->populate_entries) driver.menu_ctx->populate_entries(driver.menu, path, label, type); return 0; } static int deferred_push_performance_counters(void *data, void *userdata, const char *path, const char *label, unsigned type) { file_list_t *list = (file_list_t*)data; file_list_t *menu_list = (file_list_t*)userdata; if (!list || !menu_list) return -1; menu_list_clear(list); menu_list_push(list, "Frontend Counters", "frontend_counters", MENU_SETTING_ACTION, 0); menu_list_push(list, "Core Counters", "core_counters", MENU_SETTING_ACTION, 0); if (driver.menu_ctx && driver.menu_ctx->populate_entries) driver.menu_ctx->populate_entries(driver.menu, path, label, type); return 0; } static inline struct video_shader *shader_manager_get_current_shader( menu_handle_t *menu, const char *label, unsigned type) { if (!strcmp(label, "video_shader_preset_parameters")) return menu->shader; else if (!strcmp(label, "video_shader_parameters") && driver.video_poke && driver.video_data && driver.video_poke->get_current_shader) return driver.video_poke->get_current_shader(driver.video_data); return NULL; } static int deferred_push_video_shader_parameters_common(void *data, void *userdata, const char *path, const char *label, unsigned type, struct video_shader *shader, unsigned base_parameter) { unsigned i; file_list_t *list = (file_list_t*)data; file_list_t *menu_list = (file_list_t*)userdata; if (!list || !menu_list) return -1; menu_list_clear(list); for (i = 0; i < shader->num_parameters; i++) { menu_list_push(list, shader->parameters[i].desc, label, base_parameter + i, 0); } if (driver.menu_ctx && driver.menu_ctx->populate_entries) driver.menu_ctx->populate_entries(driver.menu, path, label, type); return 0; } static int deferred_push_video_shader_preset_parameters(void *data, void *userdata, const char *path, const char *label, unsigned type) { if (!driver.menu->shader) return 0; return deferred_push_video_shader_parameters_common(data, userdata, path, label, type, driver.menu->shader, MENU_SETTINGS_SHADER_PRESET_PARAMETER_0); } static int deferred_push_video_shader_parameters(void *data, void *userdata, const char *path, const char *label, unsigned type) { struct video_shader *shader = NULL; if (driver.video_poke && driver.video_data && driver.video_poke->get_current_shader) shader = driver.video_poke->get_current_shader(driver.video_data); if (!shader) return 0; return deferred_push_video_shader_parameters_common(data, userdata, path, label, type, shader, MENU_SETTINGS_SHADER_PARAMETER_0); } static int deferred_push_settings(void *data, void *userdata, const char *path, const char *label, unsigned type) { file_list_t *list = (file_list_t*)data; file_list_t *menu_list = (file_list_t*)userdata; if (!list || !menu_list) return -1; settings_list_free(driver.menu->list_settings); driver.menu->list_settings = (rarch_setting_t *)setting_data_new(SL_FLAG_ALL_SETTINGS); rarch_setting_t *setting = (rarch_setting_t*)setting_data_find_setting(driver.menu->list_settings, "Driver Options"); menu_list_clear(list); for (; setting->type != ST_NONE; setting++) { if (setting->type == ST_GROUP) menu_list_push(list, setting->short_description, setting->name, menu_entries_setting_set_flags(setting), 0); } if (driver.menu_ctx && driver.menu_ctx->populate_entries) driver.menu_ctx->populate_entries(driver.menu, path, label, type); return 0; } static int deferred_push_category(void *data, void *userdata, const char *path, const char *label, unsigned type) { file_list_t *list = (file_list_t*)data; file_list_t *menu_list = (file_list_t*)userdata; if (!list || !menu_list) return -1; settings_list_free(driver.menu->list_settings); driver.menu->list_settings = (rarch_setting_t *)setting_data_new(SL_FLAG_ALL_SETTINGS); rarch_setting_t *setting = (rarch_setting_t*)setting_data_find_setting( driver.menu->list_settings, label); menu_list_clear(list); if (!strcmp(label, "Video Options")) { #if defined(GEKKO) || defined(__CELLOS_LV2__) menu_list_push(list, "Screen Resolution", "", MENU_SETTINGS_VIDEO_RESOLUTION, 0); #endif menu_list_push(list, "Custom Ratio", "", MENU_SETTINGS_CUSTOM_VIEWPORT, 0); } for (; setting->type != ST_END_GROUP; setting++) { if ( setting->type == ST_GROUP || setting->type == ST_SUB_GROUP || setting->type == ST_END_SUB_GROUP ) continue; menu_list_push(list, setting->short_description, setting->name, menu_entries_setting_set_flags(setting), 0); } if (driver.menu_ctx && driver.menu_ctx->populate_entries) driver.menu_ctx->populate_entries(driver.menu, path, label, type); return 0; } static int deferred_push_shader_options(void *data, void *userdata, const char *path, const char *label, unsigned type) { unsigned i; struct video_shader *shader = NULL; file_list_t *list = (file_list_t*)data; file_list_t *menu_list = (file_list_t*)userdata; if (!list || !menu_list) return -1; shader = driver.menu->shader; if (!shader) return -1; menu_list_clear(list); menu_list_push(list, "Apply Shader Changes", "shader_apply_changes", MENU_SETTING_ACTION, 0); menu_list_push(list, "Default Filter", "video_shader_default_filter", 0, 0); menu_list_push(list, "Load Shader Preset", "video_shader_preset", MENU_SETTING_ACTION, 0); menu_list_push(list, "Shader Preset Save As", "video_shader_preset_save_as", MENU_SETTING_ACTION, 0); menu_list_push(list, "Parameters (Current)", "video_shader_parameters", MENU_SETTING_ACTION, 0); menu_list_push(list, "Parameters (Menu)", "video_shader_preset_parameters", MENU_SETTING_ACTION, 0); menu_list_push(list, "Shader Passes", "video_shader_num_passes", 0, 0); for (i = 0; i < shader->passes; i++) { char buf[64]; snprintf(buf, sizeof(buf), "Shader #%u", i); menu_list_push(list, buf, "video_shader_pass", MENU_SETTINGS_SHADER_PASS_0 + i, 0); snprintf(buf, sizeof(buf), "Shader #%u Filter", i); menu_list_push(list, buf, "video_shader_filter_pass", MENU_SETTINGS_SHADER_PASS_FILTER_0 + i, 0); snprintf(buf, sizeof(buf), "Shader #%u Scale", i); menu_list_push(list, buf, "video_shader_scale_pass", MENU_SETTINGS_SHADER_PASS_SCALE_0 + i, 0); } if (driver.menu_ctx && driver.menu_ctx->populate_entries) driver.menu_ctx->populate_entries(driver.menu, path, label, type); return 0; } static void push_perfcounter(menu_handle_t *menu, file_list_t *list, const struct retro_perf_counter **counters, unsigned num, unsigned id) { unsigned i; if (!counters || num == 0) return; for (i = 0; i < num; i++) if (counters[i] && counters[i]->ident) menu_list_push(list, counters[i]->ident, "", id + i, 0); } static int deferred_push_core_counters(void *data, void *userdata, const char *path, const char *label, unsigned type) { file_list_t *list = (file_list_t*)data; file_list_t *menu_list = (file_list_t*)userdata; if (!list || !menu_list) return -1; menu_list_clear(list); push_perfcounter(driver.menu, list, perf_counters_libretro, perf_ptr_libretro, MENU_SETTINGS_LIBRETRO_PERF_COUNTERS_BEGIN); if (driver.menu_ctx && driver.menu_ctx->populate_entries) driver.menu_ctx->populate_entries(driver.menu, path, label, type); return 0; } static int deferred_push_frontend_counters(void *data, void *userdata, const char *path, const char *label, unsigned type) { file_list_t *list = (file_list_t*)data; if (!list) return -1; menu_list_clear(list); push_perfcounter(driver.menu, list, perf_counters_rarch, perf_ptr_rarch, MENU_SETTINGS_PERF_COUNTERS_BEGIN); if (driver.menu_ctx && driver.menu_ctx->populate_entries) driver.menu_ctx->populate_entries(driver.menu, path, label, type); return 0; } static int deferred_push_core_cheat_options(void *data, void *userdata, const char *path, const char *label, unsigned type) { unsigned i; file_list_t *list = (file_list_t*)data; cheat_manager_t *cheat = g_extern.cheat; (void)userdata; (void)type; if (!list) return -1; if (!cheat) { g_extern.cheat = cheat_manager_new(0); if (!g_extern.cheat) return -1; cheat = g_extern.cheat; } menu_list_clear(list); menu_list_push(list, "Cheat File Load", "cheat_file_load", MENU_SETTING_ACTION, 0); menu_list_push(list, "Cheat File Save As", "cheat_file_save_as", MENU_SETTING_ACTION, 0); menu_list_push(list, "Cheat Passes", "cheat_num_passes", 0, 0); menu_list_push(list, "Apply Cheat Changes", "cheat_apply_changes", MENU_SETTING_ACTION, 0); for (i = 0; i < cheat->size; i++) { char cheat_label[64]; snprintf(cheat_label, sizeof(cheat_label), "Cheat #%d: ", i); if (cheat->cheats[i].desc) strlcat(cheat_label, cheat->cheats[i].desc, sizeof(cheat_label)); menu_list_push(list, cheat_label, "", MENU_SETTINGS_CHEAT_BEGIN + i, 0); } if (driver.menu_ctx && driver.menu_ctx->populate_entries) driver.menu_ctx->populate_entries(driver.menu, path, label, type); return 0; } static int deferred_push_core_input_remapping_options(void *data, void *userdata, const char *path, const char *label, unsigned type) { unsigned p, retro_id; file_list_t *list = (file_list_t*)data; (void)userdata; (void)type; if (!list) return -1; menu_list_clear(list); menu_list_push(list, "Remap File Load", "remap_file_load", MENU_SETTING_ACTION, 0); menu_list_push(list, "Remap File Save As", "remap_file_save_as", MENU_SETTING_ACTION, 0); for (p = 0; p < g_settings.input.max_users; p++) { for (retro_id = 0; retro_id < RARCH_FIRST_CUSTOM_BIND; retro_id++) { char desc_label[64]; unsigned user = p + 1; const char *description = g_extern.system.input_desc_btn[p][retro_id]; if (!description) continue; snprintf(desc_label, sizeof(desc_label), "User %u %s : ", user, description); menu_list_push(list, desc_label, "", MENU_SETTINGS_INPUT_DESC_BEGIN + (p * RARCH_FIRST_CUSTOM_BIND) + retro_id, 0); } } if (driver.menu_ctx && driver.menu_ctx->populate_entries) driver.menu_ctx->populate_entries(driver.menu, path, label, type); return 0; } static int deferred_push_core_options(void *data, void *userdata, const char *path, const char *label, unsigned type) { unsigned i; file_list_t *list = (file_list_t*)data; (void)userdata; if (!list) return -1; menu_list_clear(list); if (g_extern.system.core_options) { size_t opts = core_option_size(g_extern.system.core_options); for (i = 0; i < opts; i++) menu_list_push(list, core_option_get_desc(g_extern.system.core_options, i), "", MENU_SETTINGS_CORE_OPTION_START + i, 0); } else menu_list_push(list, "No options available.", "", MENU_SETTINGS_CORE_OPTION_NONE, 0); if (driver.menu_ctx && driver.menu_ctx->populate_entries) driver.menu_ctx->populate_entries(driver.menu, path, label, type); return 0; } static int deferred_push_disk_options(void *data, void *userdata, const char *path, const char *label, unsigned type) { file_list_t *list = (file_list_t*)data; (void)userdata; if (!list) return -1; menu_list_clear(list); menu_list_push(list, "Disk Index", "disk_idx", MENU_SETTINGS_CORE_DISK_OPTIONS_DISK_INDEX, 0); menu_list_push(list, "Disk Cycle Tray Status", "disk_cycle_tray_status", MENU_SETTINGS_CORE_DISK_OPTIONS_DISK_CYCLE_TRAY_STATUS, 0); menu_list_push(list, "Disk Image Append", "disk_image_append", MENU_SETTINGS_CORE_DISK_OPTIONS_DISK_IMAGE_APPEND, 0); if (driver.menu_ctx && driver.menu_ctx->populate_entries) driver.menu_ctx->populate_entries(driver.menu, path, label, type); return 0; } #ifdef HAVE_NETPLAY static void print_buf_lines(file_list_t *list, char *buf, int buf_size, unsigned type) { int i; char c, *line_start = buf; for (i = 0; i < buf_size; i++) { if (*(buf + i) == '\n') { /* Found a line ending, print the line and compute new line_start */ size_t ln; /* Save the next char */ c = *(buf + i + 1); /* replace with \0 */ *(buf + i + 1) = '\0'; /* We need to strip the newline. */ ln = strlen(line_start) - 1; if (line_start[ln] == '\n') line_start[ln] = '\0'; menu_list_push(list, line_start, "", type, 0); /* Restore the saved char */ *(buf + i + 1) = c; line_start = buf + i + 1; } else if (*(buf + i) == '\0') { /* The end of the buffer, print the last bit */ break; } } /* If the buffer was completely full, and didn't end with a newline, just * ignore the partial last line. */ } #endif /* HACK - we have to find some way to pass state inbetween * function pointer callback functions that don't necessarily * call each other. */ static void *core_manager_list_data; static char core_manager_list_path[PATH_MAX_LENGTH]; static char core_manager_list_label[PATH_MAX_LENGTH]; static unsigned core_manager_list_type; static int cb_core_manager_list(void *data_, size_t len) { char *data = (char*)data_; file_list_t *list = NULL; if (!data || !core_manager_list_data) return -1; list = (file_list_t*)core_manager_list_data; if (!list) return -1; menu_list_clear(list); rarch_main_command(RARCH_CMD_NETWORK_INIT); print_buf_lines(list, data, len, MENU_FILE_DOWNLOAD_CORE); driver.menu->scroll_indices_size = 0; menu_entries_build_scroll_indices(list); menu_entries_refresh(list); if (driver.menu_ctx && driver.menu_ctx->populate_entries) driver.menu_ctx->populate_entries(driver.menu, core_manager_list_path, core_manager_list_label, core_manager_list_type); return 0; } static int deferred_push_core_manager_list(void *data, void *userdata, const char *path, const char *label, unsigned type) { #ifdef HAVE_NETPLAY char url_path[PATH_MAX_LENGTH]; #endif file_list_t *list = (file_list_t*)data; core_manager_list_data = data; strlcpy(core_manager_list_path, path, sizeof(core_manager_list_path)); strlcpy(core_manager_list_label, label, sizeof(core_manager_list_label)); core_manager_list_type = type; menu_list_clear(list); if (g_settings.network.buildbot_url[0] == '\0') { #ifdef HAVE_NETPLAY menu_list_push(list, "Buildbot URL not configured.", "", 0, 0); #else menu_list_push(list, "Network not available.", "", 0, 0); #endif driver.menu->scroll_indices_size = 0; menu_entries_build_scroll_indices(list); menu_entries_refresh(list); if (driver.menu_ctx && driver.menu_ctx->populate_entries) driver.menu_ctx->populate_entries(driver.menu, path, label, type); return 0; } #ifdef HAVE_NETPLAY fill_pathname_join(url_path, g_settings.network.buildbot_url, ".index", sizeof(url_path)); msg_queue_clear(g_extern.http_msg_queue); msg_queue_push(g_extern.http_msg_queue, url_path, 0, 180); net_http_set_pending_cb(cb_core_manager_list); #endif return 0; } static int deferred_push_history_list(void *data, void *userdata, const char *path, const char *label, unsigned type) { unsigned i; size_t list_size = 0; file_list_t *list = (file_list_t*)data; (void)userdata; if (!list || !driver.menu) return -1; menu_list_clear(list); list_size = content_playlist_size(g_defaults.history); for (i = 0; i < list_size; i++) { char fill_buf[PATH_MAX_LENGTH]; const char *core_name = NULL; content_playlist_get_index(g_defaults.history, i, &path, NULL, &core_name); strlcpy(fill_buf, core_name, sizeof(fill_buf)); if (path) { char path_short[PATH_MAX_LENGTH]; fill_short_pathname_representation(path_short,path,sizeof(path_short)); snprintf(fill_buf,sizeof(fill_buf),"%s (%s)", path_short,core_name); } menu_list_push(list, fill_buf, "", MENU_FILE_PLAYLIST_ENTRY, 0); } driver.menu->scroll_indices_size = 0; menu_entries_build_scroll_indices(list); menu_entries_refresh(list); if (driver.menu_ctx && driver.menu_ctx->populate_entries) driver.menu_ctx->populate_entries(driver.menu, path, label, type); return 0; } static int deferred_push_content_actions(void *data, void *userdata, const char *path, const char *label, unsigned type) { file_list_t *list = (file_list_t*)data; (void)userdata; if (!list || !driver.menu) return -1; menu_list_clear(list); menu_list_push(list, "Run", "", MENU_SETTING_ACTION_RUN, 0); if (g_extern.main_is_init && !g_extern.libretro_dummy) { menu_list_push(list, "Save State", "savestate", MENU_SETTING_ACTION_SAVESTATE, 0); menu_list_push(list, "Load State", "loadstate", MENU_SETTING_ACTION_LOADSTATE, 0); menu_list_push(list, "Take Screenshot", "take_screenshot", MENU_SETTING_ACTION_SCREENSHOT, 0); menu_list_push(list, "Reset", "restart_content", MENU_SETTING_ACTION_RESET, 0); } driver.menu->scroll_indices_size = 0; menu_entries_build_scroll_indices(list); menu_entries_refresh(list); if (driver.menu_ctx && driver.menu_ctx->populate_entries) driver.menu_ctx->populate_entries(driver.menu, path, label, type); return 0; } static int deferred_push_content_list(void *data, void *userdata, const char *path, const char *label, unsigned type) { file_list_t *list = (file_list_t*)data; (void)userdata; if (!list || !driver.menu) return -1; menu_navigation_clear(driver.menu, true); if (driver.menu->cat_selection_ptr == 0) return menu_entries_push_main_menu_list(driver.menu, driver.menu->menu_list->selection_buf, "", "Main Menu", 0); else return menu_entries_push_horizontal_menu_list(driver.menu, driver.menu->menu_list->selection_buf, "", "Horizontal Menu", 0); } static int generic_deferred_push(void *data, void *userdata, const char *path, const char *label, unsigned type, unsigned push_type, const char *ext) { file_list_t *list = (file_list_t*)data; file_list_t *menu_list = (file_list_t*)userdata; if (!list || !menu_list) return -1; menu_entries_parse_list(list, menu_list, path, label, type, push_type, ext, NULL); return 0; } static int deferred_push_database_manager_list(void *data, void *userdata, const char *path, const char *label, unsigned type) { return generic_deferred_push(data, userdata, g_settings.content_database, label, type, MENU_FILE_RDB, "rdb"); } static int deferred_push_core_list(void *data, void *userdata, const char *path, const char *label, unsigned type) { return generic_deferred_push(data, userdata, path, label, type, MENU_FILE_PLAIN, EXT_EXECUTABLES); } static int deferred_push_configurations(void *data, void *userdata, const char *path, const char *label, unsigned type) { return generic_deferred_push(data, userdata, path, label, type, MENU_FILE_CONFIG, "cfg"); } static int deferred_push_video_shader_preset(void *data, void *userdata, const char *path, const char *label, unsigned type) { return generic_deferred_push(data, userdata, path, label, type, MENU_FILE_SHADER_PRESET, "cgp|glslp"); } static int deferred_push_video_shader_pass(void *data, void *userdata, const char *path, const char *label, unsigned type) { return generic_deferred_push(data, userdata, path, label, type, MENU_FILE_SHADER, "cg|glsl"); } static int deferred_push_video_filter(void *data, void *userdata, const char *path, const char *label, unsigned type) { return generic_deferred_push(data, userdata, path, label, type, MENU_FILE_VIDEOFILTER, "filt"); } static int deferred_push_images(void *data, void *userdata, const char *path, const char *label, unsigned type) { return generic_deferred_push(data, userdata, path, label, type, MENU_FILE_IMAGE, "png"); } static int deferred_push_audio_dsp_plugin(void *data, void *userdata, const char *path, const char *label, unsigned type) { return generic_deferred_push(data, userdata, path, label, type, MENU_FILE_AUDIOFILTER, "dsp"); } static int deferred_push_cheat_file_load(void *data, void *userdata, const char *path, const char *label, unsigned type) { return generic_deferred_push(data, userdata, path, label, type, MENU_FILE_CHEAT, "cht"); } static int deferred_push_remap_file_load(void *data, void *userdata, const char *path, const char *label, unsigned type) { return generic_deferred_push(data, userdata, path, label, type, MENU_FILE_REMAP, "rmp"); } static int deferred_push_input_overlay(void *data, void *userdata, const char *path, const char *label, unsigned type) { return generic_deferred_push(data, userdata, path, label, type, MENU_FILE_OVERLAY, "cfg"); } static int deferred_push_video_font_path(void *data, void *userdata, const char *path, const char *label, unsigned type) { return generic_deferred_push(data, userdata, path, label, type, MENU_FILE_FONT, "ttf"); } static int deferred_push_content_history_path(void *data, void *userdata, const char *path, const char *label, unsigned type) { return generic_deferred_push(data, userdata, path, label, type, MENU_FILE_PLAIN, "cfg"); } static int deferred_push_detect_core_list(void *data, void *userdata, const char *path, const char *label, unsigned type) { return generic_deferred_push(data, userdata, path, label, type, MENU_FILE_PLAIN, g_extern.core_info ? core_info_list_get_all_extensions( g_extern.core_info) : ""); } static int deferred_push_default(void *data, void *userdata, const char *path, const char *label, unsigned type) { char ext_buf[PATH_MAX_LENGTH]; const char *exts = NULL; file_list_t *list = (file_list_t*)data; file_list_t *menu_list = (file_list_t*)userdata; rarch_setting_t *setting = (rarch_setting_t*) setting_data_find_setting(driver.menu->list_settings, label); if (!list || !menu_list) return -1; if (setting && setting->browser_selection_type == ST_DIR) exts = ""; /* we ignore files anyway */ else if (g_extern.menu.info.valid_extensions) { exts = ext_buf; if (*g_extern.menu.info.valid_extensions) snprintf(ext_buf, sizeof(ext_buf), "%s", g_extern.menu.info.valid_extensions); else *ext_buf = '\0'; } else exts = g_extern.system.valid_extensions; menu_entries_parse_list(list, menu_list, path, label, type, MENU_FILE_PLAIN, exts, setting); return 0; } static int action_bind_up_or_down_generic(unsigned type, const char *label, unsigned action) { unsigned 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; } return 0; } /* Bind the OK callback function */ static int menu_entries_cbs_init_bind_ok_first(menu_file_list_cbs_t *cbs, const char *path, const char *label, unsigned type, size_t idx) { const char *menu_label = NULL; if (!driver.menu) return -1; menu_list_get_last_stack(driver.menu->menu_list, NULL, &menu_label, NULL); if (type >= MENU_SETTINGS_SHADER_PARAMETER_0 && type <= MENU_SETTINGS_SHADER_PARAMETER_LAST) cbs->action_ok = NULL; else if (type >= MENU_SETTINGS_SHADER_PRESET_PARAMETER_0 && type <= MENU_SETTINGS_SHADER_PRESET_PARAMETER_LAST) cbs->action_ok = NULL; else if (type >= MENU_SETTINGS_CHEAT_BEGIN && type <= MENU_SETTINGS_CHEAT_END) cbs->action_ok = action_ok_cheat; else if (!strcmp(label, "savestate")) cbs->action_ok = action_ok_save_state; else if (!strcmp(label, "loadstate")) cbs->action_ok = action_ok_load_state; else if (!strcmp(label, "resume_content")) cbs->action_ok = action_ok_resume_content; else if (!strcmp(label, "restart_content")) cbs->action_ok = action_ok_restart_content; else if (!strcmp(label, "take_screenshot")) cbs->action_ok = action_ok_screenshot; else switch (type) { case MENU_SETTINGS_VIDEO_RESOLUTION: cbs->action_ok = action_ok_video_resolution; break; case MENU_FILE_PLAYLIST_ENTRY: cbs->action_ok = action_ok_playlist_entry; break; case MENU_FILE_CONTENTLIST_ENTRY: cbs->action_ok = action_ok_push_generic_list; break; case MENU_FILE_CHEAT: cbs->action_ok = action_ok_cheat_file_load; break; case MENU_FILE_REMAP: cbs->action_ok = action_ok_remap_file_load; break; case MENU_FILE_SHADER_PRESET: cbs->action_ok = action_ok_shader_preset_load; break; case MENU_FILE_SHADER: cbs->action_ok = action_ok_shader_pass_load; break; case MENU_FILE_IMAGE: cbs->action_ok = action_ok_menu_wallpaper_load; break; case MENU_FILE_USE_DIRECTORY: cbs->action_ok = action_ok_path_use_directory; break; case MENU_FILE_CONFIG: cbs->action_ok = action_ok_config_load; break; case MENU_FILE_DIRECTORY: cbs->action_ok = action_ok_directory_push; break; case MENU_FILE_CARCHIVE: cbs->action_ok = action_ok_compressed_archive_push; break; case MENU_FILE_CORE: if (!strcmp(menu_label, "deferred_core_list")) cbs->action_ok = action_ok_core_load_deferred; else if (!strcmp(menu_label, "core_list")) cbs->action_ok = action_ok_core_load; else if (!strcmp(menu_label, "core_manager_list")) cbs->action_ok = action_ok_core_download; else return -1; break; case MENU_FILE_DOWNLOAD_CORE: cbs->action_ok = action_ok_core_manager_list; break; case MENU_FILE_DOWNLOAD_CORE_INFO: break; case MENU_FILE_RDB: if (!strcmp(menu_label, "deferred_database_manager_list")) cbs->action_ok = action_ok_database_manager_list_deferred; else if (!strcmp(menu_label, "database_manager_list")) cbs->action_ok = action_ok_database_manager_list; else return -1; break; case MENU_FILE_FONT: case MENU_FILE_OVERLAY: case MENU_FILE_AUDIOFILTER: case MENU_FILE_VIDEOFILTER: cbs->action_ok = action_ok_set_path; break; #ifdef HAVE_COMPRESSION case MENU_FILE_IN_CARCHIVE: #endif case MENU_FILE_PLAIN: if (!strcmp(menu_label, "detect_core_list")) cbs->action_ok = action_ok_file_load_with_detect_core; else if (!strcmp(menu_label, "disk_image_append")) cbs->action_ok = action_ok_disk_image_append; else cbs->action_ok = action_ok_file_load; break; case MENU_SETTINGS_CUSTOM_VIEWPORT: cbs->action_ok = action_ok_custom_viewport; break; case MENU_SETTINGS: case MENU_SETTING_GROUP: cbs->action_ok = action_ok_push_default; break; case MENU_SETTINGS_CORE_DISK_OPTIONS_DISK_CYCLE_TRAY_STATUS: cbs->action_ok = action_ok_disk_cycle_tray_status; break; default: return -1; } return 0; } static int action_start_lookup_setting(unsigned type, const char *label, unsigned action) { return menu_action_setting_set(type, label, MENU_ACTION_START); } static void menu_entries_cbs_init_bind_start(menu_file_list_cbs_t *cbs, const char *path, const char *label, unsigned type, size_t idx) { if (!cbs) return; cbs->action_start = action_start_lookup_setting; if (!strcmp(label, "video_shader_pass")) cbs->action_start = action_start_shader_pass; else if (!strcmp(label, "video_shader_scale_pass")) cbs->action_start = action_start_shader_scale_pass; else if (!strcmp(label, "video_shader_filter_pass")) cbs->action_start = action_start_shader_filter_pass; else if (!strcmp(label, "video_shader_num_passes")) cbs->action_start = action_start_shader_num_passes; else if (!strcmp(label, "cheat_num_passes")) cbs->action_start = action_start_cheat_num_passes; else if (type >= MENU_SETTINGS_SHADER_PARAMETER_0 && type <= MENU_SETTINGS_SHADER_PARAMETER_LAST) cbs->action_start = action_start_shader_action_parameter; else if (type >= MENU_SETTINGS_SHADER_PRESET_PARAMETER_0 && type <= MENU_SETTINGS_SHADER_PRESET_PARAMETER_LAST) cbs->action_start = action_start_shader_action_preset_parameter; else if (type >= MENU_SETTINGS_LIBRETRO_PERF_COUNTERS_BEGIN && type <= MENU_SETTINGS_LIBRETRO_PERF_COUNTERS_END) cbs->action_start = action_start_performance_counters_core; else if (type >= MENU_SETTINGS_INPUT_DESC_BEGIN && type <= MENU_SETTINGS_INPUT_DESC_END) cbs->action_start = action_start_input_desc; else if (type >= MENU_SETTINGS_PERF_COUNTERS_BEGIN && type <= MENU_SETTINGS_PERF_COUNTERS_END) cbs->action_start = action_start_performance_counters_frontend; else if ((type >= MENU_SETTINGS_CORE_OPTION_START)) cbs->action_start = action_start_core_setting; } static void menu_entries_cbs_init_bind_content_list_switch(menu_file_list_cbs_t *cbs, const char *path, const char *label, unsigned type, size_t idx) { if (!cbs) return; cbs->action_content_list_switch = deferred_push_content_list; } static void menu_entries_cbs_init_bind_cancel(menu_file_list_cbs_t *cbs, const char *path, const char *label, unsigned type, size_t idx) { if (!cbs) return; cbs->action_cancel = action_cancel_lookup_setting; /* TODO - add some stuff here. */ cbs->action_cancel = action_cancel_pop_default; } static void menu_entries_cbs_init_bind_ok(menu_file_list_cbs_t *cbs, const char *path, const char *label, unsigned type, size_t idx) { rarch_setting_t *setting = (rarch_setting_t*) setting_data_find_setting(driver.menu->list_settings, label); if (!cbs) return; cbs->action_ok = action_ok_lookup_setting; if (menu_entries_cbs_init_bind_ok_first(cbs, path, label, type, idx) == 0) return; else if (!strcmp(label, "quit_retroarch")) cbs->action_ok = action_ok_quit; else if (!strcmp(label, "save_new_config")) cbs->action_ok = action_ok_save_new_config; else if (!strcmp(label, "help")) cbs->action_ok = action_ok_help; else if (!strcmp(label, "video_shader_pass")) cbs->action_ok = action_ok_shader_pass; else if (!strcmp(label, "video_shader_preset")) cbs->action_ok = action_ok_shader_preset; else if (!strcmp(label, "cheat_file_load")) cbs->action_ok = action_ok_cheat_file; else if (!strcmp(label, "remap_file_load")) cbs->action_ok = action_ok_remap_file; else if (!strcmp(label, "video_shader_parameters") || !strcmp(label, "video_shader_preset_parameters") ) cbs->action_ok = action_ok_shader_parameters; else if ( !strcmp(label, "Shader Options") || !strcmp(label, "Input Options") || !strcmp(label, "core_options") || !strcmp(label, "core_cheat_options") || !strcmp(label, "core_input_remapping_options") || !strcmp(label, "core_information") || !strcmp(label, "disk_options") || !strcmp(label, "settings") || !strcmp(label, "performance_counters") || !strcmp(label, "frontend_counters") || !strcmp(label, "core_counters") ) cbs->action_ok = action_ok_push_default; else if ( !strcmp(label, "load_content") || !strcmp(label, "detect_core_list") ) cbs->action_ok = action_ok_push_content_list; else if (!strcmp(label, "history_list") || !strcmp(label, "core_manager_list") || !strcmp(label, "database_manager_list") || (setting && setting->browser_selection_type == ST_DIR) ) cbs->action_ok = action_ok_push_generic_list; else if (!strcmp(label, "shader_apply_changes")) cbs->action_ok = action_ok_shader_apply_changes; else if (!strcmp(label, "cheat_apply_changes")) cbs->action_ok = action_ok_cheat_apply_changes; else if (!strcmp(label, "video_shader_preset_save_as")) cbs->action_ok = action_ok_shader_preset_save_as; else if (!strcmp(label, "cheat_file_save_as")) cbs->action_ok = action_ok_cheat_file_save_as; else if (!strcmp(label, "remap_file_save_as")) cbs->action_ok = action_ok_remap_file_save_as; else if (!strcmp(label, "core_list")) cbs->action_ok = action_ok_core_list; else if (!strcmp(label, "disk_image_append")) cbs->action_ok = action_ok_disk_image_append_list; else if (!strcmp(label, "configurations")) cbs->action_ok = action_ok_configurations_list; } static void menu_entries_cbs_init_bind_up_or_down(menu_file_list_cbs_t *cbs, const char *path, const char *label, unsigned type, size_t idx) { if (!cbs) return; cbs->action_up_or_down = action_bind_up_or_down_generic; } static void menu_entries_cbs_init_bind_toggle(menu_file_list_cbs_t *cbs, const char *path, const char *label, unsigned type, size_t idx) { if (!cbs) return; cbs->action_toggle = menu_action_setting_set; switch (type) { case MENU_SETTINGS_CORE_DISK_OPTIONS_DISK_INDEX: cbs->action_toggle = disk_options_disk_idx_toggle; break; case MENU_FILE_PLAIN: case MENU_FILE_DIRECTORY: case MENU_FILE_CARCHIVE: case MENU_FILE_CORE: case MENU_FILE_RDB: case MENU_FILE_RDB_ENTRY: case MENU_FILE_SHADER: case MENU_FILE_IMAGE: case MENU_FILE_OVERLAY: case MENU_FILE_VIDEOFILTER: case MENU_FILE_AUDIOFILTER: case MENU_FILE_CONFIG: case MENU_FILE_USE_DIRECTORY: case MENU_FILE_PLAYLIST_ENTRY: cbs->action_toggle = action_toggle_scroll; break; case MENU_SETTING_ACTION: case MENU_FILE_CONTENTLIST_ENTRY: cbs->action_toggle = action_toggle_mainmenu; break; } if (type >= MENU_SETTINGS_SHADER_PARAMETER_0 && type <= MENU_SETTINGS_SHADER_PARAMETER_LAST) cbs->action_toggle = shader_action_parameter_toggle; else if (type >= MENU_SETTINGS_SHADER_PRESET_PARAMETER_0 && type <= MENU_SETTINGS_SHADER_PRESET_PARAMETER_LAST) cbs->action_toggle = shader_action_parameter_preset_toggle; else if (type >= MENU_SETTINGS_CHEAT_BEGIN && type <= MENU_SETTINGS_CHEAT_END) cbs->action_toggle = action_toggle_cheat; else if (type >= MENU_SETTINGS_INPUT_DESC_BEGIN && type <= MENU_SETTINGS_INPUT_DESC_END) cbs->action_toggle = action_toggle_input_desc; else if (!strcmp(label, "savestate") || !strcmp(label, "loadstate")) cbs->action_toggle = action_toggle_save_state; else if (!strcmp(label, "video_shader_scale_pass")) cbs->action_toggle = action_toggle_shader_scale_pass; else if (!strcmp(label, "video_shader_filter_pass")) cbs->action_toggle = action_toggle_shader_filter_pass; else if (!strcmp(label, "video_shader_default_filter")) cbs->action_toggle = action_toggle_shader_filter_default; else if (!strcmp(label, "video_shader_num_passes")) cbs->action_toggle = action_toggle_shader_num_passes; else if (!strcmp(label, "cheat_num_passes")) cbs->action_toggle = action_toggle_cheat_num_passes; else if (type == MENU_SETTINGS_VIDEO_RESOLUTION) cbs->action_toggle = action_toggle_video_resolution; else if ((type >= MENU_SETTINGS_CORE_OPTION_START)) cbs->action_toggle = core_setting_toggle; } static void menu_entries_cbs_init_bind_deferred_push(menu_file_list_cbs_t *cbs, const char *path, const char *label, unsigned type, size_t idx) { const char *menu_label = NULL; if (!cbs || !driver.menu) return; menu_list_get_last_stack(driver.menu->menu_list, NULL, &menu_label, NULL); cbs->action_deferred_push = deferred_push_default; if (!strcmp(label, "core_manager_list")) cbs->action_deferred_push = deferred_push_core_manager_list; else if (!strcmp(label, "history_list")) cbs->action_deferred_push = deferred_push_history_list; else if (!strcmp(label, "database_manager_list")) cbs->action_deferred_push = deferred_push_database_manager_list; else if (!strcmp(label, "cheat_file_load")) cbs->action_deferred_push = deferred_push_cheat_file_load; else if (!strcmp(label, "remap_file_load")) cbs->action_deferred_push = deferred_push_remap_file_load; else if (!strcmp(label, "content_actions")) cbs->action_deferred_push = deferred_push_content_actions; else if (!strcmp(label, "Shader Options")) cbs->action_deferred_push = deferred_push_shader_options; else if (type == MENU_SETTING_GROUP) cbs->action_deferred_push = deferred_push_category; else if (!strcmp(label, "deferred_core_list")) cbs->action_deferred_push = deferred_push_core_list_deferred; else if (!strcmp(label, "deferred_database_manager_list")) cbs->action_deferred_push = deferred_push_database_manager_list_deferred; else if (!strcmp(label, "core_information")) cbs->action_deferred_push = deferred_push_core_information; else if (!strcmp(label, "performance_counters")) cbs->action_deferred_push = deferred_push_performance_counters; else if (!strcmp(label, "core_counters")) cbs->action_deferred_push = deferred_push_core_counters; else if (!strcmp(label, "video_shader_preset_parameters")) cbs->action_deferred_push = deferred_push_video_shader_preset_parameters; else if (!strcmp(label, "video_shader_parameters")) cbs->action_deferred_push = deferred_push_video_shader_parameters; else if (!strcmp(label, "settings")) cbs->action_deferred_push = deferred_push_settings; else if (!strcmp(label, "frontend_counters")) cbs->action_deferred_push = deferred_push_frontend_counters; else if (!strcmp(label, "core_options")) cbs->action_deferred_push = deferred_push_core_options; else if (!strcmp(label, "core_cheat_options")) cbs->action_deferred_push = deferred_push_core_cheat_options; else if (!strcmp(label, "core_input_remapping_options")) cbs->action_deferred_push = deferred_push_core_input_remapping_options; else if (!strcmp(label, "disk_options")) cbs->action_deferred_push = deferred_push_disk_options; else if (!strcmp(label, "core_list")) cbs->action_deferred_push = deferred_push_core_list; else if (!strcmp(label, "configurations")) cbs->action_deferred_push = deferred_push_configurations; else if (!strcmp(label, "video_shader_preset")) cbs->action_deferred_push = deferred_push_video_shader_preset; else if (!strcmp(label, "video_shader_pass")) cbs->action_deferred_push = deferred_push_video_shader_pass; else if (!strcmp(label, "video_filter")) cbs->action_deferred_push = deferred_push_video_filter; else if (!strcmp(label, "menu_wallpaper")) cbs->action_deferred_push = deferred_push_images; else if (!strcmp(label, "audio_dsp_plugin")) cbs->action_deferred_push = deferred_push_audio_dsp_plugin; else if (!strcmp(label, "input_overlay")) cbs->action_deferred_push = deferred_push_input_overlay; else if (!strcmp(label, "video_font_path")) cbs->action_deferred_push = deferred_push_video_font_path; else if (!strcmp(label, "game_history_path")) cbs->action_deferred_push = deferred_push_content_history_path; else if (!strcmp(label, "detect_core_list")) cbs->action_deferred_push = deferred_push_detect_core_list; } void menu_entries_cbs_init(void *data, const char *path, const char *label, unsigned type, size_t idx) { menu_file_list_cbs_t *cbs = NULL; file_list_t *list = (file_list_t*)data; if (!list) return; cbs = (menu_file_list_cbs_t*)list->list[idx].actiondata; if (cbs) { menu_entries_cbs_init_bind_ok(cbs, path, label, type, idx); menu_entries_cbs_init_bind_cancel(cbs, path, label, type, idx); menu_entries_cbs_init_bind_start(cbs, path, label, type, idx); menu_entries_cbs_init_bind_content_list_switch(cbs, path, label, type, idx); menu_entries_cbs_init_bind_up_or_down(cbs, path, label, type, idx); menu_entries_cbs_init_bind_toggle(cbs, path, label, type, idx); menu_entries_cbs_init_bind_deferred_push(cbs, path, label, type, idx); } }