193 lines
		
	
	
		
			5.6 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable File
		
	
	
	
	
			
		
		
	
	
			193 lines
		
	
	
		
			5.6 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable File
		
	
	
	
	
| #! /usr/bin/env python
 | |
| 
 | |
| print "OpenCV Python version of lkdemo"
 | |
| 
 | |
| import sys
 | |
| 
 | |
| # import the necessary things for OpenCV
 | |
| import cv2.cv as cv
 | |
| 
 | |
| #############################################################################
 | |
| # some "constants"
 | |
| 
 | |
| win_size = 10
 | |
| MAX_COUNT = 500
 | |
| 
 | |
| #############################################################################
 | |
| # some "global" variables
 | |
| 
 | |
| image = None
 | |
| pt = None
 | |
| add_remove_pt = False
 | |
| flags = 0
 | |
| night_mode = False
 | |
| need_to_init = False
 | |
| 
 | |
| #############################################################################
 | |
| # the mouse callback
 | |
| 
 | |
| # the callback on the trackbar
 | |
| def on_mouse (event, x, y, flags, param):
 | |
| 
 | |
|     # we will use the global pt and add_remove_pt
 | |
|     global pt
 | |
|     global add_remove_pt
 | |
| 
 | |
|     if image is None:
 | |
|         # not initialized, so skip
 | |
|         return
 | |
| 
 | |
|     if image.origin != 0:
 | |
|         # different origin
 | |
|         y = image.height - y
 | |
| 
 | |
|     if event == cv.CV_EVENT_LBUTTONDOWN:
 | |
|         # user has click, so memorize it
 | |
|         pt = (x, y)
 | |
|         add_remove_pt = True
 | |
| 
 | |
| #############################################################################
 | |
| # so, here is the main part of the program
 | |
| 
 | |
| if __name__ == '__main__':
 | |
| 
 | |
|     frames = sys.argv[1:]
 | |
|     if frames == []:
 | |
|       print "usage lkdemo.py <image files>"
 | |
|       sys.exit(1)
 | |
| 
 | |
|     # display a small howto use it
 | |
|     print "Hot keys: \n" \
 | |
|           "\tESC - quit the program\n" \
 | |
|           "\tr - auto-initialize tracking\n" \
 | |
|           "\tc - delete all the points\n" \
 | |
|           "\tn - switch the \"night\" mode on/off\n" \
 | |
|           "\tSPACE - next frame\n" \
 | |
|           "To add/remove a feature point click it\n"
 | |
| 
 | |
|     # first, create the necessary windows
 | |
|     cv.NamedWindow ('LkDemo', cv.CV_WINDOW_AUTOSIZE)
 | |
| 
 | |
|     # register the mouse callback
 | |
|     cv.SetMouseCallback ('LkDemo', on_mouse, None)
 | |
| 
 | |
|     fc = 0
 | |
|     while 1:
 | |
|         # do forever
 | |
| 
 | |
|         frame = cv.LoadImage(frames[fc])
 | |
| 
 | |
|         if image is None:
 | |
|             # create the images we need
 | |
|             image = cv.CreateImage (cv.GetSize (frame), 8, 3)
 | |
|             image.origin = frame.origin
 | |
|             grey = cv.CreateImage (cv.GetSize (frame), 8, 1)
 | |
|             prev_grey = cv.CreateImage (cv.GetSize (frame), 8, 1)
 | |
|             pyramid = cv.CreateImage (cv.GetSize (frame), 8, 1)
 | |
|             prev_pyramid = cv.CreateImage (cv.GetSize (frame), 8, 1)
 | |
|             features = []
 | |
| 
 | |
|         # copy the frame, so we can draw on it
 | |
|         cv.Copy (frame, image)
 | |
| 
 | |
|         # create a grey version of the image
 | |
|         cv.CvtColor (image, grey, cv.CV_BGR2GRAY)
 | |
| 
 | |
|         if night_mode:
 | |
|             # night mode: only display the points
 | |
|             cv.SetZero (image)
 | |
| 
 | |
|         if need_to_init:
 | |
|             # we want to search all the good points
 | |
| 
 | |
