#!/usr/bin/env python3 # -*- coding: utf-8 -*- """ 压力板管理器 负责压力传感器的连接、校准和足部压力数据采集 """ import os import ctypes import threading import time import json import numpy as np from typing import Optional, Dict, Any, List, Tuple import logging from collections import deque import cv2 import matplotlib.pyplot as plt import matplotlib.cm as cm from io import BytesIO import base64 from datetime import datetime try: from .base_device import BaseDevice from .utils.socket_manager import SocketManager from .utils.config_manager import ConfigManager except ImportError: from base_device import BaseDevice from utils.socket_manager import SocketManager from utils.config_manager import ConfigManager # 设置日志 logger = logging.getLogger(__name__) # 检查matplotlib可用性 try: import matplotlib matplotlib.use('Agg') import matplotlib.pyplot as plt import matplotlib.patches as patches MATPLOTLIB_AVAILABLE = True except ImportError: MATPLOTLIB_AVAILABLE = False logger.warning("matplotlib不可用,将使用简化的压力图像生成") class RealPressureDevice: """真实SMiTSense压力传感器设备""" def __init__(self, dll_path=None): """ 初始化SMiTSense压力传感器 Args: dll_path: DLL文件路径,如果为None则使用默认路径 """ self.dll = None self.device_handle = None self.is_connected = False self.rows = 0 self.cols = 0 self.frame_size = 0 self.buf = None # 设置DLL路径 - 使用正确的DLL文件名 if dll_path is None: # 尝试多个可能的DLL文件名 dll_candidates = [ os.path.join(os.path.dirname(__file__), '..', 'dll', 'smitsense', 'SMiTSenseUsbWrapper.dll'), os.path.join(os.path.dirname(__file__), '..', 'dll', 'smitsense', 'SMiTSenseUsb-F3.0.dll') ] dll_path = None for candidate in dll_candidates: if os.path.exists(candidate): dll_path = candidate break if dll_path is None: raise FileNotFoundError(f"未找到SMiTSense DLL文件,检查路径: {dll_candidates}") self.dll_path = dll_path logger.info(f'初始化真实压力传感器设备,DLL路径: {dll_path}') try: self._load_dll() self._initialize_device() except Exception as e: logger.error(f'压力传感器初始化失败: {e}') # 如果真实设备初始化失败,可以选择降级为模拟设备 raise def _load_dll(self): """加载SMiTSense DLL并设置函数签名""" try: if not os.path.exists(self.dll_path): raise FileNotFoundError(f"DLL文件未找到: {self.dll_path}") # 加载DLL self.dll = ctypes.WinDLL(self.dll_path) logger.info(f"成功加载DLL: {self.dll_path}") # 设置函数签名(基于testsmit.py的工作代码) self.dll.SMiTSenseUsb_Init.argtypes = [ctypes.c_int] self.dll.SMiTSenseUsb_Init.restype = ctypes.c_int self.dll.SMiTSenseUsb_ScanDevices.argtypes = [ctypes.POINTER(ctypes.c_int)] self.dll.SMiTSenseUsb_ScanDevices.restype = ctypes.c_int self.dll.SMiTSenseUsb_OpenAndStart.argtypes = [ ctypes.c_int, ctypes.POINTER(ctypes.c_uint16), ctypes.POINTER(ctypes.c_uint16) ] self.dll.SMiTSenseUsb_OpenAndStart.restype = ctypes.c_int self.dll.SMiTSenseUsb_GetLatestFrame.argtypes = [ ctypes.POINTER(ctypes.c_uint16), ctypes.c_int ] self.dll.SMiTSenseUsb_GetLatestFrame.restype = ctypes.c_int self.dll.SMiTSenseUsb_StopAndClose.argtypes = [] self.dll.SMiTSenseUsb_StopAndClose.restype = ctypes.c_int logger.info("DLL函数签名设置完成") except Exception as e: logger.error(f"加载DLL失败: {e}") raise def _initialize_device(self): """初始化设备连接""" try: # 初始化USB连接 ret = self.dll.SMiTSenseUsb_Init(0) if ret != 0: raise RuntimeError(f"USB初始化失败: {ret}") # 扫描设备 count = ctypes.c_int() ret = self.dll.SMiTSenseUsb_ScanDevices(ctypes.byref(count)) if ret != 0 or count.value == 0: raise RuntimeError(f"设备扫描失败或未找到设备: {ret}, count: {count.value}") logger.info(f"发现 {count.value} 个SMiTSense设备") # 打开并启动第一个设备 rows = ctypes.c_uint16() cols = ctypes.c_uint16() ret = self.dll.SMiTSenseUsb_OpenAndStart(0, ctypes.byref(rows), ctypes.byref(cols)) if ret != 0: raise RuntimeError(f"设备启动失败: {ret}") self.rows = rows.value self.cols = cols.value self.frame_size = self.rows * self.cols self.buf_type = ctypes.c_uint16 * self.frame_size self.buf = self.buf_type() self.is_connected = True logger.info(f"SMiTSense压力传感器初始化成功: {self.rows}行 x {self.cols}列") except Exception as e: logger.error(f"设备初始化失败: {e}") raise def read_data(self) -> Dict[str, Any]: """读取压力数据并转换为与MockPressureDevice兼容的格式""" try: if not self.is_connected or not self.dll: logger.error("设备未连接") return self._get_empty_data() # 读取原始压力数据 ret = self.dll.SMiTSenseUsb_GetLatestFrame(self.buf, self.frame_size) if ret != 0: logger.warning(f"读取数据帧失败: {ret}") return self._get_empty_data() # 转换为numpy数组 raw_data = np.frombuffer(self.buf, dtype=np.uint16).reshape((self.rows, self.cols)) # 计算足部区域压力 (基于传感器的实际布局) foot_zones = self._calculate_foot_pressure_zones(raw_data) # 生成压力图像 pressure_image_base64 = self._generate_pressure_image( foot_zones['left_front'], foot_zones['left_rear'], foot_zones['right_front'], foot_zones['right_rear'], raw_data ) return { 'foot_pressure': { 'left_front': round(foot_zones['left_front'], 2), 'left_rear': round(foot_zones['left_rear'], 2), 'right_front': round(foot_zones['right_front'], 2), 'right_rear': round(foot_zones['right_rear'], 2), 'left_total': round(foot_zones['left_total'], 2), 'right_total': round(foot_zones['right_total'], 2) }, 'pressure_image': pressure_image_base64, 'timestamp': datetime.now().isoformat() } except Exception as e: logger.error(f"读取压力数据异常: {e}") return self._get_empty_data() def _calculate_foot_pressure_zones(self, raw_data): """计算足部区域压力,返回百分比: - 左足、右足:相对于双足总压的百分比 - 左前、左后:相对于左足总压的百分比 - 右前、右后:相对于右足总压的百分比 基于原始矩阵按行列各等分为四象限(上半部为前、下半部为后,左半部为左、右半部为右)。 """ try: # 防护:空数据 if raw_data is None: raise ValueError("raw_data is None") # 转为浮点以避免 uint16 溢出 rd = np.asarray(raw_data, dtype=np.float64) rows, cols = rd.shape if rd.ndim == 2 else (0, 0) if rows == 0 or cols == 0: raise ValueError("raw_data has invalid shape") # 行列对半分(上=前,下=后;左=左,右=右) mid_r = rows // 2 mid_c = cols // 2 # 四象限求和 left_front = float(np.sum(rd[:mid_r, :mid_c], dtype=np.float64)) left_rear = float(np.sum(rd[mid_r:, :mid_c], dtype=np.float64)) right_front = float(np.sum(rd[:mid_r, mid_c:], dtype=np.float64)) right_rear = float(np.sum(rd[mid_r:, mid_c:], dtype=np.float64)) # 绝对总压 left_total_abs = left_front + left_rear right_total_abs = right_front + right_rear total_abs = left_total_abs + right_total_abs # 左右足占比(相对于双足总压) left_total_pct = float((left_total_abs / total_abs * 100) if total_abs > 0 else 0) right_total_pct = float((right_total_abs / total_abs * 100) if total_abs > 0 else 0) # 前后占比(相对于各自单足总压) left_front_pct = float((left_front / left_total_abs * 100) if left_total_abs > 0 else 0) left_rear_pct = float((left_rear / left_total_abs * 100) if left_total_abs > 0 else 0) right_front_pct = float((right_front / right_total_abs * 100) if right_total_abs > 0 else 0) right_rear_pct = float((right_rear / right_total_abs * 100) if right_total_abs > 0 else 0) return { 'left_front': round(left_front_pct), 'left_rear': round(left_rear_pct), 'right_front': round(right_front_pct), 'right_rear': round(right_rear_pct), 'left_total': round(left_total_pct), 'right_total': round(right_total_pct), 'total_pressure': round(total_abs) } except Exception as e: logger.error(f"计算足部区域压力异常: {e}") return { 'left_front': 0, 'left_rear': 0, 'right_front': 0, 'right_rear': 0, 'left_total': 0, 'right_total': 0, 'total_pressure': 0 } def _generate_pressure_image(self, left_front, left_rear, right_front, right_rear, raw_data=None) -> str: """生成足部压力图片的base64数据""" try: if MATPLOTLIB_AVAILABLE and raw_data is not None: # 使用原始数据生成更详细的热力图 return self._generate_heatmap_image(raw_data) else: # 降级到简单的区域显示图 return self._generate_simple_pressure_image(left_front, left_rear, right_front, right_rear) except Exception as e: logger.warning(f"生成压力图片失败: {e}") return "" def _generate_heatmap_image(self, raw_data) -> str: """生成基于原始数据的热力图(OpenCV实现,固定范围映射,效果与matplotlib一致)""" try: import cv2 import numpy as np import base64 from io import BytesIO from PIL import Image # 固定映射范围(与 matplotlib vmin/vmax 一致) vmin, vmax = 0, 1000 norm_data = np.clip((raw_data - vmin) / (vmax - vmin) * 255, 0, 255).astype(np.uint8) # 应用 jet 颜色映射 heatmap = cv2.applyColorMap(norm_data, cv2.COLORMAP_JET) # OpenCV 生成的是 BGR,转成 RGB heatmap_rgb = cv2.cvtColor(heatmap, cv2.COLOR_BGR2RGB) # 转成 Pillow Image img = Image.fromarray(heatmap_rgb) # 输出为 Base64 PNG buffer = BytesIO() img.save(buffer, format="PNG") buffer.seek(0) image_base64 = base64.b64encode(buffer.getvalue()).decode("utf-8") return f"data:image/png;base64,{image_base64}" except Exception as e: logger.warning(f"生成热力图失败: {e}") return self._generate_simple_pressure_image(0, 0, 0, 0) def _generate_simple_pressure_image(self, left_front, left_rear, right_front, right_rear) -> str: """生成简单的足部压力区域图""" try: import matplotlib matplotlib.use('Agg') import matplotlib.pyplot as plt import matplotlib.patches as patches from io import BytesIO # 创建图形 fig, ax = plt.subplots(1, 1, figsize=(6, 8)) ax.set_xlim(0, 10) ax.set_ylim(0, 12) ax.set_aspect('equal') ax.axis('off') # 定义颜色映射 max_pressure = max(left_front, left_rear, right_front, right_rear) if max_pressure > 0: left_front_color = plt.cm.Reds(left_front / max_pressure) left_rear_color = plt.cm.Reds(left_rear / max_pressure) right_front_color = plt.cm.Reds(right_front / max_pressure) right_rear_color = plt.cm.Reds(right_rear / max_pressure) else: left_front_color = left_rear_color = right_front_color = right_rear_color = 'lightgray' # 绘制足部区域 left_front_rect = patches.Rectangle((1, 6), 2, 4, linewidth=1, edgecolor='black', facecolor=left_front_color) left_rear_rect = patches.Rectangle((1, 2), 2, 4, linewidth=1, edgecolor='black', facecolor=left_rear_color) right_front_rect = patches.Rectangle((7, 6), 2, 4, linewidth=1, edgecolor='black', facecolor=right_front_color) right_rear_rect = patches.Rectangle((7, 2), 2, 4, linewidth=1, edgecolor='black', facecolor=right_rear_color) ax.add_patch(left_front_rect) ax.add_patch(left_rear_rect) ax.add_patch(right_front_rect) ax.add_patch(right_rear_rect) # 添加标签 ax.text(2, 8, f'{left_front:.1f}', ha='center', va='center', fontsize=10, weight='bold') ax.text(2, 4, f'{left_rear:.1f}', ha='center', va='center', fontsize=10, weight='bold') ax.text(8, 8, f'{right_front:.1f}', ha='center', va='center', fontsize=10, weight='bold') ax.text(8, 4, f'{right_rear:.1f}', ha='center', va='center', fontsize=10, weight='bold') ax.text(2, 0.5, '左足', ha='center', va='center', fontsize=12, weight='bold') ax.text(8, 0.5, '右足', ha='center', va='center', fontsize=12, weight='bold') # 保存为base64 buffer = BytesIO() plt.savefig(buffer, format='png', bbox_inches='tight', dpi=100, facecolor='black') buffer.seek(0) image_base64 = base64.b64encode(buffer.getvalue()).decode('utf-8') plt.close(fig) return f"data:image/png;base64,{image_base64}" except Exception as e: logger.warning(f"生成简单压力图片失败: {e}") return "" def _get_empty_data(self): """返回空的压力数据""" return { 'foot_pressure': { 'left_front': 0.0, 'left_rear': 0.0, 'right_front': 0.0, 'right_rear': 0.0, 'left_total': 0.0, 'right_total': 0.0 }, 'pressure_image': "", 'timestamp': datetime.now().isoformat() } def close(self): """显式关闭压力传感器连接""" try: if self.is_connected and self.dll: self.dll.SMiTSenseUsb_StopAndClose() self.is_connected = False logger.info('SMiTSense压力传感器连接已关闭') except Exception as e: logger.error(f'关闭压力传感器连接异常: {e}') def __del__(self): """析构函数,确保资源清理""" self.close() class MockPressureDevice: """模拟压力传感器设备,模拟真实SMiTSense设备的行为""" def __init__(self): self.base_pressure = 500 # 基础压力值 self.noise_level = 10 self.rows = 4 # 模拟传感器矩阵行数 self.cols = 4 # 模拟传感器矩阵列数 self.time_offset = np.random.random() * 10 # 随机时间偏移,让每个实例的波形不同 def read_data(self) -> Dict[str, Any]: """读取压力数据,模拟基于矩阵数据的真实设备行为""" try: # 生成模拟的传感器矩阵数据 raw_data = self._generate_simulated_matrix_data() # 使用与真实设备相同的计算逻辑 foot_zones = self._calculate_foot_pressure_zones(raw_data) # 生成压力图像 pressure_image_base64 = self._generate_pressure_image( foot_zones['left_front'], foot_zones['left_rear'], foot_zones['right_front'], foot_zones['right_rear'], raw_data ) return { 'foot_pressure': { 'left_front': round(foot_zones['left_front'], 2), 'left_rear': round(foot_zones['left_rear'], 2), 'right_front': round(foot_zones['right_front'], 2), 'right_rear': round(foot_zones['right_rear'], 2), 'left_total': round(foot_zones['left_total'], 2), 'right_total': round(foot_zones['right_total'], 2) }, 'pressure_image': pressure_image_base64, 'timestamp': datetime.now().isoformat() } except Exception as e: logger.error(f"模拟压力设备读取数据异常: {e}") return self._get_empty_data() def _generate_simulated_matrix_data(self): """生成模拟的传感器矩阵数据,模拟真实的足部压力分布""" import time current_time = time.time() + self.time_offset # 创建4x4的传感器矩阵 matrix_data = np.zeros((self.rows, self.cols)) # 模拟动态的压力分布,使用正弦波叠加噪声 for i in range(self.rows): for j in range(self.cols): # 基础压力值,根据传感器位置不同 base_value = self.base_pressure * (0.3 + 0.7 * np.random.random()) # 添加时间变化(模拟人体重心变化) time_variation = np.sin(current_time * 0.5 + i * 0.5 + j * 0.3) * 0.3 # 添加噪声 noise = np.random.normal(0, self.noise_level) # 确保压力值非负 matrix_data[i, j] = max(0, base_value * (1 + time_variation) + noise) return matrix_data def _calculate_foot_pressure_zones(self, raw_data): """计算足部区域压力,返回百分比: - 左足、右足:相对于双足总压的百分比 - 左前、左后:相对于左足总压的百分比 - 右前、右后:相对于右足总压的百分比 基于原始矩阵按行列各等分为四象限(上半部为前、下半部为后,左半部为左、右半部为右)。 """ try: # 防护:空数据 if raw_data is None: raise ValueError("raw_data is None") # 转为浮点以避免 uint16 溢出 rd = np.asarray(raw_data, dtype=np.float64) rows, cols = rd.shape if rd.ndim == 2 else (0, 0) if rows == 0 or cols == 0: raise ValueError("raw_data has invalid shape") # 行列对半分(上=前,下=后;左=左,右=右) mid_r = rows // 2 mid_c = cols // 2 # 四象限求和 left_front = float(np.sum(rd[:mid_r, :mid_c], dtype=np.float64)) left_rear = float(np.sum(rd[mid_r:, :mid_c], dtype=np.float64)) right_front = float(np.sum(rd[:mid_r, mid_c:], dtype=np.float64)) right_rear = float(np.sum(rd[mid_r:, mid_c:], dtype=np.float64)) # 绝对总压 left_total_abs = left_front + left_rear right_total_abs = right_front + right_rear total_abs = left_total_abs + right_total_abs # 左右足占比(相对于双足总压) left_total_pct = float((left_total_abs / total_abs * 100) if total_abs > 0 else 0) right_total_pct = float((right_total_abs / total_abs * 100) if total_abs > 0 else 0) # 前后占比(相对于各自单足总压) left_front_pct = float((left_front / left_total_abs * 100) if left_total_abs > 0 else 0) left_rear_pct = float((left_rear / left_total_abs * 100) if left_total_abs > 0 else 0) right_front_pct = float((right_front / right_total_abs * 100) if right_total_abs > 0 else 0) right_rear_pct = float((right_rear / right_total_abs * 100) if right_total_abs > 0 else 0) return { 'left_front': left_front_pct, 'left_rear': left_rear_pct, 'right_front': right_front_pct, 'right_rear': right_rear_pct, 'left_total': left_total_pct, 'right_total': right_total_pct, 'total_pressure': float(total_abs) } except Exception as e: logger.error(f"计算足部区域压力异常: {e}") return { 'left_front': 0, 'left_rear': 0, 'right_front': 0, 'right_rear': 0, 'left_total': 0, 'right_total': 0, 'total_pressure': 0 } def _generate_pressure_image(self, left_front, left_rear, right_front, right_rear, raw_data=None) -> str: """生成足部压力图片的base64数据""" try: if MATPLOTLIB_AVAILABLE and raw_data is not None: # 使用原始数据生成更详细的热力图 return self._generate_heatmap_image(raw_data) else: # 降级到简单的区域显示图 return self._generate_simple_pressure_image(left_front, left_rear, right_front, right_rear) except Exception as e: logger.warning(f"生成模拟压力图片失败: {e}") return "" def _generate_heatmap_image(self, raw_data) -> str: """生成基于原始数据的热力图""" try: import matplotlib matplotlib.use('Agg') import matplotlib.pyplot as plt from io import BytesIO # 参考 tests/testsmit.py 的渲染方式:使用 jet 色图、nearest 插值、固定范围并关闭坐标轴 fig, ax = plt.subplots() im = ax.imshow(raw_data, cmap='jet', interpolation='nearest', vmin=0, vmax=1000) ax.axis('off') # 紧凑布局并导出为 base64 from io import BytesIO buffer = BytesIO() plt.savefig(buffer, format='png', bbox_inches='tight', dpi=100, pad_inches=0) buffer.seek(0) image_base64 = base64.b64encode(buffer.getvalue()).decode('utf-8') plt.close(fig) return f"data:image/png;base64,{image_base64}" except Exception as e: logger.warning(f"生成热力图失败: {e}") return self._generate_simple_pressure_image(0, 0, 0, 0) def _generate_simple_pressure_image(self, left_front, left_rear, right_front, right_rear) -> str: """生成简单的足部压力区域图""" try: import matplotlib matplotlib.use('Agg') # 设置非交互式后端,避免Tkinter错误 import matplotlib.pyplot as plt import matplotlib.patches as patches from io import BytesIO # 临时禁用PIL的调试日志 pil_logger = logging.getLogger('PIL') original_level = pil_logger.level pil_logger.setLevel(logging.WARNING) # 创建图形 fig, ax = plt.subplots(1, 1, figsize=(6, 8)) ax.set_xlim(0, 10) ax.set_ylim(0, 12) ax.set_aspect('equal') ax.axis('off') # 定义颜色映射(根据压力值) max_pressure = max(left_front, left_rear, right_front, right_rear) if max_pressure > 0: left_front_color = plt.cm.Reds(left_front / max_pressure) left_rear_color = plt.cm.Reds(left_rear / max_pressure) right_front_color = plt.cm.Reds(right_front / max_pressure) right_rear_color = plt.cm.Reds(right_rear / max_pressure) else: left_front_color = left_rear_color = right_front_color = right_rear_color = 'lightgray' # 绘制左脚 left_front_rect = patches.Rectangle((1, 6), 2, 4, linewidth=1, edgecolor='black', facecolor=left_front_color) left_rear_rect = patches.Rectangle((1, 2), 2, 4, linewidth=1, edgecolor='black', facecolor=left_rear_color) # 绘制右脚 right_front_rect = patches.Rectangle((7, 6), 2, 4, linewidth=1, edgecolor='black', facecolor=right_front_color) right_rear_rect = patches.Rectangle((7, 2), 2, 4, linewidth=1, edgecolor='black', facecolor=right_rear_color) # 添加到图形 ax.add_patch(left_front_rect) ax.add_patch(left_rear_rect) ax.add_patch(right_front_rect) ax.add_patch(right_rear_rect) # 添加标签 ax.text(2, 8, f'{left_front:.1f}', ha='center', va='center', fontsize=10, weight='bold') ax.text(2, 4, f'{left_rear:.1f}', ha='center', va='center', fontsize=10, weight='bold') ax.text(8, 8, f'{right_front:.1f}', ha='center', va='center', fontsize=10, weight='bold') ax.text(8, 4, f'{right_rear:.1f}', ha='center', va='center', fontsize=10, weight='bold') ax.text(2, 0.5, '左足', ha='center', va='center', fontsize=12, weight='bold') ax.text(8, 0.5, '右足', ha='center', va='center', fontsize=12, weight='bold') # 保存为base64 buffer = BytesIO() plt.savefig(buffer, format='png', bbox_inches='tight', dpi=100, facecolor='white') buffer.seek(0) image_base64 = base64.b64encode(buffer.getvalue()).decode('utf-8') plt.close(fig) # 恢复PIL的日志级别 pil_logger.setLevel(original_level) return f"data:image/png;base64,{image_base64}" except Exception as e: # 确保在异常情况下也恢复PIL的日志级别 try: pil_logger.setLevel(original_level) except: pass logger.warning(f"生成压力图片失败: {e}") # 返回一个简单的占位符base64图片 return "" def _get_empty_data(self): """返回空的压力数据""" return { 'foot_pressure': { 'left_front': 0.0, 'left_rear': 0.0, 'right_front': 0.0, 'right_rear': 0.0, 'left_total': 0.0, 'right_total': 0.0 }, 'pressure_image': "", 'timestamp': datetime.now().isoformat() } class PressureManager(BaseDevice): """压力板管理器""" def __init__(self, socketio, config_manager: Optional[ConfigManager] = None): """ 初始化压力板管理器 Args: socketio: SocketIO实例 config_manager: 配置管理器实例 """ # 配置管理 self.config_manager = config_manager or ConfigManager() self.config = self.config_manager.get_device_config('pressure') super().__init__("pressure", self.config) # 保存socketio实例 self._socketio = socketio # 设备实例 self.device = None self.device_type = self.config.get('device_type', 'mock') # 'real' 或 'mock' # 数据流相关 self.streaming_thread = None self.is_streaming = False self.stream_interval = self.config.get('stream_interval', 0.1) # 100ms间隔 # 校准相关 self.is_calibrated = False self.calibration_data = None # 性能统计 self.packet_count = 0 self.error_count = 0 self.last_data_time = None self.logger.info(f"压力板管理器初始化完成 - 设备类型: {self.device_type}") def initialize(self) -> bool: """ 初始化压力板设备 Returns: bool: 初始化是否成功 """ try: self.logger.info(f"正在初始化压力板设备 - 类型: {self.device_type}") # 根据设备类型创建设备实例 if self.device_type == 'real': self.device = RealPressureDevice() else: self.device = MockPressureDevice() self.is_connected = True self._device_info.update({ 'device_type': self.device_type, 'matrix_size': '4x4' if hasattr(self.device, 'rows') else 'unknown' }) self.logger.info(f"压力板初始化成功 - 设备类型: {self.device_type}") return True except Exception as e: self.logger.error(f"压力板初始化失败: {e}") self.is_connected = False self.device = None return False def start_streaming(self) -> bool: """ 开始压力数据流 Args: socketio: SocketIO实例 Returns: bool: 启动是否成功 """ try: if not self.is_connected or not self.device: self.logger.error("设备未连接,无法启动数据流") return False if self.is_streaming: self.logger.warning("压力数据流已在运行") return True self.is_streaming = True self.streaming_thread = threading.Thread(target=self._pressure_streaming_thread, daemon=True) self.streaming_thread.start() self.logger.info("压力数据流启动成功") return True except Exception as e: self.logger.error(f"启动压力数据流失败: {e}") self.is_streaming = False return False def stop_streaming(self) -> bool: """ 停止压力数据流 Returns: bool: 停止是否成功 """ try: if not self.is_streaming: return True self.is_streaming = False if self.streaming_thread and self.streaming_thread.is_alive(): self.streaming_thread.join(timeout=2.0) self.logger.info("压力数据流已停止") return True except Exception as e: self.logger.error(f"停止压力数据流失败: {e}") return False def _pressure_streaming_thread(self): """ 压力数据流处理线程 """ self.logger.info("压力数据流线程启动") try: while self.is_streaming and self.is_connected: try: # 从设备读取数据 if self.device: pressure_data = self.device.read_data() if pressure_data: # 更新统计信息 self.packet_count += 1 self.last_data_time = time.time() # 发送数据到前端 if self._socketio: self._socketio.emit('pressure_data', pressure_data, namespace='/pressure') else: self.logger.warning("SocketIO实例为空,无法发送压力数据") time.sleep(self.stream_interval) except Exception as e: self.error_count += 1 self.logger.error(f"压力数据流处理异常: {e}") time.sleep(0.1) except Exception as e: self.logger.error(f"压力数据流线程异常: {e}") finally: self.logger.info("压力数据流线程结束") def get_status(self) -> Dict[str, Any]: """ 获取设备状态 Returns: Dict[str, Any]: 设备状态信息 """ return { 'device_type': self.device_type, 'is_connected': self.is_connected, 'is_streaming': self.is_streaming, 'is_calibrated': self.is_calibrated, 'packet_count': self.packet_count, 'error_count': self.error_count, 'last_data_time': self.last_data_time, 'device_info': self.get_device_info() } def calibrate(self) -> bool: """ 校准压力传感器 Returns: bool: 校准是否成功 """ try: self.logger.info("开始压力传感器校准...") # 这里可以添加具体的校准逻辑 # 目前简单设置为已校准状态 self.is_calibrated = True self.calibration_data = { 'timestamp': datetime.now().isoformat(), 'baseline': 'calibrated' } self.logger.info("压力传感器校准完成") return True except Exception as e: self.logger.error(f"压力传感器校准失败: {e}") return False def disconnect(self) -> bool: """ 断开设备连接 Returns: bool: 断开是否成功 """ try: # 停止数据流 self.stop_streaming() # 关闭设备连接 if self.device and hasattr(self.device, 'close'): self.device.close() self.device = None self.is_connected = False self.logger.info("压力板设备连接已断开") return True except Exception as e: self.logger.error(f"断开压力板设备连接失败: {e}") return False def cleanup(self) -> None: """清理资源""" try: self.stop_streaming() self.disconnect() self.logger.info("压力板设备资源清理完成") except Exception as e: self.logger.error(f"压力板设备资源清理失败: {e}")