Shadertoy_入门基础

官网:https://www.shadertoy.com/
Shadertoy_入门基础

一、基本变量

uniform vec3      iResolution;   	 // 窗口分辨率,单位像素
uniform float     iTime;        	 // 程序运行的时间,单位秒
uniform float     iTimeDelta;   	 // 渲染时间,单位秒
uniform int       iFrame;       	 // 帧率
uniform float     iChannelTime[4];       // 信道播放时间(秒)
uniform vec3      iChannelResolution[4]; // 通道分辨率(像素)
uniform vec4      iMouse;        	 // 鼠标位置
uniform samplerXX iChannel0..3;          // 输入通道  XX = 2D/Cube
uniform vec4      iDate;          // 日期(年,月,日,时)
uniform float     iSampleRate;    // 声音采样率 (i.e., 44100)
void mainImage(){}				  //main函数

示例:

void mainImage( out vec4 fragColor, in vec2 fragCoord )
{
    // 将像素位置映射到0-1
    vec2 uv = fragCoord.xy/iResolution.xy;
    
    // 获取通道0纹理在uv出的像素颜色
    fragColor = texture(iChannel0, uv);
    
    // 让红色分量的值随时间改变。
    fragColor.r = abs(sin(iTime));
}

首先我们来具体解释下:
这是ShaderToy的主函数,参数一个入,一个出。输入的像素坐标向量,输出的是像素的颜色向量。主要作用就是根据屏幕上的像素坐标,算出像素的颜色向量,简单来说完成像素坐标到颜色的变换或者是映射。屏幕分辨率是800乘600的话,就计算800乘600区域内的所有像素。ShaderToy当前窗口的每个像素坐标都要经过这个主函数的处理以决定其颜色,所以看似这个主程序是一段代码,其实逻辑上被嵌在了一个像素坐标的大循环里面:
uv这里做了归一化处理, uv.x, uv.y的取值都在0~1;

当我们在通道0中未定义纹理的时候,我们可以你看到:
Shadertoy_入门基础
当我们在通道0中添加纹理素材后,我们可以看到:
Shadertoy_入门基础

二、圆绘制

2.1 1/4圆绘制

void mainImage( out vec4 fragColor, in vec2 fragCoord )
{
	// 将像素位置映射到0-1
    vec2 uv = fragCoord/iResolution.xy;
    
    float d = length(uv);
    float c = d;
    
    fragColor = vec4(vec3(c), 1.0);
}

length函数是求向量的模,即是 sqrt(( u v . x ) 2 + ( u v . y ) 2) ,其实这是空间符号距离函数SDF的雏形,这个公式可以理解为向量到原点 向量 vec3(0,0,0) 的距离;还有向量里各分量的值类型一般都是float类型,所以在赋值时要加小数点。

vec3©:只给出一个值c,表示这个向量的x , y , z都是c,即是 vec3(c,c,c)

fragColor = vec4(vec3( c ), 1.0); ,这一行返回的是一个4维向量,由 r g b a四个元素组成。对基本的颜色要有一点直观感知,例如vec4(1.,1.,1.,1.)是白色;vec4(0.,0.,0.,1.)是黑色。
Shadertoy_入门基础
从图中可以看出:屏幕像素坐标原点 vec2(0,0) 映射出来的颜色是黑色。向右向上逐渐变淡,1/4椭圆外是白色,屏幕的颜色时白色和黑色,以及这两者之间的过渡色,这是由返回的向量rgb三个元素取值一样决定的。

2.2 基点平移

我们可以通过改变uv的取值范围,将圆点移动到屏幕中心。

void mainImage( out vec4 fragColor, in vec2 fragCoord )
{
	// 将像素位置映射到0-1
    vec2 uv = fragCoord/iResolution.xy;
    // 基点平移
    uv-=0.5;
    float d = length(uv);
    float c = d;
    
    fragColor = vec4(vec3(c), 1.0);
}

Shadertoy_入门基础

2.3 锐化成圆

我们通过上边平移uv的原点放到了中心,现在我们可以根据uv坐标与原点的具体具体把屏幕的颜色分成黑白两色:

void mainImage( out vec4 fragColor, in vec2 fragCoord )
{
	// 将像素位置映射到0-1
    vec2 uv = fragCoord/iResolution.xy;
    // 基点平移
    uv-=0.5;
    float d = length(uv);
    float c = d;
    //锐化
    if(d < .3) c=1.; else c = 0.;
    
    fragColor = vec4(vec3(c), 1.0);
}

Shadertoy_入门基础
从图可以看到,因为图片非正方形所以我们绘制的是一个椭圆,下边我们调整x向比例,使得椭圆成圆:

void mainImage( out vec4 fragColor, in vec2 fragCoord )
{
	// 将像素位置映射到0-1
    vec2 uv = fragCoord/iResolution.xy;
    // 基点平移
    uv-=0.5;
    //调整x方向比例
    uv.x *= iResolution.x/iResolution.y;
    float d = length(uv);
    float c = d;
    //锐化
    if(d < .3) c=1.; else c = 0.;
    
    fragColor = vec4(vec3(c), 1.0);
}

