function simRes = simulateRandom(HA,params,varargin)
% simulateRandom - simulates a hybrid automaton for random initial points
%    and random inputs
%
% Syntax:
%    res = simulateRandom(HA,params,options)
%
% Inputs:
%    HA - hybridAutomaton object
%    params - system parameters
%    options - settings for random simulation
%       .points - nr of simulation runs
%       .fracVert - fraction of initial states starting from vertices
%       .fracInpVert - fraction of input values taken from the 
%                       vertices of the input set
%       .nrConstInp - number of piecewise constant inputs
%
% Outputs:
%    simRes - simResult object storing simulation results
%
% Other m-files required: none
% Subfunctions: none
% MAT-files required: none
%
% See also: none

% Authors:       Niklas Kochdumper, Mark Wetzlinger
% Written:       03-December-2019 
% Last update:   08-May-2020 (MW, update interface)
% Last revision: ---

% ------------------------------ BEGIN CODE -------------------------------

% input argument validation
options = struct();
if nargin == 3 && isstruct(varargin{1})
    options = varargin{1};
end

% options preprocessing
[params,options] = validateOptions(HA,params,options);

% initialize random inputs
u = cell(size(params.Uloc));

for j = 1:length(u)
    u{j} = randPoint(params.Uloc{j});
end

% determine random points inside the initial set
nrEx = ceil(options.points*options.fracVert);
nrNor = options.points - nrEx;
points = [];
if nrEx > 0
    points = [points, randPoint(params.R0,nrEx,'extreme')]; 
end
if nrNor > 0
    points = [points, randPoint(params.R0,nrNor,'standard')];
end

time = linspace(params.tStart,params.tFinal,options.nrConstInp+1);
startLoc = params.startLoc;

% initialization
simRes = [];

% simulate the hybrid automaton
for i = 1:options.points

    counter = 1;
    locCur = startLoc;
    t = []; x = []; loc = [];
    
    % loop over all input changes
    for g = 1:length(time)-1
    
        % compute random input
        optsSim.u = u;
        
        if counter < options.nrConstInp * options.fracInpVert 
            optsSim.u{locCur} = randPoint(params.Uloc{locCur},1,'extreme');
        else
            optsSim.u{locCur} = randPoint(params.Uloc{locCur});
        end              
        
        % simulate hybrid automaton
        optsSim.x0 = points(:,i);
        if time(g) ~= 0
            optsSim.tStart = time(g);
        end
        optsSim.tFinal = time(g+1);
        optsSim.startLoc = locCur;
        optsSim.finalLoc = params.finalLoc;
        
        [tTemp,xTemp,locTemp] = simulate(HA,optsSim);
        
        % concatenate to one full trajectory
        t = [t; tTemp];
        x = [x; xTemp];
        loc = [loc; locTemp];
        
        % update location and initial point
        points(:,i) = xTemp{end}(end,:)';
        locCur = locTemp(end);
        optsSim = [];
        counter = counter + 1;
    end

    % init simRes object for trajectory
    simRes = [simRes;simResult(x,t,loc)];

end

% ------------------------------ END OF CODE ------------------------------
