3dmax的python通过普通的摄像头动捕表情

1、安装python
进入cdm,打python要能显示版本号
>>>(进入python提示符模式)
import sys
sys.path显示python的安装路径,
进入到python.exe的路径
在python目录中安装(ctrl+z退出python交互模式)
2、pip install mediapipe    mediapipe 官网 

用阿里云镜像     mediapipe的github托管

pip install mediapipe -i https://mirrors.aliyun.com/pypi/simple/

3、pip install opencv -python

pip install opencv-python -i https://mirrors.aliyun.com/pypi/simple/

opencv官方网站 (有关摄像头动捕关键点说明)

进入到python.exe的路径,新建 FaceCapture.py

import cv2
import mediapipe as mp
import socket
import json
import time# 初始化 mediapipe
mp_face_mesh = mp.solutions.face_mesh
face_mesh = mp_face_mesh.FaceMesh(static_image_mode=False,max_num_faces=1,refine_landmarks=True,min_detection_confidence=0.8,  # 增加检测置信度min_tracking_confidence=0.8  # 增加检测置信度
)# 切换摄像头请切换设备索引
device_index = 0
cap = cv2.VideoCapture(device_index, cv2.CAP_DSHOW)
sock = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
target_addr = ('127.0.0.1', 5005)total_duration = 15  # 总运行时间,单位:秒
start_time = time.time()
frame_count = 0
prev_time = 0# 定义需要的关键点索引
# 眉毛
left_eyebrow = [70, 74, 78, 336]
right_eyebrow = [300, 304, 308, 105]
# 眼睛
left_eye = [36, 70, 105, 133]
right_eye = [267, 301, 336, 362]
# 嘴巴
mouth = [61, 146, 178, 291]
# 鼻翼
left_nostril = [129]
right_nostril = [358]
# 下巴
chin = [152]
# 眼角
left_inner_corner = [133]
left_outer_corner = [33]
right_inner_corner = [362]
right_outer_corner = [263]
# 嘴角
left_mouth_corner = [61]
right_mouth_corner = [291]# 合并所有需要的关键点索引
selected_indices = left_eyebrow + right_eyebrow + left_eye + right_eye + mouth + left_nostril + right_nostril + chin + left_inner_corner + left_outer_corner + right_inner_corner + right_outer_corner + left_mouth_corner + right_mouth_cornerif not cap.isOpened():print("无法打开摄像头")
else:try:while True:elapsed = time.time() - start_time  # 提前计算运行时间remaining = total_duration - elapsedtry:ret, frame = cap.read()frame_count += 1print(f"正在读取第 {frame_count} 帧,状态: {ret},程序将在 {remaining:.2f} 秒后自动关闭")if remaining <= 0:print("运行时间已到,自动退出")breakexcept Exception as e:print("读取视频帧异常:", e)continueif not ret:print("无法获取摄像头数据,帧号:", frame_count)breakcurrent_time = time.time()fps = 1 / (current_time - prev_time)prev_time = current_timergb_frame = cv2.cvtColor(frame, cv2.COLOR_BGR2RGB)result = face_mesh.process(rgb_frame)data_to_send = {}if result.multi_face_landmarks:landmarks = result.multi_face_landmarks[0].landmarkkeypoints = [{"id": i, "x": landmarks[i].x, "y": landmarks[i].y, "z": landmarks[i].z} for i in selected_indices]data_to_send['keypoints'] = keypoints# 绘制跟踪点height, width, _ = frame.shapefor keypoint in keypoints:x = int(keypoint["x"] * width)y = int(keypoint["y"] * height)cv2.circle(frame, (x, y), 2, (0, 255, 0), -1)else:data_to_send['keypoints'] = []data_to_send['fps'] = int(fps)json_data = json.dumps(data_to_send)# 实时驱动sock.sendto(json_data.encode('utf-8'), target_addr)# 离线驱动# with open(f"frames/frame_{frame_count:04d}.json", "w") as f:#    json.dump(data_to_send, f)# 文字提示,添加 fps 信息countdown_text = f"remainTime: {remaining:.1f} second, FPS: {int(fps)}"cv2.putText(frame, countdown_text, (10, 30), cv2.FONT_HERSHEY_SIMPLEX, 1,(0, 255, 0), 2, cv2.LINE_AA)# 进度条提示bar_x, bar_y = 10, 60bar_width = 300bar_height = 20progress = remaining / total_durationcv2.rectangle(frame, (bar_x, bar_y), (bar_x + bar_width, bar_y + bar_height), (180, 180, 180), 2)cv2.rectangle(frame, (bar_x, bar_y), (bar_x + int(bar_width * progress), bar_y + bar_height), (0, 255, 0), -1)cv2.imshow('Face Capture', frame)if cv2.waitKey(1) & 0xFF == ord('q'):print("用户手动退出")break# 自动退出机制(30 秒)if time.time() - start_time > total_duration:message = "已运行 {} 秒,自动退出".format(total_duration)print(message)breakfinally:# 发送关闭信息close_message = {'type': 'close'}json_close_message = json.dumps(close_message)sock.sendto(json_close_message.encode('utf-8'), target_addr)cap.release()sock.close()cv2.destroyAllWindows()

