OpenGL ES——打光

前言

在物理世界中,光是视觉的根基。在计算机的世界中,亦是如此。我们看到一个物体,除了它的形态外,还能感知到它的色彩。这个色彩就是由物体的颜色、材质和外部的光照共同决定的。

基础概念

光照vs材质vs颜色

想象一下,任意一个物体,比如一个乒乓球。他们材质相同,在同样的太阳光下,黄色和白色的乒乓球,我们一定能够分辨出,这就是颜色的差别。同样的乒乓球,在阳光和酒吧中彩灯下,样貌也必定大不相同,这是光照的差别。此外,同样的白色的球,乒乓球和棒球也完全不同,这是材质的差别。

specular VS diffuse VS ambient

那么,我们在定义一种材质时,该如何定义它的光学特性呢?

OpenGL ES为我们提供的思路是:将材料对光的反应分为三部分。

specular

光线直接照射到物体反射到观察者眼中,这束光被称作镜面反射光。我们称之为specular。通常我们像这样定义它:

float[] specular = {1.0f, 1.0f, 1.0f, 1.0f};
float[] materialSpec = {1.0f, 0.5f, 0.0f, 1.0f,};

以上是镜面光和镜面反射的颜色。

diffuse

除了镜面反射外,我们对物体的大部分视觉认知,来源于漫反射。材料对于光进行漫反射后的颜色,我们称之为diffuse。通常我们像这样定义:

float[] diffuse = {0.5f, 0.5f, 0.5f, 1.0f};
float[] materialDiff = {0.0f, 0.0f, 1.0f, 1.0f,};

以上是漫反射光和材质漫反射的颜色。

ambient

除了两种反射外,我们还有一部分完全照不到光的地方,它只受环境光的影响。这块地方的颜色,我们称之为ambient。通常我们像这样定义:

float[] ambient = {0.9f, 0.9f, 0.9f, 1.0f};
float[] materialAmb = {0.4f, 0.4f, 1.0f, 1.0f,};

以上是环境光和环境光反射的颜色。

位置

光从不同角度打向物体,会产生不同的打光效果。因此,我们也会有控制光源位置的需求。

我们通常这样控制打光的位置:

float[] lightPosition = {0.5f, 0.5f, 0.5f, 0.0f};
gl.glLightfv(GL10.GL_LIGHT0, GL10.GL_POSITION, STLUtils.floatToBuffer(lightPosition));

如何使用

我们在上一节的代码的基础上,进行修改。上一节中,我们从STL文件中导入了3D模型,但因为没有光源,我们看到的都是纯色。

今天在上面的例子中,onSurfaceCreated周期内,加入启动光源和设置材质的方法即可:

    @Override
    public void onSurfaceCreated(GL10 gl, EGLConfig config) {
        gl.glEnable(GL10.GL_DEPTH_TEST); // 启用深度缓存
        gl.glClearColor(0f, 0f, 0f, 0f);// 设置深度缓存值
        gl.glDepthFunc(GL10.GL_LEQUAL); // 设置深度缓存比较函数
        gl.glShadeModel(GL10.GL_SMOOTH);// 设置阴影模式GL_SMOOTH

        //开启光
        openLight(gl);
        enableMaterial(gl);
        float r = model.getR();

        //r是半径,不是直径,因此用0.5/r可以算出放缩比例
        mScalef = 0.5f / r;
        mCenterPoint = model.getCentrePoint();
    }

    float[] ambient = {0.9f, 0.9f, 0.9f, 1.0f};
    float[] diffuse = {0.5f, 0.5f, 0.5f, 1.0f};
    float[] specular = {1.0f, 1.0f, 1.0f, 1.0f};
    float[] lightPosition = {0.5f, 0.5f, 0.5f, 0.0f};

    private void openLight(GL10 gl) {

        gl.glEnable(GL10.GL_LIGHTING);
        gl.glEnable(GL10.GL_LIGHT0);
        gl.glLightfv(GL10.GL_LIGHT0, GL10.GL_AMBIENT, STLUtils.floatToBuffer(ambient));
        gl.glLightfv(GL10.GL_LIGHT0, GL10.GL_DIFFUSE, STLUtils.floatToBuffer(diffuse));
        gl.glLightfv(GL10.GL_LIGHT0, GL10.GL_SPECULAR, STLUtils.floatToBuffer(specular));
        gl.glLightfv(GL10.GL_LIGHT0, GL10.GL_POSITION, STLUtils.floatToBuffer(lightPosition));


    }

    float[] materialAmb = {0.4f, 0.4f, 1.0f, 1.0f,};
    float[] materialDiff = {0.0f, 0.0f, 1.0f, 1.0f,};
    float[] materialSpec = {1.0f, 0.5f, 0.0f, 1.0f,};

    public void enableMaterial(GL10 gl) {

        //材料对环境光的反射情况
        gl.glMaterialfv(GL10.GL_FRONT_AND_BACK, GL10.GL_AMBIENT, STLUtils.floatToBuffer(materialAmb));
        //散射光的反射情况
        gl.glMaterialfv(GL10.GL_FRONT_AND_BACK, GL10.GL_DIFFUSE, STLUtils.floatToBuffer(materialDiff));
        //镜面光的反射情况
        gl.glMaterialfv(GL10.GL_FRONT_AND_BACK, GL10.GL_SPECULAR, STLUtils.floatToBuffer(materialSpec));

    }

