import numpy as np
from minigrid.core.world_object import WorldObj
from minigrid.utils.baby_ai_bot import Subgoal, ObjDesc, manhattan_distance

from babyai_utils import find_pos_everywhere, shortest_path_everywhere


class ActionInfo:
    def __init__(self, success, data):
        self.data = data
        self.success = success


class GoNextToSubgoal(Subgoal):
    """The subgoal for going next to objects or positions.

    Args:
        datum (int, int): tuple or `ObjDesc` or object reference
            The position or the description of the object or
            the object to which we are going.
        reason (str): One of the following:
            - `None`: go the position (object) and face it
            - `"PutNext"`: go face an empty position next to the object specified by `datum`
            - `"Explore"`: going to a position, just like when the reason is `None`. The only
                difference is that with this reason the subgoal will be considered exploratory

    """

    def replan_before_action(self):
        #print("ACTIONS:", self.actions.right)
        #print(type(self.bot))
        target_obj = None
        if isinstance(self.datum, ObjDesc):
            target_obj, target_pos = find_pos_everywhere(
                self.bot,
                self.datum, self.reason == "PutNext"
            )
            if not target_pos:
                # No path found -> Explore the world
                # print("Exploring... fix!!")
                self.bot.stack.append(ExploreSubgoal(self.bot))
                raise Exception("Explore")
                return
        elif isinstance(self.datum, WorldObj):
            target_obj = self.datum
            target_pos = target_obj.cur_pos
        else:
            target_pos = tuple(self.datum)

        # Suppore we are walking towards the door that we would like to open,
        # it is locked, and we don't have the key. What do we do? If we are carrying
        # something, it makes to just continue, as we still need to bring this object
        # close to the door. If we are not carrying anything though, then it makes
        # sense to change the plan and go straight for the required key.


        # The position we are on is the one we should go next to
        # -> Move away from it
        if manhattan_distance(target_pos, self.pos) == (
                1 if self.reason == "PutNext" else 0
        ):

            def steppable(cell):
                return cell is None or (cell.type == "door" and cell.is_open)

            if steppable(self.fwd_cell):
                return self.actions.forward
            if steppable(
                    self.bot.mission.unwrapped.grid.get(*(self.pos + self.right_vec))
            ):
                return self.actions.right
            if steppable(
                    self.bot.mission.unwrapped.grid.get(*(self.pos - self.right_vec))
            ):
                return self.actions.left
            # Spin and hope for the best
            return self.actions.left

        # We are facing the target cell
        # -> subgoal completed
        if self.reason == "PutNext":
            if manhattan_distance(target_pos, self.fwd_pos) == 1:
                if self.fwd_cell is None:
                    self.bot.stack.pop()
                    return
                if self.fwd_cell.type == "door" and self.fwd_cell.is_open:
                    # We can't drop an object in the cell where the door is.
                    # Instead, we add a subgoal on the stack that will force
                    # the bot to move the target object.
                    self.bot.stack.append(
                        GoNextToSubgoal(self.bot, self.fwd_pos + 2 * self.dir_vec)
                    )
                    return
        else:
            if np.array_equal(target_pos, self.fwd_pos):
                self.bot.stack.pop()
                return

        # We are still far from the target
        # -> try to find a non-blocker path
        path, _, _ = shortest_path_everywhere(self.bot,
            lambda pos, cell: pos == target_pos,
        )

        # No non-blocker path found and
        # reexploration within the room is not allowed or there is nothing to explore
        # -> Look for blocker paths
        if not path:
            path, _, _ = shortest_path_everywhere(self.bot,
                lambda pos, cell: pos == target_pos, try_with_blockers=True
            )

        # No path found
        # -> explore the world
        if not path:
            # self.bot.stack.append(ExploreSubgoal(self.bot))
            print("no path found")
            return

        # So there is a path (blocker, or non-blockers)
        # -> try following it
        next_cell = np.asarray(path[0])

        # Choose the action in the case when the forward cell
        # is the one we should go next to
        if np.array_equal(next_cell, self.fwd_pos):
            if self.fwd_cell:
               if self.fwd_cell.type == "door":
                    if not self.fwd_cell.is_open:
                        return ActionInfo(False, f"Mistake: The {self.fwd_cell.color} door is not open")
                    else:
                        return ActionInfo(True, self.actions.forward)
            else:
                return self.actions.forward

        # The forward cell is not the one we should go to
        # -> turn towards the direction we need to go
        if np.array_equal(next_cell - self.pos, self.right_vec):
            return self.actions.right
        elif np.array_equal(next_cell - self.pos, -self.right_vec):
            return self.actions.left

        # If we reach this point in the code,  then the cell is behind us.
        # Instead of choosing left or right randomly,
        # let's do something that might be useful:
        # Because when we're GoingNextTo for the purpose of exploring,
        # things might change while on the way to the position we're going to, we should
        # pick this right or left wisely.
        # The simplest thing we should do is: pick the one
        # that doesn't lead you to face a non empty cell.
        # One better thing would be to go to the direction
        # where the closest wall/door is the furthest
        distance_right = self.bot._closest_wall_or_door_given_dir(
            self.pos, self.right_vec
        )
        distance_left = self.bot._closest_wall_or_door_given_dir(
            self.pos, -self.right_vec
        )
        if distance_left > distance_right:
            return self.actions.left
        return self.actions.right

