Choosing the correct upper and lower HSV boundaries for color detection with`cv::inRange` (OpenCV)

匿名 (未验证) 提交于 2019-12-03 02:15:02

问题:

I have an image of a coffee can with an orange lid position of which I want to find. Here is it

.

gcolor2 utility shows HSV at the center of the lid to be (22, 59, 100). The question is how to choose the limits of the color then? I tried min = (18, 40, 90) and max = (27, 255, 255), but have got unexpected

Here is the Python code:

import cv  in_image = 'kaffee.png' out_image = 'kaffee_out.png' out_image_thr = 'kaffee_thr.png'  ORANGE_MIN = cv.Scalar(18, 40, 90) ORANGE_MAX = cv.Scalar(27, 255, 255) COLOR_MIN = ORANGE_MIN COLOR_MAX = ORANGE_MAX  def test1():     frame = cv.LoadImage(in_image)     frameHSV = cv.CreateImage(cv.GetSize(frame), 8, 3)     cv.CvtColor(frame, frameHSV, cv.CV_RGB2HSV)     frame_threshed = cv.CreateImage(cv.GetSize(frameHSV), 8, 1)     cv.InRangeS(frameHSV, COLOR_MIN, COLOR_MAX, frame_threshed)     cv.SaveImage(out_image_thr, frame_threshed)  if __name__ == '__main__':     test1() 

回答1:

Problem 1 : Different applications use different scales for HSV. For example gimp uses H = 0-360, S = 0-100 and V = 0-100. But OpenCV uses H: 0 - 180, S: 0 - 255, V: 0 - 255. Here i got a hue value of 22 in gimp. So I took half of it, 11, and defined range for that. ie (5,50,50) - (15,255,255).

Problem 2: And also, OpenCV uses BGR format, not RGB. So change your code which converts RGB to HSV as follows:

cv.CvtColor(frame, frameHSV, cv.CV_BGR2HSV) 

Now run it. I got an output as follows:

Hope that is what you wanted. There are some false detections, but they are small, so you can choose biggest contour which is your lid.

EDIT:

As Karl Philip told in his comment, it would be good to add new code. But there is change of only a single line. So, I would like to add the same code implemented in new cv2 module, so users can compare the easiness and flexibility of new cv2 module.

import cv2 import numpy as np  img = cv2.imread('sof.jpg')  ORANGE_MIN = np.array([5, 50, 50],np.uint8) ORANGE_MAX = np.array([15, 255, 255],np.uint8)  hsv_img = cv2.cvtColor(img,cv2.COLOR_BGR2HSV)  frame_threshed = cv2.inRange(hsv_img, ORANGE_MIN, ORANGE_MAX) cv2.imwrite('output2.jpg', frame_threshed) 

It gives the same result as above. But code is much more simpler.



回答2:

I Created this simple program to get HSV Codes in realtime

import cv2 import numpy as np   cap = cv2.VideoCapture(0)  def nothing(x):     pass # Creating a window for later use cv2.namedWindow('result')  # Starting with 100's to prevent error while masking h,s,v = 100,100,100  # Creating track bar cv2.createTrackbar('h', 'result',0,179,nothing) cv2.createTrackbar('s', 'result',0,255,nothing) cv2.createTrackbar('v', 'result',0,255,nothing)  while(1):      _, frame = cap.read()      #converting to HSV     hsv = cv2.cvtColor(frame,cv2.COLOR_BGR2HSV)      # get info from track bar and appy to result     h = cv2.getTrackbarPos('h','result')     s = cv2.getTrackbarPos('s','result')     v = cv2.getTrackbarPos('v','result')      # Normal masking algorithm     lower_blue = np.array([h,s,v])     upper_blue = np.array([180,255,255])      mask = cv2.inRange(hsv,lower_blue, upper_blue)      result = cv2.bitwise_and(frame,frame,mask = mask)      cv2.imshow('result',result)      k = cv2.waitKey(5) & 0xFF     if k == 27:         break  cap.release()  cv2.destroyAllWindows() 


回答3:

Ok, find color in HSV space is an old but common question. I made a hsv-colormap to fast look up special color. Here it is:

The x-axis represents Hue in [0,180), the y-axis1 represents Saturation in [0,255], the y-axis2 represents S = 255, while keep V = 255.

To find a color, usually just look up for the range of H and S, and set v in range(20, 255).

To find the orange color, we look up for the map, and find the best range: H :[10, 25], S: [100, 255], and V: [20, 255]. So the mask is cv2.inRange(hsv,(10, 100, 20), (25, 255, 255) )

Then we use the found range to look for the orange color, this is the result:


The method is simple but common to use:

#!/usr/bin/python3 # 2018.01.21 20:46:41 CST import cv2  img = cv2.imread("test.jpg") hsv = cv2.cvtColor(img, cv2.COLOR_BGR2HSV) mask = cv2.inRange(hsv,(10, 100, 20), (25, 255, 255) ) cv2.imshow("orange", mask);cv2.waitKey();cv2.destroyAllWindows() 

Similar answers:

  1. How to define a threshold value to detect only green colour objects in an image :Opencv

  2. Choosing correct HSV values for OpenCV thresholding with InRangeS



回答4:

OpenCV HSV range is: H: 0 to 179 S: 0 to 255 V: 0 to 255

On Gimp (or other photo manipulation sw) Hue range from 0 to 360, since opencv put color info in a single byte, the maximum number value in a single byte is 255 therefore openCV Hue values are equivalent to Hue values from gimp divided by 2.

I found when trying to do object detection based on HSV color space that a range of 5 (opencv range) was sufficient to filter out a specific color. I would advise you to use an HSV color palate to figure out the range that works best for your application.



标签
易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!