''' This demo show the communication interface of MR813 motion control board based on Lcm. Dependency: - robot_control_cmd_lcmt.py - robot_control_response_lcmt.py ''' import lcm import sys import os import time import threading from threading import Thread, Lock from robot_control_cmd_lcmt import robot_control_cmd_lcmt from robot_control_response_lcmt import robot_control_response_lcmt from camera import MiroCameraROS2 from pyzbar import pyzbar from PIL import Image import rclpy from qreader import QReader import cv2 class Robot_Ctrl(object): def __init__(self): self.rec_thread = Thread(target=self.rec_responce) self.send_thread = Thread(target=self.send_publish) self.lc_r = lcm.LCM("udpm://239.255.76.67:7670?ttl=255") self.lc_s = lcm.LCM("udpm://239.255.76.67:7671?ttl=255") self.subscription = None self.cmd_msg = robot_control_cmd_lcmt() self.rec_msg = robot_control_response_lcmt() self.send_lock = Lock() self.delay_cnt = 0 self.mode_ok = 0 self.gait_ok = 0 self.runing = 1 def run(self): self.subscription = self.lc_r.subscribe("robot_control_response", self.msg_handler) self.send_thread.start() self.rec_thread.start() def msg_handler(self, channel, data): try: self.rec_msg = robot_control_response_lcmt().decode(data) if self.rec_msg.order_process_bar >= 95: self.mode_ok = self.rec_msg.mode else: self.mode_ok = 0 except Exception as e: print(f"消息解析失败: {str(e)}") def rec_responce(self): while self.runing: try: self.lc_r.handle() except Exception as e: print(f"接收线程错误: {str(e)}") time.sleep(0.002) def Wait_finish(self, mode, gait_id): count = 0 while self.runing and count < 2000: if self.mode_ok == mode and self.gait_ok == gait_id: return True time.sleep(0.005) count += 1 return False def send_publish(self): while self.runing: try: self.send_lock.acquire() if self.delay_cnt > 20: self.lc_s.publish("robot_control_cmd", self.cmd_msg.encode()) self.delay_cnt = 0 self.delay_cnt += 1 self.send_lock.release() time.sleep(0.005) except Exception as e: print(f"发送线程错误: {str(e)}") def Send_cmd(self, msg): try: self.send_lock.acquire() self.delay_cnt = 50 self.cmd_msg = msg finally: self.send_lock.release() def quit(self): self.runing = 0 if self.rec_thread.is_alive(): self.rec_thread.join(timeout=1.0) if self.send_thread.is_alive(): self.send_thread.join(timeout=1.0) self.lc_r.unsubscribe(self.subscription) reset_msg = robot_control_cmd_lcmt() reset_msg.mode = 0 reset_msg.life_count = 0 self.lc_s.publish("robot_control_cmd", reset_msg.encode()) class MiroCameraThread(threading.Thread): def __init__(self): super().__init__(daemon=True) self.ros_initialized = False self.camera = None self.image = None def run(self): rclpy.init() self.ros_initialized = True self.camera = MiroCameraROS2() self.image = self.camera.get_latest_image() rclpy.shutdown() def main(): try: Ctrl = Robot_Ctrl() Ctrl.run() msg = robot_control_cmd_lcmt() qreader = QReader() msg.mode = 12 msg.gait_id = 0 msg.duration = 5000 msg.life_count += 1 Ctrl.Send_cmd(msg) Ctrl.Wait_finish(12,0) # msg.mode = 21 # msg.gait_id = 0 # msg.rpy_des = [0, 0.3, 0] # msg.duration = 9000 # msg.life_count += 1 # Ctrl.Send_cmd(msg) # while Ctrl.mode_ok != 12: # time.sleep(0.01) # camera_thread2 = MiroCameraThread() # camera_thread2.start() # time.sleep(10) # #等待动作和拍照完成 # try: # # 等待抬头动作完成(根据实际情况调整超时时间) # # 等待相机线程完成 # camera_thread2.join(timeout=10) # # 处理拍摄结果 # if camera_thread2.ros_initialized and camera_thread2.image is not None: # # 保存或显示图像 # camera_thread2.camera.save_image(camera_thread2.image, "camera_image2.png") # else: # print("Camera Failed") # except Exception as e: # print(f"Fail: {e}") msg.mode = 7 msg.gait_id = 1 msg.duration = 5000 msg.life_count += 1 Ctrl.Send_cmd(msg) Ctrl.Wait_finish(7, 0) except KeyboardInterrupt: print("\n用户中断") except Exception as e: print(f"严重错误: {str(e)}") finally: if Ctrl: Ctrl.quit() sys.exit() if __name__ == '__main__': main()