% =========================================================================
% Script: verify_k_video.m
%
% Purpose:
%   This script evaluates the performance of two tensor completion methods:
%       - GD with early stopping (GD\_ES)
%       - GD selecting the best iterate (GD\_best)
%   on several video tensors for different rank parameters R.
%
%   For each video in `videoList`:
%     1. Load the video tensor X (size: n1 x n2 x nFrames) from a .mat file.
%     2. Use the first up-to-30 frames as the ground-truth tensor X_star.
%     3. Normalize X_star into [0,1].
%     4. Generate a single sampling mask Omega and Gaussian noise for *all* R,
%        so comparisons across R are fair.
%     5. For each rank R in R_list:
%          - Run GD_TC_ES (early stopping) and GD_TC_best.
%          - Compute PSNR and relative error (RSE) with respect to X_star.
%          - Print results to the console.
%          - Save all numeric results into a tab-separated text file.
%
% Output:
%   verify_k_video.txt, containing rows of:
%       video_name, R, p, sigma, PSNR_GD_ES, RSE_GD_ES, PSNR_GD_best, RSE_GD_best
% =========================================================================

clc; clear all;

addpath(genpath('D:\SynchroFile\project\ICLR_rebuttal\tsvd_operation'));
addpath(genpath('methods'));
addpath(genpath('videos'));

%% List of video .mat files to process (each contains variable X)
videoList = {'suzie.mat','miss-america.mat','akiyo.mat'};

%% Fixed parameters
p      = 0.3;            % sampling ratio
sigma  = 0.05;           % noise standard deviation
T      = 4000;           % maximum number of iterations
R_list = [40, 50, 60, 70, 80];   % different rank parameters

%% Output result file
resultFile = 'verify_R_video.txt';
fid = fopen(resultFile, 'w');

% Write header line (tab separated)
fprintf(fid, ['video_name\tR\tp\tsigma\t' ...
              'PSNR_GD_ES\tRSE_GD_ES\t' ...
              'PSNR_GD_best\tRSE_GD_best\n']);

%% Loop over each video
for vIdx = 1:length(videoList)
    matName = videoList{vIdx};
    fprintf('================ Video: %s ================\n', matName);

    % Load .mat file; variable X has size n1 x n2 x nFrames
    data = load(matName, 'X');
    X = double(data.X);

    % Use the first 30 frames (or fewer if not enough frames)
    maxFrames = min(30, size(X, 3));
    X_star = X(:, :, 1:maxFrames);

    % Normalize to [0,1] using the maximum absolute value
    X_star = X_star / max(abs(X_star(:)));
    [n1, n2, n3] = size(X_star);
    maxP = max(abs(X_star(:)));

    % For fair comparison across different R:
    % generate Omega and noise only once per video
    fprintf('Generating observation mask and noise: p = %.2f, sigma = %.2f\n', p, sigma);
    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

    %% For each R, run GD_ES and GD_best
    for iR = 1:length(R_list)
        R = R_list(iR);
        fprintf('---- R = %d ----\n', R);

        %% GD with early stopping (GD_ES)
        eta = 0.0002;
        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 (R = ' num2str(R) ') : ' num2str(PSNR_GD_ES) ...
              ' , time : ' num2str(time_GD_ES) ' , RE ' num2str(RSE_GD_ES)]);

        %% GD selecting best iterate (GD_best)
        eta = 0.0002;
        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 (R = ' num2str(R) ') : ' num2str(PSNR_GD_best) ...
              ' , time : ' num2str(time_GD_best) ' , RE ' num2str(RSE_GD_best)]);

        %% Write one result line into the txt file
        fprintf(fid, '%s\t%d\t%.2f\t%.2f\t', matName, R, p, sigma);
        fprintf(fid, '%.6f\t%.6f\t', PSNR_GD_ES,   RSE_GD_ES);
        fprintf(fid, '%.6f\t%.6f\n', PSNR_GD_best, RSE_GD_best);

    end % R loop
end % video loop

fclose(fid);
fprintf('All experiments finished. Results saved to %s\n', resultFile);
