glPushMatrix() and glPopMatrix()以及glMatrixMode的用法

所有的这几个函数的本质都是对坐标系的操作。

前提知识:

1。每一种矩阵对应着一种坐标系。见图一

2。每一次对矩阵的操作都是对当前矩阵进行的。确切的说,作为一种状态机,每一时刻在矩阵方面只能有一种状态。

3。每一次压栈和出栈都对应着一种对坐标系的操作。

4。我们所有对矩阵的操作都是对矩阵栈的栈顶矩阵进行的。

图一

如图:全局坐标系中有一个局部坐标系,这个坐标系的状态对应着一种矩阵。假如当前矩阵为该矩阵,则该矩阵为矩阵栈中的栈顶元素。假如此时又进行了一 系列glTranslate(),glRotatef()等操作,都是对当前矩阵也就是栈顶矩阵进行了修改。若此时调用glPushMatrix()函 数,则将该矩阵压入栈中第二层的位置,当然此时栈顶位置的矩阵也与此矩阵相同。

当进行完一系列变换之后,栈顶矩阵为该矩阵。若此时调用glPopMatrix()函数,则恢复当前矩阵为该坐标系的状态。以前所进行的有关修改都被忽略了。

记住一句话,所有对矩阵进行的操作都是对栈顶矩阵进行的,压进去的矩阵式哪个矩阵,当前矩阵就是哪个矩阵,弹出来的是哪个矩阵,当前矩阵就为哪个矩阵。

记住:一个矩阵对应一个局部坐标系。
glPushMatrix
pushes the current matrix stack down by one, duplicating the current matrix. That is, after a
glPushMatrix call, the matrix on top of the stack is identical to the one below it.

见图二:

图二

glMatrixMode()的作用就是指定当前操作的矩阵为哪一个矩阵。总共有三种参数类型

1。GL_PROJECTION 表示当前操作的矩阵为透视矩阵,用于设定视景体的矩阵

2.GL_MODEVIEW 表示当前操作的矩阵为模型视点矩阵,用于对三维场景中坐标系的变换操作。