class OpenSubgoal(Subgoal):
    def replan_before_action(self):
        if self.fwd_cell is None or self.fwd_cell.type != 'door':
            return ActionInfo(False, "Mistake: There is no door to open")
            # return "There is no door to open"

        got_the_key = (
                self.carrying
                and self.carrying.type == "key"
                and self.carrying.color == self.fwd_cell.color
        )

        if self.fwd_cell.is_locked and not got_the_key:
            return ActionInfo(False, f"Mistake: The {self.fwd_cell.color} door is locked, go next to {self.fwd_cell.color} key, pick it up and keep it  and try open it again while carrying the key!!!")

        if self.fwd_cell.is_open:
            return ActionInfo(False, f"Mistake: The {self.fwd_cell.color} door is already open")
        print("Ich öffne die Tür")
        return ActionInfo(True, self.actions.toggle)

    def replan_after_action(self, action_taken):
        if action_taken is None or action_taken == self.actions.toggle:
            self.bot.stack.pop()
            return

class CloseSubgoal(Subgoal):
    def replan_before_action(self):
        if self.fwd_cell is None or self.fwd_cell.type != 'door':
            return ActionInfo(False, "Mistake: There is no door to close")
            # return "There is no door to close"

        if not self.fwd_cell.is_open:
            return ActionInfo(False, f"Mistake: The {self.fwd_cell.color} door is already closed")

        return ActionInfo(True, self.actions.toggle)

    def replan_after_action(self, action_taken):
        if action_taken is None or action_taken == self.actions.toggle:
            self.bot.stack.pop()
            return


class PickupSubgoal(Subgoal):
    def replan_before_action(self):
        if self.fwd_cell is None or not self.fwd_cell.can_pickup():
            return ActionInfo(False, "Mistake: There is nothing to pick up")

        if self.carrying:
            return ActionInfo(False, f"Mistake: Already carrying {self.carrying.color}{self.carrying.type}")

        print(self.actions.pickup)
        return ActionInfo(True, self.actions.pickup)

    def replan_after_action(self, action_taken):
        if action_taken is None or action_taken == self.actions.pickup:
            self.bot.stack.pop()

class FindDropLocationSubgoal(Subgoal):
    def replan_before_action(self):

        self.bot.stack.pop()
        drop_pos = self.bot._find_drop_pos()
        if not drop_pos:
            return ActionInfo(False, "Mistake: No empty cell to drop the object")

        self.bot.stack.append(GoNextToSubgoal(self.bot, drop_pos))
        # return ActionInfo(False, "Mistake: The cell is not empty")


        return ActionInfo(True, self.actions.drop)

class DropSubgoal(Subgoal):
    def replan_before_action(self):
        if not self.carrying:
            return ActionInfo(False, "Mistake: Nothing to drop")

        if self.fwd_cell is not None:
            return ActionInfo(False, "Mistake: The cell is not empty")

        return ActionInfo(True, self.actions.drop)

    def replan_after_action(self, action_taken):
        if action_taken is None or action_taken == self.actions.drop:
            self.bot.stack.pop()


class ExploreSubgoal(Subgoal):
    def replan_before_action(self):
        # Find the closest unseen position
        _, unseen_pos, with_blockers = self.bot._shortest_path(
            lambda pos, cell: not self.bot.vis_mask[pos], try_with_blockers=True
        )

        if unseen_pos:
            self.bot.stack.append(
                GoNextToSubgoal(self.bot, unseen_pos, reason="Explore")
            )
            return None

        # Find the closest unlocked unopened door
        def unopened_unlocked_door(pos, cell):
            return (
                    cell and cell.type == "door" and not cell.is_locked and not cell.is_open
            )

        # Find the closest unopened door
        def unopened_door(pos, cell):
            return cell and cell.type == "door" and not cell.is_open

        # # Try to find an unlocked door first.
        # # We do this because otherwise, opening a locked door as
        # # a subgoal may try to open the same door for exploration,
        # # resulting in an infinite loop.
        # _, door_pos, _ = self.bot._shortest_path(
        #     unopened_unlocked_door, try_with_blockers=True
        # )
        # if not door_pos:
        #     # Try to find a locker door if an unlocked one is not available.
        #     _, door_pos, _ = self.bot._shortest_path(
        #         unopened_door, try_with_blockers=True
        #     )
        #
        # # Open the door
        # if door_pos:
        #     door_obj = self.bot.mission.unwrapped.grid.get(*door_pos)
        #     # If we are going to a locked door, there are two cases:
        #     # - we already have the key, then we should not drop it
        #     # - we don't have the key, in which case eventually we should drop it
        #     got_the_key = (
        #             self.carrying
        #             and self.carrying.type == "key"
        #             and self.carrying.color == door_obj.color
        #     )
        #     open_reason = "KeepKey" if door_obj.is_locked and got_the_key else None
        #     self.bot.stack.pop()
        #     self.bot.stack.append(OpenSubgoal(self.bot, reason=open_reason))
        #     self.bot.stack.append(GoNextToSubgoal(self.bot, door_obj, reason="Open"))
        #     return
        self.bot.stack.pop()
        return
        assert False, "nothing left to explore"

    def is_exploratory(self):
        return True


class FinishSubgoal(Subgoal):
    def replan_before_action(self):
        self.bot.stack.pop()
        return ActionInfo(True, None)