#include "viewmanager.h" #include "mxdirectx/mxstopwatch.h" #include "tgl/d3drm/impl.h" #include "vec.h" #include "viewlod.h" DECOMP_SIZE_ASSERT(ViewManager, 0x1bc) // GLOBAL: LEGO1 0x100dbcd8 int g_unk0x100dbcd8[18] = {0, 1, 5, 6, 2, 3, 3, 0, 4, 1, 2, 6, 0, 3, 2, 4, 5, 6}; // GLOBAL: LEGO1 0x10101050 float g_unk0x10101050 = 4.0F; // GLOBAL: LEGO1 0x10101054 float g_unk0x10101054 = 0.00097656297; // GLOBAL: LEGO1 0x10101058 int g_unk0x10101058 = 6; // GLOBAL: LEGO1 0x1010105c float g_unk0x1010105c = 0.000125F; // GLOBAL: LEGO1 0x10101060 float g_elapsedSeconds = 0; inline void SetAppData(ViewROI* p_roi, DWORD data); inline undefined4 GetD3DRM(IDirect3DRM2*& d3drm, Tgl::Renderer* pRenderer); inline undefined4 GetFrame(IDirect3DRMFrame2*& frame, Tgl::Group* scene); // FUNCTION: LEGO1 0x100a5eb0 ViewManager::ViewManager(Tgl::Renderer* pRenderer, Tgl::Group* scene, const OrientableROI* point_of_view) : scene(scene), flags(c_bit1 | c_bit2 | c_bit3 | c_bit4) { SetPOVSource(point_of_view); unk0x28 = 0.09; GetD3DRM(d3drm, pRenderer); GetFrame(frame, scene); width = 0.0; height = 0.0; view_angle = 0.0; pov.SetIdentity(); front = 0.0; back = 0.0; memset(unk0xf0, 0, sizeof(unk0xf0)); seconds_allowed = 1.0; } // FUNCTION: LEGO1 0x100a60c0 ViewManager::~ViewManager() { SetPOVSource(NULL); } // FUNCTION: LEGO1 0x100a6410 void ViewManager::Remove(ViewROI* p_roi) { for (CompoundObject::iterator it = rois.begin(); it != rois.end(); it++) { if (*it == p_roi) { rois.erase(it); if (p_roi->GetUnknown0xe0() >= 0) { FUN_100a66a0(p_roi); } const CompoundObject* comp = p_roi->GetComp(); if (comp != NULL) { for (CompoundObject::const_iterator it = comp->begin(); !(it == comp->end()); it++) { if (((ViewROI*) *it)->GetUnknown0xe0() >= 0) { FUN_100a66a0((ViewROI*) *it); } } } return; } } } // FUNCTION: LEGO1 0x100a64d0 void ViewManager::RemoveAll(ViewROI* p_roi) { if (p_roi == NULL) { for (CompoundObject::iterator it = rois.begin(); it != rois.end(); it++) { RemoveAll((ViewROI*) *it); } rois.erase(rois.begin(), rois.end()); } else { if (p_roi->GetUnknown0xe0() >= 0) { FUN_100a66a0(p_roi); } p_roi->SetUnknown0xe0(-1); const CompoundObject* comp = p_roi->GetComp(); if (comp != NULL) { for (CompoundObject::const_iterator it = comp->begin(); !(it == comp->end()); it++) { if ((ViewROI*) *it != NULL) { RemoveAll((ViewROI*) *it); } } } } } // FUNCTION: LEGO1 0x100a65b0 void ViewManager::FUN_100a65b0(ViewROI* p_roi, int p_und) { if (p_roi->GetLODCount() <= p_und) { p_und = p_roi->GetLODCount() - 1; } int unk0xe0 = p_roi->GetUnknown0xe0(); if (unk0xe0 == p_und) { return; } Tgl::Group* group = p_roi->GetGeometry(); Tgl::MeshBuilder* meshBuilder; ViewLOD* lod; if (unk0xe0 < 0) { lod = (ViewLOD*) p_roi->GetLOD(p_und); if (lod->GetUnknown0x08() & ViewLOD::c_bit4) { scene->Add((Tgl::MeshBuilder*) group); SetAppData(p_roi, (DWORD) p_roi); } } else { lod = (ViewLOD*) p_roi->GetLOD(unk0xe0); if (lod != NULL) { meshBuilder = lod->GetMeshBuilder(); if (meshBuilder != NULL) { group->Remove(meshBuilder); } } lod = (ViewLOD*) p_roi->GetLOD(p_und); } if (lod->GetUnknown0x08() & ViewLOD::c_bit4) { meshBuilder = lod->GetMeshBuilder(); if (meshBuilder != NULL) { group->Add(meshBuilder); SetAppData(p_roi, (DWORD) p_roi); p_roi->SetUnknown0xe0(p_und); return; } } p_roi->SetUnknown0xe0(-1); } // FUNCTION: LEGO1 0x100a66a0 void ViewManager::FUN_100a66a0(ViewROI* p_roi) { const ViewLOD* lod = (const ViewLOD*) p_roi->GetLOD(p_roi->GetUnknown0xe0()); if (lod != NULL) { const Tgl::MeshBuilder* meshBuilder = NULL; Tgl::Group* roiGeometry = p_roi->GetGeometry(); meshBuilder = lod->GetMeshBuilder(); if (meshBuilder != NULL) { roiGeometry->Remove(meshBuilder); } scene->Remove(roiGeometry); } p_roi->SetUnknown0xe0(-1); } // FUNCTION: LEGO1 0x100a66f0 inline void ViewManager::FUN_100a66f0(ViewROI* p_roi, int p_und) { if (p_roi->GetUnknown0x0c() == FALSE && p_und != -2) { FUN_100a66f0(p_roi, -2); } else { const CompoundObject* comp = p_roi->GetComp(); if (p_und == -1) { if (p_roi->GetWorldBoundingSphere().Radius() > 0.001F) { float und = ProjectedSize(p_roi->GetWorldBoundingSphere()); if (und < seconds_allowed * g_unk0x1010105c) { if (p_roi->GetUnknown0xe0() == -2) { return; } FUN_100a66f0(p_roi, -2); return; } p_und = Unknown2(und, RealtimeView::GetUserMaxLodPower() * seconds_allowed, p_roi); } } if (p_und == -2) { if (p_roi->GetUnknown0xe0() >= 0) { FUN_100a66a0(p_roi); p_roi->SetUnknown0xe0(-2); } if (comp != NULL) { for (CompoundObject::const_iterator it = comp->begin(); !(it == comp->end()); it++) { FUN_100a66f0((ViewROI*) *it, p_und); } } } else if (comp == NULL) { if (p_roi->GetLODs() != NULL && p_roi->GetLODCount() > 0) { FUN_100a65b0(p_roi, p_und); return; } } else { p_roi->SetUnknown0xe0(-1); for (CompoundObject::const_iterator it = comp->begin(); !(it == comp->end()); it++) { FUN_100a66f0((ViewROI*) *it, p_und); } } } } // FUNCTION: LEGO1 0x100a6930 void ViewManager::Update(float p_previousRenderTime, float) { MxStopWatch stopWatch; stopWatch.Start(); unk0x28 = p_previousRenderTime; flags |= c_bit1; if (flags & c_bit3) { Unknown(); } else if (flags & c_bit2) { FUN_100a6b90(); } for (CompoundObject::iterator it = rois.begin(); it != rois.end(); it++) { FUN_100a66f0((ViewROI*) *it, -1); } stopWatch.Stop(); g_elapsedSeconds = stopWatch.ElapsedSeconds(); } inline int ViewManager::Unknown() { flags &= ~c_bit3; if (height == 0.0F || front == 0.0F) { return -1; } else { float fVar7 = tan(view_angle / 2.0F); view_area_at_one = view_angle * view_angle * 4.0F; float fVar1 = front * fVar7; float fVar2 = (width / height) * fVar1; float uVar6 = front; float fVar3 = back + front; float fVar4 = fVar3 / front; float fVar5 = fVar4 * fVar1; fVar4 = fVar4 * fVar2; float* unk0x90 = (float*) this->unk0x90; // clang-format off *unk0x90 = fVar2; unk0x90++; *unk0x90 = fVar1; unk0x90++; *unk0x90 = uVar6; unk0x90++; *unk0x90 = fVar2; unk0x90++; *unk0x90 = -fVar1; unk0x90++; *unk0x90 = uVar6; unk0x90++; *unk0x90 = -fVar2; unk0x90++; *unk0x90 = -fVar1; unk0x90++; *unk0x90 = uVar6; unk0x90++; *unk0x90 = -fVar2; unk0x90++; *unk0x90 = fVar1; unk0x90++; *unk0x90 = uVar6; unk0x90++; *unk0x90 = fVar4; unk0x90++; *unk0x90 = fVar5; unk0x90++; *unk0x90 = fVar3; unk0x90++; *unk0x90 = fVar4; unk0x90++; *unk0x90 = -fVar5; unk0x90++; *unk0x90 = fVar3; unk0x90++; *unk0x90 = -fVar4; unk0x90++; *unk0x90 = -fVar5; unk0x90++; *unk0x90 = fVar3; unk0x90++; *unk0x90 = -fVar4; unk0x90++; *unk0x90 = fVar5; unk0x90++; *unk0x90 = fVar3; // clang-format on FUN_100a6b90(); return 0; } } inline int ViewManager::Unknown2(float p_und1, float p_und2, ViewROI* p_roi) { int result; float i; if (Unknown3(p_roi) != 0) { if (p_und1 < g_unk0x10101054) { return 0; } result = 1; } else { result = 0; } for (i = p_und2; result < g_unk0x10101058 && p_und1 >= i; i *= g_unk0x10101050) { result++; } return result; } inline int ViewManager::Unknown3(ViewROI* p_roi) { const LODListBase* lods = p_roi->GetLODs(); if (lods != NULL && lods->Size() > 0) { if (((ViewLOD*) p_roi->GetLOD(0))->GetUnknown0x08Test8()) { return 1; } return 0; } const CompoundObject* comp = p_roi->GetComp(); if (comp != NULL) { for (CompoundObject::const_iterator it = comp->begin(); !(it == comp->end()); it++) { const LODListBase* lods = ((ViewROI*) *it)->GetLODs(); if (lods != NULL && lods->Size() > 0) { if (((ViewLOD*) ((ViewROI*) *it)->GetLOD(0))->GetUnknown0x08Test8()) { return 1; } return 0; } } } return 0; } // FUNCTION: LEGO1 0x100a6b90 void ViewManager::FUN_100a6b90() { flags &= ~c_bit2; // TODO: Should be signed, but worsens match unsigned int i, j, k; for (i = 0; i < 8; i++) { for (j = 0; j < 3; j++) { unk0xf0[i][j] = pov[3][j]; for (k = 0; k < 3; k++) { unk0xf0[i][j] += pov[k][j] * unk0x90[i][k]; } } } for (i = 0; i < 6; i++) { Vector3 a(unk0xf0[g_unk0x100dbcd8[i * 3]]); Vector3 b(unk0xf0[g_unk0x100dbcd8[i * 3 + 1]]); Vector3 c(unk0xf0[g_unk0x100dbcd8[i * 3 + 2]]); Mx3DPointFloat x; Mx3DPointFloat y; Vector3 u(unk0x150[i]); x = c; ((Vector3&) x).Sub(&b); // TODO: Fix call y = a; ((Vector3&) y).Sub(&b); // TODO: Fix call u.EqualsCross(&x, &y); u.Unitize(); unk0x150[i][3] = -u.Dot(&u, &a); } flags |= c_bit4; } // FUNCTION: LEGO1 0x100a6d50 void ViewManager::SetResolution(int width, int height) { flags |= c_bit3; this->width = width; this->height = height; } // FUNCTION: LEGO1 0x100a6d70 void ViewManager::SetFrustrum(float fov, float front, float back) { this->front = front; this->back = back; flags |= c_bit3; view_angle = fov * 0.017453292519944444; } // FUNCTION: LEGO1 0x100a6da0 void ViewManager::SetPOVSource(const OrientableROI* point_of_view) { if (point_of_view != NULL) { pov = point_of_view->GetLocal2World(); flags |= c_bit2; } } // FUNCTION: LEGO1 0x100a6dc0 float ViewManager::ProjectedSize(const BoundingSphere& p_bounding_sphere) { // The algorithm projects the radius of bounding sphere onto the perpendicular // plane one unit in front of the camera. That value is simply the ratio of the // radius to the distance from the camera to the sphere center. The projected size // is then the ratio of the area of that projected circle to the view surface area // at Z == 1.0. // float sphere_projected_area = 3.14159265359 * (p_bounding_sphere.Radius() * p_bounding_sphere.Radius()); float square_dist_to_sphere = DISTSQRD3(p_bounding_sphere.Center(), pov[3]); return sphere_projected_area / view_area_at_one / square_dist_to_sphere; } // STUB: LEGO1 0x100a6e00 ViewROI* ViewManager::Pick(Tgl::View* p_view, unsigned long x, unsigned long y) { // TODO return NULL; } inline void SetAppData(ViewROI* p_roi, DWORD data) { IDirect3DRMFrame2* frame = NULL; if (GetFrame(frame, p_roi->GetGeometry()) == 0) { frame->SetAppData(data); } } inline undefined4 GetD3DRM(IDirect3DRM2*& d3drm, Tgl::Renderer* pRenderer) { d3drm = ((TglImpl::RendererImpl*) pRenderer)->ImplementationData(); return 0; } inline undefined4 GetFrame(IDirect3DRMFrame2*& frame, Tgl::Group* scene) { frame = ((TglImpl::GroupImpl*) scene)->ImplementationData(); return 0; }