2024-03-14 04:17:00 +00:00
|
|
|
extends Node2D
|
|
|
|
class_name GolfLevel
|
|
|
|
|
2024-03-15 18:57:15 +00:00
|
|
|
|
|
|
|
## Emitted when it's time to change to the next level.
|
|
|
|
signal level_completed
|
|
|
|
|
|
|
|
|
2024-03-14 04:17:00 +00:00
|
|
|
@export_category("Level Data")
|
|
|
|
|
|
|
|
## Whether the [field camera] follows the active player or not.
|
|
|
|
@export var camera_follows_player: bool = false
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@export_category("References")
|
|
|
|
|
|
|
|
## The [Camera2D] of this level.
|
|
|
|
@export var camera: Camera2D = null
|
|
|
|
|
|
|
|
## The [GolfTee] of this level.
|
|
|
|
@export var tee: GolfTee = null
|
|
|
|
|
|
|
|
## The [GolfHole] of this level.
|
|
|
|
@export var hole: GolfHole = null
|
|
|
|
|
|
|
|
## The [TileMap] of this level.
|
|
|
|
@export var map: TileMap = null
|
|
|
|
|
|
|
|
|
2024-03-15 18:57:15 +00:00
|
|
|
## If on server, this variable indicates the [GolfLevel] to replicate.
|
2024-03-14 04:17:00 +00:00
|
|
|
##
|
|
|
|
## The [GolfLevel] in question should not be added to the scene tree, or it will cause problems on the client acting as server.
|
|
|
|
##
|
|
|
|
## On clients, this variable is ignored.
|
|
|
|
var target: GolfLevel = null
|
|
|
|
|
|
|
|
|
2024-03-15 18:57:15 +00:00
|
|
|
## The [PlayerNodeDirectory] to use to determine which players to spawn.
|
2024-03-14 04:17:00 +00:00
|
|
|
##
|
|
|
|
## It should be set on instantiation of a new [GolfLevel].
|
|
|
|
var player_dir: PlayerNodeDirectory = null
|
|
|
|
|
|
|
|
|
2024-03-15 18:57:15 +00:00
|
|
|
## The [PackedScene] to initialize as [TileMap].
|
2024-03-14 04:17:00 +00:00
|
|
|
const tilemap_scene: PackedScene = preload("res://scenes/golf_tilemap.tscn")
|
|
|
|
|
2024-03-15 18:57:15 +00:00
|
|
|
## The [PackedScene] to initialize as [GolfTee].
|
2024-03-14 04:17:00 +00:00
|
|
|
const tee_scene: PackedScene = preload("res://scenes/golf_tee.tscn")
|
|
|
|
|
2024-03-15 18:57:15 +00:00
|
|
|
## The [PackedScene] to initialize as [GolfHole].
|
2024-03-14 04:17:00 +00:00
|
|
|
const hole_scene: PackedScene = preload("res://scenes/golf_hole.tscn")
|
|
|
|
|
2024-03-15 18:57:15 +00:00
|
|
|
## The [PackedScene] to initialize as [Camera2D].
|
2024-03-14 04:17:00 +00:00
|
|
|
const camera_scene: PackedScene = preload("res://scenes/follow_camera.tscn")
|
|
|
|
|
|
|
|
|
|
|
|
## Replicate the [field target] from the server to the clients via RPC.
|
|
|
|
func build() -> void:
|
|
|
|
build_tilemap()
|
|
|
|
build_tilemap_cells()
|
|
|
|
build_tee()
|
2024-03-15 18:57:15 +00:00
|
|
|
build_balls()
|
2024-03-14 04:17:00 +00:00
|
|
|
build_hole()
|
|
|
|
build_camera()
|
|
|
|
|
|
|
|
|
|
|
|
## Replicate the [field map] of the [field target] to the remote [field map].
|
|
|
|
func build_tilemap() -> void:
|
|
|
|
Log.peer(self, "Replicating map...")
|
|
|
|
var tmap: TileMap = target.map
|
|
|
|
rpc_build_tilemap.rpc(tmap.global_position, tmap.global_rotation, tmap.global_scale)
|
|
|
|
|
2024-03-15 18:57:15 +00:00
|
|
|
## Create the [field map].
|
|
|
|
@rpc("authority", "call_local", "reliable")
|
2024-03-14 04:17:00 +00:00
|
|
|
func rpc_build_tilemap(tposition: Vector2, trotation: float, tscale: Vector2):
|
|
|
|
Log.peer(self, "Building map...")
|
|
|
|
map = tilemap_scene.instantiate()
|
|
|
|
map.global_position = tposition
|
|
|
|
map.global_rotation = trotation
|
|
|
|
map.global_scale = tscale
|
|
|
|
add_child(map)
|
|
|
|
|
|
|
|
## Replicate the cells of [field target]'s [field map] to the remote [field map].
|
|
|
|
##
|
2024-03-15 18:57:15 +00:00
|
|
|
## The [field map] must be already created.
|
2024-03-14 04:17:00 +00:00
|
|
|
##
|
|
|
|
## Only takes layer 0 into consideration.
|
|
|
|
func build_tilemap_cells() -> void:
|
|
|
|
Log.peer(self, "Replicating map cells...")
|
|
|
|
var tmap: TileMap = target.map
|
|
|
|
const layer = 0
|
|
|
|
for coords in tmap.get_used_cells(layer):
|
|
|
|
var source_id: int = tmap.get_cell_source_id(0, coords)
|
|
|
|
var atlas_coords: Vector2i = tmap.get_cell_atlas_coords(0, coords)
|
|
|
|
var alternative_tile: int = tmap.get_cell_alternative_tile(0, coords)
|
|
|
|
rpc_build_tilemap_cell.rpc(layer, coords, source_id, atlas_coords, alternative_tile)
|
|
|
|
|
2024-03-15 18:57:15 +00:00
|
|
|
## Create a cell of [field map].
|
|
|
|
@rpc("authority", "call_local", "reliable")
|
2024-03-14 04:17:00 +00:00
|
|
|
func rpc_build_tilemap_cell(
|
|
|
|
layer: int,
|
|
|
|
coords: Vector2i,
|
|
|
|
source_id: int,
|
|
|
|
atlas_coords: Vector2i = Vector2i(-1, -1),
|
|
|
|
alternative_tile: int = 0
|
|
|
|
):
|
|
|
|
Log.peer(self, "Building map cell: %s" % coords)
|
|
|
|
map.set_cell(layer, coords, source_id, atlas_coords, alternative_tile)
|
|
|
|
|
|
|
|
|
|
|
|
## Replicate the [field tee] of the [field target] to the remote [field tee].
|
|
|
|
func build_tee() -> void:
|
|
|
|
Log.peer(self, "Replicating tee...")
|
|
|
|
var ttee: GolfTee = target.tee
|
|
|
|
rpc_build_tee.rpc(ttee.global_position)
|
|
|
|
|
2024-03-15 18:57:15 +00:00
|
|
|
## Create the [GolfTee] object.
|
|
|
|
@rpc("authority", "call_local", "reliable")
|
2024-03-14 04:17:00 +00:00
|
|
|
func rpc_build_tee(tposition: Vector2):
|
|
|
|
Log.peer(self, "Building tee...")
|
|
|
|
tee = tee_scene.instantiate()
|
|
|
|
tee.global_position = tposition
|
2024-03-15 18:57:15 +00:00
|
|
|
tee.everyone_entered_hole.connect(_on_everyone_entered_hole)
|
2024-03-14 04:17:00 +00:00
|
|
|
add_child(tee)
|
|
|
|
|
2024-03-15 18:57:15 +00:00
|
|
|
|
|
|
|
## Replicate the currently connected players' [GolfBall]s to the remote [field tee].
|
|
|
|
func build_balls() -> void:
|
2024-03-14 04:17:00 +00:00
|
|
|
for playernode in player_dir.get_children():
|
2024-03-15 18:57:15 +00:00
|
|
|
rpc_build_ball.rpc(playernode.player_name)
|
|
|
|
|
2024-03-14 04:17:00 +00:00
|
|
|
|
|
|
|
## Create and initialize a [GolfBall] for a single [PlayerNode] with the given name.
|
2024-03-15 18:57:15 +00:00
|
|
|
@rpc("authority", "call_local", "reliable")
|
|
|
|
func rpc_build_ball(player_name: String):
|
2024-03-14 04:17:00 +00:00
|
|
|
Log.peer(self, "Building tee ball for: %s" % player_name)
|
|
|
|
var playernode: PlayerNode = player_dir.get_playernode(player_name)
|
2024-03-15 18:57:15 +00:00
|
|
|
tee.spawn(playernode)
|
2024-03-14 04:17:00 +00:00
|
|
|
|
|
|
|
|
|
|
|
## Replicate the [field hole] of the [field target] to the remote [field hole].
|
|
|
|
func build_hole() -> void:
|
|
|
|
Log.peer(self, "Replicating hole...")
|
|
|
|
var thole: GolfHole = target.hole
|
|
|
|
rpc_build_hole.rpc(thole.global_position, thole.global_scale)
|
|
|
|
|
2024-03-15 18:57:15 +00:00
|
|
|
## Create the [GolfHole] object.
|
|
|
|
@rpc("authority", "call_local", "reliable")
|
2024-03-14 04:17:00 +00:00
|
|
|
func rpc_build_hole(tposition: Vector2, tscale: Vector2):
|
|
|
|
Log.peer(self, "Building hole...")
|
|
|
|
hole = hole_scene.instantiate()
|
|
|
|
hole.global_position = tposition
|
|
|
|
hole.global_scale = tscale
|
|
|
|
add_child(hole)
|
|
|
|
|
|
|
|
|
|
|
|
## Replicate the [field camera] of the [field target] to the remote [field camera].
|
|
|
|
func build_camera() -> void:
|
|
|
|
Log.peer(self, "Replicating camera...")
|
|
|
|
var tcamera: FollowCamera = target.camera
|
|
|
|
rpc_build_camera.rpc(tcamera.global_position, target.camera_follows_player)
|
|
|
|
|
2024-03-15 18:57:15 +00:00
|
|
|
## Create the [Camera2D] object.
|
|
|
|
@rpc("authority", "call_local", "reliable")
|
2024-03-14 04:17:00 +00:00
|
|
|
func rpc_build_camera(tposition: Vector2, tfocus: bool):
|
2024-03-15 18:57:15 +00:00
|
|
|
if multiplayer.is_server():
|
|
|
|
Log.peer(self, "Not building camera on the server.")
|
|
|
|
return
|
2024-03-14 04:17:00 +00:00
|
|
|
Log.peer(self, "Building camera...")
|
|
|
|
camera = camera_scene.instantiate()
|
|
|
|
camera.global_position = tposition
|
|
|
|
if tfocus:
|
2024-03-15 18:57:15 +00:00
|
|
|
# This supports only a single player per peer, for now.
|
|
|
|
for child in player_dir.get_children():
|
|
|
|
var playernode = child as PlayerNode
|
|
|
|
if playernode.is_multiplayer_authority():
|
|
|
|
var ctarget = tee.get_node("GolfBall__%s" % child.player_name)
|
|
|
|
camera.target = ctarget
|
2024-03-14 04:17:00 +00:00
|
|
|
add_child(camera)
|
|
|
|
|
|
|
|
|
2024-03-15 18:57:15 +00:00
|
|
|
func _ready() -> void:
|
|
|
|
player_dir.child_entered_tree.connect(_on_playernode_created)
|
|
|
|
if multiplayer.is_server():
|
|
|
|
set_physics_process(false)
|
|
|
|
hide()
|
|
|
|
|
|
|
|
|
|
|
|
func _on_playernode_created(node: Node) -> void:
|
2024-03-14 04:17:00 +00:00
|
|
|
Log.peer(self, "Spawning new player...")
|
|
|
|
var playernode: PlayerNode = node as PlayerNode
|
2024-03-15 18:57:15 +00:00
|
|
|
rpc_build_ball.rpc(playernode.player_name)
|
|
|
|
|
|
|
|
|
|
|
|
func _on_everyone_entered_hole() -> void:
|
|
|
|
level_completed.emit()
|
|
|
|
|
|
|
|
|
|
|
|
func _on_tree_exiting() -> void:
|
|
|
|
player_dir.child_entered_tree.disconnect(_on_playernode_created)
|
|
|
|
if target:
|
|
|
|
target.queue_free()
|
|
|
|
target = null
|