2024-03-10 14:05:44 +01:00
|
|
|
|
from ShootResult import ShootResult
|
2024-04-21 19:07:27 +02:00
|
|
|
|
from ShipDirection import ShipDirection
|
|
|
|
|
from ShipMode import ShipMode
|
2024-03-05 05:39:03 +01:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class ShipField:
|
|
|
|
|
|
|
|
|
|
def __init__(self):
|
2024-03-10 14:05:44 +01:00
|
|
|
|
self.field = [' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ',
|
|
|
|
|
' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ',
|
|
|
|
|
' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ',
|
|
|
|
|
' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ',
|
|
|
|
|
' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ',
|
|
|
|
|
' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ',
|
|
|
|
|
' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ',
|
|
|
|
|
' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ',
|
|
|
|
|
' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ',
|
|
|
|
|
' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ']
|
|
|
|
|
|
2024-03-17 19:34:18 +01:00
|
|
|
|
self.ships = [4, 3, 3, 2, 2, 2, 1, 1, 1, 1]
|
2024-03-10 14:05:44 +01:00
|
|
|
|
self.field_size = 10
|
2024-04-21 19:07:27 +02:00
|
|
|
|
self.field_mode = ShipMode.PUT
|
2024-03-10 14:05:44 +01:00
|
|
|
|
self.ship_size = 4
|
2024-04-21 19:07:27 +02:00
|
|
|
|
self.ship_direction = ShipDirection.VERTICAL
|
2024-03-10 14:05:44 +01:00
|
|
|
|
|
|
|
|
|
def __getitem__(self, item):
|
|
|
|
|
if item is None:
|
|
|
|
|
return None
|
|
|
|
|
|
|
|
|
|
if type(item) is not int and item.isnumeric():
|
|
|
|
|
item = int(item)
|
|
|
|
|
|
|
|
|
|
if type(item) is int and 0 <= item < len(self.field):
|
|
|
|
|
return self.field[item]
|
|
|
|
|
|
|
|
|
|
return None
|
|
|
|
|
|
|
|
|
|
def action(self, row, col):
|
|
|
|
|
self.clear_marker()
|
|
|
|
|
|
2024-04-21 19:07:27 +02:00
|
|
|
|
if self.field_mode == ShipMode.PUT:
|
|
|
|
|
if self.check_ship(row, col):
|
|
|
|
|
self.get_ship(row, col)
|
|
|
|
|
|
|
|
|
|
elif self.ship_size in self.ships and self.check_possible(row, col):
|
2024-03-10 14:05:44 +01:00
|
|
|
|
self.set_ship(row, col)
|
|
|
|
|
|
2024-04-21 19:07:27 +02:00
|
|
|
|
elif self.field_mode == ShipMode.SHOOT:
|
2024-03-10 14:05:44 +01:00
|
|
|
|
self.shoot(row, col)
|
|
|
|
|
|
|
|
|
|
def target(self, row, col):
|
|
|
|
|
self.clear_marker()
|
|
|
|
|
|
2024-04-21 19:07:27 +02:00
|
|
|
|
if self.field_mode == ShipMode.PUT:
|
2024-03-10 14:05:44 +01:00
|
|
|
|
if self.check_possible(row, col):
|
2024-04-21 19:07:27 +02:00
|
|
|
|
if self.ship_direction == ShipDirection.VERTICAL:
|
2024-03-10 14:05:44 +01:00
|
|
|
|
for r in range(row, row + self.ship_size):
|
2024-03-17 19:34:18 +01:00
|
|
|
|
if self.ship_size in self.ships:
|
|
|
|
|
self.field[r * self.field_size + col] = "p"
|
|
|
|
|
else:
|
|
|
|
|
self.field[r * self.field_size + col] = "r"
|
2024-03-10 14:05:44 +01:00
|
|
|
|
|
2024-04-21 19:07:27 +02:00
|
|
|
|
if self.ship_direction == ShipDirection.HORIZONTAL:
|
2024-03-10 14:05:44 +01:00
|
|
|
|
for c in range(col, col + self.ship_size):
|
2024-03-17 19:34:18 +01:00
|
|
|
|
if self.ship_size in self.ships:
|
|
|
|
|
self.field[row * self.field_size + c] = "p"
|
|
|
|
|
else:
|
2024-04-21 19:07:27 +02:00
|
|
|
|
self.field[row * self.field_size + c] = "r"
|
2024-03-10 14:05:44 +01:00
|
|
|
|
else:
|
|
|
|
|
self.field[row * self.field_size + col] += "+"
|
|
|
|
|
|
|
|
|
|
def clear_marker(self):
|
|
|
|
|
for i in range(0, len(self.field)):
|
2024-03-17 19:34:18 +01:00
|
|
|
|
if self.field[i] == "p" or self.field[i] == "r":
|
2024-03-10 14:05:44 +01:00
|
|
|
|
self.field[i] = ""
|
|
|
|
|
|
|
|
|
|
if "+" in self.field[i]:
|
|
|
|
|
self.field[i] = self.field[i].replace("+", "")
|
2024-03-05 05:39:03 +01:00
|
|
|
|
|
|
|
|
|
def set_ship(self, row, col):
|
|
|
|
|
if row < 0 or row > self.field_size:
|
|
|
|
|
return
|
|
|
|
|
if col < 0 or col > self.field_size:
|
|
|
|
|
return
|
|
|
|
|
index = row * self.field_size + col
|
2024-04-21 19:07:27 +02:00
|
|
|
|
if self.ship_direction == ShipDirection.VERTICAL:
|
2024-03-05 05:39:03 +01:00
|
|
|
|
if self.field_size - row < self.ship_size:
|
|
|
|
|
return
|
|
|
|
|
for r in range(row, row + self.ship_size):
|
|
|
|
|
index = r * self.field_size + col
|
|
|
|
|
self.field[index] = "1"
|
2024-04-21 19:07:27 +02:00
|
|
|
|
if self.ship_direction == ShipDirection.HORIZONTAL:
|
2024-03-05 05:39:03 +01:00
|
|
|
|
if self.field_size - col < self.ship_size:
|
|
|
|
|
return
|
|
|
|
|
for c in range(col, col + self.ship_size):
|
|
|
|
|
index = row * self.field_size + c
|
|
|
|
|
self.field[index] = "1"
|
|
|
|
|
|
2024-03-17 19:34:18 +01:00
|
|
|
|
if self.ship_size in self.ships:
|
|
|
|
|
self.ships.remove(self.ship_size)
|
|
|
|
|
self.print_field()
|
|
|
|
|
|
2024-04-21 19:07:27 +02:00
|
|
|
|
def get_ship(self, row, col):
|
|
|
|
|
if row < 0 or row > self.field_size:
|
|
|
|
|
return
|
|
|
|
|
if col < 0 or col > self.field_size:
|
|
|
|
|
return
|
|
|
|
|
|
|
|
|
|
self.field[row * self.field_size + col] = ""
|
|
|
|
|
|
|
|
|
|
ship_size = 1
|
|
|
|
|
ship_direction = ShipDirection.UNKNOWN
|
|
|
|
|
|
|
|
|
|
# проверим вертикаль
|
|
|
|
|
for r in range(row + 1, self.field_size):
|
|
|
|
|
if self.check_ship(r, col):
|
|
|
|
|
ship_size += 1
|
|
|
|
|
ship_direction = ShipDirection.VERTICAL
|
|
|
|
|
self.field[r * self.field_size + col] = ""
|
|
|
|
|
else:
|
|
|
|
|
break
|
|
|
|
|
|
|
|
|
|
for r in range(row - 1, -1, -1):
|
|
|
|
|
if self.check_ship(r, col):
|
|
|
|
|
ship_size += 1
|
|
|
|
|
ship_direction = ShipDirection.VERTICAL
|
|
|
|
|
self.field[row * self.field_size + col] = ""
|
|
|
|
|
else:
|
|
|
|
|
break
|
|
|
|
|
|
|
|
|
|
if ship_direction == ShipDirection.UNKNOWN:
|
|
|
|
|
|
|
|
|
|
# проверим горизонталь
|
|
|
|
|
for c in range(col + 1, self.field_size):
|
|
|
|
|
if self.check_ship(row, c):
|
|
|
|
|
ship_size += 1
|
|
|
|
|
ship_direction = ShipDirection.HORIZONTAL
|
|
|
|
|
self.field[row * self.field_size + c] = ""
|
|
|
|
|
else:
|
|
|
|
|
break
|
|
|
|
|
|
|
|
|
|
for c in range(col - 1, -1, -1):
|
|
|
|
|
if self.check_ship(row, c):
|
|
|
|
|
ship_size += 1
|
|
|
|
|
ship_direction = ShipDirection.HORIZONTAL
|
|
|
|
|
self.field[row * self.field_size + c] = ""
|
|
|
|
|
else:
|
|
|
|
|
break
|
|
|
|
|
self.set_ship_direction(ship_direction)
|
|
|
|
|
self.set_ship_size(ship_size)
|
|
|
|
|
self.ships.append(ship_size)
|
|
|
|
|
|
2024-03-05 05:39:03 +01:00
|
|
|
|
def shoot(self, row, col):
|
|
|
|
|
if row < 0 or row > self.field_size - 1:
|
|
|
|
|
return ShootResult.UNDEFINED
|
|
|
|
|
if col < 0 or col > self.field_size - 1:
|
|
|
|
|
return ShootResult.UNDEFINED
|
|
|
|
|
index = row * self.field_size + col
|
|
|
|
|
if (self.field[index]).strip() == "":
|
|
|
|
|
self.field[index] = "0"
|
|
|
|
|
return ShootResult.EMPTY
|
|
|
|
|
elif (self.field[index]).strip() == "1":
|
|
|
|
|
self.field[index] = "\\"
|
|
|
|
|
return ShootResult.DAMAGED
|
|
|
|
|
else:
|
|
|
|
|
return ShootResult.UNDEFINED
|
2024-03-10 14:05:44 +01:00
|
|
|
|
|
2024-04-21 19:07:27 +02:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def check_ship(self, row, col):
|
|
|
|
|
#функция должна возвражать тру, если в заданной клетке есть корабль
|
|
|
|
|
# в противном случае фолс
|
|
|
|
|
return self.field[row * self.field_size + col].strip() == "1"
|
|
|
|
|
|
|
|
|
|
|
2024-03-10 14:05:44 +01:00
|
|
|
|
def check_possible(self, row, col):
|
|
|
|
|
# Функция должна возвращать True, если можно поставить сюда корабль,
|
|
|
|
|
# в противном случае - False
|
2024-04-21 19:07:27 +02:00
|
|
|
|
if self.ship_direction == ShipDirection.VERTICAL:
|
2024-03-10 14:05:44 +01:00
|
|
|
|
# Здесь мы знаем, что корабль помещается на поле.
|
|
|
|
|
if self.field_size - row >= self.ship_size:
|
|
|
|
|
# Теперь нужно проверить, не заблокировано ли какое-то из полей,
|
|
|
|
|
for r in range(row, row + self.ship_size):
|
|
|
|
|
if not self.check_blocked(r, col):
|
|
|
|
|
return False
|
|
|
|
|
return True
|
|
|
|
|
|
2024-04-21 19:07:27 +02:00
|
|
|
|
if self.ship_direction == ShipDirection.HORIZONTAL:
|
2024-03-10 14:05:44 +01:00
|
|
|
|
if self.field_size - col >= self.ship_size:
|
|
|
|
|
for c in range(col, col + self.ship_size):
|
|
|
|
|
if not self.check_blocked(row, c):
|
|
|
|
|
return False
|
|
|
|
|
return True
|
|
|
|
|
|
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
|
def check_blocked(self, row, col):
|
|
|
|
|
# Функция возвращает True, если все клетки вокруг клетки с координатами row, col
|
|
|
|
|
# либо находятся за пределами поля, либо в них нет корабля/они пустые
|
|
|
|
|
for r in range(row - 1, row + 2):
|
|
|
|
|
for c in range(col - 1, col + 2):
|
|
|
|
|
if 0 <= r < self.field_size and 0 <= c < self.field_size:
|
|
|
|
|
cell = (self.field[r * self.field_size + c]).strip()
|
|
|
|
|
if cell != '' and cell != 'p':
|
|
|
|
|
return False
|
|
|
|
|
return True
|
|
|
|
|
|
|
|
|
|
def set_ship_size(self, value):
|
|
|
|
|
if value is None:
|
|
|
|
|
return
|
|
|
|
|
|
|
|
|
|
if type(value) is str and value.isnumeric():
|
|
|
|
|
value = int(value)
|
|
|
|
|
|
|
|
|
|
if type(value) is int and 1 <= value <= 4:
|
|
|
|
|
self.ship_size = value
|
|
|
|
|
|
|
|
|
|
def set_ship_direction(self, value):
|
|
|
|
|
if value is None:
|
|
|
|
|
return
|
|
|
|
|
|
2024-04-21 19:07:27 +02:00
|
|
|
|
if type(value) is not ShipDirection:
|
|
|
|
|
return
|
2024-03-10 14:05:44 +01:00
|
|
|
|
|
2024-04-21 19:07:27 +02:00
|
|
|
|
if value != ShipDirection.UNKNOWN:
|
2024-03-10 14:05:44 +01:00
|
|
|
|
self.ship_direction = value
|
|
|
|
|
|
2024-04-21 19:07:27 +02:00
|
|
|
|
|
2024-03-10 14:05:44 +01:00
|
|
|
|
def toggle_ship_direction(self):
|
2024-04-21 19:07:27 +02:00
|
|
|
|
if self.ship_direction == ShipDirection.VERTICAL:
|
|
|
|
|
self.ship_direction = ShipDirection.HORIZONTAL
|
2024-03-10 14:05:44 +01:00
|
|
|
|
else:
|
2024-04-21 19:07:27 +02:00
|
|
|
|
self.ship_direction = ShipDirection.VERTICAL
|
2024-03-10 14:05:44 +01:00
|
|
|
|
|
|
|
|
|
def toggle_field_mode(self):
|
2024-04-21 19:07:27 +02:00
|
|
|
|
if self.field_mode == ShipMode.PUT:
|
|
|
|
|
self.field_mode = ShipMode.SHOOT
|
2024-03-10 14:05:44 +01:00
|
|
|
|
else:
|
2024-04-21 19:07:27 +02:00
|
|
|
|
self.field_mode = ShipMode.PUT
|
2024-03-10 14:05:44 +01:00
|
|
|
|
|
|
|
|
|
def print_field(self):
|
2024-03-17 19:34:18 +01:00
|
|
|
|
print(self.ships)
|
2024-03-10 14:05:44 +01:00
|
|
|
|
for r in range(0, self.field_size):
|
|
|
|
|
blocked_string = ""
|
|
|
|
|
ship_string = ""
|
|
|
|
|
for c in range(0, self.field_size):
|
|
|
|
|
blocked_string += str(self.check_blocked(r, c))[0] + ", "
|
|
|
|
|
ship_string += self.field[r * self.field_size + c] + ', '
|
|
|
|
|
print(ship_string[:-2])
|
2024-03-17 19:34:18 +01:00
|
|
|
|
# print(blocked_string[:-2] + ' ' + ship_string[:-2])
|
|
|
|
|
print("********************************************************************")
|