godot-camera-systems
Camera Systems
Expert guidance for creating smooth, responsive cameras in 2D and 3D games.
NEVER Do
- NEVER use
global_position = target.global_positionevery frame — Instant position matching causes jittery movement. Uselerp()orposition_smoothing_enabled = true[12]. - NEVER use
offsetfor permanent camera positioning —offsetis for shake, sway, or temporary recoil effects only. Usepositionfor permanent framing to avoid logic conflicts [14]. - NEVER forget
limit_smoothed = trueforCamera2D— Hard boundaries cause jarring visual stops. Smoothing against limits ensures a professional feel [13]. - NEVER enable multiple
Camera2Dnodes in the same viewport simultaneously — Only the last enabled camera takes precedence. Explicitly disable inactive cameras [15]. - NEVER use
SpringArm3Dwithout a collision mask — It will clip through terrain and walls. Set it to the world/environment layer [16]. - NEVER implement screen shake by randomizing
positiondirectly — This overwrites follow-logic. Useoffsetor a dedicated Trauma/Noise system to Layer shake over the follow-position [27, 28]. - NEVER parent the Camera directly to a high-speed physics body — Physics stutter or parent rotation will cause motion sickness. Use
RemoteTransform2D/3Dwith rotation sync disabled for a stable view [30]. - NEVER use
look_at()in 3D without a fallback for the 'Up' vector — If the target is directly above/below, the camera will flip wildly. Use guards orQuaternionmath for vertical tracking. - NEVER rely on
SubViewportdefaults for Mini-maps — Viewports are expensive; explicitly setrender_target_update_modetoUPDATE_WHEN_VISIBLEor a fixed lower framerate to save GPU [156]. - NEVER use linear interpolation for Zoom — It feels 'robotic'. Use exponential lerp or a
TweenwithTRANS_CUBICfor a more natural tactical feel.
Available Scripts
MANDATORY: Read before implementing camera behaviors.
camera_shake_trauma_pro.gd
Advanced noise-based screenshake (Trauma system) for organic, non-jittery explosions and impacts.
cinematic_framing_logic.gd
Rule of Thirds and Lead Room management in code, ensuring high-quality cinematic composition.
camera_state_machine.gd
Managing transitions between 'Follow', 'Static', and 'Cinematic' camera states with Tweens.
minimap_viewport_manager.gd
SubViewport optimization for Mini-maps and UI overlays. Reduces render updates for better FPS.
split_screen_setup.gd
Dynamic split-screen architecture for local multiplayer, handling viewport stretching and audio listeners.
remote_transform_decoupling.gd
Decoupling camera position from player rotation/scale using RemoteTransform2D for high-speed stability.
zoom_damping_controller.gd
Non-linear, smooth zoom logic with tactical overview bounds and mouse-wheel support.
spring_lerp_camera_3d.gd
Physics-stable 3D follow camera using spring-mass interpolation to reduce follow-latency jitter.
first_person_sway.gd
Procedural 8-figure head bob and weapon sway logic for immersive First-Person systems.
deadzone_drag_margins.gd
Platformer-specific deadzone management using code to control follow-margins and drag-center behavior.
Camera2D Basics
extends Camera2D
@export var target: Node2D
@export var follow_speed := 5.0
func _process(delta: float) -> void:
if target:
global_position = global_position.lerp(
target.global_position,
follow_speed * delta
)
Position Smoothing
extends Camera2D
func _ready() -> void:
# Built-in smoothing
position_smoothing_enabled = true
position_smoothing_speed = 5.0
Camera Limits
extends Camera2D
func _ready() -> void:
# Constrain camera to level bounds
limit_left = 0
limit_top = 0
limit_right = 1920
limit_bottom = 1080
# Smooth against limits
limit_smoothed = true
Camera Shake
extends Camera2D
var shake_amount := 0.0
var shake_decay := 5.0
func _process(delta: float) -> void:
if shake_amount > 0:
shake_amount = max(shake_amount - shake_decay * delta, 0)
offset = Vector2(
randf_range(-shake_amount, shake_amount),
randf_range(-shake_amount, shake_amount)
)
else:
offset = Vector2.ZERO
func shake(intensity: float) -> void:
shake_amount = intensity
# Usage:
$Camera2D.shake(10.0) # Screen shake on explosion
Zoom Controls
extends Camera2D
@export var zoom_speed := 0.1
@export var min_zoom := 0.5
@export var max_zoom := 2.0
func _unhandled_input(event: InputEvent) -> void:
if event is InputEventMouseButton:
if event.button_index == MOUSE_BUTTON_WHEEL_UP:
zoom_in()
elif event.button_index == MOUSE_BUTTON_WHEEL_DOWN:
zoom_out()
func zoom_in() -> void:
zoom = zoom.move_toward(
Vector2.ONE * max_zoom,
zoom_speed
)
func zoom_out() -> void:
zoom = zoom.move_toward(
Vector2.ONE * min_zoom,
zoom_speed
)
Look-Ahead Camera
extends Camera2D
@export var look_ahead_distance := 50.0
@export var target: CharacterBody2D
func _process(delta: float) -> void:
if target:
var look_ahead := target.velocity.normalized() * look_ahead_distance
global_position = target.global_position + look_ahead
Split-Screen (Multiple Cameras)
# Player 1 Camera
@onready var cam1: Camera2D = $Player1/Camera2D
# Player 2 Camera
@onready var cam2: Camera2D = $Player2/Camera2D
func _ready() -> void:
# Split viewport
cam1.anchor_mode = Camera2D.ANCHOR_MODE_DRAG_CENTER
cam2.anchor_mode = Camera2D.ANCHOR_MODE_DRAG_CENTER
Camera3D Patterns
Third-Person Camera
extends Camera3D
@export var target: Node3D
@export var distance := 5.0
@export var height := 2.0
@export var rotation_speed := 3.0
var rotation_angle := 0.0
func _process(delta: float) -> void:
if not target:
return
# Rotate around target
rotation_angle += Input.get_axis("camera_left", "camera_right") * rotation_speed * delta
# Calculate position
var offset := Vector3(
sin(rotation_angle) * distance,
height,
cos(rotation_angle) * distance
)
global_position = target.global_position + offset
look_at(target.global_position, Vector3.UP)
First-Person Camera
extends Camera3D
@export var mouse_sensitivity := 0.002
@export var max_pitch := deg_to_rad(80)
var pitch := 0.0
func _ready() -> void:
Input.mouse_mode = Input.MOUSE_MODE_CAPTURED
func _input(event: InputEvent) -> void:
if event is InputEventMouseMotion:
# Yaw (horizontal)
get_parent().rotate_y(-event.relative.x * mouse_sensitivity)
# Pitch (vertical)
pitch -= event.relative.y * mouse_sensitivity
pitch = clamp(pitch, -max_pitch, max_pitch)
rotation.x = pitch
Camera Transitions
# Smooth camera position change
func move_to_position(target_pos: Vector2, duration: float = 1.0) -> void:
var tween := create_tween()
tween.tween_property(self, "global_position", target_pos, duration)
tween.set_ease(Tween.EASE_IN_OUT)
tween.set_trans(Tween.TRANS_CUBIC)
Cinematic Cameras
# Camera path following
extends Path2D
@onready var path_follow: PathFollow2D = $PathFollow2D
@onready var camera: Camera2D = $PathFollow2D/Camera2D
func play_cutscene(duration: float) -> void:
var tween := create_tween()
tween.tween_property(path_follow, "progress_ratio", 1.0, duration)
await tween.finished
Best Practices
1. One Active Camera
# Only one Camera2D should be enabled at a time
# Others should have enabled = false
2. Parent Camera to Player
# Scene structure:
# Player (CharacterBody2D)
# └─ Camera2D
3. Use Anchors for UI
# Camera doesn't affect UI positioned with anchors
# UI stays in screen space
4. Deadzone for Platformers
extends Camera2D
func _ready() -> void:
drag_horizontal_enabled = true
drag_vertical_enabled = true
drag_left_margin = 0.3
drag_right_margin = 0.3
Reference
Related
- Master Skill: godot-master