mi-task/basic_motion/yellow.py
2025-08-21 15:29:20 +08:00

179 lines
5.2 KiB
Python

'''
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()