classdef conHyperplane < contSet
% conHyperplane - object constructor for constrained hyperplanes
%
% Description:
%    This class represents constrained hyperplane objects defined as
%    {x | a*x = b, C*x <= d}.
%
% Syntax:
%    obj = conHyperplane(hs)
%    obj = conHyperplane(a,b)
%    obj = conHyperplane(hs,C,d)
%    obj = conHyperplane(a,b,C,d)
%
% Inputs:
%    hs - halfspace object defining the constraint a*x = b
%    a - normal vector of the hyperplane a*x = b
%    b - offset of the hyperplane a*x = b
%    C - constraint matrix for the inequality constraints C*x <= d
%    d - constraint vector for the inequality constraints C*x <= d
%
% Outputs:
%    obj - conHyperplane object
%
% Example:
%    a = [-0.5; -1; 0.1];
%    b = -1;
%    hs = halfspace(a,b);
%    C = [-0.6 0.8 -1.7;...
%          0.6 0.5 -0.8];
%    d = [1; 0.5];
%    hyp = conHyperplane(hs,C,d);
%
% Other m-files required: none
% Subfunctions: none
% MAT-files required: none
%
% See also: halfspace, example_conHyperplane.m

% Authors:       Matthias Althoff, Niklas Kochdumper, Victor Gassmann
% Written:       10-August-2011
% Last update:   22-November-2019 (NK, renamed + added additional constructors)
%                02-May-2020 (MW, added property validation)
%                19-March-2021 (MW, error messages)
%                22-March-2021 (VG, added 1D case)
%                14-December-2022 (TL, property check in inputArgsCheck)
%                16-August-2023 (TL, removed 1D check)
% Last revision: 16-June-2023 (MW, restructure using auxiliary functions)

% ------------------------------ BEGIN CODE -------------------------------


properties (SetAccess = private, GetAccess = public)
    h;  % halfspace
    C;  % constraint matrix
    d;  % constraint offset
end
    
methods
    
    % class constructor
    function obj = conHyperplane(varargin)
        
        % 1. copy constructor
        if nargin == 1 && isa(varargin{1},'conHyperplane')
            obj = varargin{1}; return
        end

        % 2. parse input arguments: varargin -> vars
        [h,C,d] = aux_parseInputArgs(varargin{:});

        % 3. check correctness of input arguments
        aux_checkInputArgs(h,C,d,nargin);

        % 4. assign properties
        obj.h = h;
        obj.C = C;
        obj.d = d;
        
    end
         
    % methods in seperate files
    n = dim(hyp)
    val = distance(hyp,S)
    res = isequal(hyp1,hyp2,varargin)
    res = isHyperplane(hyp)
    P = polytope(hyp)
    han = plot(hyp,varargin)
    hyp = lift_(hyp,N,dims)
    Sproj = projectOnHyperplane(hyp,S)
    [res,S] = representsa_(hyp,type,tol,varargin)
        
    % display functions
    display(hyp)

end

methods (Static = true)
    hyp = generateRandom(varargin) % generate random conHyperplane
end


end


% Auxiliary functions -----------------------------------------------------

function [h,C,d] = aux_parseInputArgs(varargin)
% parse input arguments from user and assign to variables

    % check number of input arguments
    if nargin > 4
        throw(CORAerror('CORA:tooManyInputArgs',4));
    end

    % no input arguments
    if nargin == 0
        h = halfspace(); C = []; d = 0;
        return
    end

    % set default values depending on nargin
    if nargin == 1 || nargin == 3
        [h,C,d] = setDefaultValues({[],[],0},varargin);
    elseif nargin == 2 || nargin == 4
        [a,b,C,d] = setDefaultValues({[],[],[],0},varargin);
        h = halfspace(a,b);
    end

end

function aux_checkInputArgs(h,C,d,n_in)
% check correctness of input arguments

    % only check if macro set to true
    if CHECKS_ENABLED && n_in > 0
            
        inputArgsCheck({ ...
            {h, 'att', 'halfspace'}; ...
            {C, 'att', 'numeric', {'finite', 'matrix'}}; ...
            {d, 'att', 'numeric', {'finite', 'column'}}; ...
        })

        if ~isempty(C)
            if length(h.c) ~= size(C,2)
                throw(CORAerror('CORA:wrongInputInConstructor',...
                    ['The dimension of the constraint matrix does not '...
                    'match the dimension of the halfspace.']));
            elseif size(C,1) ~= length(d)
                throw(CORAerror('CORA:wrongInputInConstructor',...
                    ['The length of the constraint offset does not '...
                    'match the dimension of the constraint matrix.']));
            end
        end
    end
    
end

% ------------------------------ END OF CODE ------------------------------
