function [S,y] = calcSensitivity(obj, x, varargin)
% calcSensitivity - calculates input-output sensitivity matrix at x
%       rows correspond to output neurons, columns to input neurons
%       sensitivity of layer i will be stored in obj.layers{i}.sensitivity
%
% Syntax:
%    [S,y] = calcSensitivity(obj, x)
%
% Inputs:
%    obj - object of class neuralNetwork
%    x - point from input space
%    options - options for neural network evaluation (stored in options.nn)
%    store_sensitivty - {0,1} if sensitivity should be stored in each
%       layer; default: 1
%    idxLayer - indices of layers to be evaluated
%
% Outputs:
%    S - sensitivity matrix at x
%    y - output of the neural network for x
%
% References:
%    [1] Zurada, J. M., et al. "Sensitivity analysis for minimization of
%           input data dimension for feedforward neural network"
%
% Other m-files required: none
% Subfunctions: none
% MAT-files required: none
%
% See also: neuralNetwork, nnHelper/validateNNoptions

% Authors:       ---
% Written:       14-April-2022
% Last update:   16-January-2024 (return neural network output)
% Last revision: ---

% ------------------------------ BEGIN CODE -------------------------------

% parse input
narginchk(2,5);
[options,storeSensitivty,idxLayer] = setDefaultValues(...
    {struct,true,1:length(obj.layers)}, varargin);
options = nnHelper.validateNNoptions(options);

% forward propagation
xs = cell(length(idxLayer), 1);
for i = idxLayer
    xs{i} = x;
    layer_i = obj.layers{i};
    x = layer_i.evaluateNumeric(x, options);
end
y = x;

% calculate sensitivity [1]
% S = ones([1 1 size(y,2)],'like',y);
S = repmat(eye(size(y,1),'like',y),1,1,size(y,2));

% backward propagation
for i = fliplr(idxLayer)
    layer_i = obj.layers{i};
    S = layer_i.evaluateSensitivity(S, xs{i}, options);
    % save sensitivity at layer i for refinement
    if storeSensitivty
        layer_i.sensitivity = S;
    end
end

end

% ------------------------------ END OF CODE ------------------------------
