2021-04-19 17:20:40 +02:00
|
|
|
from enum import Enum
|
2021-05-09 18:11:25 +02:00
|
|
|
from queue import PriorityQueue
|
2021-05-21 20:14:00 +02:00
|
|
|
from typing import Tuple, List
|
2021-04-19 17:20:40 +02:00
|
|
|
|
2021-06-06 19:55:55 +02:00
|
|
|
from survival.components.direction_component import DirectionChangeComponent
|
|
|
|
from survival.components.moving_component import MovingComponent
|
2021-04-19 17:20:40 +02:00
|
|
|
from survival.components.position_component import PositionComponent
|
2021-05-24 17:20:08 +02:00
|
|
|
from survival.components.resource_component import ResourceComponent
|
2021-06-19 00:09:14 +02:00
|
|
|
from survival.game.enums import Direction
|
2021-05-24 17:20:08 +02:00
|
|
|
from survival.esper import World
|
2021-06-20 16:27:38 +02:00
|
|
|
from survival.systems.consumption_system import ConsumeComponent
|
2021-04-19 17:20:40 +02:00
|
|
|
|
|
|
|
|
|
|
|
class Action(Enum):
|
|
|
|
ROTATE_LEFT = 0
|
|
|
|
ROTATE_RIGHT = 1
|
|
|
|
MOVE = 2
|
|
|
|
|
2021-06-06 19:55:55 +02:00
|
|
|
@staticmethod
|
|
|
|
def from_array(action):
|
|
|
|
if action[0] == 1:
|
|
|
|
return Action.MOVE
|
|
|
|
if action[1] == 1:
|
|
|
|
return Action.ROTATE_LEFT
|
|
|
|
if action[2] == 1:
|
|
|
|
return Action.ROTATE_RIGHT
|
|
|
|
raise Exception("Unknown action.")
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def perform(world, entity, action):
|
|
|
|
if world.has_component(entity, MovingComponent):
|
|
|
|
raise Exception(f"Entity was already moving. Could not perform action: {action}")
|
|
|
|
if world.has_component(entity, DirectionChangeComponent):
|
|
|
|
raise Exception(f"Entity was already rotating. Could not perform action: {action}")
|
|
|
|
|
|
|
|
if action == Action.ROTATE_LEFT:
|
|
|
|
world.add_component(entity, DirectionChangeComponent(
|
|
|
|
Direction.rotate_left(world.component_for_entity(entity, PositionComponent).direction)))
|
2021-06-20 16:27:38 +02:00
|
|
|
world.add_component(entity, ConsumeComponent(0.2))
|
2021-06-06 19:55:55 +02:00
|
|
|
elif action == Action.ROTATE_RIGHT:
|
|
|
|
world.add_component(entity, DirectionChangeComponent(
|
|
|
|
Direction.rotate_right(world.component_for_entity(entity, PositionComponent).direction)))
|
2021-06-20 16:27:38 +02:00
|
|
|
world.add_component(entity, ConsumeComponent(0.2))
|
2021-06-06 19:55:55 +02:00
|
|
|
else:
|
|
|
|
world.add_component(entity, MovingComponent())
|
|
|
|
return action
|
|
|
|
|
2021-04-19 17:20:40 +02:00
|
|
|
|
|
|
|
class State:
|
2021-05-21 20:14:00 +02:00
|
|
|
def __init__(self, position: Tuple[int, int], direction: Direction):
|
2021-04-19 17:20:40 +02:00
|
|
|
self.position = position
|
|
|
|
self.direction = direction
|
|
|
|
|
|
|
|
|
|
|
|
class Node:
|
2021-05-09 18:11:25 +02:00
|
|
|
def __init__(self, state: State, parent=None, action=None, cost=None):
|
2021-04-19 17:20:40 +02:00
|
|
|
self.state = state
|
|
|
|
self.parent = parent
|
|
|
|
self.action = action
|
2021-05-09 18:11:25 +02:00
|
|
|
self.cost = cost
|
2021-04-19 17:20:40 +02:00
|
|
|
|
2021-05-09 18:11:25 +02:00
|
|
|
def __lt__(self, other):
|
|
|
|
return self.cost < other.cost
|
2021-04-19 17:20:40 +02:00
|
|
|
|
2021-05-09 18:11:25 +02:00
|
|
|
def __eq__(self, other):
|
|
|
|
return self.cost == other.cost
|
|
|
|
|
|
|
|
|
2021-05-21 20:14:00 +02:00
|
|
|
def get_moved_position(position: Tuple[int, int], direction: Direction):
|
2021-04-19 17:20:40 +02:00
|
|
|
vector = Direction.get_vector(direction)
|
|
|
|
return position[0] + vector[0], position[1] + vector[1]
|
|
|
|
|
|
|
|
|
2021-06-06 19:55:55 +02:00
|
|
|
def get_states(state: State, game_map, world: World) -> List[Tuple[Action, State, int]]:
|
2021-04-19 17:20:40 +02:00
|
|
|
states = list()
|
|
|
|
|
2021-05-09 18:11:25 +02:00
|
|
|
states.append((Action.ROTATE_LEFT, State(state.position, state.direction.rotate_left(state.direction)), 1))
|
|
|
|
states.append((Action.ROTATE_RIGHT, State(state.position, state.direction.rotate_right(state.direction)), 1))
|
2021-04-19 17:20:40 +02:00
|
|
|
|
2021-05-09 18:11:25 +02:00
|
|
|
target_position = get_moved_position(state.position, state.direction)
|
|
|
|
if not game_map.is_colliding(target_position):
|
|
|
|
states.append((Action.MOVE, State(target_position, state.direction), game_map.get_cost(target_position)))
|
2021-05-24 17:20:08 +02:00
|
|
|
elif game_map.get_entity(target_position) is not None:
|
|
|
|
ent = game_map.get_entity(target_position)
|
|
|
|
if world.has_component(ent, ResourceComponent):
|
|
|
|
states.append((Action.MOVE, State(target_position, state.direction), 3))
|
2021-04-19 17:20:40 +02:00
|
|
|
|
|
|
|
return states
|
|
|
|
|
|
|
|
|
2021-05-09 18:11:25 +02:00
|
|
|
def build_path(node: Node):
|
2021-06-06 19:55:55 +02:00
|
|
|
cost = 0
|
2021-05-09 18:11:25 +02:00
|
|
|
actions = [node.action]
|
|
|
|
parent = node.parent
|
|
|
|
|
|
|
|
while parent is not None:
|
|
|
|
if parent.action is not None:
|
|
|
|
actions.append(parent.action)
|
2021-06-06 19:55:55 +02:00
|
|
|
cost += parent.cost
|
2021-05-09 18:11:25 +02:00
|
|
|
parent = parent.parent
|
|
|
|
|
|
|
|
actions.reverse()
|
2021-06-06 19:55:55 +02:00
|
|
|
return actions, cost
|
2021-05-09 18:11:25 +02:00
|
|
|
|
|
|
|
|
2021-05-21 20:14:00 +02:00
|
|
|
def heuristic(new_node: Node, goal: Tuple[int, int]):
|
2021-05-09 18:11:25 +02:00
|
|
|
return abs(new_node.state.position[0] - goal[0]) + abs(new_node.state.position[1] - goal[1])
|
|
|
|
|
|
|
|
|
2021-06-06 19:55:55 +02:00
|
|
|
def graph_search(game_map, start: PositionComponent, goal: tuple, world: World):
|
2021-05-09 18:11:25 +02:00
|
|
|
fringe = PriorityQueue()
|
2021-04-19 17:20:40 +02:00
|
|
|
explored = list()
|
|
|
|
|
|
|
|
explored_states = set()
|
2021-05-09 18:11:25 +02:00
|
|
|
fringe_states = set() # Stores positions and directions of states
|
2021-04-19 17:20:40 +02:00
|
|
|
|
|
|
|
start = State(start.grid_position, start.direction)
|
2021-05-09 18:11:25 +02:00
|
|
|
fringe.put((0, Node(start, cost=0)))
|
2021-04-19 17:20:40 +02:00
|
|
|
fringe_states.add((tuple(start.position), start.direction))
|
|
|
|
|
|
|
|
while True:
|
|
|
|
# No solutions found
|
2021-05-09 18:11:25 +02:00
|
|
|
if fringe.empty():
|
2021-06-06 19:55:55 +02:00
|
|
|
return [], 0
|
2021-04-19 17:20:40 +02:00
|
|
|
|
2021-05-09 18:11:25 +02:00
|
|
|
node = fringe.get()
|
|
|
|
node_priority = node[0]
|
|
|
|
node = node[1]
|
2021-04-19 17:20:40 +02:00
|
|
|
fringe_states.remove((tuple(node.state.position), node.state.direction))
|
|
|
|
|
|
|
|
# Check goal
|
|
|
|
if node.state.position == goal:
|
2021-05-09 18:11:25 +02:00
|
|
|
return build_path(node)
|
2021-04-19 17:20:40 +02:00
|
|
|
|
|
|
|
explored.append(node)
|
|
|
|
explored_states.add((tuple(node.state.position), node.state.direction))
|
|
|
|
|
|
|
|
# Get all possible states
|
2021-05-24 17:20:08 +02:00
|
|
|
for state in get_states(node.state, game_map, world):
|
2021-04-19 17:20:40 +02:00
|
|
|
sub_state = (tuple(state[1].position), state[1].direction)
|
2021-05-09 18:11:25 +02:00
|
|
|
new_node = Node(state=state[1],
|
|
|
|
parent=node,
|
|
|
|
action=state[0],
|
|
|
|
cost=(state[2] + node.cost))
|
2021-06-06 19:55:55 +02:00
|
|
|
|
2021-05-09 18:11:25 +02:00
|
|
|
priority = new_node.cost + heuristic(new_node, goal)
|
2021-04-19 17:20:40 +02:00
|
|
|
if sub_state not in fringe_states and sub_state not in explored_states:
|
2021-05-09 18:11:25 +02:00
|
|
|
fringe.put((priority, new_node))
|
|
|
|
fringe_states.add((tuple(new_node.state.position), new_node.state.direction))
|
|
|
|
elif sub_state in fringe_states and node.cost > new_node.cost:
|
|
|
|
fringe.get(node)
|
|
|
|
fringe.put((priority, new_node))
|
2021-04-19 17:20:40 +02:00
|
|
|
fringe_states.add((tuple(new_node.state.position), new_node.state.direction))
|