function [R,tcomp] = observe_ESO_C(obj,params,options)
% observe_ESO_C - computes the guaranteed state estimation approach of [1].
%
% Syntax:
%    [R,tcomp] = observe_ESO_C(obj,params,options)
%
% Inputs:
%    obj - discrete-time linear system object
%    params - model parameters
%    options - options for the guaranteed state estimation
%
% Outputs:
%    R - observed set of points in time
%    tcomp - computation time
%
% Reference:
%    [1] Nassim Loukkas, John J. Martinez, and Nacim Meslem. Set-
%        membership observer design based on ellipsoidal invariant
%        sets. IFAC-Papers On Line, 50(1):6471-6476, 2017.
%
% Other m-files required: none
% Subfunctions: none
% MAT-files required: none
%
% See also: none

% Authors:       Matthias Althoff
% Written:       04-March-2021
% Last update:   ---
% Last revision: ---

% ------------------------------ BEGIN CODE -------------------------------

% obtain offline gains
[L,P,gamma,lambda] = observe_gain_ESO_C(obj,params,options);

%%initialize computation
%time period
tVec = params.tStart:options.timeStep:params.tFinal-options.timeStep;
timeSteps = length(tVec);

% initialize parameter for the output equation
R = cell(length(tVec),1);

% store first reachable set
Rnext.tp = params.R0;
R{1} = Rnext.tp;
x = Rnext.tp.center;

% compute e_inf, eq. (50) in [1]
n = size(obj.A,1); 
nrOfOutputs = size(obj.C,1);
w_bar = ones(n+nrOfOutputs,1); % without loss of generality, the disturbances are bounded by 1
c_bar = gamma^2/lambda*(w_bar'*w_bar);
e_inf = diag((P/c_bar)^-0.5);

% init mu according to (51) in [1]
% enclosing ball of initial set
r = radius(params.R0);
lambda_max = eigs(P,1);
mu = lambda_max*r^2/c_bar;

tic

%% loop over all time steps
for k = 1:timeSteps-1
    
    % center, eq. (46) in [1]
    x = obj.A*x + obj.B*params.uTransVec(:,k) + obj.c + L*(params.y(:,k) - obj.C*x);
    % update mu, eq. (47) in [1]
    mu = (1-lambda)*mu + lambda;
    
    % upper and lower bound, eq. (48), (49) in [1]
    delta = e_inf*mu^0.5;
    Rnext.tp = interval(x - delta, x + delta);

    % Store result
    R{k+1} = Rnext.tp;
end

tcomp = toc;

% ------------------------------ END OF CODE ------------------------------
