function [Rfirst,options] = initReach_Krylov(obj,Rinit,options)
% initReach_Krylov - computes the reachable continuous set for the first
%    time step using Krylov subspace methods
%
% Syntax:
%    [Rfirst,options] = initReach_Krylov(obj,Rinit,options)
%
% Inputs:
%    obj - linearSys object
%    Rinit - initial reachable set
%    options - options for the computation of the reachable set
%
% Outputs:
%    Rfirst - first reachable set 
%    options - options for the computation of the reachable set
%
% Example: 
%
% Other m-files required: none
% Subfunctions: none
% MAT-files required: none
%
% See also: none

% Authors:       Matthias Althoff
% Written:       15-November-2016
% Last update:   29-November-2016
%                19-December-2016
%                03-March-2017
%                02-November-2018
%                24-July-2020 (box inputs removed)
% Last revision: ---

% ------------------------------ BEGIN CODE -------------------------------

% retrieve C matrix for projection
C = obj.C;

%% precompute solutions for state solution
[V_c,H_c,V_g,H_g] = subspace_Krylov(obj,Rinit,options);

% save precomputed solutions and project them
% center
if ~isempty(V_c)
    obj.krylov.state.V_c = V_c;
    obj.krylov.state.V_c_proj = C*V_c;
else
    obj.krylov.state.V_c_proj = [];
end
%generators
if ~isempty(V_g)
    for i=1:length(V_g)
        obj.krylov.state.V_g{i} = V_g{i};
        obj.krylov.state.V_g_proj{i} = C*V_g{i};
    end
else
    obj.krylov.state.V_g_proj = [];
end
obj.krylov.state.H_c = H_c;
obj.krylov.state.H_g = H_g;


% INITIAL SET SOLUTION------------------------------------------------------
% obtain center and generators of initial set
c = sparse(center(Rinit));
G = sparse(generators(Rinit));

% check if center is zero
c_norm = norm(c);
if c_norm == 0
    c_new = c;
    tie_error = 0*c;
else
    % generate reduced order system for the center
    A_red = obj.krylov.state.H_c;
    B_red = obj.krylov.state.V_c'*obj.B;
    linRedSys_c = linearSys('linearReducedDynamics',A_red,B_red); %initialize linear reduced dynamics
    
    % compute exponential matrix
    linRedSys_c = exponential(linRedSys_c,options);
    
    % compute time interval error (tie)
    linRedSys_c = tie(linRedSys_c,options);
    F_tmp = linRedSys_c.taylor.F;
    %tie_error        
    Fmid = center(F_tmp);
    Frad = rad(F_tmp);
    norm_c = norm(c);
    new_tie_error_mid = norm_c*V_c*Fmid(:,1);
    new_tie_error_rad = norm_c*abs(V_c*Frad(:,1));
    tie_error = interval(new_tie_error_mid - new_tie_error_rad, ...
        new_tie_error_mid + new_tie_error_rad);

    
    %Compute new center
    eAt = expm(A_red*options.timeStep);
    c_new = c_norm*V_c*eAt(:,1);
    
    %store error
    %taylor_error_c = linRed_c.taylor.error; %required for input solution
end


% preallocation
dim = length(c);
nrOfGens = length(G(1,:));
g_norm = zeros(nrOfGens,1);
% V_g = obj.krylov.state.V_g;
% H_g = obj.krylov.state.H_g;
G_new = zeros(dim,nrOfGens);

