function X_hat = GD_TC_ES(X_star,Y,Omega,p,eta,k,T)  

% img = double(imread('1.jpg'));
% img = img/255;
% [n1,n2,n3] = size(img);
% r = 50;
% X_star = tsvd_r(img,r);
% X_star = tprod(tprod(U,S),tran(V));
% norm(X_star(:)-img(:))^2/norm(img(:))^2;
% 
% maxP = max(abs(X_star(:)));
% 
% p=0.3;
% 
% Omega_seed = rand(n1,n2,n3);
% Omega = Omega_seed < p;
% Y = Omega.*(X_star +normrnd(0,0,[n1,n2,n3])); % missing
% 
% T = 100;
% eta = 0.001;


rate = 0.95;
[n1,n2,n3] = size(Y);
L = rand(n1,k,n3)/k*1e-10;
R = rand(n2,k,n3)/k*1e-10;
X = tprod(L,tran(R));
[Omega_train,Omega_val,Y_train,Y_val]=split_logical_tensor(Omega,Y,rate);
p=p*rate;


Z = Omega_train.*tprod(L,tran(R)) - Y_train;
loss = [];
error = [];
X_hat = zeros(n1,n2,n3);
for t=1:T
    t1 = tic;
    
    obj = norm(Z, 'fro')^2/(2*p);
    if ~isfinite(obj) 
        break;
    end
    Z = Omega_train.*X - Y_train;
    Z_val = Omega_val.*X-Y_val; 
    L = L - eta/p *tprod(Z,R);
    R = R- eta/p *tprod(tran(Z),L);
    X = tprod(L,tran(R));
    re = norm(X - X_star, 'fro')/norm(X_star, 'fro');
    loss  = [loss,norm(Z_val(:))];
    if t>1
        if loss(t)<=loss(t-1)
            X_hat = X;
        end
    end
    error = [error,re^2];
end
% [~,idx] = min(loss);
% errors = error(idx)
% 
% norm(X_hat - X_star, 'fro')^2/norm(X_star, 'fro')^2


end