mirror of
				https://github.com/JHDev2006/Super-Mario-Bros.-Remastered-Public.git
				synced 2025-10-26 09:20:49 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			150 lines
		
	
	
		
			5.5 KiB
		
	
	
	
		
			GDScript
		
	
	
	
	
	
			
		
		
	
	
			150 lines
		
	
	
		
			5.5 KiB
		
	
	
	
		
			GDScript
		
	
	
	
	
	
| class_name CameraHandler
 | |
| extends Node2D
 | |
| 
 | |
| @onready var last_position = global_position
 | |
| @export var camera: Camera2D = null
 | |
| 
 | |
| @export var camera_center_joint: Node2D = null
 | |
| 
 | |
| var camera_position := Vector2.ZERO
 | |
| var camera_offset := Vector2(8, 0)
 | |
| 
 | |
| var camera_right_limit := 9999999
 | |
| 
 | |
| var player_offset := 0.0
 | |
| 
 | |
| var can_scroll_left := true
 | |
| var can_scroll_right := true
 | |
| 
 | |
| static var cam_locked := false
 | |
| 
 | |
| var scrolling := false
 | |
| var cam_direction := 1
 | |
| 
 | |
| # how far between the center and the edge of the screen before scrolling to the center
 | |
| const SCROLL_DIFFERENCE := 48.0
 | |
| 
 | |
| func _exit_tree() -> void:
 | |
| 	cam_locked = false
 | |
| 
 | |
| func _physics_process(delta: float) -> void:
 | |
| 	handle_camera(delta)
 | |
| 	last_position = global_position
 | |
| 
 | |
| func handle_camera(delta: float) -> void:
 | |
| 	
 | |
| 	can_scroll_left = camera_position.x + camera_offset.x > -255
 | |
| 	can_scroll_right = camera_position.x + camera_offset.x < camera_right_limit - 1
 | |
| 	
 | |
| 	if ["Pipe", "Climb", "FlagPole"].has(owner.state_machine.state.name):
 | |
| 		handle_vertical_scrolling(delta)
 | |
| 		do_limits()
 | |
| 		camera.global_position = camera_position + camera_offset
 | |
| 		return
 | |
| 	
 | |
| 	if not cam_locked:
 | |
| 		handle_horizontal_scrolling(delta)
 | |
| 		handle_vertical_scrolling(delta)
 | |
| 		handle_offsets(delta)
 | |
| 	
 | |
| 	do_limits()
 | |
| 	camera.global_position = camera_position + camera_offset
 | |
| 	update_camera_barriers()
 | |
| 
 | |
| func update_camera_barriers() -> void:
 | |
| 	if get_viewport() != null:
 | |
| 		camera_center_joint.global_position = get_viewport().get_camera_2d().get_screen_center_position()
 | |
| 		camera_center_joint.get_node("LeftWall").position.x = -(get_viewport_rect().size.x / 2)
 | |
| 		camera_center_joint.get_node("RightWall").position.x = (get_viewport_rect().size.x / 2)
 | |
| 
 | |
| func handle_horizontal_scrolling(delta: float) -> void:
 | |
| 	scrolling = false
 | |
| 	var true_velocity = (global_position - last_position) / delta
 | |
| 	var true_vel_dir = sign(true_velocity.x)
 | |
| 	if (owner.is_on_wall() and owner.direction == -owner.get_wall_normal().x):
 | |
| 		true_vel_dir = 0
 | |
| 		true_velocity.x = 0
 | |
| 	## RIGHT MOVEMENT
 | |
| 	if true_vel_dir == 1 and can_scroll_right:
 | |
| 		cam_direction = 1
 | |
| 		if global_position.x >= camera_position.x:
 | |
| 			var offset = 0
 | |
| 			if camera_position.x <= global_position.x - 4:
 | |
| 				offset = camera_position.x - global_position.x + abs(true_velocity.x * delta)
 | |
| 			scrolling = true
 | |
| 			camera_position.x = global_position.x + offset
 | |
| 	
 | |
| 	## LEFT MOVEMENT
 | |
| 	elif true_vel_dir == -1 and can_scroll_left and Global.current_level.can_backscroll:
 | |
| 		cam_direction = -1
 | |
| 		if global_position.x <= camera_position.x:
 | |
| 			scrolling = true
 | |
| 			var offset = 0
 | |
| 			if camera_position.x >= global_position.x + 4:
 | |