python安装目录,进去后cmd命令提示符,然后 python   FaceCapture.py

 遇到 ImportError: DLL load failed while importing _framework_bindings: 动态链接库(DLL)初始化例程失败。

最新受支持的 Visual C++ 可再发行程序包下载 | Microsoft Learn

进入到python.exe的路径,新建 StartCam.bat 文件

@echo off
python FaceCapture.py
pause

进入到python.exe的路径,新建 启动摄像头.vbs 文件,这样就可以隐藏掉cmd命令行的界面只显示摄像头界面

Set objShell = CreateObject("WScript.Shell")
Dim strPath, strBatFile
strPath = CreateObject("Scripting.FileSystemObject").GetParentFolderName(WScript.ScriptFullName)
strBatFile = strPath & "\StartCam.bat"
objShell.Run strBatFile, 0, False

新建一个max的ms文件GetUDP.ms ,这个可以创建十个dmmy物体,用来接受摄像头的表情捕捉数据,

--引入必要的 .NET类
dotNet.loadAssembly "System.Threading"
dotNet.loadAssembly "System.Net"
dotNet.loadAssembly "System.Net.Sockets"
dotNet.loadAssembly "System.Text"
dotNet.loadAssembly "Newtonsoft.Json"
-- dotNet.loadAssembly @"C:\Windows\Microsoft.NET\Framework\v4.0.30319\System.Threading.dll"--清理之前的资源
try (if isProperty udpClient "Close" do (udpClient.Close())) catch ()
try (if isProperty thread "Abort" do (thread.Abort())) catch ()--初始化UDP接收器
global localEndpoint = dotNetObject "System.Net.IPEndPoint" (dotNetClass "System.Net.IPAddress").Any  5005
global udpClient = dotNetObject "System.Net.Sockets.UdpClient"
udpClient.ExclusiveAddressUse = false 
udpClient.Client.SetSocketOption (dotNetClass "System.Net.Sockets.SocketOptionLevel").Socket (dotNetClass "System.Net.Sockets.SocketOptionName").ReuseAddress true
udpClient.Client.Bind localEndpoint
--定义定时器	
global delayTimer = dotnetobject "Windows.Forms.Timer"	
-- 定义后台线程
global BgThread = dotNetObject "System.ComponentModel.BackgroundWorker"
BgThread.WorkerSupportsCancellation = true--重置场景
resetMaxFile #noPrompt-- 创建 10 个 Dummy 物体并命名为 Dummy001, Dummy002, ..., Dummy010
global dummyArray = for i = 1 to 10 collect (local mydummy = dummy()  -- 创建 Dummy 物体mydummy.name = "Dummy" + formattedPrint i format:"03d"  -- 命名为 Dummy001, Dummy002, ...mydummy.position = [i * 50, 0, 0]  -- 设置一个简单的初始位置,您可以根据需要调整mydummy  -- 返回创建的 Dummy 物体
)-- 接收 UDP 数据并解析为 JSON 结构体的函数
fn receiveAndParseJson =
(local remoteEndpoint = dotNetObject "System.Net.IPEndPoint" (dotNetClass "System.Net.IPAddress").Any 0local receiveBytes = udpClient.Receive remoteEndpointlocal jsonString = (dotNetClass "System.Text.Encoding").UTF8.GetString receiveBytestry (-- 引用 System.Web.Extensions(包含 JSON 解析器)dotNet.loadAssembly "System.Web.Extensions"-- 创建内建 JSON 解析器jsonParser = dotNetObject "System.Web.Script.Serialization.JavaScriptSerializer"-- 反序列化 JSON 字符串为 MaxScript 可用的结构体(Hashtable/Array等)jsonStruct = jsonParser.DeserializeObject jsonStringreturn jsonStruct)catch (print("解析 JSON 时出错:" + getCurrentException())return undefined)
)-- 处理关键点数据并更新 Dummy 位置的函数
fn processKeypointsData keypoints =
(if keypoints != undefined do(for id = 0 to 9 do (for pt in keypoints do (if pt.Item["id"] == id then (local x = pt.Item["x"] * 100local y = pt.Item["y"] * 100local z = pt.Item["z"] * 100--dotNetClass "System.Windows.Forms.Control".Invoke (dotNetDelegate updateDummyPosition id x y z)local mydummy = dummyArray[id+1]if isValidNode mydummy do (mydummy.position = [pt.Item["x"]*100, pt.Item["y"]*100, pt.Item["z"]*100] )))print("Dummies 位置更新完成!")))
)-- 定时器触发时执行的函数
fn timerCallback sender e =
(if not BgThread.CancellationPending do(local jsonStruct = receiveAndParseJson()format "json结构体为:% \n" jsonStructif jsonStruct != undefined do(-- 获取帧率local fps = jsonStruct.Item["fps"]if fps != undefined do(-- 根据帧率计算定时器间隔(毫秒)local interval = 1000 / fpsdelayTimer.Interval = interval)-- 处理关键点数据local keypoints = jsonStruct.Item["keypoints"]processKeypointsData keypoints-- 检查是否接收到关闭信息,则关闭后台线程if jsonStruct.Item["type"] == "close" do(BgThread.CancelAsync()return true))sleep 0.1)
)--主执行命令
if (timerCallback sender e )then(delayTimer.enabled = true-- 添加定时器事件处理程序dotnet.AddEventHandler delayTimer "Tick" (timerCallback sender e)-- 添加后台线程事件处理程序dotnet.AddEventHandler BgThread "DoWork" (fn sender e = timerCallback sender e)-- 监听键盘事件,设置取消标志if keyboard.escPressed do (BgThread.CancelAsync()print "已发送取消请求")
)
else(	delayTimer.enabled = falseudpClient.Close()BgThread.CancelAsync()
)

