soildworks模型导入MFC对话框

最近项目需要把solidworks创建的设备模型导入MFC对话框中显示,并且能鼠标控制移动、缩放。

软件平台包括 solidworks 2008 、3DS-Max 7.0 、 vs 2008 sp1;

openGL配置

提供最全的openGL库,支持鼠标滚轮,配置方法


将下载的压缩包解开,将得到5个文件(glut.dll, glut32.dll, glut.lib, glut32.lib,glut.h)
(1)把glut.h复制到x:\Program Files\Microsoft\Visual Studio 10.0\VC\include\gl文件夹中,如果没有gl这个文件夹则可以自己新建一个。(x是你安装VS的盘符号)
(2)把解压得到的glut.lib和glut32.lib放到静态函数库所在文件夹(即与include并排的lib文件夹下)。
(3)把解压得到的glut.dll和glut32.dll放到操作系统目录下面的system32文件夹内。(典型的位置为:C:\Windows\System32)
(注:如在开发应用程序时用到OpenGL辅助库函数,则还需下载相应动态链接库,包含glaux.dll, glaux.lib, glaux.h,相应步骤同上)


在Visual C++中先右击项目,选择属性,找到连接器标签,最后在输入中的附加依赖库加上opengl32.lib glut32.lib glu32.lib.

3 #include<gl/glut.h>
 4 
 5 //#include<gl/glu.h>  //glut.h自动包含了glu.h 和 gl.h
 6 
 7 //#include<gl/gl.h>


点击打开链接

1.在solildworks中创建设备模型,编辑好纹理,这里为了方便直接为模型上的颜色,没有进行贴图(关于soildworks的使用可另外参考教程)

2.保存模型文件为 .wtl格式

3.在3DS-Max 7.0中导入上面的.wtl文件,然后导出文件为.OBJ格式,同时导出素材库即.mtl格式文件。


4.导入模型到MFC对话框

创建一个MFC对话框在OnInitDialog()中设置RC;

	hrenderDC=::GetDC(this->m_hWnd);
	if(SetWindowPixelFormat(hrenderDC)==FALSE)// 设置像素格式
		return ;

	if(CreateViewGLContext(hrenderDC)==FALSE)//创建RC并选为所用
		return ;
SetWindowPixelFormat( HDC hDC )
{
PIXELFORMATDESCRIPTOR pixelDesc;

	pixelDesc.nSize = sizeof(PIXELFORMATDESCRIPTOR);
	pixelDesc.nVersion = 1;

	pixelDesc.dwFlags = PFD_DRAW_TO_WINDOW | 
		PFD_SUPPORT_OPENGL |
		PFD_DOUBLEBUFFER |
		PFD_TYPE_RGBA;

	pixelDesc.iPixelType = PFD_TYPE_RGBA;
	pixelDesc.cColorBits = 32;
	pixelDesc.cRedBits = 0;
	pixelDesc.cRedShift = 0;
	pixelDesc.cGreenBits = 0;
	pixelDesc.cGreenShift = 0;
	pixelDesc.cBlueBits = 0;
	pixelDesc.cBlueShift = 0;
	pixelDesc.cAlphaBits = 0;
	pixelDesc.cAlphaShift = 0;
	pixelDesc.cAccumBits = 0;
	pixelDesc.cAccumRedBits = 0;
	pixelDesc.cAccumGreenBits = 0;
	pixelDesc.cAccumBlueBits = 0;
	pixelDesc.cAccumAlphaBits = 0;
	pixelDesc.cDepthBits = 0;
	pixelDesc.cStencilBits = 1;
	pixelDesc.cAuxBuffers = 0;
	pixelDesc.iLayerType = PFD_MAIN_PLANE;
	pixelDesc.bReserved = 0;
	pixelDesc.dwLayerMask = 0;
	pixelDesc.dwVisibleMask = 0;
	pixelDesc.dwDamageMask = 0;

	PixelFormat = ChoosePixelFormat(hDC,&pixelDesc);
	if(PixelFormat==0) // Choose default
	{
		PixelFormat = 1;
		if(DescribePixelFormat(hDC,PixelFormat,
			sizeof(PIXELFORMATDESCRIPTOR),&pixelDesc)==0)
		{
			return FALSE;
		}
	}

	if(SetPixelFormat(hDC,PixelFormat,&pixelDesc)==FALSE)

	{ 
		return FALSE;
	}

	return TRUE;
}

	hrenderRC = wglCreateContext(hDC);

	if(hrenderRC==NULL)
		return FALSE;

	if(wglMakeCurrent(hDC,hrenderRC)==FALSE)
		return FALSE;



	
