% =========================================================================
% This script evaluates two tensor completion algorithms (GD-ES and GD-best)
% on a set of images using different rank parameters R.
%
% For each image in the folder:
%   1. The image is normalized and a random sampling mask Omega is generated.
%   2. Gaussian noise with standard deviation sigma is added.
%   3. The same observation Y is used for all R values to ensure comparability.
%   4. GD_TC_ES (gradient descent with early stopping) and
%      GD_TC_best (best iterate along GD trajectory) are applied.
%   5. PSNR and Relative Error (RSE) are computed for each R.
%   6. Results are saved in a tab-separated txt file.
%
% Output:
%   results.txt — containing one row per (image, R) with evaluation metrics.
% =========================================================================

clc; clear all;

addpath(genpath('tsvd_operation'));
addpath(genpath('methods'));

%% Image folder
imgDir = 'figures';
imgFiles = dir(fullfile(imgDir, '*.jpg'));   % Add '*.png' if needed

%% Fixed parameters
p      = 0.2;
sigma  = 0.05;

T = 2000;                     % Maximum number of GD iterations
R_list = [50, 75, 100, 125, 150];   % Rank values R to test

%% Output result file
resultFile = 'results.txt';
fid = fopen(resultFile, 'w');

% Write header line (tab-separated)
fprintf(fid, ['image_name\tR\tp\tsigma\t' ...
    'PSNR_GD_ES\tRSE_GD_ES\t' ...
    'PSNR_GD_best\tRSE_GD_best\t']);

%% Process each image
for imgIdx = 1:length(imgFiles)
    imgName = imgFiles(imgIdx).name;
    imgPath = fullfile(imgDir, imgName);
    fprintf('================ Image: %s ================\n', imgName);

    % Load image and normalize to [0,1]
    img = im2double(imread(imgPath));
    if ndims(img) == 2
        % Convert grayscale image to 3-channel RGB for consistency
        img = repmat(img, [1, 1, 3]);
    end

    [n1, n2, n3] = size(img);
    X_star = img;
    maxP = max(abs(X_star(:)));

    % Generate one mask + noise per image,
    % reuse same Y for all R to ensure fair comparison across ranks
    Omega_seed = rand(n1, n2, n3);
    Omega = Omega_seed < p;
    noise = sigma * randn(n1, n2, n3);
    Y = Omega .* (X_star + noise);   % Missing + Gaussian-noisy observation

    %% Loop over different rank values R
    for iR = 1:length(R_list)
        R = R_list(iR);
        fprintf('---- R = %d, p = %.2f, sigma = %.2f ----\n', R, p, sigma);

        %% GD with early stopping
        eta = 0.001;
        t1 = tic;
        [X_GD_ES, err_GD_ES, loss_GD_ES] = GD_TC_ES(X_star, Y, Omega, p, eta, R, T);
        time_GD_ES = toc(t1);
        PSNR_GD_ES = PSNR(X_GD_ES, X_star, maxP);
        RSE_GD_ES  = norm(X_GD_ES - X_star, 'fro') / norm(X_star, 'fro');
        disp([' PSNR of GD_ES : ' num2str(PSNR_GD_ES) ...
              ' , time : ' num2str(time_GD_ES)]);

        %% GD selecting best iterate
        eta = 0.001;
        t2 = tic;
        [X_GD_best, error_GD] = GD_TC_best(X_star, Y, Omega, p, eta, R, T);
        time_GD_best = toc(t2);
        PSNR_GD_best = PSNR(X_GD_best, X_star, maxP);
        RSE_GD_best  = norm(X_GD_best - X_star, 'fro') / norm(X_star, 'fro');
        disp([' PSNR of GD_best : ' num2str(PSNR_GD_best) ...
              ' , time : ' num2str(time_GD_best)]);

        %% Write a line of results
        fprintf(fid, '%s\t%d\t%.2f\t%.2f\t', imgName, R, p, sigma);
        fprintf(fid, '%.6f\t%.6f\t', PSNR_GD_ES,   RSE_GD_ES);
        fprintf(fid, '%.6f\t%.6f\t', PSNR_GD_best, RSE_GD_best);

    end % end for each R
end % end for each image

fclose(fid);
fprintf('All experiments finished. Results saved to %s\n', resultFile);
