"""
Implements useful utility functions:

- Discretized (gym.space): A gym space corresponding to a gym.spaces.Box space
    which was discretized per dimension. Extends gym.spaces.Discrete to expose
    the number of dimensions and the granularity of the discretization
- DiscretizedActionEnv: Wraps a continuous action environment into a
    discrete action environment by discretizing per-dimension
- ImageEnv: Wraps a Multiworld env to change observation space into images
    (copied primarily from _multiworld)
"""
import numpy as np
import warnings

import gym
from gym.spaces import Dict, Box, Discrete

from her_modules.wrapper_env import ProxyEnv

class Discretized(Discrete):
    def __init__(self, n, n_dims, granularity):
        self.n_dims = n_dims
        self.granularity = granularity
        assert n == granularity ** n_dims
        super(Discretized, self).__init__(n)

class DiscretizedActionEnv(ProxyEnv):
    def __init__(self, wrapped_env, possible_actions=None, granularity=3):
        self.quick_init(locals())
        ProxyEnv.__init__(self, wrapped_env)
        if possible_actions is not None:
            self.base_actions = possible_actions
            n_dims = 1
            granularity = len(self.base_actions)
        else:
            actions_meshed = np.meshgrid(*[np.linspace(lo, hi, granularity)
                                           for lo, hi in zip(self.wrapped_env.action_space.low,
                                                             self.wrapped_env.action_space.high)])
            self.base_actions = np.array([a.flat[:] for a in actions_meshed]).T
            n_dims = self.wrapped_env.action_space.shape[0]

        self.action_space = Discretized(len(self.base_actions), n_dims, granularity)

    def step(self, action):
        return  self.wrapped_env.step(self.base_actions[action])
