python 如何使中置管手检戴手套在手上进行

gr8qqesn  于 2023-05-27  发布在  Python
关注(0)|答案(1)|浏览(150)

我试图使mediapipe的手部检测能够真实的地在戴着蓝色手套的手上工作。但它不能正常工作。如果颜色与肤色相似,则检测仍然可以在手套上工作。所以,我尝试做一些预处理,其中我改变了蓝色像素上发现的帧到裸色。结果是它无法准确地检测到手-有时它可以,但它后来会消失。
请帮我解决这个问题。我在某个地方读到过,这种手部检测可以通过将手套的颜色改变为肤色,同时保留手部的阴影来真实的工作在蓝色手套上。但我不知道该怎么做。如果你能帮助我,我将不胜感激。

  1. def findHands(self, img, draw=True, flipType=True):
  2. imgRGB = cv2.cvtColor(img, cv2.COLOR_BGR2RGB)
  3. blue = [0,0,255]
  4. nude=[225, 190, 160]
  5. #Make mask
  6. Bmask = np.all(imgRGB == blue, axis=-1)
  7. #Replace blue pixel into nude
  8. imgRGB[Bmask] = nude
  9. img_res=imgRGB
  10. #Send frame to mediapipe
  11. self.results = self.hands.process(img_res)
8wigbo56

8wigbo561#

最简单的解决方案是省略cv2.cvtColor(),这将导致图像通道交换,有效地使blue看起来像orange。下面是一个例子(对于mediapipe==0.9.0):

  1. import cv2
  2. import mediapipe as mp
  3. mp_drawing = mp.solutions.drawing_utils
  4. mp_drawing_styles = mp.solutions.drawing_styles
  5. mp_hands = mp.solutions.hands
  6. with mp_hands.Hands(
  7. static_image_mode=True,
  8. max_num_hands=2,
  9. min_detection_confidence=0.5) as hands:
  10. image = cv2.imread("gloves.jpg")
  11. results = hands.process(image)
  12. if results.multi_hand_landmarks:
  13. image_height, image_width, _ = image.shape
  14. for hand_landmarks in results.multi_hand_landmarks:
  15. mp_drawing.draw_landmarks(
  16. image,
  17. hand_landmarks,
  18. mp_hands.HAND_CONNECTIONS,
  19. mp_drawing_styles.get_default_hand_landmarks_style(),
  20. mp_drawing_styles.get_default_hand_connections_style())
  21. else:
  22. print("no hands were found")
  23. cv2.imshow("result", image)
  24. cv2.waitKey(0)

这应该足以让MediaPipe识别手牌:

或者,您也可以使用HSV模式(adapted from here)中的色调偏移对颜色变化进行微调。这将是一个“更好”的解决方案,因为它只改变色调,但不交换通道。
**编辑:**我创建了一个简单的色调调整“类似GUI”的脚本,可以帮助找到合适的色调偏移:

  1. import cv2
  2. import mediapipe as mp
  3. mp_drawing = mp.solutions.drawing_utils
  4. mp_drawing_styles = mp.solutions.drawing_styles
  5. mp_hands = mp.solutions.hands
  6. window_title = 'Recognition result'
  7. trackbar_title = 'Hue offset'
  8. class HueHelper:
  9. def __init__(self):
  10. self.mp_hands = mp_hands.Hands(static_image_mode=True,
  11. max_num_hands=2,
  12. min_detection_confidence=0.5)
  13. self.img_bgr = None
  14. def apply_hue_offset(self,image, hue_offset):# 0 is no change; 0<=huechange<=180
  15. # convert img to hsv
  16. img_hsv = cv2.cvtColor(image, cv2.COLOR_BGR2HSV)
  17. h = img_hsv[:,:,0]
  18. s = img_hsv[:,:,1]
  19. v = img_hsv[:,:,2]
  20. # shift the hue
  21. # cv2 will clip automatically to avoid color wrap-around
  22. img_hsv = cv2.add(h, hue_offset)
  23. # combine new hue with s and v
  24. img_hsv = cv2.merge([img_hsv,s,v])
  25. # convert from HSV to BGR
  26. return cv2.cvtColor(img_hsv, cv2.COLOR_HSV2BGR)
  27. def on_trackbar_change(self, trackbar_hue_offset):
  28. img_bgr_modified = self.recognize(self.img_bgr.copy(), trackbar_hue_offset)
  29. cv2.imshow(window_title, img_bgr_modified)
  30. def recognize(self, img_bgr, hue_offset):
  31. img_bgr = self.apply_hue_offset(img_bgr, hue_offset)
  32. img_rgb = cv2.cvtColor(img_bgr, cv2.COLOR_BGR2RGB)
  33. results = self.mp_hands.process(img_rgb)
  34. if results.multi_hand_landmarks:
  35. for hand_landmarks in results.multi_hand_landmarks:
  36. mp_drawing.draw_landmarks(
  37. img_bgr,
  38. hand_landmarks,
  39. mp_hands.HAND_CONNECTIONS,
  40. mp_drawing_styles.get_default_hand_landmarks_style(),
  41. mp_drawing_styles.get_default_hand_connections_style())
  42. else:
  43. print('no hands were found')
  44. return img_bgr
  45. def run(self, img_path):
  46. self.img_bgr = cv2.imread(img_path)
  47. if self.img_bgr is None: print('Image was not found!')
  48. self.on_trackbar_change(0)
  49. # Hue range is 0-179: https://docs.opencv.org/4.x/df/d9d/tutorial_py_colorspaces.html
  50. cv2.createTrackbar(trackbar_title, window_title, 0, 179, self.on_trackbar_change)
  51. cv2.waitKey(0)
  52. cv2.destroyAllWindows()
  53. if __name__ == '__main__':
  54. h = HueHelper()
  55. h.run('gloves.jpg')

再来一个demo:

展开查看全部

相关问题