function [e,u_out,ftilde,time,X,Xd] = ObserverSimIF(n,step_size,simtime,X_init,kr,kf,alpha_1,alpha_2,intervals,SNR)
% This function simulates a system using an observer-based controller for trajectory tracking with noisy state measurements.

% Initialize parameters and states
time_length = simtime / step_size;
X = X_init;
fhat_i = zeros(n, 1);
fhat_int = zeros(n, 1);
rhat_i = zeros(n, 1);


for i = 1:time_length
    A = 2;
    B = 0.5;
    f1 = 0.5;
    
    t = (i - 1) * step_size;       % Time   
    time(i) = t;
    
    flag = true;  % Initialize flag to true

    % Loop through each row in the intervals matrix
    for j = 1:size(intervals, 1)
        % Check if t is outside the current interval
        if ~(t <= intervals(j, 1) || t >= intervals(j, 2))
            flag = false;  % Set flag to false if t is inside any interval
            break;  % Exit the loop early if the condition is not met
        end
    end

    if flag
        sigma = 1;
    else
        sigma = 2;
    end
    
    Xi = X(:, i);  
    xi = X(1:n, i);               % State (positions)
    xi_dot = X(n+1:2*n, i);       % State (velocities)
    
    % Desired Trajectory
    xdi = [A*cos(f1*t); A*sin(f1*t); B*f1*t; 0; 0; -0.5*B*f1*t];
    xdi_dot = [-A*f1*sin(f1*t); A*f1*cos(f1*t); B*f1; 0; 0; -0.5*B*f1];
    xdi_ddot = [-A*f1^2*cos(f1*t); -A*f1^2*sin(f1*t); 0; 0; 0; 0];
    
    % Calculate noise standard deviation based on SNR
    signal_power = mean(xi.^2)+ 1e-6;
    noise_power = signal_power / (10^(SNR / 10));
    noise_std_dev = sqrt(noise_power* step_size);
    
    % Add measurement noise to the state
    noisy_xi = xi + noise_std_dev * randn(size(xi));
    noisy_xi_dot = xi_dot + noise_std_dev * randn(size(xi_dot));
    noisy_Xi = [noisy_xi; noisy_xi_dot];

    % Noisy Errors for Control Development
    noisy_ei = noisy_xi - xdi;
    noisy_ei_dot = noisy_xi_dot - xdi_dot;
    noisy_ri = noisy_ei_dot + alpha_1 * noisy_ei;

    % Desired trajectory and errors
    Xdi = [xdi; xdi_dot];
    Xd(:, i) = Xdi;
    ei = xi - xdi;
    e(:, i) = ei;
    
    ei_dot = xi_dot - xdi_dot;
    ri = ei_dot + alpha_1 * ei;
    
    % Observer-based Control
    if sigma == 1
        [~, g] = dynamics(noisy_Xi);
        u = pinv(g) * (xdi_ddot - fhat_i - (alpha_1 + kr) * noisy_ri + (alpha_1^2 - 1) * noisy_ei);

        rhatdot = g * u - xdi_ddot + fhat_i + alpha_2 * (noisy_ri - rhat_i) + alpha_1 * (ri - alpha_1 * noisy_ei);
        fhat_int_dot = (kf * alpha_2 + 1) * (noisy_ri - rhat_i);

        rhat_i = rhat_i + step_size * rhatdot;
        fhat_int = fhat_int + step_size * fhat_int_dot;

        fhat_i = kf * (noisy_ri - rhat_i) + fhat_int;
        Xhati = noisy_Xi;
        Xhat(:, i) = Xhati;
    else
        [~, gh] = dynamics(Xhati);
        u = pinv(gh) * (xdi_ddot);
        xhatdot = Xhati(n+1:2*n);
        xhatdotdot = gh * u;
        Xhatdot = [xhatdot; xhatdotdot];
        Xhati = Xhati + step_size * Xhatdot;
        Xhat(:, i) = Xhati;       
    end
    
    [f, g, J] = dynamics(Xi);
    u_body = J' * u;
    Xdot = [xi_dot; f] + [zeros(n, 1); g * u];
    X(:, i + 1) = Xi + step_size * Xdot;

    ftilde(:, i) = f - fhat_i;
    
    u_out(:, i) = u_body;
    e(:, i) = ei;
end