以上就是OpenGL ES中关于光照的基本操作。

如有问题,欢迎指正。

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏opengps

新手入门百度地图开发的(0,0)坐标问题

        对于大部分人来讲,由于百度地图资料众多,过度依赖搜索引擎等等原因。新接触百度地图开发工作其实并不容易。今天说说关于坐标(0,0)的问题。 ...

3749
来自专栏小樱的经验随笔

元胞自动机实现多数分类算法

元胞自动机(Cellular automaton) 元胞自动机是由元胞组成的网格,每个元胞都根据邻域的状态来选择开或关。所有的元胞都遵循同样的规则,也称为元胞的...

3185
来自专栏智慧教育

比起WE大会“救命的AI”,这个AI已经悄悄进入人们的学习中

“未来人工智能要进一步发展的话,就需从脑科学得到启发,包括机器学习过程,怎么从脑启发的这个概念来设计新的计算模式,新的类似人脑的神经元结构的器件、芯片,甚至是机...

2164
来自专栏生信技能树

【生信菜鸟经】如何系统入门R语言

写在前面 R语言不仅在生物信息数据处理中发挥着重要作用,也是其它主流数据处理人士的首选工具。现在非常多自学生物信息学的小伙伴必须学的就是R,所以写一个R的系统性...

1K9
来自专栏架构师之路

洗稿,技术上怎么判断文章相似性?

这几天“差评洗稿”的事情,闹得沸沸扬扬,楼主本身也是内容创作的手艺人,简单说两句。

1494
来自专栏企鹅号快讯

数据分析:寻找Python最优计算性能

1、场景描述 在数据统计分析过程中,求累计值(总和)是最常用的统计指标之一,市面上的各种流行数据库均支持的查询方式基本如下: select sum(c) fro...

2137
来自专栏程序员互动联盟

【编程基础第七讲】如何编写有界面的程序?

存在问题: 好多小伙伴都有一个疑问,我我们学编程貌似都是看输出,怎么才能搞个像window上程序带个又界面的,眼见为实嘛 解决方案: 很多群友都在问学习了C语言...

3847
来自专栏Crossin的编程教室

【每周一坑】双色球选号器

说到做到,每周一坑 没有跳票,今天又来了! 我反复说,学会语法只是编程的开始,不断地写代码才是提升技能的必经之路。所以这个栏目我们肯定是要做下去的,哪怕它的阅读...

3135
来自专栏程序员叨叨叨

6.4 移位操作符

Cg语言中的移位操作符,功能和C语言中的一样,也可以作用在向量上,但是向量类型必须是int类型。例如:

992
来自专栏数据派THU

一文盘点三大顶级Python库(附代码)

Python在许多方面有着强大的吸引力 - 例如效率、代码可读性和速度方面,也正因为如此,对于希望提升应用程序功能的数据科学家和机器学习专家来说,Python通...

1484

扫码关注云+社区