mirror of
				https://github.com/isledecomp/isle.git
				synced 2025-10-25 09:24:17 +00:00 
			
		
		
		
	 48febaf5c5
			
		
	
	48febaf5c5
	
	
	
		
			
			* Match Mx3DPointFloat overloads on BETA10 --------- Co-authored-by: jonschz <jonschz@users.noreply.github.com>
		
			
				
	
	
		
			573 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			573 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| #include "viewmanager.h"
 | |
| 
 | |
| #include "mxdirectx/mxstopwatch.h"
 | |
| #include "tgl/d3drm/impl.h"
 | |
| #include "viewlod.h"
 | |
| 
 | |
| #include <vec.h>
 | |
| 
 | |
| DECOMP_SIZE_ASSERT(ViewManager, 0x1bc)
 | |
| 
 | |
| // GLOBAL: LEGO1 0x100dbc78
 | |
| int g_boundingBoxCornerMap[8][3] =
 | |
| 	{{0, 0, 0}, {0, 0, 1}, {0, 1, 0}, {1, 0, 0}, {0, 1, 1}, {1, 0, 1}, {1, 1, 0}, {1, 1, 1}};
 | |
| 
 | |
| // GLOBAL: LEGO1 0x100dbcd8
 | |
| int g_planePointIndexMap[18] = {0, 1, 5, 6, 2, 3, 3, 0, 4, 1, 2, 6, 0, 3, 2, 4, 5, 6};
 | |
| 
 | |
| // GLOBAL: LEGO1 0x10101050
 | |
| float g_LODScaleFactor = 4.0F;
 | |
| 
 | |
| // GLOBAL: LEGO1 0x10101054
 | |
| float g_minLODThreshold = 0.00097656297;
 | |
| 
 | |
| // GLOBAL: LEGO1 0x10101058
 | |
| int g_maxLODLevels = 6;
 | |
| 
 | |
| // GLOBAL: LEGO1 0x1010105c
 | |
| float g_viewDistance = 0.000125F;
 | |
| 
 | |
| // GLOBAL: LEGO1 0x10101060
 | |
| float g_elapsedSeconds = 0;
 | |
| 
 | |
| inline void SetAppData(ViewROI* p_roi, LPD3DRM_APPDATA 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);
 | |
| 	prev_render_time = 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(transformed_points, 0, sizeof(transformed_points));
 | |
| 	seconds_allowed = 1.0;
 | |
| }
 | |
| 
 | |
| // FUNCTION: LEGO1 0x100a60c0
 | |
| ViewManager::~ViewManager()
 | |
| {
 | |
| 	SetPOVSource(NULL);
 | |
| }
 | |
| 
 | |
| // FUNCTION: LEGO1 0x100a6150
 | |
| // FUNCTION: BETA10 0x10172164
 | |
| unsigned int ViewManager::IsBoundingBoxInFrustum(const BoundingBox& p_bounding_box)
 | |
