% Add paths to CVX

addpath('../cvx')
cvx_setup
cvx_solver mosek
cvx_save_prefs

% Add paths to existing and proposed methods

addpath('../Functions_Library_MSDW/LR')
addpath('../Functions_Library_MSDW/KMM')
addpath('../Functions_Library_MSDW/DWGCS')
addpath('../Functions_Library_MSDW/2SWMDA')
addpath('../Functions_Library_MSDW/MSDRL')
addpath('../Functions_Library_MSDW/CWKMM')
addpath('../Functions_Library_MSDW/DWMSCS')
addpath('../Functions_Library_MSDW/Auxiliar')

%% Fix the training and testing distributions

mu_tr1    = [-1,0];
Sigma_tr1 = 0.5^2*[1,0;0,1];
mu_tr2    = [1,0];
Sigma_tr2 = 0.5^2*[1,0;0,1];
mu_te     = [0,0];
Sigma_te  = 0.5^2*eye(2);
ndistribs = 2;
n1        = 200;
n2        = 200;
t         = 200;

% Initialize the base model

Mdl.intercept = false;
Mdl.fmap = 'linear';
Mdl.loss = 'log';
Mdl.deterministic = true;
Mdl.lambda0 = 0;
Mdl.labels = 2;
Mdl.distribs = 2;
Mdl.sigma = 1;

