mirror of
				https://github.com/JohnBreaux/Boat-Battle.git
				synced 2025-02-04 12:28:35 +00:00 
			
		
		
		
	Ship Battle: Draw the rest of the freaking owl
This commit is contained in:
		| @@ -10,6 +10,7 @@ enum  {MISS = -1, READY = 0, HIT = 1, SUNK = 2, LOST = 3} | ||||
| var bottom_board:Array # Player board | ||||
| var top_board:Array    # Opponent board | ||||
| var ships = []         # list of Ships | ||||
| var ship_data = []     # Data used to generate ships | ||||
| var ship_count = 0     # number of 'active' (un-sunk) ships | ||||
|  | ||||
| # a board is square. This is its side length | ||||
| @@ -33,50 +34,55 @@ func hit(pos): | ||||
| 	var res = MISS | ||||
| 	# Get the ship-metadata for that location | ||||
| 	var ship = bottom_board[pos.x][pos.y] | ||||
| 	# If there's a ship there, which exists, and hasn't been hit, | ||||
| 	if ship and ship[0] > NO_SHIP and ship[1] == READY: | ||||
| 		# Hit the ship, and store whether HIT or SUNK | ||||
| 	# If the ship's already been hit here, don't bother beating it again | ||||
| 	if ship[1] != READY: | ||||
| 		return ship[1] | ||||
| 	if ship[0] > NO_SHIP: | ||||
| 		# Decide whether HIT or SUNK | ||||
| 		res = ships[ship[0]].hit(pos) | ||||
| 		# TODO: display KABOOM | ||||
| 		# Update the ship | ||||
| 		ships[ship[0]].update() | ||||
| 		# Mark the ship as hit | ||||
| 		ship[1] = HIT | ||||
| 	else: | ||||
| 		# Mark that position as a miss, with no ship | ||||
| 		bottom_board[pos.x][pos.y] = [NO_SHIP, MISS] | ||||
| 	# If ship sunk, | ||||
| 	if res == SUNK: | ||||
| 		# remove it from the count | ||||
| 		ship_count -= 1 | ||||
| 	# If no ships left, | ||||
| 	# If we have no more ships left, we LOST | ||||
| 	if ship_count == 0: | ||||
| 		# Game has been lost | ||||
| 		res = LOST; | ||||
| 		res = LOST | ||||
| 	# Record the result on the board, and return it | ||||
| 	ship[1] = res | ||||
| 	return res | ||||
|  | ||||
| # fire: Store the results of firing on an opponent | ||||
| #   pos: board position fired on | ||||
| #   res: result of firing on the opponent | ||||
| func fire(pos, res): | ||||
| 	if top_board[pos.x][pos.y] == null: | ||||
| 	if top_board[pos.x][pos.y] == READY: | ||||
| 		top_board[pos.x][pos.y] = res | ||||
| 		return true | ||||
| 	return false | ||||
| 		return res | ||||
| 	else: | ||||
| 		return top_board[pos.x][pos.y] | ||||
|  | ||||
| # Place a ship on the board at board-space coordinates | ||||
| func place_ship(in_position, in_size, in_orientation, in_variant = 0): | ||||
| 	# Save the ship data | ||||
| 	ship_data.append([in_position, in_size, in_orientation]) | ||||
| 	# Create a new Ship, and give it some data | ||||
| 	var ship = Ship.instance() | ||||
| 	ship._init(in_position, in_size, in_orientation, in_variant) | ||||
| 	# Mark the ship on the board | ||||
| 	for pos in ship.get_extent(): | ||||
| 		bottom_board[pos.x][pos.y] = [ships.size(), READY] | ||||
| 	# Add the ship to the ships array, and keep count | ||||
| 	ships.append(ship) | ||||
| 	ship_count += 1 | ||||
| 	# Add the ship to the scene tree | ||||
| 	add_child(ship) | ||||
|  | ||||
| # Not sure why this is necessary yet | ||||
| func get_bottom_board(): | ||||
| 	return bottom_board | ||||
|  | ||||
| func query_bottom(pos): | ||||
| 	return bottom_board[pos.x][pos.y] | ||||
|  | ||||
| func query_top(pos): | ||||
| 	return top_board[pos.x][pos.y] | ||||
|  | ||||
| # Get the number of live ships | ||||
| func get_ship_count(): | ||||
| @@ -84,16 +90,16 @@ func get_ship_count(): | ||||
|  | ||||
| # _init: Constructor | ||||
| func _init(): | ||||
| 	# Initialize the bottom_board to a 10x10 array | ||||
| 	for _row in range(board_len): | ||||
| 	# Initialize the bottom_board to a len*len array | ||||
| 	for x in board_len: | ||||
| 		bottom_board.append([]) | ||||
| 	for column in bottom_board: | ||||
| 		column.resize(10) | ||||
| 	# Initialize the top_board to a 10x10 array | ||||
| 	for _row in range(board_len): | ||||
| 		for y in board_len: | ||||
| 			bottom_board[x].append([NO_SHIP, READY]) | ||||
| 	# Initialize the top_board to a len*len array | ||||
| 	for x in board_len: | ||||
| 		top_board.append([]) | ||||
| 	for column in top_board: | ||||
| 		column.resize(board_len) | ||||
| 		for y in board_len: | ||||
| 			top_board[x].append(READY) | ||||
|  | ||||
| #   worldspace_to_boardspace: convert a Vector2 in world-space to board-space | ||||
| func worldspace_to_boardspace(coordinate:Vector2): | ||||
|   | ||||
| @@ -1,19 +1,20 @@ | ||||
| extends Control | ||||
|  | ||||
| # Signal to pass the fire location back to parent | ||||
| signal fire_at | ||||
|  | ||||
| var atlas = preload("res://assets/game/HitMissAtlas.png") | ||||
| var sprites = [] | ||||
| var hits = [] | ||||
| var hits | ||||
|  | ||||
| # Called when the node enters the scene tree for the first time. | ||||
| func _ready(): | ||||
| 	print("Fire: _ready()") | ||||
| 	for x in 10: | ||||
| 		for y in 10: | ||||
| 			texture(Vector2(x,y)) | ||||
| 	pass # Replace with function body. | ||||
|  | ||||
| # Signal to pass the fire location back to yet-unknown nodes | ||||
| signal fire_at | ||||
|  | ||||
| func _init(topBoard): | ||||
| 	hits = topBoard | ||||
|  | ||||
| func _on_Fire_pressed(): | ||||
| 	var crosshair = get_node("Crosshair") | ||||
| 	# Check if the crosshair is in a valid position | ||||
| @@ -21,10 +22,7 @@ func _on_Fire_pressed(): | ||||
| 		var crosshair_pos = crosshair.world_to_board_space(crosshair.position) | ||||
| 		if(hits[crosshair_pos.x][crosshair_pos.y] == 0): | ||||
| 			# fires at position | ||||
| 			print("Fire at position: ", crosshair_pos) | ||||
| 			emit_signal("fire_at", crosshair_pos) | ||||
| 			# Close the Firing menu | ||||
| 			queue_free() | ||||
| 			return | ||||
| 	#if invalid position popup appears | ||||
| 	var dialog = get_node("FireDialog") | ||||
| @@ -32,7 +30,7 @@ func _on_Fire_pressed(): | ||||
|  | ||||
| func _on_FireDialog_confirmed(): | ||||
| 	get_node("Crosshair").visible = true | ||||
| 	 | ||||
|  | ||||
| const OFFSET = Vector2(18, 18) | ||||
|  | ||||
| func texture(index): | ||||
| @@ -52,7 +50,4 @@ func texture(index): | ||||
| 		var sprite = Sprite.new() | ||||
| 		sprite.texture = t | ||||
| 		sprite.position = Vector2(index.x, index.y) * textureSize + OFFSET | ||||
| 		 | ||||
| 		printt(t.get_height(), t.get_width()) | ||||
| 		 | ||||
| 		$board_blue.add_child(sprite) | ||||
|   | ||||
| @@ -1,11 +1,18 @@ | ||||
| extends Control | ||||
|  | ||||
| # warning-ignore-all:unused_signal | ||||
| # warning-ignore-all:return_value_discarded | ||||
|  | ||||
| enum  {MISS = -1, READY = 0, HIT = 1, SUNK = 2, LOST = 3} | ||||
|  | ||||
| # Signals | ||||
| signal fire # fire(position) | ||||
| signal hit  # hit (state: see Miss/Ready/Hit/Sunk enum in Board.gd) | ||||
| signal win  # win (): sent when opponent player lost | ||||
| signal hit  # hit (state): see Miss/Ready/Hit/Sunk enum in Board.gd) | ||||
| signal miss | ||||
| signal loss | ||||
| signal forfeit | ||||
|  | ||||
| signal game_ready | ||||
|  | ||||
| # Path to Player class, for instantiating new Players in code | ||||
| var Player = preload("res://scenes/Game/Player.tscn") | ||||
| @@ -16,10 +23,6 @@ var Victory = preload("res://scenes/Game/Victory.tscn") | ||||
| # Array of instances of the Player class; stores the Players | ||||
| var player | ||||
| var players_ready = [] | ||||
| # turn counter | ||||
| var turn = 0 | ||||
| # winner | ||||
| var winner = 0 | ||||
|  | ||||
| # Every game is a multiplayer game, even the ones that aren't. | ||||
| # We're taking the Minecraft approach, baby | ||||
| @@ -28,129 +31,159 @@ var network_id | ||||
| # Called when the node enters the scene tree for the first time. | ||||
| func _ready(): | ||||
|  | ||||
| 	get_node("ConfirmationDialog").get_ok().text = "Yes" | ||||
| 	get_node("ConfirmationDialog").get_cancel().text = "No" | ||||
| 	get_node("ConfirmationDialog").get_ok().rect_min_size.x = 100 | ||||
| 	get_node("ConfirmationDialog").get_cancel().rect_min_size.x = 100 | ||||
| 	if multiplayer: | ||||
| 		# TODO: Spawn a lobby where people can either connect to a peer or create a server | ||||
| 	get_node("Forfeit Confirmation").get_ok().text = "Yes" | ||||
| 	get_node("Forfeit Confirmation").get_cancel().text = "No" | ||||
| 	get_node("Forfeit Confirmation").get_ok().rect_min_size.x = 100 | ||||
| 	get_node("Forfeit Confirmation").get_cancel().rect_min_size.x = 100 | ||||
|  | ||||
| 	if Net.connected: | ||||
| 		Net.connect("disconnected", self, "connection_error") | ||||
| 		Net.connect("incoming",     self, "_on_Net_incoming") | ||||
| 		pass | ||||
| 	game_setup() | ||||
|  | ||||
| # Function used to keep track of which players are ready | ||||
| # TODO: Change this to keep track of ready states only | ||||
| func player_ready(): | ||||
| 	players_ready.append(Net.get_network_id()) | ||||
| 	pass | ||||
| func player_ready(sender): | ||||
| 	print("player_ready(%s), %d" % [sender, players_ready.size()]) | ||||
| 	players_ready.append(sender) | ||||
| 	if (players_ready.size() >= Net.peer_info.size()): | ||||
| 		emit_signal("game_ready") | ||||
|  | ||||
| # Member functions: | ||||
| #   game_start: starts the game | ||||
| #   game_setup: starts the game | ||||
| sync func game_setup(): | ||||
| 	# If there's no server connected, create one | ||||
| 	if not Net.connected: | ||||
| 		# TODO: Create a fake peer who we can automate, for single-player mode | ||||
| 		Net.start_host() | ||||
| 	network_id = Net.get_network_id() | ||||
| 	player = Player.instance() | ||||
| 	player.connect("player_ready", self, "_on_player_ready") | ||||
| 	add_child(player) | ||||
| 	player.set_up_begin() | ||||
| 	yield(self, "game_ready") | ||||
| 	if Net.hosting: | ||||
| 		state_fire() | ||||
|  | ||||
| #   state_fire: The firing state. Displays fire menu, then notifies opponent. | ||||
| remote func state_fire(): | ||||
| 	var pos = player.turn_start() | ||||
| 	if pos is GDScriptFunctionState: | ||||
| 		pos = yield(pos, "completed") | ||||
| 	rpc("state_check", pos) | ||||
|  | ||||
| #   state_check: The checking state. Branches out to the other states. | ||||
| #     pos: Position which the opponent is trying to fire upon | ||||
| remote func state_check(pos): | ||||
| 	var res = player.hit(pos) | ||||
| 	# Tell the opponent | ||||
| 	Net.send(0, ["hit", res], Net.REPLY) | ||||
| 	rpc("play_hit_sound", res) | ||||
| 	match res: | ||||
| 		LOST: | ||||
| 			# the other player wins | ||||
| 			rpc("state_win", player.board.ship_data) | ||||
| 			victory_screen(null, false) | ||||
| 		SUNK, HIT: | ||||
| 			# Hit | ||||
| 			rpc("state_fire") | ||||
| 		MISS: | ||||
| 			# Our turn to fire | ||||
| 			state_fire() | ||||
| 	pass | ||||
|  | ||||
| #   game_start: Runs on host. Controls the game. | ||||
| func game_start(): | ||||
| 	var state = "P1_fire" | ||||
| 	# Make sure we're the server | ||||
| 	while true: | ||||
| 		match state: | ||||
| 			"P1_fire": | ||||
| 				# Tell local player to fire | ||||
|  | ||||
| 				# Wait for result | ||||
|  | ||||
| 				# Send fire REQUEST to P2 | ||||
| 				pass | ||||
| 			"P2_check": | ||||
| 				# Wait for hit | ||||
| 				var ret = yield(self, "hit") | ||||
| 				# Record the hit | ||||
|  | ||||
| 				# | ||||
| 				pass | ||||
| 			"P2_fire": | ||||
| 				pass | ||||
| 			"P1_check": | ||||
| 				# Check if | ||||
| 				pass | ||||
| 			"P1_win": | ||||
| 				pass | ||||
| 			"P2_win": | ||||
| 				pass | ||||
| #   state_win: The winning state. If you reach here, someone's won. | ||||
| #     ships: The opponent's ship data, so that their board can be shown | ||||
| remote func state_win(ships): | ||||
| 	victory_screen(ships) | ||||
| 	pass | ||||
|  | ||||
| func fire_on(id, pos:Vector2): | ||||
| 	# REQUEST fire on opponent | ||||
| 	Net.send(id, ["fire", pos], Net.REQUEST) | ||||
| 	# Wait for REPLY | ||||
| #   play_hit_sound: Play a hit sound depending on the severity of the hit | ||||
| #     value: Lost/Sunk/Hit/Miss | ||||
| sync func play_hit_sound(value): | ||||
| 	match value: | ||||
| 		LOST, SUNK: | ||||
| 			AudioBus.emit_signal("ship_sunk") | ||||
| 		HIT: | ||||
| 			AudioBus.emit_signal("ship_hit") | ||||
| 		MISS: | ||||
| 			AudioBus.emit_signal("ship_missed") | ||||
|  | ||||
| func return_hit(id, ship_status): | ||||
| #   hit: Update the local player's board when the opponent fires | ||||
| #     pos: Opponent's target | ||||
| func hit(pos): | ||||
| 	pos = Vector2(pos[0], pos[1]) | ||||
| 	var res = player.hit(pos) | ||||
| 	return res | ||||
|  | ||||
| 	Net.send(id, ["hit", ship_status], Net.REPLY) | ||||
|  | ||||
| func _on_win(): | ||||
| 	pass | ||||
| #   mark: Update the local player's hit/miss board when opponent replies | ||||
| func mark(res): | ||||
| 	return player.mark(res) | ||||
|  | ||||
| #   _on_Net_incoming: Handle mail. | ||||
| func _on_Net_incoming(mail): | ||||
| 	print ("mail: ", mail, mail.size()) | ||||
| 	if mail.size() == 3: | ||||
| 		var sender   = mail[0] | ||||
| 		print ("mail: ", mail, mail.size()) | ||||
| 		var sender   = int(mail[0]) | ||||
| 		var message  = mail[1] | ||||
| 		var mailtype = mail[2] | ||||
| 		var mailtype = int(mail[2]) | ||||
| 		printt(sender, message, mailtype) | ||||
| 		match mailtype: | ||||
| 			# if message is a REQUEST (to perform an action) | ||||
| 			Net.REQUEST: | ||||
| 				match message[0]: | ||||
| 					# Opponent asks for player.fire() | ||||
| 					"fire": | ||||
| 						emit_signal("fire", message[1]) | ||||
| 					# Opponent asks for hit(pos) | ||||
| 					"hit": | ||||
| 						pass | ||||
| 					_: | ||||
| 						pass | ||||
| 			Net.REPLY: | ||||
| 				print ("got REPLY") | ||||
| 					# message is a REPLY (return value) | ||||
| 				match message[0]: | ||||
| 					# on "fire": fire(result) | ||||
| 					"fire": | ||||
| 						emit_signal("hit", message[1]) | ||||
| 					# Return value of | ||||
| 						hit(message[1]) | ||||
| 					# on "hit": mark(state) | ||||
| 					"hit": | ||||
| 						mark(message[1]) | ||||
| 					"forfeit": | ||||
| 						pass | ||||
| 				pass | ||||
| 			Net.READY: | ||||
| 				print ("got READY") | ||||
| 				# Add player to the ready array | ||||
| 				players_ready.append(sender) | ||||
| 				pass | ||||
| 				player_ready(sender) | ||||
| 			_: | ||||
| 				print ("got ", mailtype) | ||||
|  | ||||
| #   _on_player_ready: Player Ready signal handler | ||||
| func _on_player_ready(): | ||||
| 	print ("_on_player_ready") | ||||
| 	Net.send(1, [], Net.READY) | ||||
| 	Net.send(0, [], Net.READY) | ||||
| 	player_ready(Net.get_network_id()) | ||||
|  | ||||
| #   victory_screen: display the victory screen | ||||
| func victory_screen(): | ||||
| 	# TODO: Create the victory screen, fill it with knowledge | ||||
| 	pass | ||||
|  | ||||
| #   display_turn: display which turn it is on the screen | ||||
| func display_turn(): | ||||
| 	# TODO: Update the turn display, if there is one? | ||||
| 	pass | ||||
| func victory_screen(ships, winner = true): | ||||
| 	if winner: | ||||
| 		# Hide the buttons | ||||
| 		get_node("Bittons").hide() | ||||
| 		# Create a new Victory screen | ||||
| 		var victory = Victory.instance() | ||||
| 		# Give it the ships received from the opponent | ||||
| 		victory.reveal_ships(ships) | ||||
| 		# Print a nice message to stdout | ||||
| 		print("You won!") | ||||
| 		# Add victory to the scene tree | ||||
| 		add_child(victory) | ||||
| 	else: | ||||
| 		end() | ||||
|  | ||||
| #   _on_Forfeit_pressed: Handle forfeit button press | ||||
| func _on_Forfeit_pressed(): | ||||
| 	AudioBus.emit_signal("button_clicked") | ||||
| 	get_node("ConfirmationDialog").popup() | ||||
| 	get_node("Forfeit Confirmation").popup_centered() | ||||
|  | ||||
| #   end: end the Game | ||||
| func end(): | ||||
| sync func end(): | ||||
| 	queue_free() | ||||
|  | ||||
|  | ||||
| func connection_error(): | ||||
| 	get_node("Connection Error").popup_centered() | ||||
|  | ||||
| #   _on_Button_button_down: Handle win button press | ||||
| #   TODO: This isn't a thing any more | ||||
| func _on_Button_button_down(): | ||||
| @@ -159,6 +192,13 @@ func _on_Button_button_down(): | ||||
| 	add_child(victory) | ||||
| 	victory.connect("exit_main", self, "end") | ||||
|  | ||||
| func _on_ConfirmationDialog_confirmed(): | ||||
| func _on_Forfeit_Confirmation_confirmed(): | ||||
| 	if Net.connected: | ||||
| 		# Send forfeit request to all users | ||||
| 		rpc("end") | ||||
| 	end() | ||||
|  | ||||
| func _on_Connection_Error_confirmed(): | ||||
| 	# End the game | ||||
| 	queue_free() | ||||
|  | ||||
|   | ||||
| @@ -12,15 +12,13 @@ var Setup = preload("res://scenes/Game/Setup.tscn") | ||||
| var Fire  = preload("res://scenes/Game/Fire.tscn") | ||||
|  | ||||
| # Members | ||||
| var pid   # Player ID | ||||
| var board # Board | ||||
|  | ||||
| var fire_at_position # Position to fire at | ||||
| var fire_pos = Vector2(-1,-1) | ||||
| var target = Vector2(-1,-1) | ||||
|  | ||||
| # Called when the node enters the scene tree for the first time. | ||||
| func _ready(): | ||||
| 	# Set the player ID according to which network peer ID we are | ||||
| 	pid = int(name) | ||||
| 	pass | ||||
|  | ||||
| func set_up_begin(): | ||||
| 	var setup = Setup.instance() | ||||
| @@ -32,22 +30,23 @@ func set_up_begin(): | ||||
| #   hit: Called when opponent fires on us. | ||||
| #     Update internal state, and return hit/miss/sunk | ||||
| func hit(pos): | ||||
| 	target = pos | ||||
| 	var res = board.hit(pos) | ||||
| 	return res | ||||
|  | ||||
| #   mark: Called when the opponent returns hit/miss/sunk | ||||
| #     Update internal state, return ack/nak | ||||
| func mark(pos, value): | ||||
| func mark(value): | ||||
| 	# Mark the position on the top board | ||||
| 	board.fire(pos, value) | ||||
| 	return board.fire(fire_pos, value) | ||||
|  | ||||
| #   place_ship: called when ships are placed. | ||||
| #     forwards Ship locations to the Board, so that it may construct a ship | ||||
| #     ship: a list of ship properties {position, orientation, size, variant} | ||||
| func place_ship(pos, size, orientation, variant): | ||||
| 	board.place_ship(pos, size, orientation, variant) | ||||
| 	return board.place_ship(pos, size, orientation, variant) | ||||
|  | ||||
| #   setup: set up the board given the placed ship locations | ||||
| #   set_up: set up the board given the placed ship locations | ||||
| #     Places each ship onto the board | ||||
| #     ships: a list of lists of ship properties [[position, orientation, size, variant], ...] | ||||
| func set_up(ships): | ||||
| @@ -62,17 +61,22 @@ func set_up(ships): | ||||
| #     Initiates the player's turn, and blocks until the player selects a location to fire upon | ||||
| #     returns: fire = [player id, target coordinates] | ||||
| func turn_start(): | ||||
| 	print("turn_start") | ||||
| 	var fire = Fire.instance() | ||||
| 	 | ||||
| 	fire.hits = board.top_board | ||||
| 	add_child(fire) | ||||
| 	var pos = yield(fire, "fire_at") | ||||
| 	return pos | ||||
| 	fire_pos = yield(fire, "fire_at") | ||||
| 	fire.queue_free() | ||||
| 	return fire_pos | ||||
|  | ||||
|  | ||||
| #   getBoard: returns the player's board | ||||
| #     returns: board | ||||
| func getBoard(): | ||||
| 	return board | ||||
| func board_query(boardname): | ||||
| 	match boardname: | ||||
| 		"top": | ||||
| 			return board.query_top  (fire_pos) | ||||
| 		"bottom": | ||||
| 			return board.quert_bottom (target) | ||||
|  | ||||
| #   forfeit: ends game for player | ||||
| #     Sinks all ships | ||||
| @@ -86,7 +90,3 @@ func forfeit(): | ||||
| #   getShipCount: get the number of ships the player has left alive | ||||
| func getShipCount(): | ||||
| 	return board.get_ship_count() | ||||
|  | ||||
|  | ||||
| func _on_fire_at(pos): | ||||
| 	fire_at_position = pos | ||||
|   | ||||
| @@ -59,6 +59,8 @@ func hit(pos): | ||||
| 		# Sink the ship. | ||||
| 		set_sunk() | ||||
| 		res = SUNK | ||||
| 	# Update graphics | ||||
| 	update() | ||||
| 	return res | ||||
|  | ||||
| # update: (re)calculates extents and textures | ||||
|   | ||||
		Reference in New Issue
	
	Block a user