| 				offset = camera_position.x - global_position.x - abs(true_velocity.x * delta)
 | |
| 			camera_position.x = global_position.x + offset
 | |
| 	
 | |
| 	# horizontal adjusgments
 | |
| 	# if the position is matching the camera, start scrolling towards the center
 | |
| 	if global_position.x >= camera_position.x:
 | |
| 		position.x = move_toward(position.x,0.0,delta * max(30.0, abs(true_velocity.x / 2.3)))
 | |
| 	else:
 | |
| 		# if the camera if behind the middle of the screen, calculate the current difference
 | |
| 		position.x = max(min(camera_position.x-global_position.x,SCROLL_DIFFERENCE),position.x)
 | |
| 
 | |
| 
 | |
| func handle_vertical_scrolling(_delta: float) -> void:
 | |
| 	## VERTICAL MOVEMENT
 | |
| 	if global_position.y < camera_position.y and owner.is_on_floor():
 | |
| 		camera_position.y = move_toward(camera_position.y, global_position.y, 3)
 | |
| 	elif global_position.y < camera_position.y - 64:
 | |
| 		camera_position.y = global_position.y + 64
 | |
| 	elif global_position.y > camera_position.y + 32:
 | |
| 		camera_position.y = global_position.y - 32
 | |
| 
 | |
| func tween_ahead() -> void:
 | |
| 	if scrolling == false: return
 | |
| 	await get_tree().create_timer(0.25).timeout
 | |
| 	var tween = create_tween()
 | |
| 	tween.tween_property(self, "camera_position:x", camera_position.x + (32 * cam_direction), 0.25)
 | |
| 
 | |
| func recenter_camera() -> void:
 | |
| 	camera_position = global_position
 | |
| 	last_position = camera_position
 | |
| 	camera_position += camera_offset
 | |
| 	do_limits()
 | |
| 	camera.global_position = camera_position
 | |
| 
 | |
| func handle_offsets(delta: float) -> void:
 | |
| 	var true_velocity = (global_position - last_position) / delta
 | |
| 	var true_vel_dir = sign(true_velocity.x)
 | |
| 	if owner.velocity.x == 0 or (owner.is_on_wall() and owner.direction == -owner.get_wall_normal().x):
 | |
| 		true_vel_dir = 0
 | |
| 		true_velocity.x = 0
 | |
| 	if Global.current_level.can_backscroll:
 | |
| 		if true_vel_dir != 0 and abs(true_velocity.x) > 80:
 | |
| 			if abs(camera_position.x - global_position.x) <= 64:
 | |
| 				camera_offset.x = move_toward(camera_offset.x, 8 * true_vel_dir, abs(true_velocity.x) / 200)
 | |
| 	else:
 | |
| 		camera_offset.x = 8
 | |
| 
 | |
| func do_limits() -> void:
 | |
| 	camera_right_limit = clamp(Player.camera_right_limit, -256 + (get_viewport().get_visible_rect().size.x), INF)
 | |
| 	camera_position.x = clamp(camera_position.x, point_to_camera_limit(-256 - camera_offset.x, -1), point_to_camera_limit(camera_right_limit - camera_offset.x, 1))
 | |
| 	camera_position.y = clamp(camera_position.y, point_to_camera_limit_y(Global.current_level.vertical_height, -1), point_to_camera_limit_y(32, 1))
 | |
| 	var wall_enabled := true
 | |
| 	if is_instance_valid(Global.level_editor):
 | |
| 		if Global.level_editor.playing_level == false:
 | |
| 			wall_enabled = false
 | |
| 	$"../CameraCenterJoint/LeftWall".set_collision_layer_value(1, wall_enabled)
 | |
| 	var level_exit = false
 | |
| 	if owner.state_machine != null:
 | |
| 		level_exit = owner.state_machine.state.name == "LevelExit"
 | |
| 	$"../CameraCenterJoint/RightWall".set_collision_layer_value(1, wall_enabled and level_exit == false)
 | |
| 	
 | |
| func point_to_camera_limit(point := 0, point_dir := -1) -> float:
 | |
| 	return point + ((get_viewport_rect().size.x / 2.0) * -point_dir)
 | |
| 
 | |
| func point_to_camera_limit_y(point := 0, point_dir := -1) -> float:
 | |
| 	return point + ((get_viewport_rect().size.y / 2.0) * -point_dir)
 | 
