import numpy as np
import copy
from Network.network_utils import pytorch_model
from Environment.Normalization.full_norm import compute_norm, compute_reverse, generate_multiobject_norm

MINVAR = 1E-6

def create_dict(value_dict, object_length):
	completed_dict = dict()
	lim_dict = dict()
	all_vals = []
	for n in value_dict.keys():
		pad_length = object_length - value_dict[n][1].shape[0]
		mean = (value_dict[n][1] + value_dict[n][0])/2
		var = abs(value_dict[n][1] - value_dict[n][0])/2 + MINVAR
		low, high = value_dict[n][0], value_dict[n][1]
		if pad_length > 0:
			mean, var = np.concatenate((mean, np.zeros(pad_length)), axis=-1), np.concatenate((var, np.ones(pad_length)), axis=-1) + MINVAR
			low, high = np.concatenate((low, -np.ones(pad_length)), axis=-1), np.concatenate((high, np.ones(pad_length)), axis=-1) + MINVAR
		completed_dict[n] = (mean, var)
		lim_dict[n] = (low, high)
	return completed_dict, lim_dict

class PadNormalizationModule(): # TODO: FULL REWRITE TO HANDLE INSTANCED-COUNTED STATE
	def __init__(self, lim_dict, true_lim_dict, dynamics_dict, object_counts, object_names, pad_object_length, expand_length, all=False):
		# @param inter_names is the ordering of the names used for representation of the environment
		# pad length is the amount of padding to make sure all the objects are in the same space
		# @param append_length is the amount to append to include object 1-hot "positional class" embeddings
		self.lim_dict = lim_dict # the bounds of positions for where an object can be
		self.true_lim_dict = true_lim_dict
		self.dynamics_dict = dynamics_dict # the bounds for the amount an object can change in a single timestep
		# convert min and max in lim_dict to mean and range/2 in norm dict
		self.pad_size = int(pad_object_length)
		self.expand_size = int(expand_length)
		print(lim_dict)
		self.norm_dict, self.pad_lim_dict = create_dict(self.lim_dict, self.pad_size)
		self.dynamics_norm_dict, self.pad_dynamics_dict = create_dict(self.dynamics_dict, self.pad_size)
		self.difference_norm, self.difference_lim = {n: (np.zeros(self.norm_dict[n][0].shape), self.norm_dict[n][1] * 2) for n in object_names}, \
							{n: (self.pad_lim_dict[n][0] - self.pad_lim_dict[n][1], self.pad_lim_dict[n][0] + self.pad_lim_dict[n][1]) for n in object_names}
		self.norm_dict["all"], self.pad_lim_dict["all"] = generate_multiobject_norm(self.norm_dict, object_names, object_counts), generate_multiobject_norm(self.pad_lim_dict, object_names, object_counts)
		self.dynamics_norm_dict["all"], self.pad_dynamics_dict["all"] = generate_multiobject_norm(self.dynamics_norm_dict, object_names, object_counts), generate_multiobject_norm(self.pad_dynamics_dict, object_names, object_counts)
		self.difference_norm["all"], self.difference_lim["all"] = generate_multiobject_norm(self.difference_norm, object_names, object_counts), generate_multiobject_norm(self.difference_lim, object_names, object_counts)
		self.multi_names = set(["target", "dyn", "diff"])
		self.inter_names = object_names
		self.object_counts = object_counts # environment object counts

		# specify different norms
		self.raw_norm, self.raw_lim = (128, 128), (0, 256) # assumes images are ranged 256
		# interaction state norm
		self.append_norm_dict, self.append_lim_dict = create_dict(self.lim_dict, self.expand_size)		
		self.append_norm_dict["all"], self.append_lim_dict["all"] = generate_multiobject_norm(self.append_norm_dict, self.inter_names, object_counts), generate_multiobject_norm(self.append_lim_dict, self.inter_names, object_counts)
		
		self.rel_norm, self.rel_lim = {n: (np.zeros(self.append_norm_dict[n][0].shape), self.append_norm_dict[n][1] * 2) for n in object_names + ['all']}, \
							{n: (np.zeros(self.append_lim_dict[n][0].shape), self.append_lim_dict[n][1] * 2) for n in object_names + ["all"]}

		# gets the appropriate normalization values based on the target
		self.norm_forms = {"target": self.norm_dict, "dyn": self.dynamics_norm_dict, "diff": self.difference_norm,
					"obs": self.append_norm_dict, "rel": self.rel_norm, "raw": self.raw_norm}
		self.lim_forms = {"target": self.pad_lim_dict, "dyn": self.pad_dynamics_dict, "diff": self.difference_lim,
					"obs": self.append_lim_dict, "rel": self.rel_lim, "raw": self.raw_lim}
		# TODO: handle relative norm between block and obstacles (internal relative?)

	def get_mean_var(self, form, names, idxes):
		# logic for partial additional uss an integer after "additional"
		if names is None or names == "all": names = ["all"]
		if type(names) == str: names = [names]
		if len(names) == 1: 
			name = names[0]
			norm = self.norm_forms[form][name]
		else: norm = generate_multiobject_norm(self.norm_forms[form], names, self.object_counts)
		if norm is None: return None, None
		mean = norm[0][idxes] if idxes is not None else norm[0]
		var = norm[1][idxes] if idxes is not None else norm[1]
		return mean, var

	def __call__(self, state, form="target", name=None, idxes=None):
		'''
		takes the normalization of the state, the form decides which norm to use
		valid forms: target, inter, parent, difference, relative
		'''
		mean, var = self.get_mean_var(form, name, idxes)
		if mean is None: return state
		return compute_norm(mean, var, state)

	def reverse(self, state, form = "target", name=None, idxes=None):
		mean, var = self.get_mean_var(form, name, idxes)
		if mean is None: return state
		return compute_reverse(mean, var, state)

	def clip(self, state, form="target", name=None):
		if form in self.multi_names: lims = self.lim_forms[form][name] # can't have an object named anything in the norm dict
		else: lims = self.lim_forms[form]
		return np.clip(state, lims[0], lims[1])