opengl自学记录_键盘控制图形平移_opengl键盘控制图形平移-程序员宅基地

技术标签: 算法  elasticsearch  大数据  

自学目标:
1.掌握二维变换 数学原理
2.学会应用二维变换

#define GLUT_DISABLE_ATEXIT_HACK
#include "GLUT.H"
#include<math.h>
#include <string.h> 
#define ZVALUE 20.0f

int w_width = 600;
int w_height = 600;
int lineWidth;

//非齐次二维几何变换
struct my_v_inhomogenous
{
    
	int x;
	int y;
};

struct my_v_inhomogeneous rectangle[4];
///

///
///
//
//有一行小弟忘了
/
void init(void)
{
    
	rectangle[0].x = 0;
	rectangle[0].y = 0;

	rectangle[1].x = 80;
	rectangle[1].y = 0;

	rectangle[2].x = 80;
	rectangle[2].y = 40;

	rectangle[3].x = 0;
	rectangle[3].y = 40;
}

//DDA绘制直线
void DDA(int x0, int y0, int x1, int y1)
{
    
	float dx = x1 - x0;
	float dy = y1 - y0;
	float k = dy / dx;
	float x = x0;
	float y = y0;
//y(i+1)=yi+k;运用的是y=kx+b公式
//xi,yi,int(y(i+1)+0.5)。
//斜率过大会离散
//一般来说,k和y都是整数,每一步运算都要对y进行进行四舍五入,运行效率比较低
	while (x <= x1)
	{
    
		glVertex2d(x, y);
		x++;
		y = floor(y + k + 0.5);
	}

}


void Bresenham(int x0, int y0, int x1, int y1)
//只有了int类型的加减和比较
//把水平和垂直的直线单独绘制,其他的划分为八个部分
//这里只写了第一象限的部分
//遍历每一个横坐标 然后找到横坐标所对应的纵坐标	
{
    
	int dy = y1 - y0;
	int dx = x1 - x0;
	int xf = x0;
	int yf = y0;
	int e = -dx;
	while (xf < x1)
	{
    
		glColor3f(1.0, 0.0, 0.0);
		glVertex2d(xf, yf);
		e = e + 2 * dy;
		if (e > 0)
		{
    
			yf++;
			xf++;
			e = e - 2 * dx;
		}
		else
		{
    
			xf++;
		}
	}
}


//绘制正圆
void draw_circle(int r)
{
    
	int x, y;
	x = 0, y = r;
	int d = 3 - 2 * r;
	while (x <= y)
	{
    
		glColor3f(0.0, 1.0, 0.0);
		glVertex2d(x, y);
		glVertex2d(-x, y);
		glVertex2d(x, -y);
		glVertex2d(y, x);
		glVertex2d(-x, -y);
		glVertex2d(-y, -x);
		glVertex2d(y, -x);
		glVertex2d(-y, x);
		if (d > 0)
		{
    
			d += 4 * (x - y) + 10;
			y--;
		}
		else
		{
    
			d += 4 * x + 6;
		}
		x++;
	}
}





//绘制内容
void display(void)
{
    
	glClearColor(1.f, 1.f, 1.f, 0.f);
	glClear(GL_COLOR_BUFFER_BIT);


	glPushMatrix();

	
	glColor3b(0, 0, 1);
	glBegin(GL_POINTS); //GL_POINTS
		DDA(0, 0, 300, 1200);
		Bresenham(-100, -100, +200, 12200);
		draw_circle(100);
	glEnd();
	glBegin(GL_POLYGON); //GL_LINE_LOOP
		for (int vIndex = 0; vIndex < 3; vIndex++)
		{
    
			glVertex2i(rectangle[vIndex].x, rectangle[vIndex].y);
			glVertex2i(rectangle[vIndex + 1].x, rectangle[vIndex + 1].y);
		}	
	glEnd();
	glutSwapBuffers();
	glPopMatrix();
	glFlush();
}