Shadertoy_入门基础

2.4 边缘模糊

从上图我们可以看圆边缘有细微锯齿,不平滑,需要引入平滑函数smoothstep: 理解float d = length(uv); ,这个可以理解为uv向量的长度,可以认为是到原点的距离,在当前的情况下,原点就是白圆的圆心,这个距离函数d就是离圆心的距离。所以,我们可以定义一个变量r,用来表示半径:

void mainImage( out vec4 fragColor, in vec2 fragCoord )
{
	// 将像素位置映射到0-1
    vec2 uv = fragCoord/iResolution.xy;
    // 基点平移
    uv-=0.5;
    //调整x方向比例
    uv.x *= iResolution.x/iResolution.y;
    float d = length(uv);
    float c = d;
    //模糊
    float r = .3;
    c = smoothstep(r, r-0.02, d);
    
    fragColor = vec4(vec3(c), 1.0);
}

Shadertoy_入门基础

圆方法封装:

//圆绘制
float Circle(vec2 uv, vec2 p, float r, float blur) {
    float d = length(uv-p);
    float c = smoothstep(r, r-blur, d); 
    return c;
}

void mainImage( out vec4 fragColor, in vec2 fragCoord )
{
	// 将像素位置映射到0-1
    vec2 uv = fragCoord/iResolution.xy;
    // 基点平移
    uv-=0.5;
    //调整x方向比例
    uv.x *= iResolution.x/iResolution.y;
 
    vec2 p = vec2(0., 0.);
    float c = Circle(uv, p, .4, .05);
    
    fragColor = vec4(vec3(c), 1.0);
}

三、矩形绘制

我们可以主要用smoothstep对uv两个方向做分界处理形成矩形区域:

void mainImage( out vec4 fragColor, in vec2 fragCoord )
{
	// 将像素位置映射到0-1
    vec2 uv = fragCoord/iResolution.xy;
    // 基点平移
    uv-=0.5;
    //调整x方向比例
    uv.x *= iResolution.x/iResolution.y;
 
    vec3 color=vec3(0.);
    
    float mask=smoothstep(-.2,.2,uv.x);
    
    color=vec3(1.,1.,1.)*mask;
    
    fragColor = vec4(color, 1.0);
}

运行可以看到,我们从-0.2到0.2为过滤带,分两侧颜色;
Shadertoy_入门基础
接下来,我们可以将此部分封装,其中可用参数为uv变量、起点位置、终点位置、模糊大小:

float Band(float t,float start,float end,float blur)
{
    float stepL =smoothstep(start-blur,start+blur,t);
    float stepR =smoothstep(end+blur,end-blur,t);
    return stepL*stepR;
}

void mainImage( out vec4 fragColor, in vec2 fragCoord )
{
	// 将像素位置映射到0-1
    vec2 uv = fragCoord/iResolution.xy;
    // 基点平移
    uv-=0.5;
    //调整x方向比例
    uv.x *= iResolution.x/iResolution.y;
 
    vec3 color=vec3(0.);
    
    float mask=Band(uv.x,-.2,.2,.01);
    color=vec3(1.,1.,1.)*mask;
    
    fragColor = vec4(color, 1.0);
}

运行可见如下效果:
Shadertoy_入门基础
在接下来,我们对uv的v方向进行处理,将其封装为一个正方形方法:

float Band(float t,float start,float end,float blur)
{
    float step1 =smoothstep(start-blur,start+blur,t);
    float step2 =smoothstep(end+blur,end-blur,t);
    return step1*step2;
}

float Rect(vec2 uv,float left,float right,float bottom,float top,float blur)
{
    float band1=Band(uv.x,left,right,blur);
    float band2=Band(uv.y,bottom,top,blur);
    return band1*band2;
}

void mainImage( out vec4 fragColor, in vec2 fragCoord )
{
	// 将像素位置映射到0-1
    vec2 uv = fragCoord/iResolution.xy;
    // 基点平移
    uv-=0.5;
    //调整x方向比例
    uv.x *= iResolution.x/iResolution.y;
 
    vec3 color=vec3(0.);
    
    float mask=Rect(uv,-.2,.2,-.2,.2,.01);
    color=vec3(1.,1.,1.)*mask;
    
    fragColor = vec4(color, 1.0);
}


Shadertoy_入门基础
你也可以尝试使用不同的参数达成多边形效果:
Shadertoy_入门基础
Shadertoy_入门基础

四、波浪线绘制

4.1 基础绘制

首先,我们需要绘制一个长方形,接着上一部分的内容调整参数如下(你也可以通过对UV的比例进行调整改变):
Shadertoy_入门基础

