Tracking of planar surface using Fern classifier.
#include <visp3/core/vpConfig.h>
#include <visp3/core/vpDebug.h>
#if ((defined(VISP_HAVE_X11) || defined(VISP_HAVE_GTK) || defined(VISP_HAVE_GDI)) && \
(VISP_HAVE_OPENCV_VERSION >= 0x020000) && (VISP_HAVE_OPENCV_VERSION < 0x030000))
#include <iomanip>
#include <iostream>
#include <stdlib.h>
#include <visp3/core/vpConfig.h>
#include <visp3/core/vpImage.h>
#include <visp3/core/vpIoTools.h>
#include <visp3/core/vpTime.h>
#include <visp3/gui/vpDisplayGDI.h>
#include <visp3/gui/vpDisplayGTK.h>
#include <visp3/gui/vpDisplayX.h>
#include <visp3/io/vpImageIo.h>
#include <visp3/io/vpParseArgv.h>
#include <visp3/sensor/vp1394TwoGrabber.h>
#include <visp3/sensor/vpV4l2Grabber.h>
#include <visp3/vision/vpHomography.h>
#include <visp3/vision/vpPlanarObjectDetector.h>
#define GETOPTARGS "hlcdb:i:p"
void usage(const char *name, const char *badparam);
bool getOptions(int argc, const char **argv, bool &isLearning, std::string &dataFile, bool &click_allowed,
bool &display, bool &displayPoints, std::string &ipath);
void usage(const char *name, const char *badparam)
{
#if VISP_HAVE_DATASET_VERSION >= 0x030600
std::string ext("png");
#else
std::string ext("pgm");
#endif
fprintf(stdout, "\n\
Test of detection of planar surface using a Fern classifier. The object needs \
first to be learned (-l option). This learning process will create a file used\
to detect the object.\n\
\n\
SYNOPSIS\n\
%s [-l] [-h] [-b] [-c] [-d] [-p] [-i] [-s]\n",
name);
fprintf(stdout, "\n\
OPTIONS: \n\
-l\n\
learn an object.\n\
\n\
-i <input image path> \n\
Set image input path.\n\
From this path read \"line/image.%%04d.%s\"\n\
images. \n\
Setting the VISP_INPUT_IMAGE_PATH environment\n\
variable produces the same behaviour than using\n\
this option.\n\
\n\
-b\n\
database filename to use (default is ./dataPlanar).\n\
\n\
-c\n\
Disable the mouse click. Useful to automaze the \n\
execution of this program without humain intervention.\n\
\n\
-d \n\
Turn off the display.\n\
\n\
-s \n\
Turn off the use of the sequence and use a webcam.\n\
\n\
-p \n\
display points of interest.\n\
\n\
-h\n\
Print this help.\n", ext.c_str());
if (badparam)
fprintf(stdout, "\nERROR: Bad parameter [%s]\n", badparam);
}
bool getOptions(int argc, const char **argv, bool &isLearning, std::string &dataFile, bool &click_allowed,
bool &display, bool &displayPoints, std::string &ipath)
{
const char *optarg_;
int c;
switch (c) {
case 'c':
click_allowed = false;
break;
case 'd':
display = false;
break;
case 'l':
isLearning = true;
break;
case 'h':
usage(argv[0], NULL);
return false;
break;
case 'b':
dataFile = optarg_;
break;
case 'p':
displayPoints = true;
break;
case 'i':
ipath = optarg_;
break;
default:
usage(argv[0], optarg_);
return false;
break;
}
}
if ((c == 1) || (c == -1)) {
usage(argv[0], NULL);
std::cerr << "ERROR: " << std::endl;
std::cerr << " Bad argument " << optarg_ << std::endl << std::endl;
return false;
}
return true;
}
int main(int argc, const char **argv)
{
try {
bool isLearning = false;
std::string dataFile("./dataPlanar");
bool opt_click_allowed = true;
bool opt_display = true;
std::string objectName("object");
bool displayPoints = false;
std::string opt_ipath;
std::string ipath;
std::string env_ipath;
std::string dirname;
std::string filename;
#if VISP_HAVE_DATASET_VERSION >= 0x030600
std::string ext("png");
#else
std::string ext("pgm");
#endif
if (!env_ipath.empty()) {
ipath = env_ipath;
}
if (getOptions(argc, argv, isLearning, dataFile, opt_click_allowed, opt_display, displayPoints, opt_ipath) ==
false) {
return EXIT_FAILURE;
}
if (!opt_ipath.empty()) {
ipath = opt_ipath;
}
if (!opt_ipath.empty() && !env_ipath.empty()) {
if (ipath != env_ipath) {
std::cout << std::endl << "WARNING: " << std::endl;
std::cout << " Since -i <visp image path=" << ipath << "> "
<< " is different from VISP_IMAGE_PATH=" << env_ipath << std::endl
<< " we skip the environment variable." << std::endl;
}
}
if (opt_ipath.empty() && env_ipath.empty()) {
usage(argv[0], NULL);
std::cerr << std::endl << "ERROR:" << std::endl;
std::cerr << " Use -i <visp image path> option or set VISP_INPUT_IMAGE_PATH " << std::endl
<< " environment variable to specify the location of the " << std::endl
<< " image path where test images are located." << std::endl
<< std::endl;
return EXIT_FAILURE;
}
unsigned iter = 0;
std::ostringstream s;
s.setf(std::ios::right, std::ios::adjustfield);
s << "image." << std::setw(4) << std::setfill('0') << iter << "." << ext;
try {
std::cout << "Load: " << filename << std::endl;
I = Iref;
} catch (...) {
std::cerr << std::endl << "ERROR:" << std::endl;
std::cerr << " Cannot read " << filename << std::endl;
std::cerr << " Check your -i " << ipath << " option " << std::endl
<< " or VISP_INPUT_IMAGE_PATH environment variable." << std::endl;
return EXIT_FAILURE;
}
#if defined VISP_HAVE_X11
#elif defined VISP_HAVE_GTK
#elif defined VISP_HAVE_GDI
#endif
#if defined VISP_HAVE_X11
#elif defined VISP_HAVE_GTK
#elif defined VISP_HAVE_GDI
#endif
vpPlanarObjectDetector planar;
if (isLearning) {
if (opt_display) {
displayRef.
init(Iref, 100, 100,
"Reference image");
}
if (opt_display && opt_click_allowed) {
std::cout << "Click on the top left and the bottom right corners to "
"define the reference plane"
<< std::endl;
for (int i = 0; i < 2; i++) {
std::cout << corners[i] << std::endl;
}
} else {
}
if (opt_display) {
}
if (opt_click_allowed) {
std::cout << "Click on the image to continue" << std::endl;
}
vpRect roi(corners[0], corners[1]);
std::cout << "> train the classifier on the selected plane (may take "
"up to several minutes)."
<< std::endl;
if (opt_display) {
}
planar.buildReference(Iref, roi);
planar.recordDetector(objectName, dataFile);
} else {
vpERROR_TRACE(
"cannot load the database with the specified name. Has "
"the object been learned with the -l option? ");
return EXIT_FAILURE;
}
try {
planar.load(dataFile, objectName);
} catch (...) {
vpERROR_TRACE(
"cannot load the database with the specified name. Has "
"the object been learned with the -l option? ");
return EXIT_FAILURE;
}
}
if (opt_display) {
display.init(I, 110 + (
int)Iref.
getWidth(), 100,
"Current image");
}
if (opt_display && opt_click_allowed) {
std::cout << "Click on the reference image to continue" << std::endl;
}
for (;;) {
iter++;
if (iter >= 80) {
break;
}
s.str("");
s << "image." << std::setw(4) << std::setfill('0') << iter << "." << ext;
if (opt_display) {
}
bool isDetected = planar.matchPoint(I);
if (isDetected) {
planar.getHomography(H);
std::cout << " > computed homography:" << std::endl << H << std::endl;
if (opt_display) {
if (isLearning) {
planar.display(Iref, I, displayPoints);
} else {
planar.display(I, displayPoints);
}
}
} else {
std::cout << " > reference is not detected in the image" << std::endl;
}
if (opt_display) {
break;
}
}
}
return EXIT_SUCCESS;
std::cout << "Catch an exception: " << e << std::endl;
return EXIT_FAILURE;
}
}
#else
int main()
{
#if (!(defined(VISP_HAVE_X11) || defined(VISP_HAVE_GTK) || defined(VISP_HAVE_GDI)))
std::cout << "You do not have X11, or GTK, or GDI (Graphical Device Interface) functionalities to display images..."
<< std::endl;
std::cout << "Tip if you are on a unix-like system:" << std::endl;
std::cout << "- Install X11, configure again ViSP using cmake and build again this example" << std::endl;
std::cout << "Tip if you are on a windows-like system:" << std::endl;
std::cout << "- Install GDI, configure again ViSP using cmake and build again this example" << std::endl;
#else
std::cout << "You do not have OpenCV functionalities" << std::endl;
std::cout << "Tip:" << std::endl;
std::cout << "- Install OpenCV, configure again ViSP using cmake and build again this example" << std::endl;
#endif
return EXIT_SUCCESS;
}
#endif
static const vpColor green
Display for windows using GDI (available on any windows 32 platform).
The vpDisplayGTK allows to display image using the GTK 3rd party library. Thus to enable this class G...
Use the X11 console to display images on unix-like OS. Thus to enable this class X11 should be instal...
void init(vpImage< unsigned char > &I, int win_x=-1, int win_y=-1, const std::string &win_title="")
static bool getClick(const vpImage< unsigned char > &I, bool blocking=true)
static void display(const vpImage< unsigned char > &I)
static void flush(const vpImage< unsigned char > &I)
static void displayRectangle(const vpImage< unsigned char > &I, const vpImagePoint &topLeft, unsigned int width, unsigned int height, const vpColor &color, bool fill=false, unsigned int thickness=1)
static void displayText(const vpImage< unsigned char > &I, const vpImagePoint &ip, const std::string &s, const vpColor &color)
error that can be emited by ViSP classes.
Implementation of an homography and operations on homographies.
static void read(vpImage< unsigned char > &I, const std::string &filename, int backend=IO_DEFAULT_BACKEND)
Class that defines a 2D point in an image. This class is useful for image processing and stores only ...
void set_ij(double ii, double jj)
unsigned int getWidth() const
unsigned int getHeight() const
static bool parse(int *argcPtr, const char **argv, vpArgvInfo *argTable, int flags)
Defines a rectangle in the plane.
VISP_EXPORT double measureTimeMs()