function res = isequal(cZ,S,varargin)
% isequal - checks if a constrained zonotope represents the same set as
%    another set
%
% Syntax:
%    res = isequal(cZ,S)
%    res = isequal(cZ,S,tol)
%
% Inputs:
%    cZ - conZonotope object
%    S - contSet object
%    tol - (optional) tolerance
%
% Outputs:
%    res - true/false
%
% Example:
%    c = [0;0]; G = [1 0 1;0 1 1];
%    A = [1 1 1]; b = 4;
%    cZ = conZonotope([c,G],A,b);
%
%    isequal(cZ,cZ + [0;eps],0)
%    isequal(cZ,cZ + [0;eps],1e-14)
%
% Other m-files required: none
% Subfunctions: none
% MAT-files required: none
%
% See also: none

% Authors:       Mark Wetzlinger
% Written:       19-December-2022
% Last update:   16-December-2023 (MW, support comparison to polytope)
%                18-July-2024 (MW, centers may be different)
% Last revision: ---

% ------------------------------ BEGIN CODE -------------------------------

% too many input arguments
if nargin > 3
    throw(CORAerror('CORA:tooManyInputArgs',3));
end

% set default tolerance
tol = setDefaultValues({1e-12},varargin);

% check input arguments
inputArgsCheck({ {cZ,'att','conZonotope'}, ...
                 {S,'att','contSet'}, ...
                 {tol,'att','numeric',{'nonempty','scalar','finite','nonnegative'}}});

% only implemented for conZonotope - conZonotope/zonotope/interval
if ~( isa(S,'conZonotope') || isa(S,'zonotope') || isa(S,'interval') ...
        || isa(S,'polytope'))
    throw(CORAerror('CORA:noops',cZ,S));
end

% second set: polytope
if isa(S,'polytope')
    res = contains(S,cZ,'exact',tol) && contains_(cZ,S,'exact',tol);
    return
end

% second set: zonotope or interval
if isa(S,'zonotope') || isa(S,'interval')
    % first set: conZonotope without constraints
    if isempty(cZ.A) && isempty(cZ.b)
        res = isequal(zonotope(cZ),zonotope(S),tol);
    else
        % constraints given
        res = false;
    end
    return
end

% two constrained zonotopes


% check point cases
[cZ_isPoint, cZ_p] = representsa_(cZ, 'point', tol);
[S_isPoint, S_p] = representsa_(S, 'point', tol);
if cZ_isPoint && S_isPoint
    res = all(withinTol(cZ_p, S_p, tol));
    return
elseif (~cZ_isPoint && S_isPoint) || ...
        (cZ_isPoint && ~S_isPoint)
    res = false;
    return
end

% check empty cases
if representsa_(cZ, 'emptySet', tol) && representsa_(S, 'emptySet', tol)
    res = true;
    return
elseif (~representsa_(cZ, 'emptySet', tol) && representsa_(S, 'emptySet', tol)) || ...
        (representsa_(cZ, 'emptySet', tol) && ~representsa_(S, 'emptySet', tol))
    res = false;
    return
end

% check if both are represented completely equally
if all(withinTol(cZ.c,S.c,tol)) ...
        && compareMatrices([cZ.G; cZ.A],[S.G; S.A],tol) ...
        && compareMatrices([cZ.A cZ.b],[S.A S.b],tol)
    res = true;
    return
end

% general method: check vertices (computationally expensive!)
res = compareMatrices(vertices(cZ),vertices(S),tol);

% ------------------------------ END OF CODE ------------------------------
