import numpy as np

class Object:
    def __init__(self):
        self.state = None
        self.interaction_trace = list()

class Action(Object):
    def __init__(self):
        super().__init__()
        self.name = "Action"
        self.action = 0
        self.interaction_trace = list()

    def step(self, action):
        self.action = action

    def get_state(self):
        return np.array([self.action])

class Bound(Object):
    def __init__(self, limits, n_dim=2):
        super().__init__()
        self.name = "Bound"
        self.limits = limits

    def get_state(self):
        return np.array(self.limits)

class Pusher(Object):
    def __init__(self, pos, bound):
        super().__init__()
        self.name = "Pusher"
        self.pos = pos
        self.pos_change = np.zeros(pos.shape)
        self.bound = bound
        self.movement_type = "coordinate" # could have a different movement type

    def out_of_bounds(self, pos):
        return 0 > pos[0] or pos[0] >= self.bound.limits[0] or 0 > pos[1] or pos[1] >= self.bound.limits[1]  

    def step(self, action, occupancy_matrix):
        self.pos_change = np.zeros(self.pos.shape)
        action.interaction_trace = [self]
        self.interaction_trace = [action]
        if self.movement_type == "coordinate":
            if action.action == 0: self.pos_change[0] -= 1
            if action.action == 1: self.pos_change[0] += 1
            if action.action == 2: self.pos_change[1] -= 1
            if action.action == 3: self.pos_change[1] += 1
            new_pos = self.pos + self.pos_change
            if self.out_of_bounds(new_pos): return # there will be no changes if moving out of bounds in update
            obj_at = occupancy_matrix[int(new_pos[0])][int(new_pos[1])]
            if obj_at is not None:
                if type(obj_at) == Block:
                    if not obj_at.moveable(self.pos_change, occupancy_matrix):
                        self.interaction_trace += [obj_at]
                        self.pos_change = np.zeros(self.pos.shape)
                if type(obj_at) == tuple:
                    obj_at = obj_at[0]
                    if not obj_at.moveable(self.pos_change, occupancy_matrix):
                        self.interaction_trace += [obj_at]
                        self.pos_change = np.zeros(self.pos.shape)
                elif type(obj_at) == Obstacle:
                    self.interaction_trace += [obj_at]
                    self.pos_change = np.zeros(self.pos.shape)

    def update(self): # TODO: bounds don't have trace
        old_pos = self.pos.copy()
        self.pos[0] = np.clip(self.pos[0] + self.pos_change[0], 0, self.bound.limits[0]-1)
        self.pos[1] = np.clip(self.pos[1] + self.pos_change[1], 0, self.bound.limits[1]-1)
        return old_pos

    def get_state(self):
        return np.array(self.pos.tolist())



class Obstacle(Object):
    def __init__(self, pos, idx, bound):
        super().__init__()
        self.pos = pos
        self.name = "Obstacle" + str(idx) if idx >= 0 else "Obstacle"
    
    def get_state(self):
        return np.array(self.pos.tolist())

class Block(Object):
    def __init__(self, pos, idx, bound):
        super().__init__()
        self.pos = pos
        self.pushed = False
        self.name = "Block" + str(idx) if idx >= 0 else "Block"
        self.bound = bound

    def moveable(self, change, occupancy_matrix):
        self.pushed = True
        new_pos = self.pos + change
        # print("new pos", new_pos)
        if 0 > new_pos[0] or new_pos[0] >= self.bound.limits[0] or 0 > new_pos[1] or new_pos[1] >= self.bound.limits[1]:
            return False
        obj_at = occupancy_matrix[int(new_pos[0])][int(new_pos[1])]
        if obj_at is not None:
            if type(obj_at) != Target:
                return False
        return True 

    def step(self, pusher, occupancy_matrix):
        self.pos_change = self.pos
        self.interaction_trace = list()
        if self.pushed:
            self.interaction_trace.append(pusher)
            pusher.interaction_trace.append(self)
            change = pusher.pos_change
            if self.moveable(change, occupancy_matrix):
                self.pos_change = self.pos + change.copy()

    def update(self):
        self.pushed = False
        old = self.pos.copy()
        self.pos = self.pos_change
        moved = None
        # print(np.linalg.norm(old - self.pos), self.pos.copy())
        if np.linalg.norm(old - self.pos) > 0.01:
            moved = self
        return old, moved

    def get_state(self):
        return np.array(self.pos.tolist())

class Target(Object):
    def __init__(self, pos, idx, bound):
        super().__init__()
        self.pos = pos
        self.attribute = 0
        self.attribute_change = 0
        self.name = "Target" + str(idx) if idx >= 0 else "Target"

    def step(self, occupancy_matrix):
        obj_at = occupancy_matrix[self.pos[0]][self.pos[1]]
        if type(obj_at) == tuple: 
            if type(obj_at[0]) == Block:
                self.interaction_trace.append(obj_at[0])
                self.attribute_change = 1
        else:
            self.attribute_change = 0

    def update(self): 
        self.attribute = self.attribute_change

    def get_state(self):
        return np.array(self.pos.tolist() + [self.attribute])
