Merge pull request #6384 from StevenPuttemans:update_annotation_tool_2.4

This commit is contained in:
Maksim Shabunin
2016-04-13 10:02:56 +00:00

View File

@@ -46,6 +46,9 @@ USAGE:
./opencv_annotation -images <folder location> -annotations <ouput file> ./opencv_annotation -images <folder location> -annotations <ouput file>
Created by: Puttemans Steven - February 2015 Created by: Puttemans Steven - February 2015
Adapted by: Puttemans Steven - April 2016 - Vectorize the process to enable better processing
+ early leave and store by pressing an ESC key
+ enable delete `d` button, to remove last annotation
*****************************************************************************************************/ *****************************************************************************************************/
#include <opencv2/core/core.hpp> #include <opencv2/core/core.hpp>
@@ -66,16 +69,15 @@ using namespace cv;
// Function prototypes // Function prototypes
void on_mouse(int, int, int, int, void*); void on_mouse(int, int, int, int, void*);
string int2string(int); vector<Rect> get_annotations(Mat);
void get_annotations(Mat, stringstream*);
// Public parameters // Public parameters
Mat image; Mat image;
int roi_x0 = 0, roi_y0 = 0, roi_x1 = 0, roi_y1 = 0, num_of_rec = 0; int roi_x0 = 0, roi_y0 = 0, roi_x1 = 0, roi_y1 = 0, num_of_rec = 0;
bool start_draw = false; bool start_draw = false, stop = false;
// Window name for visualisation purposes // Window name for visualisation purposes
const string window_name="OpenCV Based Annotation Tool"; const string window_name = "OpenCV Based Annotation Tool";
// FUNCTION : Mouse response for selecting objects in images // FUNCTION : Mouse response for selecting objects in images
// If left button is clicked, start drawing a rectangle as long as mouse moves // If left button is clicked, start drawing a rectangle as long as mouse moves
@@ -83,7 +85,7 @@ const string window_name="OpenCV Based Annotation Tool";
void on_mouse(int event, int x, int y, int , void * ) void on_mouse(int event, int x, int y, int , void * )
{ {
// Action when left button is clicked // Action when left button is clicked
if(event == CV_EVENT_LBUTTONDOWN) if(event == EVENT_LBUTTONDOWN)
{ {
if(!start_draw) if(!start_draw)
{ {
@@ -96,8 +98,9 @@ void on_mouse(int event, int x, int y, int , void * )
start_draw = false; start_draw = false;
} }
} }
// Action when mouse is moving
if((event == CV_EVENT_MOUSEMOVE) && start_draw) // Action when mouse is moving and drawing is enabled
if((event == EVENT_MOUSEMOVE) && start_draw)
{ {
// Redraw bounding box for annotation // Redraw bounding box for annotation
Mat current_view; Mat current_view;
@@ -107,43 +110,35 @@ void on_mouse(int event, int x, int y, int , void * )
} }
} }
// FUNCTION : snippet to convert an integer value to a string using a clean function // FUNCTION : returns a vector of Rect objects given an image containing positive object instances
// instead of creating a stringstream each time inside the main code vector<Rect> get_annotations(Mat input_image)
string int2string(int num)
{ {
stringstream temp_stream; vector<Rect> current_annotations;
temp_stream << num;
return temp_stream.str();
}
// FUNCTION : given an image containing positive object instances, add all the object // Make it possible to exit the annotation process
// annotations to a known stringstream stop = false;
void get_annotations(Mat input_image, stringstream* output_stream)
{
// Make it possible to exit the annotation
bool stop = false;
// Reset the num_of_rec element at each iteration
// Make sure the global image is set to the current image
num_of_rec = 0;
image = input_image;
// Init window interface and couple mouse actions // Init window interface and couple mouse actions
namedWindow(window_name, WINDOW_AUTOSIZE); namedWindow(window_name, WINDOW_AUTOSIZE);
setMouseCallback(window_name, on_mouse); setMouseCallback(window_name, on_mouse);
image = input_image;
imshow(window_name, image); imshow(window_name, image);
stringstream temp_stream;
int key_pressed = 0; int key_pressed = 0;
do do
{ {
// Get a temporary image clone
Mat temp_image = input_image.clone();
Rect currentRect(0, 0, 0, 0);
// Keys for processing // Keys for processing
// You need to select one for confirming a selection and one to continue to the next image // You need to select one for confirming a selection and one to continue to the next image
// Based on the universal ASCII code of the keystroke: http://www.asciitable.com/ // Based on the universal ASCII code of the keystroke: http://www.asciitable.com/
// c = 99 add rectangle to current image // <c> = 99 add rectangle to current image
// n = 110 save added rectangles and show next image // <n> = 110 save added rectangles and show next image
// <ESC> = 27 exit program // <d> = 100 delete the last annotation made
// <ESC> = 27 exit program
key_pressed = 0xFF & waitKey(0); key_pressed = 0xFF & waitKey(0);
switch( key_pressed ) switch( key_pressed )
{ {
@@ -152,31 +147,51 @@ void get_annotations(Mat input_image, stringstream* output_stream)
stop = true; stop = true;
break; break;
case 99: case 99:
// Add a rectangle to the list
num_of_rec++;
// Draw initiated from top left corner // Draw initiated from top left corner
if(roi_x0<roi_x1 && roi_y0<roi_y1) if(roi_x0<roi_x1 && roi_y0<roi_y1)
{ {
temp_stream << " " << int2string(roi_x0) << " " << int2string(roi_y0) << " " << int2string(roi_x1-roi_x0) << " " << int2string(roi_y1-roi_y0); currentRect.x = roi_x0;
currentRect.y = roi_y0;
currentRect.width = roi_x1-roi_x0;
currentRect.height = roi_y1-roi_y0;
} }
// Draw initiated from bottom right corner // Draw initiated from bottom right corner
if(roi_x0>roi_x1 && roi_y0>roi_y1) if(roi_x0>roi_x1 && roi_y0>roi_y1)
{ {
temp_stream << " " << int2string(roi_x1) << " " << int2string(roi_y1) << " " << int2string(roi_x0-roi_x1) << " " << int2string(roi_y0-roi_y1); currentRect.x = roi_x1;
currentRect.y = roi_y1;
currentRect.width = roi_x0-roi_x1;
currentRect.height = roi_y0-roi_y1;
} }
// Draw initiated from top right corner // Draw initiated from top right corner
if(roi_x0>roi_x1 && roi_y0<roi_y1) if(roi_x0>roi_x1 && roi_y0<roi_y1)
{ {
temp_stream << " " << int2string(roi_x1) << " " << int2string(roi_y0) << " " << int2string(roi_x0-roi_x1) << " " << int2string(roi_y1-roi_y0); currentRect.x = roi_x1;
currentRect.y = roi_y0;
currentRect.width = roi_x0-roi_x1;
currentRect.height = roi_y1-roi_y0;
} }
// Draw initiated from bottom left corner // Draw initiated from bottom left corner
if(roi_x0<roi_x1 && roi_y0>roi_y1) if(roi_x0<roi_x1 && roi_y0>roi_y1)
{ {
temp_stream << " " << int2string(roi_x0) << " " << int2string(roi_y1) << " " << int2string(roi_x1-roi_x0) << " " << int2string(roi_y0-roi_y1); currentRect.x = roi_x0;
currentRect.y = roi_y1;
currentRect.width = roi_x1-roi_x0;
currentRect.height = roi_y0-roi_y1;
} }
// Draw the rectangle on the canvas
rectangle(input_image, Point(roi_x0,roi_y0), Point(roi_x1,roi_y1), Scalar(0,255,0), 1); // Add the rectangle to the vector of annotations
current_annotations.push_back(currentRect);
break;
case 100:
// Remove the last annotation
if(current_annotations.size() > 0){
current_annotations.pop_back();
}
break;
default:
// Default case --> do nothing at all
// Other keystrokes can simply be ignored
break; break;
} }
@@ -185,35 +200,40 @@ void get_annotations(Mat input_image, stringstream* output_stream)
{ {
break; break;
} }
// Draw all the current rectangles onto the top image and make sure that the global image is linked
for(int i=0; i < (int)current_annotations.size(); i++){
rectangle(temp_image, current_annotations[i], Scalar(0,255,0), 1);
}
image = temp_image;
// Force an explicit redraw of the canvas --> necessary to visualize delete correctly
imshow(window_name, image);
} }
// Continue as long as the next image key has not been pressed // Continue as long as the next image key has not been pressed
while(key_pressed != 110); while(key_pressed != 110);
// If there are annotations AND the next image key is pressed
// Write the image annotations to the file
if(num_of_rec>0 && key_pressed==110)
{
*output_stream << " " << num_of_rec << temp_stream.str() << endl;
}
// Close down the window // Close down the window
destroyWindow(window_name); destroyWindow(window_name);
// Return the data
return current_annotations;
} }
int main( int argc, const char** argv ) int main( int argc, const char** argv )
{ {
// If no arguments are given, then supply some information on how this tool works // If no arguments are given, then supply some information on how this tool works
if( argc == 1 ){ if( argc == 1 ){
cout << "Usage: " << argv[0] << endl; cout << "Usage: " << argv[0] << endl;
cout << " -images <folder_location> [example - /data/testimages/]" << endl; cout << " -images <folder_location> [example - /data/testimages/]" << endl;
cout << " -annotations <ouput_file> [example - /data/annotations.txt]" << endl; cout << " -annotations <ouput_file> [example - /data/annotations.txt]" << endl;
cout << "TIP: Use absolute paths to avoid any problems with the software!" << endl;
return -1; return -1;
} }
// Read in the input arguments // Read in the input arguments
string image_folder; string image_folder;
string annotations; string annotations_file;
for(int i = 1; i < argc; ++i ) for(int i = 1; i < argc; ++i )
{ {
if( !strcmp( argv[i], "-images" ) ) if( !strcmp( argv[i], "-images" ) )
@@ -222,7 +242,7 @@ int main( int argc, const char** argv )
} }
else if( !strcmp( argv[i], "-annotations" ) ) else if( !strcmp( argv[i], "-annotations" ) )
{ {
annotations = argv[++i]; annotations_file = argv[++i];
} }
} }
@@ -247,14 +267,9 @@ int main( int argc, const char** argv )
} }
#endif #endif
// Create the outputfilestream // Start by processing the data
ofstream output(annotations.c_str());
if ( !output.is_open() ){
cerr << "The path for the output file contains an error and could not be opened. Please check again!" << endl;
return 0;
}
// Return the image filenames inside the image folder // Return the image filenames inside the image folder
vector< vector<Rect> > annotations;
vector<String> filenames; vector<String> filenames;
String folder(image_folder); String folder(image_folder);
glob(folder, filenames); glob(folder, filenames);
@@ -272,15 +287,33 @@ int main( int argc, const char** argv )
continue; continue;
} }
// Perform annotations & generate corresponding output // Perform annotations & store the result inside the vectorized structure
stringstream output_stream; vector<Rect> current_annotations = get_annotations(current_image);
get_annotations(current_image, &output_stream); annotations.push_back(current_annotations);
// Store the annotations, write to the output file // Check if the ESC key was hit, then exit earlier then expected
if (output_stream.str() != ""){ if(stop){
output << filenames[i] << output_stream.str(); break;
} }
} }
// When all data is processed, store the data gathered inside the proper file
// This now even gets called when the ESC button was hit to store preliminary results
ofstream output(annotations_file.c_str());
if ( !output.is_open() ){
cerr << "The path for the output file contains an error and could not be opened. Please check again!" << endl;
return 0;
}
// Store the annotations, write to the output file
for(int i = 0; i < (int)annotations.size(); i++){
output << filenames[i] << " " << annotations[i].size();
for(int j=0; j < (int)annotations[i].size(); j++){
Rect temp = annotations[i][j];
output << " " << temp.x << " " << temp.y << " " << temp.width << " " << temp.height;
}
output << endl;
}
return 0; return 0;
} }