/*************************************************************************\

  Copyright 2001 The University of North Carolina at Chapel Hill.
  All Rights Reserved.

  Permission to use, copy, modify OR distribute this software and its
  documentation for educational, research and non-profit purposes, without
  fee, and without a written agreement is hereby granted, provided that the
  above copyright notice and the following three paragraphs appear in all
  copies.

  IN NO EVENT SHALL THE UNIVERSITY OF NORTH CAROLINA AT CHAPEL HILL BE
  LIABLE TO ANY PARTY FOR DIRECT, INDIRECT, SPECIAL, INCIDENTAL, OR
  CONSEQUENTIAL DAMAGES, INCLUDING LOST PROFITS, ARISING OUT OF THE
  USE OF THIS SOFTWARE AND ITS DOCUMENTATION, EVEN IF THE UNIVERSITY
  OF NORTH CAROLINA HAVE BEEN ADVISED OF THE POSSIBILITY OF SUCH
  DAMAGES.

  THE UNIVERSITY OF NORTH CAROLINA SPECIFICALLY DISCLAIM ANY
  WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
  MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.  THE SOFTWARE
  PROVIDED HEREUNDER IS ON AN "AS IS" BASIS, AND THE UNIVERSITY OF
  NORTH CAROLINA HAS NO OBLIGATIONS TO PROVIDE MAINTENANCE, SUPPORT,
  UPDATES, ENHANCEMENTS, OR MODIFICATIONS.

  The authors may be contacted via:

  US Mail:             S. Ehmann, M. Lin
                       Department of Computer Science
                       Sitterson Hall, CB #3175
                       University of N. Carolina
                       Chapel Hill, NC 27599-3175

  Phone:               (919) 962-1749

  EMail:               geom@cs.unc.edu
                       ehmann@cs.unc.edu
                       lin@cs.unc.edu

\**************************************************************************/


//////////////////////////////////////////////////////////////////////////////
//
// SWIFT_pair.h
//
// Description:
//      Classes to manage a pair of objects.
//
//////////////////////////////////////////////////////////////////////////////

#ifndef _SWIFT_PAIR_H_
#define _SWIFT_PAIR_H_


#include "SWIFT_config.h"
#include "SWIFT_common.h"
#include "SWIFT_linalg.h"
#ifdef SWIFT_FRONT_TRACKING
#include "SWIFT_front.h"
#endif

class SWIFT_Object;
struct SWIFT_PairWorkspace;
void new_PairWorkspace(SWIFT_PairWorkspace*& s);   //called by scene creator
void delete_PairWorkspace(SWIFT_PairWorkspace*& s);

//////////////////////////////////////////////////////////////////////////////
// SWIFT_Pair
//
// Description:
//      Class to manage a pair of objects.
//////////////////////////////////////////////////////////////////////////////
class SWIFT_Pair {
  public:
    SWIFT_Pair( )
    {
      s = nullptr;
      /* (mt) Creating an own workspace for each pair would be too much.
         We create a single PairWorkspace on creation of a SWIFT_Scene
         and take care that the s pointer is being set to this space. */
         //new SWIFT_PairWorkspace;
#ifdef SWIFT_PIECE_CACHING
        cache_bv0 = cache_bv1 = nullptr;
#endif
#ifdef SWIFT_FRONT_TRACKING
        // Set up front
        front.Root().Set_Uninitialized( );
#else
        feat0 = nullptr; feat1 = nullptr;
#endif
        bit_field = 0x7;   // No overlap, not initialized or active
        next = nullptr; prev = nullptr;
        Set_Active();
    }
    ~SWIFT_Pair( ) { }

    void Delete( ) {
#ifdef SWIFT_FRONT_TRACKING
                front.Delete();
#endif
                id1 = -1; }

  // Get functions

    int Id0( ) const { return id0; }
    int Id1( ) const { return id1; }

    RESULT_TYPE State( ) const { return (RESULT_TYPE)((0x38 & bit_field) >> 3); }
    bool Overlapping( ) const { return !(0x7 & bit_field); }

#ifdef SWIFT_PIECE_CACHING
    SWIFT_BV* Cache_BV0( ) { return cache_bv0; }
    SWIFT_BV* Cache_BV1( ) { return cache_bv1; }
    void Set_Cache_BV0( SWIFT_BV* bv0 ) { cache_bv0 = bv0; }
    void Set_Cache_BV1( SWIFT_BV* bv1 ) { cache_bv1 = bv1; }
    void* Cache_Feature0( ) { return cache_feat0; }
    void* Cache_Feature1( ) { return cache_feat1; }
    void Set_Cache_Feature0( void* cf0 ) { cache_feat0 = cf0; }
    void Set_Cache_Feature1( void* cf1 ) { cache_feat1 = cf1; }
    RESULT_TYPE Cache_State( ) { return cache_state; }
    void Set_Cache_State( RESULT_TYPE s ) { cache_state = s; }
#ifdef SWIFT_PIECE_CACHING
    inline void Save_To_Cache_State();
#endif
#endif

#ifdef SWIFT_FRONT_TRACKING
    bool Active( ) { return (bool)(bit_field & 0x40); }
    bool Inactive( ) { return !Active(); }
    bool Initialized( ) { return (bool)(bit_field & 0x80); }
    bool Uninitialized( ) { return !Initialized(); }
#else
    bool Active( ) const { return (feat0 != nullptr); }
    bool Inactive( ) const { return (feat0 == nullptr); }
    bool Initialized( ) const { return (feat1 != nullptr); }
    bool Uninitialized( ) const { return (feat1 == nullptr); }
#endif

