当前位置:   article > 正文

MediaPipe实现手指关键点检测及追踪,人脸识别及追踪_手部界标模型(mediapipe hands)定位到手部关键节点这个网络的流程图

手部界标模型(mediapipe hands)定位到手部关键节点这个网络的流程图

OpenCV 是一个用于计算机视觉应用程序的库。在 OpenCV 的帮助下,我们可以构建大量实时运行更好的应用程序。主要用于图像和视频处理。

可以在此处获取有关 OpenCV 的更多信息 (https://opencv.org/)

除了 OpenCV,我们将使用 MediaPipe 库。

1.MediaPipe简介

MediaPipe是一个主要用于构建音频、视频或任何时间序列数据的框架。在 MediaPipe 框架的帮助下,我们可以为不同的媒体处理功能构建管道。

MediaPipe 的一些主要应用。

  • 多手追踪

  • 人脸检测

  • 对象检测和跟踪

  • Objection:3D 对象检测和跟踪

  • AutoFlip:自动视频裁剪管道等。

MediaPipe 使用单次手掌检测模型,一旦完成,它会对检测到的手部区域中的 21 个 3D 手掌坐标执行精确的关键点定位。

MediaPipe 管道使用多个模型,例如,从完整图像返回定向手边界框的手掌检测模型。裁剪后的图像区域被馈送到由手掌检测器定义的手部标志模型,并返回高保真 3D 手部关键点。

现在让我们实现手部跟踪模型。

安装所需的模块

–> pip install opencv-python

–> pip install mediapipe

注意:这里的python版本尽量在3.8以上,不然会报各种错误!!

首先,让我们检查网络摄像头的工作情况。

  1. import cv2
  2. import mediapipe as mp
  3. import time
  4. cap = cv2.VideoCapture(0)
  5. mpHands = mp.solutions.hands
  6. hands = mpHands.Hands(static_image_mode=False,
  7.                      max_num_hands=2,
  8.                      min_detection_confidence=0.5,
  9.                      min_tracking_confidence=0.5)
  10. mpDraw = mp.solutions.drawing_utils
  11. pTime = 0
  12. cTime = 0
  13. while True:
  14.    success, img = cap.read()
  15.    img = cv2.flip(img, 1)
  16.    imgRGB = cv2.cvtColor(img, cv2.COLOR_BGR2RGB)
  17.    results = hands.process(imgRGB)
  18.    #print(results.multi_hand_landmarks)
  19.    if results.multi_hand_landmarks:
  20.        for handLms in results.multi_hand_landmarks:
  21.            for id, lm in enumerate(handLms.landmark):
  22.                #print(id,lm)
  23.                h, w, c = img.shape
  24.                cx, cy = int(lm.x *w), int(lm.y*h)
  25.                #if id ==0:
  26.                cv2.circle(img, (cx,cy), 3, (255,0,255), cv2.FILLED)
  27.            mpDraw.draw_landmarks(img, handLms, mpHands.HAND_CONNECTIONS)
  28.    cTime = time.time()
  29.    fps = 1/(cTime-pTime)
  30.    pTime = cTime
  31.    cv2.putText(img,str(int(fps)), (10,70), cv2.FONT_HERSHEY_PLAIN, 3, (255,0,255), 3)
  32.    cv2.imshow("Image", img)
  33.    cv2.waitKey(1)

结果见图示:

image-20211125121105554

2.mediapipe实现手掌拖拽屏幕中的方块

效果如示:

image-20211125122000091

image-20211125122027286

image-20211125122052025

源码如下:

  1. """
  2. 功能:手势虚拟拖拽
  3. 1、使用OpenCV读取摄像头视频流;
  4. 2、识别手掌关键点像素坐标;
  5. 3、根据食指和中指指尖的坐标,利用勾股定理计算距离,当距离较小且都落在矩形内,则触发拖拽(矩形变色);
  6. 4、矩形跟着手指动;
  7. 5、两指放开,则矩形停止移动
  8. """
  9. # 导入OpenCV
  10. import cv2
  11. # 导入mediapipe
  12. import mediapipe as mp
  13. # 导入其他依赖包
  14. import time
  15. import math
  16. # 方块管理类
  17. class SquareManager:
  18.    def __init__(self, rect_width):
  19.        
  20.        
  21.        # 方框长度
  22.        self.rect_width = rect_width
  23.        
  24.        # 方块list
  25.        self.square_count = 0
  26.        self.rect_left_x_list = []
  27.        self.rect_left_y_list = []
  28.        self.alpha_list = []
  29.        # 中指与矩形左上角点的距离
  30.        self.L1 = 0
  31.        self.L2 = 0
  32.        # 激活移动模式
  33.        self.drag_active = False
  34.        # 激活的方块ID
  35.        self.active_index = -1
  36.        
  37.    
  38.    # 创建一个方块,但是没有显示
  39.    def create(self,rect_left_x,rect_left_y,alpha=0.4):
  40.        self.rect_left_x_list.append(rect_left_x)
  41.        self.rect_left_y_list.append(rect_left_y)
  42.        self.alpha_list.append(alpha)
  43.        self.square_count+=1
  44.        
  45.    # 更新位置
  46.    def display(self,class_obj):
  47.        for i in range(0,self.square_count):
  48.            x= self.rect_left_x_list[i]
  49.            y= self.rect_left_y_list[i]
  50.            alpha = self.alpha_list[i]
  51.            overlay = class_obj.image.copy()
  52.            if(i == self.active_index):
  53.                cv2.rectangle(overlay,(x,y),(x+self.rect_width,y+self.rect_width),(255, 0, 255),-1)
  54.            else:
  55.                cv2.rectangle(overlay,(x,y),(x+self.rect_width,y+self.rect_width),(255, 0, 0),-1)
  56.            
  57.            # Following line overlays transparent rectangle over the self.image
  58.            class_obj.image = cv2.addWeighted(overlay, alpha, class_obj.image, 1 - alpha, 0)
  59.            
  60.    # 判断落在哪个方块上,返回方块的ID
  61.    def checkOverlay(self,check_x,check_y):
  62.        for i in range(0,self.square_count):
  63.            x= self.rect_left_x_list[i]
  64.            y= self.rect_left_y_list[i]
  65.            if (x <  check_x < (x+self.rect_width) ) and ( y < check_y < (y+self.rect_width)):
  66.                
  67.                # 保存被激活的方块ID
  68.                self.active_index = i
  69.                return i
  70.        
  71.        return -1
  72.    # 计算与指尖的距离
  73.    def setLen(self,check_x,check_y):
  74.        # 计算距离
  75.        self.L1 = check_x - self.rect_left_x_list[self.active_index]
  76.        self.L2 = check_y - self.rect_left_y_list[self.active_index]
  77.    # 更新方块    
  78.    def updateSquare(self,new_x,new_y):
  79.        # print(self.rect_left_x_list[self.active_index])
  80.        self.rect_left_x_list[self.active_index] = new_x - self.L1
  81.        self.rect_left_y_list[self.active_index] = new_y - self.L2
  82. # 识别控制类
  83. class HandControlVolume:
  84.    def __init__(self):
  85.        # 初始化medialpipe
  86.        self.mp_drawing = mp.solutions.drawing_utils
  87.        self.mp_drawing_styles = mp.solutions.drawing_styles
  88.        self.mp_hands = mp.solutions.hands
  89.        
  90.        # 中指与矩形左上角点的距离
  91.        self.L1 = 0
  92.        self.L2 = 0
  93.        # image实例,以便另一个类调用
  94.        self.image=None
  95.        
  96.    # 主函数
  97.    def recognize(self):
  98.        # 计算刷新率
  99.        fpsTime = time.time()
  100.        # OpenCV读取视频流
  101.        cap = cv2.VideoCapture(0)
  102.        # 视频分辨率
  103.        resize_w = 1280
  104.        resize_h = 720
  105.        # 画面显示初始化参数
  106.        rect_percent_text = 0
  107.        # 初始化方块管理器
  108.        squareManager = SquareManager(150)
  109.        # 创建多个方块
  110.        for i in range(0, 7):
  111.            squareManager.create(180*i+20, 180, 0.3) ##原来是0.6
  112.        with self.mp_hands.Hands(min_detection_confidence=0.7,
  113.                                 min_tracking_confidence=0.5,
  114.                                 max_num_hands=2) as hands:
  115.            while cap.isOpened():
  116.                # 初始化矩形
  117.                success, self.image = cap.read()
  118.                self.image = cv2.resize(self.image, (resize_w, resize_h))
  119.                if not success:
  120.                    print("空帧.")
  121.                    continue
  122.                # 提高性能
  123.                self.image.flags.writeable = False
  124.                # 转为RGB
  125.                self.image = cv2.cvtColor(self.image, cv2.COLOR_BGR2RGB)
  126.                # 镜像
  127.                self.image = cv2.flip(self.image, 1)
  128.                # mediapipe模型处理
  129.                results = hands.process(self.image)
  130.                self.image.flags.writeable = True
  131.                self.image = cv2.cvtColor(self.image, cv2.COLOR_RGB2BGR)
  132.                # 判断是否有手掌
  133.                if results.multi_hand_landmarks:
  134.                    # 遍历每个手掌
  135.                    for hand_landmarks in results.multi_hand_landmarks:
  136.                        # 在画面标注手指
  137.                        self.mp_drawing.draw_landmarks(
  138.                            self.image,
  139.                            hand_landmarks,
  140.                            self.mp_hands.HAND_CONNECTIONS,
  141.                            self.mp_drawing_styles.get_default_hand_landmarks_style(),
  142.                            self.mp_drawing_styles.get_default_hand_connections_style())
  143.                        # 解析手指,存入各个手指坐标
  144.                        landmark_list = []
  145.                        # 用来存储手掌范围的矩形坐标
  146.                        paw_x_list = []
  147.                        paw_y_list = []
  148.                        for landmark_id, finger_axis in enumerate(
  149.                                hand_landmarks.landmark):
  150.                            landmark_list.append([
  151.                                landmark_id, finger_axis.x, finger_axis.y,
  152.                                finger_axis.z
  153.                           ])
  154.                            paw_x_list.append(finger_axis.x)
  155.                            paw_y_list.append(finger_axis.y)
  156.                        if landmark_list:
  157.                            # 比例缩放到像素
  158.                            ratio_x_to_pixel = lambda x: math.ceil(x * resize_w)
  159.                            ratio_y_to_pixel = lambda y: math.ceil(y * resize_h)
  160.                            
  161.                            # 设计手掌左上角、右下角坐标
  162.                            paw_left_top_x,paw_right_bottom_x = map(ratio_x_to_pixel,[min(paw_x_list),max(paw_x_list)])
  163.                            paw_left_top_y,paw_right_bottom_y = map(ratio_y_to_pixel,[min(paw_y_list),max(paw_y_list)])
  164.                            # 给手掌画框框
  165.                            cv2.rectangle(self.image,(paw_left_top_x-30,paw_left_top_y-30),(paw_right_bottom_x+30,paw_right_bottom_y+30),(0, 255,0),2)
  166.                            # 获取中指指尖坐标
  167.                            middle_finger_tip = landmark_list[12]
  168.                            middle_finger_tip_x =ratio_x_to_pixel(middle_finger_tip[1])
  169.                            middle_finger_tip_y =ratio_y_to_pixel(middle_finger_tip[2])
  170.                            # 获取食指指尖坐标
  171.                            index_finger_tip = landmark_list[8]
  172.                            index_finger_tip_x =ratio_x_to_pixel(index_finger_tip[1])
  173.                            index_finger_tip_y =ratio_y_to_pixel(index_finger_tip[2])
  174.                            # 中间点
  175.                            between_finger_tip = (middle_finger_tip_x+index_finger_tip_x)//2, (middle_finger_tip_y+index_finger_tip_y)//2
  176.                            # print(middle_finger_tip_x)
  177.                            thumb_finger_point = (middle_finger_tip_x,middle_finger_tip_y)
  178.                            index_finger_point = (index_finger_tip_x,index_finger_tip_y)
  179.                            # 画指尖2点
  180.                            circle_func = lambda point: cv2.circle(self.image,point,10,(255,0,255),-1)
  181.                            self.image = circle_func(thumb_finger_point)
  182.                            self.image = circle_func(index_finger_point)
  183.                            self.image = circle_func(between_finger_tip)
  184.                            # 画2点连线
  185.                            self.image = cv2.line(self.image,thumb_finger_point,index_finger_point,(255,0,255),5)
  186.                            # 勾股定理计算长度
  187.                            line_len = math.hypot((index_finger_tip_x-middle_finger_tip_x),(index_finger_tip_y-middle_finger_tip_y))
  188.                            # 将指尖距离映射到文字
  189.                            rect_percent_text = math.ceil(line_len)
  190.                            # 激活模式,需要让矩形跟随移动
  191.                            if squareManager.drag_active:
  192.                                # 更新方块
  193.                                squareManager.updateSquare(between_finger_tip[0],between_finger_tip[1])
  194.                                if(line_len>100):
  195.                                    # 取消激活
  196.                                    squareManager.drag_active =False
  197.                                    squareManager.active_index = -1
  198.                            elif (line_len<100) and (squareManager.checkOverlay(between_finger_tip[0],between_finger_tip[1]) != -1 )and( squareManager.drag_active == False):
  199.                                    # 激活
  200.                                    squareManager.drag_active =True
  201.                                    # 计算距离
  202.                                    squareManager.setLen(between_finger_tip[0],between_finger_tip[1])
  203.                # 显示方块,传入本实例,主要为了半透明的处理
  204.                squareManager.display(self)
  205.                
  206.                # 显示距离
  207.                cv2.putText(self.image, "Distance:"+str(rect_percent_text), (10, 120),cv2.FONT_HERSHEY_PLAIN, 3, (255, 0, 0), 3)
  208.                # 显示当前激活
  209.                cv2.putText(self.image, "Active:"+( "None" if squareManager.active_index == -1 else str(squareManager.active_index)), (10, 170),cv2.FONT_HERSHEY_PLAIN, 3, (255, 0, 0), 3)
  210.                # 显示刷新率FPS
  211.                cTime = time.time()
  212.                fps_text = 1/(cTime-fpsTime)
  213.                fpsTime = cTime
  214.                cv2.putText(self.image, "FPS: " + str(int(fps_text)), (10, 70),
  215.                            cv2.FONT_HERSHEY_PLAIN, 3, (255, 0, 0), 3)
  216.                # 显示画面
  217.                # self.image = cv2.resize(self.image, (resize_w//2, resize_h//2))
  218.                cv2.imshow('virtual drag and drop', self.image)
  219.                
  220.                if cv2.waitKey(5) & 0xFF == 27:
  221.                    break
  222.            cap.release()
  223. # 开始程序
  224. control = HandControlVolume()
  225. control.recognize()

3.mediapipe实现手势控制音量大小

效果如示:

image-20211125122351771

image-20211125122411531

源码如下:

  1. """
  2. 功能:手势操作电脑音量
  3. 1、使用OpenCV读取摄像头视频流;
  4. 2、识别手掌关键点像素坐标;
  5. 3、根据拇指和食指指尖的坐标,利用勾股定理计算距离;
  6. 4、将距离等比例转为音量大小,控制电脑音量
  7. """
  8. # 导入OpenCV
  9. import cv2
  10. # 导入mediapipe
  11. import mediapipe as mp
  12. # 导入电脑音量控制模块
  13. from ctypes import cast, POINTER
  14. from comtypes import CLSCTX_ALL
  15. from pycaw.pycaw import AudioUtilities, IAudioEndpointVolume
  16. # 导入其他依赖包
  17. import time
  18. import math
  19. import numpy as np
  20. class HandControlVolume:
  21.    def __init__(self):
  22.        # 初始化medialpipe
  23.        self.mp_drawing = mp.solutions.drawing_utils
  24.        self.mp_drawing_styles = mp.solutions.drawing_styles
  25.        self.mp_hands = mp.solutions.hands
  26.        # 获取电脑音量范围
  27.        devices = AudioUtilities.GetSpeakers()
  28.        interface = devices.Activate(
  29.            IAudioEndpointVolume._iid_, CLSCTX_ALL, None)
  30.        self.volume = cast(interface, POINTER(IAudioEndpointVolume))
  31.        self.volume_range = self.volume.GetVolumeRange()
  32.    # 主函数
  33.    def recognize(self):
  34.        # 计算刷新率
  35.        fpsTime = time.time()
  36.        # OpenCV读取视频流
  37.        cap = cv2.VideoCapture(0)
  38.        # 视频分辨率
  39.        resize_w = 640
  40.        resize_h = 480
  41.        # 画面显示初始化参数
  42.        rect_height = 0
  43.        rect_percent_text = 0
  44.        with self.mp_hands.Hands(min_detection_confidence=0.7,
  45.                                 min_tracking_confidence=0.5,
  46.                                 max_num_hands=2) as hands:
  47.            while cap.isOpened():
  48.                success, image = cap.read()
  49.                image = cv2.resize(image, (resize_w, resize_h))
  50.                if not success:
  51.                    print("空帧.")
  52.                    continue
  53.                # 提高性能
  54.                image.flags.writeable = False
  55.                # 转为RGB
  56.                image = cv2.cvtColor(image, cv2.COLOR_BGR2RGB)
  57.                # 镜像
  58.                image = cv2.flip(image, 1)
  59.                # mediapipe模型处理
  60.                results = hands.process(image)
  61.                image.flags.writeable = True
  62.                image = cv2.cvtColor(image, cv2.COLOR_RGB2BGR)
  63.                # 判断是否有手掌
  64.                if results.multi_hand_landmarks:
  65.                    # 遍历每个手掌
  66.                    for hand_landmarks in results.multi_hand_landmarks:
  67.                        # 在画面标注手指
  68.                        self.mp_drawing.draw_landmarks(
  69.                            image,
  70.                            hand_landmarks,
  71.                            self.mp_hands.HAND_CONNECTIONS,
  72.                            self.mp_drawing_styles.get_default_hand_landmarks_style(),
  73.                            self.mp_drawing_styles.get_default_hand_connections_style())
  74.                        # 解析手指,存入各个手指坐标
  75.                        landmark_list = []
  76.                        for landmark_id, finger_axis in enumerate(
  77.                                hand_landmarks.landmark):
  78.                            landmark_list.append([
  79.                                landmark_id, finger_axis.x, finger_axis.y,
  80.                                finger_axis.z
  81.                           ])
  82.                        if landmark_list:
  83.                            # 获取大拇指指尖坐标
  84.                            thumb_finger_tip = landmark_list[4]
  85.                            thumb_finger_tip_x = math.ceil(thumb_finger_tip[1] * resize_w)
  86.                            thumb_finger_tip_y = math.ceil(thumb_finger_tip[2] * resize_h)
  87.                            # 获取食指指尖坐标
  88.                            index_finger_tip = landmark_list[8]
  89.                            index_finger_tip_x = math.ceil(index_finger_tip[1] * resize_w)
  90.                            index_finger_tip_y = math.ceil(index_finger_tip[2] * resize_h)
  91.                            # 中间点
  92.                            finger_middle_point = (thumb_finger_tip_x+index_finger_tip_x)//2, (thumb_finger_tip_y+index_finger_tip_y)//2
  93.                            # print(thumb_finger_tip_x)
  94.                            thumb_finger_point = (thumb_finger_tip_x,thumb_finger_tip_y)
  95.                            index_finger_point = (index_finger_tip_x,index_finger_tip_y)
  96.                            # 画指尖2点
  97.                            image = cv2.circle(image,thumb_finger_point,10,(255,0,255),-1)
  98.                            image = cv2.circle(image,index_finger_point,10,(255,0,255),-1)
  99.                            image = cv2.circle(image,finger_middle_point,10,(255,0,255),-1)
  100.                            # 画2点连线
  101.                            image = cv2.line(image,thumb_finger_point,index_finger_point,(255,0,255),5)
  102.                            # 勾股定理计算长度
  103.                            line_len = math.hypot((index_finger_tip_x-thumb_finger_tip_x),(index_finger_tip_y-thumb_finger_tip_y))
  104.                            # 获取电脑最大最小音量
  105.                            min_volume = self.volume_range[0]
  106.                            max_volume = self.volume_range[1]
  107.                            # 将指尖长度映射到音量上
  108.                            vol = np.interp(line_len,[50,300],[min_volume,max_volume])
  109.                            # 将指尖长度映射到矩形显示上
  110.                            rect_height = np.interp(line_len,[50,300],[0,200])
  111.                            rect_percent_text = np.interp(line_len,[50,300],[0,100])
  112.                            # 设置电脑音量
  113.                            self.volume.SetMasterVolumeLevel(vol, None)
  114.                # 显示矩形
  115.                cv2.putText(image, str(math.ceil(rect_percent_text))+"%", (10, 350),
  116.                            cv2.FONT_HERSHEY_PLAIN, 3, (255, 0, 0), 3)
  117.                image = cv2.rectangle(image,(30,100),(70,300),(255, 0, 0),3)
  118.                image = cv2.rectangle(image,(30,math.ceil(300-rect_height)),(70,300),(255, 0, 0),-1)
  119.                # 显示刷新率FPS
  120.                cTime = time.time()
  121.                fps_text = 1/(cTime-fpsTime)
  122.                fpsTime = cTime
  123.                cv2.putText(image, "FPS: " + str(int(fps_text)), (10, 70),
  124.                            cv2.FONT_HERSHEY_PLAIN, 3, (255, 0, 0), 3)
  125.                # 显示画面
  126.                cv2.imshow('MediaPipe Hands', image)
  127.                if cv2.waitKey(5) & 0xFF == 27:
  128.                    break
  129.            cap.release()
  130. # 开始程序
  131. control = HandControlVolume()
  132. control.recognize()
​
​
​

reference:

使用 Python+OpenCV 构建手部跟踪系统 - 知乎 (zhihu.com)

源码来源:恩培大佬

enpeizhao/CVprojects: computer vision projects | 计算机视觉等好玩的AI项目 (github.com)

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/盐析白兔/article/detail/906106
推荐阅读
相关标签
  

闽ICP备14008679号