function [Y,histroy,ts] = NonnegativePCA_ADMM(A,r,X0,max_iter,time_limit)
% min_{X} trace(X'AX), s.t. X>=0, X is orth
% min_{X} trace(Y'AY) + mu*norm(X,1), X = Y, Y is orth
% L(X,Y,pi) = trace(Y'AY) + mu*norm(X,1) + <X-Y,Pi> + 0.5 beta ||X-Y||_F^2, s.t. Y is orth
mu = 0;


X = X0;

% A = A + abs(laneig((A+A')/2,1))*eye(size(A,1));
HandleObjTrue  = @(X)ComputeObj(X,A,mu);

% histroy(1) = HandleObjTrue(X);
Best_X = X;
Best_X_f = HandleObjTrue(X);
t00000 = clock();
tttt1 = clock();
each = time_limit/150;
scale = 2;
rho = 1*mu;


nr = size(X,1)* size(X,2);
normA = abs(laneig((A+A')/2,1));
rho_max = 2*normA*sqrt(r) + mu*sqrt(nr);
rho_max  = rho_max * 10;


rho = rho_max/(scale^20); %scale = 1;
rho = rho_max ; scale = 2;

rho = 1;
Pi = randn(size(X))*1;
ts = [];
histroy = [];
% Y =  OrthProj(randn(size(X)));
Y =  X;
% beta = 100;

oooo = [];
for iter = 1:max_iter
    
    %     Lag = mdot(Y,A*Y) + mu*norm(X(:),1) + mdot(X-Y,Pi) + 0.5*beta*norm(X-Y,'fro')^2;
    % oooo = [oooo;Lag];
    
    % update X
    % trace(Y'AY) + mu*norm(X,1) + <X-Y,Pi> + 0.5 beta ||X-Y||_F^2
    % mu*norm(X,1) + 0.5 beta ||X-Y + Pi/beta||_F^2
    [X] = max(0,Y-Pi/rho);
 
    % Lag = mdot(Y,A*Y) + mu*norm(X(:),1) + mdot(X-Y,Pi) + 0.5*beta*norm(X-Y,'fro')^2;
    % oooo = [oooo;Lag];
    
    % update Y
    % trace(Y'AY) + mu*norm(X,1) + <X-Y,Pi> + 0.5 beta ||X-Y||_F^2
    % trace(Y'AY) + <X-Y,Pi> + 0.5 beta ||X-Y||_F^2
    % trace(Y'AY) + <X-Y,Pi> + 0.5 beta ||X-Y + Pi/beta||_F^2
    
    % Lag = mdot(Y,A*Y) + mu*norm(X(:),1) + mdot(X-Y,Pi) + 0.5*beta*norm(X-Y,'fro')^2;
    % mdot(Y,A*Y) + 0.5*beta*norm(X-Y+Pi/beta,'fro')^2;
    O = X + Pi/rho;
    % HandleObj_X = @(Y)mdot(Y,A*Y) + 0.5*beta*norm(Y-O,'fro')^2;
    L = 2*normA + rho;
    %      L = 2*normA ;
    
    grad = 2*A*Y + rho*(Y-O);
    
    Y = OrthProj(Y - grad/L);
    
    diff = X - Y;
    Pi = Pi + rho * diff;
    %     rho = rho + 10* norm(diff);
    true_fobj = HandleObjTrue(Y);
    % true_fobj = HandleObjPenalty(X,Y,rho);
    dist = norm(diff,'fro');
    %     fprintf('iter:%d, dist: %f, fobj:%e, rho:%e\n',iter,dist,true_fobj,rho);
 
    histroy =[histroy;true_fobj];
    if(true_fobj< Best_X_f)
        Best_X = Y;
        Best_X_f = true_fobj;
    end
    
    
    %      if(~mod(iter,30))
    %         fprintf('iter:%d, true_fobj:%f, pen_fobj:%f, rho:%f, dist:%.4e, normX2:%.4e \n',...
    %             iter,true_fobj,fff_XY,rho,dist,norm(X,'fro')^2);
    %      end
    
    
    
    
    cur_t = etime(clock(),t00000);
    ts = [ts;cur_t];
    if (cur_t > time_limit),
        break;
    end
    
    if ~mod(iter,30) % && (dist > 1e-3)
        rho = min(1e10,rho * scale);
    end
    
    
    
    %     hhhhh = clock();
    % %     if( etime(hhhhh,tttt1) > each ) && (dist > 1e-3)
    % %         tttt1 = hhhhh;
    % %         rho = rho * scale;
    % % %           rho = rho_max;
    % %     end
    %
    %
    %     if(norm(X,'fro')^2 < r*0.9)
    %         rho = rho * scale;
    % %           rho = rho_max;
    %     end
    %     rho = min(1e8,rho);
    
    
    
end
% plot(histroy)
% dd
% Best_X = OrthProj(Best_X);