    bool Deleted( ) { return id1 == -1; }

    SWIFT_Pair* Next( ) const { return next; }
    SWIFT_Pair* Prev( ) const { return prev; }

  // Set functions

    void Set_Id0( int id ) { id0 = id; }
    void Set_Id1( int id ) { id1 = id; }
    void Toggle_Overlap( int axis ) { bit_field ^= (1 << axis); }
    void Set_State( RESULT_TYPE state )
                    { bit_field = (0xffffffc7 & bit_field) | (state << 3); }
#ifdef SWIFT_FRONT_TRACKING
    void Set_Active( ) { bit_field |= 0x40; }
    void Set_Inactive( ) { bit_field &= 0xffffffbf; }
    void Set_Initialized( ) { bit_field |= 0x80; }
    void Set_Uninitialized( ) { bit_field &= 0xffffff7f; }
    void Set_Roots( SWIFT_BV* r0, SWIFT_BV* r1 )
            { front.Root().Set_BV0( r0 ); front.Root().Set_BV1( r1 ); }
#else
    void Set_Active( )
                { if( feat0 == nullptr ) { feat0 = (void*) 0x1; feat1 = nullptr; } }
    void Set_Inactive( ) { feat0 = nullptr; }
    void Set_Uninitialized( ) { feat1 = nullptr; }
    void Set_Features( SWIFT_Tri_Vertex* vert1, SWIFT_Tri_Vertex* vert2 )
    {
        feat0 = (void*)vert1; feat1 = (void*)vert2;
        Set_State( CONTINUE_VV );
    }
#endif
    void Set_Next( SWIFT_Pair* n ) { next = n; }
    void Set_Prev( SWIFT_Pair* p ) { prev = p; }

  // Query functions

    // Returns true if there is a tolerance violation, otherwise false.
    bool Tolerance( SWIFT_Object* o0, SWIFT_Object* o1,
                    SWIFT_Real tolerance );


    // Returns true if there is an intersection, otherwise false.
    // Sets the exact distance if the tolerance was met.  It is assumed that
    // tolerance >= 0.
    // Can do approximate distance as well (for non-convex objects)
    bool Distance( SWIFT_Object* o0, SWIFT_Object* o1,
                   SWIFT_Real tolerance,
                   SWIFT_Real abs_error, SWIFT_Real rel_error,
                   SWIFT_Real& distance );

    // Returns true if there is an intersection, otherwise false.
    // Sets num_contacts to the number of contacts within the given tolerance
    // that were found.  Sets the exact distance if the tolerance was met.
    // It is assumed that tolerance >= 0.
    bool Contacts( SWIFT_Object* o0, SWIFT_Object* o1,
                   SWIFT_Real tolerance, SWIFT_Real& distance,
                   int& num_contacts );

    // These will fill the arrays with whatever number of contacts was returned
    // on the last distance call.  Based on the number of contacts and the
    // maximum number of spaces required in the arrays for each call, the
    // caller is responsible to ensure that there is enough room at the end of
    // the arrays.
    void Distances( SWIFT_Array<SWIFT_Real>& dists );
    void Contact_Features( SWIFT_Array<int>& ftypes, SWIFT_Array<int>& fids );
    void Contact_Points( SWIFT_Array<SWIFT_Real>& points );
    void Contact_Normals( SWIFT_Array<SWIFT_Real>& normals );


    SWIFT_PairWorkspace *s; //set by SWIFT_Scene on creation of a new scene
  private:
    inline void Report_Edge0( int i, SWIFT_Array<int>& fids );
    inline void Report_Edge1( int i, SWIFT_Array<int>& fids );
#ifndef SWIFT_FRONT_TRACKING
    inline void Save_State();
#endif
#ifdef SWIFT_PRIORITY_DIRECTION
    bool Handle_Full_Queue( SWIFT_Real tolerance, SWIFT_Real abs_error,
                            SWIFT_Real rel_error, SWIFT_Real& distance
#ifdef SWIFT_FRONT_TRACKING
                            , SWIFT_Front_Node* parent
#endif
                            );
#endif

  // Data members (per pair)

    // The ids of the objects.  If id0 == -1 then the pair is deleted and
    // should not be used.
    int id0;
    int id1;

    // overlap status in the lowest 3 bits then state in the next 3 bits then
    // level of object1 is in the next 5 bits then level of object2 is in the
    // next 5 bits then the index into the lut is in the next 11 bits
    unsigned int bit_field;

#ifdef SWIFT_PIECE_CACHING
    SWIFT_BV* cache_bv0;
    SWIFT_BV* cache_bv1;
    void* cache_feat0;
    void* cache_feat1;
    RESULT_TYPE cache_state;
#endif
#ifdef SWIFT_FRONT_TRACKING
    SWIFT_Front front;
#else
    // if feat0 == nullptr then the pair is not active
    void* feat0;
    // if feat1 == nullptr then the pair is not initialized
    void* feat1;
#endif

    // next and previous pointers for the box overlap list
    SWIFT_Pair* next;
    SWIFT_Pair* prev;
};

#endif