帮我注释以下代码#include <GL/glut.h> #include <math.h> GLfloat theta = 0; void init() { glClearColor(1.0, 0.0, 0.0, 0.0); glShadeModel(GL_SMOOTH); glMatrixMode(GL_PROJECTION); gluOrtho2D(-2.0, 2.0, -2.0, 2.0); glMatrixMode(GL_MODELVIEW); } void mydraw() { glBegin(GL_TRIANGLES); glColor3f(1.0, 0.0, 0.0); glVertex3f(0.0, 1.0, 0.0); glColor3f(0.0, 1.0, 0.0); glVertex3f(0.0, 0.0, 0.0); glColor3f(0.0, 0.0, 1.0); glVertex3f(0.5, 0.5, 0.0); glEnd(); } void Mydisplay(void) { glClear(GL_COLOR_BUFFER_BIT); void glPushMatrix(void); mydraw(); void glPopMatrix(void); void glPushMatrix(void); glRotatef(theta, 0.0, 0.0, 1.0); mydraw(); void glPopMatrix(void); void glPushMatrix(void); glRotatef(theta, 0.0, 0.0, 1.0); mydraw(); void glPopMatrix(void); void glPushMatrix(void); glRotatef(theta, 0.0, 0.0, 1.0); mydraw(); void glPopMatrix(void); glFlush(); } void MyIdle(void) { theta += 15; if (theta >= 360) theta = 0; glutPostRedisplay(); } void reshape(int width, int height) { glViewport(0, 0, (GLsizei)width, (GLsizei)height); glMatrixMode(GL_PROJECTION); glLoadIdentity(); gluPerspective(60.0, (GLfloat)width / (GLfloat)height, 1.0, 100.0); glMatrixMode(GL_MODELVIEW); glLoadIdentity(); glTranslatef(0.0, 0.0, -3.0); } int main(int argc, char** argv) { glutInit(&argc, argv); glutInitDisplayMode(GLUT_SINGLE | GLUT_RGB); glutInitWindowSize(500, 500); glutInitWindowPosition(100, 100); glutCreateWindow("动画"); init(); glutDisplayFunc(Mydisplay); glutReshapeFunc(reshape); glutIdleFunc(&MyIdle); glutMainLoop(); return 0; }
05-10
有了freeglut.lib和freeglut.dll为什么小熊猫c++的编译器会报错:[错误] ld returned 1 exit status #include <GL/glut.h> #include <cmath> #include <cstdio> // 游戏状态 float basketballPos[3] = {0.0f, 1.0f, 0.0f}; float basketPos[3] = {0.0f, 3.0f, -5.0f}; int score = 0; bool isShooting = false; float velocity = 0.0f; float angle = 45.0f; float time = 0.0f; void init() { glEnable(GL_DEPTH_TEST); glClearColor(0.53f, 0.81f, 0.98f, 1.0f); // 天空蓝背景 } void drawBasketball() { glColor3f(1.0f, 0.5f, 0.0f); // 橙色 glPushMatrix(); glTranslatef(basketballPos[0], basketballPos[1], basketballPos[2]); glutSolidSphere(0.3f, 32, 32); glPopMatrix(); } void drawBasket() { // 篮板 glColor3f(0.8f, 0.8f, 0.8f); glPushMatrix(); glTranslatef(basketPos[0], basketPos[1], basketPos[2]); glScalef(2.0f, 1.5f, 0.1f); glutSolidCube(1.0f); glPopMatrix(); // 篮筐 glColor3f(1.0f, 0.0f, 0.0f); glPushMatrix(); glTranslatef(basketPos[0], basketPos[1]-0.3f, basketPos[2]); glutSolidTorus(0.05f, 0.3f, 16, 16); glPopMatrix(); } void displayScore() { glColor3f(1.0f, 1.0f, 1.0f); glMatrixMode(GL_PROJECTION); glPushMatrix(); glLoadIdentity(); gluOrtho2D(0, 400, 0, 300); glMatrixMode(GL_MODELVIEW); glPushMatrix(); glLoadIdentity(); char buffer[50]; snprintf(buffer, sizeof(buffer), "Score: %d", score); glRasterPos2i(10, 280); for (char* c = buffer; *c != '\0'; c++) { glutBitmapCharacter(GLUT_BITMAP_HELVETICA_18, *c); } glPopMatrix(); glMatrixMode(GL_PROJECTION); glPopMatrix(); glMatrixMode(GL_MODELVIEW); } void updatePhysics() { if (isShooting) { time += 0.016f; // 约60FPS // 抛物线运动公式 basketballPos[0] = velocity * time * cos(angle * 3.14159f / 180.0f); basketballPos[1] = velocity * time * sin(angle * 3.14159f / 180.0f) - 0.5f * 9.8f * time * time; basketballPos[2] = -velocity * time * 0.5f; // 碰撞检测 if (fabs(basketballPos[0] - basketPos[0]) < 0.3f && fabs(basketballPos[1] - (basketPos[1]-0.3f)) < 0.2f && fabs(basketballPos[2] - basketPos[2]) < 0.3f) { score++
03-29
#include <windows.h> #include <GL/glut.h> void init (void) { glClearColor (1.0, 1.0, 1.0, 0.0); glMatrixMode (GL_PROJECTION); gluOrtho2D (-5.0, 5.0, -5.0, 5.0); //设置显示的范围是X:-5.0~5.0, Y:-5.0~5.0 glMatrixMode (GL_MODELVIEW); } void drawSquare(void) //绘制中心在原点,边长为2的正方形 { glBegin (GL_POLYGON); //顶点指定需要按逆时针方向 glVertex2f (-1.0f,-1.0f); //左下点 glVertex2f (1.0f,-1.0f); //右下点 glVertex2f (1.0f, 1.0f); //右上点 glVertex2f (-1.0f,1.0f); //左上点 glEnd ( ); } void myDraw (void) { glClear (GL_COLOR_BUFFER_BIT); //清空 glLoadIdentity(); //将当前矩阵设为单位矩阵 glPushMatrix(); glTranslatef(0.0f,2.0f,0.0f); glScalef(3.0,0.5,1.0); glColor3f (1.0, 0.0, 0.0); drawSquare(); //上面红色矩形 glPopMatrix(); glPushMatrix(); glTranslatef(-3.0,0.0,0.0); glPushMatrix(); glRotatef(45.0,0.0,0.0,1.0); glColor3f (0.0, 1.0, 0.0); drawSquare(); //中间左菱形 glPopMatrix(); glTranslatef(3.0,0.0,0.0); glPushMatrix(); glRotatef(45.0,0.0,0.0,1.0); glColor3f (0.0, 0.7, 0.0); drawSquare(); //中间中菱形 glPopMatrix(); glTranslatef(3.0,0.0,0.0); glPushMatrix(); glRotatef(45.0,0.0,0.0,1.0); glColor3f (0.0, 0.4, 0.0); drawSquare(); //中间右菱形 glPopMatrix(); glPopMatrix(); glTranslatef(0.0,-3.0,0.0); glScalef(4.0,1.5,1.0); glColor3f (0.0, 0.0, 1.0); drawSquare(); //下面蓝色矩形 glFlush ( ); } void main (int argc, char** argv) { glutInit (&argc, argv); glutInitDisplayMode (GLUT_SINGLE | GLUT_RGB); glutInitWindowPosition (0, 0); glutInitWindowSize (600, 600); glutCreateWindow ("几何变换示例"); init(); glutDisplayFunc (myDraw); glutMainLoop(); } 设计 2-3 个不同的二维图形(如 “太阳 + 行星 + 卫星” 组合、“花朵 + 花茎 + 叶子” 组合),实现 “多图形联动旋转”:每个图形有独立的旋转参数(角度、方向、速度),且支持通过交互(如键盘按键)保存 / 恢复所有图形的旋转状态(如当前角度、速度)。按照上述示例
最新发布
11-13
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值