從書上和別人的言論里,聽到很多說法:“先平移后旋轉,先旋轉后平移是不一樣的” “平移和旋轉的操作對象是物體,還是坐標系?” “旋轉的第一個角度參數是為逆時針指定的” “程序指定順序和機器執行順序是相反的”..所謂“眼見為實”,親自試驗,才可信:
先上圖:
平移操作: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; }