% ========================================================================
% This script performs tensor video completion experiments on a list of
% video .mat files (each containing a 3D tensor X).
%
% For each video and each (p, sigma) setting:
%   1. Partial observations are generated by random subsampling with rate p
%      and additive Gaussian noise with std = sigma.
%   2. Several tensor completion algorithms are applied:
%         - GD-ES (Gradient Descent with Early Stopping)
%         - GD-best (Best iterate along GD trajectory)
%         - TNN   (Tensor Nuclear Norm minimization)
%         - GTNN  (Generalized TNN, two versions)
%         - UTF   (Unified Tensor Factorization)
%         - TC_RE (Tensor Completion via Rank Estimation)
%   3. For each method, PSNR and RSE are computed.
%   4. The first reconstructed frame is saved as an image.
%   5. All quantitative results are appended into a txt file.
%
% Output:
%   - results_video.txt : Tab-separated quantitative results
%   - recover_video/*   : First-frame reconstructions for visualization
%
% ========================================================================

clc; clear all;

addpath(genpath('tsvd_operation'));
addpath(genpath('methods'));
addpath(genpath('videos'));

%% List of video MAT files to process (each contains variable X)
videoList = {'akiyo.mat'};

%% Experiment parameters
p_list      = [0.3, 0.4];
sigma_list  = [0.03, 0.05];

T = 4000;   % Maximum number of iterations for iterative methods
k = 50;     % Rank  parameter

%% Output result file
resultFile = 'results_video.txt';
fid = fopen(resultFile, 'w');

% Write header (tab-separated)
fprintf(fid, ['video_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_GTNN1\tRSE_GTNN1\t' ...
    'PSNR_UTF\tRSE_UTF\t' ...
    'PSNR_TC_RE\tRSE_TC_RE\n']);

%% Root folder for saving recovered first-frame images
recoverRoot = 'recover_video';
if ~exist(recoverRoot, 'dir')
    mkdir(recoverRoot);
end

%% Loop over each video
for vIdx = 1:length(videoList)
    matName = videoList{vIdx};
    fprintf('================ Video: %s ================\n', matName);

    % Create a separate folder for this video's outputs
    [~, videoBase, ~] = fileparts(matName);
    videoSaveDir = fullfile(recoverRoot, videoBase);
    if ~exist(videoSaveDir, 'dir')
        mkdir(videoSaveDir);
    end

    % Load MAT file (variable should be named X)
    data = load(matName, 'X');
    X = double(data.X);

    % Select first 30 frames (or fewer if video is shorter)
    maxFrames = min(30, size(X, 3));
    X_star = X(:, :, 1:maxFrames);
    X_star = X_star / max(abs(X_star(:)));     % Normalize
    [n1, n2, n3] = size(X_star);
    maxP = max(abs(X_star(:)));

    % Save original first frame
    save_first_frame(X_star, videoSaveDir, videoBase, 0, 0, 'Original');

    %% Sweep over p and 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 mask and noisy observation
            Omega_seed = rand(n1, n2, n3);
            Omega = Omega_seed < p;
            noise = sigma * randn(n1, n2, n3);
            Y = Omega .* (X_star + noise);   % Missing + Gaussian noise

            % Save observed first frame
            save_first_frame(Y, videoSaveDir, videoBase, p, sigma, 'Observed');

            %% -------- GD with Early Stopping --------
            eta = 0.0003;
            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), ' RE ', num2str(RSE_GD_ES)]);

            save_first_frame(X_GD_ES, videoSaveDir, videoBase, p, sigma, 'GD_ES');

            %% -------- GD best iterate --------
            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), ' RE ', num2str(RSE_GD_best)]);

            save_first_frame(X_GD_best, videoSaveDir, videoBase, p, sigma, 'GD_best');

            %% -------- TNN --------
            t4 = tic;
            opts.DEBUG    = 0;
            opts.max_iter = T;
            [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), ' RE ', num2str(RSE_TNN)]);

            save_first_frame(X_TNN, videoSaveDir, videoBase, p, sigma, 'TNN');

            %% -------- GTNN & GTNN1 --------
            t3 = tic;
            [Xhat_p3, Xhat_p6] = 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');
            PSNR_GTNN1 = PSNR(Xhat_p6, X_star, maxP);
            RSE_GTNN1  = norm(Xhat_p6 - X_star, 'fro') / norm(X_star, 'fro');

            disp([' PSNR of GTNN  : ' num2str(PSNR_GTNN)  ...
                  ' , time : ' num2str(time_GTNN2), ' RE ', num2str(RSE_GTNN)]);
            disp([' PSNR of GTNN1 : ' num2str(PSNR_GTNN1) ...
                  ' , time : ' num2str(time_GTNN2), ' RE ', num2str(RSE_GTNN1)]);

            save_first_frame(Xhat_p3, videoSaveDir, videoBase, p, sigma, 'GTNN_p3');
            save_first_frame(Xhat_p6, videoSaveDir, videoBase, p, sigma, 'GTNN_p6');

            %% -------- 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), ' RE ', num2str(RSE_UTF)]);

            save_first_frame(X_UTF, videoSaveDir, videoBase, p, sigma, 'UTF');

            %% -------- 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, ...       % L1 regularization parameter
                maxIter, ...  % maximum iterations
                epsilon, ...  % tolerance
                InitialR);    % initial 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), ' RE ', num2str(RSE_TC_RE)]);

            save_first_frame(X_TC_RE, videoSaveDir, videoBase, p, sigma, 'TC_RE');

            %% Write result line into txt
            fprintf(fid, '%s\t%.2f\t%.2f\t', matName, 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_GTNN1,   RSE_GTNN1);
            fprintf(fid, '%.6f\t%.6f\t', PSNR_UTF,     RSE_UTF);
            fprintf(fid, '%.6f\t%.6f\n', PSNR_TC_RE,   RSE_TC_RE);

        end % sigma
    end % p
end % video loop

fclose(fid);
fprintf('All experiments finished. Results saved to %s\n', resultFile);


%% ========================================================================
%% Helper Function: Save First Frame of a Reconstructed Video
%% ========================================================================
function save_first_frame(X_rec, saveDir, videoBase, p, sigma, algoName)
    if isempty(X_rec)
        return;
    end

    frame = X_rec(:,:,1);

    % Clip into [0, 1] range for visualization
    frame = max(min(frame, 1), 0);

    % Construct output filename:  video_p0.30_sigma0.05_GD_ES.png
    fileName = sprintf('%s_p%.2f_sigma%.2f_%s.png', ...
        videoBase, p, sigma, algoName);

    savePath = fullfile(saveDir, fileName);
    imwrite(frame, savePath);
end
