#include "Runtime/AutoMapper/CAutoMapper.hpp" #include "Runtime/CInGameTweakManagerBase.hpp" #include "Runtime/CSimplePool.hpp" #include "Runtime/GameGlobalObjects.hpp" #include "Runtime/AutoMapper/CMapArea.hpp" #include "Runtime/AutoMapper/CMapUniverse.hpp" #include "Runtime/Camera/CGameCamera.hpp" #include "Runtime/GuiSys/CGuiFrame.hpp" #include "Runtime/GuiSys/CGuiTextPane.hpp" #include "Runtime/GuiSys/CGuiWidgetDrawParms.hpp" #include "Runtime/Input/ControlMapper.hpp" #include "Runtime/MP1/MP1.hpp" #include "Runtime/Particle/CGenDescription.hpp" #include "Runtime/World/CPlayer.hpp" #include <zeus/CEulerAngles.hpp> namespace urde { void CAutoMapper::SAutoMapperRenderState::InterpolateWithClamp(const SAutoMapperRenderState& a, SAutoMapperRenderState& out, const SAutoMapperRenderState& b, float t) { t = zeus::clamp(0.f, t, 1.f); const float easeIn = zeus::clamp(0.f, t * t * t, 1.f); const float omt = 1.f - t; const float easeOut = zeus::clamp(0.f, 1.f - omt * omt * omt, 1.f); float easeInOut; if (t >= 0.5f) { easeInOut = zeus::clamp(0.f, 0.5f * std::sqrt(2.f * t - 1.f) + 0.5f, 1.f); } else { easeInOut = zeus::clamp(0.f, 1.f - (0.5f * std::sqrt(2.f * omt - 1.f) + 0.5f), 1.f); } const std::array<float, 5> eases{ 0.0f, t, easeOut, easeIn, easeInOut, }; if (b.x44_viewportEase != Ease::None) { const float easeB = eases[size_t(b.x44_viewportEase)]; const float easeA = 1.f - easeB; const zeus::CVector2i vpA = a.GetViewportSize(); const zeus::CVector2i vpB = b.GetViewportSize(); out.x0_viewportSize = zeus::CVector2i(vpB.x * easeB + vpA.x * easeA, vpB.y * easeB + vpA.y * easeA); } if (t == 1.f) out.m_getViewportSize = b.m_getViewportSize; else out.m_getViewportSize = nullptr; if (b.x48_camEase != Ease::None) { const float easeB = eases[size_t(b.x48_camEase)]; const float easeA = 1.f - easeB; out.x8_camOrientation = zeus::CQuaternion::slerp(a.x8_camOrientation, b.x8_camOrientation, easeB); out.x18_camDist = b.x18_camDist * easeB + a.x18_camDist * easeA; out.x1c_camAngle = b.x1c_camAngle * easeB + a.x1c_camAngle * easeA; } if (b.x4c_pointEase != Ease::None) { const float easeB = eases[size_t(b.x4c_pointEase)]; const float easeA = 1.f - easeB; out.x20_areaPoint = b.x20_areaPoint * easeB + a.x20_areaPoint * easeA; } if (b.x50_depth1Ease != Ease::None) { const float easeB = eases[size_t(b.x50_depth1Ease)]; const float easeA = 1.f - easeB; out.x2c_drawDepth1 = b.x2c_drawDepth1 * easeB + a.x2c_drawDepth1 * easeA; } if (b.x54_depth2Ease != Ease::None) { const float easeB = eases[size_t(b.x54_depth2Ease)]; const float easeA = 1.f - easeB; out.x30_drawDepth2 = b.x30_drawDepth2 * easeB + a.x30_drawDepth2 * easeA; } if (b.x58_alphaEase != Ease::None) { const float easeB = eases[size_t(b.x58_alphaEase)]; const float easeA = 1.f - easeB; out.x34_alphaSurfaceVisited = b.x34_alphaSurfaceVisited * easeB + a.x34_alphaSurfaceVisited * easeA; out.x38_alphaOutlineVisited = b.x38_alphaOutlineVisited * easeB + a.x38_alphaOutlineVisited * easeA; out.x3c_alphaSurfaceUnvisited = b.x3c_alphaSurfaceUnvisited * easeB + a.x3c_alphaSurfaceUnvisited * easeA; out.x40_alphaOutlineUnvisited = b.x40_alphaOutlineUnvisited * easeB + a.x40_alphaOutlineUnvisited * easeA; } } CAutoMapper::CAutoMapper(CStateManager& stateMgr) : x24_world(stateMgr.GetWorld()) { x8_mapu = g_SimplePool->GetObj("MAPU_MapUniverse"); x30_miniMapSamus = g_SimplePool->GetObj("CMDL_MiniMapSamus"); x3c_hintBeacon = g_SimplePool->GetObj("TXTR_HintBeacon"); xa0_curAreaId = xa4_otherAreaId = stateMgr.GetWorld()->IGetCurrentAreaId(); zeus::CMatrix3f camRot = stateMgr.GetCameraManager()->GetCurrentCamera(stateMgr)->GetTransform().buildMatrix3f(); xa8_renderStates[0] = xa8_renderStates[1] = xa8_renderStates[2] = BuildMiniMapWorldRenderState(stateMgr, camRot, xa0_curAreaId); x48_mapIcons.emplace_back(g_SimplePool->GetObj(SObjectTag{FOURCC('TXTR'), g_tweakPlayerRes->x4_saveStationIcon})); x48_mapIcons.emplace_back(g_SimplePool->GetObj(SObjectTag{FOURCC('TXTR'), g_tweakPlayerRes->x8_missileStationIcon})); x48_mapIcons.emplace_back(g_SimplePool->GetObj(SObjectTag{FOURCC('TXTR'), g_tweakPlayerRes->xc_elevatorIcon})); x48_mapIcons.emplace_back( g_SimplePool->GetObj(SObjectTag{FOURCC('TXTR'), g_tweakPlayerRes->x10_minesBreakFirstTopIcon})); x48_mapIcons.emplace_back( g_SimplePool->GetObj(SObjectTag{FOURCC('TXTR'), g_tweakPlayerRes->x14_minesBreakFirstBottomIcon})); for (u32 i = 0; i < 9; ++i) { x210_lstick.emplace_back(g_SimplePool->GetObj(SObjectTag{FOURCC('TXTR'), g_tweakPlayerRes->x24_lStick[i]})); x25c_cstick.emplace_back(g_SimplePool->GetObj(SObjectTag{FOURCC('TXTR'), g_tweakPlayerRes->x4c_cStick[i]})); } for (u32 i = 0; i < 2; ++i) { x2a8_ltrigger.emplace_back(g_SimplePool->GetObj(SObjectTag{FOURCC('TXTR'), g_tweakPlayerRes->x74_lTrigger[i]})); x2bc_rtrigger.emplace_back(g_SimplePool->GetObj(SObjectTag{FOURCC('TXTR'), g_tweakPlayerRes->x80_rTrigger[i]})); x2d0_abutton.emplace_back(g_SimplePool->GetObj(SObjectTag{FOURCC('TXTR'), g_tweakPlayerRes->x98_aButton[i]})); } } bool CAutoMapper::CheckLoadComplete() { switch (x4_loadPhase) { case ELoadPhase::LoadResources: for (TLockedToken<CTexture>& tex : x48_mapIcons) if (!tex.IsLoaded()) return false; if (!x30_miniMapSamus.IsLoaded()) return false; if (!x3c_hintBeacon.IsLoaded()) return false; x4_loadPhase = ELoadPhase::LoadUniverse; [[fallthrough]]; case ELoadPhase::LoadUniverse: if (!x8_mapu.IsLoaded()) return false; x14_dummyWorlds.resize(x8_mapu->GetNumMapWorldDatas()); SetCurWorldAssetId(x24_world->IGetWorldAssetId()); x4_loadPhase = ELoadPhase::Done; [[fallthrough]]; case ELoadPhase::Done: return true; default: break; } return false; } bool CAutoMapper::NotHintNavigating() const { return x1e0_hintSteps.empty(); } bool CAutoMapper::CanLeaveMapScreenInternal(const CStateManager& mgr) const { if (!NotHintNavigating()) return false; if (IsRenderStateInterpolating()) return false; if (IsInMapperState(EAutoMapperState::MapScreenUniverse)) return true; if (x24_world != mgr.GetWorld()) return false; if (IsInMapperState(EAutoMapperState::MapScreen)) return true; return false; } void CAutoMapper::LeaveMapScreen(CStateManager& mgr) { if (x1c0_nextState == EAutoMapperState::MapScreenUniverse) { xa8_renderStates[1].x2c_drawDepth1 = GetMapAreaMiniMapDrawDepth(); xa8_renderStates[1].x30_drawDepth2 = GetMapAreaMiniMapDrawDepth(); xa8_renderStates[0].x2c_drawDepth1 = GetMapAreaMiniMapDrawDepth(); xa8_renderStates[0].x30_drawDepth2 = GetMapAreaMiniMapDrawDepth(); SetupMiniMapWorld(mgr); } else { x328_ = 2; xa8_renderStates[1] = xa8_renderStates[0]; xa8_renderStates[2] = xa8_renderStates[1]; xa0_curAreaId = x24_world->IGetCurrentAreaId(); xa8_renderStates[1].x20_areaPoint = GetAreaPointOfInterest(mgr, xa0_curAreaId); xa8_renderStates[1].x4c_pointEase = SAutoMapperRenderState::Ease::Linear; xa8_renderStates[1].x2c_drawDepth1 = GetMapAreaMiniMapDrawDepth(); xa8_renderStates[1].x30_drawDepth2 = GetMapAreaMiniMapDrawDepth(); xa8_renderStates[1].x50_depth1Ease = SAutoMapperRenderState::Ease::Linear; xa8_renderStates[1].x54_depth2Ease = SAutoMapperRenderState::Ease::Linear; ResetInterpolationTimer(0.25f); } } void CAutoMapper::SetupMiniMapWorld(CStateManager& mgr) { CWorld& wld = *mgr.GetWorld(); wld.GetMapWorld()->SetWhichMapAreasLoaded(wld, wld.GetCurrentAreaId(), 3); x328_ = 3; } bool CAutoMapper::HasCurrentMapUniverseWorld() const { CAssetId mlvlId = x24_world->IGetWorldAssetId(); for (const CMapUniverse::CMapWorldData& wld : *x8_mapu) if (wld.GetWorldAssetId() == mlvlId) return true; return false; } bool CAutoMapper::CheckDummyWorldLoad(CStateManager& mgr) { const CMapUniverse::CMapWorldData& mapuWld = x8_mapu->GetMapWorldData(x9c_worldIdx); auto& dummyWorld = x14_dummyWorlds[x9c_worldIdx]; if (!dummyWorld) { x32c_loadingDummyWorld = false; return false; } if (!dummyWorld->ICheckWorldComplete()) return true; CWorldState& worldState = g_GameState->StateForWorld(dummyWorld->IGetWorldAssetId()); CMapWorldInfo& mwInfo = *worldState.MapWorldInfo(); zeus::CVector3f localPoint = mapuWld.GetWorldTransform().inverse() * xa8_renderStates[0].x20_areaPoint; zeus::CMatrix3f camRot = xa8_renderStates[0].x8_camOrientation.toTransform().buildMatrix3f(); TAreaId aid = FindClosestVisibleArea(localPoint, zeus::CUnitVector3f(camRot[1]), mgr, *dummyWorld, mwInfo); if (aid == -1) { x32c_loadingDummyWorld = false; return false; } xa0_curAreaId = aid; dummyWorld->IGetMapWorld()->RecalculateWorldSphere(mwInfo, *dummyWorld); BeginMapperStateTransition(EAutoMapperState::MapScreen, mgr); x32c_loadingDummyWorld = false; return true; } void CAutoMapper::UpdateHintNavigation(float dt, CStateManager& mgr) { SAutoMapperHintStep& nextStep = x1e0_hintSteps.front(); bool oldProcessing = nextStep.x8_processing; nextStep.x8_processing = true; switch (nextStep.x0_type) { case SAutoMapperHintStep::Type::PanToArea: { if (x24_world->IGetMapWorld()->GetMapArea(nextStep.x4_areaId)) { xa8_renderStates[2] = xa8_renderStates[0]; xa8_renderStates[1].x20_areaPoint = GetAreaPointOfInterest(mgr, nextStep.x4_areaId); xa8_renderStates[1].ResetInterpolation(); xa8_renderStates[1].x4c_pointEase = SAutoMapperRenderState::Ease::Linear; ResetInterpolationTimer(2.f * g_tweakAutoMapper->GetHintPanTime()); x1e0_hintSteps.pop_front(); } break; } case SAutoMapperHintStep::Type::PanToWorld: { const CMapUniverse::CMapWorldData& mwData = x8_mapu->GetMapWorldDataByWorldId(nextStep.x4_worldId); xa8_renderStates[2] = xa8_renderStates[0]; xa8_renderStates[1].x20_areaPoint = mwData.GetWorldCenterPoint(); xa8_renderStates[1].ResetInterpolation(); xa8_renderStates[1].x4c_pointEase = SAutoMapperRenderState::Ease::Linear; ResetInterpolationTimer(2.f * g_tweakAutoMapper->GetHintPanTime()); x1e0_hintSteps.pop_front(); break; } case SAutoMapperHintStep::Type::SwitchToUniverse: { if (HasCurrentMapUniverseWorld()) { BeginMapperStateTransition(EAutoMapperState::MapScreenUniverse, mgr); x1e0_hintSteps.pop_front(); } else { x1e0_hintSteps.clear(); } break; } case SAutoMapperHintStep::Type::SwitchToWorld: { x1e0_hintSteps.pop_front(); x32c_loadingDummyWorld = true; if (CheckDummyWorldLoad(mgr)) break; x1e0_hintSteps.clear(); break; } case SAutoMapperHintStep::Type::ShowBeacon: { if (!oldProcessing) { if (xa0_curAreaId == mgr.GetNextAreaId() && x24_world == mgr.GetWorld()) CSfxManager::SfxStart(SFXui_show_local_beacon, 1.f, 0.f, false, 0x7f, false, kInvalidAreaId); else CSfxManager::SfxStart(SFXui_show_remote_beacon, 1.f, 0.f, false, 0x7f, false, kInvalidAreaId); } nextStep.x4_float = std::max(0.f, nextStep.x4_float - dt); for (SAutoMapperHintLocation& loc : x1f8_hintLocations) { if (x24_world->IGetWorldAssetId() == loc.x8_worldId && xa0_curAreaId == loc.xc_areaId) { loc.x0_showBeacon = 1; loc.x4_beaconAlpha = 1.f - std::min(nextStep.x4_float / 0.5f, 1.f); break; } } if (nextStep.x4_float != 0.f) break; x1e0_hintSteps.pop_front(); break; } case SAutoMapperHintStep::Type::ZoomOut: { xa8_renderStates[2] = xa8_renderStates[0]; xa8_renderStates[1].x18_camDist = g_tweakAutoMapper->GetMaxCamDist(); xa8_renderStates[1].ResetInterpolation(); xa8_renderStates[1].x48_camEase = SAutoMapperRenderState::Ease::Linear; ResetInterpolationTimer(0.5f); x1e0_hintSteps.pop_front(); break; } case SAutoMapperHintStep::Type::ZoomIn: { xa8_renderStates[2] = xa8_renderStates[0]; xa8_renderStates[1].x18_camDist = g_tweakAutoMapper->GetCamDist(); xa8_renderStates[1].ResetInterpolation(); xa8_renderStates[1].x48_camEase = SAutoMapperRenderState::Ease::Linear; ResetInterpolationTimer(0.5f); x1e0_hintSteps.pop_front(); break; } default: break; } } bool CAutoMapper::CanLeaveMapScreen(const CStateManager& mgr) const { return x328_ == 3 && CanLeaveMapScreenInternal(mgr); } void CAutoMapper::SetCurWorldAssetId(CAssetId mlvlId) { u32 numWorlds = x8_mapu->GetNumMapWorldDatas(); for (u32 i = 0; i < numWorlds; ++i) if (x8_mapu->GetMapWorldData(i).GetWorldAssetId() == mlvlId) { x9c_worldIdx = i; break; } } void CAutoMapper::BeginMapperStateTransition(EAutoMapperState state, CStateManager& mgr) { if (state == x1c0_nextState) return; if ((state == EAutoMapperState::MiniMap && x1c0_nextState != EAutoMapperState::MiniMap) || (state != EAutoMapperState::MiniMap && x1c0_nextState == EAutoMapperState::MiniMap)) CSfxManager::KillAll(CSfxManager::ESfxChannels::PauseScreen); x1bc_state = x1c0_nextState; x1c0_nextState = state; xa8_renderStates[2] = xa8_renderStates[0]; xa8_renderStates[1] = xa8_renderStates[0]; if (x1bc_state == EAutoMapperState::MiniMap && state == EAutoMapperState::MapScreen) { xa8_renderStates[1] = BuildMapScreenWorldRenderState(mgr, xa8_renderStates[0].x8_camOrientation, xa0_curAreaId, false); ResetInterpolationTimer(g_tweakAutoMapper->GetOpenMapScreenTime()); } else if (x1bc_state == EAutoMapperState::MapScreen && state == EAutoMapperState::MiniMap) { xa0_curAreaId = x24_world->IGetCurrentAreaId(); xa8_renderStates[1] = BuildMiniMapWorldRenderState(mgr, xa8_renderStates[0].x8_camOrientation, xa0_curAreaId); ResetInterpolationTimer(g_tweakAutoMapper->GetCloseMapScreenTime()); x1f8_hintLocations.clear(); } else if (x1bc_state == EAutoMapperState::MapScreen && state == EAutoMapperState::MapScreenUniverse) { CSfxManager::SfxStart(SFXui_map_to_universe, 1.f, 0.f, false, 0x7f, false, kInvalidAreaId); xa8_renderStates[1] = BuildMapScreenUniverseRenderState(mgr, xa8_renderStates[0].x8_camOrientation, xa0_curAreaId); TransformRenderStatesWorldToUniverse(); ResetInterpolationTimer(g_tweakAutoMapper->GetSwitchToFromUniverseTime()); } else if (x1bc_state == EAutoMapperState::MapScreenUniverse && state == EAutoMapperState::MapScreen) { CSfxManager::SfxStart(SFXui_map_from_universe, 1.f, 0.f, false, 0x7f, false, kInvalidAreaId); xa8_renderStates[1] = BuildMapScreenWorldRenderState(mgr, xa8_renderStates[0].x8_camOrientation, xa0_curAreaId, x1e0_hintSteps.size()); TransformRenderStateWorldToUniverse(xa8_renderStates[1]); ResetInterpolationTimer(g_tweakAutoMapper->GetSwitchToFromUniverseTime()); for (auto& wld : x14_dummyWorlds) { if (wld.get() != x24_world || x24_world == mgr.GetWorld()) wld.reset(); } } else if (x1bc_state == EAutoMapperState::MapScreenUniverse && state == EAutoMapperState::MiniMap) { x24_world = mgr.GetWorld(); xa0_curAreaId = x24_world->IGetCurrentAreaId(); xa8_renderStates[1] = BuildMiniMapWorldRenderState(mgr, xa8_renderStates[0].x8_camOrientation, xa0_curAreaId); SetCurWorldAssetId(x24_world->IGetWorldAssetId()); TransformRenderStateWorldToUniverse(xa8_renderStates[1]); ResetInterpolationTimer(g_tweakAutoMapper->GetCloseMapScreenTime()); x1f8_hintLocations.clear(); for (auto& wld : x14_dummyWorlds) { if (wld.get() != x24_world || x24_world == mgr.GetWorld()) wld.reset(); } } } void CAutoMapper::CompleteMapperStateTransition(CStateManager& mgr) { if (x1bc_state == EAutoMapperState::MapScreenUniverse) TransformRenderStatesUniverseToWorld(); if (x1c0_nextState == EAutoMapperState::MapScreen) { const CMapWorldInfo& mwInfo = *g_GameState->StateForWorld(x24_world->IGetWorldAssetId()).MapWorldInfo(); x24_world->IGetMapWorld()->RecalculateWorldSphere(mwInfo, *x24_world); x1d8_flashTimer = 0.f; x1dc_playerFlashPulse = 0.f; } if (x1c0_nextState == EAutoMapperState::MiniMap) { x28_frmeMapScreen = TLockedToken<CGuiFrame>(); m_frmeInitialized = false; x2fc_textpane_hint = nullptr; x300_textpane_instructions = nullptr; x304_textpane_instructions1 = nullptr; x308_textpane_instructions2 = nullptr; x2f8_textpane_areaname = nullptr; x30c_basewidget_leftPane = nullptr; x310_basewidget_yButtonPane = nullptr; x314_basewidget_bottomPane = nullptr; SetResLockState(x210_lstick, false); SetResLockState(x25c_cstick, false); SetResLockState(x2a8_ltrigger, false); SetResLockState(x2bc_rtrigger, false); SetResLockState(x2d0_abutton, false); } if (x1c0_nextState == EAutoMapperState::MapScreenUniverse && x328_ == 1) LeaveMapScreen(mgr); x1bc_state = x1c0_nextState; } void CAutoMapper::ResetInterpolationTimer(float duration) { x1c4_interpDur = duration; x1c8_interpTime = 0.f; } CAutoMapper::SAutoMapperRenderState CAutoMapper::BuildMiniMapWorldRenderState(const CStateManager& stateMgr, const zeus::CQuaternion& rot, TAreaId area) const { zeus::CQuaternion camOrient = GetMiniMapCameraOrientation(stateMgr); zeus::CQuaternion useOrient = (camOrient.dot(rot) >= 0.f) ? camOrient : camOrient.buildEquivalent(); SAutoMapperRenderState ret( GetMiniMapViewportSize, useOrient, g_tweakAutoMapper->GetMiniCamDist(), g_tweakAutoMapper->GetMiniCamAngle(), GetAreaPointOfInterest(stateMgr, area), GetMapAreaMiniMapDrawDepth(), GetMapAreaMiniMapDrawDepth(), GetMapAreaMiniMapDrawAlphaSurfaceVisited(stateMgr), GetMapAreaMiniMapDrawAlphaOutlineVisited(stateMgr), GetMapAreaMiniMapDrawAlphaSurfaceUnvisited(stateMgr), GetMapAreaMiniMapDrawAlphaOutlineUnvisited(stateMgr)); ret.x44_viewportEase = SAutoMapperRenderState::Ease::Out; ret.x48_camEase = SAutoMapperRenderState::Ease::Out; ret.x4c_pointEase = SAutoMapperRenderState::Ease::Out; ret.x50_depth1Ease = SAutoMapperRenderState::Ease::Linear; ret.x54_depth2Ease = SAutoMapperRenderState::Ease::In; ret.x58_alphaEase = SAutoMapperRenderState::Ease::Linear; return ret; } CAutoMapper::SAutoMapperRenderState CAutoMapper::BuildMapScreenWorldRenderState(const CStateManager& mgr, const zeus::CQuaternion& rot, TAreaId area, bool doingHint) const { float camDist = doingHint ? g_tweakAutoMapper->GetMaxCamDist() : g_tweakAutoMapper->GetCamDist(); SAutoMapperRenderState ret(GetMapScreenViewportSize, rot, camDist, g_tweakAutoMapper->GetCamAngle(), GetAreaPointOfInterest(mgr, area), GetMapAreaMaxDrawDepth(mgr, area), GetMapAreaMaxDrawDepth(mgr, area), g_tweakAutoMapper->GetAlphaSurfaceVisited(), g_tweakAutoMapper->GetAlphaOutlineVisited(), g_tweakAutoMapper->GetAlphaSurfaceUnvisited(), g_tweakAutoMapper->GetAlphaOutlineUnvisited()); ret.x44_viewportEase = SAutoMapperRenderState::Ease::Out; ret.x48_camEase = SAutoMapperRenderState::Ease::Linear; ret.x4c_pointEase = SAutoMapperRenderState::Ease::Out; ret.x50_depth1Ease = SAutoMapperRenderState::Ease::Linear; ret.x54_depth2Ease = SAutoMapperRenderState::Ease::Out; ret.x58_alphaEase = SAutoMapperRenderState::Ease::Linear; return ret; } CAutoMapper::SAutoMapperRenderState CAutoMapper::BuildMapScreenUniverseRenderState(const CStateManager& mgr, const zeus::CQuaternion& rot, TAreaId area) const { SAutoMapperRenderState ret(GetMapScreenViewportSize, rot, g_tweakAutoMapper->GetUniverseCamDist(), g_tweakAutoMapper->GetCamAngle(), GetAreaPointOfInterest(mgr, area), GetMapAreaMaxDrawDepth(mgr, area), GetMapAreaMaxDrawDepth(mgr, area), 0.f, 0.f, 0.f, 0.f); ret.x44_viewportEase = SAutoMapperRenderState::Ease::Out; ret.x48_camEase = SAutoMapperRenderState::Ease::Linear; ret.x4c_pointEase = SAutoMapperRenderState::Ease::Out; ret.x50_depth1Ease = SAutoMapperRenderState::Ease::Linear; ret.x54_depth2Ease = SAutoMapperRenderState::Ease::Out; ret.x58_alphaEase = SAutoMapperRenderState::Ease::Linear; return ret; } void CAutoMapper::LeaveMapScreenState() { SetShouldPanningSoundBePlaying(false); SetShouldZoomingSoundBePlaying(false); SetShouldRotatingSoundBePlaying(false); } float CAutoMapper::GetBaseMapScreenCameraMoveSpeed() { return g_tweakAutoMapper->GetBaseMapScreenCameraMoveSpeed(); } float CAutoMapper::GetFinalMapScreenCameraMoveSpeed() const { float ret = GetBaseMapScreenCameraMoveSpeed(); if (g_tweakAutoMapper->GetScaleMoveSpeedWithCamDist()) ret = ret * xa8_renderStates[0].x18_camDist / g_tweakAutoMapper->GetCamDist(); return ret; } void CAutoMapper::ProcessMapRotateInput(const CFinalInput& input, const CStateManager& mgr) { const float up = ControlMapper::GetAnalogInput(ControlMapper::ECommands::MapCircleUp, input); const float down = ControlMapper::GetAnalogInput(ControlMapper::ECommands::MapCircleDown, input); const float left = ControlMapper::GetAnalogInput(ControlMapper::ECommands::MapCircleLeft, input); const float right = ControlMapper::GetAnalogInput(ControlMapper::ECommands::MapCircleRight, input); std::array<float, 4> dirs{}; bool mouseHeld = false; if (const auto& kbm = input.GetKBM()) { if (kbm->m_mouseButtons[size_t(boo::EMouseButton::Primary)]) { mouseHeld = true; if (float(m_mouseDelta.x()) < 0.f) dirs[3] = -m_mouseDelta.x(); else if (float(m_mouseDelta.x()) > 0.f) dirs[2] = m_mouseDelta.x(); if (float(m_mouseDelta.y()) < 0.f) dirs[0] = -m_mouseDelta.y(); else if (float(m_mouseDelta.y()) > 0.f) dirs[1] = m_mouseDelta.y(); } } float maxMag = up; size_t dirSlot = 0; if (down > up) { maxMag = down; dirSlot = 1; } if (left > maxMag) { maxMag = left; dirSlot = 2; } if (right > maxMag) { maxMag = right; dirSlot = 3; } dirs[dirSlot] += maxMag; if (dirs[0] > 0.f || dirs[1] > 0.f || dirs[2] > 0.f || dirs[3] > 0.f || mouseHeld) { int flags = 0x0; if (up > 0.f) flags |= 0x2; if (down > 0.f) flags |= 0x1; if (left > 0.f) flags |= 0x4; if (right > 0.f) flags |= 0x8; switch (flags) { case 1: // Down x2e4_lStickPos = 1; break; case 2: // Up x2e4_lStickPos = 5; break; case 4: // Left x2e4_lStickPos = 3; break; case 5: // Down-Left x2e4_lStickPos = 2; break; case 6: // Up-Left x2e4_lStickPos = 4; break; case 8: // Right x2e4_lStickPos = 7; break; case 9: // Down-Right x2e4_lStickPos = 8; break; case 10: // Up-Right x2e4_lStickPos = 6; break; default: break; } float deltaFrames = input.DeltaTime() * 60.f; SetShouldRotatingSoundBePlaying(dirs[0] > 0.f || dirs[1] > 0.f || dirs[2] > 0.f || dirs[3] > 0.f); zeus::CEulerAngles eulers(xa8_renderStates[0].x8_camOrientation); zeus::CRelAngle angX(eulers.x()); angX.makeRel(); zeus::CRelAngle angZ(eulers.z()); angZ.makeRel(); float dt = deltaFrames * g_tweakAutoMapper->GetCamRotateDegreesPerFrame(); angZ -= zeus::degToRad(dt * dirs[2]); angZ.makeRel(); angZ += zeus::degToRad(dt * dirs[3]); angZ.makeRel(); angX -= zeus::degToRad(dt * dirs[0]); angX.makeRel(); angX += zeus::degToRad(dt * dirs[1]); angX.makeRel(); float angXDeg = angX.asDegrees(); if (angXDeg > 180.f) angXDeg -= 360.f; angX = zeus::degToRad( zeus::clamp(g_tweakAutoMapper->GetMinCamRotateX(), angXDeg, g_tweakAutoMapper->GetMaxCamRotateX())); angX.makeRel(); zeus::CQuaternion quat; quat.rotateZ(angZ); quat.rotateX(angX); quat.rotateY(0.f); xa8_renderStates[0].x8_camOrientation = quat; } else { x2e4_lStickPos = 0; SetShouldRotatingSoundBePlaying(false); } } void CAutoMapper::ProcessMapZoomInput(const CFinalInput& input, const CStateManager& mgr) { bool in = ControlMapper::GetDigitalInput(ControlMapper::ECommands::MapZoomIn, input); bool out = ControlMapper::GetDigitalInput(ControlMapper::ECommands::MapZoomOut, input); float zoomSpeed = 1.f; if (const auto& kbm = input.GetKBM()) { m_mapScroll += kbm->m_accumScroll - m_lastAccumScroll; m_lastAccumScroll = kbm->m_accumScroll; if (m_mapScroll.delta[1] > 0.0) { in = true; zoomSpeed = std::max(1.f, float(m_mapScroll.delta[1])); m_mapScroll.delta[1] = std::max(0.0, m_mapScroll.delta[1] - (15.0 / 60.0)); } else if (m_mapScroll.delta[1] < 0.0) { out = true; zoomSpeed = std::max(1.f, float(-m_mapScroll.delta[1])); m_mapScroll.delta[1] = std::min(0.0, m_mapScroll.delta[1] + (15.0 / 60.0)); } } const EZoomState nextZoomState = [this, in, out] { switch (x324_zoomState) { case EZoomState::None: case EZoomState::In: case EZoomState::Out: if (in) { return EZoomState::In; } if (out) { return EZoomState::Out; } return EZoomState::None; default: return EZoomState::None; } }(); x324_zoomState = nextZoomState; float delta = input.DeltaTime() * 60.f * (x1bc_state == EAutoMapperState::MapScreen ? 1.f : 4.f) * g_tweakAutoMapper->GetCamZoomUnitsPerFrame() * zoomSpeed; float oldDist = xa8_renderStates[0].x18_camDist; if (x324_zoomState == EZoomState::In) { xa8_renderStates[0].x18_camDist = GetClampedMapScreenCameraDistance(xa8_renderStates[0].x18_camDist - delta); x2f0_rTriggerPos = 1; x324_zoomState = EZoomState::In; } else if (x324_zoomState == EZoomState::Out) { xa8_renderStates[0].x18_camDist = GetClampedMapScreenCameraDistance(xa8_renderStates[0].x18_camDist + delta); x2ec_lTriggerPos = 1; x324_zoomState = EZoomState::Out; } if (oldDist == xa8_renderStates[0].x18_camDist) m_mapScroll.delta[1] = 0.0; SetShouldZoomingSoundBePlaying(oldDist != xa8_renderStates[0].x18_camDist); } void CAutoMapper::ProcessMapPanInput(const CFinalInput& input, const CStateManager& mgr) { float forward = ControlMapper::GetAnalogInput(ControlMapper::ECommands::MapMoveForward, input); float back = ControlMapper::GetAnalogInput(ControlMapper::ECommands::MapMoveBack, input); float left = ControlMapper::GetAnalogInput(ControlMapper::ECommands::MapMoveLeft, input); float right = ControlMapper::GetAnalogInput(ControlMapper::ECommands::MapMoveRight, input); bool mouseHeld = false; if (const auto& kbm = input.GetKBM()) { if (kbm->m_mouseButtons[size_t(boo::EMouseButton::Middle)] || kbm->m_mouseButtons[size_t(boo::EMouseButton::Secondary)]) { mouseHeld = true; if (float(m_mouseDelta.x()) < 0.f) right += -m_mouseDelta.x(); else if (float(m_mouseDelta.x()) > 0.f) left += m_mouseDelta.x(); if (float(m_mouseDelta.y()) < 0.f) forward += -m_mouseDelta.y(); else if (float(m_mouseDelta.y()) > 0.f) back += m_mouseDelta.y(); } } zeus::CTransform camRot = xa8_renderStates[0].x8_camOrientation.toTransform(); if (forward > 0.f || back > 0.f || left > 0.f || right > 0.f || mouseHeld) { float deltaFrames = 60.f * input.DeltaTime(); float speed = GetFinalMapScreenCameraMoveSpeed(); int flags = 0x0; if (forward > 0.f) flags |= 0x1; if (back > 0.f) flags |= 0x2; if (left > 0.f) flags |= 0x4; if (right > 0.f) flags |= 0x8; switch (flags) { case 1: // Forward x2e8_rStickPos = 1; break; case 2: // Back x2e8_rStickPos = 5; break; case 4: // Left x2e8_rStickPos = 3; break; case 5: // Forward-Left x2e8_rStickPos = 2; break; case 6: // Back-Left x2e8_rStickPos = 4; break; case 8: // Right x2e8_rStickPos = 7; break; case 9: // Forward-Right x2e8_rStickPos = 8; break; case 10: // Back-Right x2e8_rStickPos = 6; break; default: break; } zeus::CVector3f dirVec(right - left, 0.f, forward - back); zeus::CVector3f deltaVec = camRot * (dirVec * deltaFrames * speed); zeus::CVector3f newPoint = xa8_renderStates[0].x20_areaPoint + deltaVec; SetShouldPanningSoundBePlaying(deltaVec.magnitude() > input.DeltaTime()); if (x1bc_state == EAutoMapperState::MapScreen) { xa8_renderStates[0].x20_areaPoint = x24_world->IGetMapWorld()->ConstrainToWorldVolume(newPoint, camRot.basis[1]); } else { zeus::CVector3f localPoint = newPoint - x8_mapu->GetMapUniverseCenterPoint(); if (localPoint.magnitude() > x8_mapu->GetMapUniverseRadius()) newPoint = x8_mapu->GetMapUniverseCenterPoint() + localPoint.normalized() * x8_mapu->GetMapUniverseRadius(); xa8_renderStates[0].x20_areaPoint = newPoint; } } else { x2e8_rStickPos = 0; SetShouldPanningSoundBePlaying(false); float speed = g_tweakAutoMapper->GetCamPanUnitsPerFrame() * GetBaseMapScreenCameraMoveSpeed(); if (x1bc_state == EAutoMapperState::MapScreen) { const CMapArea* area = x24_world->IGetMapWorld()->GetMapArea(xa0_curAreaId); zeus::CVector3f worldPoint = area->GetAreaPostTransform(*x24_world, xa0_curAreaId) * area->GetAreaCenterPoint(); zeus::CVector3f viewPoint = worldPoint - xa8_renderStates[0].x20_areaPoint; if (viewPoint.magnitude() < speed) xa8_renderStates[0].x20_areaPoint = worldPoint; else xa8_renderStates[0].x20_areaPoint += viewPoint.normalized() * speed; } else { std::pair<int, int> areas = FindClosestVisibleWorld(xa8_renderStates[0].x20_areaPoint, camRot.basis[1], mgr); const zeus::CTransform& hex = x8_mapu->GetMapWorldData(areas.first).GetMapAreaData(areas.second); zeus::CVector3f areaToHex = hex.origin - xa8_renderStates[0].x20_areaPoint; if (areaToHex.magnitude() < speed) xa8_renderStates[0].x20_areaPoint = hex.origin; else xa8_renderStates[0].x20_areaPoint += areaToHex.normalized() * speed; } } } void CAutoMapper::SetShouldPanningSoundBePlaying(bool shouldBePlaying) { if (shouldBePlaying) { if (!x1cc_panningSfx) x1cc_panningSfx = CSfxManager::SfxStart(SFXui_map_pan, 1.f, 0.f, false, 0x7f, true, kInvalidAreaId); } else { CSfxManager::SfxStop(x1cc_panningSfx); x1cc_panningSfx.reset(); } } void CAutoMapper::SetShouldZoomingSoundBePlaying(bool shouldBePlaying) { if (shouldBePlaying) { if (!x1d4_zoomingSfx) x1d4_zoomingSfx = CSfxManager::SfxStart(SFXui_map_zoom, 1.f, 0.f, false, 0x7f, true, kInvalidAreaId); } else { CSfxManager::SfxStop(x1d4_zoomingSfx); x1d4_zoomingSfx.reset(); } } void CAutoMapper::SetShouldRotatingSoundBePlaying(bool shouldBePlaying) { if (shouldBePlaying) { if (!x1d0_rotatingSfx) x1d0_rotatingSfx = CSfxManager::SfxStart(SFXui_map_rotate, 1.f, 0.f, false, 0x7f, true, kInvalidAreaId); } else { CSfxManager::SfxStop(x1d0_rotatingSfx); x1d0_rotatingSfx.reset(); } } void CAutoMapper::ProcessMapScreenInput(const CFinalInput& input, CStateManager& mgr) { zeus::CMatrix3f camRot = xa8_renderStates[0].x8_camOrientation.toTransform().buildMatrix3f(); if (x1bc_state == EAutoMapperState::MapScreen) { if ((input.PA() || input.PSpecialKey(boo::ESpecialKey::Enter)) && x328_ == 0 && HasCurrentMapUniverseWorld()) BeginMapperStateTransition(EAutoMapperState::MapScreenUniverse, mgr); } else if (x1bc_state == EAutoMapperState::MapScreenUniverse && (input.PA() || input.PSpecialKey(boo::ESpecialKey::Enter))) { const CMapUniverse::CMapWorldData& mapuWld = x8_mapu->GetMapWorldData(x9c_worldIdx); zeus::CVector3f pointLocal = mapuWld.GetWorldTransform().inverse() * xa8_renderStates[0].x20_areaPoint; if (mapuWld.GetWorldAssetId() != g_GameState->CurrentWorldAssetId()) { x32c_loadingDummyWorld = true; CheckDummyWorldLoad(mgr); } else { x24_world = mgr.GetWorld(); CMapWorldInfo& mwInfo = *g_GameState->StateForWorld(x24_world->IGetWorldAssetId()).MapWorldInfo(); xa0_curAreaId = FindClosestVisibleArea(pointLocal, zeus::CUnitVector3f(camRot[1]), mgr, *x24_world, mwInfo); BeginMapperStateTransition(EAutoMapperState::MapScreen, mgr); } } x2f4_aButtonPos = 0; if (input.PA() || input.PSpecialKey(boo::ESpecialKey::Enter)) x2f4_aButtonPos = 1; if (IsInPlayerControlState()) { x2ec_lTriggerPos = 0; x2f0_rTriggerPos = 0; if (const auto& kbm = input.GetKBM()) { zeus::CVector2f mouseCoord = zeus::CVector2f(kbm->m_mouseCoord.norm[0], kbm->m_mouseCoord.norm[1]); if (!m_lastMouseCoord) { m_lastMouseCoord.emplace(mouseCoord); } else { m_mouseDelta = mouseCoord - *m_lastMouseCoord; m_lastMouseCoord.emplace(mouseCoord); m_mouseDelta.x() *= g_Viewport.aspect; m_mouseDelta *= 100.f; } } ProcessMapRotateInput(input, mgr); ProcessMapZoomInput(input, mgr); ProcessMapPanInput(input, mgr); } } zeus::CQuaternion CAutoMapper::GetMiniMapCameraOrientation(const CStateManager& stateMgr) const { const CGameCamera* cam = stateMgr.GetCameraManager()->GetCurrentCamera(stateMgr); zeus::CEulerAngles camAngles(zeus::CQuaternion(cam->GetTransform().buildMatrix3f())); zeus::CRelAngle angle(camAngles.z()); angle.makeRel(); zeus::CQuaternion ret; ret.rotateZ(angle); ret.rotateX(zeus::degToRad(g_tweakAutoMapper->GetMiniCamXAngle())); return ret; } zeus::CVector3f CAutoMapper::GetAreaPointOfInterest(const CStateManager&, TAreaId aid) const { const CMapArea* mapa = x24_world->IGetMapWorld()->GetMapArea(aid); return mapa->GetAreaPostTransform(*x24_world, aid) * mapa->GetAreaCenterPoint(); } TAreaId CAutoMapper::FindClosestVisibleArea(const zeus::CVector3f& point, const zeus::CUnitVector3f& camDir, const CStateManager& mgr, const IWorld& wld, const CMapWorldInfo& mwInfo) const { float minDist = 9999.f; TAreaId closestArea = xa0_curAreaId; const CMapWorld* mw = wld.IGetMapWorld(); std::vector<TAreaId> areas = mw->GetVisibleAreas(wld, mwInfo); for (TAreaId areaId : areas) { const CMapArea* mapa = mw->GetMapArea(areaId); zeus::CVector3f xfPoint = mapa->GetAreaPostTransform(wld, areaId) * mapa->GetAreaCenterPoint(); zeus::CVector3f pointToArea = xfPoint - point; pointToArea = pointToArea.canBeNormalized() ? point + (pointToArea.normalized().dot(camDir) * pointToArea.magnitude()) * camDir : point; pointToArea -= xfPoint; float dist = pointToArea.magnitude(); if (dist < minDist) { minDist = dist; closestArea = areaId; } } return closestArea; } std::pair<int, int> CAutoMapper::FindClosestVisibleWorld(const zeus::CVector3f& point, const zeus::CUnitVector3f& camDir, const CStateManager& mgr) const { float minDist = 29999.f; std::pair<int, int> closestWorld = {x9c_worldIdx, xa0_curAreaId}; for (u32 w = 0; w < x8_mapu->GetNumMapWorldDatas(); ++w) { const CMapUniverse::CMapWorldData& mwData = x8_mapu->GetMapWorldData(w); const CMapWorldInfo& mwInfo = *g_GameState->StateForWorld(mwData.GetWorldAssetId()).MapWorldInfo(); if (!mwInfo.IsAnythingSet()) continue; for (u32 i = 0; i < mwData.GetNumMapAreaDatas(); ++i) { const zeus::CVector3f& mwOrigin = mwData.GetMapAreaData(i).origin; zeus::CVector3f pointToArea = mwOrigin - point; pointToArea = pointToArea.canBeNormalized() ? point + (pointToArea.normalized().dot(camDir) * pointToArea.magnitude()) * camDir : point; pointToArea -= mwOrigin; float dist = pointToArea.magnitude(); if (dist < minDist) { minDist = dist; closestWorld.first = w; closestWorld.second = i; } } } return closestWorld; } zeus::CVector2i CAutoMapper::GetMiniMapViewportSize() { float scaleX = g_Viewport.x8_width / 640.f; float scaleY = g_Viewport.xc_height / 480.f; return {int(scaleX * g_tweakAutoMapper->GetMiniMapViewportWidth()), int(scaleY * g_tweakAutoMapper->GetMiniMapViewportHeight())}; } zeus::CVector2i CAutoMapper::GetMapScreenViewportSize() { return {int(g_Viewport.x8_width), int(g_Viewport.xc_height)}; } float CAutoMapper::GetMapAreaMaxDrawDepth(const CStateManager&, TAreaId aid) const { return x24_world->IGetMapWorld()->GetCurrentMapAreaDepth(*x24_world, aid); } float CAutoMapper::GetMapAreaMiniMapDrawAlphaSurfaceVisited(const CStateManager& stateMgr) { float mapAlphaInterp = g_tweakGui->GetMapAlphaInterpolant(); return g_tweakAutoMapper->GetMiniAlphaSurfaceVisited() * (1.f - mapAlphaInterp) * stateMgr.Player()->GetGunAlpha() + mapAlphaInterp; } float CAutoMapper::GetMapAreaMiniMapDrawAlphaOutlineVisited(const CStateManager& stateMgr) { float mapAlphaInterp = g_tweakGui->GetMapAlphaInterpolant(); return g_tweakAutoMapper->GetMiniAlphaOutlineVisited() * (1.f - mapAlphaInterp) * stateMgr.Player()->GetGunAlpha() + mapAlphaInterp; } float CAutoMapper::GetMapAreaMiniMapDrawAlphaSurfaceUnvisited(const CStateManager& stateMgr) { float mapAlphaInterp = g_tweakGui->GetMapAlphaInterpolant(); return g_tweakAutoMapper->GetMiniAlphaSurfaceUnvisited() * (1.f - mapAlphaInterp) * stateMgr.Player()->GetGunAlpha() + mapAlphaInterp; } float CAutoMapper::GetMapAreaMiniMapDrawAlphaOutlineUnvisited(const CStateManager& stateMgr) { float mapAlphaInterp = g_tweakGui->GetMapAlphaInterpolant(); return g_tweakAutoMapper->GetMiniAlphaOutlineUnvisited() * (1.f - mapAlphaInterp) * stateMgr.Player()->GetGunAlpha() + mapAlphaInterp; } float CAutoMapper::GetDesiredMiniMapCameraDistance(const CStateManager& mgr) const { const CMapWorldInfo& mwInfo = *g_GameState->StateForWorld(x24_world->IGetWorldAssetId()).MapWorldInfo(); const CMapWorld* mw = x24_world->IGetMapWorld(); zeus::CAABox aabb; const IGameArea* area = x24_world->IGetAreaAlways(xa0_curAreaId); const CMapArea* mapa = mw->GetMapArea(xa0_curAreaId); bool oneMiniMapArea = g_tweakAutoMapper->GetShowOneMiniMapArea(); for (int i = -1; i < (oneMiniMapArea ? 0 : int(area->IGetNumAttachedAreas())); ++i) { TAreaId aid = i == -1 ? xa0_curAreaId : area->IGetAttachedAreaId(i); const CMapArea* attMapa = mw->GetMapArea(aid); if (attMapa->GetIsVisibleToAutoMapper(mwInfo.IsWorldVisible(aid), mwInfo.IsAreaVisible(aid))) { zeus::CAABox areaAABB = attMapa->GetBoundingBox().getTransformedAABox(attMapa->GetAreaPostTransform(*x24_world, aid)); aabb.accumulateBounds(areaAABB.min); aabb.accumulateBounds(areaAABB.max); } } zeus::CVector3f xfPoint = mapa->GetAreaPostTransform(*x24_world, xa0_curAreaId) * mapa->GetAreaCenterPoint(); zeus::CVector3f maxMargin; maxMargin.x() = std::max(xfPoint.x() - aabb.min.x(), aabb.max.x() - xfPoint.x()); maxMargin.y() = std::max(xfPoint.y() - aabb.min.y(), aabb.max.y() - xfPoint.y()); maxMargin.z() = std::max(xfPoint.z() - aabb.min.z(), aabb.max.z() - xfPoint.z()); zeus::CVector3f extent = mapa->GetBoundingBox().max - mapa->GetBoundingBox().min; return (0.5f * (0.5f * extent.magnitude()) + 0.5f * maxMargin.magnitude()) * g_tweakAutoMapper->GetMiniMapCamDistScale() * std::tan(M_PIF / 2.f - 0.5f * 2.f * M_PIF * (xa8_renderStates[0].x1c_camAngle / 360.f)); } float CAutoMapper::GetClampedMapScreenCameraDistance(float value) const { if (x1bc_state == EAutoMapperState::MapScreenUniverse) { return zeus::clamp(g_tweakAutoMapper->GetMinUniverseCamDist(), value, g_tweakAutoMapper->GetMaxUniverseCamDist()); } return zeus::clamp(g_tweakAutoMapper->GetMinCamDist(), value, g_tweakAutoMapper->GetMaxCamDist()); } void CAutoMapper::MuteAllLoopedSounds() { CSfxManager::SfxVolume(x1cc_panningSfx, 0.f); CSfxManager::SfxVolume(x1d0_rotatingSfx, 0.f); CSfxManager::SfxVolume(x1d4_zoomingSfx, 0.f); } void CAutoMapper::UnmuteAllLoopedSounds() { CSfxManager::SfxVolume(x1cc_panningSfx, 1.f); CSfxManager::SfxVolume(x1d0_rotatingSfx, 1.f); CSfxManager::SfxVolume(x1d4_zoomingSfx, 1.f); } void CAutoMapper::ProcessControllerInput(const CFinalInput& input, CStateManager& mgr) { if (!IsRenderStateInterpolating()) { if (IsInPlayerControlState()) { if (x32c_loadingDummyWorld) CheckDummyWorldLoad(mgr); else if (x1e0_hintSteps.size()) UpdateHintNavigation(input.DeltaTime(), mgr); else if (x328_ == 0) ProcessMapScreenInput(input, mgr); } } zeus::CMatrix3f camRot = xa8_renderStates[0].x8_camOrientation.toTransform().buildMatrix3f(); if (IsInMapperState(EAutoMapperState::MapScreen)) { CMapWorldInfo& mwInfo = *g_GameState->StateForWorld(x24_world->IGetWorldAssetId()).MapWorldInfo(); TAreaId aid = FindClosestVisibleArea(xa8_renderStates[0].x20_areaPoint, camRot[1], mgr, *x24_world, mwInfo); if (aid != xa0_curAreaId) { xa0_curAreaId = aid; xa8_renderStates[0].x2c_drawDepth1 = GetMapAreaMaxDrawDepth(mgr, xa0_curAreaId); xa8_renderStates[0].x30_drawDepth2 = GetMapAreaMaxDrawDepth(mgr, xa0_curAreaId); } } else if (IsInMapperState(EAutoMapperState::MapScreenUniverse)) { u32 oldWldIdx = x9c_worldIdx; if (x1e0_hintSteps.size()) { SAutoMapperHintStep& nextStep = x1e0_hintSteps.front(); if (nextStep.x0_type == SAutoMapperHintStep::Type::PanToWorld || nextStep.x0_type == SAutoMapperHintStep::Type::SwitchToWorld) { SetCurWorldAssetId(nextStep.x4_worldId); } else { std::pair<int, int> wld = FindClosestVisibleWorld(xa8_renderStates[0].x20_areaPoint, camRot[1], mgr); x9c_worldIdx = wld.first; } } else { std::pair<int, int> wld = FindClosestVisibleWorld(xa8_renderStates[0].x20_areaPoint, camRot[1], mgr); x9c_worldIdx = wld.first; } if (x9c_worldIdx != oldWldIdx) { CAssetId curMlvl = g_GameState->CurrentWorldAssetId(); for (u32 i = 0; i < x14_dummyWorlds.size(); ++i) { auto& wld = x14_dummyWorlds[i]; const CMapUniverse::CMapWorldData& mwData = x8_mapu->GetMapWorldData(i); if (i == x9c_worldIdx && curMlvl != mwData.GetWorldAssetId()) { if (g_ResFactory->CanBuild(SObjectTag{FOURCC('MLVL'), mwData.GetWorldAssetId()})) wld = std::make_unique<CDummyWorld>(mwData.GetWorldAssetId(), true); } else { wld.reset(); } } x24_world = (curMlvl == x8_mapu->GetMapWorldData(x9c_worldIdx).GetWorldAssetId()) ? mgr.GetWorld() : nullptr; } } if (x300_textpane_instructions) { if (x78_areaHintDesc.IsLoaded()) { x2fc_textpane_hint->TextSupport().SetText(x78_areaHintDesc->GetString(0)); x304_textpane_instructions1->TextSupport().SetText(u""); x300_textpane_instructions->TextSupport().SetText(u""); x308_textpane_instructions2->TextSupport().SetText(u""); } else { x2fc_textpane_hint->TextSupport().SetText(u""); std::u16string str = fmt::format(FMT_STRING(u"&image=SI,0.6,1.0,{};"), g_tweakPlayerRes->x24_lStick[x2e4_lStickPos]); str += g_MainStringTable->GetString(46 + (!g_Main->IsUSA() || g_Main->IsTrilogy())); // Rotate x300_textpane_instructions->TextSupport().SetText(str); str = fmt::format(FMT_STRING(u"&image=SI,0.6,1.0,{};"), g_tweakPlayerRes->x4c_cStick[x2e8_rStickPos]); str += g_MainStringTable->GetString(47 + (!g_Main->IsUSA() || g_Main->IsTrilogy())); // Move x304_textpane_instructions1->TextSupport().SetText(str); str = fmt::format(FMT_STRING(u"&image={};"), g_tweakPlayerRes->x74_lTrigger[x2ec_lTriggerPos]); str += g_MainStringTable->GetString(48 + (!g_Main->IsUSA() || g_Main->IsTrilogy())); // Zoom str += fmt::format(FMT_STRING(u"&image={};"), g_tweakPlayerRes->x80_rTrigger[x2f0_rTriggerPos]); x308_textpane_instructions2->TextSupport().SetText(str); } } if (input.PY() || input.PKey(' ')) { CPersistentOptions& sysOpts = g_GameState->SystemOptions(); switch (sysOpts.GetAutoMapperKeyState()) { case 0: sysOpts.SetAutoMapperKeyState(1); CSfxManager::SfxStart(SFXui_map_screen_key1, 1.f, 0.f, false, 0x7f, false, kInvalidAreaId); break; case 1: sysOpts.SetAutoMapperKeyState(2); CSfxManager::SfxStart(SFXui_map_screen_key2, 1.f, 0.f, false, 0x7f, false, kInvalidAreaId); break; case 2: sysOpts.SetAutoMapperKeyState(0); CSfxManager::SfxStart(SFXui_map_screen_key0, 1.f, 0.f, false, 0x7f, false, kInvalidAreaId); break; default: break; } } if (input.PZ() || input.PKey('\t') || input.PB() || input.PSpecialKey(boo::ESpecialKey::Esc)) { if (x328_ == 0) { if (CanLeaveMapScreenInternal(mgr)) { LeaveMapScreen(mgr); } else if (NotHintNavigating()) { BeginMapperStateTransition(EAutoMapperState::MapScreenUniverse, mgr); x328_ = 1; } } } } void CAutoMapper::Update(float dt, CStateManager& mgr) { if (x1bc_state != EAutoMapperState::MiniMap && x1c0_nextState != EAutoMapperState::MiniMap) { x1d8_flashTimer = std::fmod(x1d8_flashTimer + dt, 0.75f); x1dc_playerFlashPulse = x1d8_flashTimer < 0.375f ? x1d8_flashTimer / 0.375f : (0.75f - x1d8_flashTimer) / 0.375f; } if (!m_frmeInitialized && x28_frmeMapScreen.IsLoaded()) { x28_frmeMapScreen->SetMaxAspect(1.78f); m_frmeInitialized = true; static_cast<CGuiTextPane*>(x28_frmeMapScreen->FindWidget("textpane_left")) ->TextSupport() .SetText(g_MainStringTable->GetString(42 + (!g_Main->IsUSA() || g_Main->IsTrilogy()))); static_cast<CGuiTextPane*>(x28_frmeMapScreen->FindWidget("textpane_yicon")) ->TextSupport() .SetText(g_MainStringTable->GetString(43 + (!g_Main->IsUSA() || g_Main->IsTrilogy()))); x2fc_textpane_hint = static_cast<CGuiTextPane*>(x28_frmeMapScreen->FindWidget("textpane_hint")); x300_textpane_instructions = static_cast<CGuiTextPane*>(x28_frmeMapScreen->FindWidget("textpane_instructions")); x304_textpane_instructions1 = static_cast<CGuiTextPane*>(x28_frmeMapScreen->FindWidget("textpane_instructions1")); x308_textpane_instructions2 = static_cast<CGuiTextPane*>(x28_frmeMapScreen->FindWidget("textpane_instructions2")); CGuiTextPane* mapLegend = static_cast<CGuiTextPane*>(x28_frmeMapScreen->FindWidget("textpane_mapLegend")); mapLegend->TextSupport().ClearRenderBuffer(); mapLegend->TextSupport().SetImageBaseline(true); mapLegend->TextSupport().SetText(g_MainStringTable->GetString(49 + (!g_Main->IsUSA() || g_Main->IsTrilogy()))); x30c_basewidget_leftPane = x28_frmeMapScreen->FindWidget("basewidget_leftPane"); x310_basewidget_yButtonPane = x28_frmeMapScreen->FindWidget("basewidget_yButtonPane"); x314_basewidget_bottomPane = x28_frmeMapScreen->FindWidget("basewidget_bottomPane"); x2f8_textpane_areaname = static_cast<CGuiTextPane*>(x28_frmeMapScreen->FindWidget("textpane_areaname")); x2f8_textpane_areaname->SetDepthTest(false); } if (m_frmeInitialized) { x28_frmeMapScreen->Update(dt); CGuiTextPane* right1 = static_cast<CGuiTextPane*>(x28_frmeMapScreen->FindWidget("textpane_right1")); std::u16string string; if (x1bc_state == EAutoMapperState::MapScreenUniverse || (x1bc_state == EAutoMapperState::MapScreen && HasCurrentMapUniverseWorld())) string = fmt::format(FMT_STRING(u"&image={};"), g_tweakPlayerRes->x98_aButton[x2f4_aButtonPos]); right1->TextSupport().SetText(string); CGuiTextPane* right = static_cast<CGuiTextPane*>(x28_frmeMapScreen->FindWidget("textpane_right")); if (x1bc_state == EAutoMapperState::MapScreenUniverse) string = g_MainStringTable->GetString(45); else if (x1bc_state == EAutoMapperState::MapScreen && HasCurrentMapUniverseWorld()) string = g_MainStringTable->GetString(44); else string = std::u16string(); right->TextSupport().SetText(string); } float dt2 = 2.f * dt; switch (g_GameState->SystemOptions().GetAutoMapperKeyState()) { case 0: // All shown x318_leftPanePos -= dt2; x31c_yButtonPanePos -= dt2; x320_bottomPanePos -= dt2; break; case 1: // Left shown x318_leftPanePos += dt2; x31c_yButtonPanePos -= dt2; x320_bottomPanePos -= dt2; break; case 2: // All hidden x318_leftPanePos += dt2; x31c_yButtonPanePos += dt2; x320_bottomPanePos += dt2; break; default: break; } x318_leftPanePos = std::max(0.f, std::min(x318_leftPanePos, 1.f)); x31c_yButtonPanePos = std::max(0.f, std::min(x31c_yButtonPanePos, 1.f)); x320_bottomPanePos = std::max(0.f, std::min(x320_bottomPanePos, 1.f)); if (x30c_basewidget_leftPane) { float vpAspectRatio = std::max(1.78f, g_Viewport.aspect); x30c_basewidget_leftPane->SetLocalTransform( zeus::CTransform::Translate(x318_leftPanePos * vpAspectRatio * -9.f, 0.f, 0.f) * x30c_basewidget_leftPane->GetTransform()); } if (x310_basewidget_yButtonPane) { x310_basewidget_yButtonPane->SetLocalTransform(zeus::CTransform::Translate(0.f, 0.f, x31c_yButtonPanePos * -3.5f) * x310_basewidget_yButtonPane->GetTransform()); } if (x314_basewidget_bottomPane) { x314_basewidget_bottomPane->SetLocalTransform(zeus::CTransform::Translate(0.f, 0.f, x320_bottomPanePos * -7.f) * x314_basewidget_bottomPane->GetTransform()); } if (IsInMapperState(EAutoMapperState::MiniMap)) { xa8_renderStates[0].x8_camOrientation = GetMiniMapCameraOrientation(mgr); float desiredDist = GetDesiredMiniMapCameraDistance(mgr); if (std::fabs(xa8_renderStates[0].x18_camDist - desiredDist) < 3.f) xa8_renderStates[0].x18_camDist = desiredDist; else if (xa8_renderStates[0].x18_camDist < desiredDist) xa8_renderStates[0].x18_camDist += 3.f; else xa8_renderStates[0].x18_camDist -= 3.f; TAreaId curAid = x24_world->IGetCurrentAreaId(); if (curAid != xa0_curAreaId) { xa8_renderStates[2] = xa8_renderStates[0]; xa8_renderStates[1] = xa8_renderStates[0]; xa4_otherAreaId = xa0_curAreaId; xa0_curAreaId = curAid; xa8_renderStates[1].x20_areaPoint = GetAreaPointOfInterest(mgr, xa0_curAreaId); xa8_renderStates[1].x44_viewportEase = SAutoMapperRenderState::Ease::None; xa8_renderStates[1].x48_camEase = SAutoMapperRenderState::Ease::None; xa8_renderStates[1].x4c_pointEase = SAutoMapperRenderState::Ease::InOut; xa8_renderStates[1].x50_depth1Ease = SAutoMapperRenderState::Ease::Linear; xa8_renderStates[1].x54_depth2Ease = SAutoMapperRenderState::Ease::Linear; xa8_renderStates[1].x58_alphaEase = SAutoMapperRenderState::Ease::None; xa8_renderStates[1].x2c_drawDepth1 = GetMapAreaMiniMapDrawDepth(); xa8_renderStates[1].x30_drawDepth2 = GetMapAreaMiniMapDrawDepth(); xa8_renderStates[2].x2c_drawDepth1 = GetMapAreaMiniMapDrawDepth() - 1.f; xa8_renderStates[2].x30_drawDepth2 = GetMapAreaMiniMapDrawDepth() - 1.f; ResetInterpolationTimer(g_tweakAutoMapper->GetHintPanTime()); } xa8_renderStates[1].x34_alphaSurfaceVisited = GetMapAreaMiniMapDrawAlphaSurfaceVisited(mgr); xa8_renderStates[1].x38_alphaOutlineVisited = GetMapAreaMiniMapDrawAlphaOutlineVisited(mgr); xa8_renderStates[1].x3c_alphaSurfaceUnvisited = GetMapAreaMiniMapDrawAlphaSurfaceUnvisited(mgr); xa8_renderStates[1].x40_alphaOutlineUnvisited = GetMapAreaMiniMapDrawAlphaOutlineUnvisited(mgr); } else { if (x1c0_nextState == EAutoMapperState::MiniMap) { float desiredDist = GetDesiredMiniMapCameraDistance(mgr); if (std::fabs(xa8_renderStates[1].x18_camDist - desiredDist) < 3.f) xa8_renderStates[0].x18_camDist = desiredDist; else if (xa8_renderStates[1].x18_camDist < desiredDist) xa8_renderStates[1].x18_camDist += 3.f; else xa8_renderStates[1].x18_camDist -= 3.f; } else if (x1bc_state != EAutoMapperState::MiniMap && x1c0_nextState != EAutoMapperState::MiniMap && x24_world) { x24_world->IGetMapWorld()->RecalculateWorldSphere( *g_GameState->StateForWorld(x24_world->IGetWorldAssetId()).MapWorldInfo(), *x24_world); } } if (IsRenderStateInterpolating()) { x1c8_interpTime = std::min(x1c8_interpTime + dt, x1c4_interpDur); SAutoMapperRenderState::InterpolateWithClamp(xa8_renderStates[2], xa8_renderStates[0], xa8_renderStates[1], x1c8_interpTime / x1c4_interpDur); if (x1c8_interpTime == x1c4_interpDur && x328_ == 2) SetupMiniMapWorld(mgr); } else if (IsInMapperStateTransition()) { CompleteMapperStateTransition(mgr); } CAssetId stringId = x88_mapAreaStringId; if (IsInMapperState(EAutoMapperState::MapScreenUniverse)) { IWorld* wld = x14_dummyWorlds[x9c_worldIdx].get(); if (wld && wld->ICheckWorldComplete()) stringId = wld->IGetStringTableAssetId(); else if (x24_world) stringId = x24_world->IGetStringTableAssetId(); } else if (x24_world) { const IGameArea* area = x24_world->IGetAreaAlways(xa0_curAreaId); const CMapWorldInfo& mwInfo = *g_GameState->StateForWorld(x24_world->IGetWorldAssetId()).MapWorldInfo(); if (mwInfo.IsMapped(xa0_curAreaId) || mwInfo.IsAreaVisited(xa0_curAreaId)) stringId = area->IGetStringTableAssetId(); else stringId = {}; } if (x88_mapAreaStringId != stringId) { x88_mapAreaStringId = stringId; if (x88_mapAreaStringId.IsValid()) x8c_mapAreaString = g_SimplePool->GetObj(SObjectTag{FOURCC('STRG'), x88_mapAreaStringId}); else x8c_mapAreaString = TLockedToken<CStringTable>(); } if (x2f8_textpane_areaname) { if (x8c_mapAreaString) { if (x8c_mapAreaString.IsLoaded()) x2f8_textpane_areaname->TextSupport().SetText(x8c_mapAreaString->GetString(0)); } else { x2f8_textpane_areaname->TextSupport().SetText(u""); } } if (IsInMapperState(EAutoMapperState::MapScreen)) { CAssetId hintDesc = GetAreaHintDescriptionString(x24_world->IGetAreaAlways(xa0_curAreaId)->IGetAreaAssetId()); if (hintDesc != x74_areaHintDescId) { x74_areaHintDescId = hintDesc; if (x74_areaHintDescId.IsValid()) x78_areaHintDesc = g_SimplePool->GetObj(SObjectTag{FOURCC('STRG'), x74_areaHintDescId}); else x78_areaHintDesc = TLockedToken<CStringTable>(); } } for (auto& wld : x14_dummyWorlds) if (wld) wld->ICheckWorldComplete(); } void CAutoMapper::Draw(const CStateManager& mgr, const zeus::CTransform& xf, float alpha) { SCOPED_GRAPHICS_DEBUG_GROUP("CAutoMapper::Draw", zeus::skPurple); alpha *= g_GameState->GameOptions().GetHUDAlpha() / 255.f; // Blend mode alpha // Backface cull float alphaInterp; if (x1bc_state != EAutoMapperState::MiniMap && x1c0_nextState != EAutoMapperState::MiniMap) { alphaInterp = 1.f; } else if (IsInMapperState(EAutoMapperState::MiniMap)) { alphaInterp = alpha; } else if (x1c0_nextState == EAutoMapperState::MiniMap) { float t = GetInterp(); alphaInterp = alpha * t + (1.f - t); } else if (x1bc_state == EAutoMapperState::MiniMap) { float t = GetInterp(); alphaInterp = alpha * (1.f - t) + t; } else { alphaInterp = 1.f; } zeus::CVector2i vp = xa8_renderStates[0].GetViewportSize(); float aspect = vp.x / float(vp.y); if (aspect > 1.78f) aspect = 1.78f; float yScale = xa8_renderStates[0].x18_camDist / std::tan(M_PIF / 2.f - 0.5f * 2.f * M_PIF * (xa8_renderStates[0].x1c_camAngle / 360.f)); float xScale = yScale * aspect; zeus::CTransform camXf(xa8_renderStates[0].x8_camOrientation, xa8_renderStates[0].x20_areaPoint); zeus::CTransform distScale = zeus::CTransform::Scale(1.f / xScale, 0.001f, 1.f / yScale); zeus::CTransform tweakScale = zeus::CTransform::Scale(g_tweakAutoMapper->GetMapPlaneScaleX(), 0.f, g_tweakAutoMapper->GetMapPlaneScaleZ()); zeus::CTransform planeXf = xf * tweakScale * distScale * camXf.inverse(); float universeInterp = 0.f; if (x1c0_nextState == EAutoMapperState::MapScreenUniverse) { if (x1bc_state == EAutoMapperState::MapScreenUniverse) universeInterp = 1.f; else universeInterp = GetInterp(); } else if (x1bc_state == EAutoMapperState::MapScreenUniverse) { universeInterp = 1.f - GetInterp(); } zeus::CTransform preXf; if (x1bc_state == EAutoMapperState::MapScreenUniverse || x1c0_nextState == EAutoMapperState::MapScreenUniverse) preXf = x8_mapu->GetMapWorldData(x9c_worldIdx).GetWorldTransform(); float objectScale = xa8_renderStates[0].x18_camDist / g_tweakAutoMapper->GetMinCamDist(); float mapAlpha = alphaInterp * (1.f - universeInterp); if (x1bc_state != EAutoMapperState::MiniMap && x1c0_nextState != EAutoMapperState::MiniMap) { if (universeInterp < 1.f && x24_world) { const CMapWorldInfo& mwInfo = *g_GameState->StateForWorld(x24_world->IGetWorldAssetId()).MapWorldInfo(); CMapWorld* mw = x24_world->IGetMapWorld(); float hintFlash = 0.f; if (x1e0_hintSteps.size() && x1e0_hintSteps.front().x0_type == SAutoMapperHintStep::Type::ShowBeacon) { if (xa0_curAreaId == mgr.GetNextAreaId() && x24_world == mgr.GetWorld()) { float pulseTime = std::fmod(x1e0_hintSteps.front().x4_float * 8.f, 1.f); hintFlash = 2.f * (pulseTime < 0.5f ? pulseTime : 1.f - pulseTime); } else { for (const SAutoMapperHintLocation& loc : x1f8_hintLocations) { if (x24_world->IGetWorldAssetId() != loc.x8_worldId) continue; if (xa0_curAreaId != loc.xc_areaId) continue; float pulseTime = std::fmod((1.f - std::max(0.f, (x1e0_hintSteps.front().x4_float - 0.5f) / 0.5f)) * 4.f, 1.f); hintFlash = 2.f * (pulseTime < 0.5f ? pulseTime : 1.f - pulseTime); break; } } } const zeus::CTransform modelXf = planeXf * preXf; const CMapWorld::CMapWorldDrawParms parms(xa8_renderStates[0].x34_alphaSurfaceVisited * alphaInterp, xa8_renderStates[0].x38_alphaOutlineVisited * alphaInterp, xa8_renderStates[0].x3c_alphaSurfaceUnvisited * alphaInterp, xa8_renderStates[0].x40_alphaOutlineUnvisited * alphaInterp, mapAlpha, 2.f, mgr, modelXf, camXf, *x24_world, mwInfo, x1dc_playerFlashPulse, hintFlash, objectScale, true); mw->Draw(parms, xa0_curAreaId, xa0_curAreaId, xa8_renderStates[0].x2c_drawDepth1, xa8_renderStates[0].x30_drawDepth2, true); } } else if (IsInMapperState(EAutoMapperState::MiniMap)) { CMapWorld* mw = x24_world->IGetMapWorld(); const CMapWorldInfo& mwInfo = *g_GameState->StateForWorld(x24_world->IGetWorldAssetId()).MapWorldInfo(); const CMapWorld::CMapWorldDrawParms parms(xa8_renderStates[0].x34_alphaSurfaceVisited * alphaInterp, xa8_renderStates[0].x38_alphaOutlineVisited * alphaInterp, xa8_renderStates[0].x3c_alphaSurfaceUnvisited * alphaInterp, xa8_renderStates[0].x40_alphaOutlineUnvisited * alphaInterp, mapAlpha, 1.f, mgr, planeXf, camXf, *x24_world, mwInfo, 0.f, 0.f, objectScale, false); mw->Draw(parms, xa0_curAreaId, xa4_otherAreaId, xa8_renderStates[0].x2c_drawDepth1, xa8_renderStates[0].x30_drawDepth2, false); } else { CMapWorld* mw = x24_world->IGetMapWorld(); const CMapWorldInfo& mwInfo = *g_GameState->StateForWorld(x24_world->IGetWorldAssetId()).MapWorldInfo(); zeus::CTransform modelXf = planeXf * preXf; const CMapWorld::CMapWorldDrawParms parms(xa8_renderStates[0].x34_alphaSurfaceVisited * alphaInterp, xa8_renderStates[0].x38_alphaOutlineVisited * alphaInterp, xa8_renderStates[0].x3c_alphaSurfaceUnvisited * alphaInterp, xa8_renderStates[0].x40_alphaOutlineUnvisited * alphaInterp, mapAlpha, 2.f, mgr, modelXf, camXf, *x24_world, mwInfo, 0.f, 0.f, objectScale, true); mw->Draw(parms, xa0_curAreaId, xa0_curAreaId, xa8_renderStates[0].x2c_drawDepth1, xa8_renderStates[0].x30_drawDepth2, false); } if (universeInterp > 0.f) { zeus::CTransform areaXf = mgr.GetWorld() ->GetMapWorld() ->GetMapArea(mgr.GetNextAreaId()) ->GetAreaPostTransform(*mgr.GetWorld(), mgr.GetNextAreaId()); const CMapUniverse::CMapWorldData& mwData = x8_mapu->GetMapWorldDataByWorldId(g_GameState->CurrentWorldAssetId()); zeus::CTransform universeAreaXf = mwData.GetWorldTransform() * areaXf; float minMag = FLT_MAX; int hexIdx = -1; for (u32 i = 0; i < mwData.GetNumMapAreaDatas(); ++i) { float mag = (universeAreaXf.origin - mwData.GetMapAreaData(i).origin).magnitude(); if (mag < minMag) { hexIdx = i; minMag = mag; } } const CMapUniverse::CMapUniverseDrawParms parms(universeInterp, x9c_worldIdx, g_GameState->CurrentWorldAssetId(), hexIdx, x1dc_playerFlashPulse, mgr, planeXf, camXf); x8_mapu->Draw(parms, zeus::skZero3f, 0.f, 0.f); } if (!IsInMapperState(EAutoMapperState::MapScreenUniverse)) { zeus::CTransform mapXf = planeXf * preXf; if (x24_world == mgr.GetWorld()) { float func = zeus::clamp(0.f, 0.5f * (1.f + std::sin(5.f * CGraphics::GetSecondsMod900() - (M_PIF / 2.f))), 1.f); float scale = std::min(0.6f * g_tweakAutoMapper->GetMaxCamDist() / g_tweakAutoMapper->GetMinCamDist(), objectScale); zeus::CEulerAngles eulers(mgr.GetCameraManager()->GetCurrentCameraTransform(mgr)); zeus::CRelAngle angle(eulers.z()); angle.makeRel(); zeus::CTransform playerXf(zeus::CMatrix3f::RotateZ(angle), CMapArea::GetAreaPostTranslate(*x24_world, mgr.GetNextAreaId()) + mgr.GetPlayer().GetTranslation()); CGraphics::SetModelMatrix(mapXf * playerXf * zeus::CTransform::Scale(scale * (0.25f * func + 0.75f))); float colorAlpha; if (x1bc_state != EAutoMapperState::MiniMap && x1c0_nextState != EAutoMapperState::MiniMap) { colorAlpha = 1.f; } else { colorAlpha = xa8_renderStates[0].x34_alphaSurfaceVisited; } colorAlpha *= mapAlpha; zeus::CColor modColor = g_tweakAutoMapper->GetMiniMapSamusModColor(); modColor.a() *= colorAlpha; CModelFlags flags(5, 0, 8 | 1, modColor); /* Depth GEqual */ flags.m_extendedShader = EExtendedShader::DepthGEqualNoZWrite; x30_miniMapSamus->Draw(flags); } if (IsInMapperState(EAutoMapperState::MapScreen)) { CAssetId wldMlvl = x24_world->IGetWorldAssetId(); const CMapWorld* mw = x24_world->IGetMapWorld(); std::vector<CTexturedQuadFilter>& hintBeaconFilters = m_hintBeaconFilters; if (hintBeaconFilters.size() < x1f8_hintLocations.size()) { hintBeaconFilters.reserve(x1f8_hintLocations.size()); for (u32 i = hintBeaconFilters.size(); i < x1f8_hintLocations.size(); ++i) hintBeaconFilters.emplace_back(EFilterType::Add, x3c_hintBeacon); } auto locIt = x1f8_hintLocations.cbegin(); auto filterIt = hintBeaconFilters.begin(); for (; locIt != x1f8_hintLocations.cend(); ++locIt, ++filterIt) { const SAutoMapperHintLocation& loc = *locIt; CTexturedQuadFilter& filter = *filterIt; if (loc.x8_worldId != wldMlvl) continue; const CMapArea* mapa = mw->GetMapArea(loc.xc_areaId); if (!mapa) continue; zeus::CTransform camRot(camXf.buildMatrix3f(), zeus::skZero3f); CGraphics::SetModelMatrix( mapXf * zeus::CTransform::Translate(mapa->GetAreaPostTransform(*x24_world, loc.xc_areaId).origin) * zeus::CTransform::Translate(mapa->GetAreaCenterPoint()) * zeus::CTransform::Scale(objectScale) * camRot); float beaconAlpha = 0.f; if (loc.x0_showBeacon == 1) { beaconAlpha = loc.x4_beaconAlpha; } if (beaconAlpha > 0.f) { constexpr std::array<CTexturedQuadFilter::Vert, 4> verts{{ {{-4.f, -8.f, 8.f}, {0.f, 1.f}}, {{-4.f, -8.f, 0.f}, {0.f, 0.f}}, {{4.f, -8.f, 8.f}, {1.f, 1.f}}, {{4.f, -8.f, 0.f}, {1.f, 0.f}}, }}; float colorAlpha = beaconAlpha; if (x1bc_state != EAutoMapperState::MiniMap && x1c0_nextState != EAutoMapperState::MiniMap) { } else { colorAlpha *= xa8_renderStates[0].x34_alphaSurfaceVisited; } colorAlpha *= mapAlpha; zeus::CColor color = zeus::skWhite; color.a() = colorAlpha; filter.drawVerts(color, verts); } } } } // No zread, no zwrite // Ambient color white // Disable all lights if (m_frmeInitialized) { float frmeAlpha = 0.f; if (x1bc_state != EAutoMapperState::MiniMap && x1c0_nextState != EAutoMapperState::MiniMap) { frmeAlpha = 1.f; } else { if (x1c0_nextState != EAutoMapperState::MiniMap) { if (x1c4_interpDur > 0.f) frmeAlpha = x1c8_interpTime / x1c4_interpDur; } else { if (x1c4_interpDur > 0.f) frmeAlpha = x1c8_interpTime / x1c4_interpDur; frmeAlpha = 1.f - frmeAlpha; } } CGraphics::SetDepthRange(DEPTH_NEAR, DEPTH_NEAR); CGuiWidgetDrawParms parms(frmeAlpha, zeus::skZero3f); x28_frmeMapScreen->Draw(parms); CGraphics::SetDepthRange(DEPTH_NEAR, DEPTH_HUD); } } void CAutoMapper::TransformRenderStatesWorldToUniverse() { const CMapUniverse::CMapWorldData& mapuWld = x8_mapu->GetMapWorldData(x9c_worldIdx); zeus::CQuaternion rot = zeus::CQuaternion(mapuWld.GetWorldTransform().buildMatrix3f()); xa8_renderStates[2].x8_camOrientation *= rot; xa8_renderStates[2].x20_areaPoint = mapuWld.GetWorldTransform() * xa8_renderStates[2].x20_areaPoint; xa8_renderStates[0].x8_camOrientation *= rot; xa8_renderStates[0].x20_areaPoint = mapuWld.GetWorldTransform() * xa8_renderStates[0].x20_areaPoint; xa8_renderStates[1].x8_camOrientation *= rot; xa8_renderStates[1].x20_areaPoint = mapuWld.GetWorldTransform() * xa8_renderStates[1].x20_areaPoint; } void CAutoMapper::TransformRenderStatesUniverseToWorld() { const CMapUniverse::CMapWorldData& mapuWld = x8_mapu->GetMapWorldData(x9c_worldIdx); zeus::CTransform inv = mapuWld.GetWorldTransform().inverse(); zeus::CQuaternion invRot = zeus::CQuaternion(inv.buildMatrix3f()); xa8_renderStates[2].x8_camOrientation *= invRot; xa8_renderStates[2].x20_areaPoint = inv * xa8_renderStates[2].x20_areaPoint; xa8_renderStates[0].x8_camOrientation *= invRot; xa8_renderStates[0].x20_areaPoint = inv * xa8_renderStates[0].x20_areaPoint; xa8_renderStates[1].x8_camOrientation *= invRot; xa8_renderStates[1].x20_areaPoint = inv * xa8_renderStates[1].x20_areaPoint; } void CAutoMapper::TransformRenderStateWorldToUniverse(SAutoMapperRenderState& state) { state.x20_areaPoint = x8_mapu->GetMapWorldData(x9c_worldIdx).GetWorldTransform() * xa8_renderStates[1].x20_areaPoint; } void CAutoMapper::SetupHintNavigation() { if (!g_GameState->GameOptions().GetIsHintSystemEnabled()) return; x1e0_hintSteps.clear(); x1f8_hintLocations.clear(); CHintOptions& hintOpts = g_GameState->HintOptions(); const CHintOptions::SHintState* curHint = hintOpts.GetCurrentDisplayedHint(); bool navigating = false; if (curHint && curHint->CanContinue()) { navigating = true; x1e0_hintSteps.emplace_back(SAutoMapperHintStep::ShowBeacon{}, 0.75f); const CGameHintInfo::CGameHint& nextHint = g_MemoryCardSys->GetHints()[hintOpts.GetNextHintIdx()]; CAssetId curMlvl = x24_world->IGetWorldAssetId(); for (const CGameHintInfo::SHintLocation& loc : nextHint.GetLocations()) { if (loc.x0_mlvlId != curMlvl) { x1e0_hintSteps.emplace_back(SAutoMapperHintStep::SwitchToUniverse{}); x1e0_hintSteps.emplace_back(SAutoMapperHintStep::PanToWorld{}, curMlvl); x1e0_hintSteps.emplace_back(SAutoMapperHintStep::SwitchToWorld{}, curMlvl); } else { x1e0_hintSteps.emplace_back(SAutoMapperHintStep::ZoomOut{}); } x1e0_hintSteps.emplace_back(SAutoMapperHintStep::PanToArea{}, loc.x8_areaId); x1e0_hintSteps.emplace_back(SAutoMapperHintStep::ZoomIn{}); x1e0_hintSteps.emplace_back(SAutoMapperHintStep::ShowBeacon{}, 1.f); x1f8_hintLocations.push_back({0, 0.f, loc.x0_mlvlId, loc.x8_areaId}); } } for (size_t i = 0; i < hintOpts.GetHintStates().size(); ++i) { const CHintOptions::SHintState& state = hintOpts.GetHintStates()[i]; if (navigating && hintOpts.GetNextHintIdx() == i) continue; if (state.x0_state != CHintOptions::EHintState::Displaying) continue; const CGameHintInfo::CGameHint& hint = g_MemoryCardSys->GetHints()[i]; for (const CGameHintInfo::SHintLocation& loc : hint.GetLocations()) x1f8_hintLocations.push_back({1, 1.f, loc.x0_mlvlId, loc.x8_areaId}); } } CAssetId CAutoMapper::GetAreaHintDescriptionString(CAssetId mreaId) { const CHintOptions& hintOpts = g_GameState->HintOptions(); for (size_t i = 0; i < hintOpts.GetHintStates().size(); ++i) { const CHintOptions::SHintState& state = hintOpts.GetHintStates()[i]; if (state.x0_state != CHintOptions::EHintState::Displaying) continue; const CGameHintInfo::CGameHint& memHint = g_MemoryCardSys->GetHints()[i]; for (const CGameHintInfo::SHintLocation& loc : memHint.GetLocations()) { if (loc.x4_mreaId != mreaId) continue; for (const SAutoMapperHintLocation& hintLoc : x1f8_hintLocations) { if (hintLoc.xc_areaId != loc.x8_areaId) continue; if (hintLoc.x4_beaconAlpha > 0.f) return loc.xc_stringId; } } } return -1; } void CAutoMapper::OnNewInGameGuiState(EInGameGuiState state, CStateManager& mgr) { if (state == EInGameGuiState::MapScreen) { MP1::CMain::EnsureWorldPaksReady(); CWorld& wld = *mgr.GetWorld(); wld.GetMapWorld()->SetWhichMapAreasLoaded(wld, 0, 9999); SetupHintNavigation(); BeginMapperStateTransition(EAutoMapperState::MapScreen, mgr); x28_frmeMapScreen = g_SimplePool->GetObj("FRME_MapScreen"); SetResLockState(x210_lstick, true); SetResLockState(x25c_cstick, true); SetResLockState(x2a8_ltrigger, true); SetResLockState(x2bc_rtrigger, true); SetResLockState(x2d0_abutton, true); } else { MP1::CMain::EnsureWorldPakReady(g_GameState->CurrentWorldAssetId()); if (x1bc_state == EAutoMapperState::MapScreenUniverse || x24_world == mgr.GetWorld()) { BeginMapperStateTransition(EAutoMapperState::MiniMap, mgr); x328_ = 0; } LeaveMapScreenState(); } } } // namespace urde