void my_traslate_inhomogeneous(struct my_v_inhomogeneous* polygon, int polygon_vertex_count, int tx, int ty)
{
    
	for (int vIndex = 0; vIndex < polygon_vertex_count; vIndex++)
	{
    
		polygon[vIndex].x += tx;
		polygon[vIndex].y += ty;
	}
}


//投影方式、modelview方式等设置
void reshape(int w, int h)
{
    
	glViewport(0, 0, (GLsizei)w_width, (GLsizei)w_height);//视口大小
	glMatrixMode(GL_PROJECTION);//设置投影模式以及视景体大小
	glLoadIdentity();
	if (w <= h)
		glOrtho(-0.5 * w_width, 0.5 * w_width, -0.5 * w_height * (GLfloat)w_height / (GLfloat)w_width, 0.5 * w_height * (GLfloat)w_height / (GLfloat)w_width,
			-ZVALUE, ZVALUE);
	else
		glOrtho(-0.5 * w_width, 0.5 * w_width, -0.5 * w_height * (GLfloat)w_width / (GLfloat)w_height, 0.5 * w_height * (GLfloat)w_width / (GLfloat)w_height,
			-ZVALUE, ZVALUE);
	glMatrixMode(GL_MODELVIEW);
	glLoadIdentity();
}

void keyboard(unsigned char key, int x, int y)
{
    
	switch (key)
	{
    
	case 'w':
	case 'W':
	{
    
		my_traslate_inhomogeneous(rectangle, 4, 0, 1);//向Y轴正方向移动1个单元

		glutPostRedisplay();
		break;
	}
	case 's':
	case 'S':
	{
    
		my_traslate_inhomogeneous(rectangle, 4, 0, -1);//向Y轴负方向移动1个单元

		glutPostRedisplay();
		break;
	}
	case 'a':
	case 'A':
	{
    
		my_traslate_inhomogeneous(rectangle, 4, -1, 0);//向X轴负方向移动1个单元

		glutPostRedisplay();
		break;
	}
	case 'd':
	case 'D':
	{
    
		my_traslate_inhomogeneous(rectangle, 4, 1, 0);//向X轴正方向移动1个单元

		glutPostRedisplay();
		break;
	}
	case 27:
		exit(0);
		break;
	}
}


//主调函数
int main(int argc, char** argv)
{
    
	glutInit(&argc, argv);
	glutInitDisplayMode(GLUT_SINGLE | GLUT_RGB|GLUT_DEPTH);
	glutInitWindowSize(w_width, w_height);
	glutInitWindowPosition(50, 50);
	glutCreateWindow("2");
	init();
	glutReshapeFunc(reshape);
	glutDisplayFunc(display);
	glutKeyboardFunc(keyboard);
	glutMainLoop();
	return 0;
}

rectangle一直在标红线,说“表达式必须是指向完整对象类型的指针”
和可以运行的代码一行一行对照,我也没发现哪里错了。
定义了一个结构体struct,struct里面定义两个变量 int x和int y。然后又有一个实例rectangle,rectangle需要x和y。
重新复习了一下,指针和结构体的概念,总之看不出来有什么问题。然后重新打了一遍,发现是自己结构体命名少写了一个e
…好了没事了

现在分析一下,非齐次平移变换
非齐次平移变换,就是没有涉及矩阵的变换。直接改变图形的坐标位置达到效果。
需要:

1.结构体

struct my_v_inhomogeneous
{
    float x;
foat y;
}

2.void()
记录增加变量

void my_translate_inhomogeneous(struct my_v_inhomogeneous*polygon,int polygon_vertex_count,int tx,int ty)
{
    
for (int vIndex=0;vIndex<polygon_vertex_count;vIndex++)
{
    
polygon[vIndex].x+=tx;
polygon[vIndex].y+=ty;
}
}

3.键盘or鼠标交互。(这里选择了鼠标)
当按下w时,my_translate_inhomogeneous(rectangle,几个顶点,x轴距离,y距离)便会变换
my_traslate_inhomogeneous(rectangle, 4, 0, 1);//向Y轴正方向移动1个单元。
glutPostRedisplay():标记当前窗口需要重新绘制
polygon[vIndex]的x和y会被影响,tx和ty是会变化的,然后用+=赋值给x和y。

