 function compare_spca
randn('seed',111);
rand('seed',111);
clc;
clear;
close all;
addpath ../misc
addpath ../SSN_subproblem



n = 1010;
m = 100;                
mu = 31;
r = 5;



                %rng(70);
            
                B = randn(m,n);
                type = 2; % random data matrix
%                 if (type == 1) %covariance matrix
%                     scale = max(diag(B)); % Sigma=A/scale;
%                 elseif (type == 0) %data matrix
%                    B = B - repmat(mean(B,1),m,1);
%                     %                     scale = [];
                    %                     for id = 1:n
                    %                         scale = [scale norm(B(:,id))];
                    %                     end
                    %                     scale = max(scale);
                    %                     B = B/scale;
%                     B = normc(B);
                    %  Sigma=A'*A;
%                 end
  
                
                %B(abs(B)<0.1) = 0;
                HandleObj = @(X) -trace(X'*B'*B*X)+ mu*norm(X,1);

                

                [X] = ExactPenalty(B,mu,r);
                 X1 = orth (X);
 
                %rng(177);
                [phi_init,~] = svd(randn(n,r),0);  % random intialization
                %[phi_init,~] = eigs(H,r);    % singular value initialization
                option_Rsub.F_manpg = -1e10;
                option_Rsub.phi_init = phi_init; option_Rsub.maxiter = 5e2;  option_Rsub.tol = 5e-3;
                option_Rsub.r = r;    option_Rsub.n= n;  option_Rsub.mu=mu;  option_Rsub.type = type;
                [phi_init, F_Rsub,sparsity_Rsub,time_Rsub,maxit_att_Rsub,succ_flag_sub]= Re_sub_grad_spca(B,option_Rsub);
                X2 =  orth(phi_init);