启动启动摄像头.vbs,出现摄像头窗口,和倒计时进度条,30秒后自动关闭,你可以设定的不关闭。

启动GetUDP.ms,接收动捕数据,传递到做表情的dmmy物体上面
 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/bicheng/76502.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

国产Linux统信安装mysql8教程步骤

系统环境 uname -a Linux FlencherHU-PC 6.12.9-amd64-desktop-rolling #23.01.01.18 SMP PREEMPT_DYNAMIC Fri Jan 10 18:29:31 CST 2025 x86_64 GNU/Linux下载离线安装包 浏览器下载https://downloads.mysql.com/archives/get/p/23/file/mysql-test-8.0.33-linux-glibc2.28…

Vite 权限绕过导致任意文件读取(CVE-2025-32395)(附脚本)

免责申明: 本文所描述的漏洞及其复现步骤仅供网络安全研究与教育目的使用。任何人不得将本文提供的信息用于非法目的或未经授权的系统测试。作者不对任何由于使用本文信息而导致的直接或间接损害承担责任。如涉及侵权,请及时与我们联系,我们将尽快处理并删除相关内容。 前言…

poi-tl

官网地址 Poi-tl Documentationword模板引擎https://deepoove.com/poi-tl github 地址 https://github.com/Sayi/poi-tl/tree/master gitcode 加速地址 GitCode - 全球开发者的开源社区,开源代码托管平台GitCode是面向全球开发者的开源社区,包括原创博客,开源代码托管,代码…

