function res = isequal(trans1,trans2,varargin)
% isequal - checks if two transitions are equal by comparing the guard
%    sets, reset functions, target locations, and synchronization labels
%
% Syntax:
%    res = isequal(trans1,trans2)
%    res = isequal(trans1,trans2,tol)
%
% Inputs:
%    trans1 - transition object
%    trans2 - transition object
%    tol - tolerance (optional)
%
% Outputs:
%    res - true/false
%
% Example: 
%    % guard set
%    c = [-1;0]; d = 0; C = [0,1]; D = 0;
%    guard = conHyperplane(c,d,C,D);
%
%    % reset function
%    reset1 = struct('A',[1,0;0,-0.75],'c',[0;0]);
%    reset2 = struct('A',[1,0;0,-0.75],'c',[1;0]);
%
%    % transition
%    trans1 = transition(guard,reset1,1);
%    trans2 = transition(guard,reset2,1);
%
%    % comparison
%    res = isequal(trans1,trans2);
%
% Other m-files required: none
% Subfunctions: none
% MAT-files required: none
%
% See also: none

% Authors:       Mark Wetzlinger
% Written:       26-November-2022
% Last update:   21-May-2023 (MW, extend to arrays)
% Last revision: ---

% ------------------------------ BEGIN CODE -------------------------------

% too many input arguments
if nargin > 3
    throw(CORAerror('CORA:tooManyInputArgs',3));
end

% default values
tol = setDefaultValues({eps},varargin);

% check input arguments
inputArgsCheck({{trans1,'att','transition'};
                {trans2,'att','transition'};
                {tol,'att','numeric',{'scalar','nonnegative','nonnan'}}});

% check array length
if any(size(trans1) ~= size(trans2))
    res = false; return
end

% loop over all individual transition objects
[r,c] = size(trans1);
res = false(r,c);
for i=1:r
    for j=1:c
        res(i,j) = aux_isequal(trans1(i,j),trans2(i,j),tol);
    end
end

end


% Auxiliary functions -----------------------------------------------------

function res = aux_isequal(trans1,trans2,tol)

% easy checks first to facilitate quick exits
res = true;

% target location
if any(size(trans1.target) ~= size(trans2.target)) ...
        || ~all(trans1.target == trans2.target)
    res = false; return
end

% synchronization label
if ~strcmp(trans1.syncLabel,trans2.syncLabel)
    res = false; return
end

% reset function (conversion from nonlinear to linear not checked)

% check fields (either both linear or both nonlinear)
reset1fields = fields(trans1.reset);
reset2fields = fields(trans2.reset);
if ~all(ismember(reset1fields,reset2fields)) || ...
        ~all(ismember(reset2fields,reset1fields))
    res = false; return
end

% established that both reset functions have the same fields
for i=1:length(reset1fields)
    fieldname = reset1fields{i};
    switch fieldname
        case 'A'
            % state mapping
            if any(size(trans1.reset.A) ~= size(trans2.reset.A)) ...
                    || ~all(all(withinTol(trans1.reset.A,trans2.reset.A,tol)))
                res = false; return
            end
        case 'B'
            % input mapping
            if any(size(trans1.reset.B) ~= size(trans2.reset.B)) ...
                    || ~all(all(withinTol(trans1.reset.B,trans2.reset.B,tol)))
                res = false; return
            end
        case 'c'
            % constant offset
            if any(size(trans1.reset.c) ~= size(trans2.reset.c)) ...
                    || ~all(all(withinTol(trans1.reset.c,trans2.reset.c,tol)))
                res = false; return
            end
        case 'f'
            % function handle for nonlinear reset function: instantiate
            % nonlinearSys object for comparison
            if ~isequal(nonlinearSys(@(x,u) trans1.reset.f([x;u]),...
                    trans1.reset.stateDim,trans1.reset.inputDim),...
                    nonlinearSys(@(x,u) trans2.reset.f([x;u]),...
                    trans2.reset.stateDim,trans2.reset.inputDim))
                res = false; return
            end
        case 'stateDim'
            % dimension of state
            if trans1.reset.stateDim ~= trans2.reset.stateDim
                res = false; return
            end
        case 'inputDim'
            if trans1.reset.inputDim ~= trans2.reset.inputDim
                res = false; return
            end
        case 'hasInput'
            if trans1.reset.hasInput ~= trans2.reset.hasInput
                res = false; return
            end
    end
end

% guard set
if any([isnumeric(trans1.guard),isnumeric(trans2.guard)])
    % empty transition object may have .guard = []
    if xor(isnumeric(trans1.guard),isnumeric(trans2.guard))
        res = false; return
    end
elseif ~isequal(trans1.guard,trans2.guard,tol)
    res = false; return
end

end

% ------------------------------ END OF CODE ------------------------------
