用OpenGL动态绘制线段。事实上非常easy,但到如今为止。网上可參考资料并不多。

于是亲自己主动手写一个函数,方便动态绘制线段。代码例如以下:

#include<GL/glut.h>						//OpenGL有用工具包
#include <Windows.h> /*所遇问题:
1、系统API函数Sleep()不听话,睡眠时快时慢(可能跟我计算机当前执行程序有关吧)
解决方式:重写Sleep()函数。实质为空循环。 仅用于Debug下。Release会将其优化 2、动态画直线,朝某些方向画线时出现块状
解决方式:认认真真把直线方程推导一遍。斜率存在或不存在分别讨论,将数学思想转换为编程思想
*/ void init(void)
{
glClearColor(0.0,0.0,0.0,0.0); //指定清除颜色(黑色),即背景颜色
glShadeModel(GL_FLAT); //设置着色模式。採用恒定着色
} void Sleep(int ms) //覆盖系统API函数
{
for(int i=0;i<ms*ms*ms*10;i++) //数值大小依据实际情况调试
{
;
}
} //特别注意:1、斜率不存在。2、直线斜率一定要保持一致,将直线两点式方程转化为斜率式方程y=kx+c
//k=(y2-y1)/(x2-x1);
//c=(x2*y1-x1*y2)/(x2-x1);
void DrawDynamicLine(GLfloat x1, GLfloat y1, //起点坐标
GLfloat x2, GLfloat y2, //终点坐标
GLfloat red,GLfloat green,GLfloat blue, //线段颜色
int speed, //绘制速度。0-100,值越大速度越大
int type //虚线类型。16位2进制表示。1为实点,0为虚点。0xFFFF为实线
)
{
glEnable(GL_LINE_STIPPLE); //点画线模式
glColor3f(red,green,blue); //设置线段绘制颜色
glLineStipple(1,type); //dashed(虚线),type为16位2进制数。0表示实点,1表示虚点 if (x1!=x2) //假设斜率存在
{
GLfloat k=(y2-y1)/(x2-x1); //直线斜率 y=kx+c
GLfloat c=(x2*y1-x1*y2)/(x2-x1); //直线常数
//假定以A为原点建立二维坐标系,则下边4个if分别相应于:第一象限,第二象限,第三象限,第四象限
if (x1<x2&&y1<=y2) //从A(x1,y1)到B(x2,y2)画线。B点在A点的右上方
{
for (int i=0;i<=x2-x1;i++) //注意循环条件
{
glBegin(GL_LINES);
glVertex2f(x1,y1);
glVertex2f(x1+i,k*(x1+i)+c);
glEnd();
Sleep(100-speed);
glFlush(); //立马開始处理保存在缓冲区中的OpenGL函数调用
}
}
else if (x1>x2&&y1<=y2) //从A(x1,y1)到B(x2,y2)画线,B点在A点的左下方
{
for (int i=0;i>=x2-x1;i--) //特别注意循环条件。既要避免死循环又要避免少循环
{
glBegin(GL_LINES);
glVertex2f(x1,y1);
glVertex2f(x1+i,k*(x1+i)+c);
glEnd();
Sleep(100-speed);
glFlush(); //立马開始处理保存在缓冲区中的OpenGL函数调用
}
}
else if (x1>x2&&y1>=y2) //从A(x1,y1)到B(x2,y2)画线,B点在A点的左上方
{
for (int i=0;i>=x2-x1;i--) //特别注意循环条件,既要避免死循环又要避免少循环
{
glBegin(GL_LINES);
glVertex2f(x1,y1);
glVertex2f(x1+i,k*(x1+i)+c);
glEnd();
Sleep(100-speed);
glFlush(); //立马開始处理保存在缓冲区中的OpenGL函数调用
}
}
else if (x1<x2&&y1>=y2) //从A(x1,y1)到B(x2,y2)画线,B点在A点的左上方
{
for (int i=0;i<=x2-x1;i++) //特别注意循环条件,既要避免死循环又要避免少循环
{
glBegin(GL_LINES);
glVertex2f(x1,y1);
glVertex2f(x1+i,k*(x1+i)+c);
glEnd();
Sleep(100-speed);
glFlush(); //立马開始处理保存在缓冲区中的OpenGL函数调用
}
}
}
else
{
if (y1<y2) //沿y轴正方向画线
{
for (int i=0;i<=y2-y1;i++) //特别注意循环条件,既要避免死循环又要避免少循环
{
glBegin(GL_LINES);
glVertex2f(x1,y1);
glVertex2f(x1,y1+i);
glEnd();
Sleep(100-speed);
glFlush(); //立马開始处理保存在缓冲区中的OpenGL函数调用
}
}
else //沿y轴负方向画线
{
for (int i=0;i>=y2-y1;i--) //特别注意循环条件,既要避免死循环又要避免少循环
{
glBegin(GL_LINES);
glVertex2f(x1,y1);
glVertex2f(x1,y1+i);
glEnd();
Sleep(100-speed);
glFlush(); //立马開始处理保存在缓冲区中的OpenGL函数调用
}
}
} } void display(void)
{
glClear(GL_COLOR_BUFFER_BIT); //清除全部的像素
//正方形ABCD
DrawDynamicLine(100,500,500,500,1,1,0,60,0xFFFF); //AB
DrawDynamicLine(500,500,500,100,1,0,1,70,0xFFFF); //BC
DrawDynamicLine(500,100,100,100,1,1,1,80,0xFFFF); //CD
DrawDynamicLine(100,100,100,500,0,1,1,90,0xFFFF); //DA //沿顺时针方向測试动态画线:第一象限,第四象限。第三象限,第二象限
DrawDynamicLine(300,300,300,500,1,0,0,90,0xFFFF); //1
DrawDynamicLine(300,300,400,500,1,0,0,90,0x00FF);
DrawDynamicLine(300,300,500,500,1,0,0,90,0xFFFF);
DrawDynamicLine(300,300,500,400,1,0,0,90,0x00FF);
DrawDynamicLine(300,300,500,300,1,0,0,90,0xFFFF); DrawDynamicLine(300,300,500,300,1,0,0,90,0xFFFF); //2
DrawDynamicLine(300,300,500,200,1,0,0,90,0x00FF);
DrawDynamicLine(300,300,500,100,1,0,0,90,0xFFFF);
DrawDynamicLine(300,300,400,200,1,0,0,90,0x00FF);
DrawDynamicLine(300,300,400,100,1,0,0,90,0x00FF); DrawDynamicLine(300,300,300,100,1,0,0,90,0xFFFF); //3
DrawDynamicLine(300,300,200,100,1,0,0,90,0x00FF);
DrawDynamicLine(300,300,100,100,1,0,0,90,0xFFFF);
DrawDynamicLine(300,300,100,200,1,0,0,90,0x00FF);
DrawDynamicLine(300,300,100,300,1,0,0,90,0xFFFF); DrawDynamicLine(300,300,100,300,1,0,0,90,0xFFFF); //4
DrawDynamicLine(300,300,100,400,1,0,0,90,0x00FF);
DrawDynamicLine(300,300,100,500,1,0,0,90,0xFFFF);
DrawDynamicLine(300,300,200,500,1,0,0,90,0x00FF);
DrawDynamicLine(300,300,300,500,1,0,0,90,0xFFFF); } void reshape(int w,int h)
{
glViewport(0,0,(GLsizei)w,(GLsizei)h); //为了选择一个更小的画图区域,在窗体中定义一个像素矩形。将图像映射到这个矩形中
glMatrixMode(GL_PROJECTION); //指定哪一个矩阵是当前矩阵(GL_PROJECTION,对投影矩阵应用随后的矩阵操作)
glLoadIdentity(); //将当前的用户坐标系的原点移到了屏幕中心:相似于一个复位操作
gluOrtho2D(0.0,(GLdouble)w,0.0,(GLdouble)h); //将当前的可视空间设置为正投影空间,这个函数描写叙述了一个平行修剪空间,意味着离观察者较远的对象看上去不会变小
} int main(int argc,char** argv) //注意參数
{
glutInit(&argc,argv); //初始化GLUT并处理命令行參数
glutInitDisplayMode(GLUT_SINGLE|GLUT_RGB); //指定模式:单缓存OR双缓存;RGBA模式OR颜色索引模式
glutInitWindowSize(600,600); //指定窗体大小(像素)
glutInitWindowPosition(300,50); //指定窗体左上角在屏幕上的位置
glutCreateWindow(argv[0]); //使用OpenGL场景创建一个窗体,參数为窗体名称
init(); //调用初始化函数
glutDisplayFunc(display); //注冊用于显示图形的回调函数。每当GLUT觉得须要重绘窗体时。都会执行该函数。故应将重绘场景所需调用的函数都放到显示回调函数中。
glutReshapeFunc(reshape); //重绘回调函数
glutMainLoop(); //进入主循环并处理事件。此时创建的全部窗体都会显示出来。被渲染到这些窗体中的内容也将显示出来,程序開始处理事件,注冊的显示回调函数被触发
return 0; //ANSI C要求函数main()返回一个int值
}

