Tutorial on spToolkit

Transcription

Tutorial on spToolkit
Tutorial on spToolkit
A.C. Klaren
August, 2004
2
Chapter 1
Introduction
1.1
The SmartSpectra project
The SmartSpectra project started in 2002 with the objective to design and
develop a multispectral sensor with a set of features that permits its use in
commercial applications. It will be provided with a set of high-level software
tools to simplify its application as a developer’s toolkit for any possible field
of application where multispectral imaging is needed. This set of high-level
software tools has been written in C++ and can be found in a library called
spToolkit.
1.2
The spToolkit library
The spToolkit library contains a number of high-level software tools to facilitate
the process of extracting information from the images obtained by the multispectral camera. All its functions can be applied to images with any number of
bands and to images of various data types. This tutorial will show you how to
create a program in Microsoft Visual C++ 6.0, which will segment an image of
an orange, using multispectral techniques provided by the spToolkit library.
3
4
CHAPTER 1. INTRODUCTION
Chapter 2
Tutorial
2.1
Creating a Windows program
In order to visualize the results, this tutorial explains the use of the spToolkit
in a Windows environment. The Microsoft platform was chosen due to the fact
that most partners in the SmartSpectra project are using this platform. This
section shows how to create an application in Microsoft Visual C++ 6.0 that
creates windows to show images stored in memory in the spToolkit format.
1. Start Microsoft Visual C++ 6.0
2. Click File, and then New
3. Under the tab Projects, choose MFC AppWizard (exe)
4. Type Tutorial as the project name, choose a location (optional)
and click OK
5. A window with the title MFC AppWizard opens. Choose Dialog based
6. Click Finish
7. The window ”New Project Information” opens. Click OK
The result is shown in Figure 2.1
Now that you have created the project, it’s time to adjust it to your needs.
Our objective is to write a program that segments a multispectral image of an
orange. To make things easy, we will integrate the possibility to view images
in a window. The window that you have already created will be the control
window. We will create an other window to show the images.
1. Click with the right mouse button on OK in the Tutorial window
2. Choose Properties
3. Change the Caption from OK to Start
4. Change the ID from IDOK to IDSTART
5. Close the Properties window by click the ”X” on the top right
of the window
5
6
CHAPTER 2. TUTORIAL
Figure 2.1: A screenshot of the new project.
6. Find the folder Dialog in the Resource View on the left and
click on it with the right mouse button
7. Choose Insert Dialog
8. Click on the OK button in the new Dialog window and press
delete
9. Click on the Cancel button and press delete
10. Save the project
Your screen should now look like Figure 2.2. To be able to use this window,
you need to add a class to this dialog window and arrange the destruction.
1. Click with the right mouse button on the Dialog window
2. Choose ClassWizard. The window ”Adding a class ” opens
3. Make sure that ”Create a new class” is selected and click OK
4. Write CImWin as class name
5. Choose CDialog as base class
6. Click OK
7. Look in the list ”Messages” for PostNcDestroy and click Add
Function
8. Click Edit Code
9. write the following code in this function: delete(this);
2.1. CREATING A WINDOWS PROGRAM
Figure 2.2: Created a second window to show the images.
Figure 2.3: The OnNcDestroy function.
7
8
CHAPTER 2. TUTORIAL
10. Save the project
Check if you screen looks like Figure 2.3. Now that you designed a dialog window, added a class and arranged the destruction, it’s time to actually start
writing our own code. To be able to use the toolkit, we will have to insert come
includes and libaries, etc.
1. Open Windows Explorer and go to the folder /spToolkit files
which is part of this tutorial
2. copy spToolkit.dll and 1394camera.dll to your project folder
(/Tutorial)
3. Copy the subfolders /include and /lib to your project folder
4. Go back to Microsoft Visual C++
5. Choose Project in the menu bar and next Add To Project, and
then Files...
6. Double click the folder lib
7. Change Files of Type to All Files
8. Select both files and click OK
9. Choose Project again, and Settings. A window called, Project
Settings, opens
10. Choose the C/C++ tab
11. Change the Category to Preprocessor
12. Write ./include at the Additional Include Directories option
13. Choose the Link tab
14. Change the Category to Input
15. Write ./lib at the Additional Library Directories option
16. Click OK
The Project Settings Window is shown in Figure 2.4. Now continue with
inserting the includes and namespaces.
1. Click the tab FileView on the left side of the screen
2. Double click the file Tutorial.h
3. Insert the following codes at the 10th line: #include ”spToolkit.h”
and using namespace sp;
4. Click File in the menu bar and next Save All
5. Click Build in the menu bar and next Build Tutorial.exe
If the compiling went alright, you’re screen should look like Figure 2.5. Now
you are ready to use toolkit commands. The next step is to write the code
for the CImWin class. The idea is that this class will show RGB images in its
window. First we will modify the standard constructor so that it can receive
toolkit image data.
2.1. CREATING A WINDOWS PROGRAM
Figure 2.4: The Project Settings Window.
Figure 2.5: A successfully compiled project.
9
10
CHAPTER 2. TUTORIAL
1. Click on the FileView tab on the left and double click the file
ImWin.h
2. Look for the standard constructor, and add a parameter so that
it will look this: CImWin(rgb image cl image, CWnd* pParent
= NULL);
3. Now go to the ImWin.cpp file in the same way and add the
same parameter to the standard constructor
4. Now insert the code below this box in this constructor
row=image.GetRows();
col=image.GetCols();
bts = new BYTE[row*col*4];
unsigned int i=0;
for (unsigned int r=0; r < row; r++)
for (unsigned int c=0; c < col; c++) {
for (unsigned int b=0; b < 3; b++) {
// 2-b because windows images are BGR
bts[i] = image(r,c,2-b);
i++;
}
i++;
}
Create(IDD_DIALOG1,pParent);
// The size of the window is bigger than the image due to its borders
MoveWindow(0, 0, col+6, row+25, false);
ShowWindow(SW_SHOW);
5. Go to the function ”void CImWin::PostNcDestroy()” and insert
”delete[] bts;” before ”delete this”
6. Save all
Your screen should look something like Figure 2.6. The constructor has been
written, now it is time to write the actual Painting function which is responsible
for drawing the image on the screen.
1. Click on the ResourceView on the left and double click IDD DIALOG1
2. Click with the right mouse button on the Dialog window and
choose ClassWizard
3. Look in the ”Messages” list for WM PAINT
4. Click Add Function
5. click Edit Code
6. Now insert the following code in the function OnPaint
CDC bmDC; // Create compatible device context
bmDC.CreateCompatibleDC(&dc);
HBITMAP hBmp; // Create bitmap
hBmp=CreateCompatibleBitmap (dc, col, row);
SetBitmapBits(hBmp,col*row*4,bts);
2.1. CREATING A WINDOWS PROGRAM
11
Figure 2.6: The modified constructor.
bmDC.SelectObject(hBmp); //Select bitmap into
dc.BitBlt(0,0,col,row,&bmDC,0,0,SRCCOPY);
compatible dc and show
7. Save all
Your screen should now look like Figure 2.7. The only thing that is still
missing, is the declaration of several variables in the header on this class.
1. Click on the tab ClassView in the left of the screen .
2. Find CImWin in the pull down list and click it with the right
mouse button
3. Choose Add Member Variable. The window Add Member Variable opens
4. Choose in the Variable type box: unsigned int
5. Type in the Variable name box: row
6. Choose Private
7. Click OK
8. Now add unsigned int col in the same way
9. And add BYTE* bts
10. Save all
12
CHAPTER 2. TUTORIAL
Figure 2.7: The OnPaint function.
Figure 2.8: The Add Member Variable window.
2.1. CREATING A WINDOWS PROGRAM
13
Figure 2.9: Including ImWin.
The Add Member Variable window is shown in Figure 2.8. Now you have
finished writing this class. When you call the constructor of an object of this
class, passing on an rgb image, it will automatically create a window and show
the image. We will see that now.
1. Click the tab Resource View on the left side of the screen
2. Find the dialog IDD TUTORIAL DIALOG and double-click it
3. Now double-click the Start button in the Tutorial window. A
window called ”Add Member Function” opens. Click OK
4. A new function has been created in the TutorialDlg.cpp file,
called OnStart. This function will be called whenever the user
clicks on Start
5. Insert the following code:
rgb_image_cl rgb_im(200,200);
rgb_im.Clear(150);
CImWin *iw = new CImWin(rgb_im, this);
6. Scroll to the top of this file and type at the line 7: #include
”ImWin.h”
7. Save all
8. Compile project
If your project compiled successfully, your screen should look like Figure 2.9.
Now it’s time to run your program! Before running it, imagine what you expect
14
CHAPTER 2. TUTORIAL
Figure 2.10: The result!
to see. Press F5 to Run. Figure 2.10 is what I see on my screen, after pressing
Start!
2.2
Load and save multispectral images
This section will show you how you can load images, cast them from one type
to another, extract bands and save them. We will start with loading a PPM
image into an image class of unsigned chars.
1. Remove the following lines from the function OnBnClickedOk:
rgb_image_cl rgb_im(200,200);
rgb_im.Clear(150);
CImWin *iw = new CImWin(rgb_im, this);
2. Replace them by this code:
ubyte_image_cl ub_im;
input_file_cl inp_file("oranges", PPM);
inp_file >> ub_im;
CImWin *iw = new CImWin( (rgb_image_cl)ub_im, this);
3. Save all
4. Press F5 to run the project
5. Click the Start button on your project
2.2. LOAD AND SAVE MULTISPECTRAL IMAGES
15
Figure 2.11: Showing oranges on the screen.
You will now see a photo of an orange tree in your screen, see Figure 2.11.
When we study the few lines of code we just wrote, we’ll see that the casting
function is situated in the last line, casting a ubyte image cl to an rgb image cl.
Let’s expand the number of bands from 3 to 33. We’ll load an ENVI image into
the program’s memory and we’ll choose 3 bands, and show them as an RGB
image on the screen.
1. Remove the 4 lines of code that you entered before and replace
them with:
ubyte_image_cl ub_im;
input_file_cl inp_file("./images/orange_multiband_ubyte", ENVI);
inp_file >> ub_im;
rgb_image_cl rgb_im(ub_im.GetRows(), ub_im.GetCols());
rgb_im.PutBand(ub_im[15],0);
rgb_im.PutBand(ub_im[16],1);
rgb_im.PutBand(ub_im[17],2);
output_file_cl outp_file("./images/output_orange_ubyte", PPM);
outp_file << rgb_im;
CImWin *iw = new CImWin(rgb_im, this);
2. Save all
3. Press F5 to run the project
This little program loaded an ENVI image in its memory, placed 3 of its
bands into an RGB image, writes it to a file and to the screen (see Figure 2.12).
16
CHAPTER 2. TUTORIAL
Figure 2.12: Three bands taken from an ENVI image and shown as an RGB
image.
You have probably been slightly annoyed by the fact that this program took
quite a long time to be executed. This is mainly due to the fact that the ENVI
file is 11 MB big. A solution to the program is to load just the 3 bands that
you really need, instead of all of them. This can be done in the following way.
1. Replace the code:
ubyte_image_cl ub_im;
input_file_cl inp_file("./images/orange_multiband_ubyte", ENVI);
inp_file >> ub_im;
rgb_image_cl rgb_im(ub_im.GetRows(), ub_im.GetCols());
rgb_im.PutBand(ub_im[15],0);
rgb_im.PutBand(ub_im[16],1);
rgb_im.PutBand(ub_im[17],2);
2. ...with this code:
rgb_image_cl rgb_im;
input_file_cl inp_file("./images/orange_multiband_ubyte", ENVI);
inp_file.LoadImage(rgb_im,15,17);
3. Save all
4. Press F5 to run the project
As you will have noticed, the program executes much quicker now. However,
LoadImage can only load a number of successive bands.
2.3. ERROR HANDLING
17
Figure 2.13: An unclear error message.
2.3
Error handling
In this section we will explore the different ways of error detection.
1. Replace the code that you typed in the function OnBnClickedOk with the following code:
rgb_image_cl rgb_im;
input_file_cl inp_file("./images/bird",PGM);
inp_file >> rgb_im;
rgb_im(600,125,1)=255;
CImWin *iw = new CImWin(rgb_im, this);
2. Save all
3. Press F5 to run the project
You should have gotten an error message as shown in Figure 2.13. Normally,
at this moment you would start a long process of searching for the error, debugging, etc. Fortunately, the toolkit has a tool incorporated to look up errors
quickly. It is based on the C++ method ”try-throw-catch”: you try a function,
an error might be thrown which will be caught and handled. Here is how it
works.
1. Add the try and catch function as shown here:
try {
rgb_image_cl rgb_im;
input_file_cl inp_file("./images/bird",PGM);
inp_file >> rgb_im;
rgb_im(600,125,1)=255;
CImWin *iw = new CImWin(rgb_im, this);
}
catch (error_message_cl e) {MessageBox(e.ReturnReport().c_str());}
2. Save all
3. Press F5 to run the project
Obviously you still got an error, but it is not so unclear as it was before.
Now you’re being told that there is something wrong with the number of bands
18
CHAPTER 2. TUTORIAL
Figure 2.14: Still an error message, but with a hint.
of the image. Let’s have a closer look at the program. You have an RGB image,
which obviously has three bands. Then, you try to save a PGM image in it,
which has just one band. The toolkit doesn’t know what to do with the two
remaining bands and throws an error of type error message cl. Let’s adjust the
program to eliminate the error.
1. Rewrite the program as follows:
try {
ubyte_image_cl ub_im; //ubyte_image_cl can have any number of bands
input_file_cl inp_file("./images/bird",PGM);
inp_file >> ub_im;
ub_im.AddBands(2); //expand the image with 2 bands to a total of 3
ub_im.PutBand(ub_im[0],1); //copy the first band into the second
ub_im.PutBand(ub_im[0],2); //copy the first band into the third
ub_im(600,125,1)=255; //change the pixel
CImWin *iw = new CImWin( (rgb_image_cl)ub_im, this);
//don’t forget casting to rgb_image_cl!
}
catch (error_message_cl e) {MessageBox(e.ReturnReport().c_str());}
2. Save all
3. Press F5 to run the project
A new error occurred, saying that the requested row number is higher than
the available number of rows. The problem appears to be related to the part of
de code where we assign a new value to the pixel at row=600, column=125 and
band=1. The image does not have 600 rows, which explains the error.
1. Change this line of code:
ub_im(600,125,1)=255;
2. into this:
ub_im(60,125,1)=255;
3. Save all
4. Press F5 to run the project
If all went well, you did not get any errors and the bird has a little dot in
his eye (see Figure 2.16). The moral of the story: don’t forget to use try/catch
when you use toolkit instructions!
2.3. ERROR HANDLING
19
Figure 2.15: Another error.
Figure 2.16: The result.
20
CHAPTER 2. TUTORIAL
Figure 2.17: The cropped image.
2.4
Performing image operations
Now it is time to actually do something with our images! As said before, the
goal of this tutorial is to perform a segmentation on a multispectral image of
an orange. To avoid a long processing time, we’ll crop the image, concentrating
on the part that really matters.
1. Replace the code that you typed in the function OnBnClickedOk with the following code:
ubyte_image_cl ub_im;
input_file_cl inp_file("./images/orange_multiband_ubyte",ENVI);
inp_file >> ub_im;
ub_im.Crop(100,200,400,500);
CImWin *iw = new CImWin( (rgb_image_cl)ub_im, this);
//casting to RGB results in showing only the first 3 bands
2. Save all
3. Press F5 to run the project
The image has become smaller, only showing the most interesting part. See
Figure 2.17. We will now threshold the image. Since it is a multispectral image,
we need to give a threshold value for every band. This is done by means of a
vector. To keep things simple, we give the same value for every band.
1. Insert the following code after the line of code where you crop
the image.
vector<unsigned char> v(ub_im.GetBands(), 50);
ub_im = ub_im.Thresholding(v);
// Copy the first band into the second and third
// to create a gray RGB image.
ub_im.AddBands(2);
ub_im.PutBand(ub_im[0],1);
ub_im.PutBand(ub_im[0],2);
2. Save all
2.4. PERFORMING IMAGE OPERATIONS
21
Figure 2.18: The segmented image.
3. Press F5 to run the project
The result is shown in Figure 2.18. This simple thresholding function did a
pretty good job segmenting the defects of the orange. By adjusting the thresholding values for every band, the results can easily improve. You can try that
yourself.
Now we will detect the borders of the segmented image, using convolution. We
will use a convolution kernel for horizontal edge detection and one for vertical
edge detection. Then, we will combine those results into one image.
1. Insert the following code after the line of code where you threshold the image.
//create both kernels and perform the convolution
gray_ubyte_image_cl gub_im_x, gub_im_y;
mask_cl mask1(Sobel_x);
gub_im_x=ub_im.Convolution(mask1);
mask_cl mask2(Sobel_y);
gub_im_y=ub_im.Convolution(mask2);
// Create a real image and use it to write the results to
gray_real_image_cl gr_im(ub_im.GetRows(), ub_im.GetCols());
for (unsigned int r=0; r<gr_im.GetRows(); r++)
for (unsigned int c=0; c<gr_im.GetCols(); c++)
gr_im(r,c)=sqrt(pow(gub_im_x(r,c),2)+pow(gub_im_y(r,c),2));
// Normalize the image and cast it to ubyte_image_cl
gr_im=gr_im.Normalization(0,255);
ub_im=(ubyte_image_cl) gr_im;
2. Save all
3. Press F5 to run the project
The result (shown in Figure 2.19) is not very impressive. This is due to the
amount of noise present in the thresholded image. There are various techniques
22
CHAPTER 2. TUTORIAL
Figure 2.19: Edge detection.
to improve the result, like e.g. region growing. However, this is not further
discussed in this tutorial and neither is it available in the toolkit (yet).
2.5
Band selection
Working with multispectral bands is usually very time consuming, especially
when the image has a large number of bands. Try the following program and
clock the time it takes to execute (It could take up to half an hour).
1. Replace the code that you typed in the function OnBnClickedOk with the following code:
// Load the source image
ushort_image_cl us_im;
input_file_cl inp("./images/sp_orange_20_b_vis", ENVI);
inp >> us_im;
// Transform the image from 12 bit to 8 bit
ubyte_image_cl ub_im;
ub_im = (ubyte_image_cl) (us_im/16);
// Only use the part of the image that shows the orange
ub_im.Crop(100,150,400,550);
// Scale the image to reduce the classification time
ub_im=ub_im.Scale(0.4, 0.4);
// Load the trainingset
ubyte_image_cl training;
input_file_cl training_file("./images/training", ENVI);
training_file >> training;
// Perform the 1-NN classification
gray_ubyte_image_cl label(ub_im.GetRows(), ub_im.GetCols());
label.Clear();
double d;
2.5. BAND SELECTION
23
for (unsigned int r=0; r<ub_im.GetRows(); r++)
for (unsigned int c=0; c<ub_im.GetCols(); c++) {
double len=1000; // start with a great length
// Determine the distance from the pixel to every prototype
for (unsigned int rr=0; rr<training.GetRows(); rr++)
for (unsigned int cc=0; cc<training.GetCols(); cc++){
d = dist(ub_im.GetPixel(r,c),training.GetPixel(rr,cc));
if (d < len) {
// give this pixel the label of the closest neighbour
label(r,c)=training(rr,cc,training.GetBands()-1);
len = d;
}
}
}
// Save the label file
output_file_cl out("label\_all\_bands", PGM);
out << label;
// Copy the first band into the second and third to create a gray RGB image.
ub_im = label;
ub_im.AddBands(2);
ub_im.PutBand(ub_im[0],1);
ub_im.PutBand(ub_im[0],2);
CImWin *iw = new CImWin( (rgb_image_cl)ub_im, this);
2. Now include the following function in you project. Place this
function above the function OnBnClickedOk.
double dist(ubyte_pixel_tp c1, ubyte_pixel_tp c2) {
//Determine the distance between c1 and c2
double sum=0;
for (unsigned int b=0; b<c1.GetBands(); b++)
sum += pow(c2[b]-c1[b],2);
return sqrt(sum);
}
3. Save all
4. Press F5 to run the project
As you have probably guessed, this program is a 1-NN classifier. I determines the distance between a multispectral pixel of the source image and every
prototype in the trainingset. Next, it labels this pixel with the same class as
the nearest prototype belongs to. The result is shown in Figure 2.20.
Since our image has 33 bands, this is a very costly process. Especially if you
consider that you might need to repeat this process all the time, which would
be the case with classifying oranges in a warehouse!
It would safe a lot of time if you could reach a similar result with just a part of
those bands. This is what the band selection function does; it selects the bands
that together contain a large amount of information and simultaneously are the
least mutually correlated. The band selection process is time-consuming, but
consider that it only has to be performed once to know which bands are most
24
CHAPTER 2. TUTORIAL
Figure 2.20: The result of the classification process (black=background;
gray=orange; white=overripe).
useful for the type of classification that you are interested in. Clock the time
again and compare the result with the previous result.
1. Add the following code behind the part where you transformed
the image ub im from 12 bit to 8 bit:
// Perform band selection
ub_im = ub_im.BandSelection(3);
2. Add the following code behind the part where you load the
training file into the image training:
// Pick the previously selected bands from the trainingset
ubyte_image_cl training_temp(training.GetRows(),training.GetCols(),4);
vector<unsigned int> sb;
sb = ub_im.GetSelectedBands();
training_temp.PutBand(training[sb[0]],0);
training_temp.PutBand(training[sb[1]],1);
training_temp.PutBand(training[sb[2]],2);
training_temp.PutBand(training[training.GetBands()-1],3);
training = training_temp;
3. Change the output file name from label all bands to label sel bands.
4. Save all
5. Press F5 to run the project
As you can see, the result is very similar to the previous result. The band
selection function is unsupervised and can therefore be used in any situation.
There is another reason that explains the slowness of this program. The images
are stored in memory according to the Band Sequential format (BSQ). That
means that bands are stored one after another, in contrast to Band Interleaved
by Pixel (BIP). BIP means that multispectral pixels are stored one after the
other. In the classification program shown above, multispectral pixels are repeatedly asked for. Due to the BSQ format used in the toolkit, these pixels
have to be built every time. Obviously, this is very time-consuming. A solution
would be to create the pixels one time and store them for further use.
2.6. USING A FIREWIRE CAMERA
25
Figure 2.21: The result of the classification process using band selection
(black=background; gray=orange; white=overripe).
2.6
Using a Firewire camera
The toolkit is designed to work with Firewire cameras with D-cam specification, including the SmartSpectra camera system. In this version of the toolkit
the interface to the SmartSpectra camera system is not activated yet, but the
general interface to Firewire cameras is. In this section we will find out how to
use the camera, how to communicate with it and how to take photos.
Before you start this section, you should be sure that the camera driver has
been properly installed and the camera is plugged in.
1. Place the following code into the function OnBnClickedOk:
ubyte_image_cl ub_im;
camera_cl cam;
cam >> ub_im;
//
//
//
//
if
{
The camera constructor determines automatically
which video mode your camera can handle.
Therefore, it is not sure if the result will
contain 1 band (gray) or 3 bands (color)
(ub_im.GetBands() < 3)
ub_im.AddBands(2);
ub_im.PutBand(ub_im[0],1);
ub_im.PutBand(ub_im[0],2);
}
CImWin *iw = new CImWin( (rgb_image_cl)ub_im, this);
2. Save all
3. Press F5 to run the project
With some cameras you’ll get a black or gray image. This could be caused
by the fact that they need a ”warming up” period. Later on in this section a
method will be introduced that captures several images in a row. This could
26
CHAPTER 2. TUTORIAL
solve the problem, because it allows you throw away the first couple of images
and use the last one.
You can find out in which video mode your camera is running.
1. Add the following code behind the part where you read the
image from the camera into ub im:
stringstream ss;
ss << "Format: " << cam.GetVideoFormat() << endl
<< "Mode:
" << cam.GetVideoMode() << endl
<< "Rate:
" << cam.GetVideoFrameRate();
MessageBox(ss.str().c_str());
2. Save all
3. Press F5 to run the project
A very important part of the camera cl class are the ControlRegisters. These
registers handle the different features of the camera, like the shutter, gain, sharpness, etc. These registers are of type C1394CameraControl, which has the
following functions:
Construction
C1394CameraControl()
virtual C1394CameraControl()
Values
void GetValues() // gets current values from camera
void SetValues() // sets values
Modes
void TurnOn(bool on) // turns a feature on (true) or off (false)
void SetOnePush() // sets the one-push auto-mode command
void SetAutoMode(bool on) // sets the auto-mode (true) or manual-mode
(false)
Status
void Inquire() // updates feature availability
void Status() // updates current values
Image that you would like to know if you can adjust the shutter speed of your
camera. You would like to know if this control function is present in your camera, if it can be turned on and off and what its minimum and maximum allowed
values are.
1. Add the following code behind the last part you added:
// You need to call the function Inquire() to update the values
cam.ControlShutter().Inquire();
stringstream ss2;
ss2 << "Present:
" << cam.ControlShutter().m_present << endl
<< "On/off:
" << cam.ControlShutter().m_onoff << endl
2.6. USING A FIREWIRE CAMERA
27
<< "Value_min: " << cam.ControlShutter().m_min << endl
<< "Value_max: " << cam.ControlShutter().m_max << endl;
MessageBox(ss2.str().c_str());
2. Save all
3. Press F5 to run the project
Now let’s find out what the current value is and change it.
1. Add the following code behind the last part you added:
// You can update by using Status() or GetValues()
cam.ControlShutter().Status();
int old_value, new_value;
old_value = cam.ControlShutter().m_value1;
cam.ControlShutter().m_value1 = 200;
cam.ControlShutter().SetValues();
new_value = cam.ControlShutter().m_value1;
stringstream ss3;
ss3 << "Old value:
" << old_value << endl
<< "New value:
" << new_value;
MessageBox(ss3.str().c_str());
2. Save all
3. Press F5 to run the project
Luckily, writing values to these registers can be done in an easier way:
1. Replace the following code:
cam.ControlShutter().m_value1 = 200;
cam.ControlShutter().SetValues();
2. by this code:
cam.SetShutter(200);
3. Save all
4. Press F5 to run the project
As mentioned before, there is another capturing method. This method captures a certain number of images in a row at high speed. They are all saved in
one multispectral image structure.
1. Now replace all the code you wrote in this section by:
ubyte_image_cl ub_im;
camera_cl cam;
cam.Acquisition(10,ub_im);
//Move the last three bands to the first bands
//They will be shown on the screen
ub_im.PutBand(ub_im[ub_im.GetBands()-1],2);
ub_im.PutBand(ub_im[ub_im.GetBands()-2],1);
ub_im.PutBand(ub_im[ub_im.GetBands()-3],0);
CImWin *iw = new CImWin( (rgb_image_cl)ub_im, this);
28
CHAPTER 2. TUTORIAL
2. Save all
3. Press F5 to run the project
If the resulting image is to dark or to bright, you might want to adjust the
shutter.
Practically all the functions available in the camera driver are also available
in the toolkit. This is because there are situations in which it is more convenient to use those than the higher level functions of the toolkit. However, this
subject is not being dealt with in this tutorial.
2.7
Other functions
There are still a number of functions that have not been treated in this tutorial.
In this section we will write one program that shows several functions at the
same time.
First we’ll introduce the Region Of Interest (ROI). It is one of the constructors of the image classes, which allows you to construct a copy of another image
(or part of another image). The difference with the copy constructor is that
the ROI image doesn’t have its own memory. It actually points to the memory
of the source image. As a consequence, if you change the appearance of a ROI
image, its source image will change as well.
1. Place the following code into the function OnBnClickedOk:
ubyte_image_cl ub_im;
input_file_cl inp_file("./images/oranges",PPM);
inp_file >> ub_im;
// create a region of interest (ROI)
ubyte_image_cl roi(ub_im,10,70,80,80);
// rotate the region of interest
roi=roi.Rotate(90);
// Show the region of interest
CImWin *iw1 = new CImWin( (rgb_image_cl)roi, this);
// Notice that the source image changes as well
CImWin *iw2 = new CImWin( (rgb_image_cl)ub_im, this);
2. Save all
3. Press F5 to run the project
With the toolkit you can calculate the Shannon Entropy of any multispectral
image. The entropy is a measure for the amount of information that an image
contains.
1. Extend your program with the following code:
// Calculating the entropy of this image (measure of information)
stringstream ss;
ss << "The Shannon Entropy of ub_im is: " << ub_im.ShannonEntropy(256);
MessageBox(ss.str().c_str());
2.7. OTHER FUNCTIONS
29
Figure 2.22: The ROI image.
Figure 2.23: The source image that was used to create the ROI.
2. Save all
3. Press F5 to run the project
It takes about a minute to calculate the entropy. As a absolute number the
entropy doesn’t mean much, but you can use it to compare the information of
one image with another. Let’s threshold the image and calculate the entropy
again. Do you expect it to be higher or lower?
1. Extend your program with the following code:
// threshold the image and show it on the screen
vector<unsigned char> v(ub_im.GetBands(), 50);
ub_im = ub_im.Thresholding(v);
// change the 1 band label image into a 3 band image
ub_im.AddBands(2);
ub_im.PutBand(ub_im[0],1);
ub_im.PutBand(ub_im[0],2);
CImWin *iw3 = new CImWin( (rgb_image_cl)ub_im, this);
// Now calculate the entropy again. What do you expect to see?
stringstream ss2;
ss2 << "The Shannon Entropy of the thresholded ub_im is: " << ub_im.ShannonEntropy
MessageBox(ss2.str().c_str());
2. Save all
30
CHAPTER 2. TUTORIAL
3. Press F5 to run the project
2.8
epilogue
You have reached the end of the tutorial. You have been introduced to a number of basic operations to give you a basic understanding of how the toolkit
works. However, more functions will be added to the toolkit and consequently,
the tutorial will be updated too.
If you have any questions of comments, you can send them to klaren@vision.uji.es.
Good luck with your multispectral image processing!