% =========================================================================
% This script evaluates several tensor completion algorithms on a set of
% RGB images using different sampling ratios p and noise levels sigma.
%
% For each image and each (p, sigma) pair:
%   1. A random observation mask Omega and Gaussian noise are generated.
%   2. Multiple tensor completion algorithms are applied:
%        - GD with early stopping (GD-ES)
%        - GD selecting the best iterate (GD-best)
%        - TNN (Tensor Nuclear Norm minimization)
%        - GTNN (Generalized TNN, two outputs p3 and p6)
%        - TCTF (Tensor Completion by Tensor Factorization)
%        - UTF (Unfolding-based Tensor Factorization)
%        - TC_RE (Tubal-Rank Estimation model)
%   3. PSNR and RSE are computed relative to the clean ground-truth image.
%   4. All results are written to results.txt in a tab-separated format.
%
% Output:
%   results.txt — Each row corresponds to an image with a specific (p, sigma),
%                  followed by performance metrics for each algorithm.
% =========================================================================

clc; clear all;

addpath(genpath('tsvd_operation'));
addpath(genpath('methods'));

%% Image folder
imgDir = 'figures';
imgFiles = dir(fullfile(imgDir, '*.jpg'));   % Add '*.png' if needed

%% Experiment parameters
p_list      = [0.2, 0.3];
sigma_list  = [0.07, 0.1];

T = 2000;     % Maximum iterations for most methods
k = 100;      % Low-rank dimension parameter (following your original setting)

%% Output result file
resultFile = 'results.txt';
fid = fopen(resultFile, 'w');

% Write header (tab separated)
fprintf(fid, ['image_name\tp\tsigma\t' ...
    'PSNR_GD_ES\tRSE_GD_ES\t' ...
    'PSNR_GD_best\tRSE_GD_best\t' ...
    'PSNR_TNN\tRSE_TNN\t' ...
    'PSNR_GTNN\tRSE_GTNN\t' ...
    'PSNR_TCTF\tRSE_TCTF\t' ...
    'PSNR_UTF\tRSE_UTF\t' ...
    'PSNR_TC_RE\tRSE_TC_RE\n']);

%% Loop over each image
for imgIdx = 1:length(imgFiles)
    imgName = imgFiles(imgIdx).name;
    imgPath = fullfile(imgDir, imgName);
    fprintf('================ Image: %s ================\n', imgName);

    % Read and normalize the image to [0,1]
    img = im2double(imread(imgPath));
    if ndims(img) == 2
        % Convert grayscale 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(:)));

    %% Loop over (p, sigma)
    for ip = 1:length(p_list)
        p = p_list(ip);

        for is = 1:length(sigma_list)
            sigma = sigma_list(is);
            fprintf('---- p = %.2f, sigma = %.2f ----\n', p, sigma);

            % Generate sampling mask and noisy observations
            Omega_seed = rand(n1, n2, n3);
            Omega = Omega_seed < p;
            noise = sigma * randn(n1, n2, n3);
            Y = Omega .* (X_star + noise);   % Missing entries + Gaussian noise

            % -------------------- 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, k, 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, k, 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)]);

            % ------------------------------ TNN ------------------------------
            t4 = tic;
            opts.DEBUG    = 0;
            opts.max_iter = 200;
            [X_TNN, error_TNN, time_TNN] = TNN(Y, Omega, X_star, opts);
            time_TNN = toc(t4);
            PSNR_TNN = PSNR(X_TNN, X_star, maxP);
            RSE_TNN  = norm(X_TNN - X_star, 'fro') / norm(X_star, 'fro');
            disp([' PSNR of TNN : ' num2str(PSNR_TNN) ...
                  ' , time : ' num2str(time_TNN)]);

            % ------------------------------ GTNN ------------------------------
            t3 = tic;
            [Xhat_p3, ~] = GTNN(Y, Omega, T, maxP);
            time_GTNN2 = toc(t3);
            PSNR_GTNN = PSNR(Xhat_p3, X_star, maxP);
            RSE_GTNN  = norm(Xhat_p3 - X_star, 'fro') / norm(X_star, 'fro');
            disp([' PSNR of GTNN : ' num2str(PSNR_GTNN) ...
                  ' , time : ' num2str(time_GTNN2)]);

            % ------------------------------ TCTF ------------------------------
            t5 = tic;
            vec_omega = Omega(:);
            idx = find(vec_omega == 1);
            T_inner = 200;
            [X_TCTF, error_TCTF, time_TCTF] = TCTF(Y, X_star, idx, T_inner);
            time_TCTF2 = toc(t5);
            PSNR_TCTF = PSNR(X_TCTF, X_star, maxP);
            RSE_TCTF  = norm(X_TCTF - X_star, 'fro') / norm(X_star,'fro');
            disp([' PSNR of TCTF : ' num2str(PSNR_TCTF) ...
                  ' , time : ' num2str(time_TCTF2)]);

            % ------------------------------ UTF ------------------------------
            t8 = tic;
            [error_UTF, time_UTF, X_UTF] = UTF(X_star, Y, Omega, k, 500);
            time_UTF2 = toc(t8);
            PSNR_UTF = PSNR(X_UTF, X_star, maxP);
            RSE_UTF  = norm(X_UTF - X_star, 'fro') / norm(X_star, 'fro');
            disp([' PSNR of UTF : ' num2str(PSNR_UTF) ...
                  ' , time : ' num2str(time_UTF2)]);

            % ------------------------------ TC_RE ------------------------------
            DIM = size(Y); %#ok<NASGU>
            maxIter  = 300;
            epsilon  = 1e-15;
            InitialR = round(0.5 * mean(size(Y)));
            fr = 1e-3 * (p); %#ok<NASGU>
            mu = 10 / (sqrt(mean(size(Y)) * (1 - p))) * (max(Y(:)));

            t9 = tic;
            [X_TC_RE, TubalR, TimeRE] = TC_RE( ...
                Y, ...        % incomplete tensor
                Omega, ...    % observed index
                mu, ...       % regularization parameter
                maxIter, ...  % maximum iterations
                epsilon, ...  % tolerance
                InitialR);    % initial tubal-rank
            time_TC_RE = toc(t9);

            PSNR_TC_RE = PSNR(X_TC_RE, X_star, maxP);
            RSE_TC_RE  = norm(X_TC_RE - X_star, 'fro') / norm(X_star, 'fro');
            disp([' PSNR of TC_RE : ' num2str(PSNR_TC_RE) ...
                  ' , time : ' num2str(time_TC_RE)]);

            % ----------------- Write one row of results -----------------
            fprintf(fid, '%s\t%.2f\t%.2f\t', imgName, 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);
            fprintf(fid, '%.6f\t%.6f\t', PSNR_TNN,     RSE_TNN);
            fprintf(fid, '%.6f\t%.6f\t', PSNR_GTNN,    RSE_GTNN);
            fprintf(fid, '%.6f\t%.6f\t', PSNR_TCTF,    RSE_TCTF);
            fprintf(fid, '%.6f\t%.6f\t', PSNR_UTF,     RSE_UTF);
            fprintf(fid, '%.6f\t%.6f\n', PSNR_TC_RE,   RSE_TC_RE);

        end % sigma loop
    end % p loop
end % image loop

fclose(fid);
fprintf('All experiments finished. Results saved to %s\n', resultFile);
