要用Godot实现屏幕抖动效果,调查了网上的一些实现方案,效果都很不满意,于是自己实现了一个。
从渲染过程看,实际发生抖动的是场景相机。
抖动过程对应物理学上的振动(物理量在某个定值附近反复变化),也称为震荡。
一个物体振动时,其振幅逐渐减小最后停止振动,这种振荡运动是阻尼的。
相机抖动符合阻尼振动的定义。
要实现抖动效果,理论上既可以是相机在平衡位置(坐标)附近的振动,也可以是相机朝向(方向角)的振动。
Godot的Camera3D类有h_offset和v_offset属性,用它们来实现振动是最方便的。
根据物理学中阻尼运动的动力学方程(振动方程),可以解得质点的运动方程:
另外,过阻尼振动和临界阻尼振动的情况我们不需要,不作讨论。
虽然运动方程是一维的,但是很容易扩展到多维。
只要多维的系统参数相同,符合物理规律,最终表现效果就会比较“真实”。
给定初始化条件(t=0, x=0)和(t=duration, x=0)可以进一步简化得到相机的运动方程,即相机坐标随时间变化的函数。
场景相机的附加脚本:
extends Camera3D
const amp = 1.5
const cycle = 3
const duration = 0.3
const beta = 3 / duration
const omega = 2 * PI / duration * cycle
var a: Vector2
func rnd_amp(amp_max: float):
return randf_range(amp_max / 2, amp_max) * [1, -1][randi() % 2]
func _on_vibra_finished():
h_offset = 0
v_offset = 0
func damped_vibra(t: float):
h_offset = a.x * exp(-beta * t) * sin(omega * t)
v_offset = a.y * exp(-beta * t) * sin(omega * t)
func _on_Main_sig_bad():
a = Vector2(rnd_amp(amp), rnd_amp(amp))
var tween = create_tween()
tween.connect("finished", _on_vibra_finished)
tween.tween_method(damped_vibra, 0.0, duration, duration)
以上代码适配Godot 4.1.3版本