Godot里的像素化后处理
2024-03-05
通过一个星期的折腾,我总算通过Godot的shader, 实现了下面这个看起来还不错的效果:
渲染前的原图
我编写了这样的一个后处理shader, 将它附加在全屏Quad里,即可看到效果。我定义了用于设置像素大小的uniform. 值得注意的是,还没有实现Pixel perfect, 这会导致相机移动的过程中会产生奇怪的锯齿抽搐,看久了让人头晕的那种。
以下是代码(第一版)
// 这是像素相机的shader, 贴在相机下的Quad里即可使用。
// 目前还没实现Pixel Perfect, 相机在移动时会产生奇怪的锯齿
shader_type spatial;
render_mode unshaded;
uniform sampler2D SCREEN_TEXTURE: source_color, hint_screen_texture, filter_nearest;
uniform sampler2D DEPTH_TEXTURE : source_color, hint_depth_texture, filter_nearest;
uniform sampler2D NORMAL_TEXTURE: source_color, hint_normal_roughness_texture,filter_nearest;
uniform vec2 render_offset = vec2(0.0, 0.0);
uniform vec2 PIXEL_SIZE = vec2(4,4);
uniform float border_threshold:hint_range(0.0, 0.05, 0.0001) = 0.01;
uniform float inner_edge_threshold:hint_range(0.0, 1.0, 0.1) = 0.4;
varying vec2 screen_uv;
varying vec2 screen_size;
#define PER_PIXEL_UV (PIXEL_SIZE/screen_size) // 每个像素占据的UV
// Robert 算子
const mat3 Robert_Gx = mat3(vec3(0, 0, 0),vec3(0, 0, -1),vec3(0, 1, 0));
const mat3 Robert_Gy = mat3(vec3(0, 0, 0),vec3(0, -1, 0),vec3(0, 0, 1));
// 裁剪矩阵,将mat4裁剪为mat3.保留左上角的矩阵。
mat3 mat4_to_mat3(mat4 raw_mat){
return mat3(raw_mat[0].xyz, raw_mat[1].xyz, raw_mat[2].xyz);
}
vec2 get_pixelized_uv(){
return (floor((screen_uv * screen_size + render_offset) / PIXEL_SIZE)+vec2(0.5)) * PIXEL_SIZE / screen_size;
}
vec2 get_nearby_pixelized_uv(ivec2 offset){
return (floor((screen_uv * screen_size + render_offset) / PIXEL_SIZE)+vec2(0.5)+vec2(offset)) * PIXEL_SIZE / screen_size;
}
float get_depth(){ // 获取线性深度
float depth = texture(DEPTH_TEXTURE, get_pixelized_uv()).x;
return depth;
}
// 获取周围点的深度
float sample_nearby_depth(ivec2 offset) {
float depth = texture(DEPTH_TEXTURE, get_pixelized_uv() + vec2(offset)*PIXEL_SIZE/screen_size).x;
return depth;
}
// 获取当前点周围8个点内,离相机最近的点的坐标
vec2 get_nearest_pixel_uv(){
ivec2 local_closest_uv = ivec2(0,0);
for(int i=-1; i<=1; i++)
for(int j=-1; j<=1; j++)
if (sample_nearby_depth(ivec2(i, j)) < sample_nearby_depth(local_closest_uv))
local_closest_uv = ivec2(i,j);
return get_nearby_pixelized_uv(local_closest_uv);
}
// Handamard, 对应元素的乘积
mat3 handamard(mat3 mat_a, mat3 mat_b){
mat3 mat_r;
for (int i=0; i<=2; i++)
for(int j=0; j<=2; j++)
mat_r[i][j] = mat_a[i][j] * mat_b[i][j];
return mat_r;
}
// Frobenius inner product. 获取对应元素积的和
float frobenius(mat3 matA, mat3 matB){
mat3 matR = handamard(matA, matB);
float r = 0.0;
for (int i=0; i<=2; i++)
for(int j=0; j<=2; j++)
r += matR[i][j];
return r;
}
// 是外边界
bool is_outer_edge(float threshold){
mat3 nearby_pixels = mat3(0.0); // 当前像素以及周围8个像素
for(int i=-1; i<=1; i++) //-1,0,1
for(int j=-1; j<=1; j++){
// 获取该像素的临近像素. mat[1][1]代表中央点。
nearby_pixels[i+1][j+1] = texture(DEPTH_TEXTURE, get_pixelized_uv() + vec2(
PER_PIXEL_UV.x*(float(i)),
PER_PIXEL_UV.y*(float(j))
)).x;
}
// 计算梯度判断边缘。如果是边缘,那么对这个边缘四周进行采样,找到离相机最近的点的UV,返回那个边缘的材质
float gradient = sqrt(pow(frobenius(Robert_Gx, nearby_pixels),2) + pow(frobenius(Robert_Gy, nearby_pixels),2));
if ((gradient > threshold))
return true;
else
return false;
}
bool is_inner_edge(float threshold){
mat3 nearby_normal_r = mat3(0.0);
mat3 nearby_normal_g = mat3(0.0);
mat3 nearby_normal_b = mat3(0.0);
for(int i=-1; i<=1; i++)
for(int j=-1; j<=1; j++){
vec3 nearby_normal = texture(NORMAL_TEXTURE, get_pixelized_uv() + vec2(
PER_PIXEL_UV.x*(float(i)),
PER_PIXEL_UV.y*(float(j))
)).rgb;
// 分别对R、G、B三个通道进行处理
nearby_normal_r[i+1][j+1] = nearby_normal.r;
nearby_normal_g[i+1][j+1] = nearby_normal.g;
nearby_normal_b[i+1][j+1] = nearby_normal.b;
}
// 计算梯度
float gradient_r = sqrt(pow(frobenius(Robert_Gx, nearby_normal_r),2) + pow(frobenius(Robert_Gy, nearby_normal_r),2));
float gradient_g = sqrt(pow(frobenius(Robert_Gx, nearby_normal_g),2) + pow(frobenius(Robert_Gy, nearby_normal_g),2));
float gradient_b = sqrt(pow(frobenius(Robert_Gx, nearby_normal_b),2) + pow(frobenius(Robert_Gy, nearby_normal_b),2));
// 判断是否为边缘
if ((gradient_r > threshold) || (gradient_g > threshold) || (gradient_b > threshold))
return true;
else
return false;
}
void vertex() {
POSITION = vec4(VERTEX, 1.0);}
void fragment() {
screen_uv = SCREEN_UV;
screen_size = VIEWPORT_SIZE; // Varying
if (is_outer_edge(border_threshold))
ALBEDO = texture(SCREEN_TEXTURE, get_nearest_pixel_uv()).rgb*0.3;
else if (is_inner_edge(inner_edge_threshold))
ALBEDO = clamp(texture(SCREEN_TEXTURE, get_pixelized_uv()).rgb*1.4,vec3(0),vec3(1));
else // 非边缘
ALBEDO = texture(SCREEN_TEXTURE, get_pixelized_uv()).rgb;
}
我还没打算解释具体做了什么。也许后面我会考虑自己做个教程视频?
这篇博客只是我自己写着玩的,还没打算正式作为教程发布。如果我的网站不幸被搜索引擎收录,同时你又不幸(不是)看到了它,对这个效果感兴趣的话,不妨去看看原视频吧(如果可以的话,留个评论吧,让我知道这个网站居然会有我以外的人涉足)
https://www.youtube.com/watch?v=WBoApONC7bM
我的实现的代码相当丑陋,也许后期会进行优化,可有的学了。