随后初始化openGL配置

	glutInitDisplayMode(GLUT_RGBA | GLUT_DOUBLE | GLUT_DEPTH | GLUT_STENCIL);

	CRect clirc;
	GetClientRect(&clirc);

	window_width = clirc.Width();
	window_height = clirc.Height();

	glPolygonMode(GL_FRONT,GL_FILL);
	glPolygonMode(GL_BACK,GL_FILL);

	glClearColor(0.0, 0.0, 0.0, 0.0);

	//设置当前操作的矩阵为“模型”视图矩阵
	glMatrixMode(GL_PROJECTION);

	//把当前矩阵设置为单位矩阵
	glLoadIdentity();

	//将当前控件设置为透视投影空间
	gluPerspective(45.0f, (float)window_width/(float)window_height, 0.1f, 100.0f);

 	glMatrixMode(GL_MODELVIEW);

	glClearColor(0, 0, 0, 1);
	glClearDepth(1.0f); 
	//把屏幕上的颜色清空
	glClear(GL_DEPTH_BUFFER_BIT | GL_COLOR_BUFFER_BIT | GL_STENCIL_BUFFER_BIT);

	//启动深度测试
	glEnable(GL_DEPTH_TEST);

	glDepthFunc(GL_LEQUAL); // The Type Of Depth Testing To Do 用于指定深度缓冲比较值。

	//
	glEnable(GL_NORMALIZE);

	//启动剔除功能
	glEnable(GL_CULL_FACE);

	glEnableClientState(GL_VERTEX_ARRAY);
	glEnableClientState(GL_TEXTURE_COORD_ARRAY);

	// Setup other misc features.

	//打开光照功能
	glEnable( GL_LIGHTING );
	glEnable( GL_NORMALIZE );

	//设置颜色填充方式:GL_SMOOTH:平滑方式   GL_FLAT:单色方式
	glShadeModel( GL_SMOOTH );


	//glViewport(0,0,clirc.Width(),clirc.Height());

	// Setup lighting model.
	GLfloat light_model_ambient[] = {1.0f, 1.0f, 1.0f, 1.0f};
	GLfloat light0_diffuse[] = {1.0f, 1.0f, 1.0f, 1.0f};
	GLfloat light0_direction[] = {0.0f, 0.0f, 10.0f, 0.0f};
	GLfloat light_specular[] = {1.0f, 1.0f, 1.0f, 1.0f};

	//	GL_LIGHT0  0号光源,以此类推


	// 光源位置由四个值表示(X,Y,Z,W)如果第四个值W为零,则表示该光源位于无限远处,
	//前三个值表示了它所在的方向。这种光源称为方向性光源,通常,太阳可以近似的被认为是方向性光源。
	//如果第四个值W不为零,则X/W, Y/W, Z/W表示了光源的位置。这种光源称为位置性光源。对于位置性光源,
	//设置其位置与设置多边形顶点的方式相似,各种矩阵变换函数例如:glTranslate*、glRotate*等在这里也同样有效。
	//方向性光源在计算时比位置性光源快了不少,因此,在视觉效果允许的情况下,应该尽可能的使用方向性光源。
	glLightfv(GL_LIGHT0, GL_POSITION, light0_direction);


	//GL_AMBIENT、GL_DIFFUSE、GL_SPECULAR属性。这三个属性表示了光源所发出的光的反射特性(以及颜色)。
	//每个属性由四个值表示,分别代表了颜色的R, G, B, A值。GL_AMBIENT表示该光源所发出的光,经过非常多次的反射后,最终遗留在整个光照环境中的强度(颜色)。
	//GL_DIFFUSE表示该光源所发出的光,照射到粗糙表面时经过漫反射,所得到的光的强度(颜色)。
	//GL_SPECULAR表示该光源所发出的光,照射到光滑表面时经过镜面反射,所得到的光的强度(颜色)。
	glLightfv(GL_LIGHT0, GL_AMBIENT, light_model_ambient);
	glLightfv(GL_LIGHT0, GL_DIFFUSE, light0_diffuse);
	glLightfv(GL_LIGHT0, GL_SPECULAR, light_specular);
	glEnable( GL_LIGHT0 );
