function res = test_nonlinearSysDT_simulate
% test_nonlinearSysDT_simulate - unit test for simulate
%
% Syntax:
%    res = test_nonlinearSysDT_simulate
%
% Inputs:
%    -
%
% Outputs:
%    res - true/false
%
% Other m-files required: none
% Subfunctions: none
% MAT-files required: none
%
% See also: none

% Authors:       Laura Luetzow
% Written:       23-June-2023
% Last update:   ---
% Last revision: ---

% ------------------------------ BEGIN CODE -------------------------------

res = true(0);

% define linearSysDT ------------------------------------------------------

% stable system matrix: n x n
A = [0.9810    0.0143    0.0262   -0.0140
    0.0079    1.0133    0.0267    0.0416
    0.0029   -0.0054    0.9680    0.0188
    0.0503   -0.0242    0.0411    0.9877];
n_x = length(A);

% input matrix: n x m
B = 0.25 * [-2 0 3;
            2 1 0;
            0 0 1;
            0 -2 1];
n_u = size(B,2);

% output matrix: q x n
C = [1 1 0 0;
     0 -0.5 0.5 0];
n_y = 2;

% feedthrough matrix: q x m
D = [0 0 1;
     0 0 0];

% disturbance matrix: n x n
E = eye(n_x);

% noise matrix: q x q
F = eye(n_y);

% initialize linearSysDT-object
dt = 0.05;
sys_linDT = linearSysDT(A,B,[],C,D,[],E,F,dt);

% initialize nonlinearSysDT-object
B_new = [sys_linDT.B eye(n_x) zeros(n_x,n_y)]; % from combining inputs and disturbances in u
D_new = [sys_linDT.D zeros(n_y,n_x), eye(n_y)]; % from combining inputs and disturbances in u
fun = @(x,u) sys_linDT.A*x + B_new * u;
out_fun = @(x,u) sys_linDT.C*x + D_new * u;
sys = nonlinearSysDT(fun, dt, n_x, n_u+n_x+n_y, out_fun, n_y);

% model parameters --------------------------------------------------------

% time horizon
params.tFinal = 1;
dt_steps = params.tFinal / dt;
% initial set
params.R0 = zonotope(10*ones(n_x,1),0.5*diag(ones(n_x,1)));
% initial state
params.x0 = center(params.R0);
% vectors for u, w, and v
u_rand = randn(n_u, dt_steps+1);
w_rand = randn(n_x, dt_steps);
v_rand = randn(n_y, dt_steps+1);


% simulate ----------------------------------------------------------------

params_NL = params; 

% simulate linear system
params.u = u_rand;
[t_lin_u, x_lin_u, ~, y_lin_u] = simulate(sys_linDT,params);

params.w = w_rand;
params.v = v_rand;
[t_lin_uwv, x_lin_uwv, ~, y_lin_uwv] = simulate(sys_linDT,params);

% simulate nonlinear system
params_NL.u = [u_rand; zeros(n_x+n_y, dt_steps+1)];
[t_nonlin_u, x_nonlin_u, ~, y_nonlin_u] = simulate(sys,params_NL);

params_NL.u = [u_rand; [w_rand zeros(n_x, 1)]; v_rand];
[t_nonlin_uwv, x_nonlin_uwv, ~, y_nonlin_uwv] = simulate(sys,params_NL);

res(end+1,1) = isequal(t_lin_u, t_nonlin_u, t_lin_uwv, t_nonlin_uwv);
res(end+1,1) = sum(abs(y_lin_u - y_nonlin_u),'all') < 1e-6;
res(end+1,1) = sum(abs(y_lin_uwv - y_nonlin_uwv),'all') < 1e-6;
res(end+1,1) = sum(abs(x_lin_u - x_nonlin_u),'all') < 1e-6;
res(end+1,1) = sum(abs(x_lin_uwv - x_nonlin_uwv),'all') < 1e-6;

% combine results
res = all(res);

% ------------------------------ END OF CODE ------------------------------
