function res = test_hybridAutomaton_simulate
% test_hybridAutomaton_simulate - test function for simulate
%
% Syntax:
%    res = test_hybridAutomaton_simulate
%
% Inputs:
%    -
%
% Outputs:
%    res - true/false
%
% Other m-files required: none
% Subfunctions: none
% MAT-files required: none
%
% See also: none

% Authors:       Mark Wetzlinger
% Written:       16-May-2023
% Last update:   ---
% Last revision: ---

% ------------------------------ BEGIN CODE -------------------------------

% generate simple automaton:
% - 1st median is guard set
% - right part moves clockwise
% - left part moves counter-clockwise
inv1 = polytope([-1 1]/sqrt(2),0);
inv2 = polytope([1 -1]/sqrt(2),0);

guard = conHyperplane([-1 1]/sqrt(2),0);
reset1 = struct('A',eye(2),'c',[-1;1]);
trans1 = transition(guard,reset1,2);

reset2 = struct('A',eye(2),'c',[1;-1]);
trans2 = transition(guard,reset2,1);

% clockwise motion
dyn1 = linearSys([0 1; -1 0],1);
% counter-clockwise motion
dyn2 = linearSys([0 -1; 1 0],1);

loc1 = location('clockwise',inv1,trans1,dyn1);
loc2 = location('counter-clockwise',inv2,trans2,dyn2);
HA = hybridAutomaton([loc1;loc2]);

% model parameters
params.x0 = [2;-2];
params.startLoc = 1;
params.finalLoc = 0;
params.tFinal = 3;

% simulate trajectory
[t,x,loc] = simulate(HA,params); 

% must be cell-arrays
res = iscell(t) && iscell(x) && isnumeric(loc);
% must be of same length
res(end+1,1) = length(t) == length(x) && length(t) == length(loc);
% check whether points are contained in respective invariant
res(end+1,1) = all(contains_(inv1,vertcat(x{loc==1})','exact',1e-10));
res(end+1,1) = all(contains_(inv2,vertcat(x{loc==2})','exact',1e-10));
% time before and after jumps must be the same, but state not
for i=1:length(t)-1
    res(end+1,1) = withinTol(t{i}(end),t{i+1}(1));
    res(end+1,1) = ~compareMatrices(x{i}(end,:)',x{i+1}(1,:)');
end


% automaton with different number of states per location
inv1 = polytope([-1 1]/sqrt(2),0);
inv2 = polytope([1 -1 0]/sqrt(2),0);

guard1 = conHyperplane([-1 1]/sqrt(2),0);
guard2 = conHyperplane([-1 1 0]/sqrt(2),0);
reset1 = struct('A',[1 0; 0 1; 0 0],'c',[-1;1;1]);
trans1 = transition(guard1,reset1,2);

reset2 = struct('A',[1 0 0; 0 1 0],'c',[1;-1]);
trans2 = transition(guard2,reset2,1);

% clockwise motion
dyn1 = linearSys([0 1; -1 0],1);
% counter-clockwise motion
dyn2 = linearSys([0 -1 0; 1 0 0; 0 0 0],1);

loc1 = location('clockwise',inv1,trans1,dyn1);
loc2 = location('counter-clockwise',inv2,trans2,dyn2);
HA = hybridAutomaton([loc1;loc2]);

% model parameters
params.x0 = [2;-2];
params.startLoc = 1;
params.finalLoc = 0;
params.tFinal = 3;

% simulate trajectory
[t,x,loc] = simulate(HA,params); 

% must be cell-arrays
res(end+1,1) = iscell(t) && iscell(x) && isnumeric(loc);
% must be of same length
res(end+1,1) = length(t) == length(x) && length(t) == length(loc);
% check whether points are contained in respective invariant
res(end+1,1) = all(contains_(inv1,vertcat(x{loc==1})','exact',1e-10));
res(end+1,1) = all(contains_(inv2,vertcat(x{loc==2})','exact',1e-10));
% time before and after jumps must be the same, but state not
for i=1:length(t)-1
    res(end+1,1) = withinTol(t{i}(end),t{i+1}(1));
    res(end+1,1) = ~compareMatrices(x{i}(end,:)',x{i+1}(1,:)');
end


% combine results
res = all(res);

% ------------------------------ END OF CODE ------------------------------
