第十一节—混合

本文为L_Ares个人写作,包括图片皆为个人亲自操作,以任何形式转载请表明原文出处。

混合是一个非常非常重要的内容,和上两节的深度和正背面剔除一样,因为我们在绘制图形的时候,一般的图形都是带有颜色的,当两个图形存在重叠区域的时候,深度问题在上一节已经学习了一些内容了,但是如果z值较小的图形是半透明状态的话,重叠区域的颜色就不可能是这两个图形的任何一种,那么这个颜色如何来实现呢?这也就是混合的重要之处。

一、什么是混合

这里存在着和深度的配合。我们已经知道在OpenGL中存在着深度缓冲区和颜色缓冲区,其中存储了像素点的深度和颜色值,当深度测试开启的时候,深度缓冲区会对比已存储的深度值和新图形的像素点深度值,这有可能会发生改变,就需要更新深度缓冲区的值。

同时,由于该像素点的深度发生了变化,那么这个像素点之前的颜色值就已经不是现在要绘制的图形的颜色值了,新的像素点颜色值也将覆盖原来深度缓冲区中的颜色值。

同理,当关闭深度测试的时候,也可能会发生变化。

开启颜色混合

glEnable(GL_BLEND)

颜色混合不是开启了就可以的,因为它有一个颜色混合的计算方程式,我们先看方程式。

当混合功能被启用时,默认情况下的混合方程式如(1.1)式所示

Cf = (Cs * S) + (Cd * D)\tag{1.1}

其中,

Cf:计算所得最终颜色参数
Cs:源颜色
Cd:目标颜色
S :源混合因子
D : 目标混合因子

Cf好理解,那么余下4个都是什么意思呢?

源颜色

渲染命令结果进入颜色缓冲区的颜色。也就是新来的颜色

目标颜色

之前存储在颜色缓冲区中的颜色

而混合因子则需要手动设置,利用函数void glBlendFunc(GLenum S , GLenum D)
这函数的两个参数都是枚举类型,所以也是由系统提供来进行选择的,可选择的值如下图1.1所示。

其中,S是源混合因子,D是目标混合因子。

1.1.png

图中的RGBA就是正常的Red、Green、Blue、Alpha。S代表源,D代表目标,C代表常量颜色,默认是黑色。

举例:

若设置源混合因子和目标混合因子的参数如下:
glBlendFunc(GL_SRC_ALPHA,GL_ONE_MINUS_SRC_ALPHA)
若颜色缓冲区颜色即目标颜色Cd为红色(1.f,0.f,0.f,0.f),新进入的颜色即源颜色Cs为透明度为0.6的蓝色(0.f,0.f,1.f,0.6f)
那么混合完成后的颜色可如下计算:

Cd = (1,0,0,0)
Cs=(0,0,1,0.6)
S=源alpha值=0.6
D=1-源alpha值=0.4

所以方程式Cf = (Cs * S) + (Cd * D) =>(Blue * 0.6) + (Red * 0.4)

混合计算的方式就是如上所示。

可以看出,如果使用这样的混合因子,那么源颜色蓝色的alpha值越高,目标颜色的alpha值就越低,原来存储在颜色缓冲区中的红色的成分也就越少。

混合函数的使用场景:在不透明的物体前增加了一块儿透明带颜色的物体的效果。

二、扩展——其他的组合方程式和复杂一点的混合因子选择函数

这一模块作为扩展来说。混合方程式肯定不止一种,那么下面来介绍一下如何选用其他的混合方程式。另外,混合因子的选择也可以有一种适应复杂情况的函数可以了解一下。

颜色混合方程式选择

默认颜色混合方程式:Cf = (Cs * S) + (Cd * D)

选择混合方程式的函数:glBlendEquation(GLenum mode),mode的可选值如下图4.2.1所示。

4.2.1.png

复杂一点的混合因子选择函数

void glBlendFuncSeparate(GLenum strRGB,GLenum dstRGB ,GLenum strAlpha,GLenum dstAlpha)

strRGB:源颜色的混合因子
dstRGB:目标颜色的混合因子
strAlpha:源颜色的Alpha因子
dstAlpha:目标颜色的Alpha因子

也就是说这个函数可以单独的为源和目标颜色的alpha值单独指定混合因子。

在混合因子的可选列表中存在GL_CONSTANT_COLORGL_ONE_MINUS_CONSTANT_COLORGL_CONSTANT_ALPHAGL _ONE_MINUS_CONSTANT允许混合方程式中引入一个常量混合颜色。

常量混合颜色默认是黑色可以通过如下函数修改:

void glBlendColor(GLclampf red ,GLclampf green ,GLclampf blue ,GLclampf alpha )

三、代码实例


//
//  main.cpp
//  05颜色混合
//
//  Created by EasonLi on 2020/8/27.
//  Copyright © 2020 EasonLi. All rights reserved.
//

