Laboratory 1: Getting to grips with OpenCV

Introduction

The Computer Vision module consists of both lectures and laboratories. Roughly speaking, the lectures explain what techniques are used to process image or video data, how they work, and how they are combined into complete systems. The laboratories, which is what concern us here, are where you put those ideas and principles into practice.

There are two distinct parts to the programme of laboratories: the first half gets you to write and test routines that provide useful computer vision functionality, while the second half uses these functions in a set of vision applications. The laboratories are self-paced but the routine-writing ones are assessed in the first progress test mid-way through the term, while the applications part is assessed in the second progress test near the end of the term. These progress tests are open-book, meaning that you should make a record what you do and why as you work through the laboratories and then use your records to answer the progress test questions.

There are two widely-used ways of developing computer vision techniques and applications, Matlab and OpenCV. We are going to work with the latter as it makes you more employable in the industry and interfaces better to modern machine learning. OpenCV is written in C++ and has 'wrappers' that allow it to be used from a variety of programming languages; but here the focus will be on the use of Python, again because it has become the norm for research and development. All the examples you are given are in Python, and the library or module that you are expected to extend in the first half of the laboratory programme is also written in Python.

OpenCV and its Python wrapper run on Windows, Linux and macOS. In previous iterations, the laboratory programme has had to be built around Linux because the Windows port in particular was rather unreliable. However, it now appears to be stable and so you should be able to work on any of the three operating systems. Do bear in mind that the laboratories and progress tests are set on the basis of the Linux and macOS versions of OpenCV so if you're working under Windows, you should ensure that you get the same results as on the machines in our Software Labs or Horizon server under Linux. If you haven't used Linux...then why not? It's the primary free operating system and any self-respecting graduate in the computing or electronics areas should have some experience of it. You will also see that most of the examples expect you to be able to run Python programs from the command line. I have seen first-hand that many students are unable to write complete programs, instead relying on button-clicking in Jupyter or another IDE to run them --- that makes you only half a programmer.

If you want to install OpenCV on your own machine, it is normally straightforward; there are guidelines in the lecture notes. You need to make sure you install the same version as in the Software Labs and the way to do that is to start a python interpreter on one of those machines and type the incantations

>>> import cv2
>>> cv2.__version__
'4.5.3'

where >>> is the prompt from the Python interpreter. The version installed in the Software Lab will be much more up to date than that printed out here.

Getting started

The starting point for your laboratory work is a zip-file which contains:

You should download the zip-file and unpack it. (Later scripts will have separate zip-files of imagery associated with them.) You will work on sxcv.py throughout all the laboratory sessions, so don't lose it or you will have to do the work again!

To run review.py, you should bring up a terminal window and type the command:

python review.py

You should see the output:

testimage is monochrome of size 13 rows x 10 columns with uint8 pixels.

You might have to use python3 rather than python depending on how the interpreter is installed on your system.

If you run the same program with a filename:

python review.py sx.jpg

you should see the output

sx.jpg has 3 channels of size 512 rows x 512 columns with uint8 pixels.

instead. If you normally use an IDE and run programs by clicking a button, this might seem a little strange way to tun a program; but you will see as the laboratories progress that running programs this way brings benefits so do make sure you can do it --- and there will be questions in the progress tests relating to this.

Let us now review the code in review.py:

#!/usr/bin/env python
"""review.py -- a "hello world" program for sxcv and cv2"""
import sys, sxcv, cv2

# If a filename was given on the command line, read it in.  Otherwise, use
# the test image built into the sxcv module.
if len (sys.argv) > 1:
    im = cv2.imread (sys.argv[1])
    if im is None:
        print ("Couldn't read the image file '%s'!" % sys.argv[1],
               file=sys.stderr)
        exit (1)
    name = sys.argv[1]
else:
    im = sxcv.testimage1 ()
    name = "testimage"

# Output a one-line summary of the image.
print (sxcv.describe (im, name))

There a few interesting points worth bringing out.

You can use review.py as the starting point for testing the routines that you add to sxcv.py...though you are encouraged to give it a different name.

Getting going with the sxcv Python module

A good way to see what is in the module is by typing one of the commands

pydoc .\sxcv.py       # Windows
pydoc ./sxcv.py       # everywhere else

(This may need to be pydoc3 on some systems.) This pulls the documentation strings from the code and presents them to you in a nicely-formatted way. If pydoc isn't installed on your system, the longer command

python -m pydoc ./sxcv.py

(.\sxcv.py on Windows) is equivalent. Do read through the module before you proceed.

