import numpy as np
from Datasets.utils.transforms import TwoCropsTransform
from Datasets.federated_dataset.multi_domain.utils.multi_domain_dataset import MultiDomainDataset
from Datasets.utils.transforms import DeNormalize
import torchvision.transforms as transforms
from torchvision.datasets import ImageFolder, DatasetFolder
from utils.conf import multi_domain_data_path as data_path


class ImageFolder_Custom(DatasetFolder):
    def __init__(self, data_name, root, train=True, transform=None, target_transform=None, subset_train_num=7, subset_capacity=10):
        self.data_name = data_name
        self.root = root
        self.train = train
        self.transform = transform
        self.target_transform = target_transform

        self.imagefolder_obj = ImageFolder(self.root + 'VLCS/' + self.data_name.upper() + '/', self.transform, self.target_transform)

        # split train dataset
        all_data = self.imagefolder_obj.samples
        self.train_data_list = []
        self.test_data_list = []
        for i in range(len(all_data)):
            if i % subset_capacity <= subset_train_num:
                self.train_data_list.append(all_data[i])
            else:
                self.test_data_list.append(all_data[i])

        self.train_data_list = np.array(self.train_data_list)
        self.test_data_list = np.array(self.test_data_list)

    def __len__(self):
        if self.train:
            return len(self.train_data_list)
        else:
            return len(self.test_data_list)

    def __getitem__(self, index):

        if self.train:
            data_list = self.train_data_list
        else:
            data_list = self.test_data_list

        # path = self.imagefolder_obj.samples[used_index_list[index]][0]
        # target = self.imagefolder_obj.samples[used_index_list[index]][1]

        path = data_list[index][0]
        target = data_list[index][1]

        target = int(target)
        img = self.imagefolder_obj.loader(path)
        if self.transform is not None:
            img = self.transform(img)
        if self.target_transform is not None:
            target = self.target_transform(target)
        return img, target


class FLVLCS(MultiDomainDataset):
    NAME = 'VLCS'
    SETTING = 'Domain'

    N_CLASS = 4

    def __init__(self, args, cfg) -> None:
        super().__init__(args, cfg)
        self.domain_list = ['caltech', 'labelme', 'pascal', 'sun']
        self.domain_ratio = {'caltech': cfg.DATASET.domain_ratio, 'labelme': cfg.DATASET.domain_ratio,
                             'pascal': cfg.DATASET.domain_ratio, 'sun': cfg.DATASET.domain_ratio}

        self.train_eval_domain_ratio = {'caltech': cfg.DATASET.train_eval_domain_ratio, 'labelme': cfg.DATASET.train_eval_domain_ratio,
                                        'pascal': cfg.DATASET.train_eval_domain_ratio, 'sun': cfg.DATASET.train_eval_domain_ratio}

        self.train_transform = transforms.Compose(
            [transforms.RandomResizedCrop(224, scale=(0.7, 1.0)),
             transforms.RandomHorizontalFlip(),
             transforms.ColorJitter(brightness=0.4, contrast=0.4, saturation=0.4, hue=0.4),
             transforms.RandomGrayscale(),
             transforms.ToTensor(),
             self.get_normalization_transform()]
        )

        self.test_transform = transforms.Compose(
            [transforms.Resize([224, 224]),
             transforms.ToTensor(),
             self.get_normalization_transform()])

    def get_data_loaders(self, selected_domain_list=[]):
        client_domain_name_list = self.domain_list if selected_domain_list == [] else selected_domain_list

        '''
        Loading the default four domains datasets
        '''
        domain_training_dataset_dict = {}
        domain_testing_dataset_dict = {}
        domain_train_eval_dataset_dict = {}

        if self.cfg.DATASET.aug == 'two_weak':
            train_transform = TwoCropsTransform(self.train_transform, self.train_transform)
            train_val_transform = TwoCropsTransform(self.train_transform, self.train_transform)
        elif self.cfg.DATASET.aug == 'weak':
            train_transform = self.train_transform
            train_val_transform = self.train_transform

        for _, domain in enumerate(self.domain_list):

            train_dataset = ImageFolder_Custom(data_name=domain, root=data_path(), train=True,
                                               transform=train_transform)
            test_dataset = ImageFolder_Custom(data_name=domain, root=data_path(), train=False,
                                              transform=self.test_transform)
            train_eval_dataset = ImageFolder_Custom(data_name=domain, root=data_path(), train=False,
                                                    transform=train_val_transform)

            domain_training_dataset_dict[domain] = train_dataset
            domain_testing_dataset_dict[domain] = test_dataset
            domain_train_eval_dataset_dict[domain] = train_eval_dataset

        self.partition_domain_loaders(client_domain_name_list, domain_training_dataset_dict, domain_testing_dataset_dict, domain_train_eval_dataset_dict)


    @staticmethod
    def get_normalization_transform():
        transform = transforms.Normalize((0.485, 0.456, 0.406),
                                         (0.229, 0.224, 0.225))
        return transform

    @staticmethod
    def get_denormalization_transform():
        transform = DeNormalize((0.485, 0.456, 0.406),
                                (0.229, 0.224, 0.225))
        return transform