Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

How to crop the internal area of a contour?

I am working on Retinal fundus images.The image consists of a circular retina on a black background. With OpenCV, I have managed to get a contour which surrounds the whole circular Retina. What I need is to crop out the circular retina from the black background.

like image 784
Gaurav Patil Avatar asked Feb 27 '15 06:02

Gaurav Patil


People also ask

How do you measure width and height for contouring?

To obtain the height and width of a contour, you can use cv2. boundingRect . The function returns the contour information in the form of x,y,w,h . The height for a specific contour will be h and the width will be w .


2 Answers

It is unclear in your question whether you want to actually crop out the information that is defined within the contour or mask out the information that isn't relevant to the contour chosen. I'll explore what to do in both situations.


Masking out the information

Assuming you ran cv2.findContours on your image, you will have received a structure that lists all of the contours available in your image. I'm also assuming that you know the index of the contour that was used to surround the object you want. Assuming this is stored in idx, first use cv2.drawContours to draw a filled version of this contour onto a blank image, then use this image to index into your image to extract out the object. This logic masks out any irrelevant information and only retain what is important - which is defined within the contour you have selected. The code to do this would look something like the following, assuming your image is a grayscale image stored in img:

import numpy as np import cv2 img = cv2.imread('...', 0) # Read in your image # contours, _ = cv2.findContours(...) # Your call to find the contours using OpenCV 2.4.x _, contours, _ = cv2.findContours(...) # Your call to find the contours idx = ... # The index of the contour that surrounds your object mask = np.zeros_like(img) # Create mask where white is what we want, black otherwise cv2.drawContours(mask, contours, idx, 255, -1) # Draw filled contour in mask out = np.zeros_like(img) # Extract out the object and place into output image out[mask == 255] = img[mask == 255]  # Show the output image cv2.imshow('Output', out) cv2.waitKey(0) cv2.destroyAllWindows() 

If you actually want to crop...

If you want to crop the image, you need to define the minimum spanning bounding box of the area defined by the contour. You can find the top left and lower right corner of the bounding box, then use indexing to crop out what you need. The code will be the same as before, but there will be an additional cropping step:

import numpy as np import cv2 img = cv2.imread('...', 0) # Read in your image # contours, _ = cv2.findContours(...) # Your call to find the contours using OpenCV 2.4.x _, contours, _ = cv2.findContours(...) # Your call to find the contours idx = ... # The index of the contour that surrounds your object mask = np.zeros_like(img) # Create mask where white is what we want, black otherwise cv2.drawContours(mask, contours, idx, 255, -1) # Draw filled contour in mask out = np.zeros_like(img) # Extract out the object and place into output image out[mask == 255] = img[mask == 255]  # Now crop (y, x) = np.where(mask == 255) (topy, topx) = (np.min(y), np.min(x)) (bottomy, bottomx) = (np.max(y), np.max(x)) out = out[topy:bottomy+1, topx:bottomx+1]  # Show the output image cv2.imshow('Output', out) cv2.waitKey(0) cv2.destroyAllWindows() 

The cropping code works such that when we define the mask to extract out the area defined by the contour, we additionally find the smallest horizontal and vertical coordinates which define the top left corner of the contour. We similarly find the largest horizontal and vertical coordinates that define the bottom left corner of the contour. We then use indexing with these coordinates to crop what we actually need. Note that this performs cropping on the masked image - that is the image that removes everything but the information contained within the largest contour.

Note with OpenCV 3.x

It should be noted that the above code assumes you are using OpenCV 2.4.x. Take note that in OpenCV 3.x, the definition of cv2.findContours has changed. Specifically, the output is a three element tuple output where the first image is the source image, while the other two parameters are the same as in OpenCV 2.4.x. Therefore, simply change the cv2.findContours statement in the above code to ignore the first output:

_, contours, _ = cv2.findContours(...) # Your call to find contours 
like image 189
rayryeng Avatar answered Sep 18 '22 08:09

rayryeng


This is a pretty simple way. Mask the image with transparency.

  1. Read the image
  2. Make a grayscale version.
  3. Otsu Threshold
  4. Apply morphology open and close to thresholded image as a mask
  5. Put the mask into the alpha channel of the input
  6. Save the output

Input

enter image description here

Code

import cv2 import numpy as np   # load image as grayscale img = cv2.imread('retina.jpeg')  gray = cv2.cvtColor(img, cv2.COLOR_BGR2GRAY)  # threshold input image using otsu thresholding as mask and refine with morphology ret, mask = cv2.threshold(gray, 0, 255, cv2.THRESH_BINARY+cv2.THRESH_OTSU)  kernel = np.ones((9,9), np.uint8) mask = cv2.morphologyEx(mask, cv2.MORPH_CLOSE, kernel) mask = cv2.morphologyEx(mask, cv2.MORPH_OPEN, kernel)  # put mask into alpha channel of result result = img.copy() result = cv2.cvtColor(result, cv2.COLOR_BGR2BGRA) result[:, :, 3] = mask  # save resulting masked image cv2.imwrite('retina_masked.png', result) 

Output

enter image description here

like image 20
fmw42 Avatar answered Sep 19 '22 08:09

fmw42