function res = isIntersecting_(hyp,S,type,varargin)
% isIntersecting_ - determines if a constrained hyperplane intersects a set
%
% Syntax:
%    res = isIntersecting_(hyp,S)
%    res = isIntersecting_(hyp,S,type)
%
% Inputs:
%    hyp - conHyperplane object
%    S - contSet object
%    type - type of check ('exact' or 'approx')
%
% Outputs:
%    res - true/false
%
% Example: 
%    hyp = conHyperplane(halfspace([1;1],0),[1 0;-1 0],[2;2]);
%    Z = zonotope([0 1 1 0; 0 1 0 1]);
%    P = polytope([-1 -1; 1 0;-1 0; 0 1; 0 -1],[2;3;2;3;2]) + [2;2];
%
%    isIntersecting(hyp,Z)
%    isIntersecting(hyp,P)
% 
%    figure; hold on; xlim([-6,6]); ylim([-6,6]);
%    plot(hyp,[1,2],'b');
%    plot(Z,[1,2],'g');
% 
%    figure; hold on; xlim([-6,6]); ylim([-6,6]);
%    plot(hyp,[1,2],'b');
%    plot(P,[1,2],'r');
%
% Other m-files required: none
% Subfunctions: none
% MAT-files required: none
%
% See also: contSet/isIntersecting, conHyperplane/contains_

% Authors:       Niklas Kochdumper
% Written:       16-May-2018
% Last update:   14-September-2019
%                20-November-2019
% Last revision: 27-March-2023 (MW, rename isIntersecting_)