操作系统 4.1-I/O与显示器

外设工作起来 操作系统让外设工作的基本原理和过程&#xff0c;具体来说&#xff0c;它概括了以下几个关键步骤&#xff1a; 发出指令&#xff1a;操作系统通过向控制器中的寄存器发送指令来启动外设的工作。这些指令通常是通过I/O指令&#xff08;如out指令&#xff09;来实现…

琥珀扫描 2.0.5.0 | 文档处理全能助手,支持扫描、文字提取及表格识别

琥珀扫描是一款功能强大的文档处理应用程序。它不仅仅支持基本的文档扫描功能&#xff0c;还涵盖了文字提取、证件扫描、表格识别等多种实用功能。无论是学生、职员还是教师&#xff0c;都能从中找到适合自己的功能。该应用支持拍照生成电子件&#xff0c;并能自动矫正文档边缘…

jQuery UI 小部件方法调用详解

jQuery UI 小部件方法调用详解 引言 jQuery UI 是一个基于 jQuery 的用户界面和交互库,它提供了一系列小部件,如按钮、对话框、进度条等,这些小部件极大地丰富了网页的交互性和用户体验。本文将详细介绍 jQuery UI 中小部件的方法调用,帮助开发者更好地理解和应用这些小部…

浮点数比较在Eigen数学库中的处理方法

浮点数比较在Eigen数学库中的处理方法 在Eigen数学库中进行浮点数比较时&#xff0c;由于浮点数的精度问题&#xff0c;直接使用运算符通常不是推荐的做法。Eigen提供了几种更安全的方法来进行浮点数比较&#xff1a; 1. 近似相等比较 使用isApprox()函数进行近似比较&#…

Linux-----驱动

一、内核驱动与启动流程 1. Linux内核驱动 Nor Flash: 可线性访问&#xff0c;有专门的数据及地址总线&#xff08;与内存访问方式相同&#xff09;。 Nand Flash: 不可线性访问&#xff0c;访问需要控制逻辑&#xff08;软件&#xff09;。 2. Linux启动流程 ARM架构: IRAM…

Wincc脚本全部不运行

Wincc脚本全部不运行 前言解决办法操作步骤 前言 这里主要是指旧项目移植到Wincc的高版本&#xff0c;移植后界面的一些功能均会失效。&#xff08;例如脚本不执行&#xff0c;项目编辑器不可用等情况&#xff09; 解决办法 Wincc的项目文件中有Dcf文件&#xff0c;Dcf文件包…

使用numpy构建逻辑回归模型及训练流程

逻辑回归模型构建及训练流程 关于逻辑回归的数据&#xff0c;有很多学习⽤的⽰例样本。这⾥我们使⽤scikit learn提供的数据集⽣成函数来创建 具体参数可参照官网 Scikit-learn 是⽤ Python 开发的开源机器学习库&#xff0c;⼴泛⽤于数据挖掘和数据分析。 特点&#xff1a;易…

python的多线程和多进程程序编程