#pragma mark - 引用类
//这里面包括了大多数GLTool类似于C语言的类。
//包括了GLBatch三角形批次类和gltSetWorkingDirectory
#include "GLTools.h"
//包括了管理和创建管理着色器的管理者的方法,还有一组存储着色器,可用于一些初级操作
#include "GLShaderManager.h"
//利用宏定义来判断系统,根据不同系统引入不同的glut
#ifdef __APPLE__
#include <glut/glut.h>
#else
#define FREEGLUT_STATIC
#include <GL/glut.h>
#endif

#pragma mark - 定义需要用到的变量

//先定义五个三角形批次类,分别用于四个固定矩形和一个移动的带透明度的矩形
GLBatch redBatch;
GLBatch blueBatch;
GLBatch greenBatch;
GLBatch blackBatch;
GLBatch moveBatch;

//定义着色器管理者
GLShaderManager shaderManager;

//定义矩形的边长的一半
GLfloat blockSize = 0.2f;

//定义一个移动的矩形的顶点坐标数组
GLfloat moveVerts[] = {
    
    -blockSize,blockSize,0,
    blockSize,blockSize,0,
    blockSize,-blockSize,0,
    -blockSize,-blockSize,0
    
};

//定义红色矩形顶点坐标数组
GLfloat redVerts[] = {
    0.25f, 0.25f, 0.0f,
    0.75f, 0.25f, 0.0f,
    0.75f, 0.75f, 0.0f,
    0.25f, 0.75f, 0.0f
};

//定义绿色矩形顶点坐标数组
GLfloat greenVerts[] = {
    -0.75f, 0.25f, 0.0f,
    -0.25f, 0.25f, 0.0f,
    -0.25f, 0.75f, 0.0f,
    -0.75f, 0.75f, 0.0f
};

//定义蓝色矩形顶点坐标数组
GLfloat blueVerts[] = {
    -0.75f, -0.75f, 0.0f,
    -0.25f, -0.75f, 0.0f,
    -0.25f, -0.25f, 0.0f,
    -0.75f, -0.25f, 0.0f
};

//定义黑色矩形顶点坐标数组
GLfloat blackVerts[] = {
    0.25f, -0.75f, 0.0f,
    0.75f, -0.75f, 0.0f,
    0.75f, -0.25f, 0.0f,
    0.25f, -0.25f, 0.0f
};

//定义颜色数组
//可移动的矩形颜色数组
GLfloat vMove[] = {0.8,0.5,0.3,0.5};
//红色
GLfloat vRed[] = {1.f,0.f,0.f,1.f};
//绿色
GLfloat vGreen[] = {0.f,1.f,0.f,1.f};
//蓝色
GLfloat vBlue[] = {0.f,0.f,1.f,1.f};
//黑色
GLfloat vBlack[] = {0.f,0.f,0.f,1.f};


#pragma mark - 函数

/**
 创建并设置视口
 1.触发本函数的条件
 (1)视口第一次创建
 (2)视口大小发生改变
 2.处理的业务
 (1)设置OpenGL的视口
 (2)设置投影方式
 3.参数说明
 (1)x,y都是代表了窗口的左下角坐标,所以我设置了0,0
 (2)w和h是宽和高的单位都是像素
 */
void ChangeSize(int w, int h)
{
    
    glViewport(0, 0, w, h);
    
}

/**
 设置渲染环境
 1.触发本函数的条件
 (1)手动触发
 2.处理的业务
 (1)设置窗口的清屏颜色
 (2)初始化着色器管理器
 (3)设置顶点信息和绘制方式
 (4)利用三角形批次类将渲染需要的数据拷贝到存储着色器
 */
void SetUpRC()
{
    
    //设置清屏颜色
    glClearColor(0.3f, 0.3f, 0.3f, 0.f);
    //初始化着色器管理器
    shaderManager.InitializeStockShaders();
    
    //设置顶点信息和绘制方式
    //移动的矩形
    moveBatch.Begin(GL_TRIANGLE_FAN, 4);
    moveBatch.CopyVertexData3f(moveVerts);
    moveBatch.End();
    
    //红色矩形
    redBatch.Begin(GL_TRIANGLE_FAN, 4);
    redBatch.CopyVertexData3f(redVerts);
    redBatch.End();
    
    //绿色矩形
    greenBatch.Begin(GL_TRIANGLE_FAN, 4);
    greenBatch.CopyVertexData3f(greenVerts);
    greenBatch.End();
    
    //蓝色矩形
    blueBatch.Begin(GL_TRIANGLE_FAN, 4);
    blueBatch.CopyVertexData3f(blueVerts);
    blueBatch.End();
    
    //黑色矩形
    blackBatch.Begin(GL_TRIANGLE_FAN, 4);
    blackBatch.CopyVertexData3f(blackVerts);
    blackBatch.End();
    
}

/**
 渲染
 1.触发条件
 (1)手动触发
 (2)系统自动触发
 2.处理业务
 (1)清空缓冲区
 (2)调用着色器
 (3)绘制图形
 */
