mirror of
				https://github.com/isledecomp/isle.git
				synced 2025-10-26 09:54:18 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			879 lines
		
	
	
		
			20 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			879 lines
		
	
	
		
			20 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| #include "legoroi.h"
 | |
| 
 | |
| #include "anim/legoanim.h"
 | |
| #include "legolod.h"
 | |
| #include "misc/legocontainer.h"
 | |
| #include "misc/legostorage.h"
 | |
| #include "mxgeometry/mxgeometry4d.h"
 | |
| #include "realtime/realtime.h"
 | |
| #include "shape/legobox.h"
 | |
| #include "shape/legosphere.h"
 | |
| 
 | |
| #include <string.h>
 | |
| #include <vec.h>
 | |
| 
 | |
| DECOMP_SIZE_ASSERT(LegoROI, 0x108)
 | |
| DECOMP_SIZE_ASSERT(TimeROI, 0x10c)
 | |
| 
 | |
| // SIZE 0x14
 | |
| typedef struct {
 | |
| 	const char* m_name;
 | |
| 	int m_red;
 | |
| 	int m_green;
 | |
| 	int m_blue;
 | |
| 	int m_alpha;
 | |
| } ROIColorAlias;
 | |
| 
 | |
| // GLOBAL: LEGO1 0x101011b0
 | |
| ROIColorAlias g_roiColorAliases[22] = {
 | |
| 	{"lego black", 0x21, 0x21, 0x21, 0},       {"lego black f", 0x21, 0x21, 0x21, 0},
 | |
| 	{"lego black flat", 0x21, 0x21, 0x21, 0},  {"lego blue", 0x00, 0x54, 0x8c, 0},
 | |
| 	{"lego blue flat", 0x00, 0x54, 0x8c, 0},   {"lego brown", 0x4a, 0x23, 0x1a, 0},
 | |
| 	{"lego brown flt", 0x4a, 0x23, 0x1a, 0},   {"lego brown flat", 0x4a, 0x23, 0x1a, 0},
 | |
| 	{"lego drk grey", 0x40, 0x40, 0x40, 0},    {"lego drk grey flt", 0x40, 0x40, 0x40, 0},
 | |
| 	{"lego dk grey flt", 0x40, 0x40, 0x40, 0}, {"lego green", 0x00, 0x78, 0x2d, 0},
 | |
| 	{"lego green flat", 0x00, 0x78, 0x2d, 0},  {"lego lt grey", 0x82, 0x82, 0x82, 0},
 | |
| 	{"lego lt grey flt", 0x82, 0x82, 0x82, 0}, {"lego lt grey fla", 0x82, 0x82, 0x82, 0},
 | |
| 	{"lego red", 0xcb, 0x12, 0x20, 0},         {"lego red flat", 0xcb, 0x12, 0x20, 0},
 | |
| 	{"lego white", 0xfa, 0xfa, 0xfa, 0},       {"lego white flat", 0xfa, 0xfa, 0xfa, 0},
 | |
| 	{"lego yellow", 0xff, 0xb9, 0x00, 0},      {"lego yellow flat", 0xff, 0xb9, 0x00, 0},
 | |
| };
 | |
| 
 | |
| // GLOBAL: LEGO1 0x10101368
 | |
| int g_roiConfig = 100;
 | |
| 
 | |
| // GLOBAL: LEGO1 0x10101370
 | |
| const char* g_unk0x10101370[] = {"bike", "moto", NULL};
 | |
| 
 | |
| // GLOBAL: LEGO1 0x10101380
 | |
| const char* g_unk0x10101380[] = {"bike", "moto", "haus", NULL};
 | |
| 
 | |
| // GLOBAL: LEGO1 0x10101390
 | |
| const char* g_unk0x10101390[] = {"rcuser", "jsuser", "dunebugy", "chtrblad", "chtrbody", "chtrshld", NULL};
 | |
| 
 | |
| // GLOBAL: LEGO1 0x101013ac
 | |
| ColorOverride g_colorOverride = NULL;
 | |
| 
 | |
| // GLOBAL: LEGO1 0x101013b0
 | |
| TextureHandler g_textureHandler = NULL;
 | |
| 
 | |
| // FUNCTION: LEGO1 0x100a81b0
 | |
| // FUNCTION: BETA10 0x101898c0
 | |
| // FUNCTION: ALPHA 0x100bb1c0
 | |
| void LegoROI::FUN_100a81b0(const LegoChar* p_error, ...)
 | |
| {
 | |
| 	// Probably a printf-like debug function that was removed early.
 | |
| 	// No known implementation in any of the binaries.
 | |
| }
 | |
| 
 | |
| // FUNCTION: LEGO1 0x100a81c0
 | |
| void LegoROI::configureLegoROI(int p_roiConfig)
 | |
| {
 | |
| 	g_roiConfig = p_roiConfig;
 | |
| }
 | |
| 
 | |
| // FUNCTION: LEGO1 0x100a81d0
 | |
| // FUNCTION: BETA10 0x101898e8
 | |
