"""
The module contains abstract interfaces of the classes.
The interfaces are aimed to be schemas for further classes implementations.
Following these schemas will ensure compatibility of the code with the entire project.
"""
from abc import ABCMeta, abstractmethod
from future.utils import with_metaclass
import numpy as np
import torch


class AbstractStructureDesigner(with_metaclass(ABCMeta, object)):
    """The class is an abstraction for the list of atomic coordinates."""

    def __init__(self):

        self._nn_distance = None
        self._kd_tree = None

    def _get_neighbours(self, query):
        """

        Parameters
        ----------
        query :


        Returns
        -------

        """

        if isinstance(query, list) or isinstance(query, torch.Tensor):
            ans = self._kd_tree.query(query.detach(),
                                      k=25,
                                      distance_upper_bound=self._nn_distance)
        elif isinstance(query, int):

            query = list(self.atom_list.items())[query][1]

            if self._kd_tree.data.shape[1] > 3:
                query = torch.cat((query,torch.tensor(0.)))

            ans = self._kd_tree.query(query.detach(),
                                      k=25,
                                      distance_upper_bound=self._nn_distance)
        elif isinstance(query, str):
            ans = self._kd_tree.query(self.atom_list[query],
                                      k=25,
                                      distance_upper_bound=self._nn_distance)
        else:
            raise TypeError('Wrong input type for query')

        return ans

    @abstractmethod
    def get_neighbours(self, query):
        """

        Parameters
        ----------
        query :


        Returns
        -------

        """
        pass

    @property
    @abstractmethod
    def atom_list(self):
        """ """
        pass


class AbstractBasis(with_metaclass(ABCMeta, object)):
    """The class contains information about sets of quantum numbers and
    dimensionality of the Hilbert space.
    It is also equipped with the member functions translating quantum numbers
    into a raw index and vise versa.

    Parameters
    ----------

    Returns
    -------

    """

    @abstractmethod
    def qn2ind(self, qn):
        """The member function trasform a dictionary of quantum numbers into a matrix index

        Parameters
        ----------
        qn :
            type qn:

            :return ind:   index

        Returns
        -------

        """
        pass

    @abstractmethod
    def ind2qn(self, ind):
        """

        Parameters
        ----------
        ind :


        Returns
        -------

        """
        pass

    @property
    @abstractmethod
    def orbitals_dict(self):
        """ """
        pass
