classdef capsule < contSet
% capsule - object constructor for capsules
%
% Description:
%    This class represents capsule objects defined as
%    C := L + S, L = {c + g*a | a \in [-1,1]}, S = {x | ||x||_2 <= r}.
%
% Syntax:
%    obj = capsule(c)
%    obj = capsule(c,g)
%    obj = capsule(c,g,r)
%
% Inputs:
%    c - center
%    g - generator
%    r - radius
%
% Outputs:
%    obj - capsule object
%
% Example:
%    c = [1;2];
%    g = [2;1];
%    r = 1;
%    C = capsule(c,g,r);
%    plot(C);
%
% Other m-files required: none
% Subfunctions: none
% MAT-files required: none
%
% See also: interval, polytope

% Authors:       Matthias Althoff
% Written:       04-March-2019
% Last update:   02-May-2020 (MW, add property validation)
%                19-March-2021 (MW, error messages, remove capsule(r) case)
%                14-December-2022 (TL, property check in inputArgsCheck)
% Last revision: 16-June-2023 (MW, restructure using auxiliary functions)

% ------------------------------ BEGIN CODE -------------------------------

properties (SetAccess = {?contSet, ?matrixSet}, GetAccess = public)
    c; % center
    g; % generator
    r; % radius
end
   
methods

    function obj = capsule(varargin)

        % 0. avoid empty instantiation
        if nargin == 0
            throw(CORAerror('CORA:noInputInSetConstructor'));
        end
        assertNarginConstructor(1:3,nargin);

        % 1. copy constructor
        if nargin == 1 && isa(varargin{1},'capsule')
            obj = varargin{1}; return
        end

        % 2. parse input arguments: varargin -> vars
        [c,g,r] = aux_parseInputArgs(varargin{:});

        % 3. check correctness of input arguments
        aux_checkInputArgs(c,g,r);

        % 4. compute properties
        [c,g,r] = aux_computeProperties(c,g,r);

        % 5. assign properties
        obj.c = c;
        obj.g = g;
        obj.r = r;

        % 6. set precedence (fixed)
        obj.precedence = 60;

    end
end

methods (Static = true)
    C = enclosePoints(varargin) % enclose point cloud by capsule
    C = generateRandom(varargin) % generates random capsule
    C = empty(n) % instantiates an empty capsule
    C = origin(n) % instantiates a capsule representing the origin in R^n
end

methods (Access = protected)
    [abbrev,printOrder] = getPrintSetInfo(S)
end

% plotting
methods (Access = {?contSet})
    han = plot3D(S,varargin);
end

end


% Auxiliary functions -----------------------------------------------------

function [c,g,r] = aux_parseInputArgs(varargin)
% parse input arguments from user and assign to variables

    % assign center
    [c,g,r] = setDefaultValues({[],[],[]},varargin);

end

function aux_checkInputArgs(c,g,r)
% check correctness of input arguments

    % only check if macro set to true
    if CHECKS_ENABLED

        if ~isempty(c) && ~isempty(g) && ~isempty(r)
            inputArgsCheck({ ...
                {c, 'att', {'numeric','numeric'}, {{'nonempty','finite','column'},{'empty'}} }; ...
                {g, 'att', {'numeric','numeric'}, {{'nonempty','finite','column'},{'empty'}} }; ...
                {r, 'att', 'numeric', {'finite', 'nonnegative', 'scalar'}};
            })
        end

        % dimension of center and generator must match
        if ~isempty(g) && ~all(size(c) == size(g))
            throw(CORAerror('CORA:wrongInputInConstructor',...
                'Dimension of center and generator do not match.'));
        end

    end

end

function [c,g,r] = aux_computeProperties(c,g,r)
% compute properties

    if isempty(g)
        if ~isempty(c)
            g = zeros(size(c,1),1);
        else
            g = zeros(size(c,1),0);
        end
    end

    if isempty(r)
        if ~isempty(c)
            r = 0;
        else
            zeros(0,0);
        end
    end

end

% ------------------------------ END OF CODE ------------------------------