| {
 | |
| 	const Vector3* box[] = {&p_bounding_box.Min(), &p_bounding_box.Max()};
 | |
| 
 | |
| 	float box_corners[8][3];
 | |
| 	int i, j, k;
 | |
| 
 | |
| 	for (i = 0; i < 8; i++) {
 | |
| 		for (j = 0; j < 3; j++) {
 | |
| 			box_corners[i][j] = box[g_boundingBoxCornerMap[i][j]]->operator[](j);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	for (i = 0; i < 6; i++) {
 | |
| 		for (k = 0; k < 8; k++) {
 | |
| 			if (frustum_planes[i][0] * box_corners[k][0] + frustum_planes[i][2] * box_corners[k][2] +
 | |
| 					frustum_planes[i][1] * box_corners[k][1] + frustum_planes[i][3] >=
 | |
| 				0.0f) {
 | |
| 				break;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		if (k == 8) {
 | |
| 			return FALSE;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return TRUE;
 | |
| }
 | |
| 
 | |
| // 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->GetLodLevel() >= 0) {
 | |
| 				RemoveROIDetailFromScene(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)->GetLodLevel() >= 0) {
 | |
| 						RemoveROIDetailFromScene((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->GetLodLevel() >= 0) {
 | |
| 			RemoveROIDetailFromScene(p_roi);
 | |
| 		}
 | |
| 
 | |
| 		p_roi->SetLodLevel(ViewROI::c_lodLevelUnset);
 | |
| 		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::UpdateROIDetailBasedOnLOD(ViewROI* p_roi, int p_lodLevel)
 | |
| {
 | |
| 	if (p_roi->GetLODCount() <= p_lodLevel) {
 | |
| 		p_lodLevel = p_roi->GetLODCount() - 1;
 | |
| 	}
 | |
| 
 | |
| 	int lodLevel = p_roi->GetLodLevel();
 | |
| 
 | |
| 	if (lodLevel == p_lodLevel) {
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	Tgl::Group* group = p_roi->GetGeometry();
 | |
| 	Tgl::MeshBuilder* meshBuilder;
 | |
| 	ViewLOD* lod;
 | |
| 
 | |
| 	if (lodLevel < 0) {
 | |
| 		lod = (ViewLOD*) p_roi->GetLOD(p_lodLevel);
 | |
| 
 | |
| 		if (lod->GetFlags() & ViewLOD::c_hasMesh) {
 | |
| 			scene->Add((Tgl::MeshBuilder*) group);
 | |
| 			SetAppData(p_roi, reinterpret_cast<LPD3DRM_APPDATA>(p_roi));
 | |
| 		}
 | |
| 	}
 | |
| 	else {
 | |
| 		lod = (ViewLOD*) p_roi->GetLOD(lodLevel);
 | |
| 
 | |
| 		if (lod != NULL) {
 | |
| 			meshBuilder = lod->GetMeshBuilder();
 | |
| 
 | |
| 			if (meshBuilder != NULL) {
 | |
| 				group->Remove(meshBuilder);
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		lod = (ViewLOD*) p_roi->GetLOD(p_lodLevel);
 | |
| 	}
 | |
| 
 | |
| 	if (lod->GetFlags() & ViewLOD::c_hasMesh) {
 | |
| 		meshBuilder = lod->GetMeshBuilder();
 | |
| 
 | |
| 		if (meshBuilder != NULL) {
 | |
| 			group->Add(meshBuilder);
 | |
| 			SetAppData(p_roi, reinterpret_cast<LPD3DRM_APPDATA>(p_roi));
 | |
| 			p_roi->SetLodLevel(p_lodLevel);
 | |
| 			return;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	p_roi->SetLodLevel(ViewROI::c_lodLevelUnset);
 | |
| }
 | |
| 
 | |
| // FUNCTION: LEGO1 0x100a66a0
 | |
| void ViewManager::RemoveROIDetailFromScene(ViewROI* p_roi)
 | |
| {
 | |
| 	const ViewLOD* lod = (const ViewLOD*) p_roi->GetLOD(p_roi->GetLodLevel());
 | |
| 
 | |
| 	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->SetLodLevel(ViewROI::c_lodLevelUnset);
 | |
| }
 | |
| 
 | |
| // FUNCTION: LEGO1 0x100a66f0
 | |
| // FUNCTION: BETA10 0x1017297f
 | |
| inline void ViewManager::ManageVisibilityAndDetailRecursively(ViewROI* p_from, int p_lodLevel)
 | |
| {
 | |
| 	assert(p_from);
 | |
| 
 | |
| 	if (!p_from->GetVisibility() && p_lodLevel != ViewROI::c_lodLevelInvisible) {
 | |
| 		ManageVisibilityAndDetailRecursively(p_from, ViewROI::c_lodLevelInvisible);
 | |
| 	}
 | |
| 	else {
 | |
| 		const CompoundObject* comp = p_from->GetComp();
 | |
| 
 | |
| 		if (p_lodLevel == ViewROI::c_lodLevelUnset) {
 | |
| 			if (p_from->GetWorldBoundingSphere().Radius() > 0.001F) {
 | |
| 				float projectedSize = ProjectedSize(p_from->GetWorldBoundingSphere());
 | |
| 
 | |
| 				if (projectedSize < seconds_allowed * g_viewDistance) {
 | |
| 					if (p_from->GetLodLevel() != ViewROI::c_lodLevelInvisible) {
 | |
| 						ManageVisibilityAndDetailRecursively(p_from, ViewROI::c_lodLevelInvisible);
 | |
| 					}
 | |
| 
 | |
| 					return;
 | |
| 				}
 | |
| 				else {
 | |
| 					p_lodLevel =
 | |
| 						CalculateLODLevel(projectedSize, RealtimeView::GetUserMaxLodPower() * seconds_allowed, p_from);
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		if (p_lodLevel == ViewROI::c_lodLevelInvisible) {
 | |
| 			if (p_from->GetLodLevel() >= 0) {
 | |
| 				RemoveROIDetailFromScene(p_from);
 | |
| 				p_from->SetLodLevel(ViewROI::c_lodLevelInvisible);
 | |
| 			}
 | |
| 
 | |
| 			if (comp != NULL) {
 | |
| 				for (CompoundObject::const_iterator it = comp->begin(); it != comp->end(); it++) {
 | |
| 					ManageVisibilityAndDetailRecursively((ViewROI*) *it, p_lodLevel);
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 		else if (comp == NULL) {
 | |
| 			if (p_from->GetLODs() != NULL && p_from->GetLODCount() > 0) {
 | |
| 				UpdateROIDetailBasedOnLOD(p_from, p_lodLevel);
 | |
| 			}
 | |
| 		}
 | |
| 		else {
 | |
| 			p_from->SetLodLevel(ViewROI::c_lodLevelUnset);
 | |
| 
 | |
| 			for (CompoundObject::const_iterator it = comp->begin(); it != comp->end(); it++) {
 | |
| 				// LINE: BETA10 0x10172bbd
 | |
| 				ManageVisibilityAndDetailRecursively((ViewROI*) *it, p_lodLevel);
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // FUNCTION: LEGO1 0x100a6930
 | |
| void ViewManager::Update(float p_previousRenderTime, float)
 | |
| {
 | |
| 	MxStopWatch stopWatch;
 | |
| 	stopWatch.Start();
 | |
| 
 | |
| 	prev_render_time = p_previousRenderTime;
 | |
| 	flags |= c_bit1;
 | |
| 
 | |
| 	if (flags & c_bit3) {
 | |
| 		CalculateFrustumTransformations();
 | |
| 	}
 | |
| 	else if (flags & c_bit2) {
 | |
| 		UpdateViewTransformations();
 | |
| 	}
 | |
| 
 | |
| 	for (CompoundObject::iterator it = rois.begin(); it != rois.end(); it++) {
 | |
| 		ManageVisibilityAndDetailRecursively((ViewROI*) *it, ViewROI::c_lodLevelUnset);
 | |
| 	}
 | |
| 
 | |
| 	stopWatch.Stop();
 | |
| 	g_elapsedSeconds = stopWatch.ElapsedSeconds();
 | |
| }
 | |
| 
 | |
| inline int ViewManager::CalculateFrustumTransformations()
 | |
| {
 | |
| 	flags &= ~c_bit3;
 | |
| 
 | |
| 	if (height == 0.0F || front == 0.0F) {
 | |
| 		return -1;
 | |
| 	}
 | |
| 	else {
 | |
| 		float fVar7 = tan(view_angle / 2.0F);
 | |
| 		view_area_at_one = fVar7 * fVar7 * 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* frustumVertices = (float*) this->frustum_vertices;
 | |
| 
 | |
| 		// clang-format off
 | |
| 		*frustumVertices = fVar2; frustumVertices++;
 | |
| 		*frustumVertices = fVar1; frustumVertices++;
 | |
| 		*frustumVertices = uVar6; frustumVertices++;
 | |
| 		*frustumVertices = fVar2; frustumVertices++;
 | |
| 		*frustumVertices = -fVar1; frustumVertices++;
 | |
| 		*frustumVertices = uVar6; frustumVertices++;
 | |
| 		*frustumVertices = -fVar2; frustumVertices++;
 | |
| 		*frustumVertices = -fVar1; frustumVertices++;
 | |
| 		*frustumVertices = uVar6; frustumVertices++;
 | |
| 		*frustumVertices = -fVar2; frustumVertices++;
 | |
| 		*frustumVertices = fVar1; frustumVertices++;
 | |
| 		*frustumVertices = uVar6; frustumVertices++;
 | |
| 		*frustumVertices = fVar4; frustumVertices++;
 | |
| 		*frustumVertices = fVar5; frustumVertices++;
 | |
| 		*frustumVertices = fVar3; frustumVertices++;
 | |
| 		*frustumVertices = fVar4; frustumVertices++;
 | |
| 		*frustumVertices = -fVar5; frustumVertices++;
 | |
| 		*frustumVertices = fVar3; frustumVertices++;
 | |
| 		*frustumVertices = -fVar4; frustumVertices++;
 | |
| 		*frustumVertices = -fVar5; frustumVertices++;
 | |
| 		*frustumVertices = fVar3; frustumVertices++;
 | |
| 		*frustumVertices = -fVar4; frustumVertices++;
 | |
| 		*frustumVertices = fVar5; frustumVertices++;
 | |
| 		*frustumVertices = fVar3;
 | |
| 		// clang-format on
 | |
| 
 | |
| 		UpdateViewTransformations();
 | |
| 		return 0;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // FUNCTION: BETA10 0x10172be5
 | |
| inline int ViewManager::CalculateLODLevel(float p_maximumScale, float p_initialScale, ViewROI* from)
 | |
| {
 | |
| 	int lodLevel;
 | |
| 
 | |
| 	assert(from);
 | |
| 
 | |
| 	if (GetFirstLODIndex(from) != 0) {
 | |
| 		if (p_maximumScale < g_minLODThreshold) {
 | |
| 			return 0;
 | |
| 		}
 | |
| 		else {
 | |
| 			lodLevel = 1;
 | |
| 		}
 | |
| 	}
 | |
| 	else {
 | |
| 		lodLevel = 0;
 | |
| 	}
 | |
| 
 | |
| 	for (float i = p_initialScale; lodLevel < g_maxLODLevels; lodLevel++) {
 | |
| 		if (i >= p_maximumScale) {
 | |
| 			break;
 | |
| 		}
 | |
| 
 | |
| 		i *= g_LODScaleFactor;
 | |
| 	}
 | |
| 
 | |
| 	return lodLevel;
 | |
| }
 | |
| 
 | |
| // FUNCTION: BETA10 0x10172cb0
 | |
| inline int ViewManager::GetFirstLODIndex(ViewROI* p_roi)
 | |
| {
 | |
| 	const LODListBase* lods = p_roi->GetLODs();
 | |
| 
 | |
| 	if (lods != NULL && lods->Size() > 0) {
 | |
| 		if (((ViewLOD*) p_roi->GetLOD(0))->IsExtraLOD()) {
 | |
| 			return 1;
 | |
| 		}
 | |
| 		else {
 | |
| 			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))->IsExtraLOD()) {
 | |
| 					return 1;
 | |
| 				}
 | |
| 				else {
 | |
| 					return 0;
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| // FUNCTION: LEGO1 0x100a6b90
 | |
| void ViewManager::UpdateViewTransformations()
 | |
| {
 | |
| 	flags &= ~c_bit2;
 | |
| 
 | |
| 	int i, j, k;
 | |
| 
 | |
| 	for (i = 0; i < 8; i++) {
 | |
| 		for (j = 0; j < 3; j++) {
 | |
| 			transformed_points[i][j] = pov[3][j];
 | |
| 
 | |
| 			for (k = 0; k < 3; k++) {
 | |
| 				transformed_points[i][j] += pov[k][j] * frustum_vertices[i][k];
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	for (i = 0; i < 6; i++) {
 | |
| 		Vector3 a(transformed_points[g_planePointIndexMap[i * 3]]);
 | |
| 		Vector3 b(transformed_points[g_planePointIndexMap[i * 3 + 1]]);
 | |
| 		Vector3 c(transformed_points[g_planePointIndexMap[i * 3 + 2]]);
 | |
| 		Mx3DPointFloat x;
 | |
| 		Mx3DPointFloat y;
 | |
| 		Vector3 normal(frustum_planes[i]);
 | |
| 
 | |
| 		x = c;
 | |
| 		x -= b;
 | |
| 
 | |
| 		y = a;
 | |
| 		y -= b;
 | |
| 
 | |
| 		normal.EqualsCross(x, y);
 | |
| 		normal.Unitize();
 | |
| 
 | |
| 		frustum_planes[i][3] = -normal.Dot(normal, 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
 | |
| // FUNCTION: BETA10 0x101739b8
 | |
| 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;
 | |
| }
 | |
| 
 | |
| // FUNCTION: LEGO1 0x100a6e00
 | |
| ViewROI* ViewManager::Pick(Tgl::View* p_view, unsigned long x, unsigned long y)
 | |
| {
 | |
| 	LPDIRECT3DRMPICKEDARRAY picked = NULL;
 | |
| 	ViewROI* result = NULL;
 | |
| 	TglImpl::ViewImpl* view = (TglImpl::ViewImpl*) p_view;
 | |
| 	IDirect3DRMViewport* d3drm = view->ImplementationData();
 | |
| 
 | |
| 	if (d3drm->Pick(x, y, &picked) != D3DRM_OK) {
 | |
| 		return NULL;
 | |
| 	}
 | |
| 
 | |
| 	if (picked != NULL) {
 | |
| 		if (picked->GetSize() != 0) {
 | |
| 			LPDIRECT3DRMVISUAL visual;
 | |
| 			LPDIRECT3DRMFRAMEARRAY frameArray;
 | |
| 			D3DRMPICKDESC desc;
 | |
| 
 | |
| 			if (picked->GetPick(0, &visual, &frameArray, &desc) == D3DRM_OK) {
 | |
| 				if (frameArray != NULL) {
 | |
| 					int size = frameArray->GetSize();
 | |
| 
 | |
| 					if (size > 1) {
 | |
| 						for (int i = 1; i < size; i++) {
 | |
| 							LPDIRECT3DRMFRAME frame = NULL;
 | |
| 
 | |
| 							if (frameArray->GetElement(i, &frame) == D3DRM_OK) {
 | |
| 								result = (ViewROI*) frame->GetAppData();
 | |
| 
 | |
| 								if (result != NULL) {
 | |
| 									frame->Release();
 | |
| 									break;
 | |
| 								}
 | |
| 
 | |
| 								frame->Release();
 | |
| 							}
 | |
| 						}
 | |
| 					}
 | |
| 
 | |
| 					visual->Release();
 | |
| 					frameArray->Release();
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		picked->Release();
 | |
| 	}
 | |
| 
 | |
| 	return result;
 | |
| }
 | |
| 
 | |
| inline void SetAppData(ViewROI* p_roi, LPD3DRM_APPDATA 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;
 | |
| }
 |