for rep = 1:100

    xTr{1,1} = mvnrnd(mu_tr1, Sigma_tr1, n1);
    yTr{1,1} = sign(xTr{1}(:,1) .* xTr{1}(:,2));
    yTr{1}(yTr{1} == -1) = 2;

    xTr{2,1} = mvnrnd(mu_tr2,Sigma_tr2,n2);
    yTr{2,1} = sign(xTr{2}(:,1).*xTr{2}(:,2));
    yTr{2}(yTr{2} == -1) = 2;

    xTe = mvnrnd(mu_te,Sigma_te,t);
    yTe = sign(xTe(:,1).*xTe(:,2));
    yTe(yTe == -1) = 2;

    X = normalize([xTr{1};xTr{2};xTe]);
    xTr{1,1} = X(1:n1,:);
    xTr{2,1} = X(n1+1:n1+n2,:);
    xTe = X(n1+n2+1:n1+n2+t,:);

    %% Compute the classification error using ERM LR

    for d = 1:Mdl.distribs
        LR = Mdl;
        LR = LR_parameters(LR, xTr{d}, yTr{d});
        LR = LR_learning(LR, xTr{d});
        LR = LR_prediction(LR, xTe, yTe);
        ErrorLR(rep, d) = LR.error;
    end

    %% Compute the classification error using KMM

    for d = 1:Mdl.distribs
        Kmm = Mdl;
        Kmm = KMM(Kmm, xTr{d}, xTe);
        Kmm = IWMRC_parameters(Kmm, xTr{d}, yTr{d});
        Kmm = IWMRC_learning(Kmm, xTr{d});
        Kmm = MRC_prediction(Kmm, xTe, yTe);
        ErrorKMM(rep,d) = Kmm.error;
    end

    %% Compute the classification error using DW-GCS

    for d = 1:Mdl.distribs
        D = 1 ./ (1 - (0:0.1:0.9)) .^ 2;
        DWGCS = [];
        for l = 1:length(D)
            DWGCS{l}      = Mdl;
            DWGCS{l}.loss = '0-1';
            DWGCS{l}.D    = D(l);
            DWGCS{l}.B    = 1000;
            DWGCS{l}      = DWKMM(DWGCS{l}, xTr{d}, xTe);
            DWGCS{l}      = DWMRC_parameters(DWGCS{l}, xTr{d}, yTr{d}, xTe);
            DWGCS{l}      = DWMRC_learning(DWGCS{l}, xTe);
            RU(l) = DWGCS{l}.min_MRC;
        end
        [~, position] = min(RU);
        DWGCS = DWGCS{position};
        DWGCS = DWMRC_prediction(DWGCS, xTe, yTe);
        ErrorDWGCS(rep, d) = DWGCS.error;
    end

    %% Compute the classification error using ERM LR ideal

    x_te1 = [];
    x_te2 = [];
    y_te1 = [];
    y_te2 = [];
    for i = 1:t
        if xTe(i,1)<0
            x_te1 = [x_te1;xTe(i,:)];
            y_te1 = [y_te1;yTe(i)];
        else
            x_te2 = [x_te2;xTe(i,:)];
            y_te2 = [y_te2;yTe(i)];
        end
    end
    MRC_region1 = Mdl;
    MRC_region1       = MRC_parameters(MRC_region1,xTr{1},yTr{1});
    MRC_region1       = MRC_learning(MRC_region1,xTr{1});
    MRC_region1       = MRC_prediction(MRC_region1,x_te1,y_te1);
    MRC_region1_error(rep,1) = MRC_region1.error;

    MRC_region2 = Mdl;
    MRC_region2       = MRC_parameters(MRC_region2,xTr{2},yTr{2});
    MRC_region2       = MRC_learning(MRC_region2,xTr{2});
    MRC_region2       = MRC_prediction(MRC_region2,x_te2,y_te2);
    MRC_region2_error(rep,1) = MRC_region2.error;
    ErrorLRideal(rep,1) = MRC_region1.error/2+MRC_region2.error/2;

    %% Compute the classification error using 2SW-MDA

    SWMDA      = Mdl;
    SWMDA.H    = zeros(Mdl.distribs, size(xTe, 1));
    for d = 1:Mdl.distribs
        SWMDAd                            = Mdl;
        SWMDAd                            = KMM(SWMDAd, xTr{d}, xTe);
        beta_aux{d,1}                     = SWMDAd.beta;
        SWMDAd                            = IWMRC_parameters(SWMDAd, xTr{d}, yTr{d});
        SWMDAd                            = IWMRC_learning(SWMDAd, xTr{d});
        SWMDAd                            = MRC_prediction(SWMDAd, xTe, yTe);
        SWMDAd.y_pred(SWMDAd.y_pred == 2) = -1;
        SWMDA.H(d,:)                      = SWMDAd.y_pred;
    end
    SWMDA.beta = beta_aux;
    SWMDA      = SWMDA_2weights(SWMDA,xTe);
    for d = 1:Mdl.distribs
        SWMDA.beta{d, 1} = (size(cell2mat(xTr), 1) * SWMDA.w(d) / size(xTr{d}, 1)) * SWMDA.beta{d};
    end
    SWMDA.beta         = cell2mat(SWMDA.beta);
    SWMDA              = IWMRC_parameters(SWMDA, cell2mat(xTr), cell2mat(yTr));
    SWMDA              = IWMRC_learning(SWMDA, cell2mat(xTr));
    SWMDA              = MRC_prediction(SWMDA, xTe, yTe);
    Error2SWMDA(rep,1) = SWMDA.error;

    %% Compute the classification error using MS-DRL

    ErrorMSDRL(rep,1) = MSDRL_learning(Mdl,xTr,yTr,xTe,yTe);

    %% Coompute the classification error using DWC with individual classification rules obtained using KMM

    for d = 1:Mdl.distribs
        clfCWKMM = Mdl;
        clfCWKMM = KMM(clfCWKMM, xTr{d}, xTe);
        clfCWKMM = IWMRC_parameters(clfCWKMM, xTr{d}, yTr{d});
        clfCWKMM = IWMRC_learning(clfCWKMM, xTr{d});
        clfCWKMM = MRC_prediction(clfCWKMM, xTe, yTe);
        CWKMM_h{d,1} = clfCWKMM.h;
    end
    ClasWeightKMM = KDE_MultiSource(xTr,xTe);
    ErrorCWKMM(rep,1) = CWKMM_prediction(CWKMM_h,ClasWeightKMM,yTe,clfCWKMM.deterministic,clfCWKMM.labels);

    %% Compute the classification error using MS-DW

    D = 1 ./ (1 - (0:0.1:0.9)) .^ 2;
    for d = 1:Mdl.distribs
        psix_tr{d} = psi(Mdl, xTr{d});
    end
    psix_te  = psi(Mdl, xTe);
    old_fmap = Mdl.fmap;
    Mdl.fmap = 'linear';

    for k = 1:length(D)
        MSDW{k} = Mdl;
        MSDW{k}.loss = '0-1';
        MSDW{k}.distribs = Mdl.distribs;
        MSDW{k}.D = D(k);
        MSDW{k}.B = 1000;
        MSDW{k}     = MSKMM(MSDW{k},xTr,xTe);
        MSDW{k}     = MSDW_parameters(MSDW{k},psix_tr,yTr,psix_te);
        MSDW{k}     = MSDW_learning(MSDW{k},psix_te);
        min_MSDW(k) = MSDW{k}.min_MRC;
    end
    [~,position] = min(min_MSDW);
    MSDW{position}     = MSDW_prediction(MSDW{position},psix_te,yTe);
    Mdl.fmap = old_fmap;
    ErrorDWMSCS(rep,1) = MSDW{position}.error;
   
end
[~,pos] = min(mean(ErrorLR));
ErrorLR = ErrorLR(:,pos);
[~,pos] = min(mean(ErrorKMM));
ErrorKMM = ErrorKMM(:,pos);
[~,pos] = min(mean(ErrorDWGCS));
ErrorDWGCS = ErrorDWGCS(:,pos);


