mirror of https://github.com/cemu-project/Cemu.git
erreula: Rework implementation and fix bugs
- ErrEula doesn't disappear on its own anymore. The expected behavior is for the game to call Disappear once a button has been selected. This fixes issues where the dialog would softlock in some games - Modernized code a bit - Added a subtle fade in/out effect
This commit is contained in:
parent
a5717e1b11
commit
66658351c1
|
@ -637,40 +637,40 @@ namespace CafeSystem
|
||||||
fsc_unmount("/cemuBossStorage/", FSC_PRIORITY_BASE);
|
fsc_unmount("/cemuBossStorage/", FSC_PRIORITY_BASE);
|
||||||
}
|
}
|
||||||
|
|
||||||
STATUS_CODE LoadAndMountForegroundTitle(TitleId titleId)
|
PREPARE_STATUS_CODE LoadAndMountForegroundTitle(TitleId titleId)
|
||||||
{
|
{
|
||||||
cemuLog_log(LogType::Force, "Mounting title {:016x}", (uint64)titleId);
|
cemuLog_log(LogType::Force, "Mounting title {:016x}", (uint64)titleId);
|
||||||
sGameInfo_ForegroundTitle = CafeTitleList::GetGameInfo(titleId);
|
sGameInfo_ForegroundTitle = CafeTitleList::GetGameInfo(titleId);
|
||||||
if (!sGameInfo_ForegroundTitle.IsValid())
|
if (!sGameInfo_ForegroundTitle.IsValid())
|
||||||
{
|
{
|
||||||
cemuLog_log(LogType::Force, "Mounting failed: Game meta information is either missing, inaccessible or not valid (missing or invalid .xml files in code and meta folder)");
|
cemuLog_log(LogType::Force, "Mounting failed: Game meta information is either missing, inaccessible or not valid (missing or invalid .xml files in code and meta folder)");
|
||||||
return STATUS_CODE::UNABLE_TO_MOUNT;
|
return PREPARE_STATUS_CODE::UNABLE_TO_MOUNT;
|
||||||
}
|
}
|
||||||
// check base
|
// check base
|
||||||
TitleInfo& titleBase = sGameInfo_ForegroundTitle.GetBase();
|
TitleInfo& titleBase = sGameInfo_ForegroundTitle.GetBase();
|
||||||
if (!titleBase.IsValid())
|
if (!titleBase.IsValid())
|
||||||
return STATUS_CODE::UNABLE_TO_MOUNT;
|
return PREPARE_STATUS_CODE::UNABLE_TO_MOUNT;
|
||||||
if(!titleBase.ParseXmlInfo())
|
if(!titleBase.ParseXmlInfo())
|
||||||
return STATUS_CODE::UNABLE_TO_MOUNT;
|
return PREPARE_STATUS_CODE::UNABLE_TO_MOUNT;
|
||||||
cemuLog_log(LogType::Force, "Base: {}", titleBase.GetPrintPath());
|
cemuLog_log(LogType::Force, "Base: {}", titleBase.GetPrintPath());
|
||||||
// mount base
|
// mount base
|
||||||
if (!titleBase.Mount("/vol/content", "content", FSC_PRIORITY_BASE) || !titleBase.Mount(GetInternalVirtualCodeFolder(), "code", FSC_PRIORITY_BASE))
|
if (!titleBase.Mount("/vol/content", "content", FSC_PRIORITY_BASE) || !titleBase.Mount(GetInternalVirtualCodeFolder(), "code", FSC_PRIORITY_BASE))
|
||||||
{
|
{
|
||||||
cemuLog_log(LogType::Force, "Mounting failed");
|
cemuLog_log(LogType::Force, "Mounting failed");
|
||||||
return STATUS_CODE::UNABLE_TO_MOUNT;
|
return PREPARE_STATUS_CODE::UNABLE_TO_MOUNT;
|
||||||
}
|
}
|
||||||
// check update
|
// check update
|
||||||
TitleInfo& titleUpdate = sGameInfo_ForegroundTitle.GetUpdate();
|
TitleInfo& titleUpdate = sGameInfo_ForegroundTitle.GetUpdate();
|
||||||
if (titleUpdate.IsValid())
|
if (titleUpdate.IsValid())
|
||||||
{
|
{
|
||||||
if (!titleUpdate.ParseXmlInfo())
|
if (!titleUpdate.ParseXmlInfo())
|
||||||
return STATUS_CODE::UNABLE_TO_MOUNT;
|
return PREPARE_STATUS_CODE::UNABLE_TO_MOUNT;
|
||||||
cemuLog_log(LogType::Force, "Update: {}", titleUpdate.GetPrintPath());
|
cemuLog_log(LogType::Force, "Update: {}", titleUpdate.GetPrintPath());
|
||||||
// mount update
|
// mount update
|
||||||
if (!titleUpdate.Mount("/vol/content", "content", FSC_PRIORITY_PATCH) || !titleUpdate.Mount(GetInternalVirtualCodeFolder(), "code", FSC_PRIORITY_PATCH))
|
if (!titleUpdate.Mount("/vol/content", "content", FSC_PRIORITY_PATCH) || !titleUpdate.Mount(GetInternalVirtualCodeFolder(), "code", FSC_PRIORITY_PATCH))
|
||||||
{
|
{
|
||||||
cemuLog_log(LogType::Force, "Mounting failed");
|
cemuLog_log(LogType::Force, "Mounting failed");
|
||||||
return STATUS_CODE::UNABLE_TO_MOUNT;
|
return PREPARE_STATUS_CODE::UNABLE_TO_MOUNT;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
|
@ -682,20 +682,20 @@ namespace CafeSystem
|
||||||
// todo - support for multi-title AOC
|
// todo - support for multi-title AOC
|
||||||
TitleInfo& titleAOC = aocList[0];
|
TitleInfo& titleAOC = aocList[0];
|
||||||
if (!titleAOC.ParseXmlInfo())
|
if (!titleAOC.ParseXmlInfo())
|
||||||
return STATUS_CODE::UNABLE_TO_MOUNT;
|
return PREPARE_STATUS_CODE::UNABLE_TO_MOUNT;
|
||||||
cemu_assert_debug(titleAOC.IsValid());
|
cemu_assert_debug(titleAOC.IsValid());
|
||||||
cemuLog_log(LogType::Force, "DLC: {}", titleAOC.GetPrintPath());
|
cemuLog_log(LogType::Force, "DLC: {}", titleAOC.GetPrintPath());
|
||||||
// mount AOC
|
// mount AOC
|
||||||
if (!titleAOC.Mount(fmt::format("/vol/aoc{:016x}", titleAOC.GetAppTitleId()), "content", FSC_PRIORITY_PATCH))
|
if (!titleAOC.Mount(fmt::format("/vol/aoc{:016x}", titleAOC.GetAppTitleId()), "content", FSC_PRIORITY_PATCH))
|
||||||
{
|
{
|
||||||
cemuLog_log(LogType::Force, "Mounting failed");
|
cemuLog_log(LogType::Force, "Mounting failed");
|
||||||
return STATUS_CODE::UNABLE_TO_MOUNT;
|
return PREPARE_STATUS_CODE::UNABLE_TO_MOUNT;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
cemuLog_log(LogType::Force, "DLC: Not present");
|
cemuLog_log(LogType::Force, "DLC: Not present");
|
||||||
sForegroundTitleId = titleId;
|
sForegroundTitleId = titleId;
|
||||||
return STATUS_CODE::SUCCESS;
|
return PREPARE_STATUS_CODE::SUCCESS;
|
||||||
}
|
}
|
||||||
|
|
||||||
void UnmountForegroundTitle()
|
void UnmountForegroundTitle()
|
||||||
|
@ -723,7 +723,7 @@ namespace CafeSystem
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
STATUS_CODE SetupExecutable()
|
PREPARE_STATUS_CODE SetupExecutable()
|
||||||
{
|
{
|
||||||
// set rpx path from cos.xml if available
|
// set rpx path from cos.xml if available
|
||||||
_pathToBaseExecutable = _pathToExecutable;
|
_pathToBaseExecutable = _pathToExecutable;
|
||||||
|
@ -755,7 +755,7 @@ namespace CafeSystem
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
LoadMainExecutable();
|
LoadMainExecutable();
|
||||||
return STATUS_CODE::SUCCESS;
|
return PREPARE_STATUS_CODE::SUCCESS;
|
||||||
}
|
}
|
||||||
|
|
||||||
void SetupMemorySpace()
|
void SetupMemorySpace()
|
||||||
|
@ -769,7 +769,7 @@ namespace CafeSystem
|
||||||
memory_unmapForCurrentTitle();
|
memory_unmapForCurrentTitle();
|
||||||
}
|
}
|
||||||
|
|
||||||
STATUS_CODE PrepareForegroundTitle(TitleId titleId)
|
PREPARE_STATUS_CODE PrepareForegroundTitle(TitleId titleId)
|
||||||
{
|
{
|
||||||
CafeTitleList::WaitForMandatoryScan();
|
CafeTitleList::WaitForMandatoryScan();
|
||||||
sLaunchModeIsStandalone = false;
|
sLaunchModeIsStandalone = false;
|
||||||
|
@ -780,21 +780,21 @@ namespace CafeSystem
|
||||||
// mount mlc storage
|
// mount mlc storage
|
||||||
MountBaseDirectories();
|
MountBaseDirectories();
|
||||||
// mount title folders
|
// mount title folders
|
||||||
STATUS_CODE r = LoadAndMountForegroundTitle(titleId);
|
PREPARE_STATUS_CODE r = LoadAndMountForegroundTitle(titleId);
|
||||||
if (r != STATUS_CODE::SUCCESS)
|
if (r != PREPARE_STATUS_CODE::SUCCESS)
|
||||||
return r;
|
return r;
|
||||||
gameProfile_load();
|
gameProfile_load();
|
||||||
// setup memory space and PPC recompiler
|
// setup memory space and PPC recompiler
|
||||||
SetupMemorySpace();
|
SetupMemorySpace();
|
||||||
PPCRecompiler_init();
|
PPCRecompiler_init();
|
||||||
r = SetupExecutable(); // load RPX
|
r = SetupExecutable(); // load RPX
|
||||||
if (r != STATUS_CODE::SUCCESS)
|
if (r != PREPARE_STATUS_CODE::SUCCESS)
|
||||||
return r;
|
return r;
|
||||||
InitVirtualMlcStorage();
|
InitVirtualMlcStorage();
|
||||||
return STATUS_CODE::SUCCESS;
|
return PREPARE_STATUS_CODE::SUCCESS;
|
||||||
}
|
}
|
||||||
|
|
||||||
STATUS_CODE PrepareForegroundTitleFromStandaloneRPX(const fs::path& path)
|
PREPARE_STATUS_CODE PrepareForegroundTitleFromStandaloneRPX(const fs::path& path)
|
||||||
{
|
{
|
||||||
sLaunchModeIsStandalone = true;
|
sLaunchModeIsStandalone = true;
|
||||||
cemuLog_log(LogType::Force, "Launching executable in standalone mode due to incorrect layout or missing meta files");
|
cemuLog_log(LogType::Force, "Launching executable in standalone mode due to incorrect layout or missing meta files");
|
||||||
|
@ -812,7 +812,7 @@ namespace CafeSystem
|
||||||
if (!r)
|
if (!r)
|
||||||
{
|
{
|
||||||
cemuLog_log(LogType::Force, "Failed to mount {}", _pathToUtf8(contentPath));
|
cemuLog_log(LogType::Force, "Failed to mount {}", _pathToUtf8(contentPath));
|
||||||
return STATUS_CODE::UNABLE_TO_MOUNT;
|
return PREPARE_STATUS_CODE::UNABLE_TO_MOUNT;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -824,7 +824,7 @@ namespace CafeSystem
|
||||||
// since a lot of systems (including save folder location) rely on a TitleId, we derive a placeholder id from the executable hash
|
// since a lot of systems (including save folder location) rely on a TitleId, we derive a placeholder id from the executable hash
|
||||||
auto execData = fsc_extractFile(_pathToExecutable.c_str());
|
auto execData = fsc_extractFile(_pathToExecutable.c_str());
|
||||||
if (!execData)
|
if (!execData)
|
||||||
return STATUS_CODE::INVALID_RPX;
|
return PREPARE_STATUS_CODE::INVALID_RPX;
|
||||||
uint32 h = generateHashFromRawRPXData(execData->data(), execData->size());
|
uint32 h = generateHashFromRawRPXData(execData->data(), execData->size());
|
||||||
sForegroundTitleId = 0xFFFFFFFF00000000ULL | (uint64)h;
|
sForegroundTitleId = 0xFFFFFFFF00000000ULL | (uint64)h;
|
||||||
cemuLog_log(LogType::Force, "Generated placeholder TitleId: {:016x}", sForegroundTitleId);
|
cemuLog_log(LogType::Force, "Generated placeholder TitleId: {:016x}", sForegroundTitleId);
|
||||||
|
@ -834,7 +834,7 @@ namespace CafeSystem
|
||||||
// load executable
|
// load executable
|
||||||
SetupExecutable();
|
SetupExecutable();
|
||||||
InitVirtualMlcStorage();
|
InitVirtualMlcStorage();
|
||||||
return STATUS_CODE::SUCCESS;
|
return PREPARE_STATUS_CODE::SUCCESS;
|
||||||
}
|
}
|
||||||
|
|
||||||
void _LaunchTitleThread()
|
void _LaunchTitleThread()
|
||||||
|
|
|
@ -15,20 +15,19 @@ namespace CafeSystem
|
||||||
virtual void CafeRecreateCanvas() = 0;
|
virtual void CafeRecreateCanvas() = 0;
|
||||||
};
|
};
|
||||||
|
|
||||||
enum class STATUS_CODE
|
enum class PREPARE_STATUS_CODE
|
||||||
{
|
{
|
||||||
SUCCESS,
|
SUCCESS,
|
||||||
INVALID_RPX,
|
INVALID_RPX,
|
||||||
UNABLE_TO_MOUNT, // failed to mount through TitleInfo (most likely caused by an invalid or outdated path)
|
UNABLE_TO_MOUNT, // failed to mount through TitleInfo (most likely caused by an invalid or outdated path)
|
||||||
//BAD_META_DATA, - the title list only stores titles with valid meta, so this error code is impossible
|
|
||||||
};
|
};
|
||||||
|
|
||||||
void Initialize();
|
void Initialize();
|
||||||
void SetImplementation(SystemImplementation* impl);
|
void SetImplementation(SystemImplementation* impl);
|
||||||
void Shutdown();
|
void Shutdown();
|
||||||
|
|
||||||
STATUS_CODE PrepareForegroundTitle(TitleId titleId);
|
PREPARE_STATUS_CODE PrepareForegroundTitle(TitleId titleId);
|
||||||
STATUS_CODE PrepareForegroundTitleFromStandaloneRPX(const fs::path& path);
|
PREPARE_STATUS_CODE PrepareForegroundTitleFromStandaloneRPX(const fs::path& path);
|
||||||
void LaunchForegroundTitle();
|
void LaunchForegroundTitle();
|
||||||
bool IsTitleRunning();
|
bool IsTitleRunning();
|
||||||
|
|
||||||
|
|
|
@ -40,7 +40,12 @@ namespace coreinit
|
||||||
|
|
||||||
inline TimerTicks ConvertNsToTimerTicks(uint64 ns)
|
inline TimerTicks ConvertNsToTimerTicks(uint64 ns)
|
||||||
{
|
{
|
||||||
return ((GetTimerClock() / 31250LL) * ((ns)) / 32000LL);
|
return ((GetTimerClock() / 31250LL) * ((TimerTicks)ns) / 32000LL);
|
||||||
|
}
|
||||||
|
|
||||||
|
inline TimerTicks ConvertMsToTimerTicks(uint64 ms)
|
||||||
|
{
|
||||||
|
return (TimerTicks)ms * GetTimerClock() / 1000LL;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -9,32 +9,45 @@
|
||||||
#include <wx/msgdlg.h>
|
#include <wx/msgdlg.h>
|
||||||
|
|
||||||
#include "Cafe/OS/libs/coreinit/coreinit_FS.h"
|
#include "Cafe/OS/libs/coreinit/coreinit_FS.h"
|
||||||
|
#include "Cafe/OS/libs/coreinit/coreinit_Time.h"
|
||||||
#include "Cafe/OS/libs/vpad/vpad.h"
|
#include "Cafe/OS/libs/vpad/vpad.h"
|
||||||
|
|
||||||
namespace nn
|
namespace nn
|
||||||
{
|
{
|
||||||
namespace erreula
|
namespace erreula
|
||||||
{
|
{
|
||||||
#define RESULTTYPE_NONE 0
|
|
||||||
#define RESULTTYPE_FINISH 1
|
|
||||||
#define RESULTTYPE_NEXT 2
|
|
||||||
#define RESULTTYPE_JUMP 3
|
|
||||||
#define RESULTTYPE_PASSWORD 4
|
|
||||||
|
|
||||||
#define ERRORTYPE_CODE 0
|
enum class ErrorDialogType : uint32
|
||||||
#define ERRORTYPE_TEXT 1
|
|
||||||
#define ERRORTYPE_TEXT_ONE_BUTTON 2
|
|
||||||
#define ERRORTYPE_TEXT_TWO_BUTTON 3
|
|
||||||
|
|
||||||
#define ERREULA_STATE_HIDDEN 0
|
|
||||||
#define ERREULA_STATE_APPEARING 1
|
|
||||||
#define ERREULA_STATE_VISIBLE 2
|
|
||||||
#define ERREULA_STATE_DISAPPEARING 3
|
|
||||||
|
|
||||||
struct AppearArg_t
|
|
||||||
{
|
{
|
||||||
AppearArg_t() = default;
|
Code = 0,
|
||||||
AppearArg_t(const AppearArg_t& o)
|
Text = 1,
|
||||||
|
TextOneButton = 2,
|
||||||
|
TextTwoButton = 3
|
||||||
|
};
|
||||||
|
|
||||||
|
static const sint32 FADE_TIME = 80;
|
||||||
|
|
||||||
|
enum class ErrEulaState : uint32
|
||||||
|
{
|
||||||
|
Hidden = 0,
|
||||||
|
Appearing = 1,
|
||||||
|
Visible = 2,
|
||||||
|
Disappearing = 3
|
||||||
|
};
|
||||||
|
|
||||||
|
enum class ResultType : uint32
|
||||||
|
{
|
||||||
|
None = 0,
|
||||||
|
Finish = 1,
|
||||||
|
Next = 2,
|
||||||
|
Jump = 3,
|
||||||
|
Password = 4
|
||||||
|
};
|
||||||
|
|
||||||
|
struct AppearError
|
||||||
|
{
|
||||||
|
AppearError() = default;
|
||||||
|
AppearError(const AppearError& o)
|
||||||
{
|
{
|
||||||
errorType = o.errorType;
|
errorType = o.errorType;
|
||||||
screenType = o.screenType;
|
screenType = o.screenType;
|
||||||
|
@ -49,7 +62,7 @@ namespace erreula
|
||||||
drawCursor = o.drawCursor;
|
drawCursor = o.drawCursor;
|
||||||
}
|
}
|
||||||
|
|
||||||
uint32be errorType;
|
betype<ErrorDialogType> errorType;
|
||||||
uint32be screenType;
|
uint32be screenType;
|
||||||
uint32be controllerType;
|
uint32be controllerType;
|
||||||
uint32be holdType;
|
uint32be holdType;
|
||||||
|
@ -63,7 +76,9 @@ namespace erreula
|
||||||
bool drawCursor{};
|
bool drawCursor{};
|
||||||
};
|
};
|
||||||
|
|
||||||
static_assert(sizeof(AppearArg_t) == 0x2C); // maybe larger
|
using AppearArg = AppearError;
|
||||||
|
|
||||||
|
static_assert(sizeof(AppearError) == 0x2C); // maybe larger
|
||||||
|
|
||||||
struct HomeNixSignArg_t
|
struct HomeNixSignArg_t
|
||||||
{
|
{
|
||||||
|
@ -80,6 +95,132 @@ namespace erreula
|
||||||
|
|
||||||
static_assert(sizeof(ControllerInfo_t) == 0x14); // maybe larger
|
static_assert(sizeof(ControllerInfo_t) == 0x14); // maybe larger
|
||||||
|
|
||||||
|
class ErrEulaInstance
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
enum class BUTTON_SELECTION : uint32
|
||||||
|
{
|
||||||
|
NONE = 0xFFFFFFFF,
|
||||||
|
LEFT = 0,
|
||||||
|
RIGHT = 1,
|
||||||
|
};
|
||||||
|
|
||||||
|
void Init()
|
||||||
|
{
|
||||||
|
m_buttonSelection = BUTTON_SELECTION::NONE;
|
||||||
|
m_resultCode = -1;
|
||||||
|
m_resultCodeForLeftButton = 0;
|
||||||
|
m_resultCodeForRightButton = 0;
|
||||||
|
SetState(ErrEulaState::Hidden);
|
||||||
|
}
|
||||||
|
|
||||||
|
void DoAppearError(AppearArg* arg)
|
||||||
|
{
|
||||||
|
m_buttonSelection = BUTTON_SELECTION::NONE;
|
||||||
|
m_resultCode = -1;
|
||||||
|
m_resultCodeForLeftButton = -1;
|
||||||
|
m_resultCodeForRightButton = -1;
|
||||||
|
// for standard dialog its 0 and 1?
|
||||||
|
m_resultCodeForLeftButton = 0;
|
||||||
|
m_resultCodeForRightButton = 1;
|
||||||
|
SetState(ErrEulaState::Appearing);
|
||||||
|
}
|
||||||
|
|
||||||
|
void DoDisappearError()
|
||||||
|
{
|
||||||
|
if(m_state != ErrEulaState::Visible)
|
||||||
|
return;
|
||||||
|
SetState(ErrEulaState::Disappearing);
|
||||||
|
}
|
||||||
|
|
||||||
|
void DoCalc()
|
||||||
|
{
|
||||||
|
// appearing and disappearing state will automatically advance after some time
|
||||||
|
if (m_state == ErrEulaState::Appearing || m_state == ErrEulaState::Disappearing)
|
||||||
|
{
|
||||||
|
uint32 elapsedTick = coreinit::OSGetTime() - m_lastStateChange;
|
||||||
|
if (elapsedTick > coreinit::EspressoTime::ConvertMsToTimerTicks(FADE_TIME))
|
||||||
|
{
|
||||||
|
SetState(m_state == ErrEulaState::Appearing ? ErrEulaState::Visible : ErrEulaState::Hidden);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bool IsDecideSelectButtonError() const
|
||||||
|
{
|
||||||
|
return m_buttonSelection != BUTTON_SELECTION::NONE;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool IsDecideSelectLeftButtonError() const
|
||||||
|
{
|
||||||
|
return m_buttonSelection != BUTTON_SELECTION::LEFT;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool IsDecideSelectRightButtonError() const
|
||||||
|
{
|
||||||
|
return m_buttonSelection != BUTTON_SELECTION::RIGHT;
|
||||||
|
}
|
||||||
|
|
||||||
|
void SetButtonSelection(BUTTON_SELECTION selection)
|
||||||
|
{
|
||||||
|
cemu_assert_debug(m_buttonSelection == BUTTON_SELECTION::NONE);
|
||||||
|
m_buttonSelection = selection;
|
||||||
|
cemu_assert_debug(selection == BUTTON_SELECTION::LEFT || selection == BUTTON_SELECTION::RIGHT);
|
||||||
|
m_resultCode = selection == BUTTON_SELECTION::LEFT ? m_resultCodeForLeftButton : m_resultCodeForRightButton;
|
||||||
|
}
|
||||||
|
|
||||||
|
ErrEulaState GetState() const
|
||||||
|
{
|
||||||
|
return m_state;
|
||||||
|
}
|
||||||
|
|
||||||
|
sint32 GetResultCode() const
|
||||||
|
{
|
||||||
|
return m_resultCode;
|
||||||
|
}
|
||||||
|
|
||||||
|
ResultType GetResultType() const
|
||||||
|
{
|
||||||
|
if(m_resultCode == -1)
|
||||||
|
return ResultType::None;
|
||||||
|
if(m_resultCode < 10)
|
||||||
|
return ResultType::Finish;
|
||||||
|
if(m_resultCode >= 9999)
|
||||||
|
return ResultType::Next;
|
||||||
|
if(m_resultCode == 40)
|
||||||
|
return ResultType::Password;
|
||||||
|
return ResultType::Jump;
|
||||||
|
}
|
||||||
|
|
||||||
|
float GetFadeTransparency() const
|
||||||
|
{
|
||||||
|
if(m_state == ErrEulaState::Appearing || m_state == ErrEulaState::Disappearing)
|
||||||
|
{
|
||||||
|
uint32 elapsedTick = coreinit::OSGetTime() - m_lastStateChange;
|
||||||
|
if(m_state == ErrEulaState::Appearing)
|
||||||
|
return std::min<float>(1.0f, (float)elapsedTick / (float)coreinit::EspressoTime::ConvertMsToTimerTicks(FADE_TIME));
|
||||||
|
else
|
||||||
|
return std::max<float>(0.0f, 1.0f - (float)elapsedTick / (float)coreinit::EspressoTime::ConvertMsToTimerTicks(FADE_TIME));
|
||||||
|
}
|
||||||
|
return 1.0f;
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
void SetState(ErrEulaState state)
|
||||||
|
{
|
||||||
|
m_state = state;
|
||||||
|
m_lastStateChange = coreinit::OSGetTime();
|
||||||
|
}
|
||||||
|
|
||||||
|
ErrEulaState m_state;
|
||||||
|
uint32 m_lastStateChange;
|
||||||
|
|
||||||
|
/* +0x30 */ betype<sint32> m_resultCode;
|
||||||
|
/* +0x239C */ betype<BUTTON_SELECTION> m_buttonSelection;
|
||||||
|
/* +0x23A0 */ betype<sint32> m_resultCodeForLeftButton;
|
||||||
|
/* +0x23A4 */ betype<sint32> m_resultCodeForRightButton;
|
||||||
|
};
|
||||||
|
|
||||||
struct ErrEula_t
|
struct ErrEula_t
|
||||||
{
|
{
|
||||||
SysAllocator<coreinit::OSMutex> mutex;
|
SysAllocator<coreinit::OSMutex> mutex;
|
||||||
|
@ -87,18 +228,12 @@ namespace erreula
|
||||||
uint32 langType;
|
uint32 langType;
|
||||||
MEMPTR<coreinit::FSClient_t> fsClient;
|
MEMPTR<coreinit::FSClient_t> fsClient;
|
||||||
|
|
||||||
AppearArg_t currentDialog;
|
std::unique_ptr<ErrEulaInstance> errEulaInstance;
|
||||||
uint32 state;
|
|
||||||
bool buttonPressed;
|
|
||||||
bool rightButtonPressed;
|
|
||||||
|
|
||||||
|
AppearError currentDialog;
|
||||||
bool homeNixSignVisible;
|
bool homeNixSignVisible;
|
||||||
|
|
||||||
std::chrono::steady_clock::time_point stateTimer{};
|
|
||||||
} g_errEula = {};
|
} g_errEula = {};
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
std::wstring GetText(uint16be* text)
|
std::wstring GetText(uint16be* text)
|
||||||
{
|
{
|
||||||
std::wstringstream result;
|
std::wstringstream result;
|
||||||
|
@ -113,22 +248,61 @@ namespace erreula
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void export_ErrEulaCreate(PPCInterpreter_t* hCPU)
|
void ErrEulaCreate(void* workmem, uint32 regionType, uint32 langType, coreinit::FSClient_t* fsClient)
|
||||||
{
|
{
|
||||||
ppcDefineParamMEMPTR(thisptr, uint8, 0);
|
|
||||||
ppcDefineParamU32(regionType, 1);
|
|
||||||
ppcDefineParamU32(langType, 2);
|
|
||||||
ppcDefineParamMEMPTR(fsClient, coreinit::FSClient_t, 3);
|
|
||||||
|
|
||||||
coreinit::OSLockMutex(&g_errEula.mutex);
|
coreinit::OSLockMutex(&g_errEula.mutex);
|
||||||
|
|
||||||
g_errEula.regionType = regionType;
|
g_errEula.regionType = regionType;
|
||||||
g_errEula.langType = langType;
|
g_errEula.langType = langType;
|
||||||
g_errEula.fsClient = fsClient;
|
g_errEula.fsClient = fsClient;
|
||||||
|
cemu_assert_debug(!g_errEula.errEulaInstance);
|
||||||
|
g_errEula.errEulaInstance = std::make_unique<ErrEulaInstance>();
|
||||||
|
g_errEula.errEulaInstance->Init();
|
||||||
|
|
||||||
coreinit::OSUnlockMutex(&g_errEula.mutex);
|
coreinit::OSUnlockMutex(&g_errEula.mutex);
|
||||||
|
}
|
||||||
|
|
||||||
osLib_returnFromFunction(hCPU, 0);
|
void ErrEulaDestroy()
|
||||||
|
{
|
||||||
|
g_errEula.errEulaInstance.reset();
|
||||||
|
}
|
||||||
|
|
||||||
|
// check if any dialog button was selected
|
||||||
|
bool IsDecideSelectButtonError()
|
||||||
|
{
|
||||||
|
if(!g_errEula.errEulaInstance)
|
||||||
|
return false;
|
||||||
|
return g_errEula.errEulaInstance->IsDecideSelectButtonError();
|
||||||
|
}
|
||||||
|
|
||||||
|
// check if left dialog button was selected
|
||||||
|
bool IsDecideSelectLeftButtonError()
|
||||||
|
{
|
||||||
|
if(!g_errEula.errEulaInstance)
|
||||||
|
return false;
|
||||||
|
return g_errEula.errEulaInstance->IsDecideSelectLeftButtonError();
|
||||||
|
}
|
||||||
|
|
||||||
|
// check if right dialog button was selected
|
||||||
|
bool IsDecideSelectRightButtonError()
|
||||||
|
{
|
||||||
|
if(!g_errEula.errEulaInstance)
|
||||||
|
return false;
|
||||||
|
return g_errEula.errEulaInstance->IsDecideSelectRightButtonError();
|
||||||
|
}
|
||||||
|
|
||||||
|
sint32 GetResultCode()
|
||||||
|
{
|
||||||
|
if(!g_errEula.errEulaInstance)
|
||||||
|
return -1;
|
||||||
|
return g_errEula.errEulaInstance->GetResultCode();
|
||||||
|
}
|
||||||
|
|
||||||
|
ResultType GetResultType()
|
||||||
|
{
|
||||||
|
if(!g_errEula.errEulaInstance)
|
||||||
|
return ResultType::None;
|
||||||
|
return g_errEula.errEulaInstance->GetResultType();
|
||||||
}
|
}
|
||||||
|
|
||||||
void export_AppearHomeNixSign(PPCInterpreter_t* hCPU)
|
void export_AppearHomeNixSign(PPCInterpreter_t* hCPU)
|
||||||
|
@ -137,28 +311,24 @@ namespace erreula
|
||||||
osLib_returnFromFunction(hCPU, 0);
|
osLib_returnFromFunction(hCPU, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
void export_AppearError(PPCInterpreter_t* hCPU)
|
void ErrEulaAppearError(AppearArg* arg)
|
||||||
{
|
{
|
||||||
ppcDefineParamMEMPTR(arg, AppearArg_t, 0);
|
g_errEula.currentDialog = *arg;
|
||||||
|
if(g_errEula.errEulaInstance)
|
||||||
g_errEula.currentDialog = *arg.GetPtr();
|
g_errEula.errEulaInstance->DoAppearError(arg);
|
||||||
g_errEula.state = ERREULA_STATE_APPEARING;
|
|
||||||
g_errEula.buttonPressed = false;
|
|
||||||
g_errEula.rightButtonPressed = false;
|
|
||||||
|
|
||||||
g_errEula.stateTimer = tick_cached();
|
|
||||||
|
|
||||||
osLib_returnFromFunction(hCPU, 0);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void export_GetStateErrorViewer(PPCInterpreter_t* hCPU)
|
void ErrEulaDisappearError()
|
||||||
{
|
{
|
||||||
osLib_returnFromFunction(hCPU, g_errEula.state);
|
if(g_errEula.errEulaInstance)
|
||||||
|
g_errEula.errEulaInstance->DoDisappearError();
|
||||||
}
|
}
|
||||||
void export_DisappearError(PPCInterpreter_t* hCPU)
|
|
||||||
|
ErrEulaState ErrEulaGetStateErrorViewer()
|
||||||
{
|
{
|
||||||
g_errEula.state = ERREULA_STATE_HIDDEN;
|
if(!g_errEula.errEulaInstance)
|
||||||
osLib_returnFromFunction(hCPU, 0);
|
return ErrEulaState::Hidden;
|
||||||
|
return g_errEula.errEulaInstance->GetState();
|
||||||
}
|
}
|
||||||
|
|
||||||
void export_ChangeLang(PPCInterpreter_t* hCPU)
|
void export_ChangeLang(PPCInterpreter_t* hCPU)
|
||||||
|
@ -168,27 +338,6 @@ namespace erreula
|
||||||
osLib_returnFromFunction(hCPU, 0);
|
osLib_returnFromFunction(hCPU, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
void export_IsDecideSelectButtonError(PPCInterpreter_t* hCPU)
|
|
||||||
{
|
|
||||||
if (g_errEula.buttonPressed)
|
|
||||||
cemuLog_logDebug(LogType::Force, "IsDecideSelectButtonError: TRUE");
|
|
||||||
osLib_returnFromFunction(hCPU, g_errEula.buttonPressed);
|
|
||||||
}
|
|
||||||
|
|
||||||
void export_IsDecideSelectLeftButtonError(PPCInterpreter_t* hCPU)
|
|
||||||
{
|
|
||||||
if (g_errEula.buttonPressed)
|
|
||||||
cemuLog_logDebug(LogType::Force, "IsDecideSelectLeftButtonError: TRUE");
|
|
||||||
osLib_returnFromFunction(hCPU, g_errEula.buttonPressed);
|
|
||||||
}
|
|
||||||
|
|
||||||
void export_IsDecideSelectRightButtonError(PPCInterpreter_t* hCPU)
|
|
||||||
{
|
|
||||||
if (g_errEula.rightButtonPressed)
|
|
||||||
cemuLog_logDebug(LogType::Force, "IsDecideSelectRightButtonError: TRUE");
|
|
||||||
osLib_returnFromFunction(hCPU, g_errEula.rightButtonPressed);
|
|
||||||
}
|
|
||||||
|
|
||||||
void export_IsAppearHomeNixSign(PPCInterpreter_t* hCPU)
|
void export_IsAppearHomeNixSign(PPCInterpreter_t* hCPU)
|
||||||
{
|
{
|
||||||
osLib_returnFromFunction(hCPU, g_errEula.homeNixSignVisible);
|
osLib_returnFromFunction(hCPU, g_errEula.homeNixSignVisible);
|
||||||
|
@ -200,61 +349,19 @@ namespace erreula
|
||||||
osLib_returnFromFunction(hCPU, 0);
|
osLib_returnFromFunction(hCPU, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
void export_GetResultType(PPCInterpreter_t* hCPU)
|
void ErrEulaCalc(ControllerInfo_t* controllerInfo)
|
||||||
{
|
{
|
||||||
uint32 result = RESULTTYPE_NONE;
|
if(g_errEula.errEulaInstance)
|
||||||
if (g_errEula.buttonPressed || g_errEula.rightButtonPressed)
|
g_errEula.errEulaInstance->DoCalc();
|
||||||
{
|
|
||||||
cemuLog_logDebug(LogType::Force, "GetResultType: FINISH");
|
|
||||||
result = RESULTTYPE_FINISH;
|
|
||||||
}
|
|
||||||
|
|
||||||
osLib_returnFromFunction(hCPU, result);
|
|
||||||
}
|
|
||||||
|
|
||||||
void export_Calc(PPCInterpreter_t* hCPU)
|
|
||||||
{
|
|
||||||
ppcDefineParamMEMPTR(controllerInfo, ControllerInfo_t, 0);
|
|
||||||
// TODO: check controller buttons bla to accept dialog?
|
|
||||||
osLib_returnFromFunction(hCPU, 0);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void render(bool mainWindow)
|
void render(bool mainWindow)
|
||||||
{
|
{
|
||||||
if(g_errEula.state == ERREULA_STATE_HIDDEN)
|
if(!g_errEula.errEulaInstance)
|
||||||
return;
|
return;
|
||||||
|
if(g_errEula.errEulaInstance->GetState() != ErrEulaState::Visible && g_errEula.errEulaInstance->GetState() != ErrEulaState::Appearing && g_errEula.errEulaInstance->GetState() != ErrEulaState::Disappearing)
|
||||||
if(g_errEula.state == ERREULA_STATE_APPEARING)
|
|
||||||
{
|
|
||||||
if(std::chrono::duration_cast<std::chrono::milliseconds>(tick_cached() - g_errEula.stateTimer).count() <= 1000)
|
|
||||||
{
|
|
||||||
return;
|
return;
|
||||||
}
|
const AppearError& appearArg = g_errEula.currentDialog;
|
||||||
|
|
||||||
g_errEula.state = ERREULA_STATE_VISIBLE;
|
|
||||||
g_errEula.stateTimer = tick_cached();
|
|
||||||
}
|
|
||||||
/*else if(g_errEula.state == STATE_VISIBLE)
|
|
||||||
{
|
|
||||||
if (std::chrono::duration_cast<std::chrono::milliseconds>(tick_cached() - g_errEula.stateTimer).count() >= 1000)
|
|
||||||
{
|
|
||||||
g_errEula.state = STATE_DISAPPEARING;
|
|
||||||
g_errEula.stateTimer = tick_cached();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}*/
|
|
||||||
else if(g_errEula.state == ERREULA_STATE_DISAPPEARING)
|
|
||||||
{
|
|
||||||
if (std::chrono::duration_cast<std::chrono::milliseconds>(tick_cached() - g_errEula.stateTimer).count() >= 2000)
|
|
||||||
{
|
|
||||||
g_errEula.state = ERREULA_STATE_HIDDEN;
|
|
||||||
g_errEula.stateTimer = tick_cached();
|
|
||||||
}
|
|
||||||
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const AppearArg_t& appearArg = g_errEula.currentDialog;
|
|
||||||
std::string text;
|
std::string text;
|
||||||
const uint32 errorCode = (uint32)appearArg.errorCode;
|
const uint32 errorCode = (uint32)appearArg.errorCode;
|
||||||
if (errorCode != 0)
|
if (errorCode != 0)
|
||||||
|
@ -280,13 +387,24 @@ namespace erreula
|
||||||
std::string title;
|
std::string title;
|
||||||
if (appearArg.title)
|
if (appearArg.title)
|
||||||
title = boost::nowide::narrow(GetText(appearArg.title.GetPtr()));
|
title = boost::nowide::narrow(GetText(appearArg.title.GetPtr()));
|
||||||
if(title.empty()) // ImGui doesn't allow empty titles, so set one if appearArg.title is not set or empty
|
if (title.empty()) // ImGui doesn't allow empty titles, so set one if appearArg.title is not set or empty
|
||||||
title = "ErrEula";
|
title = "ErrEula";
|
||||||
|
|
||||||
|
float fadeTransparency = 1.0f;
|
||||||
|
if (g_errEula.errEulaInstance->GetState() == ErrEulaState::Appearing || g_errEula.errEulaInstance->GetState() == ErrEulaState::Disappearing)
|
||||||
|
{
|
||||||
|
fadeTransparency = g_errEula.errEulaInstance->GetFadeTransparency();
|
||||||
|
}
|
||||||
|
|
||||||
|
float originalAlpha = ImGui::GetStyle().Alpha;
|
||||||
|
ImGui::GetStyle().Alpha = fadeTransparency;
|
||||||
|
ImGui::SetNextWindowBgAlpha(0.9f * fadeTransparency);
|
||||||
if (ImGui::Begin(title.c_str(), nullptr, kPopupFlags))
|
if (ImGui::Begin(title.c_str(), nullptr, kPopupFlags))
|
||||||
{
|
{
|
||||||
const float startx = ImGui::GetWindowSize().x / 2.0f;
|
const float startx = ImGui::GetWindowSize().x / 2.0f;
|
||||||
|
bool hasLeftButtonPressed = false, hasRightButtonPressed = false;
|
||||||
|
|
||||||
switch ((uint32)appearArg.errorType)
|
switch (appearArg.errorType)
|
||||||
{
|
{
|
||||||
default:
|
default:
|
||||||
{
|
{
|
||||||
|
@ -294,11 +412,10 @@ namespace erreula
|
||||||
ImGui::TextUnformatted(text.c_str(), text.c_str() + text.size());
|
ImGui::TextUnformatted(text.c_str(), text.c_str() + text.size());
|
||||||
ImGui::Spacing();
|
ImGui::Spacing();
|
||||||
ImGui::SetCursorPosX(startx - 50);
|
ImGui::SetCursorPosX(startx - 50);
|
||||||
g_errEula.buttonPressed |= ImGui::Button("OK", {100, 0});
|
hasLeftButtonPressed = ImGui::Button("OK", {100, 0});
|
||||||
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case ERRORTYPE_TEXT:
|
case ErrorDialogType::Text:
|
||||||
{
|
{
|
||||||
std::string txtTmp = "Unknown Error";
|
std::string txtTmp = "Unknown Error";
|
||||||
if (appearArg.text)
|
if (appearArg.text)
|
||||||
|
@ -309,10 +426,10 @@ namespace erreula
|
||||||
ImGui::Spacing();
|
ImGui::Spacing();
|
||||||
|
|
||||||
ImGui::SetCursorPosX(startx - 50);
|
ImGui::SetCursorPosX(startx - 50);
|
||||||
g_errEula.buttonPressed |= ImGui::Button("OK", { 100, 0 });
|
hasLeftButtonPressed = ImGui::Button("OK", { 100, 0 });
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case ERRORTYPE_TEXT_ONE_BUTTON:
|
case ErrorDialogType::TextOneButton:
|
||||||
{
|
{
|
||||||
std::string txtTmp = "Unknown Error";
|
std::string txtTmp = "Unknown Error";
|
||||||
if (appearArg.text)
|
if (appearArg.text)
|
||||||
|
@ -328,10 +445,10 @@ namespace erreula
|
||||||
|
|
||||||
float width = std::max(100.0f, ImGui::CalcTextSize(button1.c_str()).x + 10.0f);
|
float width = std::max(100.0f, ImGui::CalcTextSize(button1.c_str()).x + 10.0f);
|
||||||
ImGui::SetCursorPosX(startx - (width / 2.0f));
|
ImGui::SetCursorPosX(startx - (width / 2.0f));
|
||||||
g_errEula.buttonPressed |= ImGui::Button(button1.c_str(), { width, 0 });
|
hasLeftButtonPressed = ImGui::Button(button1.c_str(), { width, 0 });
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case ERRORTYPE_TEXT_TWO_BUTTON:
|
case ErrorDialogType::TextTwoButton:
|
||||||
{
|
{
|
||||||
std::string txtTmp = "Unknown Error";
|
std::string txtTmp = "Unknown Error";
|
||||||
if (appearArg.text)
|
if (appearArg.text)
|
||||||
|
@ -352,42 +469,52 @@ namespace erreula
|
||||||
float width2 = std::max(100.0f, ImGui::CalcTextSize(button2.c_str()).x + 10.0f);
|
float width2 = std::max(100.0f, ImGui::CalcTextSize(button2.c_str()).x + 10.0f);
|
||||||
ImGui::SetCursorPosX(startx - (width1 / 2.0f) - (width2 / 2.0f) - 10);
|
ImGui::SetCursorPosX(startx - (width1 / 2.0f) - (width2 / 2.0f) - 10);
|
||||||
|
|
||||||
g_errEula.buttonPressed |= ImGui::Button(button1.c_str(), { width1, 0 });
|
hasLeftButtonPressed = ImGui::Button(button1.c_str(), { width1, 0 });
|
||||||
ImGui::SameLine();
|
ImGui::SameLine();
|
||||||
|
|
||||||
g_errEula.rightButtonPressed |= ImGui::Button(button2.c_str(), { width2, 0 });
|
hasRightButtonPressed = ImGui::Button(button2.c_str(), { width2, 0 });
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
if (!g_errEula.errEulaInstance->IsDecideSelectButtonError())
|
||||||
|
{
|
||||||
|
if (hasLeftButtonPressed)
|
||||||
|
g_errEula.errEulaInstance->SetButtonSelection(ErrEulaInstance::BUTTON_SELECTION::LEFT);
|
||||||
|
if (hasRightButtonPressed)
|
||||||
|
g_errEula.errEulaInstance->SetButtonSelection(ErrEulaInstance::BUTTON_SELECTION::RIGHT);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
ImGui::End();
|
ImGui::End();
|
||||||
ImGui::PopFont();
|
ImGui::PopFont();
|
||||||
|
ImGui::GetStyle().Alpha = originalAlpha;
|
||||||
if(g_errEula.buttonPressed || g_errEula.rightButtonPressed)
|
|
||||||
{
|
|
||||||
g_errEula.state = ERREULA_STATE_DISAPPEARING;
|
|
||||||
g_errEula.stateTimer = tick_cached();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void load()
|
void load()
|
||||||
{
|
{
|
||||||
|
g_errEula.errEulaInstance.reset();
|
||||||
|
|
||||||
OSInitMutexEx(&g_errEula.mutex, nullptr);
|
OSInitMutexEx(&g_errEula.mutex, nullptr);
|
||||||
|
|
||||||
//osLib_addFunction("erreula", "ErrEulaCreate__3RplFPUcQ3_2nn7erreula10", export_ErrEulaCreate); // copy ctor?
|
cafeExportRegisterFunc(ErrEulaCreate, "erreula", "ErrEulaCreate__3RplFPUcQ3_2nn7erreula10RegionTypeQ3_2nn7erreula8LangTypeP8FSClient", LogType::Placeholder);
|
||||||
osLib_addFunction("erreula", "ErrEulaCreate__3RplFPUcQ3_2nn7erreula10RegionTypeQ3_2nn7erreula8LangTypeP8FSClient", export_ErrEulaCreate);
|
cafeExportRegisterFunc(ErrEulaDestroy, "erreula", "ErrEulaDestroy__3RplFv", LogType::Placeholder);
|
||||||
|
|
||||||
|
cafeExportRegisterFunc(IsDecideSelectButtonError, "erreula", "ErrEulaIsDecideSelectButtonError__3RplFv", LogType::Placeholder);
|
||||||
|
cafeExportRegisterFunc(IsDecideSelectLeftButtonError, "erreula", "ErrEulaIsDecideSelectLeftButtonError__3RplFv", LogType::Placeholder);
|
||||||
|
cafeExportRegisterFunc(IsDecideSelectRightButtonError, "erreula", "ErrEulaIsDecideSelectRightButtonError__3RplFv", LogType::Placeholder);
|
||||||
|
|
||||||
|
cafeExportRegisterFunc(GetResultCode, "erreula", "ErrEulaGetResultCode__3RplFv", LogType::Placeholder);
|
||||||
|
cafeExportRegisterFunc(GetResultType, "erreula", "ErrEulaGetResultType__3RplFv", LogType::Placeholder);
|
||||||
|
|
||||||
|
cafeExportRegisterFunc(ErrEulaAppearError, "erreula", "ErrEulaAppearError__3RplFRCQ3_2nn7erreula9AppearArg", LogType::Placeholder);
|
||||||
|
cafeExportRegisterFunc(ErrEulaDisappearError, "erreula", "ErrEulaDisappearError__3RplFv", LogType::Placeholder);
|
||||||
|
cafeExportRegisterFunc(ErrEulaGetStateErrorViewer, "erreula", "ErrEulaGetStateErrorViewer__3RplFv", LogType::Placeholder);
|
||||||
|
|
||||||
|
cafeExportRegisterFunc(ErrEulaCalc, "erreula", "ErrEulaCalc__3RplFRCQ3_2nn7erreula14ControllerInfo", LogType::Placeholder);
|
||||||
|
|
||||||
osLib_addFunction("erreula", "ErrEulaAppearHomeNixSign__3RplFRCQ3_2nn7erreula14HomeNixSignArg", export_AppearHomeNixSign);
|
osLib_addFunction("erreula", "ErrEulaAppearHomeNixSign__3RplFRCQ3_2nn7erreula14HomeNixSignArg", export_AppearHomeNixSign);
|
||||||
osLib_addFunction("erreula", "ErrEulaAppearError__3RplFRCQ3_2nn7erreula9AppearArg", export_AppearError);
|
|
||||||
osLib_addFunction("erreula", "ErrEulaGetStateErrorViewer__3RplFv", export_GetStateErrorViewer);
|
|
||||||
osLib_addFunction("erreula", "ErrEulaChangeLang__3RplFQ3_2nn7erreula8LangType", export_ChangeLang);
|
osLib_addFunction("erreula", "ErrEulaChangeLang__3RplFQ3_2nn7erreula8LangType", export_ChangeLang);
|
||||||
osLib_addFunction("erreula", "ErrEulaIsDecideSelectButtonError__3RplFv", export_IsDecideSelectButtonError);
|
|
||||||
osLib_addFunction("erreula", "ErrEulaCalc__3RplFRCQ3_2nn7erreula14ControllerInfo", export_Calc);
|
|
||||||
osLib_addFunction("erreula", "ErrEulaIsDecideSelectLeftButtonError__3RplFv", export_IsDecideSelectLeftButtonError);
|
|
||||||
osLib_addFunction("erreula", "ErrEulaIsDecideSelectRightButtonError__3RplFv", export_IsDecideSelectRightButtonError);
|
|
||||||
osLib_addFunction("erreula", "ErrEulaIsAppearHomeNixSign__3RplFv", export_IsAppearHomeNixSign);
|
osLib_addFunction("erreula", "ErrEulaIsAppearHomeNixSign__3RplFv", export_IsAppearHomeNixSign);
|
||||||
osLib_addFunction("erreula", "ErrEulaDisappearHomeNixSign__3RplFv", export_DisappearHomeNixSign);
|
osLib_addFunction("erreula", "ErrEulaDisappearHomeNixSign__3RplFv", export_DisappearHomeNixSign);
|
||||||
osLib_addFunction("erreula", "ErrEulaGetResultType__3RplFv", export_GetResultType);
|
|
||||||
osLib_addFunction("erreula", "ErrEulaDisappearError__3RplFv", export_DisappearError);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -483,20 +483,20 @@ bool MainWindow::FileLoad(const fs::path launchPath, wxLaunchGameEvent::INITIATE
|
||||||
wxMessageBox(t, _("Error"), wxOK | wxCENTRE | wxICON_ERROR);
|
wxMessageBox(t, _("Error"), wxOK | wxCENTRE | wxICON_ERROR);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
CafeSystem::STATUS_CODE r = CafeSystem::PrepareForegroundTitle(baseTitleId);
|
CafeSystem::PREPARE_STATUS_CODE r = CafeSystem::PrepareForegroundTitle(baseTitleId);
|
||||||
if (r == CafeSystem::STATUS_CODE::INVALID_RPX)
|
if (r == CafeSystem::PREPARE_STATUS_CODE::INVALID_RPX)
|
||||||
{
|
{
|
||||||
cemu_assert_debug(false);
|
cemu_assert_debug(false);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
else if (r == CafeSystem::STATUS_CODE::UNABLE_TO_MOUNT)
|
else if (r == CafeSystem::PREPARE_STATUS_CODE::UNABLE_TO_MOUNT)
|
||||||
{
|
{
|
||||||
wxString t = _("Unable to mount title.\nMake sure the configured game paths are still valid and refresh the game list.\n\nFile which failed to load:\n");
|
wxString t = _("Unable to mount title.\nMake sure the configured game paths are still valid and refresh the game list.\n\nFile which failed to load:\n");
|
||||||
t.append(_pathToUtf8(launchPath));
|
t.append(_pathToUtf8(launchPath));
|
||||||
wxMessageBox(t, _("Error"), wxOK | wxCENTRE | wxICON_ERROR);
|
wxMessageBox(t, _("Error"), wxOK | wxCENTRE | wxICON_ERROR);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
else if (r != CafeSystem::STATUS_CODE::SUCCESS)
|
else if (r != CafeSystem::PREPARE_STATUS_CODE::SUCCESS)
|
||||||
{
|
{
|
||||||
wxString t = _("Failed to launch game.");
|
wxString t = _("Failed to launch game.");
|
||||||
t.append(_pathToUtf8(launchPath));
|
t.append(_pathToUtf8(launchPath));
|
||||||
|
@ -511,8 +511,8 @@ bool MainWindow::FileLoad(const fs::path launchPath, wxLaunchGameEvent::INITIATE
|
||||||
CafeTitleFileType fileType = DetermineCafeSystemFileType(launchPath);
|
CafeTitleFileType fileType = DetermineCafeSystemFileType(launchPath);
|
||||||
if (fileType == CafeTitleFileType::RPX || fileType == CafeTitleFileType::ELF)
|
if (fileType == CafeTitleFileType::RPX || fileType == CafeTitleFileType::ELF)
|
||||||
{
|
{
|
||||||
CafeSystem::STATUS_CODE r = CafeSystem::PrepareForegroundTitleFromStandaloneRPX(launchPath);
|
CafeSystem::PREPARE_STATUS_CODE r = CafeSystem::PrepareForegroundTitleFromStandaloneRPX(launchPath);
|
||||||
if (r != CafeSystem::STATUS_CODE::SUCCESS)
|
if (r != CafeSystem::PREPARE_STATUS_CODE::SUCCESS)
|
||||||
{
|
{
|
||||||
cemu_assert_debug(false); // todo
|
cemu_assert_debug(false); // todo
|
||||||
wxString t = _("Failed to launch executable. Path: ");
|
wxString t = _("Failed to launch executable. Path: ");
|
||||||
|
|
Loading…
Reference in New Issue