function[f_vec2,g_vec2,time_vec2,x,acc_vec] = Alg_Projection(fun_f,grad_f,grad_g,fun_g,TSA,param,x0)

% The update rule is given by: 
% 
% x_{k+1} = \Pi_{\mathcal{Z}}(x_k - \gamma_k(\nabla g(x_k)+\eta_k \nabla f(x_k)))
% 
% We choose \gamma_k = \gamma_0/sqrt{k+1} and \eta_k = \eta_0/(k+1)^0.25

eta_0 = 1e-3;
gamma_0 = 1e-3;
lambda=param.lam;

f_vec2 = [];
g_vec2 = [];
time_vec2 = [];
acc_vec = [];
x = x0;

%% algorithm
maxiter = param.maxiter;
maxtime = param.maxtime;
tic;
for k = 1 : maxiter 
    eta_k = (eta_0)/(k+1)^0.25;
    gamma_k = gamma_0/sqrt(k+1+10);
    % Descent step
    x = x - gamma_k*(grad_g(x)+eta_k*(grad_f(x)));
    % Projection step
    %x = ProjectOntoL1Ball(x,lambda);
    x = ProjectOntoL2Ball(x,lambda, zeros(size(x)));
    cpu_t2 = toc;
    f_vec2 = [f_vec2;fun_f(x)];
    g_vec2 = [g_vec2;fun_g(x)];
    time_vec2 = [time_vec2;cpu_t2];
    % test set accuracy
    acc_vec = [acc_vec;TSA(x)];
    if cpu_t2>maxtime
        break
    end
end