mirror of
https://github.com/libretro/RetroArch.git
synced 2025-01-09 02:00:44 +00:00
688 lines
24 KiB
C
688 lines
24 KiB
C
/* 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 <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#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 <features/features_cpu.h>
|
|
#include <retro_assert.h>
|
|
|
|
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
|
|
};
|
|
|
|
/* if menu_badge_grayscale is set to a value other than 1 or 0, it's a counter for the number of
|
|
* frames since the last time we checked for the file. When the counter reaches this value, we'll
|
|
* check for the file again. */
|
|
#define MENU_BADGE_RETRY_RELOAD_FRAMES 64
|
|
|
|
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->menu_badge_texture = 0;
|
|
cheevo->menu_badge_grayscale = MENU_BADGE_RETRY_RELOAD_FRAMES;
|
|
}
|
|
++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 >= MENU_BADGE_RETRY_RELOAD_FRAMES)
|
|
{
|
|
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(info->list,
|
|
msg_hash_to_str(MENU_ENUM_LABEL_VALUE_ACHIEVEMENT_PAUSE_CANCEL),
|
|
msg_hash_to_str(MENU_ENUM_LABEL_ACHIEVEMENT_PAUSE_CANCEL),
|
|
MENU_ENUM_LABEL_ACHIEVEMENT_PAUSE_CANCEL,
|
|
MENU_SETTING_ACTION_CLOSE, 0, 0, NULL);
|
|
menu_entries_append(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, NULL);
|
|
}
|
|
else
|
|
{
|
|
menu_entries_append(info->list,
|
|
msg_hash_to_str(MENU_ENUM_LABEL_VALUE_ACHIEVEMENT_RESUME_CANCEL),
|
|
msg_hash_to_str(MENU_ENUM_LABEL_ACHIEVEMENT_RESUME_CANCEL),
|
|
MENU_ENUM_LABEL_ACHIEVEMENT_RESUME_CANCEL,
|
|
MENU_SETTING_ACTION_CLOSE, 0, 0, NULL);
|
|
menu_entries_append(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, NULL);
|
|
}
|
|
}
|
|
}
|
|
|
|
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(info->list,
|
|
msg_hash_to_str(MENU_ENUM_LABEL_VALUE_ACHIEVEMENT_PAUSE),
|
|
msg_hash_to_str(MENU_ENUM_LABEL_ACHIEVEMENT_PAUSE_MENU),
|
|
MENU_ENUM_LABEL_ACHIEVEMENT_PAUSE_MENU,
|
|
MENU_SETTING_ACTION_PAUSE_ACHIEVEMENTS, 0, 0, NULL);
|
|
else
|
|
menu_entries_append(info->list,
|
|
msg_hash_to_str(MENU_ENUM_LABEL_VALUE_ACHIEVEMENT_RESUME),
|
|
msg_hash_to_str(MENU_ENUM_LABEL_ACHIEVEMENT_PAUSE_MENU),
|
|
MENU_ENUM_LABEL_ACHIEVEMENT_PAUSE_MENU,
|
|
MENU_SETTING_ACTION_RESUME_ACHIEVEMENTS, 0, 0, NULL);
|
|
}
|
|
|
|
/* 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(info->list, menuitem->cheevo->title,
|
|
menuitem->cheevo->description,
|
|
MENU_ENUM_LABEL_CHEEVOS_LOCKED_ENTRY,
|
|
MENU_SETTINGS_CHEEVOS_START + idx, 0, 0, NULL);
|
|
else
|
|
{
|
|
snprintf(buffer, sizeof(buffer), "----- %s -----",
|
|
msg_hash_to_str(menuitem->state_label_idx));
|
|
|
|
menu_entries_append(info->list, buffer, "",
|
|
MENU_ENUM_LABEL_CHEEVOS_LOCKED_ENTRY,
|
|
MENU_SETTINGS_CHEEVOS_START + idx, 0, 0, NULL);
|
|
}
|
|
|
|
++idx;
|
|
++menuitem;
|
|
} while (menuitem != stop);
|
|
}
|
|
else
|
|
{
|
|
/* no achievements found */
|
|
if (!rcheevos_locals->core_supports)
|
|
menu_entries_append(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, NULL);
|
|
else if (rcheevos_locals->load_info.state == RCHEEVOS_LOAD_STATE_NETWORK_ERROR)
|
|
menu_entries_append(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, NULL);
|
|
else if (!rcheevos_locals->game.id)
|
|
menu_entries_append(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, NULL);
|
|
else if (!rcheevos_locals->token[0])
|
|
menu_entries_append(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, NULL);
|
|
else
|
|
menu_entries_append(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, NULL);
|
|
}
|
|
}
|
|
|
|
#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;
|
|
}
|