#ifndef ROOT_TKDTree
#define ROOT_TKDTree

#ifndef ROOT_TObject
#include "TObject.h"
#endif

#include "TMath.h"
#include <vector>

template <typename Index, typename Value> class TKDTree : public TObject
{
public:
	
   TKDTree();
   TKDTree(Index npoints, Index ndim, UInt_t bsize);
   TKDTree(Index npoints, Index ndim, UInt_t bsize, Value **data);
   ~TKDTree();
   
   void            Build();  // build the tree
   
   Double_t        Distance(const Value *point, Index ind, Int_t type=2) const;
   void            DistanceToNode(const Value *point, Index inode, Value &min, Value &max, Int_t type=2);

   // Get indexes of left and right daughter nodes
   Int_t   GetLeft(Int_t inode)  const    {return inode*2+1;}
   Int_t   GetRight(Int_t inode) const    {return (inode+1)*2;}
   Int_t   GetParent(Int_t inode) const  {return (inode-1)/2;}
   //  
   // Other getters
   Index*  GetPointsIndexes(Int_t node) const;
   void    GetNodePointsIndexes(Int_t node, Int_t &first1, Int_t &last1, Int_t &first2, Int_t &last2) const;
   UChar_t GetNodeAxis(Int_t id) const {return (id < 0 || id >= fNNodes) ? 0 : fAxis[id];}
   Value   GetNodeValue(Int_t id) const {return (id < 0 || id >= fNNodes) ? 0 : fValue[id];}
   Int_t   GetNNodes() const {return fNNodes;}
   Int_t   GetTotalNodes() const {return fTotalNodes;}
   Value*  GetBoundaries();
   Value*  GetBoundariesExact();
   Value*  GetBoundary(const Int_t node);
   Value*  GetBoundaryExact(const Int_t node);
   Index   GetNPoints() { return fNPoints; }
   Index   GetNDim()    { return fNDim; }
   Index   GetNPointsNode(Int_t node) const;

   //Getters for internal variables.
   Int_t   GetRowT0() {return fRowT0;}      //! smallest terminal row
   Int_t   GetCrossNode() {return fCrossNode;}  //! cross node
   Int_t   GetOffset() {return fOffset;}     //! offset in fIndPoints
   Index*  GetIndPoints() {return fIndPoints;}
   Index   GetBucketSize() {return fBucketSize;}

   void    FindNearestNeighbors(const Value *point, Int_t k, Index *ind, Value *dist);
   Index   FindNode(const Value * point) const;
   void    FindPoint(Value * point, Index &index, Int_t &iter);
   void    FindInRange(Value *point, Value range, std::vector<Index> &res);
   void    FindBNodeA(Value * point, Value * delta, Int_t &inode);

   Bool_t  IsTerminal(Index inode) const {return (inode>=fNNodes);}
   Int_t   IsOwner() { return fDataOwner; }
   Value   KOrdStat(Index ntotal, Value *a, Index k, Index *index) const;


   void    MakeBoundaries(Value *range = 0x0);
   void    MakeBoundariesExact();
   void    SetData(Index npoints, Index ndim, UInt_t bsize, Value **data);
   Int_t   SetData(Index idim, Value *data);
   void    SetOwner(Int_t owner) { fDataOwner = owner; }
   void    Spread(Index ntotal, Value *a, Index *index, Value &min, Value &max) const;
   
 private:
   TKDTree(const TKDTree &); // not implemented
   TKDTree<Index, Value>& operator=(const TKDTree<Index, Value>&); // not implemented
   void CookBoundaries(const Int_t node, Bool_t left);

   void UpdateNearestNeighbors(Index inode, const Value *point, Int_t kNN, Index *ind, Value *dist);
   void UpdateRange(Index inode, Value *point, Value range, std::vector<Index> &res); 

 protected:
   Int_t   fDataOwner;  //! 0 - not owner, 2 - owner of the pointer array, 1 - owner of the whole 2-d array
   Int_t   fNNodes;     // size of node array
   Int_t   fTotalNodes; // total number of nodes (fNNodes + terminal nodes)
   Index   fNDim;       // number of dimensions
   Index   fNDimm;      // dummy 2*fNDim
   Index   fNPoints;    // number of multidimensional points
   Index   fBucketSize; // size of the terminal nodes
   UChar_t *fAxis;      //[fNNodes] nodes cutting axis
   Value   *fValue;     //[fNNodes] nodes cutting value
   //
   Value   *fRange;     //[fNDimm] range of data for each dimension
   Value   **fData;     //! data points
   Value   *fBoundaries;//! nodes boundaries


   Index   *fIndPoints; //! array of points indexes
   Int_t   fRowT0;      //! smallest terminal row - first row that contains terminal nodes
   Int_t   fCrossNode;  //! cross node - node that begins the last row (with terminal nodes only)
   Int_t   fOffset;     //! offset in fIndPoints - if there are 2 rows, that contain terminal nodes
                        //  fOffset returns the index in the fIndPoints array of the first point
                        //  that belongs to the first node on the second row.


   ClassDef(TKDTree, 1)  // KD tree
};
      
      
typedef TKDTree<Int_t, Double_t> TKDTreeID;
typedef TKDTree<Int_t, Float_t> TKDTreeIF;