% ------------------------------ BEGIN CODE -------------------------------
    
    % exact or apprxomiative algorithm
    if strcmp(type,'exact')
        
        if isempty(hyp.C) && ~isa(S,'zonoBundle')
            res = isIntersecting(hyp,S,'approx');
        else

            % init options for linprog (used in all aux_ functions)
            persistent options
            if isempty(options)
                options = optimoptions('linprog','display','off');
            end

            if isa(S,'polytope')
                res = aux_intersectPolyPoly(hyp,S,options);
            elseif isa(S,'interval') || isa(S,'zonotope')
                res = aux_intersectPolyConZono(hyp,conZonotope(S),options);
            elseif isa(S,'conZonotope')
                res = aux_intersectPolyConZono(hyp,S,options);
            elseif isa(S,'zonoBundle')
                res = aux_intersectPolyZonoBundle(hyp,S,options);
            else
                throw(CORAerror('CORA:noExactAlg',S,type));
            end
        end
        
    else
       
        res = true;
        
        % special 'approx' algorithm for zonotope bundles
        if isa(S,'zonoBundle')
        
            % loop over all parallel zonotopes
            for j = 1:length(S.Z)

                Z = S.Z{j};
            
                % check instesection with hyperplane
                lb = supportFunc_(Z,hyp.h.c,'lower');
                ub = supportFunc_(Z,hyp.h.c,'upper');

                if ~contains_(interval(lb,ub),hyp.h.d,'exact',0)
                   res = false;
                   return;
                end

                % check intersection with inequality constraints
                C = hyp.C;
                d = hyp.d;

                for i = 1:size(C,1)
                   bound = supportFunc_(Z,C(i,:)','lower');
                   if bound > d(i)
                      res = false;
                      return;
                   end
                end
            end 
            
        else

            otherOptions = {};
            if isa(S,'polyZonotope') || isa(S,'conPolyZono')
                otherOptions = {'interval',8,1e-3};
            end
            
            % check instesection with hyperplane
            lb = supportFunc_(S,hyp.h.c,'lower',otherOptions{:});
            ub = supportFunc_(S,hyp.h.c,'upper',otherOptions{:});

            if ~contains_(interval(lb,ub),hyp.h.d,'exact',0)
               res = false;
               return;
            end

            % check intersection with inequality constraints
            C = hyp.C;
            d = hyp.d;

            for i = 1:size(C,1)
               bound = supportFunc_(S,C(i,:)','lower',otherOptions{:});
               if bound > d(i)
                  res = false;
                  return;
               end
            end
        end
    end
end


% Auxiliary functions -----------------------------------------------------

function res = aux_intersectPolyPoly(obj1,obj2,options)
% check if a constrained hyperplane {x | c x = d, A x < b} and a polytope
% {x | H x <= k} intersect by solving the following linear program
%
% min sum(y)
%
% s.t. H x - y < k
%      A x - y < b
%            y > 0
%          c x = d 
    
    % construct matrices for inequality constraints
    A = [obj2.A;obj1.C];
    b = [obj2.b;obj1.d];
    
    % construct matrices for equality constraint
    Aeq = obj1.h.c';
    beq = obj1.h.d;
    
    % introduce slack variables y
    m = length(b);
    n = size(A,2);
    
    A = [A,-eye(m);zeros(m,n),-eye(m)];
    b = [b;zeros(m,1)];

    Aeq = [Aeq,zeros(1,m)];
    
    f = [zeros(n,1);ones(m,1)];
    
    % solve the dual problem using linear programming
    [~,val,exitflag] = linprog(f,A,b,Aeq,beq,[],[],options);

    % check if intersection between the two polytopes is empty
    res = true;
    
    if exitflag < 0 || val > eps
        res = false; 
    end
end

function res = aux_intersectPolyConZono(obj1,obj2,options)
% check if a constrained hyperplane {x | h x = d, H x < k}   
% and a constrained zonotope {x = c + G*a | A*a = b, a \in [-1,1]} intersect 
% by solving the following linear program:
%
% min sum(y)
%
% s.t.     cx = d
%      Hx - y < k
%           y > 0
%      c + Ga = x
%          Aa = b
%           a \in [-1,1]

    % get object properties
    h = obj1.h.c;
    d = obj1.h.d;
    
    H = obj1.C;
    k = obj1.d;
    
    c = obj2.c;
    G = obj2.G;
    
    n = length(c);
    m = size(G,2);

    % construct inequality constraints
    if isempty(H)
       H = [h';-h'];
       k = [d;-d];
    else
       H = [H;h';-h'];
       k = [k;d;-d]; 
    end 
    
    p = size(H,1);
    
    A = [H,-eye(p);zeros(p,n),-eye(p)];
    b = [k;zeros(p,1)];
    
    A = blkdiag(A,[eye(m);-eye(m)]);
    b = [b;ones(m,1);ones(m,1)];

    % construct equality constraints
    Aeq = [eye(n),zeros(n,p),-G];
    beq = c;
    
    if ~isempty(obj2.A)
        Aeq = [Aeq;[zeros(size(obj2.A,1),n+p),obj2.A]];
        beq = [beq;obj2.b];
    end
    
    % construct objective function
    f = [zeros(n,1);ones(p,1);zeros(m,1)];
    
    % solve linear program
    [~,val,exitflag] = linprog(f,A,b,Aeq,beq,[],[],options);

    % check if intersection between the two polytopes is empty
    res = true;
    
    if exitflag < 0 || val > eps
        res = false; 
    end
end
    
function res = aux_intersectPolyZonoBundle(obj1,obj2,options)
% check if a constrained hyperplane {x | h x = d, H x < k} and a bundle
% {x = c1 + G1*a|a \in [-1,1]} \cup ... \cup {x = cq + Gq*a | a \in [-1,1]}
% intersect by solving the following linear program:
%
% min sum(y)
%
% s.t.     hx = d
%      Hx - y < k
%           y > 0
%  c1 + G1 a1 = x
%             .
%             .
%             .
%  cq + Gq a2 = x
%   a1,...,aq \in [-1,1]

    % get object properties
    h = obj1.h.c;
    d = obj1.h.d;
    
    H = obj1.C;
    k = obj1.d;

    % construct inequality constraints
    if isempty(H)
       H = [h';-h'];
       k = [d;-d];
    else
       H = [H;h';-h'];
       k = [k;d;-d]; 
    end 
    
    [p,n] = size(H);
    
    A = [H,-eye(p);zeros(p,n),-eye(p)];
    b = [k;zeros(p,1)];
    
    % loop over all parallel zonotopes in the bundle
    Aeq = [];
    beq = [];
    
    for i = 1:obj2.parallelSets
       
        Z = obj2.Z{i};
        c = center(Z);
        G = generators(Z);
        m = size(G,2);
        
        % construct equality constraints 
        Aeq = blkdiag(Aeq,-G);
        beq = [beq;c];
        
        % construct inequality constraints
        A = blkdiag(A,[eye(m);-eye(m)]);
        b = [b;ones(2*m,1)];
    end
    
    temp = repmat(eye(n),[obj2.parallelSets,1]);
    Aeq = [temp,zeros(size(temp,1),p),Aeq];
    
    % construct objective function
    f = zeros(size(Aeq,2),1);
    f(n+1:n+p) = ones(p,1);
    
    % solve linear program
    [~,val,exitflag] = linprog(f,A,b,Aeq,beq,[],[],options);

    % check if intersection between the two polytopes is empty
    res = true;
    
    if exitflag < 0 || val > eps
        res = false; 
    end
end

% ------------------------------ END OF CODE ------------------------------
