mirror of https://github.com/PCSX2/pcsx2.git
Achievements: Use big picture to confirm HC mode disable
This commit is contained in:
parent
8a93c5696f
commit
acb45459f5
|
@ -392,13 +392,33 @@ void EmuThread::run()
|
||||||
// Main CPU thread loop.
|
// Main CPU thread loop.
|
||||||
while (!m_shutdown_flag.load())
|
while (!m_shutdown_flag.load())
|
||||||
{
|
{
|
||||||
if (!VMManager::HasValidVM())
|
switch (VMManager::GetState())
|
||||||
{
|
{
|
||||||
m_event_loop->exec();
|
case VMState::Initializing:
|
||||||
continue;
|
pxFailRel("Shouldn't be in the starting state");
|
||||||
}
|
continue;
|
||||||
|
|
||||||
executeVM();
|
case VMState::Shutdown:
|
||||||
|
case VMState::Paused:
|
||||||
|
m_event_loop->exec();
|
||||||
|
continue;
|
||||||
|
|
||||||
|
case VMState::Running:
|
||||||
|
m_event_loop->processEvents(QEventLoop::AllEvents);
|
||||||
|
VMManager::Execute();
|
||||||
|
continue;
|
||||||
|
|
||||||
|
case VMState::Resetting:
|
||||||
|
VMManager::Reset();
|
||||||
|
continue;
|
||||||
|
|
||||||
|
case VMState::Stopping:
|
||||||
|
destroyVM();
|
||||||
|
continue;
|
||||||
|
|
||||||
|
default:
|
||||||
|
continue;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Teardown in reverse order.
|
// Teardown in reverse order.
|
||||||
|
@ -423,40 +443,6 @@ void EmuThread::destroyVM()
|
||||||
m_save_state_on_shutdown = false;
|
m_save_state_on_shutdown = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
void EmuThread::executeVM()
|
|
||||||
{
|
|
||||||
for (;;)
|
|
||||||
{
|
|
||||||
switch (VMManager::GetState())
|
|
||||||
{
|
|
||||||
case VMState::Initializing:
|
|
||||||
pxFailRel("Shouldn't be in the starting state state");
|
|
||||||
continue;
|
|
||||||
|
|
||||||
case VMState::Paused:
|
|
||||||
m_event_loop->exec();
|
|
||||||
continue;
|
|
||||||
|
|
||||||
case VMState::Running:
|
|
||||||
m_event_loop->processEvents(QEventLoop::AllEvents);
|
|
||||||
VMManager::Execute();
|
|
||||||
continue;
|
|
||||||
|
|
||||||
case VMState::Resetting:
|
|
||||||
VMManager::Reset();
|
|
||||||
continue;
|
|
||||||
|
|
||||||
case VMState::Stopping:
|
|
||||||
destroyVM();
|
|
||||||
m_event_loop->processEvents(QEventLoop::AllEvents);
|
|
||||||
return;
|
|
||||||
|
|
||||||
default:
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void EmuThread::createBackgroundControllerPollTimer()
|
void EmuThread::createBackgroundControllerPollTimer()
|
||||||
{
|
{
|
||||||
pxAssert(!m_background_controller_polling_timer);
|
pxAssert(!m_background_controller_polling_timer);
|
||||||
|
|
|
@ -194,7 +194,6 @@ private:
|
||||||
static constexpr u32 FULLSCREEN_UI_CONTROLLER_POLLING_INTERVAL = 8;
|
static constexpr u32 FULLSCREEN_UI_CONTROLLER_POLLING_INTERVAL = 8;
|
||||||
|
|
||||||
void destroyVM();
|
void destroyVM();
|
||||||
void executeVM();
|
|
||||||
|
|
||||||
void createBackgroundControllerPollTimer();
|
void createBackgroundControllerPollTimer();
|
||||||
void destroyBackgroundControllerPollTimer();
|
void destroyBackgroundControllerPollTimer();
|
||||||
|
|
|
@ -1825,6 +1825,43 @@ bool Achievements::ConfirmHardcoreModeDisable(const char* trigger)
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void Achievements::ConfirmHardcoreModeDisableAsync(const char* trigger, std::function<void(bool)> callback)
|
||||||
|
{
|
||||||
|
#ifdef ENABLE_RAINTEGRATION
|
||||||
|
if (IsUsingRAIntegration())
|
||||||
|
{
|
||||||
|
const bool result = (RA_WarnDisableHardcore(trigger) != 0);
|
||||||
|
callback(result);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
if (!FullscreenUI::Initialize())
|
||||||
|
{
|
||||||
|
Host::AddOSDMessage(fmt::format(TRANSLATE_FS("Cannot {} while hardcode mode is active.", trigger)),
|
||||||
|
Host::OSD_WARNING_DURATION);
|
||||||
|
callback(false);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto real_callback = [callback = std::move(callback)](bool res) mutable {
|
||||||
|
// don't run the callback in the middle of rendering the UI
|
||||||
|
Host::RunOnCPUThread([callback = std::move(callback), res]() {
|
||||||
|
if (res)
|
||||||
|
DisableHardcoreMode();
|
||||||
|
callback(res);
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
ImGuiFullscreen::OpenConfirmMessageDialog(
|
||||||
|
TRANSLATE_STR("Achievements", "Confirm Hardcore Mode"),
|
||||||
|
fmt::format(TRANSLATE_FS("Achievements", "{0} cannot be performed while hardcore mode is active. Do you "
|
||||||
|
"want to disable hardcore mode? {0} will be cancelled if you select No."),
|
||||||
|
trigger),
|
||||||
|
std::move(real_callback), fmt::format(ICON_FA_CHECK " {}", TRANSLATE_SV("Achievements", "Yes")),
|
||||||
|
fmt::format(ICON_FA_TIMES " {}", TRANSLATE_SV("Achievements", "No")));
|
||||||
|
}
|
||||||
|
|
||||||
void Achievements::ClearUIState()
|
void Achievements::ClearUIState()
|
||||||
{
|
{
|
||||||
if (FullscreenUI::IsAchievementsWindowOpen() || FullscreenUI::IsLeaderboardsWindowOpen())
|
if (FullscreenUI::IsAchievementsWindowOpen() || FullscreenUI::IsLeaderboardsWindowOpen())
|
||||||
|
|
|
@ -19,6 +19,7 @@
|
||||||
|
|
||||||
#include "Config.h"
|
#include "Config.h"
|
||||||
|
|
||||||
|
#include <functional>
|
||||||
#include <mutex>
|
#include <mutex>
|
||||||
#include <string>
|
#include <string>
|
||||||
#include <utility>
|
#include <utility>
|
||||||
|
@ -83,6 +84,7 @@ namespace Achievements
|
||||||
|
|
||||||
/// Prompts the user to disable hardcore mode, if they agree, returns true.
|
/// Prompts the user to disable hardcore mode, if they agree, returns true.
|
||||||
bool ConfirmHardcoreModeDisable(const char* trigger);
|
bool ConfirmHardcoreModeDisable(const char* trigger);
|
||||||
|
void ConfirmHardcoreModeDisableAsync(const char* trigger, std::function<void(bool)> callback);
|
||||||
|
|
||||||
/// Returns true if hardcore mode is active, and functionality should be restricted.
|
/// Returns true if hardcore mode is active, and functionality should be restricted.
|
||||||
bool IsHardcoreModeActive();
|
bool IsHardcoreModeActive();
|
||||||
|
|
|
@ -855,17 +855,12 @@ void FullscreenUI::DoStartPath(const std::string& path, std::optional<s32> state
|
||||||
params.fast_boot = fast_boot;
|
params.fast_boot = fast_boot;
|
||||||
|
|
||||||
// switch to nothing, we'll get brought back if init fails
|
// switch to nothing, we'll get brought back if init fails
|
||||||
const MainWindowType prev_window = s_current_main_window;
|
Host::RunOnCPUThread([params = std::move(params)]() {
|
||||||
s_current_main_window = MainWindowType::None;
|
|
||||||
|
|
||||||
Host::RunOnCPUThread([params = std::move(params), prev_window]() {
|
|
||||||
if (VMManager::HasValidVM())
|
if (VMManager::HasValidVM())
|
||||||
return;
|
return;
|
||||||
|
|
||||||
if (VMManager::Initialize(std::move(params)))
|
if (VMManager::Initialize(std::move(params)))
|
||||||
VMManager::SetState(VMState::Running);
|
VMManager::SetState(VMState::Running);
|
||||||
else
|
|
||||||
s_current_main_window = prev_window;
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -5514,6 +5509,18 @@ void FullscreenUI::DrawGameListWindow()
|
||||||
default:
|
default:
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (VMManager::GetState() != VMState::Shutdown)
|
||||||
|
{
|
||||||
|
// Dummy window to prevent interacting with the game list while loading.
|
||||||
|
ImGui::PushStyleColor(ImGuiCol_WindowBg, ImVec4(1.0f, 1.0f, 1.0f, 1.0f));
|
||||||
|
ImGui::SetNextWindowSize(ImGui::GetIO().DisplaySize);
|
||||||
|
ImGui::SetNextWindowPos(ImVec2(0, 0));
|
||||||
|
ImGui::SetNextWindowBgAlpha(0.25f);
|
||||||
|
ImGui::Begin("##dummy", nullptr, ImGuiWindowFlags_NoDecoration);
|
||||||
|
ImGui::End();
|
||||||
|
ImGui::PopStyleColor();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void FullscreenUI::DrawGameList(const ImVec2& heading_size)
|
void FullscreenUI::DrawGameList(const ImVec2& heading_size)
|
||||||
|
|
|
@ -658,7 +658,7 @@ void VMManager::Internal::UpdateEmuFolders()
|
||||||
|
|
||||||
AutoEject::SetAll();
|
AutoEject::SetAll();
|
||||||
|
|
||||||
if(!GSDumpReplayer::IsReplayingDump())
|
if (!GSDumpReplayer::IsReplayingDump())
|
||||||
FileMcd_Reopen(memcardFilters.empty() ? s_disc_serial : memcardFilters);
|
FileMcd_Reopen(memcardFilters.empty() ? s_disc_serial : memcardFilters);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1085,6 +1085,12 @@ bool VMManager::Initialize(VMBootParameters boot_params)
|
||||||
s_elf_override = {};
|
s_elf_override = {};
|
||||||
ClearELFInfo();
|
ClearELFInfo();
|
||||||
ClearDiscDetails();
|
ClearDiscDetails();
|
||||||
|
|
||||||
|
Achievements::GameChanged(0, 0);
|
||||||
|
FullscreenUI::GameChanged(s_title, std::string(), s_disc_serial, 0, 0);
|
||||||
|
UpdateDiscordPresence();
|
||||||
|
Host::OnGameChanged(s_title, std::string(), std::string(), s_disc_serial, 0, 0);
|
||||||
|
|
||||||
UpdateGameSettingsLayer();
|
UpdateGameSettingsLayer();
|
||||||
s_state.store(VMState::Shutdown, std::memory_order_release);
|
s_state.store(VMState::Shutdown, std::memory_order_release);
|
||||||
Host::OnVMDestroyed();
|
Host::OnVMDestroyed();
|
||||||
|
@ -1184,7 +1190,6 @@ bool VMManager::Initialize(VMBootParameters boot_params)
|
||||||
if (!FileSystem::FileExists(s_elf_override.c_str()))
|
if (!FileSystem::FileExists(s_elf_override.c_str()))
|
||||||
{
|
{
|
||||||
Host::ReportErrorAsync("Error", fmt::format("Requested boot ELF '{}' does not exist.", s_elf_override));
|
Host::ReportErrorAsync("Error", fmt::format("Requested boot ELF '{}' does not exist.", s_elf_override));
|
||||||
DoCDVDclose();
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1200,11 +1205,34 @@ bool VMManager::Initialize(VMBootParameters boot_params)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check for resuming with hardcore mode.
|
// Check for resuming with hardcore mode.
|
||||||
Achievements::ResetHardcoreMode();
|
// Why do we need the boot param? Because we need some way of telling BootSystem() that
|
||||||
if (!state_to_load.empty() && Achievements::IsHardcoreModeActive() &&
|
// the user allowed HC mode to be disabled, because otherwise we'll ResetHardcoreMode()
|
||||||
!Achievements::ConfirmHardcoreModeDisable(TRANSLATE("VMManager", "Resuming state")))
|
// and send ourselves into an infinite loop.
|
||||||
|
if (boot_params.disable_achievements_hardcore_mode)
|
||||||
|
Achievements::DisableHardcoreMode();
|
||||||
|
else
|
||||||
|
Achievements::ResetHardcoreMode();
|
||||||
|
if (!state_to_load.empty() && Achievements::IsHardcoreModeActive())
|
||||||
{
|
{
|
||||||
return false;
|
if (FullscreenUI::IsInitialized())
|
||||||
|
{
|
||||||
|
boot_params.elf_override = std::move(s_elf_override);
|
||||||
|
boot_params.save_state = std::move(state_to_load);
|
||||||
|
boot_params.disable_achievements_hardcore_mode = true;
|
||||||
|
s_elf_override = {};
|
||||||
|
|
||||||
|
Achievements::ConfirmHardcoreModeDisableAsync(TRANSLATE("VMManager", "Resuming state"),
|
||||||
|
[boot_params = std::move(boot_params)](bool approved) mutable {
|
||||||
|
if (approved && Initialize(std::move(boot_params)))
|
||||||
|
SetState(VMState::Running);
|
||||||
|
});
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
else if (!Achievements::ConfirmHardcoreModeDisable(TRANSLATE("VMManager", "Resuming state")))
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
s_limiter_mode = GetInitialLimiterMode();
|
s_limiter_mode = GetInitialLimiterMode();
|
||||||
|
@ -1706,8 +1734,15 @@ u32 VMManager::DeleteSaveStates(const char* game_serial, u32 game_crc, bool also
|
||||||
|
|
||||||
bool VMManager::LoadState(const char* filename)
|
bool VMManager::LoadState(const char* filename)
|
||||||
{
|
{
|
||||||
if (Achievements::IsHardcoreModeActive() && !Achievements::ConfirmHardcoreModeDisable(TRANSLATE("VMManager", "Loading state")))
|
if (Achievements::IsHardcoreModeActive())
|
||||||
|
{
|
||||||
|
Achievements::ConfirmHardcoreModeDisableAsync(TRANSLATE("VMManager", "Loading state"),
|
||||||
|
[filename = std::string(filename)](bool approved) {
|
||||||
|
if (approved)
|
||||||
|
LoadState(filename.c_str());
|
||||||
|
});
|
||||||
return false;
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
// TODO: Save the current state so we don't need to reset.
|
// TODO: Save the current state so we don't need to reset.
|
||||||
if (DoLoadState(filename))
|
if (DoLoadState(filename))
|
||||||
|
@ -1719,7 +1754,7 @@ bool VMManager::LoadState(const char* filename)
|
||||||
|
|
||||||
bool VMManager::LoadStateFromSlot(s32 slot)
|
bool VMManager::LoadStateFromSlot(s32 slot)
|
||||||
{
|
{
|
||||||
const std::string filename(GetCurrentSaveStateFileName(slot));
|
const std::string filename = GetCurrentSaveStateFileName(slot);
|
||||||
if (filename.empty())
|
if (filename.empty())
|
||||||
{
|
{
|
||||||
Host::AddIconOSDMessage("LoadStateFromSlot", ICON_FA_EXCLAMATION_TRIANGLE,
|
Host::AddIconOSDMessage("LoadStateFromSlot", ICON_FA_EXCLAMATION_TRIANGLE,
|
||||||
|
@ -1728,8 +1763,15 @@ bool VMManager::LoadStateFromSlot(s32 slot)
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (Achievements::IsHardcoreModeActive() && !Achievements::ConfirmHardcoreModeDisable(TRANSLATE("VMManager", "Loading state")))
|
if (Achievements::IsHardcoreModeActive())
|
||||||
|
{
|
||||||
|
Achievements::ConfirmHardcoreModeDisableAsync(TRANSLATE("VMManager", "Loading state"),
|
||||||
|
[slot](bool approved) {
|
||||||
|
if (approved)
|
||||||
|
LoadStateFromSlot(slot);
|
||||||
|
});
|
||||||
return false;
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
Host::AddIconOSDMessage("LoadStateFromSlot", ICON_FA_FOLDER_OPEN,
|
Host::AddIconOSDMessage("LoadStateFromSlot", ICON_FA_FOLDER_OPEN,
|
||||||
fmt::format(TRANSLATE_FS("VMManager", "Loading state from slot {}..."), slot), Host::OSD_QUICK_DURATION);
|
fmt::format(TRANSLATE_FS("VMManager", "Loading state from slot {}..."), slot), Host::OSD_QUICK_DURATION);
|
||||||
|
@ -1916,8 +1958,16 @@ void VMManager::FrameAdvance(u32 num_frames /*= 1*/)
|
||||||
if (!HasValidVM())
|
if (!HasValidVM())
|
||||||
return;
|
return;
|
||||||
|
|
||||||
if (Achievements::IsHardcoreModeActive() && !Achievements::ConfirmHardcoreModeDisable(TRANSLATE("VMManager", "Frame advancing")))
|
if (Achievements::IsHardcoreModeActive())
|
||||||
|
{
|
||||||
|
Achievements::ConfirmHardcoreModeDisableAsync(TRANSLATE("VMManager", "Frame advancing"),
|
||||||
|
[num_frames](bool approved) {
|
||||||
|
if (approved)
|
||||||
|
FrameAdvance(num_frames);
|
||||||
|
});
|
||||||
|
|
||||||
return;
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
s_frame_advance_count = num_frames;
|
s_frame_advance_count = num_frames;
|
||||||
SetState(VMState::Running);
|
SetState(VMState::Running);
|
||||||
|
@ -1961,7 +2011,7 @@ bool VMManager::ChangeDisc(CDVD_SourceType source, std::string path)
|
||||||
if (!DoCDVDopen(&error))
|
if (!DoCDVDopen(&error))
|
||||||
{
|
{
|
||||||
Host::AddIconOSDMessage("ChangeDisc", ICON_FA_COMPACT_DISC,
|
Host::AddIconOSDMessage("ChangeDisc", ICON_FA_COMPACT_DISC,
|
||||||
fmt::format(TRANSLATE_FS("VMManager", "Failed to switch back to old disc image. Removing disc.\nError was: {}"),
|
fmt::format(TRANSLATE_FS("VMManager", "Failed to switch back to old disc image. Removing disc.\nError was: {}"),
|
||||||
error.GetDescription()),
|
error.GetDescription()),
|
||||||
Host::OSD_CRITICAL_ERROR_DURATION);
|
Host::OSD_CRITICAL_ERROR_DURATION);
|
||||||
CDVDsys_ChangeSource(CDVD_SourceType::NoDisc);
|
CDVDsys_ChangeSource(CDVD_SourceType::NoDisc);
|
||||||
|
@ -2482,8 +2532,7 @@ void VMManager::WarnAboutUnsafeSettings()
|
||||||
return;
|
return;
|
||||||
|
|
||||||
std::string messages;
|
std::string messages;
|
||||||
auto append = [&messages](const char* icon, const std::string_view& msg)
|
auto append = [&messages](const char* icon, const std::string_view& msg) {
|
||||||
{
|
|
||||||
messages += icon;
|
messages += icon;
|
||||||
messages += ' ';
|
messages += ' ';
|
||||||
messages += msg;
|
messages += msg;
|
||||||
|
@ -2860,11 +2909,12 @@ static void InitializeCPUInfo()
|
||||||
s_big_cores = 0;
|
s_big_cores = 0;
|
||||||
s_small_cores = 0;
|
s_small_cores = 0;
|
||||||
std::vector<DarwinMisc::CPUClass> classes = DarwinMisc::GetCPUClasses();
|
std::vector<DarwinMisc::CPUClass> classes = DarwinMisc::GetCPUClasses();
|
||||||
for (size_t i = 0; i < classes.size(); i++) {
|
for (size_t i = 0; i < classes.size(); i++)
|
||||||
|
{
|
||||||
const DarwinMisc::CPUClass& cls = classes[i];
|
const DarwinMisc::CPUClass& cls = classes[i];
|
||||||
const bool is_big = i == 0 || i < classes.size() - 1; // Assume only one group is small
|
const bool is_big = i == 0 || i < classes.size() - 1; // Assume only one group is small
|
||||||
DevCon.WriteLn("(VMManager) Found %u physical cores and %u logical cores in perf level %u (%s), assuming %s",
|
DevCon.WriteLn("(VMManager) Found %u physical cores and %u logical cores in perf level %u (%s), assuming %s",
|
||||||
cls.num_physical, cls.num_logical, i, cls.name.c_str(), is_big ? "big" : "small");
|
cls.num_physical, cls.num_logical, i, cls.name.c_str(), is_big ? "big" : "small");
|
||||||
(is_big ? s_big_cores : s_small_cores) += cls.num_physical;
|
(is_big ? s_big_cores : s_small_cores) += cls.num_physical;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -47,6 +47,7 @@ struct VMBootParameters
|
||||||
|
|
||||||
std::optional<bool> fast_boot;
|
std::optional<bool> fast_boot;
|
||||||
std::optional<bool> fullscreen;
|
std::optional<bool> fullscreen;
|
||||||
|
bool disable_achievements_hardcore_mode = false;
|
||||||
};
|
};
|
||||||
|
|
||||||
namespace VMManager
|
namespace VMManager
|
||||||
|
|
Loading…
Reference in New Issue