然后读取模型文件

	// Center of the model
	float modelCenter[] = {0.0f, 0.0f, 0.0f};

	if ( pModel )
	{
		glmDelete( pModel );
		pModel = NULL;
	}

	CString FileName;
	FileName = CommonFunction::GetApplicationPath() + _T("ESR.obj");


	//声明标识符   
	USES_CONVERSION ;
	char * pFileName = T2A(FileName); 


	// Load the new obj model
	pModel = glmReadOBJ( pFileName);

	// Generate normal for the model
	glmFacetNormals( pModel );

	// Scale the model to fit the screen
	glmUnitize( pModel, modelCenter );

	// Init the modelview matrix as an identity matrix
	glMatrixMode(GL_MODELVIEW);
	glLoadIdentity();
	glGetDoublev( GL_MODELVIEW_MATRIX, pModelViewMatrix );	

绘制场景及模型

	glClear( GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT );

	glMatrixMode(GL_MODELVIEW);
	glLoadIdentity();

	glTranslated( 0.0, 0.0, -5.0 );
	glMultMatrixd( pModelViewMatrix );

	if ( pModel )
	{
		glmDraw( pModel, GLM_FLAT|GLM_MATERIAL );
	}

窗口改变大小时候重绘模型
OnSize(UINT nType, int cx, int cy)
{
	CDialog::OnSize(nType, cx, cy);

	window_width  = cx;
	window_height = cy;

	// Reset the viewport
	//把像素绘制到窗口的哪个区域,前两个参数定义了视口的左下脚(0,0表示最左下方),后两个参数分别是宽度和高度
	glViewport(0, 0, window_width, window_height);

	//使用glMatrixMode来指定当前操作的究竟是模型视图矩阵还是投影矩阵。
	glMatrixMode(GL_PROJECTION);
	glLoadIdentity();

	//为了得到透视效果,我们使用gluPerspective来设置可视空间。假定可视角为60度(如果调试时发现该角度不合适,可修改之。我在最后选择的数值是75。),
	//高宽比为1.0。最近可视距离为1.0,最远可视距离为200000000*2=400000000。即:gluPerspective(60, 1, 1, 400000000);
	gluPerspective(45.0f, (float)window_width/(float)window_height, 0.01f, 200.0f);
}
鼠标操作

OnLButtonDown(UINT nFlags, CPoint point)
{
	// TODO: 在此添加消息处理程序代码和/或调用默认值
	OldX = point.x;
	OldY = point.y;
	bLeftBntDown = true;
	CDialog::OnLButtonDown(nFlags, point);
}
OnLButtonUp(UINT nFlags, CPoint point)
{
	// TODO: 在此添加消息处理程序代码和/或调用默认值
	bLeftBntDown = false;
	CDialog::OnLButtonUp(nFlags, point);
}
OnMouseMove(UINT nFlags, CPoint point)
{
	// TODO: 在此添加消息处理程序代码和/或调用默认值

	if ( bLeftBntDown && pModel )
	{
		float fOldX =  2.0f*OldX/(float)window_width  - 1.0f;
		float fOldY = -2.0f*OldY/(float)window_height + 1.0f;
		float fNewX =  2.0f*point.x/(float)window_width  - 1.0f;
		float fNewY = -2.0f*point.y/(float)window_height + 1.0f;

		double pMatrix[16];
		trackball_opengl_matrix( pMatrix, fOldX, fOldY, fNewX, fNewY);

		glMatrixMode( GL_MODELVIEW );
		glLoadIdentity();
		glLoadMatrixd( pMatrix );
		glMultMatrixd( pModelViewMatrix );
		glGetDoublev( GL_MODELVIEW_MATRIX, pModelViewMatrix );

		OldX = point.x;
		OldY = point.y;
	}
	CDialog::OnMouseMove(nFlags, point);
}

滚轮操作

OnMouseWheel(UINT nFlags, short zDelta, CPoint pt)
{
	// TODO: 在此添加消息处理程序代码和/或调用默认值

	if (zDelta > 0)
	{
		glMatrixMode( GL_MODELVIEW );
		glLoadIdentity();
		glLoadMatrixd( pModelViewMatrix );
		glScaled( 1.05, 1.05, 1.05 );
		glGetDoublev( GL_MODELVIEW_MATRIX, pModelViewMatrix );
	}
	if (zDelta < 0)
	{
		glMatrixMode( GL_MODELVIEW );
		glLoadIdentity();
		glLoadMatrixd( pModelViewMatrix );
		glScaled( 0.95, 0.95, 0.95 );
		glGetDoublev( GL_MODELVIEW_MATRIX, pModelViewMatrix );
	}
	return CDialog::OnMouseWheel(nFlags, zDelta, pt);
}

源码源码

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章