function [R,res] = reach(obj,params,varargin)
% reach - computes the reachable set for linear systems
%
% Syntax:
%    R = reach(obj,params)
%    R = reach(obj,params,options)
%    [R,res] = reach(obj,params,spec)
%    [R,res] = reach(obj,params,options,spec)
%
% Inputs:
%    obj - continuous system object
%    params - model parameters
%    options - options for the computation of reachable sets
%    spec - object of class specification 
%
% Outputs:
%    R - object of class reachSet storing the reachable set
%    res - true/false whether specifications are satisfied
%
% Example: 
%
% Other m-files required: none
% Subfunctions: none
% MAT-files required: none
%
% See also: none

% Authors:       Mark Wetzlinger
% Written:       26-June-2019
% Last update:   08-October-2019
%                23-April-2020 (added params)
% Last revision: ---

% ------------------------------ BEGIN CODE -------------------------------

% parse input arguments
spec = [];
if nargin == 2
    options.linAlg = 'adaptive'; 
elseif nargin == 3
    if isa(varargin{1},'specification')
        spec = varargin{1};
        options.linAlg = 'adaptive';
    else
        options = varargin{1};
    end
else
    options = varargin{1};
    spec = varargin{2};
end

% options preprocessing
[params,options] = validateOptions(obj,params,options);

specLogic = [];
if ~isempty(spec)
    [spec,specLogic] = splitLogic(spec);
    if ~isempty(spec)
	    options.specification = spec;
    end
end

% hybrid systems: if invariant is empty set (used to model instant
% transitions), exit immediately with only start set as reachable set
% same goes for tStart = tFinal, which may occur in hybrid systems
if withinTol(params.tStart,params.tFinal) || ...
        ( isfield(options,'specification') && ~isempty(options.specification) ...
        && strcmp(options.specification(1).type,'invariant') ...
        && representsa_(options.specification(1).set,'emptySet',eps))
    timePoint.set{1} = params.R0; timePoint.time{1} = params.tStart;
    timeInt = [];
    res = false;
else

    % decide which reach function to execute by options.linAlg
    if strcmp(options.linAlg,'adaptive')
        [timeInt,timePoint,res] = reach_adaptive(obj,params,options);
    else
        % all below, const. time step sizes
        switch options.linAlg
            case 'standard'
                [timeInt,timePoint,res] = reach_standard(obj,params,options);
            case 'wrapping-free'
                [timeInt,timePoint,res] = reach_wrappingfree(obj,params,options);
            case 'fromStart'
                [timeInt,timePoint,res] = reach_fromStart(obj,params,options);
            case 'decomp'
                [timeInt,timePoint,res] = reach_decomp(obj,params,options);
            case 'krylov'
                [timeInt,timePoint,res] = reach_krylov(obj,params,options);
        end
        % error vector (initial set: no error; error not computed -> NaN)
        timePoint.error = [0; NaN(length(timePoint.set)-1,1)];
        if isfield(timeInt,'set')
            timeInt.error = NaN(length(timeInt.set),1);
        end
    end
end

% create object of class reachSet
R = reachSet.initReachSet(timePoint,timeInt);

% check temporal logic specifications
if res && ~isempty(specLogic)
    res = check(specLogic,R);
end

% ------------------------------ END OF CODE ------------------------------
