Godot屏幕抖动效果的原理与实现

要用Godot实现屏幕抖动效果,调查了网上的一些实现方案,效果都很不满意,于是自己实现了一个。

原理

从渲染过程看,实际发生抖动的是场景相机。
抖动过程对应物理学上的振动(物理量在某个定值附近反复变化),也称为震荡。
一个物体振动时,其振幅逐渐减小最后停止振动,这种振荡运动是阻尼的。
相机抖动符合阻尼振动的定义。

要实现抖动效果,理论上既可以是相机在平衡位置(坐标)附近的振动,也可以是相机朝向(方向角)的振动。
Godot的Camera3D类有h_offsetv_offset属性,用它们来实现振动是最方便的。

根据物理学中阻尼运动的动力学方程(振动方程),可以解得质点的运动方程:
阻尼振动的运动方程
另外,过阻尼振动和临界阻尼振动的情况我们不需要,不作讨论。
虽然运动方程是一维的,但是很容易扩展到多维。
只要多维的系统参数相同,符合物理规律,最终表现效果就会比较“真实”。
给定初始化条件(t=0, x=0)和(t=duration, x=0)可以进一步简化得到相机的运动方程,即相机坐标随时间变化的函数。

GDScript 代码

场景相机的附加脚本:

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版本

说明

  • 将外部信号sig_bad连接到场景相机脚本中的函数_on_Main_sig_bad();
  • 函数_on_Main_sig_bad()初始化振动过程参数,并触发tween的补间(多次插值)过程;
  • 系统会通过Tween机制每隔一段时间触发函数damped_vibra(),函数采用阻尼振动的运动方程来更新相机参数h_offsetv_offset,实现振动效果;
  • 当补间结束时,finished信号触发函数_on_vibra_finished(),重置相机参数h_offsetv_offset

参考

知乎:阻尼运动、参变共振
百度百科:阻尼振动
Godot: Tween

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值