metaforce/Runtime/World/CWorld.cpp

727 lines
21 KiB
C++
Raw Normal View History

#include "Runtime/World/CWorld.hpp"
#include <algorithm>
#include <iterator>
#include "Runtime/CGameState.hpp"
#include "Runtime/CInGameTweakManagerBase.hpp"
#include "Runtime/CSimplePool.hpp"
#include "Runtime/CStateManager.hpp"
#include "Runtime/IMain.hpp"
#include "Runtime/Audio/CAudioGroupSet.hpp"
#include "Runtime/Audio/CStreamAudioManager.hpp"
#include "Runtime/Graphics/CBooRenderer.hpp"
#include "Runtime/World/CScriptAreaAttributes.hpp"
#include "Runtime/World/CScriptRoomAcoustics.hpp"
2016-04-24 02:46:13 +00:00
2018-12-08 05:30:43 +00:00
namespace urde {
2016-04-24 02:46:13 +00:00
2018-12-08 05:30:43 +00:00
CWorld::CSoundGroupData::CSoundGroupData(int grpId, CAssetId agsc) : x0_groupId(grpId), x4_agscId(agsc) {
x1c_groupData = g_SimplePool->GetObj(SObjectTag{FOURCC('AGSC'), agsc});
}
2018-12-08 05:30:43 +00:00
CDummyWorld::CDummyWorld(CAssetId mlvlId, bool loadMap) : x4_loadMap(loadMap), xc_mlvlId(mlvlId) {
SObjectTag tag{FOURCC('MLVL'), mlvlId};
2019-03-09 08:58:27 +00:00
x38_bufSz = g_ResFactory->ResourceSize(tag);
x34_loadBuf.reset(new u8[x38_bufSz]);
2018-12-08 05:30:43 +00:00
x30_loadToken = g_ResFactory->LoadResourceAsync(tag, x34_loadBuf.get());
2018-05-08 05:10:24 +00:00
}
2018-12-08 05:30:43 +00:00
CDummyWorld::~CDummyWorld() {
if (x30_loadToken)
x30_loadToken->PostCancelRequest();
}
2017-08-13 05:26:14 +00:00
CAssetId CDummyWorld::IGetWorldAssetId() const { return xc_mlvlId; }
2017-08-13 05:26:14 +00:00
CAssetId CDummyWorld::IGetStringTableAssetId() const { return x10_strgId; }
2017-08-13 05:26:14 +00:00
CAssetId CDummyWorld::IGetSaveWorldAssetId() const { return x14_savwId; }
2017-01-07 01:58:05 +00:00
const CMapWorld* CDummyWorld::IGetMapWorld() const { return x2c_mapWorld.GetObj(); }
CMapWorld* CDummyWorld::IGetMapWorld() { return x2c_mapWorld.GetObj(); }
2017-01-07 01:58:05 +00:00
const IGameArea* CDummyWorld::IGetAreaAlways(TAreaId id) const { return &x18_areas.at(id); }
2017-01-07 01:58:05 +00:00
TAreaId CDummyWorld::IGetCurrentAreaId() const { return x3c_curAreaId; }
2018-12-08 05:30:43 +00:00
TAreaId CDummyWorld::IGetAreaId(CAssetId id) const {
if (!id.IsValid()) {
2016-07-24 04:46:32 +00:00
return kInvalidAreaId;
2018-12-08 05:30:43 +00:00
}
const auto iter =
2020-04-15 01:15:38 +00:00
std::find_if(x18_areas.cbegin(), x18_areas.cend(), [id](const auto& area) { return area.xc_mrea == id; });
if (iter == x18_areas.cend()) {
return kInvalidAreaId;
}
return TAreaId(std::distance(x18_areas.cbegin(), iter));
2018-12-08 05:30:43 +00:00
}
CWorld::CRelay::CRelay(CInputStream& in) {
x0_relay = in.readUint32Big();
x4_target = in.readUint32Big();
x8_msg = in.readUint16Big();
xa_active = in.readBool();
}
std::vector<CWorld::CRelay> CWorld::CRelay::ReadMemoryRelays(athena::io::MemoryReader& r) {
std::vector<CWorld::CRelay> ret;
u32 count = r.readUint32Big();
ret.reserve(count);
for (u32 i = 0; i < count; ++i)
ret.emplace_back(r);
return ret;
}
void CWorldLayers::ReadWorldLayers(athena::io::MemoryReader& r, int version, CAssetId mlvlId) {
if (version <= 14) {
2018-12-08 05:30:43 +00:00
return;
}
2018-12-08 05:30:43 +00:00
CWorldLayers ret;
u32 areaCount = r.readUint32Big();
ret.m_areas.reserve(areaCount);
for (u32 i = 0; i < areaCount; ++i) {
auto& area = ret.m_areas.emplace_back();
area.m_layerCount = r.readUint32Big();
area.m_layerBits = r.readUint64Big();
2018-12-08 05:30:43 +00:00
}
const u32 nameCount = r.readUint32Big();
2018-12-08 05:30:43 +00:00
ret.m_names.reserve(nameCount);
for (u32 i = 0; i < nameCount; ++i) {
2018-12-08 05:30:43 +00:00
ret.m_names.push_back(r.readString());
}
2018-12-08 05:30:43 +00:00
areaCount = r.readUint32Big();
for (u32 i = 0; i < areaCount; ++i) {
2018-12-08 05:30:43 +00:00
ret.m_areas[i].m_startNameIdx = r.readUint32Big();
}
2018-12-08 05:30:43 +00:00
CWorldState& wldState = g_GameState->StateForWorld(mlvlId);
wldState.GetLayerState()->InitializeWorldLayers(ret.m_areas);
}
bool CDummyWorld::ICheckWorldComplete() {
switch (x8_phase) {
case Phase::Loading: {
2019-03-09 08:58:27 +00:00
if (!x30_loadToken->IsComplete())
2018-12-08 05:30:43 +00:00
return false;
2019-03-09 08:58:27 +00:00
athena::io::MemoryReader r(x34_loadBuf.get(), x38_bufSz);
2018-12-08 05:30:43 +00:00
r.readUint32Big();
int version = r.readUint32Big();
x10_strgId = r.readUint32Big();
if (version >= 15)
x14_savwId = r.readUint32Big();
if (version >= 12)
r.readUint32Big();
if (version >= 17)
CWorld::CRelay::ReadMemoryRelays(r);
u32 areaCount = r.readUint32Big();
2018-12-08 05:30:43 +00:00
r.readUint32Big();
2018-12-08 05:30:43 +00:00
x18_areas.reserve(areaCount);
2017-01-07 01:58:05 +00:00
for (u32 i = 0; i < areaCount; ++i)
2018-12-08 05:30:43 +00:00
x18_areas.emplace_back(r, i, version);
2018-12-08 05:30:43 +00:00
x28_mapWorldId = r.readUint32Big();
if (x4_loadMap)
x2c_mapWorld = g_SimplePool->GetObj(SObjectTag{FOURCC('MAPW'), x28_mapWorldId});
2018-12-08 05:30:43 +00:00
r.readByte();
r.readUint32Big();
2018-12-08 05:30:43 +00:00
if (version > 10) {
u32 audioGroupCount = r.readUint32Big();
for (u32 i = 0; i < audioGroupCount; ++i) {
r.readUint32Big();
r.readUint32Big();
2018-12-08 05:30:43 +00:00
}
}
2018-12-08 05:30:43 +00:00
if (version > 12)
r.readString();
2018-12-08 05:30:43 +00:00
CWorldLayers::ReadWorldLayers(r, version, xc_mlvlId);
2019-03-09 08:58:27 +00:00
x30_loadToken.reset();
x34_loadBuf.reset();
x38_bufSz = 0;
2018-12-08 05:30:43 +00:00
if (x4_loadMap)
x8_phase = Phase::LoadingMap;
else {
x8_phase = Phase::Done;
return false;
}
2019-02-18 05:47:46 +00:00
[[fallthrough]];
2018-12-08 05:30:43 +00:00
}
case Phase::LoadingMap: {
if (!x2c_mapWorld.IsLoaded() || !x2c_mapWorld.GetObj())
return false;
2018-12-08 05:30:43 +00:00
x2c_mapWorld->SetWhichMapAreasLoaded(*this, 0, 9999);
x8_phase = Phase::LoadingMapAreas;
2019-02-18 05:47:46 +00:00
[[fallthrough]];
2018-12-08 05:30:43 +00:00
}
case Phase::LoadingMapAreas: {
if (x2c_mapWorld->IsMapAreasStreaming())
return false;
2018-12-08 05:30:43 +00:00
x8_phase = Phase::Done;
2019-02-18 05:47:46 +00:00
[[fallthrough]];
2018-12-08 05:30:43 +00:00
}
case Phase::Done:
return true;
default:
return false;
}
}
2017-01-07 01:58:05 +00:00
std::string CDummyWorld::IGetDefaultAudioTrack() const { return {}; }
2017-01-07 01:58:05 +00:00
int CDummyWorld::IGetAreaCount() const { return x18_areas.size(); }
2016-07-24 04:46:32 +00:00
2017-08-13 05:26:14 +00:00
CWorld::CWorld(IObjectStore& objStore, IFactory& resFactory, CAssetId mlvlId)
: x8_mlvlId(mlvlId)
, x60_objectStore(objStore)
, x64_resFactory(resFactory) {
2018-12-08 05:30:43 +00:00
SObjectTag tag{FOURCC('MLVL'), mlvlId};
2019-03-09 08:58:27 +00:00
x44_bufSz = resFactory.ResourceSize(tag);
x40_loadBuf.reset(new u8[x44_bufSz]);
x3c_loadToken = resFactory.LoadResourceAsync(tag, x40_loadBuf.get());
}
2018-12-08 05:30:43 +00:00
CWorld::~CWorld() {
StopSounds();
if (g_GameState->GetWorldTransitionManager()->IsTransitionEnabled() && g_Main->GetFlowState() == EFlowState::None)
CStreamAudioManager::StopOneShot();
else
CStreamAudioManager::StopAll();
UnloadSoundGroups();
CScriptRoomAcoustics::DisableAuxCallbacks();
2017-11-27 05:06:53 +00:00
}
CAssetId CWorld::IGetWorldAssetId() const { return GetWorldAssetId(); }
2017-08-13 05:26:14 +00:00
CAssetId CWorld::IGetStringTableAssetId() const { return xc_strgId; }
2017-08-13 05:26:14 +00:00
CAssetId CWorld::IGetSaveWorldAssetId() const { return x10_savwId; }
2017-01-07 01:58:05 +00:00
const CMapWorld* CWorld::IGetMapWorld() const { return const_cast<CWorld*>(this)->GetMapWorld(); }
CMapWorld* CWorld::IGetMapWorld() { return const_cast<CMapWorld*>(GetMapWorld()); }
2017-01-07 01:58:05 +00:00
const CGameArea* CWorld::GetAreaAlways(TAreaId id) const { return x18_areas.at(id).get(); }
2017-01-07 01:58:05 +00:00
CGameArea* CWorld::GetArea(TAreaId id) { return const_cast<CGameArea*>(GetAreaAlways(id)); }
2016-08-09 03:58:19 +00:00
2017-01-07 01:58:05 +00:00
const IGameArea* CWorld::IGetAreaAlways(TAreaId id) const { return GetAreaAlways(id); }
2016-08-09 03:58:19 +00:00
2017-01-07 01:58:05 +00:00
TAreaId CWorld::IGetCurrentAreaId() const { return x68_curAreaId; }
2018-12-08 05:30:43 +00:00
TAreaId CWorld::IGetAreaId(CAssetId id) const {
if (!id.IsValid()) {
2016-07-24 04:46:32 +00:00
return kInvalidAreaId;
2018-12-08 05:30:43 +00:00
}
const auto iter =
std::find_if(x18_areas.cbegin(), x18_areas.cend(), [id](const auto& area) { return area->x84_mrea == id; });
if (iter == x18_areas.cend()) {
return kInvalidAreaId;
}
return TAreaId(std::distance(x18_areas.cbegin(), iter));
2016-07-24 04:46:32 +00:00
}
2018-12-08 05:30:43 +00:00
void CWorld::MoveToChain(CGameArea* area, EChain chain) {
if (area->x138_curChain == chain) {
2018-12-08 05:30:43 +00:00
return;
}
if (area->x138_curChain != EChain::Invalid) {
if (x4c_chainHeads[size_t(area->x138_curChain)] == area) {
x4c_chainHeads[size_t(area->x138_curChain)] = area->x130_next;
}
}
area->SetChain(x4c_chainHeads[size_t(chain)], chain);
x4c_chainHeads[size_t(chain)] = area;
2016-07-24 04:46:32 +00:00
}
2018-12-08 05:30:43 +00:00
void CWorld::MoveAreaToAliveChain(TAreaId aid) { MoveToChain(x18_areas[aid].get(), EChain::Alive); }
2017-02-19 09:27:01 +00:00
2018-12-08 05:30:43 +00:00
void CWorld::LoadSoundGroup(int groupId, CAssetId agscId, CSoundGroupData& data) {
if (!CAudioSys::SysLoadGroupSet(g_SimplePool, agscId)) {
auto name = CAudioSys::SysGetGroupSetName(agscId);
CAudioSys::SysAddGroupIntoAmuse(name);
data.xc_name = name;
++x6c_loadedAudioGrpCount;
}
2017-11-27 05:06:53 +00:00
}
2017-01-07 01:58:05 +00:00
void CWorld::LoadSoundGroups() {}
2018-12-08 05:30:43 +00:00
void CWorld::UnloadSoundGroups() {
for (CSoundGroupData& data : x74_soundGroupData) {
CAudioSys::SysRemoveGroupFromAmuse(data.xc_name);
CAudioSys::SysUnloadAudioGroupSet(data.xc_name);
}
}
void CWorld::StopSounds() {
for (CSfxHandle& hnd : xc8_globalSfxHandles)
CSfxManager::RemoveEmitter(hnd);
xc8_globalSfxHandles.clear();
}
bool CWorld::CheckWorldComplete(CStateManager* mgr, TAreaId id, CAssetId mreaId) {
if (mreaId.IsValid()) {
x68_curAreaId = 0;
TAreaId aid = 0;
for (const std::unique_ptr<CGameArea>& area : x18_areas) {
if (area->x84_mrea == mreaId) {
x68_curAreaId = aid;
break;
}
++aid;
}
} else
x68_curAreaId = id;
switch (x4_phase) {
case Phase::Loading: {
2019-03-09 08:58:27 +00:00
if (!x3c_loadToken->IsComplete())
2018-12-08 05:30:43 +00:00
return false;
2019-03-09 08:58:27 +00:00
athena::io::MemoryReader r(x40_loadBuf.get(), x44_bufSz);
2018-12-08 05:30:43 +00:00
r.readUint32Big();
int version = r.readUint32Big();
xc_strgId = r.readUint32Big();
if (version >= 15)
x10_savwId = r.readUint32Big();
if (version >= 12) {
CAssetId skyboxId = r.readUint32Big();
if (skyboxId.IsValid() && mgr)
2019-11-06 18:12:20 +00:00
x94_skyboxWorld = g_SimplePool->GetObj(SObjectTag{FOURCC('CMDL'), skyboxId});
2018-12-08 05:30:43 +00:00
}
if (version >= 17)
x2c_relays = CWorld::CRelay::ReadMemoryRelays(r);
2016-07-24 04:46:32 +00:00
2018-12-08 05:30:43 +00:00
u32 areaCount = r.readUint32Big();
r.readUint32Big();
2016-07-24 04:46:32 +00:00
2018-12-08 05:30:43 +00:00
x18_areas.reserve(areaCount);
for (u32 i = 0; i < areaCount; ++i) {
2018-12-08 05:30:43 +00:00
x18_areas.push_back(std::make_unique<CGameArea>(r, i, version));
}
2016-07-24 04:46:32 +00:00
if (x48_chainCount < x4c_chainHeads.size()) {
for (size_t i = x48_chainCount; i < x4c_chainHeads.size(); ++i) {
2018-12-08 05:30:43 +00:00
x4c_chainHeads[i] = nullptr;
}
x48_chainCount = x4c_chainHeads.size();
2018-12-08 05:30:43 +00:00
}
2016-07-24 04:46:32 +00:00
2018-12-08 05:30:43 +00:00
for (std::unique_ptr<CGameArea>& area : x18_areas)
MoveToChain(area.get(), EChain::Deallocated);
2016-07-24 04:46:32 +00:00
2018-12-08 05:30:43 +00:00
x24_mapwId = r.readUint32Big();
x28_mapWorld = g_SimplePool->GetObj(SObjectTag{FOURCC('MAPW'), x24_mapwId});
2016-07-24 04:46:32 +00:00
2018-12-08 05:30:43 +00:00
if (mgr) {
std::vector<TEditorId> ids;
mgr->LoadScriptObjects(kInvalidAreaId, r, ids);
mgr->InitScriptObjects(ids);
2016-07-24 04:46:32 +00:00
}
2018-12-08 05:30:43 +00:00
if (version > 10) {
u32 audioGroupCount = r.readUint32Big();
x74_soundGroupData.reserve(audioGroupCount);
for (u32 i = 0; i < audioGroupCount; ++i) {
int grpId = r.readUint32Big();
CAssetId agscId = r.readUint32Big();
x74_soundGroupData.emplace_back(grpId, agscId);
}
2016-07-24 04:46:32 +00:00
}
2018-12-08 05:30:43 +00:00
if (version > 12) {
x84_defAudioTrack = r.readString();
2020-04-11 22:51:39 +00:00
std::string trackKey = fmt::format(FMT_STRING("WorldDefault: {}"), x8_mlvlId);
2018-12-08 05:30:43 +00:00
if (g_TweakManager->HasTweakValue(trackKey))
x84_defAudioTrack = g_TweakManager->GetTweakValue(trackKey)->GetAudio().GetFileName();
}
2018-12-08 05:30:43 +00:00
CWorldLayers::ReadWorldLayers(r, version, x8_mlvlId);
2019-03-09 08:58:27 +00:00
x3c_loadToken.reset();
x40_loadBuf.reset();
x44_bufSz = 0;
2018-12-08 05:30:43 +00:00
x4_phase = Phase::LoadingMap;
2019-02-18 05:47:46 +00:00
[[fallthrough]];
2018-12-08 05:30:43 +00:00
}
case Phase::LoadingMap: {
if (!x28_mapWorld.IsLoaded() || !x28_mapWorld.GetObj())
return false;
2016-07-24 04:46:32 +00:00
2018-12-08 05:30:43 +00:00
if (x68_curAreaId == kInvalidAreaId)
x28_mapWorld->SetWhichMapAreasLoaded(*this, 0, 9999);
else
x28_mapWorld->SetWhichMapAreasLoaded(*this, x68_curAreaId, 3);
x4_phase = Phase::LoadingMapAreas;
2019-02-18 05:47:46 +00:00
[[fallthrough]];
2018-12-08 05:30:43 +00:00
}
case Phase::LoadingMapAreas: {
if (x28_mapWorld->IsMapAreasStreaming())
return false;
x4_phase = Phase::LoadingSkyBox;
2019-02-18 05:47:46 +00:00
[[fallthrough]];
2018-12-08 05:30:43 +00:00
}
case Phase::LoadingSkyBox: {
x70_26_skyboxActive = true;
x70_27_skyboxVisible = false;
if (x94_skyboxWorld) {
CModel* skybox = x94_skyboxWorld->GetObj();
if (!skybox)
return false;
2018-12-08 05:30:43 +00:00
skybox->GetInstance().Touch(0);
if (!skybox->IsLoaded(0))
return false;
2018-12-08 05:30:43 +00:00
xa4_skyboxWorldLoaded = x94_skyboxWorld;
}
2018-12-08 05:30:43 +00:00
for (CSoundGroupData& group : x74_soundGroupData)
group.x1c_groupData.Lock();
x4_phase = Phase::LoadingSoundGroups;
2019-02-18 05:47:46 +00:00
[[fallthrough]];
2018-12-08 05:30:43 +00:00
}
case Phase::LoadingSoundGroups: {
bool allLoaded = true;
for (CSoundGroupData& group : x74_soundGroupData) {
bool loaded = group.x1c_groupData.IsLoaded();
allLoaded &= loaded;
if (loaded) {
CAudioGroupSet* groupData = group.x1c_groupData.GetObj();
if (groupData)
LoadSoundGroup(group.x0_groupId, group.x4_agscId, group);
}
}
if (!allLoaded)
return false;
LoadSoundGroups();
x4_phase = Phase::Done;
2019-02-18 05:47:46 +00:00
[[fallthrough]];
2018-12-08 05:30:43 +00:00
}
case Phase::Done:
return true;
default:
2019-11-09 23:48:46 +00:00
break;
2018-12-08 05:30:43 +00:00
}
2018-12-08 05:30:43 +00:00
return false;
}
2018-12-08 05:30:43 +00:00
bool CWorld::ScheduleAreaToLoad(CGameArea* area, CStateManager& mgr) {
if (!area->IsPostConstructed()) {
MoveToChain(area, EChain::Loading);
return true;
} else {
if (area->x138_curChain != EChain::Alive) {
if (area->x138_curChain != EChain::AliveJudgement) {
2017-04-13 19:28:31 +00:00
x70_24_currentAreaNeedsAllocation = true;
2018-12-08 05:30:43 +00:00
}
MoveToChain(area, EChain::Alive);
}
2018-12-08 05:30:43 +00:00
return false;
}
}
void CWorld::TravelToArea(TAreaId aid, CStateManager& mgr, bool skipLoadOther) {
if (aid < 0 || aid >= x18_areas.size())
return;
x70_24_currentAreaNeedsAllocation = false;
x68_curAreaId = aid;
CGameArea* toDeallocateAreas = x4c_chainHeads[0];
while (toDeallocateAreas) {
if (toDeallocateAreas->Invalidate(&mgr)) {
MoveToChain(toDeallocateAreas, EChain::Deallocated);
break;
}
toDeallocateAreas = toDeallocateAreas->x130_next;
}
CGameArea* aliveAreas = x4c_chainHeads[3];
while (aliveAreas) {
CGameArea* aliveArea = aliveAreas;
aliveAreas = aliveAreas->x130_next;
MoveToChain(aliveArea, EChain::AliveJudgement);
}
CGameArea* loadingAreas = x4c_chainHeads[2];
while (loadingAreas) {
CGameArea* loadingArea = loadingAreas;
loadingAreas = loadingAreas->x130_next;
MoveToChain(loadingArea, EChain::ToDeallocate);
}
CGameArea* area = x18_areas[aid].get();
if (area->x138_curChain != EChain::AliveJudgement)
x70_24_currentAreaNeedsAllocation = true;
area->Validate(mgr);
MoveToChain(area, EChain::Alive);
area->SetOcclusionState(CGameArea::EOcclusionState::Visible);
CGameArea* otherLoadArea = nullptr;
if (!skipLoadOther) {
bool otherLoading = false;
for (CGameArea::Dock& dock : area->xcc_docks) {
u32 dockRefCount = dock.GetDockRefs().size();
for (u32 i = 0; i < dockRefCount; ++i) {
if (!dock.ShouldLoadOtherArea(i))
continue;
TAreaId connArea = dock.GetConnectedAreaId(i);
CGameArea* cArea = x18_areas[connArea].get();
if (!cArea->xf0_25_active)
continue;
if (!otherLoading) {
otherLoading = ScheduleAreaToLoad(cArea, mgr);
if (!otherLoading)
continue;
otherLoadArea = cArea;
} else
ScheduleAreaToLoad(cArea, mgr);
}
}
2018-12-08 05:30:43 +00:00
}
CGameArea* judgementAreas = x4c_chainHeads[4];
while (judgementAreas) {
CGameArea* judgementArea = judgementAreas;
judgementAreas = judgementArea->x130_next;
MoveToChain(judgementArea, EChain::ToDeallocate);
}
2018-12-08 05:30:43 +00:00
size_t toStreamCount = 0;
toDeallocateAreas = x4c_chainHeads[0];
while (toDeallocateAreas) {
toDeallocateAreas->RemoveStaticGeometry();
toDeallocateAreas = toDeallocateAreas->x130_next;
++toStreamCount;
}
2018-12-08 05:30:43 +00:00
if (!toStreamCount && otherLoadArea && !x70_25_loadPaused)
otherLoadArea->StartStreamIn(mgr);
2018-12-08 05:30:43 +00:00
x28_mapWorld->SetWhichMapAreasLoaded(*this, aid, 3);
}
2018-12-08 05:30:43 +00:00
void CWorld::SetLoadPauseState(bool paused) {
for (auto it = GetChainHead(EChain::Loading); it != AliveAreasEnd(); ++it)
it->SetLoadPauseState(paused);
x70_25_loadPaused = paused;
2017-02-18 02:19:50 +00:00
}
2018-12-08 05:30:43 +00:00
void CWorld::CycleLoadPauseState() {
if (!x70_25_loadPaused) {
SetLoadPauseState(true);
SetLoadPauseState(false);
}
2018-05-14 04:38:58 +00:00
}
2017-01-07 01:58:05 +00:00
bool CWorld::ICheckWorldComplete() { return CheckWorldComplete(nullptr, kInvalidAreaId, -1); }
2017-01-07 01:58:05 +00:00
std::string CWorld::IGetDefaultAudioTrack() const { return x84_defAudioTrack; }
2017-01-07 01:58:05 +00:00
int CWorld::IGetAreaCount() const { return x18_areas.size(); }
bool CWorld::DoesAreaExist(TAreaId area) const { return (area >= 0 && area < x18_areas.size()); }
2018-12-08 05:30:43 +00:00
void CWorld::PropogateAreaChain(CGameArea::EOcclusionState occlusionState, CGameArea* area, CWorld* world) {
if (!area->GetPostConstructed() || occlusionState == area->GetOcclusionState())
return;
2017-01-07 01:58:05 +00:00
2018-12-08 05:30:43 +00:00
if (occlusionState == CGameArea::EOcclusionState::Visible)
area->SetOcclusionState(CGameArea::EOcclusionState::Visible);
2018-12-08 05:30:43 +00:00
for (CGameArea& areaItr : *world) {
if (&areaItr == area)
continue;
if (areaItr.IsPostConstructed() && areaItr.GetOcclusionState() == CGameArea::EOcclusionState::Visible)
areaItr.OtherAreaOcclusionChanged();
}
for (CGameArea& areaItr : *world) {
if (&areaItr == area)
continue;
if (areaItr.IsPostConstructed() && areaItr.GetOcclusionState() == CGameArea::EOcclusionState::Occluded)
areaItr.OtherAreaOcclusionChanged();
}
if (occlusionState == CGameArea::EOcclusionState::Occluded)
area->SetOcclusionState(CGameArea::EOcclusionState::Occluded);
}
void CWorld::Update(float dt) {
xc4_neededFx = EEnvFxType::None;
CAssetId overrideSkyId;
bool needsSky = false;
bool skyVisible = false;
u32 areaCount = 0;
for (auto head = GetChainHead(EChain::Alive); head != AliveAreasEnd(); ++head, ++areaCount) {
2018-12-08 05:30:43 +00:00
head->AliveUpdate(dt);
if (head->DoesAreaNeedSkyNow()) {
const CScriptAreaAttributes* attrs = head->GetPostConstructed()->x10d8_areaAttributes;
if (attrs && attrs->GetSkyModel().IsValid())
overrideSkyId = attrs->GetSkyModel();
needsSky = true;
CGameArea::EOcclusionState occlusionState =
(head->IsPostConstructed() ? head->GetPostConstructed()->x10dc_occlusionState
: CGameArea::EOcclusionState::Occluded);
if (occlusionState == CGameArea::EOcclusionState::Visible)
skyVisible = true;
}
EEnvFxType envFxType = head->DoesAreaNeedEnvFx();
if (envFxType != EEnvFxType::None)
xc4_neededFx = envFxType;
}
if (areaCount == 0)
return;
if (overrideSkyId.IsValid() && needsSky) {
x70_26_skyboxActive = true;
x70_27_skyboxVisible = skyVisible;
2019-11-06 18:12:20 +00:00
xb4_skyboxOverride = g_SimplePool->GetObj({SBIG('CMDL'), overrideSkyId});
xa4_skyboxWorldLoaded.reset();
2018-12-08 05:30:43 +00:00
if (x94_skyboxWorld)
x94_skyboxWorld->Unlock();
2018-12-08 05:30:43 +00:00
} else {
2019-11-06 18:12:20 +00:00
xb4_skyboxOverride.reset();
2018-12-08 05:30:43 +00:00
if (!x94_skyboxWorld) {
x70_26_skyboxActive = false;
x70_27_skyboxVisible = false;
} else if (!needsSky) {
2019-11-06 18:12:20 +00:00
xa4_skyboxWorldLoaded.reset();
x94_skyboxWorld->Unlock();
2018-12-08 05:30:43 +00:00
x70_26_skyboxActive = false;
x70_27_skyboxVisible = false;
} else {
if (!xa4_skyboxWorldLoaded) {
x94_skyboxWorld->Lock();
if (x94_skyboxWorld->IsLoaded()) {
x94_skyboxWorld.value()->Touch(0);
if (x94_skyboxWorld.value()->IsLoaded(0))
2018-12-08 05:30:43 +00:00
xa4_skyboxWorldLoaded = x94_skyboxWorld;
2017-11-12 05:14:57 +00:00
}
2018-12-08 05:30:43 +00:00
}
x70_26_skyboxActive = true;
x70_27_skyboxVisible = skyVisible;
}
2018-12-08 05:30:43 +00:00
}
2017-03-24 05:30:16 +00:00
}
2018-12-08 05:30:43 +00:00
void CWorld::PreRender() {
for (auto head = GetChainHead(EChain::Alive); head != AliveAreasEnd(); ++head) {
2018-12-08 05:30:43 +00:00
head->PreRender();
}
}
2018-12-08 05:30:43 +00:00
void CWorld::TouchSky() {
if (xa4_skyboxWorldLoaded && xa4_skyboxWorldLoaded->IsLoaded())
xa4_skyboxWorldLoaded.value()->Touch(0);
if (xb4_skyboxOverride && xb4_skyboxOverride->IsLoaded())
xb4_skyboxOverride.value()->Touch(0);
}
2018-12-08 05:30:43 +00:00
void CWorld::DrawSky(const zeus::CTransform& xf) const {
const CModel* model;
if (xa4_skyboxWorldLoaded)
model = xa4_skyboxWorldLoaded->GetObj();
2018-12-08 05:30:43 +00:00
else if (xb4_skyboxOverride)
model = xb4_skyboxOverride->GetObj();
2018-12-08 05:30:43 +00:00
else
return;
2018-12-08 05:30:43 +00:00
if (!x70_27_skyboxVisible)
return;
2019-07-21 08:42:52 +00:00
SCOPED_GRAPHICS_DEBUG_GROUP("CWorld::DrawSky", zeus::skCyan);
2018-12-08 05:30:43 +00:00
CGraphics::DisableAllLights();
CGraphics::SetModelMatrix(xf);
g_Renderer->SetAmbientColor(zeus::skWhite);
2018-12-08 05:30:43 +00:00
CGraphics::SetDepthRange(DEPTH_SKY, DEPTH_FAR);
2020-10-21 05:23:34 +00:00
constexpr CModelFlags flags(0, 0, 1, zeus::skWhite);
2018-12-08 05:30:43 +00:00
model->Draw(flags);
2018-12-08 05:30:43 +00:00
CGraphics::SetDepthRange(DEPTH_WORLD, DEPTH_FAR);
}
2017-07-07 12:23:20 +00:00
2018-12-08 05:30:43 +00:00
void CWorld::StopGlobalSound(u16 id) {
auto search = std::find_if(xc8_globalSfxHandles.begin(), xc8_globalSfxHandles.end(),
[id](CSfxHandle& hnd) { return hnd->GetSfxId() == id; });
if (search != xc8_globalSfxHandles.end()) {
CSfxManager::RemoveEmitter(*search);
xc8_globalSfxHandles.erase(search);
}
2017-11-27 05:06:53 +00:00
}
2018-12-08 05:30:43 +00:00
bool CWorld::HasGlobalSound(u16 id) const {
auto search = std::find_if(xc8_globalSfxHandles.begin(), xc8_globalSfxHandles.end(),
[id](const CSfxHandle& hnd) { return hnd->GetSfxId() == id; });
return search != xc8_globalSfxHandles.end();
2017-11-27 05:06:53 +00:00
}
2018-12-08 05:30:43 +00:00
void CWorld::AddGlobalSound(const CSfxHandle& hnd) {
if (xc8_globalSfxHandles.size() >= xc8_globalSfxHandles.capacity())
return;
xc8_globalSfxHandles.push_back(hnd);
2017-07-07 12:23:20 +00:00
}
2018-12-08 05:30:43 +00:00
bool CWorld::AreSkyNeedsMet() const {
if (!x70_26_skyboxActive)
return true;
if (xb4_skyboxOverride && xb4_skyboxOverride.value()->IsLoaded(0))
2018-12-08 05:30:43 +00:00
return true;
if (xa4_skyboxWorldLoaded && xa4_skyboxWorldLoaded.value()->IsLoaded(0))
2018-12-08 05:30:43 +00:00
return true;
return x94_skyboxWorld && x94_skyboxWorld.value()->IsLoaded(0);
}
2018-12-08 05:30:43 +00:00
TAreaId CWorld::GetAreaIdForSaveId(s32 saveId) const {
if (saveId == -1) {
2018-12-08 05:30:43 +00:00
return kInvalidAreaId;
}
if (x18_areas.empty()) {
2018-12-08 05:30:43 +00:00
return kInvalidAreaId;
}
const auto iter = std::find_if(x18_areas.cbegin(), x18_areas.cend(),
[saveId](const auto& area) { return area->x88_areaId == saveId; });
if (iter == x18_areas.cend()) {
return kInvalidAreaId;
2018-12-08 05:30:43 +00:00
}
return TAreaId(std::distance(x18_areas.cbegin(), iter));
2016-04-24 02:46:13 +00:00
}
2018-12-08 05:30:43 +00:00
} // namespace urde