2010-05-12 01:44:00 +08:00
|
|
|
/*M///////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
//
|
|
|
|
// IMPORTANT: READ BEFORE DOWNLOADING, COPYING, INSTALLING OR USING.
|
|
|
|
//
|
|
|
|
// By downloading, copying, installing or using the software you agree to this license.
|
|
|
|
// If you do not agree to this license, do not download, install,
|
|
|
|
// copy or use the software.
|
|
|
|
//
|
|
|
|
//
|
|
|
|
// License Agreement
|
|
|
|
// For Open Source Computer Vision Library
|
|
|
|
//
|
|
|
|
// Copyright (C) 2000-2008, Intel Corporation, all rights reserved.
|
|
|
|
// Copyright (C) 2009, Willow Garage Inc., all rights reserved.
|
|
|
|
// Third party copyrights are property of their respective owners.
|
|
|
|
//
|
|
|
|
// Redistribution and use in source and binary forms, with or without modification,
|
|
|
|
// are permitted provided that the following conditions are met:
|
|
|
|
//
|
|
|
|
// * Redistribution's of source code must retain the above copyright notice,
|
|
|
|
// this list of conditions and the following disclaimer.
|
|
|
|
//
|
|
|
|
// * Redistribution's in binary form must reproduce the above copyright notice,
|
|
|
|
// this list of conditions and the following disclaimer in the documentation
|
|
|
|
// and/or other materials provided with the distribution.
|
|
|
|
//
|
|
|
|
// * The name of the copyright holders may not be used to endorse or promote products
|
|
|
|
// derived from this software without specific prior written permission.
|
|
|
|
//
|
|
|
|
// This software is provided by the copyright holders and contributors "as is" and
|
|
|
|
// any express or implied warranties, including, but not limited to, the implied
|
|
|
|
// warranties of merchantability and fitness for a particular purpose are disclaimed.
|
|
|
|
// In no event shall the Intel Corporation or contributors be liable for any direct,
|
|
|
|
// indirect, incidental, special, exemplary, or consequential damages
|
|
|
|
// (including, but not limited to, procurement of substitute goods or services;
|
|
|
|
// loss of use, data, or profits; or business interruption) however caused
|
|
|
|
// and on any theory of liability, whether in contract, strict liability,
|
|
|
|
// or tort (including negligence or otherwise) arising in any way out of
|
|
|
|
// the use of this software, even if advised of the possibility of such damage.
|
|
|
|
//
|
|
|
|
//M*/
|
|
|
|
|
|
|
|
#ifndef __OPENCV_FEATURES_2D_HPP__
|
|
|
|
#define __OPENCV_FEATURES_2D_HPP__
|
|
|
|
|
|
|
|
#include "opencv2/core/core.hpp"
|
2010-10-13 03:47:50 +08:00
|
|
|
#include "opencv2/flann/flann.hpp"
|
2010-05-12 01:44:00 +08:00
|
|
|
|
|
|
|
#ifdef __cplusplus
|
2010-05-12 15:33:21 +08:00
|
|
|
#include <limits>
|
|
|
|
|
2010-05-12 01:44:00 +08:00
|
|
|
extern "C" {
|
|
|
|
#endif
|
|
|
|
|
|
|
|
typedef struct CvSURFPoint
|
|
|
|
{
|
|
|
|
CvPoint2D32f pt;
|
|
|
|
int laplacian;
|
|
|
|
int size;
|
|
|
|
float dir;
|
|
|
|
float hessian;
|
|
|
|
} CvSURFPoint;
|
|
|
|
|
|
|
|
CV_INLINE CvSURFPoint cvSURFPoint( CvPoint2D32f pt, int laplacian,
|
|
|
|
int size, float dir CV_DEFAULT(0),
|
|
|
|
float hessian CV_DEFAULT(0))
|
|
|
|
{
|
|
|
|
CvSURFPoint kp;
|
|
|
|
kp.pt = pt;
|
|
|
|
kp.laplacian = laplacian;
|
|
|
|
kp.size = size;
|
|
|
|
kp.dir = dir;
|
|
|
|
kp.hessian = hessian;
|
|
|
|
return kp;
|
|
|
|
}
|
|
|
|
|
|
|
|
typedef struct CvSURFParams
|
|
|
|
{
|
|
|
|
int extended;
|
|
|
|
double hessianThreshold;
|
|
|
|
|
|
|
|
int nOctaves;
|
|
|
|
int nOctaveLayers;
|
|
|
|
} CvSURFParams;
|
|
|
|
|
|
|
|
CVAPI(CvSURFParams) cvSURFParams( double hessianThreshold, int extended CV_DEFAULT(0) );
|
|
|
|
|
|
|
|
// If useProvidedKeyPts!=0, keypoints are not detected, but descriptors are computed
|
|
|
|
// at the locations provided in keypoints (a CvSeq of CvSURFPoint).
|
|
|
|
CVAPI(void) cvExtractSURF( const CvArr* img, const CvArr* mask,
|
|
|
|
CvSeq** keypoints, CvSeq** descriptors,
|
|
|
|
CvMemStorage* storage, CvSURFParams params, int useProvidedKeyPts CV_DEFAULT(0) );
|
|
|
|
|
2010-06-01 00:47:13 +08:00
|
|
|
/*!
|
|
|
|
Maximal Stable Regions Parameters
|
|
|
|
*/
|
2010-05-12 01:44:00 +08:00
|
|
|
typedef struct CvMSERParams
|
|
|
|
{
|
2010-06-01 00:47:13 +08:00
|
|
|
//! delta, in the code, it compares (size_{i}-size_{i-delta})/size_{i-delta}
|
2010-05-12 01:44:00 +08:00
|
|
|
int delta;
|
2010-06-01 00:47:13 +08:00
|
|
|
//! prune the area which bigger than maxArea
|
2010-05-12 01:44:00 +08:00
|
|
|
int maxArea;
|
2010-06-01 00:47:13 +08:00
|
|
|
//! prune the area which smaller than minArea
|
2010-05-12 01:44:00 +08:00
|
|
|
int minArea;
|
2010-06-01 00:47:13 +08:00
|
|
|
//! prune the area have simliar size to its children
|
2010-05-12 01:44:00 +08:00
|
|
|
float maxVariation;
|
2010-06-01 00:47:13 +08:00
|
|
|
//! trace back to cut off mser with diversity < min_diversity
|
2010-05-12 01:44:00 +08:00
|
|
|
float minDiversity;
|
2010-06-01 00:47:13 +08:00
|
|
|
|
|
|
|
/////// the next few params for MSER of color image
|
|
|
|
|
|
|
|
//! for color image, the evolution steps
|
2010-05-12 01:44:00 +08:00
|
|
|
int maxEvolution;
|
2010-06-01 00:47:13 +08:00
|
|
|
//! the area threshold to cause re-initialize
|
2010-05-12 01:44:00 +08:00
|
|
|
double areaThreshold;
|
2010-06-01 00:47:13 +08:00
|
|
|
//! ignore too small margin
|
2010-05-12 01:44:00 +08:00
|
|
|
double minMargin;
|
2010-06-01 00:47:13 +08:00
|
|
|
//! the aperture size for edge blur
|
2010-05-12 01:44:00 +08:00
|
|
|
int edgeBlurSize;
|
|
|
|
} CvMSERParams;
|
|
|
|
|
|
|
|
CVAPI(CvMSERParams) cvMSERParams( int delta CV_DEFAULT(5), int min_area CV_DEFAULT(60),
|
|
|
|
int max_area CV_DEFAULT(14400), float max_variation CV_DEFAULT(.25f),
|
|
|
|
float min_diversity CV_DEFAULT(.2f), int max_evolution CV_DEFAULT(200),
|
|
|
|
double area_threshold CV_DEFAULT(1.01),
|
|
|
|
double min_margin CV_DEFAULT(.003),
|
|
|
|
int edge_blur_size CV_DEFAULT(5) );
|
|
|
|
|
|
|
|
// Extracts the contours of Maximally Stable Extremal Regions
|
|
|
|
CVAPI(void) cvExtractMSER( CvArr* _img, CvArr* _mask, CvSeq** contours, CvMemStorage* storage, CvMSERParams params );
|
|
|
|
|
|
|
|
|
|
|
|
typedef struct CvStarKeypoint
|
|
|
|
{
|
|
|
|
CvPoint pt;
|
|
|
|
int size;
|
|
|
|
float response;
|
|
|
|
} CvStarKeypoint;
|
|
|
|
|
|
|
|
CV_INLINE CvStarKeypoint cvStarKeypoint(CvPoint pt, int size, float response)
|
|
|
|
{
|
|
|
|
CvStarKeypoint kpt;
|
|
|
|
kpt.pt = pt;
|
|
|
|
kpt.size = size;
|
|
|
|
kpt.response = response;
|
|
|
|
return kpt;
|
|
|
|
}
|
|
|
|
|
|
|
|
typedef struct CvStarDetectorParams
|
|
|
|
{
|
|
|
|
int maxSize;
|
|
|
|
int responseThreshold;
|
|
|
|
int lineThresholdProjected;
|
|
|
|
int lineThresholdBinarized;
|
|
|
|
int suppressNonmaxSize;
|
|
|
|
} CvStarDetectorParams;
|
|
|
|
|
|
|
|
CV_INLINE CvStarDetectorParams cvStarDetectorParams(
|
|
|
|
int maxSize CV_DEFAULT(45),
|
|
|
|
int responseThreshold CV_DEFAULT(30),
|
|
|
|
int lineThresholdProjected CV_DEFAULT(10),
|
|
|
|
int lineThresholdBinarized CV_DEFAULT(8),
|
|
|
|
int suppressNonmaxSize CV_DEFAULT(5))
|
|
|
|
{
|
|
|
|
CvStarDetectorParams params;
|
|
|
|
params.maxSize = maxSize;
|
|
|
|
params.responseThreshold = responseThreshold;
|
|
|
|
params.lineThresholdProjected = lineThresholdProjected;
|
|
|
|
params.lineThresholdBinarized = lineThresholdBinarized;
|
|
|
|
params.suppressNonmaxSize = suppressNonmaxSize;
|
|
|
|
|
|
|
|
return params;
|
|
|
|
}
|
|
|
|
|
|
|
|
CVAPI(CvSeq*) cvGetStarKeypoints( const CvArr* img, CvMemStorage* storage,
|
|
|
|
CvStarDetectorParams params CV_DEFAULT(cvStarDetectorParams()));
|
|
|
|
|
|
|
|
#ifdef __cplusplus
|
|
|
|
}
|
|
|
|
|
2010-05-26 20:34:48 +08:00
|
|
|
namespace cv
|
|
|
|
{
|
2010-07-26 16:58:46 +08:00
|
|
|
struct CV_EXPORTS DefaultRngAuto
|
|
|
|
{
|
|
|
|
const uint64 old_state;
|
|
|
|
|
2010-10-12 20:36:20 +08:00
|
|
|
DefaultRngAuto() : old_state(theRNG().state) { theRNG().state = (uint64)-1; }
|
2010-07-26 16:58:46 +08:00
|
|
|
~DefaultRngAuto() { theRNG().state = old_state; }
|
|
|
|
|
|
|
|
DefaultRngAuto& operator=(const DefaultRngAuto&);
|
|
|
|
};
|
|
|
|
|
2010-05-26 20:34:48 +08:00
|
|
|
|
2010-05-18 01:36:58 +08:00
|
|
|
// CvAffinePose: defines a parameterized affine transformation of an image patch.
|
|
|
|
// An image patch is rotated on angle phi (in degrees), then scaled lambda1 times
|
|
|
|
// along horizontal and lambda2 times along vertical direction, and then rotated again
|
|
|
|
// on angle (theta - phi).
|
|
|
|
class CV_EXPORTS CvAffinePose
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
float phi;
|
|
|
|
float theta;
|
|
|
|
float lambda1;
|
|
|
|
float lambda2;
|
|
|
|
};
|
2010-05-12 01:44:00 +08:00
|
|
|
|
2010-06-01 00:47:13 +08:00
|
|
|
/*!
|
|
|
|
The Keypoint Class
|
|
|
|
|
|
|
|
The class instance stores a keypoint, i.e. a point feature found by one of many available keypoint detectors, such as
|
|
|
|
Harris corner detector, cv::FAST, cv::StarDetector, cv::SURF, cv::SIFT, cv::LDetector etc.
|
|
|
|
|
|
|
|
The keypoint is characterized by the 2D position, scale
|
|
|
|
(proportional to the diameter of the neighborhood that needs to be taken into account),
|
|
|
|
orientation and some other parameters. The keypoint neighborhood is then analyzed by another algorithm that builds a descriptor
|
|
|
|
(usually represented as a feature vector). The keypoints representing the same object in different images can then be matched using
|
|
|
|
cv::KDTree or another method.
|
|
|
|
*/
|
2010-05-12 01:44:00 +08:00
|
|
|
class CV_EXPORTS KeyPoint
|
|
|
|
{
|
2010-06-01 00:47:13 +08:00
|
|
|
public:
|
|
|
|
//! the default constructor
|
2010-05-12 01:44:00 +08:00
|
|
|
KeyPoint() : pt(0,0), size(0), angle(-1), response(0), octave(0), class_id(-1) {}
|
2010-06-01 00:47:13 +08:00
|
|
|
//! the full constructor
|
2010-05-12 01:44:00 +08:00
|
|
|
KeyPoint(Point2f _pt, float _size, float _angle=-1,
|
|
|
|
float _response=0, int _octave=0, int _class_id=-1)
|
|
|
|
: pt(_pt), size(_size), angle(_angle),
|
|
|
|
response(_response), octave(_octave), class_id(_class_id) {}
|
2010-06-01 00:47:13 +08:00
|
|
|
//! another form of the full constructor
|
2010-05-12 01:44:00 +08:00
|
|
|
KeyPoint(float x, float y, float _size, float _angle=-1,
|
|
|
|
float _response=0, int _octave=0, int _class_id=-1)
|
|
|
|
: pt(x, y), size(_size), angle(_angle),
|
|
|
|
response(_response), octave(_octave), class_id(_class_id) {}
|
2010-06-01 00:47:13 +08:00
|
|
|
//! converts vector of keypoints to vector of points
|
2010-10-17 04:34:34 +08:00
|
|
|
static void convert(const std::vector<KeyPoint>& keypoints,
|
|
|
|
CV_OUT std::vector<Point2f>& points2f,
|
2010-06-15 23:19:11 +08:00
|
|
|
const std::vector<int>& keypointIndexes=std::vector<int>());
|
2010-06-01 00:47:13 +08:00
|
|
|
//! converts vector of points to the vector of keypoints, where each keypoint is assigned the same size and the same orientation
|
2010-10-17 04:34:34 +08:00
|
|
|
static void convert(const std::vector<Point2f>& points2f,
|
|
|
|
CV_OUT std::vector<KeyPoint>& keypoints,
|
2010-05-12 01:44:00 +08:00
|
|
|
float size=1, float response=1, int octave=0, int class_id=-1);
|
2010-07-28 18:47:48 +08:00
|
|
|
|
|
|
|
//! computes overlap for pair of keypoints;
|
|
|
|
//! overlap is a ratio between area of keypoint regions intersection and
|
|
|
|
//! area of keypoint regions union (now keypoint region is circle)
|
|
|
|
static float overlap(const KeyPoint& kp1, const KeyPoint& kp2);
|
|
|
|
|
2010-06-01 00:47:13 +08:00
|
|
|
Point2f pt; //!< coordinates of the keypoints
|
|
|
|
float size; //!< diameter of the meaningfull keypoint neighborhood
|
|
|
|
float angle; //!< computed orientation of the keypoint (-1 if not applicable)
|
|
|
|
float response; //!< the response by which the most strong keypoints have been selected. Can be used for the further sorting or subsampling
|
|
|
|
int octave; //!< octave (pyramid layer) from which the keypoint has been extracted
|
|
|
|
int class_id; //!< object class (if the keypoints need to be clustered by an object they belong to)
|
2010-05-12 01:44:00 +08:00
|
|
|
};
|
|
|
|
|
2010-06-01 00:47:13 +08:00
|
|
|
//! writes vector of keypoints to the file storage
|
2010-10-28 02:26:39 +08:00
|
|
|
CV_EXPORTS_W void write(FileStorage& fs, const string& name, const vector<KeyPoint>& keypoints);
|
2010-06-01 00:47:13 +08:00
|
|
|
//! reads vector of keypoints from the specified file storage node
|
2010-10-28 02:26:39 +08:00
|
|
|
CV_EXPORTS_W void read(const FileNode& node, CV_OUT vector<KeyPoint>& keypoints);
|
2010-05-12 01:44:00 +08:00
|
|
|
|
2010-06-01 00:47:13 +08:00
|
|
|
/*!
|
|
|
|
SIFT implementation.
|
|
|
|
|
|
|
|
The class implements SIFT algorithm by D. Lowe.
|
|
|
|
*/
|
2010-05-12 01:44:00 +08:00
|
|
|
class CV_EXPORTS SIFT
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
struct CommonParams
|
|
|
|
{
|
|
|
|
static const int DEFAULT_NOCTAVES = 4;
|
|
|
|
static const int DEFAULT_NOCTAVE_LAYERS = 3;
|
|
|
|
static const int DEFAULT_FIRST_OCTAVE = -1;
|
2010-05-20 00:02:30 +08:00
|
|
|
enum{ FIRST_ANGLE = 0, AVERAGE_ANGLE = 1 };
|
|
|
|
|
2010-06-09 22:23:15 +08:00
|
|
|
CommonParams();
|
|
|
|
CommonParams( int _nOctaves, int _nOctaveLayers, int _firstOctave, int _angleMode );
|
2010-05-12 01:44:00 +08:00
|
|
|
int nOctaves, nOctaveLayers, firstOctave;
|
2010-05-20 00:02:30 +08:00
|
|
|
int angleMode;
|
2010-05-12 01:44:00 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
struct DetectorParams
|
|
|
|
{
|
|
|
|
static double GET_DEFAULT_THRESHOLD() { return 0.04 / SIFT::CommonParams::DEFAULT_NOCTAVE_LAYERS / 2.0; }
|
|
|
|
static double GET_DEFAULT_EDGE_THRESHOLD() { return 10.0; }
|
2010-05-20 00:02:30 +08:00
|
|
|
|
2010-06-09 22:23:15 +08:00
|
|
|
DetectorParams();
|
|
|
|
DetectorParams( double _threshold, double _edgeThreshold );
|
2010-05-12 01:44:00 +08:00
|
|
|
double threshold, edgeThreshold;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct DescriptorParams
|
|
|
|
{
|
|
|
|
static double GET_DEFAULT_MAGNIFICATION() { return 3.0; }
|
|
|
|
static const bool DEFAULT_IS_NORMALIZE = true;
|
2010-05-18 01:36:58 +08:00
|
|
|
static const int DESCRIPTOR_SIZE = 128;
|
2010-06-09 22:23:15 +08:00
|
|
|
|
|
|
|
DescriptorParams();
|
|
|
|
DescriptorParams( double _magnification, bool _isNormalize, bool _recalculateAngles );
|
2010-05-12 01:44:00 +08:00
|
|
|
double magnification;
|
|
|
|
bool isNormalize;
|
2010-05-20 00:02:30 +08:00
|
|
|
bool recalculateAngles;
|
2010-05-12 01:44:00 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
SIFT();
|
2010-06-01 00:47:13 +08:00
|
|
|
//! sift-detector constructor
|
2010-05-20 00:02:30 +08:00
|
|
|
SIFT( double _threshold, double _edgeThreshold,
|
2010-05-12 01:44:00 +08:00
|
|
|
int _nOctaves=CommonParams::DEFAULT_NOCTAVES,
|
|
|
|
int _nOctaveLayers=CommonParams::DEFAULT_NOCTAVE_LAYERS,
|
2010-05-20 00:02:30 +08:00
|
|
|
int _firstOctave=CommonParams::DEFAULT_FIRST_OCTAVE,
|
|
|
|
int _angleMode=CommonParams::FIRST_ANGLE );
|
2010-06-01 00:47:13 +08:00
|
|
|
//! sift-descriptor constructor
|
2010-05-12 01:44:00 +08:00
|
|
|
SIFT( double _magnification, bool _isNormalize=true,
|
2010-05-20 00:02:30 +08:00
|
|
|
bool _recalculateAngles = true,
|
2010-05-12 01:44:00 +08:00
|
|
|
int _nOctaves=CommonParams::DEFAULT_NOCTAVES,
|
|
|
|
int _nOctaveLayers=CommonParams::DEFAULT_NOCTAVE_LAYERS,
|
2010-05-20 00:02:30 +08:00
|
|
|
int _firstOctave=CommonParams::DEFAULT_FIRST_OCTAVE,
|
|
|
|
int _angleMode=CommonParams::FIRST_ANGLE );
|
2010-05-12 01:44:00 +08:00
|
|
|
SIFT( const CommonParams& _commParams,
|
|
|
|
const DetectorParams& _detectorParams = DetectorParams(),
|
|
|
|
const DescriptorParams& _descriptorParams = DescriptorParams() );
|
|
|
|
|
2010-06-01 00:47:13 +08:00
|
|
|
//! returns the descriptor size in floats (128)
|
2010-05-12 01:44:00 +08:00
|
|
|
int descriptorSize() const { return DescriptorParams::DESCRIPTOR_SIZE; }
|
2010-06-01 00:47:13 +08:00
|
|
|
//! finds the keypoints using SIFT algorithm
|
2010-05-12 01:44:00 +08:00
|
|
|
void operator()(const Mat& img, const Mat& mask,
|
|
|
|
vector<KeyPoint>& keypoints) const;
|
2010-06-09 22:23:15 +08:00
|
|
|
//! finds the keypoints and computes descriptors for them using SIFT algorithm.
|
|
|
|
//! Optionally it can compute descriptors for the user-provided keypoints
|
2010-05-12 01:44:00 +08:00
|
|
|
void operator()(const Mat& img, const Mat& mask,
|
|
|
|
vector<KeyPoint>& keypoints,
|
|
|
|
Mat& descriptors,
|
|
|
|
bool useProvidedKeypoints=false) const;
|
2010-06-04 13:30:09 +08:00
|
|
|
|
|
|
|
CommonParams getCommonParams () const { return commParams; }
|
|
|
|
DetectorParams getDetectorParams () const { return detectorParams; }
|
|
|
|
DescriptorParams getDescriptorParams () const { return descriptorParams; }
|
2010-05-12 01:44:00 +08:00
|
|
|
protected:
|
|
|
|
CommonParams commParams;
|
|
|
|
DetectorParams detectorParams;
|
|
|
|
DescriptorParams descriptorParams;
|
|
|
|
};
|
|
|
|
|
2010-06-01 00:47:13 +08:00
|
|
|
|
|
|
|
/*!
|
|
|
|
SURF implementation.
|
|
|
|
|
|
|
|
The class implements SURF algorithm by H. Bay et al.
|
|
|
|
*/
|
2010-10-28 02:26:39 +08:00
|
|
|
class CV_EXPORTS_W SURF : public CvSURFParams
|
2010-05-12 01:44:00 +08:00
|
|
|
{
|
|
|
|
public:
|
2010-06-01 00:47:13 +08:00
|
|
|
//! the default constructor
|
2010-10-28 02:26:39 +08:00
|
|
|
CV_WRAP SURF();
|
2010-06-01 00:47:13 +08:00
|
|
|
//! the full constructor taking all the necessary parameters
|
2010-10-28 02:26:39 +08:00
|
|
|
CV_WRAP SURF(double _hessianThreshold, int _nOctaves=4,
|
2010-05-12 01:44:00 +08:00
|
|
|
int _nOctaveLayers=2, bool _extended=false);
|
|
|
|
|
2010-06-01 00:47:13 +08:00
|
|
|
//! returns the descriptor size in float's (64 or 128)
|
2010-10-28 02:26:39 +08:00
|
|
|
CV_WRAP int descriptorSize() const;
|
2010-06-01 00:47:13 +08:00
|
|
|
//! finds the keypoints using fast hessian detector used in SURF
|
2010-10-17 04:34:34 +08:00
|
|
|
CV_WRAP_AS(detect) void operator()(const Mat& img, const Mat& mask,
|
|
|
|
CV_OUT vector<KeyPoint>& keypoints) const;
|
2010-06-01 00:47:13 +08:00
|
|
|
//! finds the keypoints and computes their descriptors. Optionally it can compute descriptors for the user-provided keypoints
|
2010-10-17 04:34:34 +08:00
|
|
|
CV_WRAP_AS(detect) void operator()(const Mat& img, const Mat& mask,
|
|
|
|
CV_OUT vector<KeyPoint>& keypoints,
|
|
|
|
CV_OUT vector<float>& descriptors,
|
2010-05-12 01:44:00 +08:00
|
|
|
bool useProvidedKeypoints=false) const;
|
|
|
|
};
|
|
|
|
|
2010-06-01 00:47:13 +08:00
|
|
|
/*!
|
|
|
|
Maximal Stable Extremal Regions class.
|
|
|
|
|
|
|
|
The class implements MSER algorithm introduced by J. Matas.
|
|
|
|
Unlike SIFT, SURF and many other detectors in OpenCV, this is salient region detector,
|
|
|
|
not the salient point detector.
|
|
|
|
|
|
|
|
It returns the regions, each of those is encoded as a contour.
|
|
|
|
*/
|
2010-10-28 02:26:39 +08:00
|
|
|
class CV_EXPORTS_W MSER : public CvMSERParams
|
2010-05-12 01:44:00 +08:00
|
|
|
{
|
|
|
|
public:
|
2010-06-01 00:47:13 +08:00
|
|
|
//! the default constructor
|
2010-10-28 02:26:39 +08:00
|
|
|
CV_WRAP MSER();
|
2010-06-01 00:47:13 +08:00
|
|
|
//! the full constructor
|
2010-10-28 02:26:39 +08:00
|
|
|
CV_WRAP MSER( int _delta, int _min_area, int _max_area,
|
2010-07-16 20:54:53 +08:00
|
|
|
double _max_variation, double _min_diversity,
|
2010-05-12 01:44:00 +08:00
|
|
|
int _max_evolution, double _area_threshold,
|
|
|
|
double _min_margin, int _edge_blur_size );
|
2010-06-01 00:47:13 +08:00
|
|
|
//! the operator that extracts the MSERs from the image or the specific part of it
|
2010-10-17 04:34:34 +08:00
|
|
|
CV_WRAP_AS(detect) void operator()( const Mat& image,
|
|
|
|
CV_OUT vector<vector<Point> >& msers, const Mat& mask ) const;
|
2010-05-12 01:44:00 +08:00
|
|
|
};
|
|
|
|
|
2010-06-01 00:47:13 +08:00
|
|
|
/*!
|
|
|
|
The "Star" Detector.
|
|
|
|
|
|
|
|
The class implements the keypoint detector introduced by K. Konolige.
|
|
|
|
*/
|
2010-10-28 02:26:39 +08:00
|
|
|
class CV_EXPORTS_W StarDetector : public CvStarDetectorParams
|
2010-05-12 01:44:00 +08:00
|
|
|
{
|
|
|
|
public:
|
2010-06-01 00:47:13 +08:00
|
|
|
//! the default constructor
|
2010-10-28 02:26:39 +08:00
|
|
|
CV_WRAP StarDetector();
|
2010-06-01 00:47:13 +08:00
|
|
|
//! the full constructor
|
2010-10-28 02:26:39 +08:00
|
|
|
CV_WRAP StarDetector(int _maxSize, int _responseThreshold,
|
2010-05-12 01:44:00 +08:00
|
|
|
int _lineThresholdProjected,
|
|
|
|
int _lineThresholdBinarized,
|
|
|
|
int _suppressNonmaxSize);
|
2010-06-01 00:47:13 +08:00
|
|
|
//! finds the keypoints in the image
|
2010-10-17 04:34:34 +08:00
|
|
|
CV_WRAP_AS(detect) void operator()(const Mat& image,
|
|
|
|
CV_OUT vector<KeyPoint>& keypoints) const;
|
2010-05-12 01:44:00 +08:00
|
|
|
};
|
|
|
|
|
2010-06-01 00:47:13 +08:00
|
|
|
//! detects corners using FAST algorithm by E. Rosten
|
2010-10-17 04:34:34 +08:00
|
|
|
CV_EXPORTS void FAST( const Mat& image, CV_OUT vector<KeyPoint>& keypoints,
|
|
|
|
int threshold, bool nonmaxSupression=true );
|
2010-05-12 01:44:00 +08:00
|
|
|
|
2010-06-01 00:47:13 +08:00
|
|
|
/*!
|
2010-07-26 16:58:46 +08:00
|
|
|
The Patch Generator class
|
2010-06-01 00:47:13 +08:00
|
|
|
*/
|
2010-05-12 01:44:00 +08:00
|
|
|
class CV_EXPORTS PatchGenerator
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
PatchGenerator();
|
|
|
|
PatchGenerator(double _backgroundMin, double _backgroundMax,
|
|
|
|
double _noiseRange, bool _randomBlur=true,
|
|
|
|
double _lambdaMin=0.6, double _lambdaMax=1.5,
|
|
|
|
double _thetaMin=-CV_PI, double _thetaMax=CV_PI,
|
|
|
|
double _phiMin=-CV_PI, double _phiMax=CV_PI );
|
2010-10-28 02:26:39 +08:00
|
|
|
void operator()(const Mat& image, Point2f pt, Mat& patch, Size patchSize, RNG& rng) const;
|
|
|
|
void operator()(const Mat& image, const Mat& transform, Mat& patch,
|
2010-05-12 01:44:00 +08:00
|
|
|
Size patchSize, RNG& rng) const;
|
|
|
|
void warpWholeImage(const Mat& image, Mat& matT, Mat& buf,
|
2010-10-17 04:34:34 +08:00
|
|
|
CV_OUT Mat& warped, int border, RNG& rng) const;
|
2010-05-12 01:44:00 +08:00
|
|
|
void generateRandomTransform(Point2f srcCenter, Point2f dstCenter,
|
2010-10-17 04:34:34 +08:00
|
|
|
CV_OUT Mat& transform, RNG& rng,
|
|
|
|
bool inverse=false) const;
|
2010-05-12 01:44:00 +08:00
|
|
|
void setAffineParam(double lambda, double theta, double phi);
|
|
|
|
|
|
|
|
double backgroundMin, backgroundMax;
|
|
|
|
double noiseRange;
|
|
|
|
bool randomBlur;
|
|
|
|
double lambdaMin, lambdaMax;
|
|
|
|
double thetaMin, thetaMax;
|
|
|
|
double phiMin, phiMax;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class CV_EXPORTS LDetector
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
LDetector();
|
|
|
|
LDetector(int _radius, int _threshold, int _nOctaves,
|
|
|
|
int _nViews, double _baseFeatureSize, double _clusteringDistance);
|
2010-10-28 02:26:39 +08:00
|
|
|
void operator()(const Mat& image,
|
2010-10-17 04:34:34 +08:00
|
|
|
CV_OUT vector<KeyPoint>& keypoints,
|
|
|
|
int maxCount=0, bool scaleCoords=true) const;
|
2010-10-28 02:26:39 +08:00
|
|
|
void operator()(const vector<Mat>& pyr,
|
2010-10-17 04:34:34 +08:00
|
|
|
CV_OUT vector<KeyPoint>& keypoints,
|
|
|
|
int maxCount=0, bool scaleCoords=true) const;
|
|
|
|
void getMostStable2D(const Mat& image, CV_OUT vector<KeyPoint>& keypoints,
|
2010-05-12 01:44:00 +08:00
|
|
|
int maxCount, const PatchGenerator& patchGenerator) const;
|
|
|
|
void setVerbose(bool verbose);
|
|
|
|
|
|
|
|
void read(const FileNode& node);
|
|
|
|
void write(FileStorage& fs, const String& name=String()) const;
|
|
|
|
|
|
|
|
int radius;
|
|
|
|
int threshold;
|
|
|
|
int nOctaves;
|
|
|
|
int nViews;
|
|
|
|
bool verbose;
|
|
|
|
|
|
|
|
double baseFeatureSize;
|
|
|
|
double clusteringDistance;
|
|
|
|
};
|
|
|
|
|
|
|
|
typedef LDetector YAPE;
|
|
|
|
|
|
|
|
class CV_EXPORTS FernClassifier
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
FernClassifier();
|
|
|
|
FernClassifier(const FileNode& node);
|
2010-07-26 16:58:46 +08:00
|
|
|
FernClassifier(const vector<vector<Point2f> >& points,
|
|
|
|
const vector<Mat>& refimgs,
|
|
|
|
const vector<vector<int> >& labels=vector<vector<int> >(),
|
2010-05-12 01:44:00 +08:00
|
|
|
int _nclasses=0, int _patchSize=PATCH_SIZE,
|
|
|
|
int _signatureSize=DEFAULT_SIGNATURE_SIZE,
|
|
|
|
int _nstructs=DEFAULT_STRUCTS,
|
|
|
|
int _structSize=DEFAULT_STRUCT_SIZE,
|
|
|
|
int _nviews=DEFAULT_VIEWS,
|
|
|
|
int _compressionMethod=COMPRESSION_NONE,
|
|
|
|
const PatchGenerator& patchGenerator=PatchGenerator());
|
|
|
|
virtual ~FernClassifier();
|
|
|
|
virtual void read(const FileNode& n);
|
|
|
|
virtual void write(FileStorage& fs, const String& name=String()) const;
|
|
|
|
virtual void trainFromSingleView(const Mat& image,
|
|
|
|
const vector<KeyPoint>& keypoints,
|
|
|
|
int _patchSize=PATCH_SIZE,
|
|
|
|
int _signatureSize=DEFAULT_SIGNATURE_SIZE,
|
|
|
|
int _nstructs=DEFAULT_STRUCTS,
|
|
|
|
int _structSize=DEFAULT_STRUCT_SIZE,
|
|
|
|
int _nviews=DEFAULT_VIEWS,
|
|
|
|
int _compressionMethod=COMPRESSION_NONE,
|
|
|
|
const PatchGenerator& patchGenerator=PatchGenerator());
|
2010-07-26 16:58:46 +08:00
|
|
|
virtual void train(const vector<vector<Point2f> >& points,
|
|
|
|
const vector<Mat>& refimgs,
|
|
|
|
const vector<vector<int> >& labels=vector<vector<int> >(),
|
2010-05-12 01:44:00 +08:00
|
|
|
int _nclasses=0, int _patchSize=PATCH_SIZE,
|
|
|
|
int _signatureSize=DEFAULT_SIGNATURE_SIZE,
|
|
|
|
int _nstructs=DEFAULT_STRUCTS,
|
|
|
|
int _structSize=DEFAULT_STRUCT_SIZE,
|
|
|
|
int _nviews=DEFAULT_VIEWS,
|
|
|
|
int _compressionMethod=COMPRESSION_NONE,
|
|
|
|
const PatchGenerator& patchGenerator=PatchGenerator());
|
|
|
|
virtual int operator()(const Mat& img, Point2f kpt, vector<float>& signature) const;
|
|
|
|
virtual int operator()(const Mat& patch, vector<float>& signature) const;
|
|
|
|
virtual void clear();
|
|
|
|
void setVerbose(bool verbose);
|
|
|
|
|
|
|
|
int getClassCount() const;
|
|
|
|
int getStructCount() const;
|
|
|
|
int getStructSize() const;
|
|
|
|
int getSignatureSize() const;
|
|
|
|
int getCompressionMethod() const;
|
|
|
|
Size getPatchSize() const;
|
|
|
|
|
|
|
|
struct Feature
|
|
|
|
{
|
|
|
|
uchar x1, y1, x2, y2;
|
|
|
|
Feature() : x1(0), y1(0), x2(0), y2(0) {}
|
|
|
|
Feature(int _x1, int _y1, int _x2, int _y2)
|
|
|
|
: x1((uchar)_x1), y1((uchar)_y1), x2((uchar)_x2), y2((uchar)_y2)
|
|
|
|
{}
|
|
|
|
template<typename _Tp> bool operator ()(const Mat_<_Tp>& patch) const
|
|
|
|
{ return patch(y1,x1) > patch(y2, x2); }
|
|
|
|
};
|
|
|
|
|
|
|
|
enum
|
|
|
|
{
|
|
|
|
PATCH_SIZE = 31,
|
|
|
|
DEFAULT_STRUCTS = 50,
|
|
|
|
DEFAULT_STRUCT_SIZE = 9,
|
|
|
|
DEFAULT_VIEWS = 5000,
|
|
|
|
DEFAULT_SIGNATURE_SIZE = 176,
|
|
|
|
COMPRESSION_NONE = 0,
|
|
|
|
COMPRESSION_RANDOM_PROJ = 1,
|
|
|
|
COMPRESSION_PCA = 2,
|
|
|
|
DEFAULT_COMPRESSION_METHOD = COMPRESSION_NONE
|
|
|
|
};
|
|
|
|
|
|
|
|
protected:
|
|
|
|
virtual void prepare(int _nclasses, int _patchSize, int _signatureSize,
|
|
|
|
int _nstructs, int _structSize,
|
|
|
|
int _nviews, int _compressionMethod);
|
|
|
|
virtual void finalize(RNG& rng);
|
|
|
|
virtual int getLeaf(int fidx, const Mat& patch) const;
|
|
|
|
|
|
|
|
bool verbose;
|
|
|
|
int nstructs;
|
|
|
|
int structSize;
|
|
|
|
int nclasses;
|
|
|
|
int signatureSize;
|
|
|
|
int compressionMethod;
|
|
|
|
int leavesPerStruct;
|
|
|
|
Size patchSize;
|
|
|
|
vector<Feature> features;
|
|
|
|
vector<int> classCounters;
|
|
|
|
vector<float> posteriors;
|
|
|
|
};
|
|
|
|
|
2010-10-17 04:34:34 +08:00
|
|
|
|
2010-05-12 01:44:00 +08:00
|
|
|
class CV_EXPORTS PlanarObjectDetector
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
PlanarObjectDetector();
|
|
|
|
PlanarObjectDetector(const FileNode& node);
|
|
|
|
PlanarObjectDetector(const vector<Mat>& pyr, int _npoints=300,
|
|
|
|
int _patchSize=FernClassifier::PATCH_SIZE,
|
|
|
|
int _nstructs=FernClassifier::DEFAULT_STRUCTS,
|
|
|
|
int _structSize=FernClassifier::DEFAULT_STRUCT_SIZE,
|
|
|
|
int _nviews=FernClassifier::DEFAULT_VIEWS,
|
|
|
|
const LDetector& detector=LDetector(),
|
|
|
|
const PatchGenerator& patchGenerator=PatchGenerator());
|
|
|
|
virtual ~PlanarObjectDetector();
|
|
|
|
virtual void train(const vector<Mat>& pyr, int _npoints=300,
|
|
|
|
int _patchSize=FernClassifier::PATCH_SIZE,
|
|
|
|
int _nstructs=FernClassifier::DEFAULT_STRUCTS,
|
|
|
|
int _structSize=FernClassifier::DEFAULT_STRUCT_SIZE,
|
|
|
|
int _nviews=FernClassifier::DEFAULT_VIEWS,
|
|
|
|
const LDetector& detector=LDetector(),
|
|
|
|
const PatchGenerator& patchGenerator=PatchGenerator());
|
|
|
|
virtual void train(const vector<Mat>& pyr, const vector<KeyPoint>& keypoints,
|
|
|
|
int _patchSize=FernClassifier::PATCH_SIZE,
|
|
|
|
int _nstructs=FernClassifier::DEFAULT_STRUCTS,
|
|
|
|
int _structSize=FernClassifier::DEFAULT_STRUCT_SIZE,
|
|
|
|
int _nviews=FernClassifier::DEFAULT_VIEWS,
|
|
|
|
const LDetector& detector=LDetector(),
|
|
|
|
const PatchGenerator& patchGenerator=PatchGenerator());
|
|
|
|
Rect getModelROI() const;
|
|
|
|
vector<KeyPoint> getModelPoints() const;
|
|
|
|
const LDetector& getDetector() const;
|
|
|
|
const FernClassifier& getClassifier() const;
|
|
|
|
void setVerbose(bool verbose);
|
|
|
|
|
|
|
|
void read(const FileNode& node);
|
|
|
|
void write(FileStorage& fs, const String& name=String()) const;
|
2010-10-28 02:26:39 +08:00
|
|
|
bool operator()(const Mat& image, CV_OUT Mat& H, CV_OUT vector<Point2f>& corners) const;
|
|
|
|
bool operator()(const vector<Mat>& pyr, const vector<KeyPoint>& keypoints,
|
2010-10-17 04:34:34 +08:00
|
|
|
CV_OUT Mat& H, CV_OUT vector<Point2f>& corners,
|
|
|
|
CV_OUT vector<int>* pairs=0) const;
|
2010-05-12 01:44:00 +08:00
|
|
|
|
|
|
|
protected:
|
|
|
|
bool verbose;
|
|
|
|
Rect modelROI;
|
|
|
|
vector<KeyPoint> modelPoints;
|
|
|
|
LDetector ldetector;
|
|
|
|
FernClassifier fernClassifier;
|
|
|
|
};
|
|
|
|
|
|
|
|
/****************************************************************************************\
|
2010-07-26 16:58:46 +08:00
|
|
|
* Calonder Classifier *
|
2010-05-12 01:44:00 +08:00
|
|
|
\****************************************************************************************/
|
2010-07-27 20:36:48 +08:00
|
|
|
|
|
|
|
struct RTreeNode;
|
|
|
|
|
|
|
|
struct CV_EXPORTS BaseKeypoint
|
|
|
|
{
|
|
|
|
int x;
|
|
|
|
int y;
|
|
|
|
IplImage* image;
|
|
|
|
|
|
|
|
BaseKeypoint()
|
|
|
|
: x(0), y(0), image(NULL)
|
|
|
|
{}
|
|
|
|
|
|
|
|
BaseKeypoint(int x, int y, IplImage* image)
|
|
|
|
: x(x), y(y), image(image)
|
|
|
|
{}
|
|
|
|
};
|
|
|
|
|
|
|
|
class CV_EXPORTS RandomizedTree
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
friend class RTreeClassifier;
|
|
|
|
|
2010-07-27 21:28:21 +08:00
|
|
|
static const uchar PATCH_SIZE = 32;
|
2010-07-27 20:36:48 +08:00
|
|
|
static const int DEFAULT_DEPTH = 9;
|
|
|
|
static const int DEFAULT_VIEWS = 5000;
|
|
|
|
static const size_t DEFAULT_REDUCED_NUM_DIM = 176;
|
2010-07-27 21:28:21 +08:00
|
|
|
static float GET_LOWER_QUANT_PERC() { return .03f; }
|
|
|
|
static float GET_UPPER_QUANT_PERC() { return .92f; }
|
2010-07-27 20:36:48 +08:00
|
|
|
|
|
|
|
RandomizedTree();
|
|
|
|
~RandomizedTree();
|
|
|
|
|
|
|
|
void train(std::vector<BaseKeypoint> const& base_set, RNG &rng,
|
|
|
|
int depth, int views, size_t reduced_num_dim, int num_quant_bits);
|
|
|
|
void train(std::vector<BaseKeypoint> const& base_set, RNG &rng,
|
|
|
|
PatchGenerator &make_patch, int depth, int views, size_t reduced_num_dim,
|
|
|
|
int num_quant_bits);
|
|
|
|
|
|
|
|
// following two funcs are EXPERIMENTAL (do not use unless you know exactly what you do)
|
|
|
|
static void quantizeVector(float *vec, int dim, int N, float bnds[2], int clamp_mode=0);
|
2010-07-27 21:28:21 +08:00
|
|
|
static void quantizeVector(float *src, int dim, int N, float bnds[2], uchar *dst);
|
2010-07-27 20:36:48 +08:00
|
|
|
|
|
|
|
// patch_data must be a 32x32 array (no row padding)
|
|
|
|
float* getPosterior(uchar* patch_data);
|
|
|
|
const float* getPosterior(uchar* patch_data) const;
|
2010-07-27 21:28:21 +08:00
|
|
|
uchar* getPosterior2(uchar* patch_data);
|
|
|
|
const uchar* getPosterior2(uchar* patch_data) const;
|
2010-07-27 20:36:48 +08:00
|
|
|
|
|
|
|
void read(const char* file_name, int num_quant_bits);
|
|
|
|
void read(std::istream &is, int num_quant_bits);
|
|
|
|
void write(const char* file_name) const;
|
|
|
|
void write(std::ostream &os) const;
|
|
|
|
|
|
|
|
int classes() { return classes_; }
|
|
|
|
int depth() { return depth_; }
|
|
|
|
|
|
|
|
//void setKeepFloatPosteriors(bool b) { keep_float_posteriors_ = b; }
|
|
|
|
void discardFloatPosteriors() { freePosteriors(1); }
|
|
|
|
|
|
|
|
inline void applyQuantization(int num_quant_bits) { makePosteriors2(num_quant_bits); }
|
|
|
|
|
|
|
|
// debug
|
|
|
|
void savePosteriors(std::string url, bool append=false);
|
|
|
|
void savePosteriors2(std::string url, bool append=false);
|
|
|
|
|
|
|
|
private:
|
|
|
|
int classes_;
|
|
|
|
int depth_;
|
|
|
|
int num_leaves_;
|
|
|
|
std::vector<RTreeNode> nodes_;
|
|
|
|
float **posteriors_; // 16-bytes aligned posteriors
|
2010-07-27 21:28:21 +08:00
|
|
|
uchar **posteriors2_; // 16-bytes aligned posteriors
|
2010-07-27 20:36:48 +08:00
|
|
|
std::vector<int> leaf_counts_;
|
|
|
|
|
|
|
|
void createNodes(int num_nodes, RNG &rng);
|
|
|
|
void allocPosteriorsAligned(int num_leaves, int num_classes);
|
|
|
|
void freePosteriors(int which); // which: 1=posteriors_, 2=posteriors2_, 3=both
|
|
|
|
void init(int classes, int depth, RNG &rng);
|
|
|
|
void addExample(int class_id, uchar* patch_data);
|
|
|
|
void finalize(size_t reduced_num_dim, int num_quant_bits);
|
|
|
|
int getIndex(uchar* patch_data) const;
|
|
|
|
inline float* getPosteriorByIndex(int index);
|
|
|
|
inline const float* getPosteriorByIndex(int index) const;
|
2010-07-27 21:28:21 +08:00
|
|
|
inline uchar* getPosteriorByIndex2(int index);
|
|
|
|
inline const uchar* getPosteriorByIndex2(int index) const;
|
2010-07-27 20:36:48 +08:00
|
|
|
//void makeRandomMeasMatrix(float *cs_phi, PHI_DISTR_TYPE dt, size_t reduced_num_dim);
|
|
|
|
void convertPosteriorsToChar();
|
|
|
|
void makePosteriors2(int num_quant_bits);
|
|
|
|
void compressLeaves(size_t reduced_num_dim);
|
|
|
|
void estimateQuantPercForPosteriors(float perc[2]);
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
inline uchar* getData(IplImage* image)
|
|
|
|
{
|
|
|
|
return reinterpret_cast<uchar*>(image->imageData);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline float* RandomizedTree::getPosteriorByIndex(int index)
|
|
|
|
{
|
|
|
|
return const_cast<float*>(const_cast<const RandomizedTree*>(this)->getPosteriorByIndex(index));
|
|
|
|
}
|
|
|
|
|
|
|
|
inline const float* RandomizedTree::getPosteriorByIndex(int index) const
|
|
|
|
{
|
|
|
|
return posteriors_[index];
|
|
|
|
}
|
|
|
|
|
2010-07-27 21:28:21 +08:00
|
|
|
inline uchar* RandomizedTree::getPosteriorByIndex2(int index)
|
2010-07-27 20:36:48 +08:00
|
|
|
{
|
2010-07-27 21:28:21 +08:00
|
|
|
return const_cast<uchar*>(const_cast<const RandomizedTree*>(this)->getPosteriorByIndex2(index));
|
2010-07-27 20:36:48 +08:00
|
|
|
}
|
|
|
|
|
2010-07-27 21:28:21 +08:00
|
|
|
inline const uchar* RandomizedTree::getPosteriorByIndex2(int index) const
|
2010-07-27 20:36:48 +08:00
|
|
|
{
|
|
|
|
return posteriors2_[index];
|
|
|
|
}
|
|
|
|
|
|
|
|
struct CV_EXPORTS RTreeNode
|
|
|
|
{
|
|
|
|
short offset1, offset2;
|
|
|
|
|
|
|
|
RTreeNode() {}
|
|
|
|
RTreeNode(uchar x1, uchar y1, uchar x2, uchar y2)
|
|
|
|
: offset1(y1*RandomizedTree::PATCH_SIZE + x1),
|
|
|
|
offset2(y2*RandomizedTree::PATCH_SIZE + x2)
|
|
|
|
{}
|
|
|
|
|
|
|
|
//! Left child on 0, right child on 1
|
|
|
|
inline bool operator() (uchar* patch_data) const
|
|
|
|
{
|
|
|
|
return patch_data[offset1] > patch_data[offset2];
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
class CV_EXPORTS RTreeClassifier
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
static const int DEFAULT_TREES = 48;
|
|
|
|
static const size_t DEFAULT_NUM_QUANT_BITS = 4;
|
|
|
|
|
|
|
|
RTreeClassifier();
|
|
|
|
void train(std::vector<BaseKeypoint> const& base_set,
|
|
|
|
RNG &rng,
|
|
|
|
int num_trees = RTreeClassifier::DEFAULT_TREES,
|
|
|
|
int depth = RandomizedTree::DEFAULT_DEPTH,
|
|
|
|
int views = RandomizedTree::DEFAULT_VIEWS,
|
|
|
|
size_t reduced_num_dim = RandomizedTree::DEFAULT_REDUCED_NUM_DIM,
|
|
|
|
int num_quant_bits = DEFAULT_NUM_QUANT_BITS);
|
|
|
|
void train(std::vector<BaseKeypoint> const& base_set,
|
|
|
|
RNG &rng,
|
|
|
|
PatchGenerator &make_patch,
|
|
|
|
int num_trees = RTreeClassifier::DEFAULT_TREES,
|
|
|
|
int depth = RandomizedTree::DEFAULT_DEPTH,
|
|
|
|
int views = RandomizedTree::DEFAULT_VIEWS,
|
|
|
|
size_t reduced_num_dim = RandomizedTree::DEFAULT_REDUCED_NUM_DIM,
|
|
|
|
int num_quant_bits = DEFAULT_NUM_QUANT_BITS);
|
|
|
|
|
2010-07-27 21:28:21 +08:00
|
|
|
// sig must point to a memory block of at least classes()*sizeof(float|uchar) bytes
|
|
|
|
void getSignature(IplImage *patch, uchar *sig) const;
|
2010-07-27 20:36:48 +08:00
|
|
|
void getSignature(IplImage *patch, float *sig) const;
|
|
|
|
void getSparseSignature(IplImage *patch, float *sig, float thresh) const;
|
|
|
|
// TODO: deprecated in favor of getSignature overload, remove
|
|
|
|
void getFloatSignature(IplImage *patch, float *sig) const { getSignature(patch, sig); }
|
|
|
|
|
|
|
|
static int countNonZeroElements(float *vec, int n, double tol=1e-10);
|
2010-07-27 21:28:21 +08:00
|
|
|
static inline void safeSignatureAlloc(uchar **sig, int num_sig=1, int sig_len=176);
|
|
|
|
static inline uchar* safeSignatureAlloc(int num_sig=1, int sig_len=176);
|
2010-07-27 20:36:48 +08:00
|
|
|
|
|
|
|
inline int classes() const { return classes_; }
|
|
|
|
inline int original_num_classes() const { return original_num_classes_; }
|
|
|
|
|
|
|
|
void setQuantization(int num_quant_bits);
|
|
|
|
void discardFloatPosteriors();
|
|
|
|
|
|
|
|
void read(const char* file_name);
|
|
|
|
void read(std::istream &is);
|
|
|
|
void write(const char* file_name) const;
|
|
|
|
void write(std::ostream &os) const;
|
|
|
|
|
|
|
|
// experimental and debug
|
|
|
|
void saveAllFloatPosteriors(std::string file_url);
|
|
|
|
void saveAllBytePosteriors(std::string file_url);
|
|
|
|
void setFloatPosteriorsFromTextfile_176(std::string url);
|
|
|
|
float countZeroElements();
|
|
|
|
|
|
|
|
std::vector<RandomizedTree> trees_;
|
|
|
|
|
|
|
|
private:
|
|
|
|
int classes_;
|
|
|
|
int num_quant_bits_;
|
2010-07-27 21:28:21 +08:00
|
|
|
mutable uchar **posteriors_;
|
|
|
|
mutable unsigned short *ptemp_;
|
2010-07-27 20:36:48 +08:00
|
|
|
int original_num_classes_;
|
|
|
|
bool keep_floats_;
|
|
|
|
};
|
|
|
|
|
2010-05-12 01:44:00 +08:00
|
|
|
/****************************************************************************************\
|
|
|
|
* One-Way Descriptor *
|
|
|
|
\****************************************************************************************/
|
|
|
|
|
|
|
|
class CV_EXPORTS OneWayDescriptor
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
OneWayDescriptor();
|
|
|
|
~OneWayDescriptor();
|
|
|
|
|
|
|
|
// allocates memory for given descriptor parameters
|
|
|
|
void Allocate(int pose_count, CvSize size, int nChannels);
|
|
|
|
|
|
|
|
// GenerateSamples: generates affine transformed patches with averaging them over small transformation variations.
|
|
|
|
// If external poses and transforms were specified, uses them instead of generating random ones
|
|
|
|
// - pose_count: the number of poses to be generated
|
|
|
|
// - frontal: the input patch (can be a roi in a larger image)
|
|
|
|
// - norm: if nonzero, normalizes the output patch so that the sum of pixel intensities is 1
|
|
|
|
void GenerateSamples(int pose_count, IplImage* frontal, int norm = 0);
|
|
|
|
|
|
|
|
// GenerateSamplesFast: generates affine transformed patches with averaging them over small transformation variations.
|
|
|
|
// Uses precalculated transformed pca components.
|
|
|
|
// - frontal: the input patch (can be a roi in a larger image)
|
|
|
|
// - pca_hr_avg: pca average vector
|
|
|
|
// - pca_hr_eigenvectors: pca eigenvectors
|
|
|
|
// - pca_descriptors: an array of precomputed descriptors of pca components containing their affine transformations
|
|
|
|
// pca_descriptors[0] corresponds to the average, pca_descriptors[1]-pca_descriptors[pca_dim] correspond to eigenvectors
|
|
|
|
void GenerateSamplesFast(IplImage* frontal, CvMat* pca_hr_avg,
|
|
|
|
CvMat* pca_hr_eigenvectors, OneWayDescriptor* pca_descriptors);
|
|
|
|
|
|
|
|
// sets the poses and corresponding transforms
|
|
|
|
void SetTransforms(CvAffinePose* poses, CvMat** transforms);
|
|
|
|
|
|
|
|
// Initialize: builds a descriptor.
|
|
|
|
// - pose_count: the number of poses to build. If poses were set externally, uses them rather than generating random ones
|
|
|
|
// - frontal: input patch. Can be a roi in a larger image
|
|
|
|
// - feature_name: the feature name to be associated with the descriptor
|
|
|
|
// - norm: if 1, the affine transformed patches are normalized so that their sum is 1
|
|
|
|
void Initialize(int pose_count, IplImage* frontal, const char* feature_name = 0, int norm = 0);
|
|
|
|
|
|
|
|
// InitializeFast: builds a descriptor using precomputed descriptors of pca components
|
|
|
|
// - pose_count: the number of poses to build
|
|
|
|
// - frontal: input patch. Can be a roi in a larger image
|
|
|
|
// - feature_name: the feature name to be associated with the descriptor
|
|
|
|
// - pca_hr_avg: average vector for PCA
|
|
|
|
// - pca_hr_eigenvectors: PCA eigenvectors (one vector per row)
|
|
|
|
// - pca_descriptors: precomputed descriptors of PCA components, the first descriptor for the average vector
|
|
|
|
// followed by the descriptors for eigenvectors
|
|
|
|
void InitializeFast(int pose_count, IplImage* frontal, const char* feature_name,
|
|
|
|
CvMat* pca_hr_avg, CvMat* pca_hr_eigenvectors, OneWayDescriptor* pca_descriptors);
|
|
|
|
|
|
|
|
// ProjectPCASample: unwarps an image patch into a vector and projects it into PCA space
|
|
|
|
// - patch: input image patch
|
|
|
|
// - avg: PCA average vector
|
|
|
|
// - eigenvectors: PCA eigenvectors, one per row
|
|
|
|
// - pca_coeffs: output PCA coefficients
|
|
|
|
void ProjectPCASample(IplImage* patch, CvMat* avg, CvMat* eigenvectors, CvMat* pca_coeffs) const;
|
|
|
|
|
|
|
|
// InitializePCACoeffs: projects all warped patches into PCA space
|
|
|
|
// - avg: PCA average vector
|
|
|
|
// - eigenvectors: PCA eigenvectors, one per row
|
|
|
|
void InitializePCACoeffs(CvMat* avg, CvMat* eigenvectors);
|
|
|
|
|
|
|
|
// EstimatePose: finds the closest match between an input patch and a set of patches with different poses
|
|
|
|
// - patch: input image patch
|
|
|
|
// - pose_idx: the output index of the closest pose
|
|
|
|
// - distance: the distance to the closest pose (L2 distance)
|
|
|
|
void EstimatePose(IplImage* patch, int& pose_idx, float& distance) const;
|
|
|
|
|
|
|
|
// EstimatePosePCA: finds the closest match between an input patch and a set of patches with different poses.
|
|
|
|
// The distance between patches is computed in PCA space
|
|
|
|
// - patch: input image patch
|
|
|
|
// - pose_idx: the output index of the closest pose
|
|
|
|
// - distance: distance to the closest pose (L2 distance in PCA space)
|
|
|
|
// - avg: PCA average vector. If 0, matching without PCA is used
|
|
|
|
// - eigenvectors: PCA eigenvectors, one per row
|
|
|
|
void EstimatePosePCA(CvArr* patch, int& pose_idx, float& distance, CvMat* avg, CvMat* eigenvalues) const;
|
|
|
|
|
|
|
|
// GetPatchSize: returns the size of each image patch after warping (2 times smaller than the input patch)
|
|
|
|
CvSize GetPatchSize() const
|
|
|
|
{
|
|
|
|
return m_patch_size;
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetInputPatchSize: returns the required size of the patch that the descriptor is built from
|
|
|
|
// (2 time larger than the patch after warping)
|
|
|
|
CvSize GetInputPatchSize() const
|
|
|
|
{
|
|
|
|
return cvSize(m_patch_size.width*2, m_patch_size.height*2);
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetPatch: returns a patch corresponding to specified pose index
|
|
|
|
// - index: pose index
|
|
|
|
// - return value: the patch corresponding to specified pose index
|
|
|
|
IplImage* GetPatch(int index);
|
|
|
|
|
|
|
|
// GetPose: returns a pose corresponding to specified pose index
|
|
|
|
// - index: pose index
|
|
|
|
// - return value: the pose corresponding to specified pose index
|
|
|
|
CvAffinePose GetPose(int index) const;
|
|
|
|
|
|
|
|
// Save: saves all patches with different poses to a specified path
|
|
|
|
void Save(const char* path);
|
|
|
|
|
|
|
|
// ReadByName: reads a descriptor from a file storage
|
|
|
|
// - fs: file storage
|
|
|
|
// - parent: parent node
|
|
|
|
// - name: node name
|
|
|
|
// - return value: 1 if succeeded, 0 otherwise
|
|
|
|
int ReadByName(CvFileStorage* fs, CvFileNode* parent, const char* name);
|
|
|
|
|
2010-06-04 13:30:09 +08:00
|
|
|
// ReadByName: reads a descriptor from a file node
|
|
|
|
// - parent: parent node
|
|
|
|
// - name: node name
|
|
|
|
// - return value: 1 if succeeded, 0 otherwise
|
|
|
|
int ReadByName(const FileNode &parent, const char* name);
|
|
|
|
|
2010-05-12 01:44:00 +08:00
|
|
|
// Write: writes a descriptor into a file storage
|
|
|
|
// - fs: file storage
|
|
|
|
// - name: node name
|
|
|
|
void Write(CvFileStorage* fs, const char* name);
|
|
|
|
|
|
|
|
// GetFeatureName: returns a name corresponding to a feature
|
|
|
|
const char* GetFeatureName() const;
|
|
|
|
|
|
|
|
// GetCenter: returns the center of the feature
|
|
|
|
CvPoint GetCenter() const;
|
|
|
|
|
|
|
|
void SetPCADimHigh(int pca_dim_high) {m_pca_dim_high = pca_dim_high;};
|
|
|
|
void SetPCADimLow(int pca_dim_low) {m_pca_dim_low = pca_dim_low;};
|
|
|
|
|
|
|
|
int GetPCADimLow() const;
|
|
|
|
int GetPCADimHigh() const;
|
|
|
|
|
|
|
|
CvMat** GetPCACoeffs() const {return m_pca_coeffs;}
|
|
|
|
|
|
|
|
protected:
|
|
|
|
int m_pose_count; // the number of poses
|
|
|
|
CvSize m_patch_size; // size of each image
|
|
|
|
IplImage** m_samples; // an array of length m_pose_count containing the patch in different poses
|
|
|
|
IplImage* m_input_patch;
|
|
|
|
IplImage* m_train_patch;
|
|
|
|
CvMat** m_pca_coeffs; // an array of length m_pose_count containing pca decomposition of the patch in different poses
|
|
|
|
CvAffinePose* m_affine_poses; // an array of poses
|
|
|
|
CvMat** m_transforms; // an array of affine transforms corresponding to poses
|
|
|
|
|
2010-07-26 16:58:46 +08:00
|
|
|
string m_feature_name; // the name of the feature associated with the descriptor
|
2010-05-12 01:44:00 +08:00
|
|
|
CvPoint m_center; // the coordinates of the feature (the center of the input image ROI)
|
|
|
|
|
|
|
|
int m_pca_dim_high; // the number of descriptor pca components to use for generating affine poses
|
|
|
|
int m_pca_dim_low; // the number of pca components to use for comparison
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
// OneWayDescriptorBase: encapsulates functionality for training/loading a set of one way descriptors
|
|
|
|
// and finding the nearest closest descriptor to an input feature
|
|
|
|
class CV_EXPORTS OneWayDescriptorBase
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
|
|
|
|
// creates an instance of OneWayDescriptor from a set of training files
|
|
|
|
// - patch_size: size of the input (large) patch
|
|
|
|
// - pose_count: the number of poses to generate for each descriptor
|
|
|
|
// - train_path: path to training files
|
|
|
|
// - pca_config: the name of the file that contains PCA for small patches (2 times smaller
|
|
|
|
// than patch_size each dimension
|
|
|
|
// - pca_hr_config: the name of the file that contains PCA for large patches (of patch_size size)
|
|
|
|
// - pca_desc_config: the name of the file that contains descriptors of PCA components
|
|
|
|
OneWayDescriptorBase(CvSize patch_size, int pose_count, const char* train_path = 0, const char* pca_config = 0,
|
|
|
|
const char* pca_hr_config = 0, const char* pca_desc_config = 0, int pyr_levels = 1,
|
|
|
|
int pca_dim_high = 100, int pca_dim_low = 100);
|
|
|
|
|
2010-06-02 13:19:09 +08:00
|
|
|
OneWayDescriptorBase(CvSize patch_size, int pose_count, const string &pca_filename, const string &train_path = string(), const string &images_list = string(),
|
2010-06-02 14:55:03 +08:00
|
|
|
float _scale_min = 0.7f, float _scale_max=1.5f, float _scale_step=1.2f, int pyr_levels = 1,
|
2010-06-02 13:19:09 +08:00
|
|
|
int pca_dim_high = 100, int pca_dim_low = 100);
|
|
|
|
|
|
|
|
|
2010-06-02 13:39:29 +08:00
|
|
|
virtual ~OneWayDescriptorBase();
|
2010-06-02 14:55:03 +08:00
|
|
|
void clear ();
|
|
|
|
|
2010-05-12 01:44:00 +08:00
|
|
|
|
|
|
|
// Allocate: allocates memory for a given number of descriptors
|
|
|
|
void Allocate(int train_feature_count);
|
|
|
|
|
|
|
|
// AllocatePCADescriptors: allocates memory for pca descriptors
|
|
|
|
void AllocatePCADescriptors();
|
|
|
|
|
|
|
|
// returns patch size
|
|
|
|
CvSize GetPatchSize() const {return m_patch_size;};
|
|
|
|
// returns the number of poses for each descriptor
|
|
|
|
int GetPoseCount() const {return m_pose_count;};
|
|
|
|
|
|
|
|
// returns the number of pyramid levels
|
|
|
|
int GetPyrLevels() const {return m_pyr_levels;};
|
|
|
|
|
|
|
|
// returns the number of descriptors
|
|
|
|
int GetDescriptorCount() const {return m_train_feature_count;};
|
|
|
|
|
|
|
|
// CreateDescriptorsFromImage: creates descriptors for each of the input features
|
|
|
|
// - src: input image
|
|
|
|
// - features: input features
|
|
|
|
// - pyr_levels: the number of pyramid levels
|
|
|
|
void CreateDescriptorsFromImage(IplImage* src, const std::vector<cv::KeyPoint>& features);
|
|
|
|
|
|
|
|
// CreatePCADescriptors: generates descriptors for PCA components, needed for fast generation of feature descriptors
|
|
|
|
void CreatePCADescriptors();
|
|
|
|
|
|
|
|
// returns a feature descriptor by feature index
|
|
|
|
const OneWayDescriptor* GetDescriptor(int desc_idx) const {return &m_descriptors[desc_idx];};
|
|
|
|
|
|
|
|
// FindDescriptor: finds the closest descriptor
|
|
|
|
// - patch: input image patch
|
|
|
|
// - desc_idx: output index of the closest descriptor to the input patch
|
|
|
|
// - pose_idx: output index of the closest pose of the closest descriptor to the input patch
|
|
|
|
// - distance: distance from the input patch to the closest feature pose
|
|
|
|
// - _scales: scales of the input patch for each descriptor
|
|
|
|
// - scale_ranges: input scales variation (float[2])
|
|
|
|
void FindDescriptor(IplImage* patch, int& desc_idx, int& pose_idx, float& distance, float* _scale = 0, float* scale_ranges = 0) const;
|
|
|
|
|
|
|
|
// - patch: input image patch
|
|
|
|
// - n: number of the closest indexes
|
|
|
|
// - desc_idxs: output indexes of the closest descriptor to the input patch (n)
|
|
|
|
// - pose_idx: output indexes of the closest pose of the closest descriptor to the input patch (n)
|
|
|
|
// - distances: distance from the input patch to the closest feature pose (n)
|
|
|
|
// - _scales: scales of the input patch
|
|
|
|
// - scale_ranges: input scales variation (float[2])
|
|
|
|
void FindDescriptor(IplImage* patch, int n, std::vector<int>& desc_idxs, std::vector<int>& pose_idxs,
|
|
|
|
std::vector<float>& distances, std::vector<float>& _scales, float* scale_ranges = 0) const;
|
|
|
|
|
|
|
|
// FindDescriptor: finds the closest descriptor
|
|
|
|
// - src: input image
|
|
|
|
// - pt: center of the feature
|
|
|
|
// - desc_idx: output index of the closest descriptor to the input patch
|
|
|
|
// - pose_idx: output index of the closest pose of the closest descriptor to the input patch
|
|
|
|
// - distance: distance from the input patch to the closest feature pose
|
|
|
|
void FindDescriptor(IplImage* src, cv::Point2f pt, int& desc_idx, int& pose_idx, float& distance) const;
|
|
|
|
|
|
|
|
// InitializePoses: generates random poses
|
|
|
|
void InitializePoses();
|
|
|
|
|
|
|
|
// InitializeTransformsFromPoses: generates 2x3 affine matrices from poses (initializes m_transforms)
|
|
|
|
void InitializeTransformsFromPoses();
|
|
|
|
|
|
|
|
// InitializePoseTransforms: subsequently calls InitializePoses and InitializeTransformsFromPoses
|
|
|
|
void InitializePoseTransforms();
|
|
|
|
|
|
|
|
// InitializeDescriptor: initializes a descriptor
|
|
|
|
// - desc_idx: descriptor index
|
|
|
|
// - train_image: image patch (ROI is supported)
|
|
|
|
// - feature_label: feature textual label
|
|
|
|
void InitializeDescriptor(int desc_idx, IplImage* train_image, const char* feature_label);
|
|
|
|
|
|
|
|
void InitializeDescriptor(int desc_idx, IplImage* train_image, const cv::KeyPoint& keypoint, const char* feature_label);
|
|
|
|
|
|
|
|
// InitializeDescriptors: load features from an image and create descriptors for each of them
|
|
|
|
void InitializeDescriptors(IplImage* train_image, const vector<cv::KeyPoint>& features,
|
|
|
|
const char* feature_label = "", int desc_start_idx = 0);
|
|
|
|
|
2010-06-08 14:56:35 +08:00
|
|
|
// Write: writes this object to a file storage
|
|
|
|
// - fs: output filestorage
|
|
|
|
void Write (FileStorage &fs) const;
|
|
|
|
|
|
|
|
// Read: reads OneWayDescriptorBase object from a file node
|
|
|
|
// - fn: input file node
|
|
|
|
void Read (const FileNode &fn);
|
2010-06-04 13:30:09 +08:00
|
|
|
|
2010-05-12 01:44:00 +08:00
|
|
|
// LoadPCADescriptors: loads PCA descriptors from a file
|
|
|
|
// - filename: input filename
|
|
|
|
int LoadPCADescriptors(const char* filename);
|
|
|
|
|
2010-06-04 13:30:09 +08:00
|
|
|
// LoadPCADescriptors: loads PCA descriptors from a file node
|
|
|
|
// - fn: input file node
|
|
|
|
int LoadPCADescriptors(const FileNode &fn);
|
|
|
|
|
2010-05-12 01:44:00 +08:00
|
|
|
// SavePCADescriptors: saves PCA descriptors to a file
|
|
|
|
// - filename: output filename
|
|
|
|
void SavePCADescriptors(const char* filename);
|
|
|
|
|
2010-06-02 13:19:09 +08:00
|
|
|
// SavePCADescriptors: saves PCA descriptors to a file storage
|
|
|
|
// - fs: output file storage
|
2010-06-04 13:30:09 +08:00
|
|
|
void SavePCADescriptors(CvFileStorage* fs) const;
|
2010-06-02 13:19:09 +08:00
|
|
|
|
|
|
|
// GeneratePCA: calculate and save PCA components and descriptors
|
|
|
|
// - img_path: path to training PCA images directory
|
|
|
|
// - images_list: filename with filenames of training PCA images
|
2010-06-02 14:55:03 +08:00
|
|
|
void GeneratePCA(const char* img_path, const char* images_list, int pose_count=500);
|
2010-06-02 13:19:09 +08:00
|
|
|
|
2010-05-12 01:44:00 +08:00
|
|
|
// SetPCAHigh: sets the high resolution pca matrices (copied to internal structures)
|
|
|
|
void SetPCAHigh(CvMat* avg, CvMat* eigenvectors);
|
|
|
|
|
|
|
|
// SetPCALow: sets the low resolution pca matrices (copied to internal structures)
|
|
|
|
void SetPCALow(CvMat* avg, CvMat* eigenvectors);
|
|
|
|
|
|
|
|
int GetLowPCA(CvMat** avg, CvMat** eigenvectors)
|
|
|
|
{
|
|
|
|
*avg = m_pca_avg;
|
|
|
|
*eigenvectors = m_pca_eigenvectors;
|
|
|
|
return m_pca_dim_low;
|
|
|
|
};
|
|
|
|
|
|
|
|
int GetPCADimLow() const {return m_pca_dim_low;};
|
|
|
|
int GetPCADimHigh() const {return m_pca_dim_high;};
|
|
|
|
|
|
|
|
void ConvertDescriptorsArrayToTree(); // Converting pca_descriptors array to KD tree
|
|
|
|
|
2010-06-02 13:19:09 +08:00
|
|
|
// GetPCAFilename: get default PCA filename
|
|
|
|
static string GetPCAFilename () { return "pca.yml"; }
|
2010-05-12 01:44:00 +08:00
|
|
|
|
|
|
|
protected:
|
|
|
|
CvSize m_patch_size; // patch size
|
|
|
|
int m_pose_count; // the number of poses for each descriptor
|
|
|
|
int m_train_feature_count; // the number of the training features
|
|
|
|
OneWayDescriptor* m_descriptors; // array of train feature descriptors
|
|
|
|
CvMat* m_pca_avg; // PCA average Vector for small patches
|
|
|
|
CvMat* m_pca_eigenvectors; // PCA eigenvectors for small patches
|
|
|
|
CvMat* m_pca_hr_avg; // PCA average Vector for large patches
|
|
|
|
CvMat* m_pca_hr_eigenvectors; // PCA eigenvectors for large patches
|
|
|
|
OneWayDescriptor* m_pca_descriptors; // an array of PCA descriptors
|
|
|
|
|
2010-05-27 03:50:20 +08:00
|
|
|
cv::flann::Index* m_pca_descriptors_tree;
|
2010-05-12 01:44:00 +08:00
|
|
|
CvMat* m_pca_descriptors_matrix;
|
|
|
|
|
|
|
|
CvAffinePose* m_poses; // array of poses
|
|
|
|
CvMat** m_transforms; // array of affine transformations corresponding to poses
|
|
|
|
|
|
|
|
int m_pca_dim_high;
|
|
|
|
int m_pca_dim_low;
|
|
|
|
|
|
|
|
int m_pyr_levels;
|
2010-06-08 14:56:35 +08:00
|
|
|
float scale_min;
|
|
|
|
float scale_max;
|
|
|
|
float scale_step;
|
|
|
|
|
|
|
|
// SavePCAall: saves PCA components and descriptors to a file storage
|
|
|
|
// - fs: output file storage
|
|
|
|
void SavePCAall (FileStorage &fs) const;
|
|
|
|
|
|
|
|
// LoadPCAall: loads PCA components and descriptors from a file node
|
|
|
|
// - fn: input file node
|
|
|
|
void LoadPCAall (const FileNode &fn);
|
2010-05-12 01:44:00 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
class CV_EXPORTS OneWayDescriptorObject : public OneWayDescriptorBase
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
// creates an instance of OneWayDescriptorObject from a set of training files
|
|
|
|
// - patch_size: size of the input (large) patch
|
|
|
|
// - pose_count: the number of poses to generate for each descriptor
|
|
|
|
// - train_path: path to training files
|
|
|
|
// - pca_config: the name of the file that contains PCA for small patches (2 times smaller
|
|
|
|
// than patch_size each dimension
|
|
|
|
// - pca_hr_config: the name of the file that contains PCA for large patches (of patch_size size)
|
|
|
|
// - pca_desc_config: the name of the file that contains descriptors of PCA components
|
|
|
|
OneWayDescriptorObject(CvSize patch_size, int pose_count, const char* train_path, const char* pca_config,
|
|
|
|
const char* pca_hr_config = 0, const char* pca_desc_config = 0, int pyr_levels = 1);
|
|
|
|
|
2010-06-02 13:19:09 +08:00
|
|
|
OneWayDescriptorObject(CvSize patch_size, int pose_count, const string &pca_filename,
|
2010-06-02 14:55:03 +08:00
|
|
|
const string &train_path = string (), const string &images_list = string (),
|
|
|
|
float _scale_min = 0.7f, float _scale_max=1.5f, float _scale_step=1.2f, int pyr_levels = 1);
|
2010-06-02 13:19:09 +08:00
|
|
|
|
|
|
|
|
2010-06-02 13:39:29 +08:00
|
|
|
virtual ~OneWayDescriptorObject();
|
2010-05-12 01:44:00 +08:00
|
|
|
|
|
|
|
// Allocate: allocates memory for a given number of features
|
|
|
|
// - train_feature_count: the total number of features
|
|
|
|
// - object_feature_count: the number of features extracted from the object
|
|
|
|
void Allocate(int train_feature_count, int object_feature_count);
|
|
|
|
|
|
|
|
|
|
|
|
void SetLabeledFeatures(const vector<cv::KeyPoint>& features) {m_train_features = features;};
|
|
|
|
vector<cv::KeyPoint>& GetLabeledFeatures() {return m_train_features;};
|
|
|
|
const vector<cv::KeyPoint>& GetLabeledFeatures() const {return m_train_features;};
|
|
|
|
vector<cv::KeyPoint> _GetLabeledFeatures() const;
|
|
|
|
|
|
|
|
// IsDescriptorObject: returns 1 if descriptor with specified index is positive, otherwise 0
|
|
|
|
int IsDescriptorObject(int desc_idx) const;
|
|
|
|
|
|
|
|
// MatchPointToPart: returns the part number of a feature if it matches one of the object parts, otherwise -1
|
|
|
|
int MatchPointToPart(CvPoint pt) const;
|
|
|
|
|
|
|
|
// GetDescriptorPart: returns the part number of the feature corresponding to a specified descriptor
|
|
|
|
// - desc_idx: descriptor index
|
|
|
|
int GetDescriptorPart(int desc_idx) const;
|
|
|
|
|
|
|
|
|
|
|
|
void InitializeObjectDescriptors(IplImage* train_image, const vector<cv::KeyPoint>& features,
|
|
|
|
const char* feature_label, int desc_start_idx = 0, float scale = 1.0f,
|
|
|
|
int is_background = 0);
|
|
|
|
|
|
|
|
// GetObjectFeatureCount: returns the number of object features
|
|
|
|
int GetObjectFeatureCount() const {return m_object_feature_count;};
|
|
|
|
|
|
|
|
protected:
|
|
|
|
int* m_part_id; // contains part id for each of object descriptors
|
|
|
|
vector<cv::KeyPoint> m_train_features; // train features
|
|
|
|
int m_object_feature_count; // the number of the positive features
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/****************************************************************************************\
|
|
|
|
* FeatureDetector *
|
|
|
|
\****************************************************************************************/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Abstract base class for 2D image feature detectors.
|
|
|
|
*/
|
|
|
|
class CV_EXPORTS FeatureDetector
|
|
|
|
{
|
|
|
|
public:
|
2010-07-09 05:15:09 +08:00
|
|
|
virtual ~FeatureDetector() {}
|
|
|
|
|
2010-05-12 01:44:00 +08:00
|
|
|
/*
|
2010-10-29 16:44:42 +08:00
|
|
|
* Detect keypoints in an image. Must be implemented by the subclass.
|
2010-05-12 01:44:00 +08:00
|
|
|
*
|
|
|
|
* image The image.
|
|
|
|
* keypoints The detected keypoints.
|
2010-07-26 16:58:46 +08:00
|
|
|
* mask Mask specifying where to look for keypoints (optional). Must be a char
|
|
|
|
* matrix with non-zero values in the region of interest.
|
2010-05-12 01:44:00 +08:00
|
|
|
*/
|
2010-10-29 16:44:42 +08:00
|
|
|
virtual void detect( const Mat& image, vector<KeyPoint>& keypoints, const Mat& mask=Mat() ) const = 0;
|
2010-07-09 05:15:09 +08:00
|
|
|
|
2010-10-29 16:44:42 +08:00
|
|
|
/*
|
|
|
|
* Detect keypoints in an image set.
|
|
|
|
*
|
|
|
|
* images Image collection.
|
|
|
|
* pointCollection Collection of keypoints detected in an input images.
|
|
|
|
* masks Masks for each input image.
|
|
|
|
*/
|
|
|
|
void detect( const vector<Mat>& imageCollection, vector<vector<KeyPoint> >& pointCollection, const vector<Mat>& masks=vector<Mat>() ) const;
|
|
|
|
|
2010-07-09 05:15:09 +08:00
|
|
|
virtual void read(const FileNode&) {}
|
|
|
|
virtual void write(FileStorage&) const {}
|
2010-06-04 13:30:09 +08:00
|
|
|
|
2010-05-12 01:44:00 +08:00
|
|
|
protected:
|
|
|
|
/*
|
|
|
|
* Remove keypoints that are not in the mask.
|
|
|
|
*
|
|
|
|
* Helper function, useful when wrapping a library call for keypoint detection that
|
|
|
|
* does not support a mask argument.
|
|
|
|
*/
|
|
|
|
static void removeInvalidPoints( const Mat& mask, vector<KeyPoint>& keypoints );
|
|
|
|
};
|
|
|
|
|
|
|
|
class CV_EXPORTS FastFeatureDetector : public FeatureDetector
|
|
|
|
{
|
|
|
|
public:
|
2010-06-04 13:30:09 +08:00
|
|
|
FastFeatureDetector( int _threshold = 1, bool _nonmaxSuppression = true );
|
2010-10-29 16:44:42 +08:00
|
|
|
virtual void detect( const Mat& image, vector<KeyPoint>& keypoints, const Mat& mask=Mat() ) const;
|
2010-06-04 13:30:09 +08:00
|
|
|
|
|
|
|
virtual void read (const FileNode& fn);
|
|
|
|
virtual void write (FileStorage& fs) const;
|
2010-05-12 01:44:00 +08:00
|
|
|
|
|
|
|
protected:
|
|
|
|
int threshold;
|
|
|
|
bool nonmaxSuppression;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class CV_EXPORTS GoodFeaturesToTrackDetector : public FeatureDetector
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
GoodFeaturesToTrackDetector( int _maxCorners, double _qualityLevel, double _minDistance,
|
|
|
|
int _blockSize=3, bool _useHarrisDetector=false, double _k=0.04 );
|
2010-10-29 16:44:42 +08:00
|
|
|
virtual void detect( const Mat& image, vector<KeyPoint>& keypoints, const Mat& mask=Mat() ) const;
|
2010-06-04 13:30:09 +08:00
|
|
|
|
|
|
|
virtual void read (const FileNode& fn);
|
|
|
|
virtual void write (FileStorage& fs) const;
|
|
|
|
|
2010-05-12 01:44:00 +08:00
|
|
|
protected:
|
|
|
|
int maxCorners;
|
|
|
|
double qualityLevel;
|
|
|
|
double minDistance;
|
|
|
|
int blockSize;
|
|
|
|
bool useHarrisDetector;
|
|
|
|
double k;
|
|
|
|
};
|
|
|
|
|
|
|
|
class CV_EXPORTS MserFeatureDetector : public FeatureDetector
|
|
|
|
{
|
|
|
|
public:
|
2010-06-04 13:30:09 +08:00
|
|
|
MserFeatureDetector( CvMSERParams params = cvMSERParams () );
|
2010-07-16 20:54:53 +08:00
|
|
|
MserFeatureDetector( int delta, int minArea, int maxArea, double maxVariation, double minDiversity,
|
2010-05-12 01:44:00 +08:00
|
|
|
int maxEvolution, double areaThreshold, double minMargin, int edgeBlurSize );
|
2010-10-29 16:44:42 +08:00
|
|
|
virtual void detect( const Mat& image, vector<KeyPoint>& keypoints, const Mat& mask=Mat() ) const;
|
2010-06-04 13:30:09 +08:00
|
|
|
|
|
|
|
virtual void read (const FileNode& fn);
|
|
|
|
virtual void write (FileStorage& fs) const;
|
|
|
|
|
2010-05-12 01:44:00 +08:00
|
|
|
protected:
|
|
|
|
MSER mser;
|
|
|
|
};
|
|
|
|
|
|
|
|
class CV_EXPORTS StarFeatureDetector : public FeatureDetector
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
StarFeatureDetector( int maxSize=16, int responseThreshold=30, int lineThresholdProjected = 10,
|
|
|
|
int lineThresholdBinarized=8, int suppressNonmaxSize=5 );
|
2010-10-29 16:44:42 +08:00
|
|
|
virtual void detect( const Mat& image, vector<KeyPoint>& keypoints, const Mat& mask=Mat() ) const;
|
2010-05-12 01:44:00 +08:00
|
|
|
|
2010-06-04 13:30:09 +08:00
|
|
|
virtual void read (const FileNode& fn);
|
|
|
|
virtual void write (FileStorage& fs) const;
|
|
|
|
|
2010-05-12 01:44:00 +08:00
|
|
|
protected:
|
|
|
|
StarDetector star;
|
|
|
|
};
|
|
|
|
|
|
|
|
class CV_EXPORTS SiftFeatureDetector : public FeatureDetector
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
SiftFeatureDetector( double threshold=SIFT::DetectorParams::GET_DEFAULT_THRESHOLD(),
|
|
|
|
double edgeThreshold=SIFT::DetectorParams::GET_DEFAULT_EDGE_THRESHOLD(),
|
|
|
|
int nOctaves=SIFT::CommonParams::DEFAULT_NOCTAVES,
|
|
|
|
int nOctaveLayers=SIFT::CommonParams::DEFAULT_NOCTAVE_LAYERS,
|
2010-05-20 00:02:30 +08:00
|
|
|
int firstOctave=SIFT::CommonParams::DEFAULT_FIRST_OCTAVE,
|
|
|
|
int angleMode=SIFT::CommonParams::FIRST_ANGLE );
|
2010-10-29 16:44:42 +08:00
|
|
|
virtual void detect( const Mat& image, vector<KeyPoint>& keypoints, const Mat& mask=Mat() ) const;
|
2010-06-04 13:30:09 +08:00
|
|
|
|
|
|
|
virtual void read (const FileNode& fn);
|
|
|
|
virtual void write (FileStorage& fs) const;
|
|
|
|
|
2010-05-12 01:44:00 +08:00
|
|
|
protected:
|
|
|
|
SIFT sift;
|
|
|
|
};
|
|
|
|
|
|
|
|
class CV_EXPORTS SurfFeatureDetector : public FeatureDetector
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
SurfFeatureDetector( double hessianThreshold = 400., int octaves = 3, int octaveLayers = 4 );
|
2010-10-29 16:44:42 +08:00
|
|
|
virtual void detect( const Mat& image, vector<KeyPoint>& keypoints, const Mat& mask=Mat() ) const;
|
2010-06-04 13:30:09 +08:00
|
|
|
virtual void read (const FileNode& fn);
|
|
|
|
virtual void write (FileStorage& fs) const;
|
|
|
|
|
2010-05-12 01:44:00 +08:00
|
|
|
protected:
|
|
|
|
SURF surf;
|
|
|
|
};
|
|
|
|
|
2010-09-30 22:21:22 +08:00
|
|
|
class CV_EXPORTS DenseFeatureDetector : public FeatureDetector
|
2010-09-25 00:55:12 +08:00
|
|
|
{
|
|
|
|
public:
|
|
|
|
DenseFeatureDetector() : initFeatureScale(1), featureScaleLevels(1), featureScaleMul(0.1f),
|
|
|
|
initXyStep(6), initImgBound(0), varyXyStepWithScale(true), varyImgBoundWithScale(false) {}
|
|
|
|
DenseFeatureDetector( float _initFeatureScale, int _featureScaleLevels=1, float _featureScaleMul=0.1f,
|
|
|
|
int _initXyStep=6, int _initImgBound=0, bool _varyXyStepWithScale=true, bool _varyImgBoundWithScale=false ) :
|
|
|
|
initFeatureScale(_initFeatureScale), featureScaleLevels(_featureScaleLevels), featureScaleMul(_featureScaleMul),
|
|
|
|
initXyStep(_initXyStep), initImgBound(_initImgBound), varyXyStepWithScale(_varyXyStepWithScale), varyImgBoundWithScale(_varyImgBoundWithScale) {}
|
2010-10-29 16:44:42 +08:00
|
|
|
virtual void detect( const Mat& image, vector<KeyPoint>& keypoints, const Mat& mask=Mat() ) const;
|
|
|
|
// todo read/write
|
2010-09-25 00:55:12 +08:00
|
|
|
protected:
|
|
|
|
float initFeatureScale;
|
|
|
|
int featureScaleLevels;
|
|
|
|
float featureScaleMul;
|
|
|
|
|
|
|
|
int initXyStep;
|
|
|
|
int initImgBound;
|
|
|
|
|
|
|
|
bool varyXyStepWithScale;
|
|
|
|
bool varyImgBoundWithScale;
|
|
|
|
};
|
|
|
|
|
2010-08-04 00:28:52 +08:00
|
|
|
/*
|
|
|
|
* Adapts a detector to partition the source image into a grid and detect
|
|
|
|
* points in each cell.
|
|
|
|
*/
|
|
|
|
class CV_EXPORTS GridAdaptedFeatureDetector : public FeatureDetector
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
GridAdaptedFeatureDetector( const Ptr<FeatureDetector>& _detector, int _maxTotalKeypoints,
|
|
|
|
int _gridRows=4, int _gridCols=4 );
|
2010-10-29 16:44:42 +08:00
|
|
|
virtual void detect( const Mat& image, vector<KeyPoint>& keypoints, const Mat& mask=Mat() ) const;
|
2010-08-04 00:28:52 +08:00
|
|
|
// todo read/write
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Ptr<FeatureDetector> detector;
|
|
|
|
int maxTotalKeypoints;
|
|
|
|
int gridRows;
|
|
|
|
int gridCols;
|
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Adapts a detector to detect points over multiple levels of a Gaussian
|
|
|
|
* pyramid. Useful for detectors that are not inherently scaled.
|
|
|
|
*/
|
|
|
|
class PyramidAdaptedFeatureDetector : public FeatureDetector
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
PyramidAdaptedFeatureDetector( const Ptr<FeatureDetector>& _detector, int _levels=2 );
|
2010-10-29 16:44:42 +08:00
|
|
|
virtual void detect( const Mat& image, vector<KeyPoint>& keypoints, const Mat& mask=Mat() ) const;
|
2010-08-04 00:28:52 +08:00
|
|
|
// todo read/write
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Ptr<FeatureDetector> detector;
|
|
|
|
int levels;
|
|
|
|
};
|
|
|
|
|
2010-08-05 20:19:26 +08:00
|
|
|
CV_EXPORTS Mat windowedMatchingMask( const vector<KeyPoint>& keypoints1, const vector<KeyPoint>& keypoints2,
|
|
|
|
float maxDeltaX, float maxDeltaY );
|
|
|
|
|
2010-10-29 16:44:42 +08:00
|
|
|
CV_EXPORTS Ptr<FeatureDetector> createFeatureDetector( const string& detectorType );
|
2010-05-12 01:44:00 +08:00
|
|
|
/****************************************************************************************\
|
|
|
|
* DescriptorExtractor *
|
|
|
|
\****************************************************************************************/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Abstract base class for computing descriptors for image keypoints.
|
|
|
|
*
|
|
|
|
* In this interface we assume a keypoint descriptor can be represented as a
|
|
|
|
* dense, fixed-dimensional vector of some basic type. Most descriptors used
|
|
|
|
* in practice follow this pattern, as it makes it very easy to compute
|
|
|
|
* distances between descriptors. Therefore we represent a collection of
|
|
|
|
* descriptors as a cv::Mat, where each row is one keypoint descriptor.
|
|
|
|
*/
|
|
|
|
class CV_EXPORTS DescriptorExtractor
|
|
|
|
{
|
|
|
|
public:
|
2010-07-09 05:15:09 +08:00
|
|
|
virtual ~DescriptorExtractor() {}
|
2010-05-12 01:44:00 +08:00
|
|
|
/*
|
|
|
|
* Compute the descriptors for a set of keypoints in an image.
|
|
|
|
* Must be implemented by the subclass.
|
|
|
|
*
|
|
|
|
* image The image.
|
|
|
|
* keypoints The keypoints. Keypoints for which a descriptor cannot be computed are removed.
|
|
|
|
* descriptors The descriptors. Row i is the descriptor for keypoint i.
|
|
|
|
*/
|
|
|
|
virtual void compute( const Mat& image, vector<KeyPoint>& keypoints, Mat& descriptors ) const = 0;
|
|
|
|
|
2010-10-29 16:44:42 +08:00
|
|
|
/*
|
|
|
|
* Compute the descriptors for a keypoints collection detected in image collection.
|
|
|
|
*
|
|
|
|
* imageCollection Image collection.
|
|
|
|
* pointCollection Keypoints collection. pointCollection[i] is keypoints detected in imageCollection[i].
|
|
|
|
* descCollection Descriptor collection. descCollection[i] is descriptors computed for pointCollection[i].
|
|
|
|
*/
|
|
|
|
void compute( const vector<Mat>& imageCollection, vector<vector<KeyPoint> >& pointCollection, vector<Mat>& descCollection ) const;
|
|
|
|
|
|
|
|
virtual void read( const FileNode& ) {}
|
|
|
|
virtual void write( FileStorage& ) const {}
|
2010-06-04 13:30:09 +08:00
|
|
|
|
2010-09-23 18:53:36 +08:00
|
|
|
virtual int descriptorSize() const = 0;
|
|
|
|
virtual int descriptorType() const = 0;
|
|
|
|
|
2010-05-12 01:44:00 +08:00
|
|
|
protected:
|
|
|
|
/*
|
|
|
|
* Remove keypoints within border_pixels of an image edge.
|
|
|
|
*/
|
|
|
|
static void removeBorderKeypoints( vector<KeyPoint>& keypoints,
|
|
|
|
Size imageSize, int borderPixels );
|
|
|
|
};
|
|
|
|
|
|
|
|
class CV_EXPORTS SiftDescriptorExtractor : public DescriptorExtractor
|
|
|
|
{
|
|
|
|
public:
|
2010-05-21 00:16:28 +08:00
|
|
|
SiftDescriptorExtractor( double magnification=SIFT::DescriptorParams::GET_DEFAULT_MAGNIFICATION(),
|
|
|
|
bool isNormalize=true, bool recalculateAngles=true,
|
2010-05-12 01:44:00 +08:00
|
|
|
int nOctaves=SIFT::CommonParams::DEFAULT_NOCTAVES,
|
|
|
|
int nOctaveLayers=SIFT::CommonParams::DEFAULT_NOCTAVE_LAYERS,
|
2010-05-20 00:02:30 +08:00
|
|
|
int firstOctave=SIFT::CommonParams::DEFAULT_FIRST_OCTAVE,
|
|
|
|
int angleMode=SIFT::CommonParams::FIRST_ANGLE );
|
2010-05-12 01:44:00 +08:00
|
|
|
|
2010-07-26 16:58:46 +08:00
|
|
|
virtual void compute( const Mat& image, vector<KeyPoint>& keypoints, Mat& descriptors ) const;
|
|
|
|
virtual void read( const FileNode &fn );
|
|
|
|
virtual void write( FileStorage &fs ) const;
|
2010-05-12 01:44:00 +08:00
|
|
|
|
2010-09-23 18:53:36 +08:00
|
|
|
virtual int descriptorSize() const { return sift.descriptorSize(); }
|
|
|
|
virtual int descriptorType() const { return CV_32FC1; }
|
|
|
|
|
2010-05-12 01:44:00 +08:00
|
|
|
protected:
|
|
|
|
SIFT sift;
|
|
|
|
};
|
|
|
|
|
|
|
|
class CV_EXPORTS SurfDescriptorExtractor : public DescriptorExtractor
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
SurfDescriptorExtractor( int nOctaves=4,
|
|
|
|
int nOctaveLayers=2, bool extended=false );
|
|
|
|
|
2010-07-26 16:58:46 +08:00
|
|
|
virtual void compute( const Mat& image, vector<KeyPoint>& keypoints, Mat& descriptors ) const;
|
|
|
|
virtual void read( const FileNode &fn );
|
|
|
|
virtual void write( FileStorage &fs ) const;
|
2010-05-12 01:44:00 +08:00
|
|
|
|
2010-09-23 18:53:36 +08:00
|
|
|
virtual int descriptorSize() const { return surf.descriptorSize(); }
|
|
|
|
virtual int descriptorType() const { return CV_32FC1; }
|
|
|
|
|
2010-05-12 01:44:00 +08:00
|
|
|
protected:
|
|
|
|
SURF surf;
|
|
|
|
};
|
|
|
|
|
2010-07-26 16:58:46 +08:00
|
|
|
template<typename T>
|
2010-07-27 20:36:48 +08:00
|
|
|
class CV_EXPORTS CalonderDescriptorExtractor : public DescriptorExtractor
|
2010-07-26 16:58:46 +08:00
|
|
|
{
|
|
|
|
public:
|
|
|
|
CalonderDescriptorExtractor( const string& classifierFile );
|
|
|
|
|
|
|
|
virtual void compute( const Mat& image, vector<KeyPoint>& keypoints, Mat& descriptors ) const;
|
|
|
|
virtual void read( const FileNode &fn );
|
|
|
|
virtual void write( FileStorage &fs ) const;
|
|
|
|
|
2010-09-23 18:53:36 +08:00
|
|
|
virtual int descriptorSize() const { return classifier_.classes(); }
|
|
|
|
virtual int descriptorType() const { return DataType<T>::type; }
|
|
|
|
|
2010-07-26 16:58:46 +08:00
|
|
|
protected:
|
|
|
|
RTreeClassifier classifier_;
|
|
|
|
static const int BORDER_SIZE = 16;
|
|
|
|
};
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
CalonderDescriptorExtractor<T>::CalonderDescriptorExtractor(const std::string& classifier_file)
|
|
|
|
{
|
|
|
|
classifier_.read( classifier_file.c_str() );
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
void CalonderDescriptorExtractor<T>::compute( const cv::Mat& image,
|
|
|
|
std::vector<cv::KeyPoint>& keypoints,
|
|
|
|
cv::Mat& descriptors) const
|
|
|
|
{
|
|
|
|
// Cannot compute descriptors for keypoints on the image border.
|
|
|
|
removeBorderKeypoints(keypoints, image.size(), BORDER_SIZE);
|
|
|
|
|
|
|
|
/// @todo Check 16-byte aligned
|
|
|
|
descriptors.create(keypoints.size(), classifier_.classes(), cv::DataType<T>::type);
|
|
|
|
|
2010-07-27 20:36:48 +08:00
|
|
|
int patchSize = RandomizedTree::PATCH_SIZE;
|
|
|
|
int offset = patchSize / 2;
|
2010-07-26 16:58:46 +08:00
|
|
|
for (size_t i = 0; i < keypoints.size(); ++i) {
|
2010-07-27 20:36:48 +08:00
|
|
|
cv::Point2f pt = keypoints[i].pt;
|
2010-08-05 21:29:43 +08:00
|
|
|
IplImage ipl = image( Rect((int)(pt.x - offset), (int)(pt.y - offset), patchSize, patchSize) );
|
2010-07-27 20:36:48 +08:00
|
|
|
classifier_.getSignature( &ipl, descriptors.ptr<T>(i));
|
2010-07-26 16:58:46 +08:00
|
|
|
}
|
|
|
|
}
|
2010-07-27 20:36:48 +08:00
|
|
|
|
|
|
|
template<typename T>
|
2010-07-27 21:28:21 +08:00
|
|
|
void CalonderDescriptorExtractor<T>::read( const FileNode& )
|
2010-07-27 20:36:48 +08:00
|
|
|
{}
|
|
|
|
|
|
|
|
template<typename T>
|
2010-08-05 21:29:43 +08:00
|
|
|
void CalonderDescriptorExtractor<T>::write( FileStorage& ) const
|
2010-07-27 20:36:48 +08:00
|
|
|
{}
|
2010-07-26 16:58:46 +08:00
|
|
|
|
2010-09-23 18:53:36 +08:00
|
|
|
/*
|
|
|
|
* Adapts a descriptor extractor to compute descripors in Opponent Color Space
|
|
|
|
* (refer to van de Sande et al., CGIV 2008 "Color Descriptors for Object Category Recognition").
|
|
|
|
* Input RGB image is transformed in Opponent Color Space. Then unadapted descriptor extractor
|
|
|
|
* (set in constructor) computes descriptors on each of the three channel and concatenate
|
|
|
|
* them into a single color descriptor.
|
|
|
|
*/
|
|
|
|
class OpponentColorDescriptorExtractor : public DescriptorExtractor
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
OpponentColorDescriptorExtractor( const Ptr<DescriptorExtractor>& dextractor );
|
|
|
|
|
|
|
|
virtual void compute( const Mat& image, vector<KeyPoint>& keypoints, Mat& descriptors ) const;
|
|
|
|
|
|
|
|
virtual void read( const FileNode& );
|
|
|
|
virtual void write( FileStorage& ) const;
|
|
|
|
|
|
|
|
virtual int descriptorSize() const { return 3*dextractor->descriptorSize(); }
|
|
|
|
virtual int descriptorType() const { return dextractor->descriptorType(); }
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Ptr<DescriptorExtractor> dextractor;
|
|
|
|
};
|
|
|
|
|
2010-07-12 19:56:11 +08:00
|
|
|
CV_EXPORTS Ptr<DescriptorExtractor> createDescriptorExtractor( const string& descriptorExtractorType );
|
2010-06-12 02:44:22 +08:00
|
|
|
|
2010-05-12 01:44:00 +08:00
|
|
|
/****************************************************************************************\
|
|
|
|
* Distance *
|
|
|
|
\****************************************************************************************/
|
|
|
|
template<typename T>
|
|
|
|
struct CV_EXPORTS Accumulator
|
|
|
|
{
|
|
|
|
typedef T Type;
|
|
|
|
};
|
|
|
|
|
|
|
|
template<> struct Accumulator<unsigned char> { typedef unsigned int Type; };
|
|
|
|
template<> struct Accumulator<unsigned short> { typedef unsigned int Type; };
|
|
|
|
template<> struct Accumulator<char> { typedef int Type; };
|
|
|
|
template<> struct Accumulator<short> { typedef int Type; };
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Squared Euclidean distance functor
|
|
|
|
*/
|
|
|
|
template<class T>
|
|
|
|
struct CV_EXPORTS L2
|
|
|
|
{
|
|
|
|
typedef T ValueType;
|
|
|
|
typedef typename Accumulator<T>::Type ResultType;
|
|
|
|
|
|
|
|
ResultType operator()( const T* a, const T* b, int size ) const
|
|
|
|
{
|
|
|
|
ResultType result = ResultType();
|
|
|
|
for( int i = 0; i < size; i++ )
|
|
|
|
{
|
|
|
|
ResultType diff = a[i] - b[i];
|
|
|
|
result += diff*diff;
|
|
|
|
}
|
2010-06-10 00:57:07 +08:00
|
|
|
return sqrt(result);
|
2010-05-12 01:44:00 +08:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2010-06-28 21:06:24 +08:00
|
|
|
/*
|
|
|
|
* Manhattan distance (city block distance) functor
|
|
|
|
*/
|
|
|
|
template<class T>
|
|
|
|
struct CV_EXPORTS L1
|
|
|
|
{
|
|
|
|
typedef T ValueType;
|
|
|
|
typedef typename Accumulator<T>::Type ResultType;
|
|
|
|
|
|
|
|
ResultType operator()( const T* a, const T* b, int size ) const
|
|
|
|
{
|
|
|
|
ResultType result = ResultType();
|
|
|
|
for( int i = 0; i < size; i++ )
|
|
|
|
{
|
|
|
|
ResultType diff = a[i] - b[i];
|
|
|
|
result += fabs( diff );
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2010-06-11 17:38:39 +08:00
|
|
|
|
|
|
|
/****************************************************************************************\
|
2010-07-26 16:58:46 +08:00
|
|
|
* DMatch *
|
2010-06-11 17:38:39 +08:00
|
|
|
\****************************************************************************************/
|
|
|
|
/*
|
2010-10-29 16:44:42 +08:00
|
|
|
* Struct for matching: query descriptor index, train descriptor index, train image index and distance between descriptors.
|
2010-06-11 17:38:39 +08:00
|
|
|
*/
|
2010-08-16 17:15:15 +08:00
|
|
|
struct CV_EXPORTS DMatch
|
2010-06-11 17:38:39 +08:00
|
|
|
{
|
2010-10-29 16:44:42 +08:00
|
|
|
DMatch() : queryIdx(-1), trainIdx(-1), imgIdx(-1), distance(std::numeric_limits<float>::max()) {}
|
|
|
|
DMatch( int _queryIdx, int _trainIdx, float _distance ) :
|
|
|
|
queryIdx(_queryIdx), trainIdx(_trainIdx), imgIdx(-1), distance(_distance) {}
|
|
|
|
DMatch( int _queryIdx, int _trainIdx, int _imgIdx, float _distance ) :
|
|
|
|
queryIdx(_queryIdx), trainIdx(_trainIdx), imgIdx(_imgIdx), distance(_distance) {}
|
|
|
|
|
|
|
|
int queryIdx; // query descriptor index
|
|
|
|
int trainIdx; // train descriptor index
|
|
|
|
int imgIdx; // train image index
|
|
|
|
|
2010-06-11 17:38:39 +08:00
|
|
|
float distance;
|
|
|
|
|
|
|
|
//less is better
|
2010-06-12 03:56:52 +08:00
|
|
|
bool operator<( const DMatch &m) const
|
2010-06-11 17:38:39 +08:00
|
|
|
{
|
|
|
|
return distance < m.distance;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2010-05-12 01:44:00 +08:00
|
|
|
/****************************************************************************************\
|
|
|
|
* DescriptorMatcher *
|
|
|
|
\****************************************************************************************/
|
|
|
|
/*
|
|
|
|
* Abstract base class for matching two sets of descriptors.
|
|
|
|
*/
|
|
|
|
class CV_EXPORTS DescriptorMatcher
|
|
|
|
{
|
|
|
|
public:
|
2010-07-09 17:09:20 +08:00
|
|
|
virtual ~DescriptorMatcher() {}
|
2010-05-12 01:44:00 +08:00
|
|
|
/*
|
2010-10-29 16:44:42 +08:00
|
|
|
* Add descriptors to train descriptor collection.
|
|
|
|
* descCollection Descriptors to add. Each descCollection[i] is from one image.
|
2010-05-12 01:44:00 +08:00
|
|
|
*/
|
2010-10-29 16:44:42 +08:00
|
|
|
virtual void add( const vector<Mat>& descCollection );
|
2010-05-12 01:44:00 +08:00
|
|
|
/*
|
2010-10-29 16:44:42 +08:00
|
|
|
* Get descriptor collection.
|
2010-05-12 01:44:00 +08:00
|
|
|
*/
|
2010-10-29 16:44:42 +08:00
|
|
|
const vector<Mat>& getTrainDescCollection() const { return trainDescCollection; }
|
2010-05-12 01:44:00 +08:00
|
|
|
/*
|
2010-10-29 16:44:42 +08:00
|
|
|
* Clear inner data (train image collection).
|
2010-05-12 01:44:00 +08:00
|
|
|
*/
|
2010-10-29 16:44:42 +08:00
|
|
|
virtual void clear();
|
|
|
|
|
|
|
|
virtual bool supportMask() = 0;
|
2010-05-12 01:44:00 +08:00
|
|
|
|
|
|
|
/*
|
2010-10-29 16:44:42 +08:00
|
|
|
* Train matcher (e.g. train flann index)
|
2010-05-12 01:44:00 +08:00
|
|
|
*/
|
2010-10-29 16:44:42 +08:00
|
|
|
virtual void train() = 0;
|
2010-06-11 17:38:39 +08:00
|
|
|
/*
|
2010-10-29 16:44:42 +08:00
|
|
|
* Group of methods to match descriptors from image pair.
|
2010-06-11 17:38:39 +08:00
|
|
|
*/
|
2010-10-29 16:44:42 +08:00
|
|
|
// Find one best match for each query descriptor (if mask is empty).
|
|
|
|
void match( const Mat& queryDescs, const Mat& trainDescs, vector<DMatch>& matches,
|
|
|
|
const Mat& mask=Mat() ) const;
|
|
|
|
// Find knn best matches for each query descriptor (in increasing order of distances).
|
|
|
|
// compactResult is used when mask is not empty. If compactResult is false matches vector will have the same size as queryDescs rows.
|
|
|
|
// If compactResult is true matches vector will not contain matches for fully masked out query descriptors.
|
|
|
|
void knnMatch( const Mat& queryDescs, const Mat& trainDescs, vector<vector<DMatch> >& matches, int knn,
|
|
|
|
const Mat& mask=Mat(), bool compactResult=false ) const;
|
|
|
|
// Find best matches for each query descriptor which have distance less than maxDistance (in increasing order of distances).
|
|
|
|
void radiusMatch( const Mat& queryDescs, const Mat& trainDescs, vector<vector<DMatch> >& matches, float maxDistance,
|
|
|
|
const Mat& mask=Mat(), bool compactResult=false ) const;
|
2010-06-11 17:38:39 +08:00
|
|
|
/*
|
2010-10-29 16:44:42 +08:00
|
|
|
* Group of methods to match descriptors from one image to image set.
|
|
|
|
* See description of similar methods for matching image pair above.
|
2010-06-11 17:38:39 +08:00
|
|
|
*/
|
2010-10-29 16:44:42 +08:00
|
|
|
void match( const Mat& queryDescs, vector<DMatch>& matches,
|
|
|
|
const vector<Mat>& masks=vector<Mat>() );
|
|
|
|
void knnMatch( const Mat& queryDescs, vector<vector<DMatch> >& matches, int knn,
|
|
|
|
const vector<Mat>& masks=vector<Mat>(), bool compactResult=false );
|
|
|
|
void radiusMatch( const Mat& queryDescs, vector<vector<DMatch> >& matches, float maxDistance,
|
|
|
|
const vector<Mat>& masks=vector<Mat>(), bool compactResult=false );
|
2010-06-16 16:56:53 +08:00
|
|
|
|
2010-10-29 16:44:42 +08:00
|
|
|
// Reads matcher object from a file node
|
|
|
|
virtual void read( const FileNode& ) {}
|
2010-08-24 20:52:29 +08:00
|
|
|
|
2010-10-29 16:44:42 +08:00
|
|
|
// Writes matcher object to a file storage
|
|
|
|
virtual void write( FileStorage& ) const {}
|
|
|
|
|
|
|
|
protected:
|
2010-06-16 16:56:53 +08:00
|
|
|
/*
|
2010-10-29 16:44:42 +08:00
|
|
|
* Class to work with descriptors from several images as with one merged matrix.
|
|
|
|
* It is used e.g. in FlannBasedMatcher
|
2010-06-16 16:56:53 +08:00
|
|
|
*/
|
2010-10-29 16:44:42 +08:00
|
|
|
class DescriptorCollection
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
DescriptorCollection() {}
|
|
|
|
virtual ~DescriptorCollection() {}
|
2010-06-16 16:56:53 +08:00
|
|
|
|
2010-10-29 16:44:42 +08:00
|
|
|
// descCollection will be merged to dmatrix here
|
|
|
|
void set( const vector<Mat>& descCollection );
|
|
|
|
virtual void clear();
|
2010-06-16 16:56:53 +08:00
|
|
|
|
2010-10-29 16:44:42 +08:00
|
|
|
const Mat& getDescriptors() const { return dmatrix; }
|
|
|
|
const Mat getDescriptor( int imgIdx, int localDescIdx ) const;
|
|
|
|
const Mat getDescriptor( int globalDescIdx ) const;
|
|
|
|
void getLocalIdx( int globalDescIdx, int& imgIdx, int& localDescIdx ) const;
|
2010-05-12 01:44:00 +08:00
|
|
|
|
2010-10-29 16:44:42 +08:00
|
|
|
int size() const { return dmatrix.rows; }
|
2010-05-12 01:44:00 +08:00
|
|
|
|
2010-10-29 16:44:42 +08:00
|
|
|
protected:
|
|
|
|
Mat dmatrix;
|
|
|
|
vector<int> startIdxs;
|
|
|
|
};
|
2010-05-12 01:44:00 +08:00
|
|
|
|
2010-10-29 16:44:42 +08:00
|
|
|
// create matcher clone with current parameters but with empty data
|
|
|
|
virtual Ptr<DescriptorMatcher> cloneWithoutData() const = 0;
|
2010-06-11 17:38:39 +08:00
|
|
|
|
2010-10-29 16:44:42 +08:00
|
|
|
virtual void knnMatchImpl( const Mat& queryDescs, vector<vector<DMatch> >& matches, int knn,
|
|
|
|
const vector<Mat>& masks, bool compactResult ) = 0;
|
|
|
|
virtual void radiusMatchImpl( const Mat& queryDescs, vector<vector<DMatch> >& matches, float maxDistance,
|
|
|
|
const vector<Mat>& masks, bool compactResult ) = 0;
|
2010-06-16 16:56:53 +08:00
|
|
|
|
|
|
|
|
2010-05-12 01:44:00 +08:00
|
|
|
static bool possibleMatch( const Mat& mask, int index_1, int index_2 )
|
|
|
|
{
|
2010-10-29 16:44:42 +08:00
|
|
|
return mask.empty() || mask.at<uchar>(index_1, index_2);
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool maskedOut( const vector<Mat>& masks, int queryDescIdx )
|
|
|
|
{
|
|
|
|
size_t outCount = 0;
|
|
|
|
for( size_t i = 0; i < masks.size(); i++ )
|
|
|
|
{
|
|
|
|
if( !masks[i].empty() && (countNonZero(masks[i].row(queryDescIdx)) == 0) )
|
|
|
|
outCount++;
|
|
|
|
}
|
|
|
|
|
|
|
|
return !masks.empty() && outCount == masks.size() ;
|
2010-05-12 01:44:00 +08:00
|
|
|
}
|
2010-10-29 16:44:42 +08:00
|
|
|
|
|
|
|
vector<Mat> trainDescCollection;
|
2010-05-12 01:44:00 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
2010-10-29 16:44:42 +08:00
|
|
|
* Next two functions are used to implement BruteForceMatcher class specialization
|
2010-05-12 01:44:00 +08:00
|
|
|
*/
|
|
|
|
template<class Distance>
|
2010-10-29 16:44:42 +08:00
|
|
|
class BruteForceMatcher;
|
2010-06-16 16:56:53 +08:00
|
|
|
|
2010-10-29 16:44:42 +08:00
|
|
|
template<class Distance>
|
|
|
|
inline void bfKnnMatchImpl( BruteForceMatcher<Distance>& matcher,
|
|
|
|
const Mat& queryDescs, vector<vector<DMatch> >& matches, int knn,
|
|
|
|
const vector<Mat>& masks, bool compactResult )
|
|
|
|
{
|
|
|
|
typedef typename Distance::ValueType ValueType;
|
|
|
|
typedef typename Distance::ResultType DistanceType;
|
|
|
|
CV_Assert( DataType<ValueType>::type == queryDescs.type() || queryDescs.empty() );
|
|
|
|
CV_Assert( masks.empty() || masks.size() == matcher.trainDescCollection.size() );
|
|
|
|
int dimension = queryDescs.cols;
|
|
|
|
matches.reserve(queryDescs.rows);
|
|
|
|
|
|
|
|
size_t imgCount = matcher.trainDescCollection.size();
|
|
|
|
vector<Mat> allDists( imgCount ); // distances between one query descriptor and all train descriptors
|
|
|
|
for( size_t i = 0; i < imgCount; i++ )
|
|
|
|
allDists[i] = Mat( 1, matcher.trainDescCollection[i].rows, queryDescs.type() );
|
|
|
|
|
|
|
|
for( int qIdx = 0; qIdx < queryDescs.rows; qIdx++ )
|
|
|
|
{
|
|
|
|
if( matcher.maskedOut( masks, qIdx ) )
|
|
|
|
{
|
|
|
|
if( !compactResult ) // push empty vector
|
|
|
|
matches.push_back( vector<DMatch>() );
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// 1. compute distances between i-th query descriptor and all train descriptors
|
|
|
|
for( size_t iIdx = 0; iIdx < imgCount; iIdx++ )
|
|
|
|
{
|
|
|
|
CV_Assert( masks.empty() || masks[iIdx].empty() ||
|
|
|
|
( masks[iIdx].rows == queryDescs.rows && masks[iIdx].cols == matcher.trainDescCollection[iIdx].rows &&
|
|
|
|
masks[iIdx].type() == CV_8UC1 ) );
|
|
|
|
CV_Assert( DataType<ValueType>::type == matcher.trainDescCollection[iIdx].type() || matcher.trainDescCollection[iIdx].empty() );
|
|
|
|
CV_Assert( queryDescs.cols == matcher.trainDescCollection[iIdx].cols );
|
|
|
|
|
|
|
|
const ValueType* d1 = (const ValueType*)(queryDescs.data + queryDescs.step*qIdx);
|
|
|
|
allDists[iIdx].setTo( Scalar::all(std::numeric_limits<DistanceType>::max()) );
|
|
|
|
for( int tIdx = 0; tIdx < matcher.trainDescCollection[iIdx].rows; tIdx++ )
|
|
|
|
{
|
|
|
|
if( masks.empty() || matcher.possibleMatch(masks[iIdx], qIdx, tIdx) )
|
|
|
|
{
|
|
|
|
const ValueType* d2 = (const ValueType*)(matcher.trainDescCollection[iIdx].data +
|
|
|
|
matcher.trainDescCollection[iIdx].step*tIdx);
|
|
|
|
allDists[iIdx].at<ValueType>(0, tIdx) = matcher.distance(d1, d2, dimension);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// 2. choose knn nearest matches for query[i]
|
|
|
|
matches.push_back( vector<DMatch>() );
|
|
|
|
vector<vector<DMatch> >::reverse_iterator curMatches = matches.rbegin();
|
|
|
|
for( int k = 0; k < knn; k++ )
|
|
|
|
{
|
|
|
|
DMatch bestMatch;
|
|
|
|
for( size_t iIdx = 0; iIdx < imgCount; iIdx++ )
|
|
|
|
{
|
|
|
|
double minVal;
|
|
|
|
Point minLoc;
|
|
|
|
minMaxLoc( allDists[iIdx], &minVal, 0, &minLoc, 0 );
|
|
|
|
if( minVal < bestMatch.distance )
|
|
|
|
bestMatch = DMatch( qIdx, minLoc.x, iIdx, minVal );
|
|
|
|
}
|
|
|
|
if( bestMatch.trainIdx == -1 )
|
|
|
|
break;
|
|
|
|
|
|
|
|
allDists[bestMatch.imgIdx].at<ValueType>(0, bestMatch.trainIdx) = std::numeric_limits<DistanceType>::max();
|
|
|
|
curMatches->push_back( bestMatch );
|
|
|
|
}
|
|
|
|
std::sort( curMatches->begin(), curMatches->end() );
|
|
|
|
}
|
|
|
|
}
|
2010-06-11 17:38:39 +08:00
|
|
|
}
|
|
|
|
|
2010-10-29 16:44:42 +08:00
|
|
|
template<class Distance>
|
|
|
|
inline void bfRadiusMatchImpl( BruteForceMatcher<Distance>& matcher,
|
|
|
|
const Mat& queryDescs, vector<vector<DMatch> >& matches, float maxDistance,
|
|
|
|
const vector<Mat>& masks, bool compactResult )
|
2010-05-12 01:44:00 +08:00
|
|
|
{
|
|
|
|
typedef typename Distance::ValueType ValueType;
|
|
|
|
typedef typename Distance::ResultType DistanceType;
|
2010-10-29 16:44:42 +08:00
|
|
|
CV_Assert( DataType<ValueType>::type == queryDescs.type() || queryDescs.empty() );
|
|
|
|
CV_Assert( masks.empty() || masks.size() == matcher.trainDescCollection.size() );
|
2010-05-12 01:44:00 +08:00
|
|
|
|
2010-10-29 16:44:42 +08:00
|
|
|
int dimension = queryDescs.cols;
|
|
|
|
matches.reserve(queryDescs.rows);
|
2010-05-12 01:44:00 +08:00
|
|
|
|
2010-10-29 16:44:42 +08:00
|
|
|
size_t imgCount = matcher.trainDescCollection.size();
|
|
|
|
for( int qIdx = 0; qIdx < queryDescs.rows; qIdx++ )
|
2010-05-12 01:44:00 +08:00
|
|
|
{
|
2010-10-29 16:44:42 +08:00
|
|
|
if( matcher.maskedOut( masks, qIdx ) )
|
2010-05-12 01:44:00 +08:00
|
|
|
{
|
2010-10-29 16:44:42 +08:00
|
|
|
if( !compactResult ) // push empty vector
|
|
|
|
matches.push_back( vector<DMatch>() );
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
matches.push_back( vector<DMatch>() );
|
|
|
|
vector<vector<DMatch> >::reverse_iterator curMatches = matches.rbegin();
|
|
|
|
for( size_t iIdx = 0; iIdx < imgCount; iIdx++ )
|
2010-05-12 01:44:00 +08:00
|
|
|
{
|
2010-10-29 16:44:42 +08:00
|
|
|
CV_Assert( masks.empty() || masks[iIdx].empty() ||
|
|
|
|
( masks[iIdx].rows == queryDescs.rows && masks[iIdx].cols == matcher.trainDescCollection[iIdx].rows &&
|
|
|
|
masks[iIdx].type() == CV_8UC1 ) );
|
|
|
|
CV_Assert( DataType<ValueType>::type == matcher.trainDescCollection[iIdx].type() ||
|
|
|
|
matcher.trainDescCollection[iIdx].empty() );
|
|
|
|
CV_Assert( queryDescs.cols == matcher.trainDescCollection[iIdx].cols );
|
|
|
|
|
|
|
|
const ValueType* d1 = (const ValueType*)(queryDescs.data + queryDescs.step*qIdx);
|
|
|
|
for( int tIdx = 0; tIdx < matcher.trainDescCollection[iIdx].rows; tIdx++ )
|
2010-05-12 01:44:00 +08:00
|
|
|
{
|
2010-10-29 16:44:42 +08:00
|
|
|
if( masks.empty() || matcher.possibleMatch(masks[iIdx], qIdx, tIdx) )
|
|
|
|
{
|
|
|
|
const ValueType* d2 = (const ValueType*)(matcher.trainDescCollection[iIdx].data +
|
|
|
|
matcher.trainDescCollection[iIdx].step*tIdx);
|
|
|
|
DistanceType d = matcher.distance(d1, d2, dimension);
|
|
|
|
if( d < maxDistance )
|
|
|
|
curMatches->push_back( DMatch( qIdx, tIdx, iIdx, d ) );
|
|
|
|
}
|
2010-05-12 01:44:00 +08:00
|
|
|
}
|
|
|
|
}
|
2010-10-29 16:44:42 +08:00
|
|
|
std::sort( curMatches->begin(), curMatches->end() );
|
2010-06-11 17:38:39 +08:00
|
|
|
}
|
2010-05-12 01:44:00 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-10-29 16:44:42 +08:00
|
|
|
/*
|
|
|
|
* Brute-force descriptor matcher.
|
|
|
|
*
|
|
|
|
* For each descriptor in the first set, this matcher finds the closest
|
|
|
|
* descriptor in the second set by trying each one.
|
|
|
|
*
|
|
|
|
* For efficiency, BruteForceMatcher is templated on the distance metric.
|
|
|
|
* For float descriptors, a common choice would be cv::L2<float>.
|
|
|
|
*/
|
|
|
|
template<class Distance>
|
|
|
|
class CV_EXPORTS BruteForceMatcher : public DescriptorMatcher
|
2010-06-16 16:56:53 +08:00
|
|
|
{
|
2010-10-29 16:44:42 +08:00
|
|
|
public:
|
|
|
|
template<class bfDistance>
|
|
|
|
friend void bfKnnMatchImpl( BruteForceMatcher<bfDistance>& matcher,
|
|
|
|
const Mat& queryDescs, vector<vector<DMatch> >& matches, int knn,
|
|
|
|
const vector<Mat>& masks, bool compactResult );
|
|
|
|
template<class bfDistance>
|
|
|
|
friend void bfRadiusMatchImpl( BruteForceMatcher<bfDistance>& matcher,
|
|
|
|
const Mat& queryDescs, vector<vector<DMatch> >& matches, float maxDistance,
|
|
|
|
const vector<Mat>& masks, bool compactResult );
|
2010-06-16 16:56:53 +08:00
|
|
|
|
2010-10-29 16:44:42 +08:00
|
|
|
BruteForceMatcher( Distance d = Distance() ) : distance(d) {}
|
|
|
|
virtual ~BruteForceMatcher() {}
|
2010-06-16 16:56:53 +08:00
|
|
|
|
2010-10-29 16:44:42 +08:00
|
|
|
virtual void train() {}
|
|
|
|
virtual bool supportMask() { return true; }
|
2010-06-16 16:56:53 +08:00
|
|
|
|
2010-10-29 16:44:42 +08:00
|
|
|
protected:
|
|
|
|
virtual Ptr<DescriptorMatcher> cloneWithoutData() const { return new BruteForceMatcher(distance); }
|
2010-06-16 16:56:53 +08:00
|
|
|
|
2010-10-29 16:44:42 +08:00
|
|
|
virtual void knnMatchImpl( const Mat& queryDescs, vector<vector<DMatch> >& matches, int knn,
|
|
|
|
const vector<Mat>& masks, bool compactResult );
|
|
|
|
virtual void radiusMatchImpl( const Mat& queryDescs, vector<vector<DMatch> >& matches, float maxDistance,
|
|
|
|
const vector<Mat>& masks, bool compactResult );
|
|
|
|
Distance distance;
|
|
|
|
};
|
2010-06-16 16:56:53 +08:00
|
|
|
|
2010-10-29 16:44:42 +08:00
|
|
|
template<class Distance>
|
|
|
|
void BruteForceMatcher<Distance>::knnMatchImpl( const Mat& queryDescs, vector<vector<DMatch> >& matches, int knn,
|
|
|
|
const vector<Mat>& masks, bool compactResult )
|
|
|
|
{
|
|
|
|
bfKnnMatchImpl<Distance>( *this, queryDescs, matches, knn, masks, compactResult );
|
2010-06-16 16:56:53 +08:00
|
|
|
}
|
|
|
|
|
2010-10-29 16:44:42 +08:00
|
|
|
template<class Distance>
|
|
|
|
void BruteForceMatcher<Distance>::radiusMatchImpl( const Mat& queryDescs, vector<vector<DMatch> >& matches, float maxDistance,
|
|
|
|
const vector<Mat>& masks, bool compactResult )
|
|
|
|
{
|
|
|
|
bfRadiusMatchImpl<Distance>( *this, queryDescs, matches, maxDistance, masks, compactResult );
|
|
|
|
}
|
2010-06-16 16:56:53 +08:00
|
|
|
|
2010-10-29 16:44:42 +08:00
|
|
|
/*
|
|
|
|
* BruteForceMatcher L2 specialization
|
|
|
|
*/
|
|
|
|
template<>
|
|
|
|
void BruteForceMatcher<L2<float> >::knnMatchImpl( const Mat& queryDescs, vector<vector<DMatch> >& matches, int knn,
|
|
|
|
const vector<Mat>& masks, bool compactResult );
|
|
|
|
template<>
|
|
|
|
void BruteForceMatcher<L2<float> >::radiusMatchImpl( const Mat& queryDescs, vector<vector<DMatch> >& matches, float maxDistance,
|
|
|
|
const vector<Mat>& masks, bool compactResult );
|
2010-05-12 01:44:00 +08:00
|
|
|
|
|
|
|
/*
|
2010-10-29 16:44:42 +08:00
|
|
|
* Flann based matcher
|
2010-05-12 01:44:00 +08:00
|
|
|
*/
|
2010-10-29 16:44:42 +08:00
|
|
|
class CV_EXPORTS FlannBasedMatcher : public DescriptorMatcher
|
2010-05-12 01:44:00 +08:00
|
|
|
{
|
|
|
|
public:
|
2010-10-29 16:44:42 +08:00
|
|
|
FlannBasedMatcher( const Ptr<flann::IndexParams>& _indexParams=new flann::KDTreeIndexParams(),
|
|
|
|
const Ptr<flann::SearchParams>& _searchParams=new flann::SearchParams() );
|
2010-05-12 01:44:00 +08:00
|
|
|
|
2010-10-29 16:44:42 +08:00
|
|
|
virtual void add( const vector<Mat>& descCollection );
|
|
|
|
virtual void clear();
|
2010-05-12 01:44:00 +08:00
|
|
|
|
2010-10-29 16:44:42 +08:00
|
|
|
virtual void train();
|
|
|
|
virtual bool supportMask() { return false; }
|
|
|
|
protected:
|
|
|
|
virtual Ptr<DescriptorMatcher> cloneWithoutData() const { return new FlannBasedMatcher(indexParams, searchParams); }
|
2010-05-12 01:44:00 +08:00
|
|
|
|
2010-10-29 16:44:42 +08:00
|
|
|
// masks is ignored (unsupported)
|
|
|
|
virtual void knnMatchImpl( const Mat& queryDescs, vector<vector<DMatch> >& matches, int knn,
|
|
|
|
const vector<Mat>& masks, bool compactResult );
|
|
|
|
virtual void radiusMatchImpl( const Mat& queryDescs, vector<vector<DMatch> >& matches, float maxDistance,
|
|
|
|
const vector<Mat>& masks, bool compactResult );
|
2010-05-26 19:27:56 +08:00
|
|
|
|
2010-10-29 16:44:42 +08:00
|
|
|
static void convertToDMatches( const DescriptorCollection& collection, const Mat& indices, const Mat& dists,
|
|
|
|
vector<vector<DMatch> >& matches );
|
2010-05-12 01:44:00 +08:00
|
|
|
|
2010-10-29 16:44:42 +08:00
|
|
|
Ptr<flann::IndexParams> indexParams;
|
|
|
|
Ptr<flann::SearchParams> searchParams;
|
|
|
|
Ptr<flann::Index> flannIndex;
|
|
|
|
|
|
|
|
DescriptorCollection mergedDescriptors;
|
|
|
|
int addedDescCount;
|
2010-05-12 01:44:00 +08:00
|
|
|
};
|
|
|
|
|
2010-10-29 16:44:42 +08:00
|
|
|
|
|
|
|
CV_EXPORTS Ptr<DescriptorMatcher> createDescriptorMatcher( const string& descriptorMatcherType );
|
|
|
|
|
|
|
|
/****************************************************************************************\
|
|
|
|
* GenericDescriptorMatcher *
|
|
|
|
\****************************************************************************************/
|
2010-05-12 01:44:00 +08:00
|
|
|
/*
|
|
|
|
* Abstract interface for a keypoint descriptor
|
|
|
|
*/
|
2010-10-29 16:44:42 +08:00
|
|
|
class GenericDescriptorMatcher;
|
|
|
|
typedef GenericDescriptorMatcher GenericDescriptorMatch;
|
|
|
|
|
|
|
|
class CV_EXPORTS GenericDescriptorMatcher
|
2010-05-12 01:44:00 +08:00
|
|
|
{
|
|
|
|
public:
|
2010-10-29 16:44:42 +08:00
|
|
|
GenericDescriptorMatcher() {}
|
|
|
|
virtual ~GenericDescriptorMatcher() {}
|
2010-05-12 01:44:00 +08:00
|
|
|
|
2010-10-29 16:44:42 +08:00
|
|
|
/*
|
|
|
|
* Set train collection: images and keypoints from them.
|
|
|
|
* imgCollection Image collection.
|
|
|
|
* pointCollection Keypoint collection detected on imgCollection.
|
|
|
|
*/
|
|
|
|
virtual void add( const vector<Mat>& imgCollection,
|
|
|
|
vector<vector<KeyPoint> >& pointCollection );
|
2010-05-12 01:44:00 +08:00
|
|
|
|
2010-10-29 16:44:42 +08:00
|
|
|
const vector<Mat>& getTrainImgCollection() const { return trainPointCollection.getImages(); }
|
|
|
|
const vector<vector<KeyPoint> >& getTrainPointCollection() const { return trainPointCollection.getKeypoints(); }
|
2010-05-12 01:44:00 +08:00
|
|
|
|
2010-10-29 16:44:42 +08:00
|
|
|
// Clears keypoints storing in collection
|
|
|
|
virtual void clear();
|
2010-05-12 01:44:00 +08:00
|
|
|
|
2010-10-29 16:44:42 +08:00
|
|
|
virtual void train() = 0;
|
2010-05-12 01:44:00 +08:00
|
|
|
|
2010-10-29 16:44:42 +08:00
|
|
|
/*
|
|
|
|
* Classifies query keypoints.
|
|
|
|
* queryImage The query image
|
|
|
|
* queryPoints Keypoints from the query image
|
|
|
|
* trainImage The train image
|
|
|
|
* trainPoints Keypoints from the train image
|
|
|
|
*/
|
|
|
|
// Classify keypoints from query image under one train image.
|
|
|
|
virtual void classify( const Mat& queryImage, vector<KeyPoint>& queryPoints,
|
|
|
|
const Mat& trainImage, vector<KeyPoint>& trainPoints ) const;
|
|
|
|
// Classify keypoints from query image under train image collection.
|
|
|
|
virtual void classify( const Mat& queryImage, vector<KeyPoint>& queryPoints );
|
2010-05-12 01:44:00 +08:00
|
|
|
|
2010-10-29 16:44:42 +08:00
|
|
|
/*
|
|
|
|
* Group of methods to match keypoints from image pair.
|
|
|
|
*/
|
|
|
|
// Find one best match for each query descriptor (if mask is empty).
|
|
|
|
void match( const Mat& queryImg, vector<KeyPoint>& queryPoints,
|
|
|
|
const Mat& trainImg, vector<KeyPoint>& trainPoints,
|
|
|
|
vector<DMatch>& matches, const Mat& mask=Mat() ) const;
|
|
|
|
// Find knn best matches for each query keypoint (in increasing order of distances).
|
|
|
|
// compactResult is used when mask is not empty. If compactResult is false matches vector will have the same size as queryDescs rows.
|
|
|
|
// If compactResult is true matches vector will not contain matches for fully masked out query descriptors.
|
|
|
|
void knnMatch( const Mat& queryImg, vector<KeyPoint>& queryPoints,
|
|
|
|
const Mat& trainImg, vector<KeyPoint>& trainPoints,
|
|
|
|
vector<vector<DMatch> >& matches, int knn, const Mat& mask=Mat(), bool compactResult=false ) const;
|
|
|
|
// Find best matches for each query descriptor which have distance less than maxDistance (in increasing order of distances).
|
|
|
|
void radiusMatch( const Mat& queryImg, vector<KeyPoint>& queryPoints,
|
|
|
|
const Mat& trainImg, vector<KeyPoint>& trainPoints,
|
|
|
|
vector<vector<DMatch> >& matches, float maxDistance, const Mat& mask=Mat(), bool compactResult=false ) const;
|
|
|
|
/*
|
|
|
|
* Group of methods to match keypoints from one image to image set.
|
|
|
|
* See description of similar methods for matching image pair above.
|
|
|
|
*/
|
|
|
|
void match( const Mat& queryImg, vector<KeyPoint>& queryPoints,
|
|
|
|
vector<DMatch>& matches, const vector<Mat>& masks=vector<Mat>() );
|
|
|
|
void knnMatch( const Mat& queryImg, vector<KeyPoint>& queryPoints,
|
|
|
|
vector<vector<DMatch> >& matches, int knn, const vector<Mat>& masks=vector<Mat>(), bool compactResult=false );
|
|
|
|
void radiusMatch( const Mat& queryImg, vector<KeyPoint>& queryPoints,
|
|
|
|
vector<vector<DMatch> >& matches, float maxDistance, const vector<Mat>& masks=vector<Mat>(), bool compactResult=false );
|
|
|
|
|
|
|
|
// Reads matcher object from a file node
|
|
|
|
virtual void read( const FileNode& ) {}
|
|
|
|
|
|
|
|
// Writes matcher object to a file storage
|
|
|
|
virtual void write( FileStorage& ) const {}
|
2010-06-11 17:38:39 +08:00
|
|
|
|
2010-10-29 16:44:42 +08:00
|
|
|
protected:
|
|
|
|
virtual Ptr<GenericDescriptorMatcher> createEmptyMatcherCopy() const = 0;
|
|
|
|
|
|
|
|
virtual void knnMatchImpl( const Mat& queryImg, vector<KeyPoint>& queryPoints,
|
|
|
|
vector<vector<DMatch> >& matches, int knn,
|
|
|
|
const vector<Mat>& masks, bool compactResult ) = 0;
|
|
|
|
virtual void radiusMatchImpl( const Mat& queryImg, vector<KeyPoint>& queryPoints,
|
|
|
|
vector<vector<DMatch> >& matches, float maxDistance,
|
|
|
|
const vector<Mat>& masks, bool compactResult ) = 0;
|
|
|
|
/*
|
|
|
|
* A storage for sets of keypoints together with corresponding images and class IDs
|
|
|
|
*/
|
|
|
|
class CV_EXPORTS KeyPointCollection
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
KeyPointCollection() : size(0) {}
|
|
|
|
void add( const vector<Mat>& _images, const vector<vector<KeyPoint> >& _points );
|
|
|
|
void clear();
|
2010-06-16 16:56:53 +08:00
|
|
|
|
2010-10-29 16:44:42 +08:00
|
|
|
// Returns the total number of keypoints in the collection
|
|
|
|
size_t pointCount() const { return size; }
|
|
|
|
size_t imageCount() const { return images.size(); }
|
2010-06-04 13:30:09 +08:00
|
|
|
|
2010-10-29 16:44:42 +08:00
|
|
|
const vector<vector<KeyPoint> >& getKeypoints() const { return points; }
|
|
|
|
const vector<KeyPoint>& getKeypoints( int imgIdx ) const { CV_Assert( imgIdx < (int)imageCount() ); return points[imgIdx]; }
|
|
|
|
const KeyPoint& getKeyPoint( int imgIdx, int localPointIdx ) const;
|
|
|
|
const KeyPoint& getKeyPoint( int globalPointIdx ) const;
|
|
|
|
void getLocalIdx( int globalPointIdx, int& imgIdx, int& localPointIdx ) const;
|
2010-06-10 15:59:18 +08:00
|
|
|
|
2010-10-29 16:44:42 +08:00
|
|
|
const vector<Mat>& getImages() const { return images; }
|
|
|
|
const Mat& getImage( int imgIdx ) const { CV_Assert( imgIdx < (int)imageCount() ); return images[imgIdx]; }
|
|
|
|
|
|
|
|
protected:
|
|
|
|
int size;
|
|
|
|
|
|
|
|
vector<Mat> images;
|
|
|
|
vector<vector<KeyPoint> > points;
|
|
|
|
|
|
|
|
// global indices of the first points in each image,
|
|
|
|
// startIndices.size() = points.size()
|
|
|
|
vector<int> startIndices;
|
|
|
|
};
|
|
|
|
|
|
|
|
KeyPointCollection trainPointCollection;
|
2010-05-12 01:44:00 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
2010-10-29 16:44:42 +08:00
|
|
|
* OneWayDescriptorMatcher
|
2010-05-12 01:44:00 +08:00
|
|
|
*/
|
2010-10-29 16:44:42 +08:00
|
|
|
class OneWayDescriptorMatcher;
|
|
|
|
typedef OneWayDescriptorMatcher OneWayDescriptorMatch;
|
|
|
|
|
|
|
|
class CV_EXPORTS OneWayDescriptorMatcher : public GenericDescriptorMatcher
|
2010-05-12 01:44:00 +08:00
|
|
|
{
|
|
|
|
public:
|
|
|
|
class Params
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
static const int POSE_COUNT = 500;
|
|
|
|
static const int PATCH_WIDTH = 24;
|
|
|
|
static const int PATCH_HEIGHT = 24;
|
2010-06-02 14:55:03 +08:00
|
|
|
static float GET_MIN_SCALE() { return 0.7f; }
|
|
|
|
static float GET_MAX_SCALE() { return 1.5f; }
|
|
|
|
static float GET_STEP_SCALE() { return 1.2f; }
|
2010-05-12 01:44:00 +08:00
|
|
|
|
|
|
|
Params( int _poseCount = POSE_COUNT,
|
|
|
|
Size _patchSize = Size(PATCH_WIDTH, PATCH_HEIGHT),
|
2010-10-29 16:44:42 +08:00
|
|
|
string _pcaFilename = string(),
|
2010-05-12 01:44:00 +08:00
|
|
|
string _trainPath = string(),
|
2010-06-02 13:19:09 +08:00
|
|
|
string _trainImagesList = string(),
|
2010-05-12 01:44:00 +08:00
|
|
|
float _minScale = GET_MIN_SCALE(), float _maxScale = GET_MAX_SCALE(),
|
|
|
|
float _stepScale = GET_STEP_SCALE() ) :
|
2010-06-02 13:19:09 +08:00
|
|
|
poseCount(_poseCount), patchSize(_patchSize), pcaFilename(_pcaFilename),
|
|
|
|
trainPath(_trainPath), trainImagesList(_trainImagesList),
|
2010-05-12 01:44:00 +08:00
|
|
|
minScale(_minScale), maxScale(_maxScale), stepScale(_stepScale) {}
|
|
|
|
|
|
|
|
int poseCount;
|
|
|
|
Size patchSize;
|
2010-06-02 13:19:09 +08:00
|
|
|
string pcaFilename;
|
2010-05-12 01:44:00 +08:00
|
|
|
string trainPath;
|
2010-06-02 13:19:09 +08:00
|
|
|
string trainImagesList;
|
2010-05-12 01:44:00 +08:00
|
|
|
|
|
|
|
float minScale, maxScale, stepScale;
|
|
|
|
};
|
|
|
|
|
|
|
|
// Equivalent to calling PointMatchOneWay() followed by Initialize(_params)
|
2010-10-29 16:44:42 +08:00
|
|
|
OneWayDescriptorMatcher( const Params& _params=Params() );
|
|
|
|
virtual ~OneWayDescriptorMatcher();
|
2010-05-12 01:44:00 +08:00
|
|
|
|
2010-10-29 16:44:42 +08:00
|
|
|
void initialize( const Params& _params, const Ptr<OneWayDescriptorBase>& _base=Ptr<OneWayDescriptorBase>() );
|
2010-05-12 01:44:00 +08:00
|
|
|
|
2010-06-08 14:56:35 +08:00
|
|
|
// Clears keypoints storing in collection and OneWayDescriptorBase
|
|
|
|
virtual void clear ();
|
|
|
|
|
2010-10-29 16:44:42 +08:00
|
|
|
virtual void train();
|
|
|
|
|
2010-06-08 14:56:35 +08:00
|
|
|
// Reads match object from a file node
|
2010-07-26 16:58:46 +08:00
|
|
|
virtual void read( const FileNode &fn );
|
2010-06-08 14:56:35 +08:00
|
|
|
|
|
|
|
// Writes match object to a file storage
|
2010-07-26 16:58:46 +08:00
|
|
|
virtual void write( FileStorage& fs ) const;
|
2010-06-02 14:55:03 +08:00
|
|
|
|
2010-05-12 01:44:00 +08:00
|
|
|
protected:
|
2010-10-29 16:44:42 +08:00
|
|
|
virtual Ptr<GenericDescriptorMatcher> createEmptyMatcherCopy() const { return new OneWayDescriptorMatcher( params ); }
|
|
|
|
|
|
|
|
// Matches a set of keypoints from a single image of the training set. A rectangle with a center in a keypoint
|
|
|
|
// and size (patch_width/2*scale, patch_height/2*scale) is cropped from the source image for each
|
|
|
|
// keypoint. scale is iterated from DescriptorOneWayParams::min_scale to DescriptorOneWayParams::max_scale.
|
|
|
|
// The minimum distance to each training patch with all its affine poses is found over all scales.
|
|
|
|
// The class ID of a match is returned for each keypoint. The distance is calculated over PCA components
|
|
|
|
// loaded with DescriptorOneWay::Initialize, kd tree is used for finding minimum distances.
|
|
|
|
virtual void knnMatchImpl( const Mat& queryImg, vector<KeyPoint>& queryPoints,
|
|
|
|
vector<vector<DMatch> >& matches, int knn,
|
|
|
|
const vector<Mat>& masks, bool compactResult );
|
|
|
|
virtual void radiusMatchImpl( const Mat& queryImg, vector<KeyPoint>& queryPoints,
|
|
|
|
vector<vector<DMatch> >& matches, float maxDistance,
|
|
|
|
const vector<Mat>& masks, bool compactResult );
|
|
|
|
|
2010-05-12 01:44:00 +08:00
|
|
|
Ptr<OneWayDescriptorBase> base;
|
|
|
|
Params params;
|
2010-10-29 16:44:42 +08:00
|
|
|
int prevTrainCount;
|
2010-05-12 01:44:00 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
2010-10-29 16:44:42 +08:00
|
|
|
* FernDescriptorMatcher
|
2010-05-12 01:44:00 +08:00
|
|
|
*/
|
2010-10-29 16:44:42 +08:00
|
|
|
class FernDescriptorMatcher;
|
|
|
|
typedef FernDescriptorMatcher FernDescriptorMatch;
|
|
|
|
|
|
|
|
class CV_EXPORTS FernDescriptorMatcher : public GenericDescriptorMatcher
|
2010-05-12 01:44:00 +08:00
|
|
|
{
|
|
|
|
public:
|
|
|
|
class Params
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Params( int _nclasses=0,
|
|
|
|
int _patchSize=FernClassifier::PATCH_SIZE,
|
|
|
|
int _signatureSize=FernClassifier::DEFAULT_SIGNATURE_SIZE,
|
|
|
|
int _nstructs=FernClassifier::DEFAULT_STRUCTS,
|
|
|
|
int _structSize=FernClassifier::DEFAULT_STRUCT_SIZE,
|
|
|
|
int _nviews=FernClassifier::DEFAULT_VIEWS,
|
|
|
|
int _compressionMethod=FernClassifier::COMPRESSION_NONE,
|
|
|
|
const PatchGenerator& patchGenerator=PatchGenerator() );
|
|
|
|
|
|
|
|
Params( const string& _filename );
|
|
|
|
|
|
|
|
int nclasses;
|
|
|
|
int patchSize;
|
|
|
|
int signatureSize;
|
|
|
|
int nstructs;
|
|
|
|
int structSize;
|
|
|
|
int nviews;
|
|
|
|
int compressionMethod;
|
|
|
|
PatchGenerator patchGenerator;
|
|
|
|
|
|
|
|
string filename;
|
|
|
|
};
|
|
|
|
|
2010-10-29 16:44:42 +08:00
|
|
|
FernDescriptorMatcher( const Params& _params=Params() );
|
|
|
|
virtual ~FernDescriptorMatcher();
|
2010-05-12 01:44:00 +08:00
|
|
|
|
2010-08-16 17:15:15 +08:00
|
|
|
virtual void clear();
|
2010-06-08 14:56:35 +08:00
|
|
|
|
2010-10-29 16:44:42 +08:00
|
|
|
virtual void train();
|
2010-06-08 14:56:35 +08:00
|
|
|
|
2010-10-29 16:44:42 +08:00
|
|
|
virtual void read( const FileNode &fn );
|
2010-06-08 14:56:35 +08:00
|
|
|
virtual void write( FileStorage& fs ) const;
|
|
|
|
|
2010-05-12 01:44:00 +08:00
|
|
|
protected:
|
2010-10-29 16:44:42 +08:00
|
|
|
virtual Ptr<GenericDescriptorMatcher> createEmptyMatcherCopy() const { return new FernDescriptorMatcher( params ); }
|
|
|
|
|
|
|
|
virtual void knnMatchImpl( const Mat& queryImg, vector<KeyPoint>& queryPoints,
|
|
|
|
vector<vector<DMatch> >& matches, int knn,
|
|
|
|
const vector<Mat>& masks, bool compactResult );
|
|
|
|
virtual void radiusMatchImpl( const Mat& queryImg, vector<KeyPoint>& queryPoints,
|
|
|
|
vector<vector<DMatch> >& matches, float maxDistance,
|
|
|
|
const vector<Mat>& masks, bool compactResult );
|
|
|
|
|
2010-05-12 01:44:00 +08:00
|
|
|
void trainFernClassifier();
|
|
|
|
void calcBestProbAndMatchIdx( const Mat& image, const Point2f& pt,
|
|
|
|
float& bestProb, int& bestMatchIdx, vector<float>& signature );
|
|
|
|
Ptr<FernClassifier> classifier;
|
|
|
|
Params params;
|
2010-10-29 16:44:42 +08:00
|
|
|
int prevTrainCount;
|
2010-05-12 01:44:00 +08:00
|
|
|
};
|
|
|
|
|
2010-10-29 16:44:42 +08:00
|
|
|
CV_EXPORTS Ptr<GenericDescriptorMatcher> createGenericDescriptorMatcher( const string& genericDescritptorMatcherType,
|
|
|
|
const string ¶msFilename = string () );
|
2010-07-26 16:58:46 +08:00
|
|
|
|
2010-05-12 01:44:00 +08:00
|
|
|
/****************************************************************************************\
|
2010-10-29 16:44:42 +08:00
|
|
|
* VectorDescriptorMatcher *
|
2010-05-12 01:44:00 +08:00
|
|
|
\****************************************************************************************/
|
|
|
|
|
|
|
|
/*
|
2010-05-21 00:16:28 +08:00
|
|
|
* A class used for matching descriptors that can be described as vectors in a finite-dimensional space
|
2010-05-12 01:44:00 +08:00
|
|
|
*/
|
2010-10-29 16:44:42 +08:00
|
|
|
class VectorDescriptorMatcher;
|
|
|
|
typedef VectorDescriptorMatcher VectorDescriptorMatch;
|
|
|
|
|
|
|
|
class CV_EXPORTS VectorDescriptorMatcher : public GenericDescriptorMatcher
|
2010-05-12 01:44:00 +08:00
|
|
|
{
|
|
|
|
public:
|
2010-10-29 16:44:42 +08:00
|
|
|
VectorDescriptorMatcher( const Ptr<DescriptorExtractor>& _extractor, const Ptr<DescriptorMatcher>& _matcher )
|
|
|
|
: extractor( _extractor ), matcher( _matcher ) { CV_Assert( !extractor.empty() && !matcher.empty() ); }
|
|
|
|
virtual ~VectorDescriptorMatcher() {}
|
2010-05-12 01:44:00 +08:00
|
|
|
|
2010-10-29 16:44:42 +08:00
|
|
|
virtual void add( const vector<Mat>& imgCollection,
|
|
|
|
vector<vector<KeyPoint> >& pointCollection );
|
2010-05-12 01:44:00 +08:00
|
|
|
|
2010-10-29 16:44:42 +08:00
|
|
|
virtual void clear();
|
2010-06-16 16:56:53 +08:00
|
|
|
|
2010-10-29 16:44:42 +08:00
|
|
|
virtual void train();
|
2010-05-26 23:00:03 +08:00
|
|
|
|
2010-07-26 16:58:46 +08:00
|
|
|
virtual void read( const FileNode& fn );
|
|
|
|
virtual void write( FileStorage& fs ) const;
|
2010-06-04 13:30:09 +08:00
|
|
|
|
2010-05-12 01:44:00 +08:00
|
|
|
protected:
|
2010-10-29 16:44:42 +08:00
|
|
|
virtual Ptr<GenericDescriptorMatcher> createEmptyMatcherCopy() const { return new VectorDescriptorMatcher(extractor, matcher); }
|
|
|
|
|
|
|
|
virtual void knnMatchImpl( const Mat& queryImg, vector<KeyPoint>& queryPoints,
|
|
|
|
vector<vector<DMatch> >& matches, int knn,
|
|
|
|
const vector<Mat>& masks, bool compactResult );
|
|
|
|
virtual void radiusMatchImpl( const Mat& queryImg, vector<KeyPoint>& queryPoints,
|
|
|
|
vector<vector<DMatch> >& matches, float maxDistance,
|
|
|
|
const vector<Mat>& masks, bool compactResult );
|
|
|
|
|
2010-07-12 19:56:11 +08:00
|
|
|
Ptr<DescriptorExtractor> extractor;
|
|
|
|
Ptr<DescriptorMatcher> matcher;
|
2010-05-12 01:44:00 +08:00
|
|
|
};
|
|
|
|
|
2010-08-05 20:19:26 +08:00
|
|
|
/****************************************************************************************\
|
|
|
|
* Drawing functions *
|
|
|
|
\****************************************************************************************/
|
2010-06-12 01:15:43 +08:00
|
|
|
struct CV_EXPORTS DrawMatchesFlags
|
|
|
|
{
|
|
|
|
enum{ DEFAULT = 0, // Output image matrix will be created (Mat::create),
|
2010-06-12 18:11:57 +08:00
|
|
|
// i.e. existing memory of output image may be reused.
|
2010-06-12 01:15:43 +08:00
|
|
|
// Two source image, matches and single keypoints will be drawn.
|
2010-08-04 00:28:52 +08:00
|
|
|
// For each keypoint only the center point will be drawn (without
|
|
|
|
// the circle around keypoint with keypoint size and orientation).
|
2010-06-12 01:15:43 +08:00
|
|
|
DRAW_OVER_OUTIMG = 1, // Output image matrix will not be created (Mat::create).
|
|
|
|
// Matches will be drawn on existing content of output image.
|
2010-08-04 00:28:52 +08:00
|
|
|
NOT_DRAW_SINGLE_POINTS = 2, // Single keypoints will not be drawn.
|
|
|
|
DRAW_RICH_KEYPOINTS = 4 // For each keypoint the circle around keypoint with keypoint size and
|
2010-08-05 20:19:26 +08:00
|
|
|
// orientation will be drawn.
|
2010-06-12 01:15:43 +08:00
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2010-08-04 00:28:52 +08:00
|
|
|
// Draw keypoints.
|
|
|
|
CV_EXPORTS void drawKeypoints( const Mat& image, const vector<KeyPoint>& keypoints, Mat& outImg,
|
|
|
|
const Scalar& color=Scalar::all(-1), int flags=DrawMatchesFlags::DEFAULT );
|
|
|
|
|
2010-06-12 01:15:43 +08:00
|
|
|
// Draws matches of keypints from two images on output image.
|
2010-08-04 00:28:52 +08:00
|
|
|
CV_EXPORTS void drawMatches( const Mat& img1, const vector<KeyPoint>& keypoints1,
|
|
|
|
const Mat& img2, const vector<KeyPoint>& keypoints2,
|
|
|
|
const vector<DMatch>& matches1to2, Mat& outImg,
|
|
|
|
const Scalar& matchColor=Scalar::all(-1), const Scalar& singlePointColor=Scalar::all(-1),
|
|
|
|
const vector<char>& matchesMask=vector<char>(), int flags=DrawMatchesFlags::DEFAULT );
|
|
|
|
|
|
|
|
CV_EXPORTS void drawMatches( const Mat& img1, const vector<KeyPoint>& keypoints1,
|
|
|
|
const Mat& img2, const vector<KeyPoint>& keypoints2,
|
|
|
|
const vector<vector<DMatch> >& matches1to2, Mat& outImg,
|
|
|
|
const Scalar& matchColor=Scalar::all(-1), const Scalar& singlePointColor=Scalar::all(-1),
|
|
|
|
const vector<vector<char> >& matchesMask=vector<vector<char> >(), int flags=DrawMatchesFlags::DEFAULT );
|
2010-06-12 01:15:43 +08:00
|
|
|
|
2010-08-05 20:19:26 +08:00
|
|
|
/****************************************************************************************\
|
2010-09-24 00:17:48 +08:00
|
|
|
* Functions to evaluate the feature detectors and [generic] descriptor extractors *
|
2010-08-05 20:19:26 +08:00
|
|
|
\****************************************************************************************/
|
|
|
|
|
|
|
|
CV_EXPORTS void evaluateFeatureDetector( const Mat& img1, const Mat& img2, const Mat& H1to2,
|
|
|
|
vector<KeyPoint>* keypoints1, vector<KeyPoint>* keypoints2,
|
|
|
|
float& repeatability, int& correspCount,
|
|
|
|
const Ptr<FeatureDetector>& fdetector=Ptr<FeatureDetector>() );
|
|
|
|
|
|
|
|
CV_EXPORTS void computeRecallPrecisionCurve( const vector<vector<DMatch> >& matches1to2,
|
|
|
|
const vector<vector<uchar> >& correctMatches1to2Mask,
|
|
|
|
vector<Point2f>& recallPrecisionCurve );
|
|
|
|
CV_EXPORTS float getRecall( const vector<Point2f>& recallPrecisionCurve, float l_precision );
|
|
|
|
|
2010-09-24 00:17:48 +08:00
|
|
|
CV_EXPORTS void evaluateGenericDescriptorMatcher( const Mat& img1, const Mat& img2, const Mat& H1to2,
|
|
|
|
vector<KeyPoint>& keypoints1, vector<KeyPoint>& keypoints2,
|
|
|
|
vector<vector<DMatch> >* matches1to2, vector<vector<uchar> >* correctMatches1to2Mask,
|
|
|
|
vector<Point2f>& recallPrecisionCurve,
|
2010-10-29 16:44:42 +08:00
|
|
|
const Ptr<GenericDescriptorMatcher>& dmatch=Ptr<GenericDescriptorMatcher>() );
|
2010-08-05 20:19:26 +08:00
|
|
|
|
|
|
|
|
2010-09-24 00:17:48 +08:00
|
|
|
/****************************************************************************************\
|
|
|
|
* Bag of visual words *
|
|
|
|
\****************************************************************************************/
|
|
|
|
/*
|
|
|
|
* Abstract base class for training of a 'bag of visual words' vocabulary from a set of descriptors
|
|
|
|
*/
|
2010-09-30 22:21:22 +08:00
|
|
|
class CV_EXPORTS BOWTrainer
|
2010-09-24 00:17:48 +08:00
|
|
|
{
|
|
|
|
public:
|
2010-10-08 21:17:34 +08:00
|
|
|
BOWTrainer(){}
|
|
|
|
virtual ~BOWTrainer(){}
|
|
|
|
|
2010-09-30 22:21:22 +08:00
|
|
|
void add( const Mat& descriptors );
|
|
|
|
const vector<Mat>& getDescriptors() const { return descriptors; }
|
|
|
|
int descripotorsCount() const { return descriptors.empty() ? 0 : size; }
|
|
|
|
|
|
|
|
virtual void clear();
|
|
|
|
|
2010-09-24 00:17:48 +08:00
|
|
|
/*
|
|
|
|
* Train visual words vocabulary, that is cluster training descriptors and
|
|
|
|
* compute cluster centers.
|
2010-09-30 22:21:22 +08:00
|
|
|
* Returns cluster centers.
|
2010-09-24 00:17:48 +08:00
|
|
|
*
|
|
|
|
* descriptors Training descriptors computed on images keypoints.
|
|
|
|
*/
|
2010-09-30 22:21:22 +08:00
|
|
|
virtual Mat cluster() const = 0;
|
|
|
|
virtual Mat cluster( const Mat& descriptors ) const = 0;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
vector<Mat> descriptors;
|
|
|
|
int size;
|
2010-09-24 00:17:48 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
* This is BOWTrainer using cv::kmeans to get vocabulary.
|
|
|
|
*/
|
2010-09-30 22:21:22 +08:00
|
|
|
class CV_EXPORTS BOWKMeansTrainer : public BOWTrainer
|
2010-09-24 00:17:48 +08:00
|
|
|
{
|
|
|
|
public:
|
|
|
|
BOWKMeansTrainer( int clusterCount, const TermCriteria& termcrit=TermCriteria(),
|
|
|
|
int attempts=3, int flags=KMEANS_PP_CENTERS );
|
2010-10-08 21:17:34 +08:00
|
|
|
virtual ~BOWKMeansTrainer(){}
|
2010-09-30 22:21:22 +08:00
|
|
|
|
|
|
|
// Returns trained vocabulary (i.e. cluster centers).
|
|
|
|
virtual Mat cluster() const;
|
|
|
|
virtual Mat cluster( const Mat& descriptors ) const;
|
2010-09-24 00:17:48 +08:00
|
|
|
|
|
|
|
protected:
|
2010-09-30 22:21:22 +08:00
|
|
|
|
2010-09-24 00:17:48 +08:00
|
|
|
int clusterCount;
|
|
|
|
TermCriteria termcrit;
|
|
|
|
int attempts;
|
|
|
|
int flags;
|
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Class to compute image descriptor using bad of visual words.
|
|
|
|
*/
|
2010-09-30 22:21:22 +08:00
|
|
|
class CV_EXPORTS BOWImgDescriptorExtractor
|
2010-09-24 00:17:48 +08:00
|
|
|
{
|
|
|
|
public:
|
|
|
|
BOWImgDescriptorExtractor( const Ptr<DescriptorExtractor>& dextractor,
|
|
|
|
const Ptr<DescriptorMatcher>& dmatcher );
|
2010-10-08 21:17:34 +08:00
|
|
|
virtual ~BOWImgDescriptorExtractor(){}
|
|
|
|
|
2010-09-30 22:21:22 +08:00
|
|
|
void setVocabulary( const Mat& vocabulary );
|
|
|
|
const Mat& getVocabulary() const { return vocabulary; }
|
2010-09-24 00:17:48 +08:00
|
|
|
void compute( const Mat& image, vector<KeyPoint>& keypoints, Mat& imgDescriptor,
|
2010-10-29 16:44:42 +08:00
|
|
|
vector<vector<int> >* pointIdxsOfClusters=0 ); //not constant because DescriptorMatcher::match is not constant
|
2010-09-24 00:17:48 +08:00
|
|
|
int descriptorSize() const { return vocabulary.empty() ? 0 : vocabulary.rows; }
|
|
|
|
int descriptorType() const { return CV_32FC1; }
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Mat vocabulary;
|
|
|
|
Ptr<DescriptorExtractor> dextractor;
|
|
|
|
Ptr<DescriptorMatcher> dmatcher;
|
|
|
|
};
|
|
|
|
|
2010-08-05 20:19:26 +08:00
|
|
|
} /* namespace cv */
|
2010-05-12 01:44:00 +08:00
|
|
|
|
|
|
|
#endif /* __cplusplus */
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* End of file. */
|