效果例如以下:

OpenGL进阶演示样例1——动态画线(虚线、实线、颜色、速度等)的更多相关文章

  1. 01_MUI之Boilerplate中:HTML5演示样例,动态组件,自己定义字体演示样例,自己定义字体演示样例,图标字体演示样例

     1安装HBuilder5.0.0,安装后的界面截图例如以下: 2 依照https://www.muicss.com/docs/v1/css-js/boilerplate-html中的说明,创建上 ...

  2. cmake使用演示样例与整理总结

    本文代码托管于github  cmake_demo cmake中一些提前定义变量 PROJECT_SOURCE_DIR project的根文件夹 PROJECT_BINARY_DIR 执行cmake命 ...

  3. 最简单的视音频播放演示样例5:OpenGL播放RGB/YUV

    ===================================================== 最简单的视音频播放演示样例系列文章列表: 最简单的视音频播放演示样例1:总述 最简单的视音频 ...

  4. 【OpenGL 学习笔记01】HelloWorld演示样例

    <<OpenGL Programming Guide>>这本书是看了忘,忘了又看,赶脚还是把笔记做一做心里比較踏实,哈哈. 我的主题是,好记性不如烂笔头. ========== ...

  5. 最简单的视音频播放演示样例4:Direct3D播放RGB(通过Texture)

    ===================================================== 最简单的视音频播放演示样例系列文章列表: 最简单的视音频播放演示样例1:总述 最简单的视音频 ...

  6. WebGL自学教程——WebGL演示样例:開始

    最终開始WebGL的演示样例了,...... 開始 使用WebGL的步骤,非常easy: 1. 获得WebGL的渲染环境(也叫渲染上下文). 2. 发挥你的想象力,利用<WebGL參考手冊> ...

  7. 最简单的视音频播放演示样例8:DirectSound播放PCM

    ===================================================== 最简单的视音频播放演示样例系列文章列表: 最简单的视音频播放演示样例1:总述 最简单的视音频 ...

  8. Cocos2d-x-Lua演示样例项目HelloLua

    Cocos2d-x-Lua演示样例项目HelloLua 本篇博客介绍Cocos2d-x中Lua的实例项目,就是使用Cocos2d-x创建的初始项目执行所呈现的农场,这里笔者取名为HelloLua.本篇 ...

  9. 『HTML5梦幻之旅』 - 仿Qt演示样例Drag and Drop Robot(换装机器人)

    起源 在Qt的演示样例中看到了一个有趣的demo.截图例如以下: 这个demo的名字叫Drag and Drop Robot,简单概括而言,在这个demo中,能够把机器人四周的颜色拖动到机器人的各个部 ...

