mirror of
https://github.com/hedge-dev/UnleashedRecomp.git
synced 2026-06-12 19:21:39 +00:00
812 lines
29 KiB
C++
812 lines
29 KiB
C++
#include <user/config.h>
|
|
#include <api/SWA.h>
|
|
#include <app.h>
|
|
#include <ui/game_window.h>
|
|
#include <ui/sdl_listener.h>
|
|
#include <gpu/video.h>
|
|
|
|
// These are here for now to not recompile basically all of the project.
|
|
namespace Chao::CSD
|
|
{
|
|
struct Cast
|
|
{
|
|
SWA_INSERT_PADDING(0x144);
|
|
};
|
|
|
|
struct CastLink
|
|
{
|
|
be<uint32_t> ChildCastIndex;
|
|
be<uint32_t> SiblingCastIndex;
|
|
};
|
|
|
|
struct CastNode
|
|
{
|
|
be<uint32_t> CastCount;
|
|
xpointer<xpointer<Cast>> pCasts;
|
|
be<uint32_t> RootCastIndex;
|
|
xpointer<CastLink> pCastLinks;
|
|
};
|
|
|
|
struct CastIndex
|
|
{
|
|
xpointer<const char> pCastName;
|
|
be<uint32_t> CastNodeIndex;
|
|
be<uint32_t> CastIndex;
|
|
};
|
|
|
|
struct Scene
|
|
{
|
|
SWA_INSERT_PADDING(0x24);
|
|
be<uint32_t> CastNodeCount;
|
|
xpointer<CastNode> pCastNodes;
|
|
be<uint32_t> CastCount;
|
|
xpointer<CastIndex> pCastIndices;
|
|
};
|
|
|
|
struct SceneIndex
|
|
{
|
|
xpointer<const char> pSceneName;
|
|
be<uint32_t> SceneIndex;
|
|
};
|
|
|
|
struct SceneNodeIndex
|
|
{
|
|
xpointer<const char> pSceneNodeName;
|
|
be<uint32_t> SceneNodeIndex;
|
|
};
|
|
|
|
struct SceneNode
|
|
{
|
|
be<uint32_t> SceneCount;
|
|
xpointer<xpointer<Scene>> pScenes;
|
|
xpointer<SceneIndex> pSceneIndices;
|
|
be<uint32_t> SceneNodeCount;
|
|
xpointer<SceneNode> pSceneNodes;
|
|
xpointer<SceneNodeIndex> pSceneNodeIndices;
|
|
};
|
|
|
|
struct Project
|
|
{
|
|
xpointer<SceneNode> pRootNode;
|
|
};
|
|
}
|
|
|
|
static Mutex g_pathMutex;
|
|
static std::map<const void*, XXH64_hash_t> g_paths;
|
|
|
|
static XXH64_hash_t HashStr(const std::string_view& value)
|
|
{
|
|
return XXH3_64bits(value.data(), value.size());
|
|
}
|
|
|
|
static void EmplacePath(const void* key, const std::string_view& value)
|
|
{
|
|
std::lock_guard lock(g_pathMutex);
|
|
g_paths.emplace(key, HashStr(value));
|
|
}
|
|
|
|
static void TraverseCast(Chao::CSD::Scene* scene, uint32_t castNodeIndex, Chao::CSD::CastNode* castNode, uint32_t castIndex, const std::string& parentPath)
|
|
{
|
|
if (castIndex == ~0)
|
|
return;
|
|
|
|
TraverseCast(scene, castNodeIndex, castNode, castNode->pCastLinks[castIndex].SiblingCastIndex, parentPath);
|
|
|
|
std::string path = parentPath;
|
|
|
|
for (size_t i = 0; i < scene->CastCount; i++)
|
|
{
|
|
auto& index = scene->pCastIndices[i];
|
|
if (index.CastNodeIndex == castNodeIndex && index.CastIndex == castIndex)
|
|
{
|
|
path += index.pCastName;
|
|
break;
|
|
}
|
|
}
|
|
|
|
EmplacePath(castNode->pCasts[castIndex].get(), path);
|
|
|
|
if (castNode->RootCastIndex == castIndex)
|
|
EmplacePath(castNode, path);
|
|
|
|
path += "/";
|
|
|
|
TraverseCast(scene, castNodeIndex, castNode, castNode->pCastLinks[castIndex].ChildCastIndex, path);
|
|
}
|
|
|
|
static void TraverseScene(Chao::CSD::Scene* scene, std::string path)
|
|
{
|
|
EmplacePath(scene, path);
|
|
path += "/";
|
|
|
|
for (size_t i = 0; i < scene->CastNodeCount; i++)
|
|
{
|
|
auto& castNode = scene->pCastNodes[i];
|
|
TraverseCast(scene, i, &castNode, castNode.RootCastIndex, path);
|
|
}
|
|
}
|
|
|
|
static void TraverseSceneNode(Chao::CSD::SceneNode* sceneNode, std::string path)
|
|
{
|
|
EmplacePath(sceneNode, path);
|
|
path += "/";
|
|
|
|
for (size_t i = 0; i < sceneNode->SceneCount; i++)
|
|
{
|
|
auto& sceneIndex = sceneNode->pSceneIndices[i];
|
|
TraverseScene(sceneNode->pScenes[sceneIndex.SceneIndex], path + sceneIndex.pSceneName.get());
|
|
}
|
|
|
|
for (size_t i = 0; i < sceneNode->SceneNodeCount; i++)
|
|
{
|
|
auto& sceneNodeIndex = sceneNode->pSceneNodeIndices[i];
|
|
TraverseSceneNode(&sceneNode->pSceneNodes[sceneNodeIndex.SceneNodeIndex], path + sceneNodeIndex.pSceneNodeName.get());
|
|
}
|
|
}
|
|
|
|
void MakeCsdProjectMidAsmHook(PPCRegister& r3, PPCRegister& r29)
|
|
{
|
|
uint8_t* base = g_memory.base;
|
|
auto csdProject = reinterpret_cast<Chao::CSD::CProject*>(base + PPC_LOAD_U32(PPC_LOAD_U32(r3.u32 + 16) + 4));
|
|
auto name = reinterpret_cast<const char*>(base + PPC_LOAD_U32(r29.u32));
|
|
TraverseSceneNode(csdProject->m_pResource->pRootNode, name);
|
|
}
|
|
|
|
// Chao::CSD::CMemoryAlloc::Free
|
|
PPC_FUNC_IMPL(__imp__sub_825E2E60);
|
|
PPC_FUNC(sub_825E2E60)
|
|
{
|
|
if (ctx.r4.u32 != NULL && PPC_LOAD_U32(ctx.r4.u32) == 0x4E594946 && PPC_LOAD_U32(ctx.r4.u32 + 0x20) == 0x6E43504A) // NYIF, nCPJ
|
|
{
|
|
uint32_t fileSize = PPC_LOAD_U32(ctx.r4.u32 + 0x14);
|
|
|
|
std::lock_guard lock(g_pathMutex);
|
|
const uint8_t* key = base + ctx.r4.u32;
|
|
|
|
auto lower = g_paths.lower_bound(key);
|
|
auto upper = g_paths.lower_bound(key + fileSize);
|
|
|
|
g_paths.erase(lower, upper);
|
|
}
|
|
|
|
__imp__sub_825E2E60(ctx, base);
|
|
}
|
|
|
|
static constexpr float NARROW_ASPECT_RATIO = 4.0f / 3.0f;
|
|
static constexpr float WIDE_ASPECT_RATIO = 16.0f / 9.0f;
|
|
static constexpr float STEAM_DECK_ASPECT_RATIO = 16.0f / 10.0f;
|
|
static constexpr float TALL_ASPECT_RATIO = 1.0f / WIDE_ASPECT_RATIO;
|
|
static constexpr float TALL_SCALE = 1280.0f / 960.0f;
|
|
|
|
static float g_offsetX;
|
|
static float g_offsetY;
|
|
static float g_scale;
|
|
static float g_worldMapOffset;
|
|
static bool g_ultrawide;
|
|
|
|
static float ComputeScale(float aspectRatio)
|
|
{
|
|
return ((aspectRatio * 720.0f) / 1280.0f) / sqrt((aspectRatio * 720.0f) / 1280.0f);
|
|
}
|
|
|
|
static void ComputeOffsets(float width, float height)
|
|
{
|
|
float aspectRatio = width / height;
|
|
g_scale = 1.0f;
|
|
|
|
if (aspectRatio >= NARROW_ASPECT_RATIO)
|
|
{
|
|
// height is locked to 720 in this case
|
|
g_offsetX = 0.5f * (aspectRatio * 720.0f - 1280.0f);
|
|
g_offsetY = 0.0f;
|
|
|
|
// keep same scale above Steam Deck aspect ratio
|
|
if (aspectRatio < WIDE_ASPECT_RATIO)
|
|
{
|
|
// interpolate to original 4:3 scale
|
|
float steamDeckScale = aspectRatio / WIDE_ASPECT_RATIO;
|
|
float narrowScale = ComputeScale(NARROW_ASPECT_RATIO);
|
|
|
|
float lerpFactor = std::clamp((aspectRatio - NARROW_ASPECT_RATIO) / (STEAM_DECK_ASPECT_RATIO - NARROW_ASPECT_RATIO), 0.0f, 1.0f);
|
|
g_scale = narrowScale + (steamDeckScale - narrowScale) * lerpFactor;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// width is locked to 960 in this case to have 4:3 crop
|
|
g_offsetX = 0.5f * (960.0f - 1280.0f);
|
|
g_offsetY = 0.5f * (960.0f / aspectRatio - 720.0f);
|
|
|
|
// scale to 16:9 as the aspect ratio becomes 9:16
|
|
float factor = std::clamp((aspectRatio - TALL_ASPECT_RATIO) / (NARROW_ASPECT_RATIO - TALL_ASPECT_RATIO), 0.0f, 1.0f);
|
|
g_scale = TALL_SCALE + factor * (ComputeScale(NARROW_ASPECT_RATIO) - TALL_SCALE);
|
|
}
|
|
|
|
g_worldMapOffset = std::clamp((aspectRatio - NARROW_ASPECT_RATIO) / (WIDE_ASPECT_RATIO - NARROW_ASPECT_RATIO), 0.0f, 1.0f);
|
|
g_ultrawide = aspectRatio > WIDE_ASPECT_RATIO;
|
|
}
|
|
|
|
static class SDLEventListenerForCSD : public SDLEventListener
|
|
{
|
|
public:
|
|
void OnSDLEvent(SDL_Event* event) override
|
|
{
|
|
if (event->type == SDL_WINDOWEVENT && event->window.event == SDL_WINDOWEVENT_RESIZED)
|
|
ComputeOffsets(event->window.data1, event->window.data2);
|
|
}
|
|
} g_sdlEventListenerForCSD;
|
|
|
|
// Chao::CSD::SetOffsets
|
|
PPC_FUNC_IMPL(__imp__sub_830C0A78);
|
|
PPC_FUNC(sub_830C0A78)
|
|
{
|
|
__imp__sub_830C0A78(ctx, base);
|
|
|
|
ComputeOffsets(GameWindow::s_width, GameWindow::s_height);
|
|
}
|
|
|
|
// SWA::CGameDocument::ComputeScreenPosition
|
|
PPC_FUNC_IMPL(__imp__sub_8250FC70);
|
|
PPC_FUNC(sub_8250FC70)
|
|
{
|
|
__imp__sub_8250FC70(ctx, base);
|
|
|
|
auto position = reinterpret_cast<be<float>*>(base + ctx.r3.u32);
|
|
position[0] = position[0] - g_offsetX;
|
|
position[1] = position[1] - g_offsetY;
|
|
}
|
|
|
|
void ComputeScreenPositionMidAsmHook(PPCRegister& f1, PPCRegister& f2)
|
|
{
|
|
f1.f64 -= g_offsetX;
|
|
f2.f64 -= g_offsetY;
|
|
}
|
|
|
|
void WorldMapInfoMidAsmHook(PPCRegister& r4)
|
|
{
|
|
// Prevent the game from snapping "cts_parts_sun_moon"
|
|
// to "cts_guide_icon" automatically, we will do this ourselves.
|
|
r4.u32 = 0x8200A621;
|
|
}
|
|
|
|
// SWA::CTitleStateWorldMap::Update
|
|
PPC_FUNC_IMPL(__imp__sub_8258B558);
|
|
PPC_FUNC(sub_8258B558)
|
|
{
|
|
auto r3 = ctx.r3;
|
|
__imp__sub_8258B558(ctx, base);
|
|
|
|
uint32_t worldMapSimpleInfo = PPC_LOAD_U32(r3.u32 + 0x70);
|
|
|
|
auto setPosition = [&](uint32_t rcPtr, float offsetX = 0.0f, float offsetY = 0.0f)
|
|
{
|
|
uint32_t scene = PPC_LOAD_U32(rcPtr + 0x4);
|
|
if (scene != NULL)
|
|
{
|
|
scene = PPC_LOAD_U32(scene + 0x4);
|
|
if (scene != NULL)
|
|
{
|
|
ctx.r3.u32 = scene;
|
|
ctx.f1.f64 = offsetX + g_worldMapOffset * 140.0f;
|
|
ctx.f2.f64 = offsetY;
|
|
sub_830BB3D0(ctx, base);
|
|
}
|
|
}
|
|
};
|
|
|
|
setPosition(worldMapSimpleInfo + 0x2C, 299.0f, -178.0f);
|
|
setPosition(worldMapSimpleInfo + 0x34);
|
|
setPosition(worldMapSimpleInfo + 0x4C);
|
|
|
|
for (uint32_t it = PPC_LOAD_U32(worldMapSimpleInfo + 0x20); it != PPC_LOAD_U32(worldMapSimpleInfo + 0x24); it += 8)
|
|
setPosition(it);
|
|
|
|
uint32_t menuTextBox = PPC_LOAD_U32(worldMapSimpleInfo + 0x5C);
|
|
if (menuTextBox != NULL)
|
|
{
|
|
uint32_t textBox = PPC_LOAD_U32(menuTextBox + 0x4);
|
|
if (textBox != NULL)
|
|
{
|
|
float value = 708.0f + g_worldMapOffset * 140.0f;
|
|
PPC_STORE_U32(textBox + 0x38, reinterpret_cast<uint32_t&>(value));
|
|
}
|
|
}
|
|
}
|
|
|
|
enum
|
|
{
|
|
ALIGN_CENTER = 0 << 0,
|
|
|
|
ALIGN_TOP = 1 << 0,
|
|
ALIGN_LEFT = 1 << 1,
|
|
ALIGN_BOTTOM = 1 << 2,
|
|
ALIGN_RIGHT = 1 << 3,
|
|
|
|
ALIGN_TOP_LEFT = ALIGN_TOP | ALIGN_LEFT,
|
|
ALIGN_TOP_RIGHT = ALIGN_TOP | ALIGN_RIGHT,
|
|
ALIGN_BOTTOM_LEFT = ALIGN_BOTTOM | ALIGN_LEFT,
|
|
ALIGN_BOTTOM_RIGHT = ALIGN_BOTTOM | ALIGN_RIGHT,
|
|
|
|
STRETCH_HORIZONTAL = 1 << 4,
|
|
STRETCH_VERTICAL = 1 << 5,
|
|
|
|
STRETCH = STRETCH_HORIZONTAL | STRETCH_VERTICAL,
|
|
|
|
SCALE = 1 << 6,
|
|
|
|
WORLD_MAP = 1 << 7,
|
|
|
|
EXTEND_LEFT = 1 << 8,
|
|
EXTEND_RIGHT = 1 << 9,
|
|
|
|
STORE_LEFT_CORNER = 1 << 10,
|
|
STORE_RIGHT_CORNER = 1 << 11,
|
|
|
|
SKIP = 1 << 12,
|
|
|
|
OFFSET_SCALE_LEFT = 1 << 13,
|
|
OFFSET_SCALE_RIGHT = 1 << 14
|
|
};
|
|
|
|
struct CsdModifier
|
|
{
|
|
uint32_t flags{};
|
|
float cornerMax{};
|
|
};
|
|
|
|
static const ankerl::unordered_dense::map<XXH64_hash_t, CsdModifier> g_modifiers =
|
|
{
|
|
// ui_balloon
|
|
{ HashStr("ui_balloon/window/bg"), { STRETCH } },
|
|
{ HashStr("ui_balloon/window/footer"), { ALIGN_BOTTOM } },
|
|
|
|
// ui_boss_gauge
|
|
{ HashStr("ui_boss_gauge/gauge_bg"), { ALIGN_TOP_RIGHT | SCALE } },
|
|
{ HashStr("ui_boss_gauge/gauge_2"), { ALIGN_TOP_RIGHT | SCALE } },
|
|
{ HashStr("ui_boss_gauge/gauge_1"), { ALIGN_TOP_RIGHT | SCALE } },
|
|
{ HashStr("ui_boss_gauge/gauge_breakpoint"), { ALIGN_TOP_RIGHT | SCALE } },
|
|
|
|
// ui_exstage
|
|
{ HashStr("ui_exstage/shield/L_gauge"), { ALIGN_BOTTOM_LEFT | SCALE } },
|
|
{ HashStr("ui_exstage/shield/L_gauge_effect"), { ALIGN_BOTTOM_LEFT | SCALE } },
|
|
{ HashStr("ui_exstage/shield/L_gauge_effect_2"), { ALIGN_BOTTOM_LEFT | SCALE } },
|
|
{ HashStr("ui_exstage/energy/R_gauge"), { ALIGN_BOTTOM_RIGHT | SCALE } },
|
|
{ HashStr("ui_exstage/energy/R_gauge_effect"), { ALIGN_BOTTOM_RIGHT | SCALE } },
|
|
{ HashStr("ui_exstage/energy/R_gauge_effect_2"), { ALIGN_BOTTOM_RIGHT | SCALE } },
|
|
{ HashStr("ui_exstage/hit/hit_counter_bg"), { ALIGN_RIGHT | SCALE } },
|
|
{ HashStr("ui_exstage/hit/hit_counter_num"), { ALIGN_RIGHT | SCALE } },
|
|
|
|
// ui_gate
|
|
{ HashStr("ui_gate/footer/status_footer"), { ALIGN_BOTTOM } },
|
|
{ HashStr("ui_gate/header/status_title"), { ALIGN_TOP } },
|
|
{ HashStr("ui_gate/window/window_bg"), { STRETCH } },
|
|
|
|
// ui_general
|
|
{ HashStr("ui_general/bg"), { STRETCH } },
|
|
{ HashStr("ui_general/footer"), { ALIGN_BOTTOM } },
|
|
|
|
// ui_itemresult
|
|
{ HashStr("ui_itemresult/footer/result_footer"), { ALIGN_BOTTOM } },
|
|
{ HashStr("ui_itemresult/main/iresult_title"), { ALIGN_TOP } },
|
|
|
|
// ui_loading
|
|
{ HashStr("ui_loading/bg_1"), { STRETCH } },
|
|
{ HashStr("ui_loading/bg_2"), { STRETCH } },
|
|
|
|
// ui_missionscreen
|
|
{ HashStr("ui_missionscreen/player_count"), { ALIGN_TOP_LEFT | SCALE } },
|
|
{ HashStr("ui_missionscreen/time_count"), { ALIGN_TOP_LEFT | SCALE } },
|
|
{ HashStr("ui_missionscreen/score_count"), { ALIGN_TOP_LEFT | SCALE } },
|
|
{ HashStr("ui_missionscreen/item_count"), { ALIGN_TOP_LEFT | SCALE } },
|
|
{ HashStr("ui_missionscreen/laptime_count"), { ALIGN_TOP_LEFT | SCALE } },
|
|
{ HashStr("ui_missionscreen/lap_count"), { ALIGN_TOP_RIGHT | SCALE } },
|
|
|
|
// ui_misson
|
|
{ HashStr("ui_misson/bg"), { STRETCH } },
|
|
{ HashStr("ui_misson/footer/footer_B"), { ALIGN_BOTTOM } },
|
|
{ HashStr("ui_misson/header/misson_title_B"), { ALIGN_TOP } },
|
|
{ HashStr("ui_misson/window/bg_B2/position/bg"), { STRETCH } },
|
|
|
|
// ui_pause
|
|
{ HashStr("ui_pause/bg"), { STRETCH } },
|
|
{ HashStr("ui_pause/footer/footer_A"), { ALIGN_BOTTOM } },
|
|
{ HashStr("ui_pause/footer/footer_B"), { ALIGN_BOTTOM } },
|
|
{ HashStr("ui_pause/header/status_title"), { ALIGN_TOP | OFFSET_SCALE_LEFT, 585.0f } },
|
|
{ HashStr("ui_pause/header/status_title/title_bg/center"), { ALIGN_TOP | EXTEND_LEFT } },
|
|
{ HashStr("ui_pause/header/status_title/title_bg/center/h_light"), { ALIGN_TOP | EXTEND_LEFT} },
|
|
{ HashStr("ui_pause/header/status_title/title_bg/right"), { ALIGN_TOP | STORE_RIGHT_CORNER } },
|
|
|
|
// ui_playscreen
|
|
{ HashStr("ui_playscreen/player_count"), { ALIGN_TOP_LEFT | SCALE } },
|
|
{ HashStr("ui_playscreen/time_count"), { ALIGN_TOP_LEFT | SCALE } },
|
|
{ HashStr("ui_playscreen/score_count"), { ALIGN_TOP_LEFT | SCALE } },
|
|
{ HashStr("ui_playscreen/exp_count"), { ALIGN_TOP_LEFT | SCALE } },
|
|
{ HashStr("ui_playscreen/so_speed_gauge"), { ALIGN_BOTTOM_LEFT | SCALE } },
|
|
{ HashStr("ui_playscreen/so_ringenagy_gauge"), { ALIGN_BOTTOM_LEFT | SCALE } },
|
|
{ HashStr("ui_playscreen/gauge_frame"), { ALIGN_BOTTOM_LEFT | SCALE } },
|
|
{ HashStr("ui_playscreen/ring_count"), { ALIGN_BOTTOM_LEFT | SCALE } },
|
|
{ HashStr("ui_playscreen/ring_get"), { ALIGN_BOTTOM_LEFT | SCALE } },
|
|
{ HashStr("ui_playscreen/add/speed_count"), { ALIGN_RIGHT | SCALE } },
|
|
{ HashStr("ui_playscreen/add/u_info"), { ALIGN_BOTTOM_RIGHT | SCALE } },
|
|
{ HashStr("ui_playscreen/add/medal_get_s"), { ALIGN_BOTTOM_RIGHT | SCALE } },
|
|
{ HashStr("ui_playscreen/add/medal_get_m"), { ALIGN_BOTTOM_RIGHT | SCALE } },
|
|
|
|
// ui_playscreen_ev
|
|
{ HashStr("ui_playscreen_ev/player_count"), { ALIGN_TOP_LEFT | SCALE } },
|
|
{ HashStr("ui_playscreen_ev/score_count"), { ALIGN_TOP_LEFT | SCALE } },
|
|
{ HashStr("ui_playscreen_ev/ring_count"), { ALIGN_TOP_LEFT | SCALE } },
|
|
{ HashStr("ui_playscreen_ev/ring_get"), { ALIGN_TOP_LEFT | SCALE } },
|
|
{ HashStr("ui_playscreen_ev/exp_count"), { ALIGN_TOP_LEFT | SCALE } },
|
|
{ HashStr("ui_playscreen_ev/add/u_info"), { ALIGN_BOTTOM_RIGHT | SCALE } },
|
|
{ HashStr("ui_playscreen_ev/add/medal_get_s"), { ALIGN_BOTTOM_RIGHT | SCALE } },
|
|
{ HashStr("ui_playscreen_ev/add/medal_get_m"), { ALIGN_BOTTOM_RIGHT | SCALE } },
|
|
{ HashStr("ui_playscreen_ev/gauge/unleash_bg"), { ALIGN_BOTTOM_LEFT | SCALE } },
|
|
{ HashStr("ui_playscreen_ev/gauge/life_bg"), { ALIGN_BOTTOM_LEFT | SCALE } },
|
|
{ HashStr("ui_playscreen_ev/gauge/unleash_body"), { ALIGN_BOTTOM_LEFT | SCALE } },
|
|
{ HashStr("ui_playscreen_ev/gauge/unleash_bar_1"), { ALIGN_BOTTOM_LEFT | SCALE } },
|
|
{ HashStr("ui_playscreen_ev/gauge/unleash_gauge"), { ALIGN_BOTTOM_LEFT | SCALE } },
|
|
{ HashStr("ui_playscreen_ev/gauge/unleash_gauge_effect_2"), { ALIGN_BOTTOM_LEFT | SCALE } },
|
|
{ HashStr("ui_playscreen_ev/gauge/unleash_gauge_effect"), { ALIGN_BOTTOM_LEFT | SCALE } },
|
|
{ HashStr("ui_playscreen_ev/gauge/unleash_bar_2"), { ALIGN_BOTTOM_LEFT | SCALE } },
|
|
{ HashStr("ui_playscreen_ev/gauge/life"), { ALIGN_BOTTOM_LEFT | SCALE } },
|
|
{ HashStr("ui_playscreen_ev/gauge/shield_position"), { ALIGN_BOTTOM_LEFT | SCALE } },
|
|
{ HashStr("ui_playscreen_ev/gauge/shield/shield_01"), { ALIGN_BOTTOM_LEFT | SCALE } },
|
|
{ HashStr("ui_playscreen_ev/gauge/shield/shield_02"), { ALIGN_BOTTOM_LEFT | SCALE } },
|
|
{ HashStr("ui_playscreen_ev/gauge/shield/shield_03"), { ALIGN_BOTTOM_LEFT | SCALE } },
|
|
{ HashStr("ui_playscreen_ev/gauge/shield/shield_04"), { ALIGN_BOTTOM_LEFT | SCALE } },
|
|
{ HashStr("ui_playscreen_ev/gauge/shield/shield_05"), { ALIGN_BOTTOM_LEFT | SCALE } },
|
|
{ HashStr("ui_playscreen_ev/gauge/shield/shield_06"), { ALIGN_BOTTOM_LEFT | SCALE } },
|
|
{ HashStr("ui_playscreen_ev/gauge/shield/shield_07"), { ALIGN_BOTTOM_LEFT | SCALE } },
|
|
{ HashStr("ui_playscreen_ev/gauge/shield/shield_08"), { ALIGN_BOTTOM_LEFT | SCALE } },
|
|
{ HashStr("ui_playscreen_ev/gauge/shield/shield_09"), { ALIGN_BOTTOM_LEFT | SCALE } },
|
|
{ HashStr("ui_playscreen_ev/gauge/shield/shield_10"), { ALIGN_BOTTOM_LEFT | SCALE } },
|
|
{ HashStr("ui_playscreen_ev/gauge/shield/shield_11"), { ALIGN_BOTTOM_LEFT | SCALE } },
|
|
{ HashStr("ui_playscreen_ev/gauge/shield/shield_12"), { ALIGN_BOTTOM_LEFT | SCALE } },
|
|
{ HashStr("ui_playscreen_ev/gauge/shield/shield_13"), { ALIGN_BOTTOM_LEFT | SCALE } },
|
|
{ HashStr("ui_playscreen_ev/gauge/shield/shield_14"), { ALIGN_BOTTOM_LEFT | SCALE } },
|
|
{ HashStr("ui_playscreen_ev/gauge/shield/shield_15"), { ALIGN_BOTTOM_LEFT | SCALE } },
|
|
|
|
// ui_playscreen_ev_hit
|
|
{ HashStr("ui_playscreen_ev_hit/hit_counter_bg"), { ALIGN_RIGHT | SCALE } },
|
|
{ HashStr("ui_playscreen_ev_hit/hit_counter_num"), { ALIGN_RIGHT | SCALE } },
|
|
{ HashStr("ui_playscreen_ev_hit/hit_counter_txt_1"), { ALIGN_RIGHT | SCALE } },
|
|
{ HashStr("ui_playscreen_ev_hit/hit_counter_txt_2"), { ALIGN_RIGHT | SCALE } },
|
|
{ HashStr("ui_playscreen_ev_hit/chance_attack"), { ALIGN_RIGHT | SCALE } },
|
|
|
|
// ui_playscreen_su
|
|
{ HashStr("ui_playscreen_su/su_sonic_gauge"), { ALIGN_BOTTOM_LEFT | SCALE } },
|
|
{ HashStr("ui_playscreen_su/gaia_gauge"), { ALIGN_BOTTOM_LEFT | SCALE } },
|
|
{ HashStr("ui_playscreen_su/footer"), { ALIGN_BOTTOM_RIGHT | SCALE } },
|
|
|
|
// ui_prov_playscreen
|
|
{ HashStr("ui_prov_playscreen/so_speed_gauge"), { ALIGN_BOTTOM_LEFT | SCALE } },
|
|
{ HashStr("ui_prov_playscreen/so_ringenagy_gauge"), { ALIGN_BOTTOM_LEFT | SCALE } },
|
|
{ HashStr("ui_prov_playscreen/bg"), { ALIGN_TOP_LEFT | SCALE } },
|
|
{ HashStr("ui_prov_playscreen/info_1"), { ALIGN_TOP_LEFT | SCALE } },
|
|
{ HashStr("ui_prov_playscreen/info_2"), { ALIGN_TOP_LEFT | SCALE } },
|
|
{ HashStr("ui_prov_playscreen/ring_get_effect"), { ALIGN_BOTTOM_LEFT | SCALE } },
|
|
|
|
// ui_result
|
|
{ HashStr("ui_result/footer/result_footer"), { ALIGN_BOTTOM } },
|
|
{ HashStr("ui_result/main/result_title"), { ALIGN_TOP } },
|
|
|
|
// ui_result_ex
|
|
{ HashStr("ui_result_ex/footer/result_footer"), { ALIGN_BOTTOM } },
|
|
{ HashStr("ui_result_ex/main/result_title"), { ALIGN_TOP } },
|
|
|
|
// ui_shop
|
|
{ HashStr("ui_shop/footer/shop_footer"), { ALIGN_BOTTOM } },
|
|
{ HashStr("ui_shop/header/ring"), { ALIGN_TOP } },
|
|
{ HashStr("ui_shop/header/shop_nametag"), { ALIGN_TOP } },
|
|
|
|
// ui_start
|
|
{ HashStr("ui_start/Clear/position/bg/bg_1"), { STRETCH } },
|
|
{ HashStr("ui_start/Clear/position/bg/bg_2"), { STRETCH } },
|
|
{ HashStr("ui_start/Start/img/bg/bg_1"), { STRETCH } },
|
|
{ HashStr("ui_start/Start/img/bg/bg_2"), { STRETCH } },
|
|
|
|
// ui_status
|
|
{ HashStr("ui_status/footer/status_footer"), { ALIGN_BOTTOM } },
|
|
{ HashStr("ui_status/header/status_title"), { ALIGN_TOP | OFFSET_SCALE_LEFT, 617.0f } },
|
|
{ HashStr("ui_status/header/status_title/title_bg/center"), { ALIGN_TOP | EXTEND_LEFT } },
|
|
{ HashStr("ui_status/header/status_title/title_bg/center/h_light"), { ALIGN_TOP | EXTEND_LEFT } },
|
|
{ HashStr("ui_status/header/status_title/title_bg/right"), { ALIGN_TOP | STORE_RIGHT_CORNER } },
|
|
{ HashStr("ui_status/logo/logo/bg_position/c_1"), { STRETCH_HORIZONTAL } },
|
|
{ HashStr("ui_status/logo/logo/bg_position/c_2"), { STRETCH_HORIZONTAL } },
|
|
{ HashStr("ui_status/main/progless/bg/prgs_bg_1"), { OFFSET_SCALE_LEFT, 714.0f } },
|
|
{ HashStr("ui_status/main/progless/bg/prgs_bg_1/position/center/right"), { STORE_RIGHT_CORNER } },
|
|
{ HashStr("ui_status/main/progless/prgs/prgs_bar_1"), { OFFSET_SCALE_LEFT, 586.0f } },
|
|
{ HashStr("ui_status/main/progless/prgs/prgs_bar_1/position/bg/center/right"), { STORE_RIGHT_CORNER } },
|
|
{ HashStr("ui_status/main/tag/bg/tag_bg_1"), { OFFSET_SCALE_LEFT, 413.0f } },
|
|
{ HashStr("ui_status/main/tag/bg/tag_bg_1/total_1_bg/center"), { EXTEND_LEFT } },
|
|
{ HashStr("ui_status/main/tag/bg/tag_bg_1/total_1_bg/center/h_light"), { EXTEND_LEFT } },
|
|
{ HashStr("ui_status/main/tag/bg/tag_bg_1/total_1_bg/center/right"), { STORE_RIGHT_CORNER } },
|
|
{ HashStr("ui_status/main/tag/bg/tag_bg_1/total_1_bg/center/left"), { SKIP } },
|
|
{ HashStr("ui_status/main/tag/bg/tag_bg_1/total_1_bg/center/left/h_light"), { SKIP } },
|
|
{ HashStr("ui_status/main/tag/txt/tag_txt_1"), { OFFSET_SCALE_LEFT, 352.0f } },
|
|
{ HashStr("ui_status/main/tag/txt/tag_txt_1/position/img"), { STORE_RIGHT_CORNER } },
|
|
{ HashStr("ui_status/window/bg"), { STRETCH } },
|
|
|
|
// ui_title
|
|
{ HashStr("ui_title/bg/bg"), { STRETCH } },
|
|
{ HashStr("ui_title/bg/headr"), { ALIGN_TOP | STRETCH_HORIZONTAL } },
|
|
{ HashStr("ui_title/bg/footer"), { ALIGN_BOTTOM | STRETCH_HORIZONTAL } },
|
|
{ HashStr("ui_title/bg/position"), { ALIGN_BOTTOM } },
|
|
|
|
// ui_townscreen
|
|
{ HashStr("ui_townscreen/time"), { ALIGN_TOP_RIGHT | SCALE } },
|
|
{ HashStr("ui_townscreen/time_effect"), { ALIGN_TOP_RIGHT | SCALE } },
|
|
{ HashStr("ui_townscreen/info"), { ALIGN_TOP_LEFT | SCALE } },
|
|
{ HashStr("ui_townscreen/cam"), { ALIGN_TOP_RIGHT | SCALE } },
|
|
{ HashStr("ui_townscreen/footer"), { ALIGN_BOTTOM } },
|
|
|
|
// ui_worldmap
|
|
{ HashStr("ui_worldmap/contents/choices/cts_choices_bg"), { STRETCH } },
|
|
{ HashStr("ui_worldmap/contents/info/bg/cts_info_bg"), { ALIGN_TOP_LEFT | WORLD_MAP } },
|
|
{ HashStr("ui_worldmap/contents/info/bg/info_bg_1"), { ALIGN_TOP_LEFT | WORLD_MAP } },
|
|
{ HashStr("ui_worldmap/contents/info/img/info_img_1"), { ALIGN_TOP_LEFT | WORLD_MAP } },
|
|
{ HashStr("ui_worldmap/contents/info/img/info_img_2"), { ALIGN_TOP_LEFT | WORLD_MAP } },
|
|
{ HashStr("ui_worldmap/contents/info/img/info_img_3"), { ALIGN_TOP_LEFT | WORLD_MAP } },
|
|
{ HashStr("ui_worldmap/contents/info/img/info_img_4"), { ALIGN_TOP_LEFT | WORLD_MAP } },
|
|
{ HashStr("ui_worldmap/footer/worldmap_footer_bg"), { ALIGN_BOTTOM } },
|
|
{ HashStr("ui_worldmap/footer/worldmap_footer_img_A"), { ALIGN_BOTTOM } },
|
|
{ HashStr("ui_worldmap/header/worldmap_header_bg"), { ALIGN_TOP } },
|
|
{ HashStr("ui_worldmap/header/worldmap_header_img"), { ALIGN_TOP_LEFT | WORLD_MAP } }
|
|
};
|
|
|
|
// Reserve space in managed scene class to store the corner in.
|
|
// Scenes can be duplicated, so we can't rely on storing them by YNCP scene pointer.
|
|
static constexpr float UNINITIALIZED_CORNER = FLT_MAX;
|
|
|
|
struct SceneEx
|
|
{
|
|
float corner = UNINITIALIZED_CORNER;
|
|
float cornerNew{};
|
|
uint64_t frame{};
|
|
};
|
|
|
|
static SceneEx* g_sceneEx = nullptr;
|
|
|
|
void SceneByteSizeMidAsmHook(PPCRegister& r4)
|
|
{
|
|
r4.u32 += sizeof(SceneEx);
|
|
}
|
|
|
|
// Chao::CSD::CScene::CScene
|
|
PPC_FUNC_IMPL(__imp__sub_830BE7F8);
|
|
PPC_FUNC(sub_830BE7F8)
|
|
{
|
|
new (base + ctx.r3.u32 + 0xE0) SceneEx();
|
|
__imp__sub_830BE7F8(ctx, base);
|
|
}
|
|
|
|
static std::optional<CsdModifier> FindModifier(uint32_t data)
|
|
{
|
|
XXH64_hash_t path;
|
|
{
|
|
std::lock_guard lock(g_pathMutex);
|
|
|
|
auto findResult = g_paths.find(g_memory.Translate(data));
|
|
if (findResult == g_paths.end())
|
|
return {};
|
|
|
|
path = findResult->second;
|
|
}
|
|
|
|
auto findResult = g_modifiers.find(path);
|
|
if (findResult != g_modifiers.end())
|
|
return findResult->second;
|
|
|
|
return {};
|
|
}
|
|
|
|
static std::optional<CsdModifier> g_sceneModifier;
|
|
|
|
// Chao::CSD::CScene::Render
|
|
PPC_FUNC_IMPL(__imp__sub_830BC640);
|
|
PPC_FUNC(sub_830BC640)
|
|
{
|
|
g_sceneModifier = FindModifier(PPC_LOAD_U32(ctx.r3.u32 + 0xC));
|
|
g_sceneEx = reinterpret_cast<SceneEx*>(base + ctx.r3.u32 + 0xE0);
|
|
|
|
if (g_sceneModifier.has_value() && ((g_sceneEx->frame + 1) != App::s_frame || g_sceneEx->corner == UNINITIALIZED_CORNER))
|
|
g_sceneEx->corner = (g_sceneModifier->flags & OFFSET_SCALE_LEFT) != 0 ? -100000.0f : 100000.0f;
|
|
|
|
__imp__sub_830BC640(ctx, base);
|
|
|
|
#if 1
|
|
if (g_sceneModifier.has_value() && (g_sceneModifier->flags & (OFFSET_SCALE_LEFT | OFFSET_SCALE_RIGHT)) != 0 && g_sceneModifier->cornerMax == 0.0f)
|
|
fmt::println("Corner: {}", g_sceneEx->cornerNew);
|
|
#endif
|
|
|
|
g_sceneEx->corner = g_sceneEx->cornerNew;
|
|
g_sceneEx->frame = App::s_frame;
|
|
|
|
g_sceneEx = nullptr;
|
|
}
|
|
|
|
static std::optional<CsdModifier> g_castNodeModifier;
|
|
|
|
void RenderCsdCastNodeMidAsmHook(PPCRegister& r10, PPCRegister& r27)
|
|
{
|
|
g_castNodeModifier = FindModifier(r10.u32 + r27.u32);
|
|
}
|
|
|
|
static std::optional<CsdModifier> g_castModifier;
|
|
|
|
void RenderCsdCastMidAsmHook(PPCRegister& r4)
|
|
{
|
|
g_castModifier = FindModifier(r4.u32);
|
|
}
|
|
|
|
static void Draw(PPCContext& ctx, uint8_t* base, PPCFunc* original, uint32_t stride)
|
|
{
|
|
CsdModifier modifier{};
|
|
|
|
if (g_castModifier.has_value())
|
|
{
|
|
modifier = g_castModifier.value();
|
|
}
|
|
else if (g_castNodeModifier.has_value())
|
|
{
|
|
modifier = g_castNodeModifier.value();
|
|
}
|
|
else if (g_sceneModifier.has_value())
|
|
{
|
|
modifier = g_sceneModifier.value();
|
|
}
|
|
|
|
if ((modifier.flags & SKIP) != 0)
|
|
return;
|
|
|
|
if (Config::UIScaleMode == EUIScaleMode::Centre && (modifier.flags & SCALE) != 0)
|
|
modifier.flags &= ~(ALIGN_TOP | ALIGN_LEFT | ALIGN_BOTTOM | ALIGN_RIGHT);
|
|
|
|
auto backBuffer = Video::GetBackBuffer();
|
|
|
|
uint32_t size = ctx.r5.u32 * stride;
|
|
ctx.r1.u32 -= size;
|
|
|
|
uint8_t* stack = base + ctx.r1.u32;
|
|
memcpy(stack, base + ctx.r4.u32, size);
|
|
|
|
float offsetX = 0.0f;
|
|
float offsetY = 0.0f;
|
|
float scaleX = 1.0f;
|
|
float scaleY = 1.0f;
|
|
|
|
if ((modifier.flags & STRETCH_HORIZONTAL) != 0)
|
|
{
|
|
scaleX = backBuffer->width / 1280.0f;
|
|
}
|
|
else
|
|
{
|
|
if ((modifier.flags & ALIGN_RIGHT) != 0)
|
|
offsetX = g_offsetX * 2.0f;
|
|
else if ((modifier.flags & ALIGN_LEFT) == 0)
|
|
offsetX = g_offsetX;
|
|
|
|
if ((modifier.flags & SCALE) != 0)
|
|
{
|
|
scaleX = g_scale;
|
|
|
|
if ((modifier.flags & ALIGN_RIGHT) != 0)
|
|
offsetX += 1280.0f * (1.0f - scaleX);
|
|
else if ((modifier.flags & ALIGN_LEFT) == 0)
|
|
offsetX += 640.0f * (1.0f - scaleX);
|
|
}
|
|
|
|
if ((modifier.flags & WORLD_MAP) != 0)
|
|
{
|
|
if ((modifier.flags & ALIGN_LEFT) != 0)
|
|
offsetX += (1.0f - g_worldMapOffset) * -20.0f;
|
|
}
|
|
}
|
|
|
|
if ((modifier.flags & STRETCH_VERTICAL) != 0)
|
|
{
|
|
scaleY = backBuffer->height / 720.0f;
|
|
}
|
|
else
|
|
{
|
|
if ((modifier.flags & ALIGN_BOTTOM) != 0)
|
|
offsetY = g_offsetY * 2.0f;
|
|
else if ((modifier.flags & ALIGN_TOP) == 0)
|
|
offsetY = g_offsetY;
|
|
|
|
if ((modifier.flags & SCALE) != 0)
|
|
{
|
|
scaleY = g_scale;
|
|
|
|
if ((modifier.flags & ALIGN_BOTTOM) != 0)
|
|
offsetY += 720.0f * (1.0f - scaleY);
|
|
else if ((modifier.flags & ALIGN_TOP) == 0)
|
|
offsetY += 360.0f * (1.0f - scaleY);
|
|
}
|
|
}
|
|
|
|
if (g_sceneEx != nullptr && g_sceneModifier.has_value())
|
|
{
|
|
if ((modifier.flags & (STORE_LEFT_CORNER | STORE_RIGHT_CORNER)) != 0)
|
|
{
|
|
uint32_t vertexIndex = ((modifier.flags & STORE_LEFT_CORNER) != 0) ? 0 : 3;
|
|
g_sceneEx->cornerNew = *reinterpret_cast<be<float>*>(stack + vertexIndex * stride);
|
|
}
|
|
|
|
if (g_ultrawide)
|
|
{
|
|
if ((g_sceneModifier->flags & OFFSET_SCALE_LEFT) != 0)
|
|
offsetX *= g_sceneEx->corner / g_sceneModifier->cornerMax;
|
|
}
|
|
}
|
|
|
|
for (size_t i = 0; i < ctx.r5.u32; i++)
|
|
{
|
|
auto position = reinterpret_cast<be<float>*>(stack + i * stride);
|
|
|
|
float x = offsetX + position[0] * scaleX;
|
|
float y = offsetY + position[1] * scaleY;
|
|
|
|
if ((modifier.flags & EXTEND_LEFT) != 0 && (i == 0 || i == 1))
|
|
{
|
|
x = std::min(x, 0.0f);
|
|
}
|
|
else if ((modifier.flags & EXTEND_RIGHT) != 0 && (i == 2 || i == 3))
|
|
{
|
|
x = std::max(x, float(backBuffer->width));
|
|
}
|
|
|
|
position[0] = round(x / backBuffer->width * GameWindow::s_width) / GameWindow::s_width * backBuffer->width;
|
|
position[1] = round(y / backBuffer->height * GameWindow::s_height) / GameWindow::s_height * backBuffer->height;
|
|
}
|
|
|
|
ctx.r4.u32 = ctx.r1.u32;
|
|
original(ctx, base);
|
|
ctx.r1.u32 += size;
|
|
}
|
|
|
|
// SWA::CCsdPlatformMirage::Draw
|
|
PPC_FUNC_IMPL(__imp__sub_825E2E70);
|
|
PPC_FUNC(sub_825E2E70)
|
|
{
|
|
Draw(ctx, base, __imp__sub_825E2E70, 0x14);
|
|
}
|
|
|
|
// SWA::CCsdPlatformMirage::DrawNoTex
|
|
PPC_FUNC_IMPL(__imp__sub_825E2E88);
|
|
PPC_FUNC(sub_825E2E88)
|
|
{
|
|
Draw(ctx, base, __imp__sub_825E2E88, 0xC);
|
|
}
|
|
|
|
// Hedgehog::MirageDebug::SetScissorRect
|
|
PPC_FUNC_IMPL(__imp__sub_82E16C70);
|
|
PPC_FUNC(sub_82E16C70)
|
|
{
|
|
auto backBuffer = Video::GetBackBuffer();
|
|
auto scissorRect = reinterpret_cast<GuestRect*>(base + ctx.r4.u32);
|
|
|
|
scissorRect->left = scissorRect->left + g_offsetX;
|
|
scissorRect->top = scissorRect->top + g_offsetY;
|
|
scissorRect->right = scissorRect->right + g_offsetX;
|
|
scissorRect->bottom = scissorRect->bottom + g_offsetY;
|
|
|
|
__imp__sub_82E16C70(ctx, base);
|
|
}
|
|
|
|
// Hedgehog::MirageDebug::CPrimitive2D::Draw
|
|
PPC_FUNC_IMPL(__imp__sub_830D1EF0);
|
|
PPC_FUNC(sub_830D1EF0)
|
|
{
|
|
__imp__sub_830D1EF0(ctx, base);
|
|
|
|
auto backBuffer = Video::GetBackBuffer();
|
|
auto position = reinterpret_cast<be<float>*>(base + ctx.r4.u32);
|
|
|
|
for (size_t i = 0; i < 4; i++)
|
|
{
|
|
position[0] = position[0] * 1280.0f / backBuffer->width;
|
|
position[1] = position[1] * 720.0f / backBuffer->height;
|
|
position += 7;
|
|
}
|
|
}
|