void keyboard(unsigned char key, int x, int y)
{
    
	switch (key)
	{
    
	case 'w':
	case 'W':
	{
    
		my_traslate_inhomogeneous(rectangle, 4, 0, 1);//向Y轴正方向移动1个单元

		glutPostRedisplay();
		break;
	}
	case 's':
	case 'S':
	{
    
		my_traslate_inhomogeneous(rectangle, 4, 0, -1);//向Y轴负方向移动1个单元

		glutPostRedisplay();
		break;
	}
	case 'a':
	case 'A':
	{
    
		my_traslate_inhomogeneous(rectangle, 4, -1, 0);//向X轴负方向移动1个单元

		glutPostRedisplay();
		break;
	}
	case 'd':
	case 'D':
	{
    
		my_traslate_inhomogeneous(rectangle, 4, 1, 0);//向X轴正方向移动1个单元

		glutPostRedisplay();
		break;
	}
	case 27:
		exit(0);
		break;
	}
}

其实我不是很清楚,这里的struct my_v_inhomogeneous*polygon是什么含义。但
是我短暂的学习经验告诉我,看不懂就算了…死嗑更加看不懂,换个时间总能看懂的。

4.display函数。
前面的操作都是为了给x和y重新赋值,但是还没有涉及到显示上面。我们还不能看出来什么变动
这个结构体rectangle的x和y (突然领悟到了struct的作用…可以夹带着改变每个rectangle[i]所对应的x,y)
于是在display里面,要能够显示出来rectangle带来的变化。
glBegin~glend不解释。
遍历四边形的每个顶点,
{画出i顶点
画出下一个顶点}
glVertex2i就是一个绘制二维图形的函数。

glBegin(GL_POLYGON); //GL_LINE_LOOP
		for (int vIndex = 0; vIndex < 3; vIndex++)
		{
    
			glVertex2i(rectangle[vIndex].x, rectangle[vIndex].y);
			glVertex2i(rectangle[vIndex + 1].x, rectangle[vIndex + 1].y);
		}
	glEnd();
版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/weixin_45510058/article/details/120925904

智能推荐

【随机生成验证码】-程序员宅基地

文章浏览阅读472次。随机生成验证码_随机生成验证码

编码规范(二)----JAVA编码规范插件_阿里 java编码规范及插件有哪些-程序员宅基地

文章浏览阅读7.5k次,点赞5次,收藏10次。一、前言在项目开发的过程中,很多时候,无法统一大家的编码规范,你有你的风格,我有我的风格,虽然每个公司都有自己的规范,但是有时候,作为开发人员的我们,写者写着,就按自己的来了,反正有没有人来检查,或者是人为的检查感觉费时间,这时候,就可以通过一些代码插件工具来提高我们的效率,目前常见的有:CheckStyle、阿里巴巴的p3c插件等。CheckStyle简介_阿里 java编码规范及插件有哪些

一次OOM问题排查过程实战记录(内存爆满)_oom时查看内存-程序员宅基地

文章浏览阅读149次。一次OOM问题排查过程实战记录(内存爆满)_oom时查看内存

android 【IM】 聊天布局问题_android多布局的im-程序员宅基地

文章浏览阅读310次。有前辈做过这个东西的布局吗? 怎么才能让里面的文字居中显示这是一个自定义的dialog布局如下: android:layout_width="match_parent" android:layout_height="match_parent" android:orientation="vertical" android:gravity="c_android多布局的im

Windows驱动开发第2课(开发及调试工具介绍)_debugview 驱动调试-程序员宅基地

文章浏览阅读972次,点赞4次,收藏3次。1.开发工具-虚拟机。2.调试工具:debugview。_debugview 驱动调试

5.4--->Ansible中的变量及加密_ansible 不收集事实-程序员宅基地

