从书上和别人的言论里,听到很多说法:“先平移后旋转,先旋转后平移是不一样的” “平移和旋转的操作对象是物体,还是坐标系?” “旋转的第一个角度参数是为逆时针指定的” “程序指定顺序和机器执行顺序是相反的”..所谓“眼见为实”,亲自试验,才可信:
先上图:

平移操作:glTranslatef(180.0f,0.0f,0.0f);
旋转操作:glRotatef(105.0f,0.0f,0.0f,1.0f);
(开始采用的是球体,但不如有棱角的立方体更能说明问题)
第一个图是先旋转,后平移的效果。它能说明:1.glTranslatef glRotatef的对象都是坐标系,否则蓝色的立方体不会出现在左上方(可能会出现在右方);2.机器执行顺序与代码指定顺序是一致的,否则蓝色的立方体不会是“绕自身中心旋转”的效果,而可能是仍然“端端正正”--绕非自身轴旋转的效果;3.glRotatef函数的第一个角度参数确实是针对逆时针(CCW)的。
第二个图是先平移,后旋转的效果。起初怎么也不解,甚至怀疑库文件是否完整(专门换了一个更早的库版本跑同一个测试程序,结果是一样的,这至少证明库文件没问题)。后来,洗了个澡,一身清爽,刹那间就来了灵感:这样才对啊,因为平移的是坐标系,也就是经过glTranslate后原点平移到了图中蓝色立方体的中心,然后坐标系(带动它上面的stuff)绕z轴(由屏幕指向外)旋转105°--关键是原点的位置。
ps:遇事多动脑筋,问题总能解决的。加油!
贴上测试代码:
// Test4GL.cpp : Defines the entry point for the console application.
//
//MBD,不能运行。dog-shit stuff.改成GLUT_SINGLE,glFlush()就能跑了。
#define FREEGLUT_STATIC
#include <GL/glut.h>
#include <GL/gl.h>
#include <GL/glu.h>
void SetupRC()
{
glClearColor(0.0f,0.0f,0.0f,1.0f);
glColor3f(1.0f,0.0f,0.0f);
glEnable(GL_DEPTH_TEST);
}
void RenderScene()
{
glClear(GL_COLOR_BUFFER_BIT|GL_DEPTH_BUFFER_BIT);
glPushMatrix();
glColor3f(1.0f,0.0f,0.0f);
//glutSolidSphere(20.0,15,15);
glutSolidCube(48);
glColor3f(0.0f,0.0f,1.0f);
//这里可以更改平移、旋转的先后顺序
glRotatef(105.0f,0.0f,0.0f,1.0f);
glTranslatef(180.0f,0.0f,0.0f);
//glutSolidSphere(20.0,14,14);
glutSolidCube(48);
glPopMatrix();
//glutSwapBuffers();
glFlush();//没有这句,只显示白茫茫的一片 。对比其他程序好像都没写glFlush().?
}
void ChangeSize(int w,int h)
{
if(h==0)
h=1;
glViewport(0,0,w,h);
glMatrixMode(GL_PROJECTION);
glLoadIdentity();
if(w<=h)
glOrtho(-200.0f,200.0f,-200.0f,200.0f*h/w,200.0f,-200.0f);
else
glOrtho(-200.0f,200.0f*w/h,-200.0f,200.0f,200.0f,-200.0f);
glMatrixMode(GL_MODELVIEW);
glLoadIdentity();
glutPostRedisplay();
}
int main(int argc,char** argv)
{
glutInit(&argc,argv);
glutInitWindowSize(400,300);
glutCreateWindow("Testing glRotatef");
glutInitDisplayMode(GLUT_RGBA|GLUT_DEPTH);
glutDisplayFunc(RenderScene);
glutReshapeFunc(ChangeSize);
SetupRC();
glutMainLoop();
return 0;
}
