function [X_full,zTraj] = simulateRRT(obj,Rcont,params,options)
% simulateRRT - simulates a DAE system using rapidly exploring random trees
%    (partially deterministic sampling)
%
% Syntax:
%    [X_full,zTraj] = simulateRRT(obj,Rcont,params,options)
%
% Inputs:
%    obj - DAE object
%    Rcont - reachable set
%    params - struct containing the parameter that define the reachability problem
%    options - struct containing settings for the random simulation
%       .points number of random initial points (positive integer)
%       .vertSamp flag that specifies if random initial points, inputs, and parameters
%           are sampled from the vertices of the corresponding sets (0 or 1)
%       .stretchFac stretching factor for enlarging the reachable sets during execution
%           of the algorithm (scalar ¿ 1).
%
% Outputs:
%    obj - linearSys object
%    t - time vector
%    x - state vector
%    index - returns the event which has been detected
%
% Example: 
%
% Other m-files required: none
% Subfunctions: none
% MAT-files required: none
%
% See also: none

% Authors:       Matthias Althoff
% Written:       06-December-2011
% Last update:   ---
% Last revision: ---

% ------------------------------ BEGIN CODE -------------------------------

% new options preprocessing
[params,options] = validateOptions(obj,params,options);

%nrOfSamples
nrOfSamples = 10;

%initialize
X_sample_size = 2*rad(interval(options.X_sample));
normMatrix = diag(1./X_sample_size);
xCenter = params.x0;

%possible extreme inputs
U_input_mat = vertices(params.U);
nrOfExtrInputs = length(U_input_mat(1,:));

%obtain vertices 
V = vertices(params.R0);
X_full = V;
Y_full = params.y0*ones(1,length(X_full(1,:)));

%timeSteps
timeSteps = ceil(params.tFinal/options.timeStep);

%init
xTraj = cell(nrOfSamples,timeSteps);

for iStep = 1:timeSteps
    
    iStep
    
    for iSample = 1:nrOfSamples
        %iSample
        
        %init
        x_full = cell(1,nrOfExtrInputs);
        x_end = [];     

        %sample
        x_sample = randPoint(options.X_sample) + xCenter;

        %nearest neighbor and selected state
        ind = aux_nearestNeighbor(x_sample,X_full,xCenter,normMatrix);
        ind
        x0 = X_full(:,ind);
        y0 = Y_full(:,ind);

        %simulate model to find out best input
        for iInput = 1:nrOfExtrInputs
            %set input
            par_options.u = params.uTrans + U_input_mat(:,iInput);
            %[x0_ss,y0] = aux_initialState(obj, x0, y0, par_options.u);
            %simulate
            z{iInput} = aux_par_simulate(obj,par_options,options.timeStep,x0,y0);
            %convert to states
            x_end(:,iInput) = z{iInput}(end,1:obj.dim);   
            y_end(:,iInput) = z{iInput}(end,obj.dim+1:obj.dim + obj.nrOfConstraints); 
        end

        %nearest neighbor and selected state
        ind_input = aux_nearestNeighbor(x_sample,x_end,xCenter,normMatrix);
        ind_input

        %add selected state 
        X_next(:,iSample) = x_end(:,ind_input);
        Y_next(:,iSample) = y_end(:,ind_input);

        %add trajectory
        zTraj{iSample,iStep} = z{ind_input}';
    end
    %update variables
    X_full = X_next;
    Y_full = Y_next;
end

end


% Auxiliary functions -----------------------------------------------------

function ind = aux_nearestNeighbor(x_sample,X,c,normMatrix)

%normalize set of points
X_delta = X - c*ones(1,length(X(1,:)));
X_norm = normMatrix*X_delta;

%normalize sample point
x_delta = x_sample - c;
x_norm = normMatrix*x_delta;

%norm of distance
X_rel = X_norm - x_norm*ones(1,length(X_norm(1,:)));
norm_val = vecnorm(X_rel); %compute 2-norm

%find index with smallest norm
[~,ind] = min(norm_val);

end


%simulate
function z_full = aux_par_simulate(obj,par_options,timeStep,x0,y0)

par_options.timeStep = timeStep;
par_options.tStart = 0;
par_options.x0 = x0;
par_options.y0 = y0;
[~,z_full] = simulate(obj,par_options);

end


%steady state solution
function [x0,y0] = aux_initialState(obj, x0, y0, u0)

while ~converged
    k = obj.dynFile(x0, y0, u0);
    l = obj.conFile(x0, y0, u0);
    [A,~,C,D,~,F] = jacobian(x0, y0, u0);

    %new steady state solution
    M = [A C; D F];
    delta_z = M\(-[k;l]);

    %update steady state solution
    delta_y = delta_z((length(x0)+1) : end);
    
    %check convergence
    if norm(delta_y)<1e-10
        break
    end
   
    %update y0
    y0 = y0 + delta_y;
end

end


% ------------------------------ END OF CODE ------------------------------
