function han = plot(pZ,varargin)
% plot - plots an over-approximative projection of a polynomial zonotope
%
% Syntax:
%    han = plot(pZ)
%    han = plot(pZ,dims)
%    han = plot(pZ,dims,type)
%
% Inputs:
%    pZ - polyZonotope object
%    dims - (optional) dimensions for projection
%    type - (optional) plot settings (LineSpec and Name-Value pairs)
%           additional Name-Value pairs:
%               <'Splits',splits> - number of splits for refinement
%
% Outputs:
%    han - handle to the graphics object
%
% Example: 
%    pZ = polyZonotope([0;0],[2 0 1;0 2 1],[0;0],[1 0 3;0 1 1]);
%     
%    figure; hold on;
%    plotRandPoint(pZ);
%    plot(pZ,[1,2],'b','Splits',4);
%
%    figure; hold on;
%    plotRandPoint(pZ);
%    plot(pZ,[1,2],'b','Splits',10);
%
% Other m-files required: none
% Subfunctions: none
% MAT-files required: none
%
% See also: plotRandPoint

% Authors:       Niklas Kochdumper, Mark Wetzlinger, Tobias Ladner
% Written:       29-March-2018
% Last update:   23-June-2020 (MW, harmonize with other plot functions)
%                14-July-2020 (MW, merge with plotFilled)
%                25-May-2022 (TL, 1D Plotting)
%                23-February-2023 (TL, enlarge polygon if 2 regions)
%                23-March-2023 (TL, bugfix Splits=0, clean up)
%                05-April-2023 (TL, clean up using plotPolygon)
%                11-July-2023 (TL, bug fix holes)
%                13-April-2024 (TL, bug fix 3d)
%                17-July-2024 (TL, minor speed up)
% Last revision: 12-July-2023 (TL, restructure)

% ------------------------------ BEGIN CODE -------------------------------

% 1. parse input
[pZ,dims,NVpairs,splits] = aux_parseInput(pZ,varargin{:});

% 2. preprocess
[pZ,dims] = aux_preprocess(pZ,dims);

% 3. plot n-dimensional set
han = aux_plotNd(pZ,dims,NVpairs,splits);

% 4. clear han
if nargout == 0
    clear han;
end

end


% Auxiliary functions -----------------------------------------------------

function [pZ,dims,NVpairs,splits] = aux_parseInput(pZ,varargin)
    % parse input

    % default values for the optional input arguments
    dims = setDefaultValues({[1,2]},varargin);
    
    % read additional name-value pairs
    NVpairs = readPlotOptions(varargin(2:end));
    % read out 'Splits', default value given
    [NVpairs,splits] = readNameValuePair(NVpairs,'Splits','isscalar',10);
    
    % check input arguments
    inputArgsCheck({{pZ,'att','polyZonotope'};
        {dims,'att','numeric',{'nonempty','integer','positive','vector'}}; ...
        {splits,'att','numeric',{'scalar','integer','nonnegative'}} ...
    });
    
    % check dimension
    if length(dims) < 1
        throw(CORAerror('CORA:plotProperties',1));
    elseif length(dims) > 3
        throw(CORAerror('CORA:plotProperties',3));
    end
end

function [pZ,dims] = aux_preprocess(pZ,dims)
    % delete all zero-generators
    pZ = compact_(pZ,'states',eps);
    
    % project to desired dimensions
    pZ = project(pZ,dims);
    dims = 1:dim(pZ);

    % compact in lower-dimensional space
    pZ = compact_(pZ,'all',eps);
end

function han = aux_plotNd(pZ,dims,NVpairs,splits)
    % plot n-dimensional set

    % check if is zonotope
    if representsa_(pZ,'zonotope',eps)
        % plot zonottope
        han = plot(zonotope(pZ), dims, NVpairs{:});
    
    elseif length(dims) == 1 % 1d
        han = plot(interval(pZ, 'split'), 1, NVpairs{:});
    
    elseif length(dims) == 2 % 2d
        han = aux_plot2d(pZ,dims,NVpairs,splits);      
    
    else % 3d
        han = aux_plot3d(pZ,dims,NVpairs,splits);
        
    end
end