// Test functions:
TKDTreeIF *  TKDTreeTestBuild();

#endif

 TKDTree.h:1
 TKDTree.h:2
 TKDTree.h:3
 TKDTree.h:4
 TKDTree.h:5
 TKDTree.h:6
 TKDTree.h:7
 TKDTree.h:8
 TKDTree.h:9
 TKDTree.h:10
 TKDTree.h:11
 TKDTree.h:12
 TKDTree.h:13
 TKDTree.h:14
 TKDTree.h:15
 TKDTree.h:16
 TKDTree.h:17
 TKDTree.h:18
 TKDTree.h:19
 TKDTree.h:20
 TKDTree.h:21
 TKDTree.h:22
 TKDTree.h:23
 TKDTree.h:24
 TKDTree.h:25
 TKDTree.h:26
 TKDTree.h:27
 TKDTree.h:28
 TKDTree.h:29
 TKDTree.h:30
 TKDTree.h:31
 TKDTree.h:32
 TKDTree.h:33
 TKDTree.h:34
 TKDTree.h:35
 TKDTree.h:36
 TKDTree.h:37
 TKDTree.h:38
 TKDTree.h:39
 TKDTree.h:40
 TKDTree.h:41
 TKDTree.h:42
 TKDTree.h:43
 TKDTree.h:44
 TKDTree.h:45
 TKDTree.h:46
 TKDTree.h:47
 TKDTree.h:48
 TKDTree.h:49
 TKDTree.h:50
 TKDTree.h:51
 TKDTree.h:52
 TKDTree.h:53
 TKDTree.h:54
 TKDTree.h:55
 TKDTree.h:56
 TKDTree.h:57
 TKDTree.h:58
 TKDTree.h:59
 TKDTree.h:60
 TKDTree.h:61
 TKDTree.h:62
 TKDTree.h:63
 TKDTree.h:64
 TKDTree.h:65
 TKDTree.h:66
 TKDTree.h:67
 TKDTree.h:68
 TKDTree.h:69
 TKDTree.h:70
 TKDTree.h:71
 TKDTree.h:72
 TKDTree.h:73
 TKDTree.h:74
 TKDTree.h:75
 TKDTree.h:76
 TKDTree.h:77
 TKDTree.h:78
 TKDTree.h:79
 TKDTree.h:80
 TKDTree.h:81
 TKDTree.h:82
 TKDTree.h:83
 TKDTree.h:84
 TKDTree.h:85
 TKDTree.h:86
 TKDTree.h:87
 TKDTree.h:88
 TKDTree.h:89
 TKDTree.h:90
 TKDTree.h:91
 TKDTree.h:92
 TKDTree.h:93
 TKDTree.h:94
 TKDTree.h:95
 TKDTree.h:96
 TKDTree.h:97
 TKDTree.h:98
 TKDTree.h:99
 TKDTree.h:100
 TKDTree.h:101
 TKDTree.h:102
 TKDTree.h:103
 TKDTree.h:104
 TKDTree.h:105
 TKDTree.h:106
 TKDTree.h:107
 TKDTree.h:108
 TKDTree.h:109
 TKDTree.h:110
 TKDTree.h:111
 TKDTree.h:112
 TKDTree.h:113