文章浏览阅读191次。目录一.变量命名:二. 变量级别:三. 变量设定和使用方式:一.在playbook中直接定义变量二. 在文件中定义变量:三.使用变量:四.设定主机变量和清单变量五.目录设定变量六.用命令覆盖变量七.使用数组设定变量,字典的方式八.注册变量 ,register九. 事实变量:十一.魔法变量 :四.JINJA2模板j2书写模板一:j2书写模板二:j2书写模板三:for循环j2书写模板四:if判定五.Ansible的加密控制一.._ansible 不收集事实

随便推点

我的前端成长之路:中医药大学毕业的业务女前端修炼之路_前端通过项目个人成长-程序员宅基地

文章浏览阅读818次。前端工程师的修炼没有捷径,踏踏实实的通过一个个项目的实践来升级打怪实现进阶;本文仅分享自己11年的前端生涯,探讨一直在业务中的技术人的成长之路,也复盘再认识下自己,每个节点我遇到的问题和我的选择。..._前端通过项目个人成长

docker inspect --format 详解-程序员宅基地

文章浏览阅读2.6k次。Docker --format 参数提供了基于 Go模板 的日志格式化输出辅助功能,并提供了一些内置的增强函数。什么是模板?上图是大家熟悉的 MVC 框架(Model View Controller): Model(模型,通常在服务端)用于处理数据、View(视图,客户端代码)用于展现结果、Controller(控制器)用于控制数据流,确保 M 和 V 的同步,即一旦 M 改变,V 也应该同步更新。而对于 View 端的处理,在很多动态语言中是通过在静态 HTML 代码中插入动态数据来实现的。例如 _docker inspect --format

centOS7.4安装jdk1.8+tomcat7_centos7 安装tomcat jdk-程序员宅基地

文章浏览阅读2.1k次。1.准备1.JDK1.8:http://www.oracle.com/technetwork/java/javase/downloads/java-archive-javase8-2177648.html jdk-8u162-linux-x64.tar.gz2.tomcat7:https://tomcat.apache.org/download-70.cgi下载之后通过scp或者winscp等工具..._centos7 安装tomcat jdk

科目一必过宝典_科目一 提取码-程序员宅基地

文章浏览阅读2.8k次,点赞15次,收藏49次。文章目录写在前面知识点速记其他总结危险驾驶交警手势扣分总结违法判刑题违法罚款题人车总结图片写在前面科目一今天满分飘过,特此分享下自己打的笔记。如果你想问我只是科目一而已用的着那么认真吗,别问,问就是因为没复习挂过一次。之所以写这篇文章的原因是提醒大家当时我是过了全部题后之后只做某考宝典的模拟题,一直是95以上,但考试的时候发现考试比手机模拟难的多,好多易错题手机模拟根本碰不到,这里建议大家用网页版的模拟。另外只要大家把我发的知识点记住图片题目看了之后过绝对没问题。知识点速记1.有线城5公7,没线_科目一 提取码

C++实现矩阵类(附代码和功能)_c++矩阵-程序员宅基地

文章浏览阅读6.7w次,点赞173次,收藏663次。本文由两部分组成,第一部分介绍一个在win10系统上运行的exe程序,第二部分介绍通过C++实现矩阵运算的方法(功能会更强大,但不如exe文件操作方便)。 用户界面如下,能够实现矩阵的加、减、乘、除运算,以及矩阵的转置,求逆,求行列式的值等。 读者可以在下载该程序,直接在自己的电脑上运行。下载地址:https://download.csdn.net/do..._c++矩阵

WinForm开发中针对TreeView控件改变当前选择节点的字体与颜色_winform treenode节点forecolor改变不了-程序员宅基地

文章浏览阅读9.1k次。WinForm开发中针对TreeView控件改变当前选择节点的字体与颜色 在B/S开发中,对TreeView控件要改变当前选中节点的颜色比较方便,其有相应的SelectedNodeChanged事件进行控制,但对于WinForm则没有这样方便。申明一下,我在这儿所说的改变当前节点的字体与颜色,主要是在WinForm中的TreeView控件,当前选中节点后,其失去鼠标焦点后节点的字体与颜色失去_winform treenode节点forecolor改变不了