The routines present in the version of sxcv.py that you downloaded are a starting point for you to develop your own routines. They include ones to examine and visualize images and data extracted from them, generate test images and some simple debugging aids. As you work through these laboratory scripts, you should add the routines that are discussed below the comment that reads

#-------------------------------------------------------------------------------
# LIBRARY ROUTINES.
#-------------------------------------------------------------------------------

To show you what to do, let us add a routine to compute the mean of an image. Your starting point is the following text:

def mean (im):
    """
    Return the mean of the pixel values an image.

    Args:
        im (image): image for which the mean value is to be found

    Returns:
        ave (float): the mean of the image

    Tests:
        >>> ave = mean (arrowhead ())
        >>> print ("OK") if abs (ave - 39.66666666) < 1.0e-5 else print ("bad")
        OK
    """

The first line is a definition of a Python routine and, as you will see, it takes an image as its parameter. The comment that follows the def line is its documentation (what pydoc will extract). This includes an example of how it is invoked after the Tests line: the >>> again represents the Python prompt. You should paste the above code lines unchanged into your copy of sxcv.py.

The routine definition does not, of course, include any code for computing the mean. You could compute this yourself, iterating over the rows and columns of the image as discussed in lectures; but finding the mean of a set of numbers is such a common thing to do that there is suitable functionality in numpy, the representation used for OpenCV images in Python. If you do a web-search for something like numpy calculate mean you'll find there is a method called mean, so all you need do is add the line

return numpy.mean (im)

at the end of your mean routine with the right indentation. The routines you have to write won't usually be one-liners like this, of course.

With your functionality added, you now need to test whether it works. Python provides functionality to pull tests out of these so-called docstring comments, execute them, and check that they produce the same output as is in the comment. In this case, it will pull out the two lines of code

ave = mean (arrowhead ())
print ("OK") if abs (ave - 39.66666666) < 1.0e-5 else print ("bad")

The first line of these generates the 'arrowhead' image which appears in the software chapter of the lecture notes, computes its mean using your newly-written routine and checks that the result is within \(10^{-5}\) of 39.66666666; if it is, it prints OK and if it isn't, it prints bad. A correctly working routine will print OK.

Rather than go to the effort of extracting and running the code manually, Python provides tooling for doing this. Simply type the command

python -m doctest sxcv.py

to run all the tests. If you see no output, thay have all succeeded. To see all the tests being run, add the -v qualifier to the above command.

We shall take the same approach for all the other functions you have to add to sxcv.py: you will be given the routine specification and docstring and it is your job to implement the functionality of the code.

Now that you have seen how to add a routine to sxcv.py, add three further ones as specified below. The first two are also numpy one-liners, while the last combines the values returned from the first two.

def highest (im):
    """
    Return the maximum of the pixel values of an image.

    Args:
        im (image): image for which the maximum value is to be found

    Returns:
        hi (of same type as image): highest value in the image

    Tests:
        >>> im = testimage1 ()
        >>> hi = highest (im)
        >>> print (hi)
        15
        >>> print (type (hi))
        <class 'int'>
    """


def lowest (im):
    """
    Return the minimum of the pixel values of an image.

    Args:
        im (image): image for which the maximum value is to be found
    
    Returns:
        lo (of same type as image): lowest value in the image

    Tests:
        >>> im = testimage1 ()
        >>> lo = lowest (im)
        >>> print (lo)
        10
        >>> print (type (lo))
        <class 'int'>
    """


def extremes (im):
    """
    Return the minimum and maximum of the pixel values of an image.

    Args:
        im (image): image for which the maximum value is to be found
    
    Returns:
        lo (of same type as image): lowest value in the image
        hi (of same type as image): highest value in the image

    Tests:
        >>> im = testimage1 ()
        >>> lo, hi = extremes (im)
        >>> print (lo)
        10
        >>> print (hi)
        15
        >>> print (type (hi))
        <class 'int'>
    """

To wrap up this experiment, copy review.py to (say) display.py. Find the routine in sxcv.py called display and make the program invoke it to display the image specified on the command line. When you run

python display.py sx.jpg

you should see the following image pop up in a window on your screen.

Experiment with the parameters in the call to find out what they do.


If you have struggled to carry out this experiment, below is a screencast which shows me working through the important parts of it myself. Note that the image displays in my terminal window; yours won't do that. I apologize for my poor typing prowess: 40 years of using computers don't appear to have improved it much!

I shan't do this for subsequent experiments; but you shouldn't need as much hand-holding.

All the laboratory scripts


Web page maintained by Adrian F. Clark using Emacs, the One True Editor ;-)