|             # create the wanted images
 | |
|             eig = cv.CreateImage (cv.GetSize (grey), 32, 1)
 | |
|             temp = cv.CreateImage (cv.GetSize (grey), 32, 1)
 | |
| 
 | |
|             # the default parameters
 | |
|             quality = 0.01
 | |
|             min_distance = 10
 | |
| 
 | |
|             # search the good points
 | |
|             features = cv.GoodFeaturesToTrack (
 | |
|                 grey, eig, temp,
 | |
|                 MAX_COUNT,
 | |
|                 quality, min_distance, None, 3, 0, 0.04)
 | |
| 
 | |
|             # refine the corner locations
 | |
|             features = cv.FindCornerSubPix (
 | |
|                 grey,
 | |
|                 features,
 | |
|                 (win_size, win_size),  (-1, -1),
 | |
|                 (cv.CV_TERMCRIT_ITER | cv.CV_TERMCRIT_EPS, 20, 0.03))
 | |
| 
 | |
|         elif features != []:
 | |
|             # we have points, so display them
 | |
| 
 | |
|             # calculate the optical flow
 | |
|             features, status, track_error = cv.CalcOpticalFlowPyrLK (
 | |
|                 prev_grey, grey, prev_pyramid, pyramid,
 | |
|                 features,
 | |
|                 (win_size, win_size), 3,
 | |
|                 (cv.CV_TERMCRIT_ITER|cv.CV_TERMCRIT_EPS, 20, 0.03),
 | |
|                 flags)
 | |
| 
 | |
|             # set back the points we keep
 | |
|             features = [ p for (st,p) in zip(status, features) if st]
 | |
| 
 | |
|             if add_remove_pt:
 | |
|                 # we have a point to add, so see if it is close to
 | |
|                 # another one. If yes, don't use it
 | |
|                 def ptptdist(p0, p1):
 | |
|                     dx = p0[0] - p1[0]
 | |
|                     dy = p0[1] - p1[1]
 | |
|                     return dx**2 + dy**2
 | |
|                 if min([ ptptdist(pt, p) for p in features ]) < 25:
 | |
|                     # too close
 | |
|                     add_remove_pt = 0
 | |
| 
 | |
|             # draw the points as green circles
 | |
|             for the_point in features:
 | |
|                 cv.Circle (image, (int(the_point[0]), int(the_point[1])), 3, (0, 255, 0, 0), -1, 8, 0)
 | |
| 
 | |
|         if add_remove_pt:
 | |
|             # we want to add a point
 | |
|             # refine this corner location and append it to 'features'
 | |
| 
 | |
|             features += cv.FindCornerSubPix (
 | |
|                 grey,
 | |
|                 [pt],
 | |
|                 (win_size, win_size),  (-1, -1),
 | |
|                 (cv.CV_TERMCRIT_ITER | cv.CV_TERMCRIT_EPS,
 | |
|                 20, 0.03))
 | |
|             # we are no longer in "add_remove_pt" mode
 | |
|             add_remove_pt = False
 | |
| 
 | |
|         # swapping
 | |
|         prev_grey, grey = grey, prev_grey
 | |
|         prev_pyramid, pyramid = pyramid, prev_pyramid
 | |
|         need_to_init = False
 | |
| 
 | |
|         # we can now display the image
 | |
|         cv.ShowImage ('LkDemo', image)
 | |
| 
 | |
|         # handle events
 | |
|         c = cv.WaitKey(10) % 0x100
 | |
| 
 | |
|         if c == 27:
 | |
|             # user has press the ESC key, so exit
 | |
|             break
 | |
| 
 | |
|         # processing depending on the character
 | |
|         if 32 <= c and c < 128:
 | |
|           cc = chr(c).lower()
 | |
|           if cc == 'r':
 | |
|               need_to_init = True
 | |
|           elif cc == 'c':
 | |
|               features = []
 | |
|           elif cc == 'n':
 | |
|               night_mode = not night_mode
 | |
|           elif cc == ' ':
 | |
|               fc = (fc + 1) % len(frames)
 | |
|     cv.DestroyAllWindows()
 | 