function han = aux_plot2d(pZ,dims,NVpairs,splits)
    % plot 2-dimensional set

    % turn off warning from polygon
    warOrig = warning;
    warning('off','all');
    
    % split the polynomial zonotope multiple times to obtain a better 
    % over-approximation of the real shape

    % init with no split
    pZsplit{1} = pZ;
    [polyUnion,V] = aux_getPolygon(pZ);
    polyPrev = polyUnion;
    Vlist{1} = V;
    polylist{1} = polyUnion;

    for i=1:splits
        
        % construct new polygon
        scounter = 0;
        pZnew = cell(1,2*length(pZsplit));
        Vnew = cell(1,2*length(pZsplit));
        polysnew = cell(1,2*length(pZsplit));
        polyUnion = [];
        
        for j = 1:length(pZsplit)
            
            % only split the set if it is part of the boundary
            if any(sum(ismembertol(Vlist{j}',polyPrev.Vertices,1e-12),2) == 2)
                % split set if on boundary

                % split the polynomial zonotope
                res = splitLongestGen(pZsplit{j});
                
                % compute the corresponding polygons
                [poly1,V1] = aux_getPolygon(res{1});
                scounter = scounter + 1;
                pZnew{scounter} = res{1};
                Vnew{scounter} = V1;
                polysnew{scounter} = poly1;
                
                [poly2,V2] = aux_getPolygon(res{2});
                scounter = scounter + 1;
                pZnew{scounter} = res{2};
                Vnew{scounter} = V2;
                polysnew{scounter} = poly2;
                
                % unite the polygons
                poly = union(poly1,poly2);
                
            else
                % no splitting if not on boundary

                % reuse previous result
                scounter = scounter + 1;
                pZnew{scounter} = pZsplit{j};
                Vnew{scounter} = Vlist{j};
                poly = polylist{j};
                polysnew{scounter} = poly;

            end
            
            % unite with previous polygons
            polyUnion = aux_unitePolygons(polyUnion,poly);
        end
        
        % update lists
        pZsplit = pZnew(1:scounter);
        Vlist = Vnew(1:scounter);
        polylist = polysnew(1:scounter);
        polyPrev = polyUnion;
    end

    % restore warning
    warning(warOrig);

    % check vertices
    if size(polyUnion.Vertices,1) > 0
        % vertices are present
        % close regions and holes for correct plotting

        V = polyUnion.Vertices;

        % find nan values
        idxNan = [find(any(isnan(V),2));size(V,1)+1];
        
        % iterate over regions
        regStart = 1;
        xVals = [];
        yVals = [];
        for i=1:length(idxNan)
            % close each region by appending start point 
            xVals = [xVals; V(regStart:idxNan(i)-1,1); V(regStart,1); nan];
            yVals = [yVals; V(regStart:idxNan(i)-1,2); V(regStart,2); nan];

            regStart = idxNan(i)+1;
        end

        % remove nan values at the end
        xVals(end) = [];
        yVals(end) = [];
    else
        % no vertices are present
        % if e.g. all generators are collinear or within tol of polyshape
        V = cell2mat(Vlist)';
        xVals = [V(:,1);V(1,1)];
        yVals = [V(:,2);V(1,2)];
    end
    
    % plot the polygon
    han = plotPolygon([xVals,yVals]', NVpairs{:});
end

function polyUnion = aux_unitePolygons(poly1,poly2)
% unites two polygons

if isempty(poly1)
    polyUnion = poly2;
elseif isempty(poly2)
    polyUnion = poly1;
else
    polyUnion = union(poly1,poly2);
end

if polyUnion.NumRegions >= 2
    % might be due to numeric instability
    % enlargen polygon slightly
    polyUnion_ = polybuffer(polyUnion, 1e-8);
    if polyUnion_.NumRegions < polyUnion.NumRegions
        polyUnion = polyUnion_;
    end
end

end

function [poly,V] = aux_getPolygon(pZ)
    % enclose polynomial zonotope with a polygon

    % zonotope over-approximation
    Z = zonotope(pZ);

    % calculate vertices of zonotope
    V = vertices(Z);

    % transform to 2D polyshape (zonotope/vertices are already 'simple')
    poly = polyshape(V(1,:),V(2,:),'Simplify',false);
end

function han = aux_plot3d(pZ,dims,NVpairs,splits)
    % plot 3-dimensional set

    % split the polynomial zonotope multiple times to obtain a better 
    % over-approximation of the real shape
    pZsplit{1} = pZ;

    for i=1:splits
        pZnew = cell(2*length(pZsplit),1);
        for j=1:length(pZsplit)
            res = splitLongestGen(pZsplit{j});
            pZnew{2*j-1} = res{1};
            pZnew{2*j} = res{2};
        end
        pZsplit = pZnew;
    end
    
    % convert all sets to a zonotope
    Zs = cell(1,numel(pZsplit));
    for i = 1:numel(pZsplit)
        Zs{i} = zonotope(pZsplit{i});
    end

    % plot all sets as one
    han = plotMultipleSetsAsOne(Zs,dims,NVpairs);
end

% ------------------------------ END OF CODE ------------------------------
