function P = polytope(cZ)
% polytope - convert a constrained zonotope object to a polytope
%    according to Prop. 3 in [1]
%
% Syntax:
%    P = polytope(cZ)
%
% Inputs:
%    cZ - conZonotope object
%
% Outputs:
%    P - polytope object
%
% Example: 
%    Z = [0 1 0 1;0 1 2 -1];
%    A = [-2 1 -1]; b = 2;
%    cZ = conZonotope(Z,A,b);
%    P = polytope(cZ);
%
%    figure; hold on;
%    plot(cZ,[1,2],'FaceColor','r');
%    plot(P,[1,2],'g');
%
% Other m-files required: none
% Subfunctions: none
% MAT-files required: none
%
% See also: polytope
%
% References: 
%   [1] J. Scott et al. "Constrained zonotope: A new tool for set-based
%       estimation and fault detection"

% Authors:       Niklas Kochdumper
% Written:       13-May-2018
% Last update:   28-April-2019 (MA, code shortened)      
%                27-December-2020 (NK, use advanced method)
% Last revision: ---

% ------------------------------ BEGIN CODE -------------------------------
    
    % 2-dimensional set -> use efficient algorithm for vertex computation
    if dim(cZ) == 2
        V = vertices(cZ);
        P = polytope(V);
        return;
    end

    % construct lifted zonotope 
    c = cZ.c; 
    G = cZ.G;
    
    Z = zonotope([c;-cZ.b],[G;cZ.A]);
    
    % compute halfspace representation of lifted zonotope
    P = polytope(Z);
    
    % extract halfspace representation for the constrained zonotope
    constraints(P);
    C = P.A;
    d = P.b;
    n = dim(cZ);
    
    % instantiate polytope
    P = polytope(C(:,1:n),d);
    
    % set properties
    P.bounded.val = true;

% ------------------------------ END OF CODE ------------------------------