CPU密集型使用多进程&#xff0c;IO密集型使用多线程 查看进程ID和线程ID的命令分别是os.getpid()和threading.current_thread() 多进程使用multiprocessing就可以了&#xff0c;通常使用进程池来完成操作&#xff0c;阻塞主进程使用join方法 多线程使用threading模块&#…

代码随想录算法训练营第十五天

LeetCode题目: 654. 最大二叉树617. 合并二叉树700. 二叉搜索树中的搜索98. 验证二叉搜索树2843. 统计对称整数的数目 其他: 今日总结 往期打卡 654. 最大二叉树 跳转: 654. 最大二叉树 学习: 代码随想录公开讲解 问题: 给定一个不重复的整数数组 nums 。 最大二叉树 可以用…

[GN] Uart协议解码器源码各个方法

系列文章目录 sigrokdecode 模块学习指南 — 准备阶段 通讯协议 - Uart sigrokdecode 模块 UART协议解码器源码解析 Uart协议解码器源码各个方法 文章目录 系列文章目录引入库parity_ok注解类型枚举options参数annotations 注解annotation_rows 注解分组接收&#xff08;RX&a…

技术分享|iTOP-RK3588开发板Ubuntu20系统旋转屏幕方案

iTOP-3588开发板采用瑞芯微RK3588处理器&#xff0c;是全新一代AloT高端应用芯片&#xff0c;采用8nmLP制程&#xff0c;搭载八核64位CPU&#xff0c;四核Cortex-A76和四核Cortex-A55架构&#xff0c;主频高达2.4GHz。是一款可用于互联网设备和其它数字多媒体的高性能产品。 在…

Unity IL2CPP内存泄漏追踪方案(基于Memory Profiler)技术详解

一、IL2CPP内存管理特性与泄漏根源 1. IL2CPP内存架构特点 内存区域管理方式常见泄漏类型托管堆(Managed)GC自动回收静态引用/事件订阅未取消原生堆(Native)手动管理非托管资源未释放桥接层GCHandle/PInvoke跨语言引用未正确释放 对惹&#xff0c;这里有一个游戏开发交流小组…

消融实验_草稿

五列数据 \begin{table}[htbp]\caption{Performance Comparison of Standalone KD Variants vs MIRKD-enhanced Variants on ACNE04 Dataset\label{AblationKD}}\centering\renewcommand{\arraystretch}{1.2}\scriptsize\begin{tabularx}{\linewidth}{{}l *{3}{>{\centering…

面向对象高级(1)

文章目录 final认识final关键字修饰类&#xff1a;修饰方法&#xff1a;修饰变量final修饰变量的注意事项 常量 单例类什么是设计模式&#xff1f;单例怎么写?饿汉式单例的特点是什么&#xff1f;单例有啥应用场景&#xff0c;有啥好处&#xff1f;懒汉式单例类。 枚举类认识枚…

不用额外下载jar包,idea快速查看使用的组件源码

以nacos为例子&#xff0c;在idea中引入了nacos依赖&#xff0c;就可以查看源码了。 2. idea选择open&#xff08;不关闭项目直接选择file-open也可以&#xff09;, 在maven的仓库里找到对应的包&#xff0c;打开 2.idea中选择 jar包&#xff0c;选择 add as library 3.这样j…

小白学习java第12天:IO流之缓冲流

1.IO缓冲流&#xff1a; 之前我们学习的都是原始流&#xff08;FileInputStream字节输入流、FileOutputStream字节输出流、FIleReader字符输入流、FIleWriter字符输出流&#xff09;其实我们可以知道对于这些其实性能都不是很好&#xff0c;要么太慢一个一个&#xff0c;要么就…

高速电路设计概述

1.1 低速设计和高速设计的例子 本节通过一个简单的例子&#xff0c;探讨高速电路设计相对于低速电路设计需要考虑哪些不同的问题。希望读者通过本例&#xff0c;对高速电路设计建立一个表象的认识。至于高速电路设计中各方面的设计要点&#xff0c;将在后续章节展开详细的讨论…