%                 %%%%%  manpg parameter
%                 option_manpg.adap = 0;    option_manpg.type =type;
%                 option_manpg.phi_init = phi_init; option_manpg.maxiter = 20000;  option_manpg.tol =1e-8*n*r;
%                 option_manpg.r = r;    option_manpg.n = n;  option_manpg.mu = mu;
%                 %option_manpg.L = L;
%                 %option_manpg.inner_tol =1e-11;
%                 option_manpg.inner_iter = 100;
%                 %%%%%% soc parameter
%                 option_soc.phi_init = phi_init; option_soc.maxiter = 20000;  option_soc.tol =1e-4;
%                 option_soc.r = r;    option_soc.n = n;  option_soc.mu=mu;
%                 %option_soc.L= L;
% 
%                 option_soc.type = type;
%                 %%%%%% PAMAL parameter
%                 option_PAMAL.phi_init = phi_init; option_PAMAL.maxiter =20000;  option_PAMAL.tol =1e-4;
%                 %option_PAMAL.L = L;   option_PAMAL.V = V;
%                 option_PAMAL.r = r;   option_PAMAL.n = n;  option_PAMAL.mu=mu;   option_PAMAL.type = type;
%                 %    B = randn(d,d)+eye(d,d); B = -B'*B;
%                 [X_manpg, F_manpg,sparsity_manpg,time_manpg,maxit_att_manpg,succ_flag_manpg, lins,in_av]= manpg_orth_sparse(B,option_manpg);
% 
%                 norm(X_manpg'*X_manpg - eye(r),'fro')
% 
%                 X3 =  orth(X_manpg);
                 

%                 option_manpg.F_manpg = F_manpg;
%                 [X_manpg_BB, F_manpg_BB,sparsity_manpg_BB,time_manpg_BB,...
%                     maxit_att_manpg_BB,succ_flag_manpg_BB,lins_adap,in_av_adap]= manpg_orth_sparse_adap(B,option_manpg);
%  
%                  X4 =  orth(X_manpg_BB);
                

                %%%%%% Riemannian subgradient parameter
%                 option_Rsub.F_manpg = F_manpg;
                option_Rsub.phi_init = phi_init; option_Rsub.maxiter = 1e1;  option_Rsub.tol = 5e-3;
                option_Rsub.r = r;    option_Rsub.n= n;  option_Rsub.mu=mu;  option_Rsub.type = 0;
                
                [X_Rsub, F_Rsub,sparsity_Rsub,time_Rsub,...
                    maxit_att_Rsub,succ_flag_sub]= Re_sub_grad_spca(B,option_Rsub);
                %phi_init = X_Rsub;
                  X5 =  orth(X_Rsub);
                  

%                 option_soc.type=0;
%                 option_soc.F_manpg = F_manpg;
%                 option_soc.X_manpg = X_manpg;
%                 option_PAMAL.F_manpg = F_manpg;
%                 option_PAMAL.X_manpg = X_manpg;
%                 [X_Soc, F_soc,sparsity_soc,time_soc,...
%                     soc_error_XPQ,maxit_att_soc,succ_flag_SOC]= soc_spca(B,option_soc);
%                 if succ_flag_SOC == 1
%                     succ_no_SOC = succ_no_SOC + 1;
%                 end
%                 option_PAMAL.F_manpg = F_manpg;
%                 [X_pamal, F_pamal,sparsity_pamal,time_pamal,...
%                     pam_error_XPQ, maxit_att_pamal,succ_flag_PAMAL]= PAMAL_spca(B,option_PAMAL);
%                 if succ_flag_PAMAL ==1
%                     succ_no_PAMAL = succ_no_PAMAL + 1;
%                 end
%                  X6 =  orth(X_Soc);
                
                
fprintf('method 1:%f\n',HandleObj(X1));
fprintf('method 2:%f\n',HandleObj(X2));
% fprintf('method 3:%f\n',HandleObj(X3));
% fprintf('method 4:%f\n',HandleObj(X4));
fprintf('method 5:%f\n',HandleObj(X5));
% fprintf('method 6:%f\n',HandleObj(X6));
%% plot

% figure(1);
% plot(n_set, time.manpg, 'r-','linewidth',1); hold on;
% plot(n_set, time.manpg_BB, 'k-','linewidth',1); hold on;
% plot(n_set, time.soc, 'b--','linewidth',1); hold on;
% plot(n_set, time.pamal, 'c-.','linewidth',2); hold on;
% %plot(n_set, time.Rsub, 'g-.','linewidth',2);
% xlabel('dimenion-n');   ylabel('CPU');
% title(['comparison on CPU: different dimension',',r=',num2str(r),',\mu=',num2str(mu)]);
% legend('ManPG','ManPG-adap','SOC','PAMAL','Location','NorthWest');
% filename_pic1 = ['SPCA_CPU_n',  '_' num2str(r) '_' num2str(mu)  '.eps'];
% saveas(gcf,filename_pic1,'epsc')
%
%
% figure(2)
% semilogy(n_set, Fval.manpg-Fval.best+1e-16, 'r-s','MarkerSize',10,'linewidth',1); hold on;
% semilogy(n_set, Fval.manpg_BB-Fval.best+1e-16, 'k-o','MarkerSize',6,'linewidth',1); hold on;
% semilogy(n_set, Fval.soc-Fval.best+1e-16, 'b-d','MarkerSize',8,'linewidth',1); hold on;
% semilogy(n_set, Fval.pamal-Fval.best+1e-16, 'c-.','MarkerSize',20,'linewidth',1.5); hold on;
% %semilogy(n_set, Fval.Rsub -Fval.best+1e-16, 'g-.','MarkerSize',20,'linewidth',1.5);
% %legend('ManPG','ManPG-BB','SOC','PAMAL','Rsub','Location','NorthWest');
% legend('ManPG','ManPG-adap','SOC','PAMAL','Location','NorthWest');
%
% xlabel('dimenion-n');   ylabel('fucntion value difference');
% title(['comparison on function value difference: different dimension',',r=',num2str(r),',\mu=',num2str(mu)]);
% filename_pic2 = ['SPCA_Fval_n',  '_' num2str(r) '_' num2str(mu)  '.eps'];
% saveas(gcf,filename_pic2,'epsc')
%
% %
% %
% figure(3)
% plot(n_set, Sp.manpg, 'r-s','MarkerSize',10,'linewidth',1); hold on;
% plot(n_set, Sp.manpg_BB, 'k-o','MarkerSize',6,'linewidth',1); hold on;
% plot(n_set, Sp.soc, 'b-d','MarkerSize',8,'linewidth',1); hold on;
% plot(n_set, Sp.pamal, 'c-.','MarkerSize',20,'linewidth',1.5); hold on;
% %plot(n_set, Sp.Rsub, 'g-.','MarkerSize',20,'linewidth',1.5); %hold on;
% xlabel('dimenion-n');   ylabel('sparsity');
% legend('ManPG','ManPG-adap','SOC','PAMAL','Location','SouthEast');
% title(['comparison on sparsity: different dimension',',r=',num2str(r),',\mu=',num2str(mu)]);
% filename_pic3 = ['SPCA_Sparsity_n',  '_' num2str(r)  '_' num2str(mu) '.eps'];
% saveas(gcf,filename_pic3,'epsc')
%
%
% figure(4)
% plot(n_set, iter.manpg, 'r-s','MarkerSize',10,'linewidth',1); hold on;
% plot(n_set, iter.manpg_BB, 'k-o','MarkerSize',6,'linewidth',1); hold on;
% plot(n_set, iter.soc, 'b-d','MarkerSize',8,'linewidth',1); hold on;
% plot(n_set, iter.pamal, 'c-.','MarkerSize',20,'linewidth',1.5); %hold on;
% %plot(n_set, iter.Rsub, 'g-.','MarkerSize',20,'linewidth',1.5); %hold on;
% xlabel('dimenion-n');   ylabel('iter');
% legend('ManPG','ManPG-adap','SOC','PAMAL','Location','NorthWest');
% title(['comparison on iter: different dimension',',r=',num2str(r),',\mu=',num2str(mu)]);
% filename_pic4= ['SPCA_iter_n',  '_'  num2str(r)  '_' num2str(mu) '.eps'];
% saveas(gcf,filename_pic4,'epsc')
%
% close(figure(1));
% close(figure(2));
% close(figure(3));
% close(figure(4));
% filename = ['SPCA_comparison_n_'  num2str(r_set(id_r)) '_'  num2str(mu_set(id_mu)) '.csv'];
% csvwrite( filename, Result);


function [X] = ExactPenalty2(B,mu,r)
A = -B'*B;
%min Tr(X'*A*X) + mu*norm(X,1) s.t. X'*X=Ir. X \in R^{n*r}
% min_{X} trace(X'AX) + mu*norm(X,1), X is orth
% min_{X} trace(X'AX) + mu*norm(X,1), ||X||_F \leq sqrt{r}, svd(X)==r
% min_{X} trace(X'AX) + mu*norm(X,1) + rho (r  -||X||_* ), s.t. ||X||_F \leq sqrt{r}


%  -433.1913
[n]=size(A,1);
maxeigA = max(abs(eig(A)));

maxrho = 2* (mu*sqrt(n*r) + sqrt(r) * maxeigA);
rho = 11;
A = A + eye(n)*maxeigA*2;

X = randn(n,r);
Lip = max(abs(eig(A)))*2;
eps1 = 1e-12;
HandleObjX = @(X)trace(X'*A*X) + mu*norm(X(:),1) + rho*(r  - sum(svd(X)) );
HandleObj = @(X)trace(X'*A*X) + mu*norm(X(:),1);

for iter = 1:500
     if(~mod(iter,50)),rho = min(maxrho,rho * 2);end
    [U,D,V]=svd(X,0);
 

     subgrad = -rho*U*V';

     %     if(~mod(iter,100)) rho = min(10,rho * 2);end
    % trace(X'AX) + mu*norm(X,1) + rho <X,subgrad>, s.t. ||X||_F \leq sqrt{r}
%     old_X = X;
%     Y = X + beta *(X-old_X);
    X = SolveConvexSubproblem(X,A,subgrad,mu,Lip);


    % min_X 0.5||X-A||_{fro}^2 + lambda ||X||_1, s.t. ||X||_F <= sqrt(r)
    his (iter) = HandleObjX(X);
    fprintf('iter:%d, fobj:%f\n',iter,HandleObj(OrthProj(X)));
end

X = orth(X);

 plot(his)

function [X] = ExactPenalty(B,mu,r)
A = -B'*B;
%min Tr(X'*A*X) + mu*norm(X,1) s.t. X'*X=Ir. X \in R^{n*r}
% min_{X} trace(X'AX) + mu*norm(X,1), X is orth
% min_{X} trace(X'AX) + mu*norm(X,1), ||X||_F \leq sqrt{r}, svd(X)==r
% min_{X} trace(X'AX) + mu*norm(X,1) + rho sqrt(r + eps1 -||X||_* ), s.t. ||X||_F \leq sqrt{r}


[n]=size(A,1);
A = A + eye(n)*max(abs(eig(A)));

maxeigA = max(abs(eig(A)));
maxrho = 2 * (mu*sqrt(n*r) + sqrt(r) * maxeigA);
rho = 0.001;


X = proj_l2(10000*randn(n,r),sqrt(r));

Lip = max(abs(eig(A)))*2;
eps1 = 0.1;
HandleObjX = @(X)trace(X'*A*X) + mu*norm(X(:),1) + rho*sqrt(r + eps1 - sum(svd(X)) );
HandleObj = @(X)trace(X'*A*X) + mu*norm(X(:),1);

machine_accuracy = 1e-10;
for iter = 1:1000
    
    normX2 = norm(X,'fro')^2;
    if(abs(normX2-r)>machine_accuracy)
            if(~mod(iter,30)|| normX2<=machine_accuracy),
                rho = min(maxrho,rho * 2);
            end
    end
    [U,D,V]=svd(X,0);

     sss = -2*sqrt(r+eps1-sum(diag(D)));
     subgrad = (rho/sss)*U*V';

     %     if(~mod(iter,100)) rho = min(10,rho * 2);end
    % trace(X'AX) + mu*norm(X,1) + rho <X,subgrad>, s.t. ||X||_F \leq sqrt{r}
%      old_X = X;
%    Y = X + 0.5 *(X-old_X);
    X = SolveConvexSubproblem(X,A,subgrad,mu,Lip);


    % min_X 0.5||X-A||_{fro}^2 + lambda ||X||_1, s.t. ||X||_F <= sqrt(r)
    trueObj=HandleObj(OrthProj(X));his (iter) = trueObj;
    fprintf('iter:%d, fobj:%f, normX:%.2f, rho:%f\n',iter,trueObj,norm(X,'fro')^2,rho);
end

X = orth(X);

 plot(his)
     
function [X] = SolveConvexSubproblem(X,A,B,mu,Lip)
% It solves the following problem:
% min_X trace(X'*A*X) + mu*norm(X(:),1) + mdot(X,B)
% s.t. trace(X'*X) <= r

% addpath('G:\cvx');
[n,r] = size(X);

% cvx_begin quiet
% variables X
% minimize( trace(X'*A*X) + mu*norm(X(:),1) + trace(X'*B) )
% subject to
% trace(X'*X) <= r
% cvx_end
% X
% return;

  HandleObjConvex= @(X)trace(X'*A*X) + mu*norm(X(:),1) + mdot(X,B); % 
     for in = 1:3000
    grad = 2*A*X + B;
    Y = prox_l1_ball (X - grad / Lip,mu/Lip,r);
    if(in>10&&norm(Y-X,'fro')<1e-12),break;end
    X = Y;
%      fs(in)=HandleObjConvex(X);
     end

 

%        plot(fs)
%      pause(1)
% dddd
function [X] = proj_l1_nn2(X,n)
for i=1:n
X(:,i)=BreakPointSearch(ones(n,1),-X(:,i),1,'==') ;
end


function [X] = proj_l1_nn(X,n)
xx = X(:);
[X] = BreakPointSearch(ones(size(xx)),-xx,n,'==');
X = reshape(X,n,n);

function [x] = proj01(x)
x(x<0.5)=0;
x(x>0.5)=1;



function [x] = BreakPointSearch(d,a,s,oper)
% This program solves the following QP: 
% min_x 0.5*x'*diag(d)*x + a'*x
% s.t. sum(x) oper s,  0 <= x <= 1
% oper can be '==', '>=', or '<='
% We assume d > 0.
% Reference:
% [1] Sparsity Constrained Minimization via Mathematical Programming with Equilibrium Constraints, arXiv, 2016.
% [2] A polynomially bounded algorithm for a singly constrained quadratic program. Mathematical Programming, 1980.

n = length(d);assert( s>=0 && s<=n );
y = sort([-a-d;-a],'ascend');
l = 1; r = 2*n; L = n; R = 0;
while 1
    if(r-l==1)
        S = (s-L)/(R-L);
        rho = y(l)+(y(r) - y(l))*S;
        break;
    else
        m = floor(0.5*(l+r));
        t = (-a-y(m))./d;
        C = sum(max(0,min(t,1)));
        if(C==s),
            rho = y(m);break;
        elseif(C>s)
            l = m;L=C;
        elseif(C<s)
            r = m;R=C;
        end
    end
end
if (oper(1)=='>')
rho = min(rho,0); 
elseif(oper(1)=='<')
rho = max(rho,0); 
end
t = (-a-rho)./d; 
x = max(0,min(t,1));



function [x] = proj_l2(a,r)
% min_{x} 0.5||x-a||_2^2, s.t. ||x||_2 <= r

x = a;
norm_x = norm(x,'fro');
 if(norm_x>r)
    x = (x/norm_x) * r;
 end

 
 function w = ProjectOntoSimplex(v, b)
% PROJECTONTOSIMPLEX Projects point onto simplex of specified radius.
%
% w = ProjectOntoSimplex(v, b) returns the vector w which is the solution
%   to the following constrained minimization problem:
%
%    min   ||w - v||_2
%    s.t.  sum(w) = b, w >= 0.
%
%   That is, performs Euclidean projection of v to the positive simplex of
%   radius b.
%
% Author: John Duchi (jduchi@cs.berkeley.edu)

if (b < 0)
  error('Radius of simplex is negative: %2.3f\n', b);
end
v = (v > 0) .* v;
u = sort(v,'descend');
sv = cumsum(u);
rho = find(u > (sv - b) ./ (1:length(u))', 1, 'last');
% theta = max(0, (sv(rho) - b) / rho);
theta = (sv(rho) - b) / rho;
w = max(v - theta, 0);


function [X] = gradinetdescent(A,B)

[n]=size(A,1);
X = OrthProj(randn(n));
Lip = max(abs(eig(A)));
for iter = 1:100,
    [fobj,grad] = ComputeObj(X,A,B);
    X = X - grad / Lip;
    X = OrthProj(X);
    fprintf('iter:%d, fobj:%f\n',iter,fobj);
end

function [fobj,grad] = ComputeObj(X,A,B)
fobj = 0.5*mdot(X,A*X) + mdot(X,B);
grad = A*X + B;

function [x] = proj_l2_nn2(a,r)
% min_{x} 0.5||x-a||_2^2, s.t. ||x||_2 <= r, x>=0
x = a;
x(x<0)=0;
norm_x = norm(x,'fro');
%  if(norm_x>r)
    x = (x/norm_x) * r;
%  end


function R = OrthProj(X)
% min_{R} 0.5 ||R-X||_F^2, 
% s.t. R'R = I_d, when n > d 
%      RR' = I_n, when n <= d
% X: n x d
% 

[n,d] = size(X);
if(n>d)
[P, S, Q] = svd(X,0);
R = P*Q';
else
[P, S, Q] = svd(X',0);
R = P*Q';
R = R';
end