在完成波浪之前,我们首先来看一下需要使用的方程式如下:
Shadertoy_入门基础
接着,我们在代码中体现:

float Band(float t,float start,float end,float blur)
{
    float step1 =smoothstep(start-blur,start+blur,t);
    float step2 =smoothstep(end+blur,end-blur,t);
    return step1*step2;
}

float Rect(vec2 uv,float left,float right,float bottom,float top,float blur)
{
    float band1=Band(uv.x,left,right,blur);
    float band2=Band(uv.y,bottom,top,blur);
    return band1*band2;
}

void mainImage( out vec4 fragColor, in vec2 fragCoord )
{
	// 将像素位置映射到0-1
    vec2 uv = fragCoord/iResolution.xy;
    // 基点平移
    uv-=0.5;
    //调整x方向比例
    uv.x *= iResolution.x/iResolution.y;
 
    vec3 color=vec3(0.);
    float x=uv.x;
    float m=-(x-.5)*(x+.5);
    float y=uv.y+m;

    float mask=Rect(vec2(x,y),-.5,.5,-.05,.05,.01);
    color=vec3(1.,1.,1.)*mask;
    
    fragColor = vec4(color, 1.0);
}

运行后可见如下:
Shadertoy_入门基础
接下来我们可以使用更多的函数来表达:
Shadertoy_入门基础
在代码中体现如下:

float Band(float t,float start,float end,float blur)
{
    float step1 =smoothstep(start-blur,start+blur,t);
    float step2 =smoothstep(end+blur,end-blur,t);
    return step1*step2;
}

float Rect(vec2 uv,float left,float right,float bottom,float top,float blur)
{
    float band1=Band(uv.x,left,right,blur);
    float band2=Band(uv.y,bottom,top,blur);
    return band1*band2;
}

void mainImage( out vec4 fragColor, in vec2 fragCoord )
{
	// 将像素位置映射到0-1
    vec2 uv = fragCoord/iResolution.xy;
    // 基点平移
    uv-=0.5;
    //调整x方向比例
    uv.x *= iResolution.x/iResolution.y;
 
    vec3 color=vec3(0.);
    float x=uv.x;
    float m=-(x-.5)*(x+.5);
    m=m*m*4.;
    float y=uv.y-m;

    float mask=Rect(vec2(x,y),-.5,.5,-.05,.05,.01);
    color=vec3(1.,1.,1.)*mask;
    
    fragColor = vec4(color, 1.0);
}


运行可见:
Shadertoy_入门基础
我们也可以使用正弦函数对长方形生效:

float Band(float t,float start,float end,float blur)
{
    float step1 =smoothstep(start-blur,start+blur,t);
    float step2 =smoothstep(end+blur,end-blur,t);
    return step1*step2;
}

float Rect(vec2 uv,float left,float right,float bottom,float top,float blur)
{
    float band1=Band(uv.x,left,right,blur);
    float band2=Band(uv.y,bottom,top,blur);
    return band1*band2;
}

void mainImage( out vec4 fragColor, in vec2 fragCoord )
{
	// 将像素位置映射到0-1
    vec2 uv = fragCoord/iResolution.xy;
    // 基点平移
    uv-=0.5;
    //调整x方向比例
    uv.x *= iResolution.x/iResolution.y;
 
    vec3 color=vec3(0.);
    float x=uv.x;
    float m=sin(x*8.)*.2;
    float y=uv.y-m;

    float mask=Rect(vec2(x,y),-.5,.5,-.05,.05,.01);
    color=vec3(1.,1.,1.)*mask;
    
    fragColor = vec4(color, 1.0);
}

Shadertoy_入门基础

4.2 动态绘制

接下来我们在正弦函数上使用一个时间变量达到动态效果:

float Band(float t,float start,float end,float blur)
{
    float step1 =smoothstep(start-blur,start+blur,t);
    float step2 =smoothstep(end+blur,end-blur,t);
    return step1*step2;
}

float Rect(vec2 uv,float left,float right,float bottom,float top,float blur)
{
    float band1=Band(uv.x,left,right,blur);
    float band2=Band(uv.y,bottom,top,blur);
    return band1*band2;
}

void mainImage( out vec4 fragColor, in vec2 fragCoord )
{
	// 将像素位置映射到0-1
    vec2 uv = fragCoord/iResolution.xy;
    // 基点平移
    uv-=0.5;
    //调整x方向比例
    uv.x *= iResolution.x/iResolution.y;
 
    float t=iTime;
    
    vec3 color=vec3(0.);
    float x=uv.x;
    float m=sin(t+x*8.)*.2;
    float y=uv.y-m;

    float mask=Rect(vec2(x,y),-.5,.5,-.05,.05,.01);
    color=vec3(1.,1.,1.)*mask;
    
    fragColor = vec4(color, 1.0);
}

Shadertoy_入门基础

上一篇:cocos游戏循环与坐标


下一篇:如何在Shader中绘制类似虚线的折线