void RenderScene()
{
    
    //清空缓冲区
    glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT | GL_STENCIL_BUFFER_BIT);
    
    //选定着色器种类和要绘制的颜色
    //种类:统一都是单元着色器
    //颜色:用各自对应的颜色
    
    //开启颜色混合
    glEnable(GL_BLEND);
    //设置混合因子
    glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
    
    //红色矩形
    shaderManager.UseStockShader(GLT_SHADER_IDENTITY,vRed);
    redBatch.Draw();
    
    //绿色矩形
    shaderManager.UseStockShader(GLT_SHADER_IDENTITY,vGreen);
    greenBatch.Draw();
    
    //蓝色矩形
    shaderManager.UseStockShader(GLT_SHADER_IDENTITY,vBlue);
    blueBatch.Draw();
    
    //黑色矩形
    shaderManager.UseStockShader(GLT_SHADER_IDENTITY,vBlack);
    blackBatch.Draw();
    
    //移动矩形
    shaderManager.UseStockShader(GLT_SHADER_IDENTITY,vMove);
    moveBatch.Draw();
    
    glDisable(GL_BLEND);
    
    glutSwapBuffers();
    
}

#pragma mark - 特殊键位

//移动方块
void SpecialKeys(int key, int x, int y)
{
    
    //每次移动的距离
    GLfloat stepSize = 0.025f;
    
    //只要计算一个坐标顶点的移动,然后加上边长就可以计算其他的顶点
    //此处拿出第一个顶点做移动
    GLfloat blockX = moveVerts[0];
    GLfloat blockY = moveVerts[1];
    
    //判定键位做移动
    switch (key) {
        case GLUT_KEY_UP:
            blockY += stepSize;
            break;
        case GLUT_KEY_DOWN:
            blockY -= stepSize;
            break;
        case GLUT_KEY_LEFT:
            blockX -= stepSize;
            break;
        case GLUT_KEY_RIGHT:
            blockX += stepSize;
            break;
    }
    
    if (blockX < -1.f) {
        blockX = -1.f;
    }
    
    if (blockX > (1.f - 2 * blockSize)) {
        blockX = 1.f - 2 * blockSize;
    }
    
    if (blockY > 1.f) {
        blockY = 1.f;
    }
    
    if (blockY < (-1.f + 2 * blockSize)) {
        blockY = -1.f + 2 * blockSize;
    }
    
    //计算每一个点的坐标重新赋值
    moveVerts[0] = blockX;
    moveVerts[1] = blockY;
    
    moveVerts[3] = blockX + 2 * blockSize;
    moveVerts[4] = blockY;
    
    moveVerts[6] = blockX + 2 * blockSize;
    moveVerts[7] = blockY - 2 * blockSize;
    
    moveVerts[9] = blockX;
    moveVerts[10] = blockY - 2 * blockSize;
    
    moveBatch.CopyVertexData3f(moveVerts);
    
    glutPostRedisplay();
    
}

#pragma mark - main函数

int main (int argc,char *argv[])
{
    
    //设置工作区间
    //MacOS和Windows不一样,工作目录和项目目录可能不在同一个文件夹下
    //需要将工作目录设置到程序的/Resource目录下
    //GLUT的优先级设定已经做过这样的操作了,但是手动设置一下可以保证百分百
    gltSetWorkingDirectory(argv[0]);
    
    //初始化GLUT库
    //GLUT是一个程序内部的本地消息循环,会不断的拦截适当的消息
    //调用我们在不同时间注册的回调函数
    glutInit(&argc, argv);
    
    //初始化显示模式
    glutInitDisplayMode(GLUT_DOUBLE | GLUT_RGBA | GLUT_DEPTH | GLUT_STENCIL);
    
    //初始化窗口大小
    glutInitWindowSize(800, 600);
    
    //创建窗口并命名
    glutCreateWindow("颜色混合");
    
    //注册函数
    //视口大小发生改变时调用的函数
    glutReshapeFunc(ChangeSize);
    
    //渲染函数
    glutDisplayFunc(RenderScene);
    
    //注册特殊键位函数
    glutSpecialFunc(SpecialKeys);
    
    //初始化一个Glew库,确保OpenGL的API完全可用
    //在做任何的渲染尝试之前,都要确认驱动程序的初始化不能出现任何的问题
    GLenum status = glewInit();
    
    if (GLEW_OK != status) {
        printf("glew init error : %s \n",glewGetErrorString(status));
        return 1;
    }
    
    //设置渲染环境
    SetUpRC();
    
    //只是一个无限循环的函数
    //在运行的过程中会一直接收并执行窗口或者系统传来的用户指令
    //但是它只执行在它之前键入的指令,在它之后键入的指令则不执行
    glutMainLoop();
    
    return 0;
    
}


效果如下图3.1所示:

3.1.png
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,417评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,921评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,850评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,945评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,069评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,188评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,239评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,994评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,409评论 1 304
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,735评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,898评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,578评论 4 336
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,205评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,916评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,156评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,722评论 2 363
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,781评论 2 351

推荐阅读更多精彩内容