% obtain generators using the Arnoldi iteration
for iGen = 1:nrOfGens
    g_norm(iGen) = norm(G(:,iGen));
    if g_norm(iGen) == 0
        G_new(:,iGen) = G(:,iGen);
    else

        % generate reduced order system for the generators
        A_red_g = H_g{iGen};
        B_red_g = V_g{iGen}'*obj.B;
        %initialize linear reduced dynamics
        linRedSys_g = linearSys('linearReducedDynamics',A_red_g,B_red_g);
        
        % compute exponential matrix
        linRedSys_g = exponential(linRedSys_g,options);
        
        % compute time interval error (tie)
        linRedSys_g = tie(linRedSys_g,options);
        F_tmp = linRedSys_g.taylor.F;
        %tie_error        
        Fmid = center(F_tmp);
        Frad = rad(F_tmp);
        norm_g = norm(G(:,iGen));
        new_tie_error_mid = norm_g*V_g{iGen}*Fmid(:,1);
        new_tie_error_rad = norm_g*abs(V_g{iGen}*Frad(:,1));
        tie_error = tie_error + interval(new_tie_error_mid - new_tie_error_rad, ...
            new_tie_error_mid + new_tie_error_rad);
        
        %Compute new generator
        eAt = expm(A_red_g*options.timeStep);
        G_new(:,iGen) = g_norm(iGen)*V_g{iGen}*eAt(:,1);
        
        %store error
        %taylor_error_g{iGen} = linRed_g{iGen}.taylor.error; %required for input solution
    end
    %iGen
end

% Krylov error computation
if options.krylovError > 2*eps
%     error_normalized = obj.krylov.errorBound_normalized;
%     error = norm(Rinit)*error_normalized;
    error = options.krylovError;
    Krylov_interval = interval(-1,1)*ones(obj.dim,1)*error;
    R_Krylov = zonotope(Krylov_interval);
    
    % initial-state-solution zonotope
    R_initSol = zonotope([c_new,G_new,error*eye(obj.dim)]);
else
    R_Krylov = 0;
    
    % initial-state-solution zonotope
    R_initSol = zonotope([c_new,G_new]);
end

% tie zonotope
R_tie = zonotope(tie_error);

%--------------------------------------------------------------------------

% INPUT SOLUTION-----------------------------------------------------------
obj = inputSolution_Krylov(obj, options);
%--------------------------------------------------------------------------

% load data
Rtrans = obj.taylor.Rtrans;
RV = obj.taylor.RV;
RV_0 = RV + Rtrans;
C_trans = C;
inputCorr = obj.taylor.inputCorr;

% first time step homogeneous solution
Rhom_tp = R_initSol + R_Krylov + Rtrans; %Rtrans not yet changed
Rhom = enclose(Rinit,Rhom_tp) + R_tie + RV + inputCorr;

%reduce zonotope
Rhom=reduce(Rhom,options.reductionTechnique,options.zonotopeOrder);
Rhom_tp=reduce(Rhom_tp,options.reductionTechnique,options.zonotopeOrder);
RV=reduce(RV,options.reductionTechnique,options.zonotopeOrder);

%% precompute solutions for input solution 
[V_c,H_c,V_g,H_g] = subspace_Krylov(obj,RV_0,options);

% save precomputed solutions and project them
% center 
if ~isempty(V_c)
    obj.krylov.input.V_c_proj = C_trans*V_c;
else
    obj.krylov.input.V_c_proj = [];
end
% generators
obj.krylov.input.V_g_proj = []; % initialize so that field 'V_g_proj' always exists
for i=1:length(V_g)
    if ~isempty(V_g{i})
        obj.krylov.input.V_g_proj{i} = C_trans*V_g{i};
    else
        obj.krylov.input.V_g_proj{i} = [];
    end
end

obj.krylov.input.H_c = H_c;
obj.krylov.input.H_g = H_g;

%save homogeneous and particulate solution
options.Rhom_proj = C*Rhom;
options.Rhom_tp_proj = C*Rhom_tp;
options.Raux_proj = C*RV;
options.Rpar_proj = interval(C*RV);
options.Rtrans_proj = C*obj.taylor.Rtrans;
options.R_tie_proj = C*R_tie;
options.inputCorr_proj = C*inputCorr;
options.Rhom_0 = Rhom;
options.RV_0 = RV_0;

%total solution
if isa(Rinit,'polytope')
    %convert zonotopes to polytopes
    Radd = polytope(RV);
    Rtotal = Rhom+Radd;
    Rtotal_tp = Rhom_tp+Radd;
else
    %original computation
    Rtotal = Rhom+RV;
    Rtotal_tp = Rhom_tp+RV;
end

%write results to reachable set struct Rfirst
Rfirst.tp = C*Rtotal_tp;
Rfirst.ti = C*Rtotal;

% ------------------------------ END OF CODE ------------------------------