随机推荐

  1. 利用RELK进行日志收集

    利用RELK进行日志收集 发布时间:April 3, 2018 // 分类:运维工作,开发笔记,python // No Comments 前不久在做应急的总是遇到要求对日志进行分析溯源,当时就想到如 ...

  2. SSL证书切换

    SSL证书:SSL证书是数字证书的一种,类似于驾驶证.护照和营业执照的电子副本.因为配置在服务器上,也称为SSL服务器证书.SSL 证书就是遵守 SSL协议,由受信任的数字证书颁发机构CA,在验证服务 ...

  3. 转 oracheck

    转 https://dbaplus.cn/news-10-220-1.html ORAchk 之前被称为RACcheck,后来它的检查范围进行了扩展,改名为了ORAchk,它是在数据库系统进行健康检查 ...

  4. Tiles框架入门教程

    1.为何选用Tiles 刚接触Java Web开发的人都知道,JSP中可以通过include标签动态插入一个JSP页面.在了解这个功能后可能会兴奋不已,因为这样可以实现多个JSP页面共用一个JSP的内 ...

  5. 数据结构---Java---LinkedList

    public class LinkedList<E> extends AbstractSequentialList<E> implements List<E>, D ...

  6. Murano Weekly Meeting 2016.07.19

    Meeting time: 2016.July.19 1:00~2:00 Chairperson:  Kirill Zaitsev, from Mirantis Meeting summary: 1. ...

  7. 【LDAP】Openldap导入数据

    原创作品,允许转载,转载时请务必以超链接形式标明文章 原始出处 .作者信息和本声明.否则将追究法律责任.http://jerry12356.blog.51cto.com/4308715/1851186 ...

  8. stm32 输入捕获学习(二)

    (本文参考STM32  开发指南 V1.3   -- ALIENTEK 战舰 STM32 开发板库函数教程 ) 1. 实验设计 我们用 TIM5 的通道 1(PA0)来做输入捕获,捕获 PA0 上高电 ...

  9. windows下自动更改IP的小工具(bat批处理文件)

    每次上线,都要先上灰度环境再上到正式环境.在上到灰度环境时,访问灰度环境通过自动获取IP和更改指定dns来实现.具体如何实现,大家可自行百度. 新建一个文本文档,将其后缀改为bat,打开该文件,拷贝一 ...

  10. HDU 4738——Caocao's Bridges——————【求割边/桥的最小权值】

     Caocao's Bridges Time Limit:1000MS     Memory Limit:32768KB     64bit IO Format:%I64d & %I64u S ...