import torch
import torch.nn as nn
from functools import partial
import clip
from einops import rearrange, repeat
import kornia


from ldm.modules.x_transformer import Encoder, TransformerWrapper  # TODO: can we directly rely on lucidrains code and simply add this as a reuirement? --> test


class AbstractEncoder(nn.Module):
    def __init__(self):
        super().__init__()

    def encode(self, *args, **kwargs):
        raise NotImplementedError



class ClassEmbedder(nn.Module):
    def __init__(self, embed_dim, n_classes=1000, key='class'):
        super().__init__()
        self.key = key
        self.embedding = nn.Embedding(n_classes, embed_dim)

    def forward(self, batch, key=None):
        if key is None:
            key = self.key
        # this is for use in crossattn
        c = batch[key][:, None]
        c = self.embedding(c)
        return c


class TransformerEmbedder(AbstractEncoder):
    """Some transformer encoder layers"""
    def __init__(self, n_embed, n_layer, vocab_size, max_seq_len=77, device="cuda"):
        super().__init__()
        self.device = device
        self.transformer = TransformerWrapper(num_tokens=vocab_size, max_seq_len=max_seq_len,
                                              attn_layers=Encoder(dim=n_embed, depth=n_layer))

    def forward(self, tokens):
        tokens = tokens.to(self.device)  # meh
        z = self.transformer(tokens, return_embeddings=True)
        return z

    def encode(self, x):
        return self(x)


class BERTTokenizer(AbstractEncoder):
    """ Uses a pretrained BERT tokenizer by huggingface. Vocab size: 30522 (?)"""
    def __init__(self, device="cuda", vq_interface=True, max_length=77):
        super().__init__()
        from transformers import BertTokenizerFast  # TODO: add to reuquirements
        self.tokenizer = BertTokenizerFast.from_pretrained("bert-base-uncased")
        self.device = device
        self.vq_interface = vq_interface
        self.max_length = max_length

    def forward(self, text):
        batch_encoding = self.tokenizer(text, truncation=True, max_length=self.max_length, return_length=True,
                                        return_overflowing_tokens=False, padding="max_length", return_tensors="pt")
        tokens = batch_encoding["input_ids"].to(self.device)
        return tokens

    @torch.no_grad()
    def encode(self, text):
        tokens = self(text)
        if not self.vq_interface:
            return tokens
        return None, None, [None, None, tokens]

    def decode(self, text):
        return text


class BERTEmbedder(AbstractEncoder):
    """Uses the BERT tokenizr model and add some transformer encoder layers"""
    def __init__(self, n_embed, n_layer, vocab_size=30522, max_seq_len=77,
                 device="cuda",use_tokenizer=True, embedding_dropout=0.0):
        super().__init__()
        self.use_tknz_fn = use_tokenizer
        if self.use_tknz_fn:
            self.tknz_fn = BERTTokenizer(vq_interface=False, max_length=max_seq_len)
        self.device = device
        self.transformer = TransformerWrapper(num_tokens=vocab_size, max_seq_len=max_seq_len,
                                              attn_layers=Encoder(dim=n_embed, depth=n_layer),
                                              emb_dropout=embedding_dropout)

    def forward(self, text):
        if self.use_tknz_fn:
            tokens = self.tknz_fn(text)#.to(self.device)
        else:
            tokens = text
        z = self.transformer(tokens, return_embeddings=True)
        return z

    def encode(self, text):
        # output of length 77
        return self(text)


class SpatialRescaler(nn.Module):
    def __init__(self,
                 n_stages=1,
                 method='bilinear',
                 multiplier=0.5,
                 in_channels=3,
                 out_channels=None,
                 bias=False):
        super().__init__()
        self.n_stages = n_stages
        assert self.n_stages >= 0
        assert method in ['nearest','linear','bilinear','trilinear','bicubic','area']
        self.multiplier = multiplier
        self.interpolator = partial(torch.nn.functional.interpolate, mode=method)
        self.remap_output = out_channels is not None
        if self.remap_output:
            print(f'Spatial Rescaler mapping from {in_channels} to {out_channels} channels after resizing.')
            self.channel_mapper = nn.Conv2d(in_channels,out_channels,1,bias=bias)

    def forward(self,x):
        for stage in range(self.n_stages):
            x = self.interpolator(x, scale_factor=self.multiplier)


        if self.remap_output:
            x = self.channel_mapper(x)
        return x

    def encode(self, x):
        return self(x)


class ChannelExpander(nn.Module):
    def __init__(self,
                 n_stages=1,
                 method='bilinear',
                 multiplier=0.5,
                 in_channels=3,
                 out_channels=None,
                 bias=False):
        super().__init__()
        self.n_stages = n_stages
        assert self.n_stages == 0
        
        self.multiplier = multiplier
        self.out_channels_ratio = out_channels #// in_channels
        
        self.remap_output = out_channels is not None
        if self.remap_output:
            print(f'Channel Expander mapping from {in_channels} to {out_channels} channels after resizing.')
            self.channel_mapper = nn.Conv2d(in_channels,out_channels,1,bias=bias)

    def forward(self,x):
        sh = list(x.shape)
        sh[-1] = x.shape[-1] * self.out_channels_ratio
        if self.remap_output:
            x = self.channel_mapper(x).view(sh)
        return x

    def encode(self, x):
        return self(x)