| LegoROI::LegoROI(Tgl::Renderer* p_renderer) : ViewROI(p_renderer, NULL)
 | |
| {
 | |
| 	m_parentROI = NULL;
 | |
| 	m_name = NULL;
 | |
| 	m_entity = NULL;
 | |
| }
 | |
| 
 | |
| // FUNCTION: LEGO1 0x100a82d0
 | |
| // FUNCTION: BETA10 0x10189994
 | |
| LegoROI::LegoROI(Tgl::Renderer* p_renderer, ViewLODList* p_lodList) : ViewROI(p_renderer, p_lodList)
 | |
| {
 | |
| 	m_parentROI = NULL;
 | |
| 	m_name = NULL;
 | |
| 	m_entity = NULL;
 | |
| }
 | |
| 
 | |
| // FUNCTION: LEGO1 0x100a83c0
 | |
| // FUNCTION: BETA10 0x10189a42
 | |
| LegoROI::~LegoROI()
 | |
| {
 | |
| 	if (comp) {
 | |
| 		CompoundObject::iterator iterator;
 | |
| 
 | |
| 		for (iterator = comp->begin(); !(iterator == comp->end()); ++iterator) {
 | |
| 			ROI* child = *iterator;
 | |
| 
 | |
| 			delete child;
 | |
| 		}
 | |
| 
 | |
| 		delete comp;
 | |
| 		comp = 0;
 | |
| 	}
 | |
| 	if (m_name) {
 | |
| 		delete[] m_name;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // FUNCTION: LEGO1 0x100a84a0
 | |
| // FUNCTION: BETA10 0x10189b99
 | |
| LegoResult LegoROI::Read(
 | |
| 	OrientableROI* p_unk0xd4,
 | |
| 	Tgl::Renderer* p_renderer,
 | |
| 	ViewLODListManager* p_viewLODListManager,
 | |
| 	LegoTextureContainer* p_textureContainer,
 | |
| 	LegoStorage* p_storage
 | |
| )
 | |
| {
 | |
| 	LegoResult result = FAILURE;
 | |
| 	LegoU32 i, j;
 | |
| 	LegoU32 numLODs, surplusLODs;
 | |
| 	LegoROI* roi;
 | |
| 	LegoLOD* lod;
 | |
| 	LegoU32 length, roiLength;
 | |
| 	LegoChar *roiName, *textureName;
 | |
| 	LegoTextureInfo* textureInfo;
 | |
| 	ViewLODList* lodList;
 | |
| 	LegoU32 numROIs;
 | |
| 	LegoSphere sphere;
 | |
| 	LegoBox box;
 | |
| 
 | |
| 	m_parentROI = p_unk0xd4;
 | |
| 
 | |
| 	if (p_storage->Read(&length, sizeof(LegoU32)) != SUCCESS) {
 | |
| 		goto done;
 | |
| 	}
 | |
| 	m_name = new LegoChar[length + 1];
 | |
| 	if (p_storage->Read(m_name, length) != SUCCESS) {
 | |
| 		goto done;
 | |
| 	}
 | |
| 	m_name[length] = '\0';
 | |
| 	strlwr(m_name);
 | |
| 
 | |
| 	if (sphere.Read(p_storage) != SUCCESS) {
 | |
| 		goto done;
 | |
| 	}
 | |
| 
 | |
| 	SET3(m_sphere.Center(), sphere.GetCenter());
 | |
| 	m_sphere.Radius() = sphere.GetRadius();
 | |
| 	m_world_bounding_sphere.Radius() = m_sphere.Radius();
 | |
| 
 | |
| 	if (box.Read(p_storage) != SUCCESS) {
 | |
| 		goto done;
 | |
| 	}
 | |
| 
 | |
| 	SET3(m_bounding_box.Min(), box.GetMin());
 | |
| 	SET3(m_bounding_box.Max(), box.GetMax());
 | |
| 
 | |
| 	if (p_storage->Read(&length, sizeof(LegoU32)) != SUCCESS) {
 | |
| 		goto done;
 | |
| 	}
 | |
| 
 | |
| 	if (length != 0) {
 | |
| 		textureName = new LegoChar[length + 1];
 | |
| 		if (p_storage->Read(textureName, length) != SUCCESS) {
 | |
| 			goto done;
 | |
| 		}
 | |
| 		textureName[length] = '\0';
 | |
| 		strlwr(textureName);
 | |
| 	}
 | |
| 	else {
 | |
| 		textureName = NULL;
 | |
| 	}
 | |
| 
 | |
| 	if (p_storage->Read(&m_unk0x100, sizeof(undefined)) != SUCCESS) {
 | |
| 		goto done;
 | |
| 	}
 | |
| 
 | |
| 	if (m_unk0x100) {
 | |
| 		for (roiLength = strlen(m_name); roiLength; roiLength--) {
 | |
| 			if (m_name[roiLength - 1] < '0' || m_name[roiLength - 1] > '9') {
 | |
| 				break;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		roiName = new LegoChar[roiLength + 1];
 | |
| 		memcpy(roiName, m_name, roiLength);
 | |
| 		roiName[roiLength] = '\0';
 | |
| 
 | |
| 		lodList = p_viewLODListManager->Lookup(roiName);
 | |
| 		delete[] roiName;
 | |
| 
 | |
| 		if (lodList == NULL) {
 | |
| 			goto done;
 | |
| 		}
 | |
| 	}
 | |
| 	else {
 | |
| 		if (p_storage->Read(&numLODs, sizeof(LegoU32)) != SUCCESS) {
 | |
| 			goto done;
 | |
| 		}
 | |
| 
 | |
| 		if (!numLODs) {
 | |
| 			lodList = NULL;
 | |
| 		}
 | |
| 		else {
 | |
| 			const LegoChar* roiName = m_name;
 | |
| 			LegoU32 offset;
 | |
| 
 | |
| 			if (p_storage->Read(&offset, sizeof(LegoU32)) != SUCCESS) {
 | |
| 				goto done;
 | |
| 			}
 | |
| 
 | |
| 			if (numLODs > g_roiConfig) {
 | |
| 				surplusLODs = numLODs - g_roiConfig;
 | |
| 				numLODs = g_roiConfig;
 | |
| 			}
 | |
| 			else {
 | |
| 				surplusLODs = 0;
 | |
| 			}
 | |
| 
 | |
| 			if (g_roiConfig <= 2) {
 | |
| 				for (i = 0; g_unk0x10101380[i] != NULL; i++) {
 | |
| 					if (!strnicmp(m_name, g_unk0x10101380[i], 4)) {
 | |
| 						roiName = g_unk0x10101380[i];
 | |
| 						break;
 | |
| 					}
 | |
| 				}
 | |
| 			}
 | |
| 			else {
 | |
| 				for (i = 0; g_unk0x10101370[i] != NULL; i++) {
 | |
| 					if (!strnicmp(m_name, g_unk0x10101370[i], 4)) {
 | |
| 						roiName = g_unk0x10101370[i];
 | |
| 						break;
 | |
| 					}
 | |
| 				}
 | |
| 			}
 | |
| 
 | |
| 			if ((lodList = p_viewLODListManager->Lookup(roiName))) {
 | |
| 				for (j = 0; g_unk0x10101390[j] != NULL; j++) {
 | |
| 					if (!strcmpi(g_unk0x10101390[j], roiName)) {
 | |
| 						break;
 | |
| 					}
 | |
| 				}
 | |
| 
 | |
| 				if (g_unk0x10101390[j] != NULL) {
 | |
| 					while (lodList->Size()) {
 | |
| 						delete const_cast<ViewLOD*>(lodList->PopBack());
 | |
| 					}
 | |
| 
 | |
| 					for (j = 0; j < numLODs; j++) {
 | |
| 						lod = new LegoLOD(p_renderer);
 | |
| 						if (lod->Read(p_renderer, p_textureContainer, p_storage) != SUCCESS) {
 | |
| 							goto done;
 | |
| 						}
 | |
| 
 | |
| 						if (j == 0) {
 | |
| 							if (surplusLODs != 0 && lod->GetUnknown0x08Test8()) {
 | |
| 								numLODs++;
 | |
| 							}
 | |
| 						}
 | |
| 
 | |
| 						lodList->PushBack(lod);
 | |
| 					}
 | |
| 				}
 | |
| 			}
 | |
| 			else {
 | |
| 				for (i = 0; i < numLODs; i++) {
 | |
| 					lod = new LegoLOD(p_renderer);
 | |
| 					if (lod->Read(p_renderer, p_textureContainer, p_storage) != SUCCESS) {
 | |
| 						goto done;
 | |
| 					}
 | |
| 
 | |
| 					if (i == 0) {
 | |
| 						if (surplusLODs != 0 && lod->GetUnknown0x08Test8()) {
 | |
| 							numLODs++;
 | |
| 						}
 | |
| 					}
 | |
| 
 | |
| 					if (i == 0 && (lodList = p_viewLODListManager->Create(roiName, numLODs)) == NULL) {
 | |
| 						goto done;
 | |
| 					}
 | |
| 
 | |
| 					lodList->PushBack(lod);
 | |
| 				}
 | |
| 			}
 | |
| 
 | |
| 			p_storage->SetPosition(offset);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	SetLODList(lodList);
 | |
| 
 | |
| 	if (lodList != NULL) {
 | |
| 		lodList->Release();
 | |
| 	}
 | |
| 
 | |
| 	if (textureName != NULL) {
 | |
| 		if (!strnicmp(textureName, "t_", 2)) {
 | |
| 			textureInfo = p_textureContainer->Get(textureName + 2);
 | |
| 
 | |
| 			if (textureInfo == NULL) {
 | |
| 				goto done;
 | |
| 			}
 | |
| 
 | |
| 			SetTextureInfo(textureInfo);
 | |
| 			SetLodColor(1.0F, 1.0F, 1.0F, 0.0F);
 | |
| 		}
 | |
| 		else {
 | |
| 			LegoFloat red = 1.0F;
 | |
| 			LegoFloat green = 0.0F;
 | |
| 			LegoFloat blue = 1.0F;
 | |
| 			LegoFloat alpha = 0.0F;
 | |
| 			GetRGBAColor(textureName, red, green, blue, alpha);
 | |
| 			SetLodColor(red, green, blue, alpha);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if (p_storage->Read(&numROIs, sizeof(LegoU32)) != SUCCESS) {
 | |
| 		goto done;
 | |
| 	}
 | |
| 
 | |
| 	if (numROIs > 0) {
 | |
| 		comp = new CompoundObject;
 | |
| 	}
 | |
| 
 | |
| 	for (i = 0; i < numROIs; i++) {
 | |
| 		// Create and initialize a sub-component
 | |
| 		roi = new LegoROI(p_renderer);
 | |
| 		if (roi->Read(this, p_renderer, p_viewLODListManager, p_textureContainer, p_storage) != SUCCESS) {
 | |
| 			goto done;
 | |
| 		}
 | |
| 		// Add the new sub-component to this ROI's protected list
 | |
| 		comp->push_back(roi);
 | |
| 	}
 | |
| 
 | |
| 	result = SUCCESS;
 | |
| 
 | |
| done:
 | |
| 	return result;
 | |
| }
 | |
| 
 | |
| // FUNCTION: LEGO1 0x100a8cb0
 | |
| // FUNCTION: BETA10 0x1018a7e8
 | |
| LegoResult LegoROI::CreateLocalTransform(LegoAnimNodeData* p_data, LegoTime p_time, Matrix4& p_matrix)
 | |
| {
 | |
| 	p_matrix.SetIdentity();
 | |
| 	p_data->CreateLocalTransform(p_time, p_matrix);
 | |
| 	return SUCCESS;
 | |
| }
 | |
| 
 | |
| // FUNCTION: LEGO1 0x100a8ce0
 | |
| // FUNCTION: BETA10 0x1018a815
 | |
| LegoROI* LegoROI::FindChildROI(const LegoChar* p_name, LegoROI* p_roi)
 | |
| {
 | |
| 	CompoundObject::iterator it;
 | |
| 	const LegoChar* name = p_roi->GetName();
 | |
| 
 | |
| 	if (name != NULL && *name != '\0' && !strcmpi(name, p_name)) {
 | |
| 		return p_roi;
 | |
| 	}
 | |
| 
 | |
| 	CompoundObject* comp = p_roi->comp;
 | |
| 	if (comp != NULL) {
 | |
| 		for (it = comp->begin(); it != comp->end(); it++) {
 | |
| 			LegoROI* roi = (LegoROI*) *it;
 | |
| 			name = roi->GetName();
 | |
| 
 | |
| 			if (name != NULL && *name != '\0' && !strcmpi(name, p_name)) {
 | |
| 				return roi;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		for (it = comp->begin(); it != comp->end(); it++) {
 | |
| 			LegoROI* roi = FindChildROI(p_name, (LegoROI*) *it);
 | |
| 
 | |
| 			if (roi != NULL) {
 | |
| 				return roi;
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return NULL;
 | |
| }
 | |
| 
 | |
| // FUNCTION: LEGO1 0x100a8da0
 | |
| // FUNCTION: BETA10 0x1018a9fb
 | |
| LegoResult LegoROI::ApplyAnimationTransformation(
 | |
| 	LegoTreeNode* p_node,
 | |
| 	const Matrix4& p_matrix,
 | |
| 	LegoTime p_time,
 | |
| 	LegoROI* p_parentROI
 | |
| )
 | |
| {
 | |
| 	MxMatrix mat;
 | |
| 	LegoAnimNodeData* data = (LegoAnimNodeData*) p_node->GetData();
 | |
| 	const LegoChar* name = data->GetName();
 | |
| 	LegoROI* roi = FindChildROI(name, p_parentROI);
 | |
| 
 | |
| 	if (roi == NULL) {
 | |
| 		roi = FindChildROI(name, this);
 | |
| 	}
 | |
| 
 | |
| 	if (roi != NULL) {
 | |
| 		CreateLocalTransform(data, p_time, mat);
 | |
| 		roi->m_local2world.Product(mat, p_matrix);
 | |
| 		roi->UpdateWorldData();
 | |
| 
 | |
| 		LegoBool und = data->GetVisibility(p_time);
 | |
| 		roi->SetVisibility(und);
 | |
| 
 | |
| 		for (LegoU32 i = 0; i < p_node->GetNumChildren(); i++) {
 | |
| 			ApplyAnimationTransformation(p_node->GetChild(i), roi->m_local2world, p_time, roi);
 | |
| 		}
 | |
| 	}
 | |
| 	else {
 | |
| 		FUN_100a81b0("%s ROI Not found\n", name);
 | |
| 	}
 | |
| 
 | |
| 	return SUCCESS;
 | |
| }
 | |
| 
 | |
| // FUNCTION: LEGO1 0x100a8e80
 | |
| // FUNCTION: BETA10 0x1018ab3a
 | |
| void LegoROI::FUN_100a8e80(LegoTreeNode* p_node, Matrix4& p_matrix, LegoTime p_time, LegoROI** p_roiMap)
 | |
| {
 | |
| 	MxMatrix mat;
 | |
| 
 | |
| 	LegoAnimNodeData* data = (LegoAnimNodeData*) p_node->GetData();
 | |
| 	CreateLocalTransform(data, p_time, mat);
 | |
| 
 | |
| 	LegoROI* roi = p_roiMap[data->GetROIIndex()];
 | |
| 	if (roi != NULL) {
 | |
| 		roi->m_local2world.Product(mat, p_matrix);
 | |
| 		roi->UpdateWorldData();
 | |
| 
 | |
| 		LegoBool und = data->GetVisibility(p_time);
 | |
| 		roi->SetVisibility(und);
 | |
| 
 | |
| 		for (LegoU32 i = 0; i < p_node->GetNumChildren(); i++) {
 | |
| 			FUN_100a8e80(p_node->GetChild(i), roi->m_local2world, p_time, p_roiMap);
 | |
| 		}
 | |
| 	}
 | |
| 	else {
 | |
| 		MxMatrix local2world;
 | |
| 		local2world.Product(mat, p_matrix);
 | |
| 
 | |
| 		for (LegoU32 i = 0; i < p_node->GetNumChildren(); i++) {
 | |
| 			FUN_100a8e80(p_node->GetChild(i), local2world, p_time, p_roiMap);
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // FUNCTION: LEGO1 0x100a8fd0
 | |
| // FUNCTION: BETA10 0x1018ac81
 | |
| void LegoROI::FUN_100a8fd0(LegoTreeNode* p_node, Matrix4& p_matrix, LegoTime p_time, LegoROI** p_roiMap)
 | |
| {
 | |
| 	MxMatrix mat;
 | |
| 
 | |
| 	LegoAnimNodeData* data = (LegoAnimNodeData*) p_node->GetData();
 | |
| 	CreateLocalTransform(data, p_time, mat);
 | |
| 
 | |
| 	LegoROI* roi = p_roiMap[data->GetROIIndex()];
 | |
| 	if (roi != NULL) {
 | |
| 		roi->m_local2world.Product(mat, p_matrix);
 | |
| 
 | |
| 		for (LegoU32 i = 0; i < p_node->GetNumChildren(); i++) {
 | |
| 			FUN_100a8fd0(p_node->GetChild(i), roi->m_local2world, p_time, p_roiMap);
 | |
| 		}
 | |
| 	}
 | |
| 	else {
 | |
| 		MxMatrix local2world;
 | |
| 		local2world.Product(mat, p_matrix);
 | |
| 
 | |
| 		for (LegoU32 i = 0; i < p_node->GetNumChildren(); i++) {
 | |
| 			FUN_100a8fd0(p_node->GetChild(i), local2world, p_time, p_roiMap);
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // FUNCTION: LEGO1 0x100a90f0
 | |
| // FUNCTION: BETA10 0x1018ada8
 | |
| LegoResult LegoROI::SetFrame(LegoAnim* p_anim, LegoTime p_time)
 | |
| {
 | |
| 	LegoTreeNode* root = p_anim->GetRoot();
 | |
| 	MxMatrix mat;
 | |
| 
 | |
| 	mat = m_local2world;
 | |
| 	mat.SetIdentity(); // this clears the matrix, assignment above is redundant
 | |
| 
 | |
| 	return ApplyAnimationTransformation(root, mat, p_time, this);
 | |
| }
 | |
| 
 | |
| // FUNCTION: LEGO1 0x100a9170
 | |
| // FUNCTION: BETA10 0x1018ae09
 | |
| LegoResult LegoROI::SetLodColor(LegoFloat p_red, LegoFloat p_green, LegoFloat p_blue, LegoFloat p_alpha)
 | |
| {
 | |
| 	LegoResult result = SUCCESS;
 | |
| 	CompoundObject::iterator it;
 | |
| 
 | |
| 	int lodCount = GetLODCount();
 | |
| 	for (LegoU32 i = 0; i < lodCount; i++) {
 | |
| 		LegoLOD* lod = (LegoLOD*) GetLOD(i);
 | |
| 
 | |
| 		if (lod->SetColor(p_red, p_green, p_blue, p_alpha) != SUCCESS) {
 | |
| 			result = FAILURE;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if (comp != NULL) {
 | |
| 		for (it = comp->begin(); it != comp->end(); it++) {
 | |
| 			if (((LegoROI*) *it)->SetLodColor(p_red, p_green, p_blue, p_alpha) != SUCCESS) {
 | |
| 				result = FAILURE;
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return result;
 | |
| }
 | |
| 
 | |
| // FUNCTION: LEGO1 0x100a9210
 | |
| // FUNCTION: BETA10 0x1018af25
 | |
| LegoResult LegoROI::SetTextureInfo(LegoTextureInfo* p_textureInfo)
 | |
| {
 | |
| 	LegoResult result = SUCCESS;
 | |
| 	CompoundObject::iterator it;
 | |
| 
 | |
| 	int lodCount = GetLODCount();
 | |
| 	for (LegoU32 i = 0; i < lodCount; i++) {
 | |
| 		LegoLOD* lod = (LegoLOD*) GetLOD(i);
 | |
| 
 | |
| 		if (lod->SetTextureInfo(p_textureInfo) != SUCCESS) {
 | |
| 			result = FAILURE;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if (comp != NULL) {
 | |
| 		for (it = comp->begin(); it != comp->end(); it++) {
 | |
| 			if (((LegoROI*) *it)->SetTextureInfo(p_textureInfo) != SUCCESS) {
 | |
| 				result = FAILURE;
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return result;
 | |
| }
 | |
| 
 | |
| // FUNCTION: LEGO1 0x100a92a0
 | |
| // FUNCTION: BETA10 0x1018b12d
 | |
| LegoResult LegoROI::GetTextureInfo(LegoTextureInfo*& p_textureInfo)
 | |
| {
 | |
| 	CompoundObject::iterator it;
 | |
| 
 | |
| 	int lodCount = GetLODCount();
 | |
| 	for (LegoU32 i = 0; i < lodCount; i++) {
 | |
| 		LegoLOD* lod = (LegoLOD*) GetLOD(i);
 | |
| 
 | |
| 		if (lod->GetTextureInfo(p_textureInfo) == SUCCESS) {
 | |
| 			return SUCCESS;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if (comp != NULL) {
 | |
| 		for (it = comp->begin(); it != comp->end(); it++) {
 | |
| 			if (((LegoROI*) *it)->GetTextureInfo(p_textureInfo) == SUCCESS) {
 | |
| 				return SUCCESS;
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return FAILURE;
 | |
| }
 | |
| 
 | |
| // FUNCTION: LEGO1 0x100a9330
 | |
| // FUNCTION: BETA10 0x1018b22c
 | |
| LegoResult LegoROI::FUN_100a9330(LegoFloat p_red, LegoFloat p_green, LegoFloat p_blue, LegoFloat p_alpha)
 | |
| {
 | |
| 	return SetLodColor(p_red, p_green, p_blue, p_alpha);
 | |
| }
 | |
| 
 | |
| // FUNCTION: LEGO1 0x100a9350
 | |
| // FUNCTION: BETA10 0x1018b25c
 | |
| LegoResult LegoROI::SetLodColor(const LegoChar* p_name)
 | |
| {
 | |
| 	MxFloat red, green, blue, alpha;
 | |
| 	if (ColorAliasLookup(p_name, red, green, blue, alpha)) {
 | |
| 		return SetLodColor(red, green, blue, alpha);
 | |
| 	}
 | |
| 
 | |
| 	return SUCCESS;
 | |
| }
 | |
| 
 | |
| // FUNCTION: LEGO1 0x100a93b0
 | |
| // FUNCTION: BETA10 0x1018b2c0
 | |
| LegoResult LegoROI::FUN_100a93b0(const LegoChar* p_name)
 | |
| {
 | |
| 	MxFloat red, green, blue, alpha;
 | |
| 	if (ColorAliasLookup(p_name, red, green, blue, alpha)) {
 | |
| 		return FUN_100a9330(red, green, blue, alpha);
 | |
| 	}
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| // FUNCTION: LEGO1 0x100a9410
 | |
| // FUNCTION: BETA10 0x1018b324
 | |
| LegoU32 LegoROI::FUN_100a9410(
 | |
| 	Vector3& p_v1,
 | |
| 	Vector3& p_v2,
 | |
| 	float p_f1,
 | |
| 	float p_f2,
 | |
| 	Vector3& p_v3,
 | |
| 	LegoBool p_collideBox
 | |
| )
 | |
| {
 | |
| 	if (p_collideBox) {
 | |
| 		Mx3DPointFloat v2(p_v2);
 | |
| 		v2 *= p_f1;
 | |
| 		v2 += p_v1;
 | |
| 
 | |
| 		Mx4DPointFloat localc0;
 | |
| 		Mx4DPointFloat local9c;
 | |
| 		Mx4DPointFloat local168;
 | |
| 		Mx4DPointFloat local70;
 | |
| 		Mx4DPointFloat local150[6];
 | |
| 
 | |
| 		Vector3 local58(&localc0[0]);
 | |
| 		Vector3 locala8(&local9c[0]);
 | |
| 		Vector3 local38(&local168[0]);
 | |
| 
 | |
| 		Mx3DPointFloat local4c(p_v1);
 | |
| 
 | |
| 		local58 = m_bounding_box.Min();
 | |
| 		locala8 = m_bounding_box.Max();
 | |
| 
 | |
| 		localc0[3] = local9c[3] = local168[3] = 1.0f;
 | |
| 
 | |
| 		local38 = local58;
 | |
| 		local38 += locala8;
 | |
| 		local38 *= 0.5f;
 | |
| 
 | |
| 		local70 = localc0;
 | |
| 		localc0.SetMatrixProduct(local70, (float*) m_local2world.GetData());
 | |
| 
 | |
| 		local70 = local9c;
 | |
| 		local9c.SetMatrixProduct(local70, (float*) m_local2world.GetData());
 | |
| 
 | |
| 		local70 = local168;
 | |
| 		local168.SetMatrixProduct(local70, (float*) m_local2world.GetData());
 | |
| 
 | |
| 		p_v3 = m_local2world[3];
 | |
| 
 | |
| 		LegoS32 i;
 | |
| 		for (i = 0; i < 6; i++) {
 | |
| 			local150[i] = m_local2world[i % 3];
 | |
| 
 | |
| 			if (i > 2) {
 | |
| 				local150[i][3] = -local58.Dot(local58, local150[i]);
 | |
| 			}
 | |
| 			else {
 | |
| 				local150[i][3] = -locala8.Dot(locala8, local150[i]);
 | |
| 			}
 | |
| 
 | |
| 			if (local150[i][3] + local38.Dot(local38, local150[i]) < 0.0f) {
 | |
| 				local150[i] *= -1.0f;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		for (i = 0; i < 6; i++) {
 | |
| 			float local50 = p_v2.Dot(p_v2, local150[i]);
 | |
| 
 | |
| 			if (local50 >= 0.01 || local50 < -0.01) {
 | |
| 				local50 = -((local150[i][3] + local4c.Dot(local4c, local150[i])) / local50);
 | |
| 
 | |
| 				if (local50 >= 0.0f && local50 <= p_f1) {
 | |
| 					Mx3DPointFloat local17c(p_v2);
 | |
| 					local17c *= local50;
 | |
| 					local17c += local4c;
 | |
| 
 | |
| 					LegoS32 j;
 | |
| 					for (j = 0; j < 6; j++) {
 | |
| 						if (i != j && i - j != 3 && j - i != 3) {
 | |
| 							if (local150[j][3] + local17c.Dot(local17c, local150[j]) < 0.0f) {
 | |
| 								break;
 | |
| 							}
 | |
| 						}
 | |
| 					}
 | |
| 
 | |
| 					if (j == 6) {
 | |
| 						return 1;
 | |
| 					}
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	else {
 | |
| 		Mx3DPointFloat v1(p_v1);
 | |
| 		v1 -= GetWorldBoundingSphere().Center();
 | |
| 
 | |
| 		float local10 = GetWorldBoundingSphere().Radius();
 | |
| 		float local8 = p_v2.Dot(p_v2, p_v2);
 | |
| 		float localc = p_v2.Dot(p_v2, v1) * 2.0f;
 | |
| 		float local14 = v1.Dot(v1, v1) - (local10 * local10);
 | |
| 
 | |
| 		if (local8 >= 0.001 || local8 <= -0.001) {
 | |
| 			float local1c = -1.0f;
 | |
| 			float local18 = (localc * localc) - (local14 * local8 * 4.0f);
 | |
| 
 | |
| 			if (local18 >= -0.001) {
 | |
| 				local8 *= 2.0f;
 | |
| 				localc = -localc;
 | |
| 
 | |
| 				if (local18 > 0.0f) {
 | |
| 					local18 = sqrt(local18);
 | |
| 					float local184 = (localc + local18) / local8;
 | |
| 					float local188 = (localc - local18) / local8;
 | |
| 
 | |
| 					if (local184 > 0.0f && local188 > local184) {
 | |
| 						local1c = local184;
 | |
| 					}
 | |
| 					else if (local188 > 0.0f) {
 | |
| 						local1c = local188;
 | |
| 					}
 | |
| 					else {
 | |
| 						return 0;
 | |
| 					}
 | |
| 				}
 | |
| 				else {
 | |
| 					local1c = localc / local8;
 | |
| 				}
 | |
| 
 | |
| 				if (local1c >= 0.0f && p_f1 >= local1c) {
 | |
| 					p_v3 = p_v2;
 | |
| 					p_v3 *= local1c;
 | |
| 					p_v3 += p_v1;
 | |
| 					return 1;
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| // FUNCTION: LEGO1 0x100a9a50
 | |
| // FUNCTION: BETA10 0x1018bb6b
 | |
| TimeROI::TimeROI(Tgl::Renderer* p_renderer, ViewLODList* p_lodList, LegoTime p_time) : LegoROI(p_renderer, p_lodList)
 | |
| {
 | |
| 	m_time = p_time;
 | |
| }
 | |
| 
 | |
| // FUNCTION: LEGO1 0x100a9b40
 | |
| // FUNCTION: BETA10 0x1018bbf0
 | |
| void TimeROI::FUN_100a9b40(Matrix4& p_matrix, LegoTime p_time)
 | |
| {
 | |
| 	LegoTime time = p_time - m_time;
 | |
| 
 | |
| 	if (time) {
 | |
| 		m_time = p_time;
 | |
| 
 | |
| 		Mx3DPointFloat targetPosition(p_matrix[3]);
 | |
| 		Vector3 vec(m_local2world[3]);
 | |
| 
 | |
| 		targetPosition -= vec;
 | |
| 		targetPosition /= time / 1000.0;
 | |
| 
 | |
| 		SetWorldVelocity(targetPosition);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // FUNCTION: LEGO1 0x100a9bf0
 | |
| // FUNCTION: BETA10 0x1018bc93
 | |
| LegoBool LegoROI::GetRGBAColor(const LegoChar* p_name, float& p_red, float& p_green, float& p_blue, float& p_alpha)
 | |
| {
 | |
| 	if (p_name == NULL) {
 | |
| 		return FALSE;
 | |
| 	}
 | |
| 
 | |
| 	char p_updatedName[32];
 | |
| 	if (g_colorOverride) {
 | |
| 		if (g_colorOverride(p_name, p_updatedName, sizeof(p_updatedName))) {
 | |
| 			p_name = p_updatedName;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return ColorAliasLookup(p_name, p_red, p_green, p_blue, p_alpha);
 | |
| }
 | |
| 
 | |
| // FUNCTION: LEGO1 0x100a9c50
 | |
| // FUNCTION: BETA10 0x1018bdd9
 | |
| LegoBool LegoROI::ColorAliasLookup(const LegoChar* p_param, float& p_red, float& p_green, float& p_blue, float& p_alpha)
 | |
| {
 | |
| 	for (LegoU32 i = 0; i < sizeOfArray(g_roiColorAliases); i++) {
 | |
| 		if (strcmpi(g_roiColorAliases[i].m_name, p_param) == 0) {
 | |
| 			p_red = g_roiColorAliases[i].m_red / 255.0;
 | |
| 			p_green = g_roiColorAliases[i].m_green / 255.0;
 | |
| 			p_blue = g_roiColorAliases[i].m_blue / 255.0;
 | |
| 			p_alpha = g_roiColorAliases[i].m_alpha / 255.0;
 | |
| 			return TRUE;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return FALSE;
 | |
| }
 | |
| 
 | |
| // FUNCTION: LEGO1 0x100a9cf0
 | |
| LegoBool LegoROI::GetPaletteEntries(const LegoChar* p_name, unsigned char* paletteEntries, LegoU32 p_numEntries)
 | |
| {
 | |
| 	if (p_name == NULL) {
 | |
| 		return FALSE;
 | |
| 	}
 | |
| 
 | |
| 	// Note: g_textureHandler is never set in the code base
 | |
| 	if (g_textureHandler != NULL) {
 | |
| 		return g_textureHandler(p_name, paletteEntries, p_numEntries);
 | |
| 	}
 | |
| 
 | |
| 	paletteEntries[0] = '\0';
 | |
| 	return FALSE;
 | |
| }
 | |
| 
 | |
| // FUNCTION: LEGO1 0x100a9d30
 | |
| void LegoROI::SetColorOverride(ColorOverride p_colorOverride)
 | |
| {
 | |
| 	g_colorOverride = p_colorOverride;
 | |
| }
 | |
| 
 | |
| // FUNCTION: LEGO1 0x100a9d40
 | |
| void LegoROI::SetName(const LegoChar* p_name)
 | |
| {
 | |
| 	if (m_name != NULL) {
 | |
| 		delete[] m_name;
 | |
| 	}
 | |
| 
 | |
| 	if (p_name != NULL) {
 | |
| 		m_name = new LegoChar[strlen(p_name) + 1];
 | |
| 		strcpy(m_name, p_name);
 | |
| 		strlwr(m_name);
 | |
| 	}
 | |
| 	else {
 | |
| 		m_name = NULL;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // FUNCTION: LEGO1 0x100a9dd0
 | |
| // FUNCTION: BETA10 0x1018bfdb
 | |
| void LegoROI::ClearMeshOffset()
 | |
| {
 | |
| 	int lodCount = GetLODCount();
 | |
| 	for (LegoS32 i = 0; i < lodCount; i++) {
 | |
| 		LegoLOD* lod = (LegoLOD*) GetLOD(i);
 | |
| 		lod->ClearMeshOffset();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // FUNCTION: LEGO1 0x100a9e10
 | |
| void LegoROI::SetDisplayBB(int p_displayBB)
 | |
| {
 | |
| 	// Intentionally empty function
 | |
| }
 | |
| 
 | |
| // FUNCTION: LEGO1 0x100aa340
 | |
| // FUNCTION: BETA10 0x1018cca0
 | |
| float LegoROI::IntrinsicImportance() const
 | |
| {
 | |
| 	return .5;
 | |
| }
 | |
| 
 | |
| // FUNCTION: LEGO1 0x100aa350
 | |
| // FUNCTION: BETA10 0x1018ccc0
 | |
| void LegoROI::UpdateWorldBoundingVolumes()
 | |
| {
 | |
| 	CalcWorldBoundingVolumes(m_sphere, m_local2world, m_world_bounding_box, m_world_bounding_sphere);
 | |
| }
 | 
