/* RetroArch - A frontend for libretro. * Copyright (C) 2019-2021 - Brian Weiss * * 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 "cheevos_locals.h" #include "../gfx/gfx_display.h" #ifdef HAVE_MENU #include "cheevos.h" #include "../deps/rcheevos/include/rc_runtime_types.h" #include "../file_path_special.h" #include "../menu/menu_driver.h" #include "../menu/menu_entries.h" #include #include enum rcheevos_menuitem_bucket { RCHEEVOS_MENUITEM_BUCKET_UNKNOWN = 0, RCHEEVOS_MENUITEM_BUCKET_LOCKED, RCHEEVOS_MENUITEM_BUCKET_UNLOCKED, RCHEEVOS_MENUITEM_BUCKET_UNSUPPORTED, RCHEEVOS_MENUITEM_BUCKET_UNOFFICIAL, RCHEEVOS_MENUITEM_BUCKET_RECENTLY_UNLOCKED, RCHEEVOS_MENUITEM_BUCKET_ACTIVE_CHALLENGE, RCHEEVOS_MENUITEM_BUCKET_ALMOST_THERE }; static void rcheevos_menu_update_bucket(rcheevos_racheevo_t* cheevo) { cheevo->menu_progress = 0; if (cheevo->active & RCHEEVOS_ACTIVE_UNSUPPORTED) { /* non-active unsupported achievement */ cheevo->menu_bucket = RCHEEVOS_MENUITEM_BUCKET_UNSUPPORTED; } else if (!(cheevo->active & RCHEEVOS_ACTIVE_HARDCORE)) { /* non-active unlocked in hardcore achievement */ cheevo->menu_bucket = RCHEEVOS_MENUITEM_BUCKET_UNLOCKED; } else { const rcheevos_locals_t* rcheevos_locals = get_rcheevos_locals(); rc_trigger_t* trigger; if (!rcheevos_locals->hardcore_active && !(cheevo->active & RCHEEVOS_ACTIVE_SOFTCORE)) { /* non-active unlocked in softcore achievement in softcore mode */ cheevo->menu_bucket = RCHEEVOS_MENUITEM_BUCKET_UNLOCKED; return; } /* active achievement */ if (cheevo->active & RCHEEVOS_ACTIVE_UNOFFICIAL) cheevo->menu_bucket = RCHEEVOS_MENUITEM_BUCKET_UNOFFICIAL; else cheevo->menu_bucket = RCHEEVOS_MENUITEM_BUCKET_LOCKED; trigger = rc_runtime_get_achievement(&rcheevos_locals->runtime, cheevo->id); if (trigger) { if (trigger->measured_value && trigger->measured_target) { const unsigned long clamped_value = (unsigned long) MIN(trigger->measured_value, trigger->measured_target); cheevo->menu_progress = (uint8_t)((clamped_value * 100) / trigger->measured_target); } if (trigger->state == RC_TRIGGER_STATE_PRIMED) cheevo->menu_bucket = RCHEEVOS_MENUITEM_BUCKET_ACTIVE_CHALLENGE; else if (cheevo->menu_progress >= 80) cheevo->menu_bucket = RCHEEVOS_MENUITEM_BUCKET_ALMOST_THERE; } } } static void rcheevos_menu_update_buckets(void) { const rcheevos_locals_t* rcheevos_locals = get_rcheevos_locals(); rcheevos_racheevo_t* cheevo = rcheevos_locals->game.achievements; rcheevos_racheevo_t* stop = cheevo + rcheevos_locals->game.achievement_count; while (cheevo < stop) { rcheevos_menu_update_bucket(cheevo); ++cheevo; } } bool rcheevos_menu_get_state(unsigned menu_offset, char *buffer, size_t len) { const rcheevos_locals_t* rcheevos_locals = get_rcheevos_locals(); if (menu_offset < rcheevos_locals->menuitem_count) { const rcheevos_menuitem_t* menuitem = &rcheevos_locals->menuitems[menu_offset]; const rcheevos_racheevo_t* cheevo = menuitem->cheevo; if (cheevo) { if (cheevo->menu_progress) { const int written = snprintf(buffer, len, "%s - ", msg_hash_to_str(menuitem->state_label_idx)); if (len - written > 0) rc_runtime_format_achievement_measured(&rcheevos_locals->runtime, cheevo->id, buffer + written, len - written); } else strlcpy(buffer, msg_hash_to_str(menuitem->state_label_idx), len); return true; } } if (buffer) buffer[0] = '\0'; return false; } bool rcheevos_menu_get_sublabel(unsigned menu_offset, char *buffer, size_t len) { const rcheevos_locals_t* rcheevos_locals = get_rcheevos_locals(); if (menu_offset < rcheevos_locals->menuitem_count) { const rcheevos_racheevo_t* cheevo = rcheevos_locals->menuitems[menu_offset].cheevo; if (cheevo && buffer) { strlcpy(buffer, cheevo->description, len); return true; } } if (buffer) buffer[0] = '\0'; return false; } void rcheevos_menu_reset_badges(void) { const rcheevos_locals_t* rcheevos_locals = get_rcheevos_locals(); rcheevos_racheevo_t* cheevo = rcheevos_locals->game.achievements; rcheevos_racheevo_t* stop = cheevo + rcheevos_locals->game.achievement_count; while (cheevo < stop) { if (cheevo->menu_badge_texture) video_driver_texture_unload(&cheevo->menu_badge_texture); ++cheevo; } } static rcheevos_menuitem_t* rcheevos_menu_allocate( rcheevos_locals_t* rcheevos_locals, rcheevos_racheevo_t* cheevo) { rcheevos_menuitem_t* menuitem; if (rcheevos_locals->menuitem_count == rcheevos_locals->menuitem_capacity) { if (rcheevos_locals->menuitems) { rcheevos_menuitem_t* new_menuitems; rcheevos_locals->menuitem_capacity += 32; new_menuitems = (rcheevos_menuitem_t*)realloc(rcheevos_locals->menuitems, rcheevos_locals->menuitem_capacity * sizeof(rcheevos_menuitem_t)); if (new_menuitems) rcheevos_locals->menuitems = new_menuitems; else { /* realloc failed */ CHEEVOS_ERR(RCHEEVOS_TAG " could not allocate space for %u menu items\n", rcheevos_locals->menuitem_capacity); rcheevos_locals->menuitem_capacity -= 32; return NULL; } } else { rcheevos_locals->menuitem_capacity = 64; rcheevos_locals->menuitems = (rcheevos_menuitem_t*) malloc(rcheevos_locals->menuitem_capacity * sizeof(rcheevos_menuitem_t)); if (!rcheevos_locals->menuitems) { /* malloc failed */ CHEEVOS_ERR(RCHEEVOS_TAG " could not allocate space for %u menu items\n", rcheevos_locals->menuitem_capacity); rcheevos_locals->menuitem_capacity = 0; return NULL; } } } menuitem = &rcheevos_locals->menuitems[rcheevos_locals->menuitem_count++]; menuitem->cheevo = cheevo; menuitem->state_label_idx = MSG_UNKNOWN; return menuitem; } static void rcheevos_menu_append_header(rcheevos_locals_t* rcheevos_locals, enum msg_hash_enums label) { rcheevos_menuitem_t* menuitem = rcheevos_menu_allocate(rcheevos_locals, NULL); if (menuitem) menuitem->state_label_idx = label; } static void rcheevos_menu_update_badge(rcheevos_racheevo_t* cheevo) { bool badge_grayscale = false; switch (cheevo->menu_bucket) { case RCHEEVOS_MENUITEM_BUCKET_LOCKED: case RCHEEVOS_MENUITEM_BUCKET_UNOFFICIAL: case RCHEEVOS_MENUITEM_BUCKET_UNSUPPORTED: case RCHEEVOS_MENUITEM_BUCKET_ALMOST_THERE: case RCHEEVOS_MENUITEM_BUCKET_ACTIVE_CHALLENGE: badge_grayscale = true; break; default: badge_grayscale = false; break; } if (!cheevo->menu_badge_texture || cheevo->menu_badge_grayscale != badge_grayscale) { uintptr_t new_badge_texture = rcheevos_get_badge_texture(cheevo->badge, badge_grayscale); if (new_badge_texture) { if (cheevo->menu_badge_texture) video_driver_texture_unload(&cheevo->menu_badge_texture); cheevo->menu_badge_texture = new_badge_texture; cheevo->menu_badge_grayscale = badge_grayscale; } /* menu_badge_grayscale is overloaded such that any value greater than 1 indicates * the server default image is being used */ else if (cheevo->menu_badge_grayscale < 2) { if (cheevo->menu_badge_texture) video_driver_texture_unload(&cheevo->menu_badge_texture); /* requested badge is not available, check for server default */ cheevo->menu_badge_texture = rcheevos_get_badge_texture("00000", false); if (cheevo->menu_badge_texture) cheevo->menu_badge_grayscale = 2; } } } static void rcheevos_menu_append_items(rcheevos_locals_t* rcheevos_locals, enum rcheevos_menuitem_bucket bucket) { rcheevos_racheevo_t* cheevo = rcheevos_locals->game.achievements; rcheevos_racheevo_t* stop = cheevo + rcheevos_locals->game.achievement_count; const unsigned first_index = rcheevos_locals->menuitem_count; while (cheevo < stop) { if (cheevo->menu_bucket == bucket) { rcheevos_menuitem_t* menuitem = rcheevos_menu_allocate(rcheevos_locals, cheevo); if (!menuitem) return; switch (cheevo->menu_bucket) { case RCHEEVOS_MENUITEM_BUCKET_UNSUPPORTED: menuitem->state_label_idx = MENU_ENUM_LABEL_VALUE_CHEEVOS_UNSUPPORTED_ENTRY; break; case RCHEEVOS_MENUITEM_BUCKET_RECENTLY_UNLOCKED: { /* insert the item such that the unlock times are descending */ unsigned entry_index = rcheevos_locals->menuitem_count - 1; while (entry_index > first_index) { rcheevos_menuitem_t* prev_menuitem = menuitem - 1; if (prev_menuitem->cheevo->unlock_time >= cheevo->unlock_time) break; memcpy(menuitem, prev_menuitem, sizeof(rcheevos_menuitem_t)); menuitem = prev_menuitem; --entry_index; } menuitem->cheevo = cheevo; } /* fallthrough to RCHEEVOS_MENUITEM_BUCKET_UNLOCKED */ case RCHEEVOS_MENUITEM_BUCKET_UNLOCKED: if (!(cheevo->active & RCHEEVOS_ACTIVE_HARDCORE)) menuitem->state_label_idx = MENU_ENUM_LABEL_VALUE_CHEEVOS_UNLOCKED_ENTRY_HARDCORE; else menuitem->state_label_idx = MENU_ENUM_LABEL_VALUE_CHEEVOS_UNLOCKED_ENTRY; break; case RCHEEVOS_MENUITEM_BUCKET_ALMOST_THERE: { /* insert the item such that the progresses are descending */ unsigned entry_index = rcheevos_locals->menuitem_count - 1; while (entry_index > first_index) { rcheevos_menuitem_t* prev_menuitem = menuitem - 1; if (prev_menuitem->cheevo->menu_progress >= cheevo->menu_progress) break; memcpy(menuitem, prev_menuitem, sizeof(rcheevos_menuitem_t)); menuitem = prev_menuitem; --entry_index; } menuitem->cheevo = cheevo; } /* fallthrough to default */ default: if (cheevo->active & RCHEEVOS_ACTIVE_UNOFFICIAL) menuitem->state_label_idx = MENU_ENUM_LABEL_VALUE_CHEEVOS_UNOFFICIAL_ENTRY; else if (!(cheevo->active & RCHEEVOS_ACTIVE_SOFTCORE)) menuitem->state_label_idx = MENU_ENUM_LABEL_VALUE_CHEEVOS_UNLOCKED_ENTRY; else menuitem->state_label_idx = MENU_ENUM_LABEL_VALUE_CHEEVOS_LOCKED_ENTRY; break; } if (cheevo->badge && cheevo->badge[0]) { #ifndef HAVE_GFX_WIDGETS const settings_t* settings = config_get_ptr(); if (settings && settings->bools.cheevos_badges_enable) #endif rcheevos_menu_update_badge(cheevo); } } ++cheevo; } } uintptr_t rcheevos_menu_get_badge_texture(unsigned menu_offset) { const rcheevos_locals_t* rcheevos_locals = get_rcheevos_locals(); if (menu_offset < rcheevos_locals->menuitem_count) { rcheevos_racheevo_t* cheevo = rcheevos_locals->menuitems[menu_offset].cheevo; if (cheevo) { /* if we're using the placeholder badge, check to see if the real badge * has become available (do this roughly once a second) */ if (cheevo->menu_badge_grayscale >= 2) { if (++cheevo->menu_badge_grayscale == 64) { cheevo->menu_badge_grayscale = 2; rcheevos_menu_update_badge(cheevo); } } return cheevo->menu_badge_texture; } } return 0; } void rcheevos_menu_populate_hardcore_pause_submenu(void* data) { const rcheevos_locals_t* rcheevos_locals = get_rcheevos_locals(); menu_displaylist_info_t* info = (menu_displaylist_info_t*)data; const settings_t* settings = config_get_ptr(); const bool cheevos_hardcore_mode_enable = settings->bools.cheevos_hardcore_mode_enable; if (cheevos_hardcore_mode_enable && rcheevos_locals->loaded) { if (rcheevos_locals->hardcore_active) { menu_entries_append_enum(info->list, msg_hash_to_str(MENU_ENUM_LABEL_VALUE_ACHIEVEMENT_PAUSE_CANCEL), msg_hash_to_str(MENU_ENUM_SUBLABEL_ACHIEVEMENT_PAUSE_CANCEL), MENU_ENUM_LABEL_ACHIEVEMENT_PAUSE_CANCEL, MENU_SETTING_ACTION_CLOSE, 0, 0); menu_entries_append_enum(info->list, msg_hash_to_str(MENU_ENUM_LABEL_VALUE_ACHIEVEMENT_PAUSE), msg_hash_to_str(MENU_ENUM_LABEL_ACHIEVEMENT_PAUSE), MENU_ENUM_LABEL_ACHIEVEMENT_PAUSE, MENU_SETTING_ACTION_PAUSE_ACHIEVEMENTS, 0, 0); } else { menu_entries_append_enum(info->list, msg_hash_to_str(MENU_ENUM_LABEL_VALUE_ACHIEVEMENT_RESUME_CANCEL), msg_hash_to_str(MENU_ENUM_SUBLABEL_ACHIEVEMENT_RESUME_CANCEL), MENU_ENUM_LABEL_ACHIEVEMENT_RESUME_CANCEL, MENU_SETTING_ACTION_CLOSE, 0, 0); menu_entries_append_enum(info->list, msg_hash_to_str(MENU_ENUM_LABEL_VALUE_ACHIEVEMENT_RESUME), msg_hash_to_str(MENU_ENUM_LABEL_ACHIEVEMENT_RESUME), MENU_ENUM_LABEL_ACHIEVEMENT_RESUME, MENU_SETTING_ACTION_RESUME_ACHIEVEMENTS, 0, 0); } } } void rcheevos_menu_populate(void* data) { menu_displaylist_info_t* info = (menu_displaylist_info_t*)data; rcheevos_locals_t* rcheevos_locals = get_rcheevos_locals(); const settings_t* settings = config_get_ptr(); unsigned num_locked = 0; unsigned num_unlocked = 0; unsigned num_recently_unlocked = 0; unsigned num_unsupported = 0; unsigned num_active_challenges = 0; unsigned num_almost_there = 0; if (rcheevos_locals->loaded) { const retro_time_t now = cpu_features_get_time_usec(); const retro_time_t recent_unlock_time = now - (10 * 60 * 1000000); /* 10 minutes ago */ rcheevos_racheevo_t* cheevo = NULL; rcheevos_racheevo_t* stop = NULL; /* first menu item is the Pause/Resume Hardcore option (unless hardcore is disabled) */ if (settings->bools.cheevos_enable && settings->bools.cheevos_hardcore_mode_enable) { if (rcheevos_locals->hardcore_active) menu_entries_append_enum(info->list, msg_hash_to_str(MENU_ENUM_LABEL_VALUE_ACHIEVEMENT_PAUSE), msg_hash_to_str(MENU_ENUM_LABEL_VALUE_ACHIEVEMENT_PAUSE_MENU), MENU_ENUM_LABEL_ACHIEVEMENT_PAUSE_MENU, MENU_SETTING_ACTION_PAUSE_ACHIEVEMENTS, 0, 0); else menu_entries_append_enum(info->list, msg_hash_to_str(MENU_ENUM_LABEL_VALUE_ACHIEVEMENT_RESUME), msg_hash_to_str(MENU_ENUM_LABEL_VALUE_ACHIEVEMENT_PAUSE_MENU), MENU_ENUM_LABEL_ACHIEVEMENT_PAUSE_MENU, MENU_SETTING_ACTION_RESUME_ACHIEVEMENTS, 0, 0); } /* update the bucket for each achievement */ rcheevos_menu_update_buckets(); /* count items in each bucket */ cheevo = rcheevos_locals->game.achievements; stop = cheevo + rcheevos_locals->game.achievement_count; while (cheevo < stop) { switch (cheevo->menu_bucket) { case RCHEEVOS_MENUITEM_BUCKET_UNLOCKED: if (cheevo->unlock_time && cheevo->unlock_time >= recent_unlock_time) { cheevo->menu_bucket = RCHEEVOS_MENUITEM_BUCKET_RECENTLY_UNLOCKED; ++num_recently_unlocked; } else ++num_unlocked; break; case RCHEEVOS_MENUITEM_BUCKET_LOCKED: case RCHEEVOS_MENUITEM_BUCKET_UNOFFICIAL: ++num_locked; break; case RCHEEVOS_MENUITEM_BUCKET_UNSUPPORTED: ++num_unsupported; break; case RCHEEVOS_MENUITEM_BUCKET_ACTIVE_CHALLENGE: ++num_active_challenges; break; case RCHEEVOS_MENUITEM_BUCKET_ALMOST_THERE: ++num_almost_there; break; } ++cheevo; } if (!rcheevos_locals->menuitems) { /* reserve space for all achievements and up to 6 headers before we need to realloc */ rcheevos_locals->menuitem_capacity = rcheevos_locals->game.achievement_count + 6; rcheevos_locals->menuitems = (rcheevos_menuitem_t*) malloc(rcheevos_locals->menuitem_capacity * sizeof(rcheevos_menuitem_t)); if (!rcheevos_locals->menuitems) rcheevos_locals->menuitem_capacity = 0; } } /* reset menu */ rcheevos_locals->menuitem_count = 0; /* active challenges */ if (num_active_challenges) { rcheevos_menu_append_header(rcheevos_locals, MENU_ENUM_LABEL_VALUE_CHEEVOS_ACTIVE_CHALLENGES_ENTRY); rcheevos_menu_append_items(rcheevos_locals, RCHEEVOS_MENUITEM_BUCKET_ACTIVE_CHALLENGE); } /* recently unlocked */ if (num_recently_unlocked) { rcheevos_menu_append_header(rcheevos_locals, MENU_ENUM_LABEL_VALUE_CHEEVOS_RECENTLY_UNLOCKED_ENTRY); rcheevos_menu_append_items(rcheevos_locals, RCHEEVOS_MENUITEM_BUCKET_RECENTLY_UNLOCKED); } /* almost there */ if (num_almost_there) { rcheevos_menu_append_header(rcheevos_locals, MENU_ENUM_LABEL_VALUE_CHEEVOS_ALMOST_THERE_ENTRY); rcheevos_menu_append_items(rcheevos_locals, RCHEEVOS_MENUITEM_BUCKET_ALMOST_THERE); } /* locked */ if (num_locked) { if (rcheevos_locals->menuitem_count > 0) rcheevos_menu_append_header(rcheevos_locals, MENU_ENUM_LABEL_VALUE_CHEEVOS_LOCKED_ENTRY); rcheevos_menu_append_items(rcheevos_locals, RCHEEVOS_MENUITEM_BUCKET_LOCKED); rcheevos_menu_append_items(rcheevos_locals, RCHEEVOS_MENUITEM_BUCKET_UNOFFICIAL); } /* unsupported */ if (num_unsupported) { if (rcheevos_locals->menuitem_count > 0) rcheevos_menu_append_header(rcheevos_locals, MENU_ENUM_LABEL_VALUE_CHEEVOS_UNSUPPORTED_ENTRY); rcheevos_menu_append_items(rcheevos_locals, RCHEEVOS_MENUITEM_BUCKET_UNSUPPORTED); } /* unlocked */ if (num_unlocked) { if (rcheevos_locals->menuitem_count > 0) rcheevos_menu_append_header(rcheevos_locals, MENU_ENUM_LABEL_VALUE_CHEEVOS_UNLOCKED_ENTRY); rcheevos_menu_append_items(rcheevos_locals, RCHEEVOS_MENUITEM_BUCKET_UNLOCKED); } if (rcheevos_locals->menuitem_count > 0) { char buffer[128]; unsigned idx = 0; /* convert to menu entries */ rcheevos_menuitem_t* menuitem = rcheevos_locals->menuitems; rcheevos_menuitem_t* stop = menuitem + rcheevos_locals->menuitem_count; do { if (menuitem->cheevo) menu_entries_append_enum(info->list, menuitem->cheevo->title, menuitem->cheevo->description, MENU_ENUM_LABEL_CHEEVOS_LOCKED_ENTRY, MENU_SETTINGS_CHEEVOS_START + idx, 0, 0); else { snprintf(buffer, sizeof(buffer), "----- %s -----", msg_hash_to_str(menuitem->state_label_idx)); menu_entries_append_enum(info->list, buffer, "", MENU_ENUM_LABEL_CHEEVOS_LOCKED_ENTRY, MENU_SETTINGS_CHEEVOS_START + idx, 0, 0); } ++idx; ++menuitem; } while (menuitem != stop); } else { /* no achievements found */ if (!rcheevos_locals->core_supports) menu_entries_append_enum(info->list, msg_hash_to_str(MENU_ENUM_LABEL_VALUE_CANNOT_ACTIVATE_ACHIEVEMENTS_WITH_THIS_CORE), msg_hash_to_str(MENU_ENUM_LABEL_CANNOT_ACTIVATE_ACHIEVEMENTS_WITH_THIS_CORE), MENU_ENUM_LABEL_CANNOT_ACTIVATE_ACHIEVEMENTS_WITH_THIS_CORE, FILE_TYPE_NONE, 0, 0); else if (rcheevos_locals->load_info.state == RCHEEVOS_LOAD_STATE_NETWORK_ERROR) menu_entries_append_enum(info->list, msg_hash_to_str(MENU_ENUM_LABEL_VALUE_NETWORK_ERROR), msg_hash_to_str(MENU_ENUM_LABEL_NETWORK_ERROR), MENU_ENUM_LABEL_NETWORK_ERROR, FILE_TYPE_NONE, 0, 0); else if (!rcheevos_locals->game.id) menu_entries_append_enum(info->list, msg_hash_to_str(MENU_ENUM_LABEL_VALUE_UNKNOWN_GAME), msg_hash_to_str(MENU_ENUM_LABEL_UNKNOWN_GAME), MENU_ENUM_LABEL_UNKNOWN_GAME, FILE_TYPE_NONE, 0, 0); else if (!rcheevos_locals->token[0]) menu_entries_append_enum(info->list, msg_hash_to_str(MENU_ENUM_LABEL_VALUE_NOT_LOGGED_IN), msg_hash_to_str(MENU_ENUM_LABEL_NOT_LOGGED_IN), MENU_ENUM_LABEL_NOT_LOGGED_IN, FILE_TYPE_NONE, 0, 0); else menu_entries_append_enum(info->list, msg_hash_to_str(MENU_ENUM_LABEL_VALUE_NO_ACHIEVEMENTS_TO_DISPLAY), msg_hash_to_str(MENU_ENUM_LABEL_NO_ACHIEVEMENTS_TO_DISPLAY), MENU_ENUM_LABEL_NO_ACHIEVEMENTS_TO_DISPLAY, FILE_TYPE_NONE, 0, 0); } } #endif /* HAVE_MENU */ uintptr_t rcheevos_get_badge_texture(const char *badge, bool locked) { char badge_file[24]; char fullpath[PATH_MAX_LENGTH]; uintptr_t tex = 0; if (!badge) return 0; /* OpenGL driver crashes if gfx_display_reset_textures_list is called on a background thread */ retro_assert(task_is_on_main_thread()); snprintf(badge_file, sizeof(badge_file), "%s%s%s", badge, locked ? "_lock" : "", FILE_PATH_PNG_EXTENSION); fill_pathname_application_special(fullpath, sizeof(fullpath), APPLICATION_SPECIAL_DIRECTORY_THUMBNAILS_CHEEVOS_BADGES); if (!gfx_display_reset_textures_list(badge_file, fullpath, &tex, TEXTURE_FILTER_MIPMAP_LINEAR, NULL, NULL)) return 0; return tex; }