python+opencv+mediapipe实现手势检测上下左右(含完整代码)

应用场景:ai换脸,根据左右手势选择图片,上下则表示选中。

版本号:python3.7(一开始是3.6,但是mediapipe最低就是3.7

因为网上检测的都不太准,所以我在判断的时候加入了如果70次里55次检测的是左才返回左,测试完之后效果还可以,蛮准的。判断方法想的头都要秃了。

def gesture3(cap, gesture_buffer_size=70, gesture_threshold=55):
    direction = None
    mp_drawing = mp.solutions.drawing_utils
    mp_hands = mp.solutions.hands

    # 初始化手部检测模型
    hands = mp_hands.Hands(static_image_mode=False, max_num_hands=2, min_detection_confidence=0.5)

    gesture_buffer = deque(maxlen=gesture_buffer_size)  # 用于存储最近的几次手势识别结果

    while cap.isOpened():
        success, image = cap.read()
        if not success:
            print("Ignoring empty camera frame.")
            continue

        # 将图像转换为RGB格式
        image = cv2.cvtColor(cv2.flip(image, 1), cv2.COLOR_BGR2RGB)

        # 执行手部检测
        results = hands.process(image)

        if results.multi_hand_landmarks:
            for hand_landmarks in results.multi_hand_landmarks:
                # 在图像上绘制手部标记
                mp_drawing.draw_landmarks(
                    image, hand_landmarks, mp_hands.HAND_CONNECTIONS,
                    mp_drawing.DrawingSpec(color=(121, 22, 76), thickness=2, circle_radius=4),
                    mp_drawing.DrawingSpec(color=(250, 44, 250), thickness=2, circle_radius=2)
                )

                # 获取手部关键点坐标
                landmarks = hand_landmarks.landmark

                # 获取手掌位置(关键点索引为0)
                palm_landmark = hand_landmarks.landmark[0]
                palm_x = palm_landmark.x
                palm_y = palm_landmark.y

                # 获取除大拇指之外的其他手指的指尖位置
                index_tip = hand_landmarks.landmark[mp.solutions.hands.HandLandmark.INDEX_FINGER_TIP]
                middle_tip = hand_landmarks.landmark[mp.solutions.hands.HandLandmark.MIDDLE_FINGER_TIP]
                ring_tip = hand_landmarks.landmark[mp.solutions.hands.HandLandmark.RING_FINGER_TIP]
                pinky_tip = hand_landmarks.landmark[mp.solutions.hands.HandLandmark.PINKY_TIP]

                # 计算其他手指指尖到手掌位置的距离
                index_distance = math.sqrt((index_tip.x - palm_x) ** 2 + (index_tip.y - palm_y) ** 2)
                middle_distance = math.sqrt((middle_tip.x - palm_x) ** 2 + (middle_tip.y - palm_y) ** 2)
                ring_distance = math.sqrt((ring_tip.x - palm_x) ** 2 + (ring_tip.y - palm_y) ** 2)
                pinky_distance = math.sqrt((pinky_tip.x - palm_x) ** 2 + (pinky_tip.y - palm_y) ** 2)

                # 假设第二个关键点代表食指指尖,第三个关键点代表拇指指尖
                index_finger_tip = landmarks[8]
                # 获取拇指 MCP (第一关节)和 TIP(指尖)的坐标
                thumb_mcp = hand_landmarks.landmark[mp_hands.HandLandmark.THUMB_MCP]
                thumb_tip = hand_landmarks.landmark[mp_hands.HandLandmark.THUMB_TIP]

                # 获取手腕的坐标
                wrist = landmarks[0]

                # 计算食指和拇指指尖之间的距离
                distance = math.sqrt((thumb_tip.x - index_finger_tip.x)**2 + (thumb_tip.y - index_finger_tip.y)**2 + (thumb_tip.z - index_finger_tip.z)**2)
                # 计算手指和手掌在X轴和Y轴上的距离
                x_distance = index_tip.x - palm_x
                y_distance = index_tip.y - palm_y
                if abs(x_distance) > abs(y_distance):
                    # 当手腕在食指的左边,并且食指和拇指指尖之间的距离较大时,判断为向左的手势。 判断手心是否超前,以及食指和拇指指尖的距离来确定手势
                    if index_tip.x < palm_x and middle_tip.x < palm_x and ring_tip.x < palm_x and pinky_tip.x < palm_x and wrist.x > index_finger_tip.x and distance > 0.2 and index_finger_tip.x < thumb_tip.x - 0.05:  # 假设手心超前且食指和拇指指尖的距离大于0.1
                        print('Left')
                        gesture_buffer.append("Left")
                    elif index_tip.x > palm_x and middle_tip.x > palm_x and ring_tip.x > palm_x and pinky_tip.x > palm_x and wrist.x < index_finger_tip.x and distance > 0.1 and index_finger_tip.x > thumb_tip.x + 0.05:  # 假设手心超前且食指和拇指指尖的距离大于0.1
                        print('Right')
                        gesture_buffer.append("Right")
                else:
                    # 当手腕在食指的上方,并且食指和拇指指尖之间的距离较大时,判断为向上的手势
                    if index_tip.y < palm_y and middle_tip.y < palm_y and ring_tip.y < palm_y and pinky_tip.y < palm_y and wrist.y > index_finger_tip.y and distance > 0.2:  # 假设手心超前且食指和拇指指尖的距离大于0.1
                        print('Up')
                        gesture_buffer.append("Up")
                    # 这里我不用下了,如果你们需要直接把注释去掉就可以。当手腕在食指的下方,并且食指和拇指指尖之间的距离较大时,判断为向下的手势
                    # elif index_tip.y > palm_y and middle_tip.y > palm_y and ring_tip.y > palm_y and pinky_tip.y > palm_y and wrist.y < index_finger_tip.y and distance > 0.1:  # 假设手心超前且食指和拇指指尖的距离大于0.1
                    #     print('Down')
                    #     gesture_buffer.append("Down")
               


        # 统计手势识别结果
        gesture_counts = {gesture: gesture_buffer.count(gesture) for gesture in set(gesture_buffer)}

        # 判断最终手势
        if len(gesture_buffer) == gesture_buffer_size and max(gesture_counts.values()) >= gesture_threshold:
            direction = max(gesture_counts, key=gesture_counts.get)

        cv2.imshow('Hand Gestures', cv2.cvtColor(image, cv2.COLOR_RGB2BGR))
        if direction or cv2.waitKey(5) & 0xFF == 27:  # 按ESC键退出
            break

    return direction

实现效果如下:
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

  • 10
    点赞
  • 14
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
### 回答1: PythonOpenCV库和MediaPipe工具包是可以一起使用的,以实现手势识别的功能。 首先,需要在Python中安装OpenCV库和MediaPipe工具包。可以使用pip命令来安装它们: ``` pip install opencv-python pip install mediapipe ``` 安装完成后,就可以开始使用了。 首先,导入必要的库: ```python import cv2 import mediapipe as mp ``` 接下来,创建一个MediaPipe的Hand对象和一个OpenCV的VideoCapture对象,用于读取摄像头输入: ```python mp_hands = mp.solutions.hands hands = mp_hands.Hands() cap = cv2.VideoCapture(0) ``` 然后,使用一个循环来读取摄像头输入并进行手势识别: ```python while True: ret, frame = cap.read() if not ret: break frame_RGB = cv2.cvtColor(frame, cv2.COLOR_BGR2RGB) results = hands.process(frame_RGB) if results.multi_handedness: for hand_landmarks in results.multi_hand_landmarks: # 在这里可以对hand_landmarks进行处理和识别手势的操作 cv2.imshow('Gesture Recognition', frame) if cv2.waitKey(1) == ord('q'): break ``` 在循环中,首先将读取到的帧转换为RGB格式,然后使用Hands对象的process方法对该帧进行手势识别。得到的结果存储在results变量中。 在对每个检测到的手部进行循环处理时,可以使用hand_landmarks来获取该手的关键点坐标。可以根据这些关键点的位置和运动轨迹来实现手势的识别和分析。 最后,通过cv2.imshow方法显示图像,并使用cv2.waitKey方法等待用户操作。当用户按下"q"键时,循环终止,程序退出。 通过以上步骤,就可以使用PythonOpenCV库和MediaPipe工具包实现手势识别的功能了。当然,实际的手势识别算法和操作需要根据具体需求进行进一步的开发和优化。 ### 回答2: Python OpenCVMediaPipe结合使用可以实现手势识别。首先,我们需要安装必要的库和工具,包括Pythonopencv-pythonmediapipe和其他依赖项。 然后,我们可以使用MediaPipe提供的HandTracking模块来检测手部的关键点。它使用机器学习模型来识别手势,并返回手部关键点的坐标。我们可以通过OpenCV的视频捕捉模块读取摄像头的实时图像。 接下来,我们通过应用MediaPipe的HandTracking模块获取手部关键点的坐标,并使用OpenCV将这些坐标绘制到图像上,以便我们可以实时看到手部的位置和动作。 完成这些基本的设置后,我们可以定义特定的手势,例如拇指和食指的指尖接触,作为一个简单的示例。我们可以通过检查特定的关键点之间的距离和角度来识别这种手势。如果关键点之间的距离较小并且角度较小,则我们可以确定手势是拇指和食指的指尖接触。 我们可以使用类似的方法来识别其他手势,比如手掌的张开和闭合,拳头的形成等等。我们可以定义一系列规则和阈值来确定特定手势的识别。 最后,我们可以根据检测到的手势执行特定的操作。例如,当识别到拇指和食指的指尖接触时,我们可以触发相机的快门,实现手势拍照。 总之,PythonOpenCVMediaPipe结合使用可以实现手势识别。我们可以利用MediaPipe的HandTracking模块检测手部关键点,并使用OpenCV实时绘制手势位置。通过定义特定手势的规则,我们可以识别各种手势并执行相应操作。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值