OpenGL图像渲染过程3.0

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了OpenGL图像渲染过程3.0相关的知识,希望对你有一定的参考价值。

参考技术A

背景: 3D图形渲染的时候,需要决定哪些部分是对观察者可见的,哪些部分对观察者是不可见的,对不可见的部分,应该丢弃。
这个过程叫做隐藏面消除(hidden surface elimination)。

先绘制场景中距离观察者较远的物体,再绘制场景中离观察者较近的物体。

OpenGL不去渲染视角看不到的面
OPenGL可以做到检查正面朝着观察者的面,丢弃背向的面,节约片元着色器的性能。

正背面的区分:按照逆时针顶点连接顺序的面为正面。

深度缓冲区(DepthBuffer)和颜色缓冲区(ColorBuffer)是对应的。深度缓冲区存储像素的深度信息,颜色缓冲区存储像素的颜色信息。在决定是否绘制一个物体表面时候,首先要将物体表面的像素的深度值和当前深度缓冲区中的数据进行比较,如果大于深度缓冲区中的深度,就丢弃物体表面的颜色值和深度值,如果小于就使用物体表面的颜色值,并更新这个像素的深度值和颜色值。这个过程叫做深度测试。

清除深度缓冲区的时候,默认值是1.范围是(0-1) 之间,值越小,表示越靠近观察者。

类型快查

启用polygon offset,让深度之间产生一个间隔,如果两个图形之间有间隔,将间隔值略微增加,让重叠的两个图形与之前的有所区分。

每个片元(Fragment)的深度值都会增加一些偏移量
m多边形深度斜率最大值
r常量,窗口坐标系能分辨的最小差值

offset是负值将使z值距离我们更近。一般都设置成-1 -1.factor和units.

当深度缓冲区被关闭的时候,新的颜色将简单的覆盖原来的颜色缓存区存在的颜色值。
当深度缓冲弄区再次被打开时候,新的颜色片段只有当离原来的值更接近裁剪平面时候才会替换原来的颜色片段。????

目标颜色:已经存储在颜色缓存区的颜色

源颜色:作为当前渲染命令结果,进入颜色缓存区的颜色

最终颜色的计算是由颜色混合方程式决定的

Cf :最终的参数颜色
Cs :源颜色
Cd :目标颜色
S :源混合因子
D: 目标混合因子

混合方程式快查

S源混合因子
D目标混合因子
C表示常量颜色,黑色

混合因子计算方法快查

混合函数经常用于实现在其他一些不透明物体前绘制一个透明物体的效果。

使用 OpenGL 将图像渲染到子类 QDeclarativeItem

【中文标题】使用 OpenGL 将图像渲染到子类 QDeclarativeItem【英文标题】:Render Image into subclassed QDeclarativeItem using OpenGL 【发布时间】:2012-01-21 11:40:28 【问题描述】:

我正在尝试使用 OpenGL ES 2.0 执行多个图像操作,并将输出显示到子类 QDeclarativeItem 中,然后将在我的 QML GUI 中使用它。 我通读了这里找到的答案:http://developer.qt.nokia.com/forums/viewthread/4109 并通过覆盖 QDeclarativeItem 的 paint() 方法成功地在我的 QML GUI 中绘制了一个从红色到蓝色的矩形:

void GLDeclarativeItem::paint(QPainter *painter, const QStyleOptionGraphicsItem *option, QWidget *widget)

    painter->beginNativePainting();
    glBegin(GL_QUADS);
    glColor3ub(0,0,255);
    glVertex2d(0, 0);
    glVertex2d(0, height());
    glColor3ub(255,0,0);
    glVertex2d(width(), height());
    glVertex2d(width(), 0);
    glEnd();
    painter->endNativePainting();

但是,我想要实现的是绘制将在我的自定义 QGLWidget 中处理的图像作为上述 QDeclarativeItem 的内容(而不是红色到蓝色的内容)。

在我使用的自定义 QGLWidget 中:

void GLWidget::paintGL()

    glClear(GL_COLOR_BUFFER_BIT);
    glDrawArrays(GL_TRIANGLE_STRIP, 0, 4);

用于绘图,效果很好。但是,我找不到我的 GLWidget 中的绘图和我的 GLDeclarativeItem 之间的转换。到目前为止,我尝试过的所有东西都只是给了我一个没有任何内容的矩形。 感谢您的帮助!

【问题讨论】:

【参考方案1】:

我做了类似的事情,但解决方法不同:我使用了叠加层

我只是将 OpenGL 渲染放在它自己的小部件中,该小部件放置在 QDeclarativeView 的顶部。

这很好用,尽管您不能在 OpenGL 渲染上进行绘制。如果您确实需要,请叠加另一个具有半透明背景的 QML 视图。

祝你好运。

【讨论】:

谢谢赫尔穆特,听起来很有希望。虽然我不确定如何将小部件作为覆盖添加到 QDeclarativeView。你有任何代码可以帮助我吗?可以很抽象,只是给我一个提示。再次感谢。 嗨,想法是这样的:我猜你使用了QDeclarativeView *myview。然后执行QGLWidget * myglwidget = new QGLWidget(myview)(使用您自己的 QGLWidget 子类)并相应地定位它。这样,myglwidgetmyview 的子代,而不是真正的QGraphicsItem。这对你有帮助吗?

以上是关于OpenGL图像渲染过程3.0的主要内容,如果未能解决你的问题,请参考以下文章

OpenGL: 渲染管线理论

基于OpenGL编写一个简易的2D渲染框架-04 绘制图片

使用 opengl 在球体上渲染图像

OpenGL入门之渲染管线pipeline,着色器Shader

将 OpenGL 渲染保存到图像文件

在 Haskell 中使用 OpenGL 渲染 PNG 图像