from abc import ABC
from abc import abstractmethod

from collections import namedtuple
import numpy as np

from .object import Object


class BaseMaze(ABC):
    def __init__(self, **kwargs):
        objects = self.make_objects()
        assert all([isinstance(obj, Object) for obj in objects])
        self.objects = namedtuple('Objects', map(lambda x: x.name, objects), defaults=objects)()
        
        for key, value in kwargs.items():
            setattr(self, key, value)
    
    @property
    @abstractmethod
    def size(self):
        r"""Returns a pair of (height, width). """
        pass
        
    @abstractmethod
    def make_objects(self):
        r"""Returns a list of defined objects. """
        pass
    
    def _convert(self, x, name):
        for obj in self.objects:
            pos = np.asarray(obj.positions)
            x[pos[:, 0], pos[:, 1]] = getattr(obj, name, None)
            #if name != 'rgb' or  obj.name != 'goal':
            #    pos = np.asarray(obj.positions)
            #    x[pos[:, 0], pos[:, 1]] = getattr(obj, name, None)
            #else:
            #    pos = np.asarray(obj.positions)
            #    x[pos[:, 0], pos[:, 1]] = getattr(self.objects[0], name, None)
        return x
    
    def to_name(self):
        x = np.empty(self.size, dtype=object)
        return self._convert(x, 'name')
    
    def to_value(self):
        x = np.empty(self.size, dtype=int)
        return self._convert(x, 'value')
    
    def to_rgb(self):
        x = np.empty((*self.size, 3), dtype=np.uint8)
        return self._convert(x, 'rgb')
    
    def to_impassable(self):
        x = np.empty(self.size, dtype=bool)
        return self._convert(x, 'impassable')
    
    def __repr__(self):
        return f'{self.__class__.__name__}{self.size}'