def tokenize_in_batches(text, clip_model, max_length=77, device='cuda'):
    # Split the text into words or phrases
    words = text[0].split(";;")
    segments = []
    current_segment = []
    num = 0
    for kk, word in enumerate(words):
        num_words = word.split(' ')
        num += len(num_words)
        # print(num)
        # Add the word to the current segment
        current_segment.append(word)
        if num > 26:
            # Tokenize the current segment to check its length
            current_tokenized_segment = clip.tokenize(";;".join(current_segment))
            num = 0
            # If the segment exceeds the max length, save the previous segment
            if len(current_tokenized_segment[0]) > max_length - 1:
                segments.append(";;".join(current_segment[:-1]))  # Add the previous segment
                current_segment = [word]  # Start a new segment with the current word

    # Add the last segment
    if current_segment:
        segments.append(";;".join(current_segment))

    # Tokenize all segments
    tokens = [clip.tokenize(segment).to(device) for segment in segments]
    tokens = torch.cat(tokens, 0)
    return tokens



class FrozenCLIPTextEmbedder(nn.Module):
    """
    Uses the CLIP transformer encoder for text.
    """
    def __init__(self, version='ViT-L/14', device="cuda", max_length=77, n_repeat=1, normalize=True, max_extended_length=1000, use_class_list=False):
        super().__init__()
        self.model, _ = clip.load(version, jit=False, device="cpu")
        self.device = device
        self.max_length = max_length
        self.use_class_list = use_class_list
        self.max_extended_length = max_extended_length
        self.n_repeat = n_repeat
        self.normalize = normalize
        
        self.clip_encodings = torch.load('/data/user/CoOp/data/combined_clip_encodings.pt', map_location='cpu')


    def freeze(self):
        self.model = self.model.eval()
        for param in self.parameters():
            param.requires_grad = False

    def is_list_of_lists(self, lst):
        return any(isinstance(i, list) for i in lst)
    
    def is_semi_in(self, lst):
        return any(';;' in i for i in lst)

    def forward(self, text, reduce_token_length=False):      
        if reduce_token_length:
            z = []
            for tex in text:
                tex = tex.split(';;')
                # print(tex)
                tensors_to_concat = [self.clip_encodings[key].view(-1, 768).to(self.device) for key in tex if key in self.clip_encodings]

                # Concatenate the tensors along dimension zero
                if tensors_to_concat:
                    z_b = torch.cat(tensors_to_concat, dim=0)
                else:
                    print("No tensors to concatenate")  
                leng = self.max_extended_length-z_b.shape[0]
                if leng > 0:
                    # nnz = torch.cat([self.clip_encodings[''].to(self.device)]*leng)
                    nnz = torch.zeros(leng, z_b.shape[1], dtype=torch.long).to(self.device)
                    z_b = torch.cat([z_b, nnz])                    
                z.append(z_b)
                
            z = torch.stack(z)
            
        else:
            if self.use_class_list: #';;' in text[0] or '' in text[0]:
                z = []
                for tex in text:
                    tex = tex.split(';;')
                    # print(tex)
                    tensors_to_concat = [self.clip_encodings[key].view(-1, 768).to(self.device) for key in tex if key in self.clip_encodings]

                    # Concatenate the tensors along dimension zero
                    if tensors_to_concat:
                        z_b = torch.cat(tensors_to_concat, dim=0)
                    else:
                        print("No tensors to concatenate")  
                    if self.normalize:
                        z_b = z_b / torch.linalg.norm(z_b, dim=0, keepdim=True)                  
                    z.append(z_b)
                # z = torch.stack(z)
                return z
            else:
                # print(len(text[0]))
                if len(text[0]) > 500:
                    tokens = tokenize_in_batches(text, clip, device=self.device)
                else:
                    tokens = clip.tokenize(text).to(self.device)
                # print("toks", tokens.shape)
                z = self.model.encode_text(tokens)
                
        if self.normalize:
            z = z / torch.linalg.norm(z, dim=1, keepdim=True)
        if len(text[0]) > 500 and not reduce_token_length and not self.use_class_list:
            return z.unsqueeze(0)
        else:
            return z

    def encode(self, text, reduce_token_length=False):
        z = self(text, reduce_token_length)
        if isinstance(z, list):
            return z
        if z.ndim==2:
            z = z[:, None, :]
        if z.shape[1] == 1:
            z = repeat(z, 'b 1 d -> b k d', k=self.n_repeat)
        return z


class FrozenClipImageEmbedder(nn.Module):
    """
        Uses the CLIP image encoder.
        """
    def __init__(
            self,
            model='ViT-L/14',
            jit=False,
            device='cuda' if torch.cuda.is_available() else 'cpu',
            antialias=False,
        ):
        super().__init__()
        self.model, _ = clip.load(name=model, device=device, jit=jit)

        self.antialias = antialias

        self.register_buffer('mean', torch.Tensor([0.48145466, 0.4578275, 0.40821073]), persistent=False)
        self.register_buffer('std', torch.Tensor([0.26862954, 0.26130258, 0.27577711]), persistent=False)

    def preprocess(self, x):
        # normalize to [0,1]
        x = kornia.geometry.resize(x, (224, 224),
                                   interpolation='bicubic',align_corners=True,
                                   antialias=self.antialias)
        x = (x + 1.) / 2.
        # renormalize according to clip
        x = kornia.enhance.normalize(x, self.mean, self.std)
        return x

    def forward(self, x):
        # x is assumed to be in range [-1,1]
        return self.model.encode_image(self.preprocess(x))

