OpenGL Shader实现光照发光体特效
作者:JulyYu 发布时间:2022-03-16 18:51:50
标签:OpenGL,Shader,光照,发光
内发光原理
内发光原理简单概况是:采样周边像素alpha取平均值叠加效果。概括来说似乎好像特别简单,但需要一定的理解和消化。发光物体可以当做是一个圆形对象,去采集圆形对象周边像素值。例如已知圆形半径是R
,角度是Angle
,然后根据半径和角度推导算出当前像素坐标位置,用当前像素坐标位置得到透明度再去做计算。
但其实在阴影遮罩效果中似乎已经介绍过了同样能够通过。不同点在于阴影遮盖是利用圆形绘制向外部晕染而内发光效果是作用于内部。
发光体实现
首先采用绘制圆的方法实现RGB
叠加。可以看到中心位置绘制圆的位置颜色较深,向外扩散颜色逐渐暗淡。效果虽然不对但已经知道下一步该怎么实现了。
void main() {
vec2 uv = gl_FragCoord.xy / iResolution.xy;
uv -= 0.5;
uv.x *= iResolution.x/iResolution.y;
vec3 color = vec3(0.);
float glow = length(uv);
color += glow;
gl_FragColor = vec4(color,1.);
}
通过取反操作,可用一个数除以length(uv)
再相乘一个小数来稍微减小值的大小。从最终结果可以看到所期望的效果。对比之前效果展示相除相当于对原结果取反,原先内部是数值最小,相除之后内部数值变成最大。
void main() {
vec2 uv = gl_FragCoord.xy / iResolution.xy;
uv -= 0.5;
uv.x *= iResolution.x/iResolution.y;
vec3 color = vec3(0.);
float glow = 0.05 * 3./length(uv);
color += glow;
gl_FragColor = vec4(color,1.);
}
但过渡效果泛白范围似乎过大了一些继续对原算法进行优化。增加pow
方法将数值变得更小一些。
float getGlow(float dist, float radius, float intensity){
return pow(radius/dist, intensity);
}
void main() {
vec2 uv = gl_FragCoord.xy / iResolution.xy;
uv -= 0.5;
uv.x *= iResolution.x/iResolution.y;
vec3 color = vec3(0.);
float glow = 0.05 * getGlow(length(uv), 1., 2.);
color += glow;
gl_FragColor = vec4(color,1.);
}
扩展效果
小太阳
改变发光位置和发光颜色模拟实现太阳光照的效果。
float getGlow(float dist, float radius, float intensity){
return radius/dist;
}
void main() {
vec2 uv = gl_FragCoord.xy / iResolution.xy;
vec3 color = vec3(0.);
vec2 uv2 = uv;
uv2 -=1.0;
float glow = 0.09 * 3./length(uv2);
color += (5.0 * vec3(0.02 * glow) + vec3(0.9686, 0.6941, 0.0) * glow);
gl_FragColor = vec4(color,1.);
}
光源移动效果
float getGlow(float dist, float radius, float intensity){
return radius/dist;
}
void main() {
vec2 uv = gl_FragCoord.xy / iResolution.xy;
vec3 color = texture(iChannel1,uv).rgb;
float position = sin(iTime) / 2.;
vec2 uv2 = uv;
uv2 -=0.5;
uv2.x *= iResolution.x/iResolution.y;
uv2 += position;
float glow = 0.09 * 3./length(uv2);
color += (5.0 * vec3(0.02 * glow));
gl_FragColor = vec4(color,1.);
}
来源:https://juejin.cn/post/7063261134811299870


猜你喜欢
- 目录1. 效果图2. 思路3. 实现步骤3.1 数据Bean类3.2 创建适配器3.3 继承Filterable接口3.4 过滤调用4. 优
- Spring @RequestParam对象绑定在Spring中,如果在方法参数列表中使用@RequestParam标注多个参数,会让映射方
- Android实现界面内嵌多种卡片视图,具体内容如下效果如图所示:1.选择某个界面时,对应的第几个小圆点亮:通过selector制造圆点和进
- 概述日常工作中,我们经常会有发送 HTTP 网络请求的需求,概括下我们常见的发送 HTTP 请求的需求内容:可以发送基本的 GET/POST
- 一、概述今天给大家带来SurfaceView的一个实战案例,话说自定义View也是各种写,一直没有写过SurfaceView,这个玩意是什么
- 游戏服务器里面总是有一大堆的配置文件需要读取, 而且这些配置文件的读取: * 要不然做成弱类型的, 就是一堆字符串或者数字, 不能看出来错误
- 一、 搭建struts2环境在myeclipse下,右击项目->MyEclipse->Project Facets->in
- 表关联上一篇介绍了JPA的简单使用,这一篇介绍JPA在表关联上的使用一对一配置参数JPA对于数据实体一对一映射使用的是@OneToOne注解
- 本文实例讲述了Java实现的Base64加密算法。分享给大家供大家参考,具体如下:一 算法实现1、JDK2、Commonc Codec3、B
- 昨天写了一篇Redis布隆过滤器相关的命令的文章,今天来说一说springboot中如何简单在代码中使用布隆过滤器吧。目前市面上也有好几种实
- 一、场景Java实现文件上传到服务器本地,并通过url访问有个需求,前端上传文件,需要用开关的方式同时支持上传七牛和服务器本地,方便不同的用
- 为了避免直接进入项目中存在的页面,使用filter过滤器新建一个类loginFilter:package com.tjcu.filter;i
- 今天新建一个springboot项目时,项目建好后,在IDEA下载依赖包时,下载了很久都没有下载完,后来仔细一看,是下载不了。解决方法:在项
- 方法一:利用两个指针p,q,首先将q往链表尾部移动n位,然后再将p、q一起往后移,那么当q达到链表尾部时,p即指向链表的倒数第n个节点。no
- 服务限流,是指通过控制请求的速率或次数来达到保护服务的目的,在微服务中,我们通常会将它和熔断、降级搭配在一起使用,来避免瞬时的大量请求对系统
- 在项目迁移到Spring Boot之后,发生内存使用量过高的问题。本文介绍了整个排查过程以及使用到的工具,也非常适用于其他堆外内存排查。背景
- 前言:今天修改项目中一个有关WebView使用的bug,激起了我总结WebView的动机,今天抽空做个总结。简介WebView是一个基于we
- 主要介绍:1.任务队列2.拒绝策略(抛出异常、直接丢弃、阻塞、临时队列)3.init( min )4.active5.maxmin<=
- 废话不多说了,直接给大家贴代码了,具体代码如下<?xml version="1.0" encoding="
- 本文实例为大家分享了六种Android常见控件的使用方法,供大家参考,具体内容如下1、TextView 主要用于界面上显示一段文本