function obj = exactTerms(obj,options)
% exactTerms - computes exact Taylor terms of
%    1. an interval matrix exponential
%    2. the series considering the input
%    additionally, the exact square of an interval matrix is computed
%
%    These different tasks are computed in one m-file to save computation
%    time: the for loop has to be executed only once and help functions do
%    not have to be called so often
%
% Syntax:
%    obj = exactTerms(obj,options)
%
% Inputs:
%    obj - linProbSys object
%    options - options struct
%
% Outputs:
%    obj - linProbSys object
%
% Example: 
%
% Other m-files required: none
% Subfunctions: none
% MAT-files required: none
%
% See also: none

% Authors:       Matthias Althoff
% Written:       20-February-2007 
% Last update:   30-April-2007
%                15-June-2016
% Last revision: ---

% ------------------------------ BEGIN CODE -------------------------------

%load data from object structure
A=obj.A;
r=options.timeStep;
n=obj.dim;

%initialize the square of A (sq), the first term of the interval exponential (H), the
%first term of the exponential for the input (Hu)
sq=0*A;
H=0*A;
Hu=0*A;
E=eye(n); %identity matrix

%get diagonal elements of A (diagA)
for i=1:n
    diagA(i,i)=A(i,i);
end

%compute elements of H, Hu and sq
for i=1:n
    %i neq j
    %auxiliary value s
    s=aux_sum(A,i);
    %auxiliary value b
    b=A(i,:); b(i)=0;
    %auxiliary matrix C
    C=E*A(i,i)+diagA;
    %compute non-diagonal elements of sq
    sq(i,:)=b*C+s;
    %compute non-diagonal elements of H
    H(i,:)=b*(E*r+0.5*C*r^2)+0.5*r^2*s;
    %compute non-diagonal elements of Hu
    Hu(i,:)=b*(0.5*E*r^2+1/6*C*r^3)+1/6*r^3*s;  
    
    %i=j
    %compute diagonal elements of sq
    sq(i,i)=sq(i,i)+A(i,i)^2;            
    %auxiliary values for H, Hu
    a_inf=infimum(A(i,i));
    a_sup=supremum(A(i,i));
    %compute diagonal elements for H
    kappa=max(a_inf*r+0.5*a_inf^2*r^2,a_sup*r+0.5*a_sup^2*r^2);
    H(i,i)=H(i,i)+interval(aux_g(A(i,i),r),kappa);
    %compute diagonal elements for Hu
    kappa_u=max(0.5*a_inf*r^2+1/6*a_inf^2*r^3,0.5*a_sup*r^2+1/6*a_sup^2*r^3);
    Hu(i,i)=Hu(i,i)+interval(aux_gu(A(i,i),r),kappa_u);            
    %--------------------------------------------------------------
end

%write results to object structure
obj.taylor.sq=sq;
obj.taylor.H=H;
obj.taylor.Hu=Hu;

end


% Auxiliary functions -----------------------------------------------------

function res = aux_g(a,r)

    if contains(interval(-1/r),a)
        res=-0.5;
    else
        a_inf=infimum(a);
        a_sup=supremum(a);
        res=min(a_inf*r+0.5*a_inf^2*r^2,a_sup*r+0.5*a_sup^2*r^2);
    end

end
    
function [res]=aux_gu(a,r)

    if contains(interval(-3/(2*r)),a)
        res=-9/24*r;
    else
        a_inf=infimum(a);
        a_sup=supremum(a);
        res=min(0.5*a_inf*r^2+1/6*a_inf^2*r^3,0.5*a_sup*r^2+1/6*a_sup^2*r^3);
    end    

end


function s=aux_sum(A,i)
%sum function:
%s=0.5 \sum_{k:k\neq i,k\neq j} a_{ik}a_{kj}t^2

    for k=1:length(A)
        A(k,k)=0;
    end
    s=A(i,:)*A;

end

% ------------------------------ END OF CODE ------------------------------
