Add/Update GUI module from wenchun
parent
e4b658d578
commit
9f1235197a
@ -0,0 +1,398 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
from PyQt6.QtWidgets import (QWidget, QVBoxLayout, QHBoxLayout, QLabel,
|
||||||
|
QPushButton, QLineEdit)
|
||||||
|
from PyQt6.QtCore import pyqtSignal
|
||||||
|
|
||||||
|
class CommPanel(QWidget):
|
||||||
|
"""通讯设置面板类"""
|
||||||
|
|
||||||
|
# 定义信号
|
||||||
|
udp_connection_added = pyqtSignal(str, int) # ip, port
|
||||||
|
udp_connection_toggled = pyqtSignal(dict, QPushButton, QLabel) # conn, btn, status_label
|
||||||
|
udp_connection_removed = pyqtSignal(dict, QWidget) # conn, panel
|
||||||
|
ws_connection_added = pyqtSignal(str) # url
|
||||||
|
ws_connection_toggled = pyqtSignal(dict, QPushButton, QLabel) # conn, btn, status_label
|
||||||
|
ws_connection_removed = pyqtSignal(dict, QWidget) # conn, panel
|
||||||
|
status_message = pyqtSignal(str, int) # message, timeout
|
||||||
|
|
||||||
|
def __init__(self, parent=None):
|
||||||
|
super().__init__(parent)
|
||||||
|
self.udp_connections = []
|
||||||
|
self.ws_connections = []
|
||||||
|
self._init_ui()
|
||||||
|
|
||||||
|
def _init_ui(self):
|
||||||
|
"""初始化UI"""
|
||||||
|
layout = QVBoxLayout(self)
|
||||||
|
layout.setContentsMargins(10, 10, 10, 10)
|
||||||
|
layout.setSpacing(10)
|
||||||
|
|
||||||
|
# ========== UDP MAVLink 區域 ==========
|
||||||
|
udp_title = QLabel("UDP")
|
||||||
|
udp_title.setStyleSheet("""
|
||||||
|
color: #DDD;
|
||||||
|
font-size: 14px;
|
||||||
|
font-weight: bold;
|
||||||
|
padding: 5px;
|
||||||
|
background-color: #333;
|
||||||
|
border-radius: 4px;
|
||||||
|
""")
|
||||||
|
layout.addWidget(udp_title)
|
||||||
|
|
||||||
|
# UDP 連接列表容器
|
||||||
|
self.udp_list_widget = QWidget()
|
||||||
|
self.udp_list_layout = QVBoxLayout(self.udp_list_widget)
|
||||||
|
self.udp_list_layout.setContentsMargins(0, 0, 0, 0)
|
||||||
|
self.udp_list_layout.setSpacing(5)
|
||||||
|
layout.addWidget(self.udp_list_widget)
|
||||||
|
|
||||||
|
# UDP 添加新連接區域
|
||||||
|
add_udp_widget = QWidget()
|
||||||
|
add_udp_layout = QHBoxLayout(add_udp_widget)
|
||||||
|
add_udp_layout.setContentsMargins(0, 0, 0, 0)
|
||||||
|
|
||||||
|
self.udp_ip_input = QLineEdit()
|
||||||
|
self.udp_ip_input.setText("127.0.0.1")
|
||||||
|
self.udp_ip_input.setPlaceholderText("IP")
|
||||||
|
self.udp_ip_input.setStyleSheet("""
|
||||||
|
QLineEdit {
|
||||||
|
background-color: #333;
|
||||||
|
color: #DDD;
|
||||||
|
border: 1px solid #555;
|
||||||
|
border-radius: 4px;
|
||||||
|
padding: 5px;
|
||||||
|
}
|
||||||
|
""")
|
||||||
|
|
||||||
|
self.udp_port_input = QLineEdit()
|
||||||
|
self.udp_port_input.setText("14550")
|
||||||
|
self.udp_port_input.setPlaceholderText("Port")
|
||||||
|
self.udp_port_input.setFixedWidth(80)
|
||||||
|
self.udp_port_input.setStyleSheet("""
|
||||||
|
QLineEdit {
|
||||||
|
background-color: #333;
|
||||||
|
color: #DDD;
|
||||||
|
border: 1px solid #555;
|
||||||
|
border-radius: 4px;
|
||||||
|
padding: 5px;
|
||||||
|
}
|
||||||
|
""")
|
||||||
|
|
||||||
|
add_udp_btn = QPushButton("添加")
|
||||||
|
add_udp_btn.clicked.connect(self._handle_add_udp)
|
||||||
|
add_udp_btn.setStyleSheet("""
|
||||||
|
QPushButton {
|
||||||
|
background-color: #4CAF50;
|
||||||
|
color: white;
|
||||||
|
border: none;
|
||||||
|
padding: 6px 12px;
|
||||||
|
border-radius: 4px;
|
||||||
|
min-width: 30px;
|
||||||
|
}
|
||||||
|
QPushButton:hover { background-color: #45a049; }
|
||||||
|
""")
|
||||||
|
|
||||||
|
add_udp_layout.addWidget(QLabel("IP:", styleSheet="color: #DDD;"))
|
||||||
|
add_udp_layout.addWidget(self.udp_ip_input)
|
||||||
|
add_udp_layout.addWidget(QLabel("Port:", styleSheet="color: #DDD;"))
|
||||||
|
add_udp_layout.addWidget(self.udp_port_input)
|
||||||
|
add_udp_layout.addWidget(add_udp_btn)
|
||||||
|
|
||||||
|
layout.addWidget(add_udp_widget)
|
||||||
|
|
||||||
|
# 分隔線
|
||||||
|
separator = QWidget()
|
||||||
|
separator.setFixedHeight(20)
|
||||||
|
layout.addWidget(separator)
|
||||||
|
|
||||||
|
# ========== WebSocket 區域 ==========
|
||||||
|
ws_title = QLabel("WebSocket")
|
||||||
|
ws_title.setStyleSheet("""
|
||||||
|
color: #DDD;
|
||||||
|
font-size: 14px;
|
||||||
|
font-weight: bold;
|
||||||
|
padding: 5px;
|
||||||
|
background-color: #333;
|
||||||
|
border-radius: 4px;
|
||||||
|
""")
|
||||||
|
layout.addWidget(ws_title)
|
||||||
|
|
||||||
|
# WebSocket 連接列表容器
|
||||||
|
self.ws_list_widget = QWidget()
|
||||||
|
self.ws_list_layout = QVBoxLayout(self.ws_list_widget)
|
||||||
|
self.ws_list_layout.setContentsMargins(0, 0, 0, 0)
|
||||||
|
self.ws_list_layout.setSpacing(5)
|
||||||
|
layout.addWidget(self.ws_list_widget)
|
||||||
|
|
||||||
|
# WebSocket 添加新連接區域
|
||||||
|
add_ws_widget = QWidget()
|
||||||
|
add_ws_layout = QHBoxLayout(add_ws_widget)
|
||||||
|
add_ws_layout.setContentsMargins(0, 0, 0, 0)
|
||||||
|
|
||||||
|
self.ws_url_input = QLineEdit()
|
||||||
|
self.ws_url_input.setPlaceholderText("host")
|
||||||
|
self.ws_url_input.setStyleSheet("""
|
||||||
|
QLineEdit {
|
||||||
|
background-color: #333;
|
||||||
|
color: #DDD;
|
||||||
|
border: 1px solid #555;
|
||||||
|
border-radius: 4px;
|
||||||
|
padding: 5px;
|
||||||
|
}
|
||||||
|
""")
|
||||||
|
|
||||||
|
add_ws_btn = QPushButton("添加")
|
||||||
|
add_ws_btn.clicked.connect(self._handle_add_ws)
|
||||||
|
add_ws_btn.setStyleSheet("""
|
||||||
|
QPushButton {
|
||||||
|
background-color: #4CAF50;
|
||||||
|
color: white;
|
||||||
|
border: none;
|
||||||
|
padding: 6px 12px;
|
||||||
|
border-radius: 4px;
|
||||||
|
min-width: 30px;
|
||||||
|
}
|
||||||
|
QPushButton:hover { background-color: #45a049; }
|
||||||
|
""")
|
||||||
|
|
||||||
|
add_ws_layout.addWidget(QLabel("URL:", styleSheet="color: #DDD;"))
|
||||||
|
add_ws_layout.addWidget(self.ws_url_input)
|
||||||
|
add_ws_layout.addWidget(add_ws_btn)
|
||||||
|
|
||||||
|
layout.addWidget(add_ws_widget)
|
||||||
|
layout.addStretch()
|
||||||
|
|
||||||
|
def _handle_add_udp(self):
|
||||||
|
"""處理添加 UDP 連接"""
|
||||||
|
ip = self.udp_ip_input.text().strip()
|
||||||
|
port_text = self.udp_port_input.text().strip()
|
||||||
|
|
||||||
|
if not ip or not port_text:
|
||||||
|
self.status_message.emit("請輸入 IP 和 Port", 3000)
|
||||||
|
return
|
||||||
|
|
||||||
|
try:
|
||||||
|
port = int(port_text)
|
||||||
|
if port < 1 or port > 65535:
|
||||||
|
raise ValueError("Port 超出範圍")
|
||||||
|
except ValueError:
|
||||||
|
self.status_message.emit("Port 必須是 1-65535 的數字", 3000)
|
||||||
|
return
|
||||||
|
|
||||||
|
# 檢查是否已存在相同連接
|
||||||
|
for conn in self.udp_connections:
|
||||||
|
if conn['ip'] == ip and conn['port'] == port:
|
||||||
|
self.status_message.emit("連接已存在", 3000)
|
||||||
|
return
|
||||||
|
|
||||||
|
# 發送信號通知主窗口
|
||||||
|
self.udp_connection_added.emit(ip, port)
|
||||||
|
|
||||||
|
# 清空輸入框
|
||||||
|
self.udp_ip_input.clear()
|
||||||
|
self.udp_port_input.clear()
|
||||||
|
|
||||||
|
def _handle_add_ws(self):
|
||||||
|
"""處理添加 WebSocket 連接"""
|
||||||
|
input_url = self.ws_url_input.text().strip()
|
||||||
|
|
||||||
|
if not input_url:
|
||||||
|
self.status_message.emit("請輸入 WebSocket URL", 3000)
|
||||||
|
return
|
||||||
|
|
||||||
|
# 自動添加 ws:// 前綴
|
||||||
|
if not input_url.startswith('ws://') and not input_url.startswith('wss://'):
|
||||||
|
url = f'ws://{input_url}'
|
||||||
|
else:
|
||||||
|
url = input_url
|
||||||
|
|
||||||
|
# 基本 URL 格式驗證
|
||||||
|
try:
|
||||||
|
if '://' in url:
|
||||||
|
parts = url.split('://', 1)
|
||||||
|
if len(parts) == 2 and ':' not in parts[1]:
|
||||||
|
self.status_message.emit("URL 格式錯誤,需要包含端口號 (例如: 127.0.0.1:8756)", 3000)
|
||||||
|
return
|
||||||
|
except:
|
||||||
|
self.status_message.emit("URL 格式錯誤", 3000)
|
||||||
|
return
|
||||||
|
|
||||||
|
# 檢查是否已存在相同連接
|
||||||
|
for conn in self.ws_connections:
|
||||||
|
if conn['url'] == url:
|
||||||
|
self.status_message.emit("連接已存在", 3000)
|
||||||
|
return
|
||||||
|
|
||||||
|
# 發送信號通知主窗口
|
||||||
|
self.ws_connection_added.emit(url)
|
||||||
|
|
||||||
|
# 清空輸入框
|
||||||
|
self.ws_url_input.clear()
|
||||||
|
|
||||||
|
def create_udp_connection_panel(self, conn):
|
||||||
|
"""創建 UDP 連接面板"""
|
||||||
|
panel = QWidget()
|
||||||
|
panel.setStyleSheet("""
|
||||||
|
QWidget {
|
||||||
|
background-color: #2A2A2A;
|
||||||
|
border-radius: 6px;
|
||||||
|
padding: 8px;
|
||||||
|
border: 1px solid #444;
|
||||||
|
}
|
||||||
|
""")
|
||||||
|
|
||||||
|
layout = QHBoxLayout(panel)
|
||||||
|
layout.setContentsMargins(8, 8, 8, 8)
|
||||||
|
|
||||||
|
# 連接資訊
|
||||||
|
info_label = QLabel(f"{conn['name']} - {conn['ip']}:{conn['port']}")
|
||||||
|
info_label.setStyleSheet("color: #DDD; font-size: 12px;")
|
||||||
|
|
||||||
|
# 狀態指示器
|
||||||
|
status_label = QLabel("●")
|
||||||
|
if conn.get('enabled', False):
|
||||||
|
status_label.setStyleSheet("color: #4CAF50; font-size: 16px;")
|
||||||
|
status_label.setToolTip("運行中")
|
||||||
|
else:
|
||||||
|
status_label.setStyleSheet("color: #888; font-size: 16px;")
|
||||||
|
status_label.setToolTip("已停止")
|
||||||
|
|
||||||
|
# 控制按鈕
|
||||||
|
toggle_btn = QPushButton("停止" if conn.get('enabled', False) else "啟動")
|
||||||
|
toggle_btn.setFixedWidth(60)
|
||||||
|
toggle_btn.clicked.connect(lambda: self.udp_connection_toggled.emit(conn, toggle_btn, status_label))
|
||||||
|
toggle_btn.setStyleSheet("""
|
||||||
|
QPushButton {
|
||||||
|
background-color: #444;
|
||||||
|
color: #DDD;
|
||||||
|
border: none;
|
||||||
|
padding: 4px 8px;
|
||||||
|
border-radius: 3px;
|
||||||
|
font-size: 11px;
|
||||||
|
}
|
||||||
|
QPushButton:hover { background-color: #555; }
|
||||||
|
""")
|
||||||
|
|
||||||
|
remove_btn = QPushButton("移除")
|
||||||
|
remove_btn.setFixedWidth(60)
|
||||||
|
remove_btn.clicked.connect(lambda: self.udp_connection_removed.emit(conn, panel))
|
||||||
|
remove_btn.setStyleSheet("""
|
||||||
|
QPushButton {
|
||||||
|
background-color: #d32f2f;
|
||||||
|
color: white;
|
||||||
|
border: none;
|
||||||
|
padding: 4px 8px;
|
||||||
|
border-radius: 3px;
|
||||||
|
font-size: 11px;
|
||||||
|
}
|
||||||
|
QPushButton:hover { background-color: #b71c1c; }
|
||||||
|
""")
|
||||||
|
|
||||||
|
layout.addWidget(status_label)
|
||||||
|
layout.addWidget(info_label)
|
||||||
|
layout.addStretch()
|
||||||
|
layout.addWidget(toggle_btn)
|
||||||
|
layout.addWidget(remove_btn)
|
||||||
|
|
||||||
|
# 儲存引用
|
||||||
|
panel.connection = conn
|
||||||
|
panel.toggle_btn = toggle_btn
|
||||||
|
panel.status_label = status_label
|
||||||
|
|
||||||
|
return panel
|
||||||
|
|
||||||
|
def create_ws_connection_panel(self, conn):
|
||||||
|
"""創建 WebSocket 連接面板"""
|
||||||
|
panel = QWidget()
|
||||||
|
panel.setStyleSheet("""
|
||||||
|
QWidget {
|
||||||
|
background-color: #2A2A2A;
|
||||||
|
border-radius: 6px;
|
||||||
|
padding: 8px;
|
||||||
|
border: 1px solid #444;
|
||||||
|
}
|
||||||
|
""")
|
||||||
|
|
||||||
|
layout = QHBoxLayout(panel)
|
||||||
|
layout.setContentsMargins(8, 8, 8, 8)
|
||||||
|
|
||||||
|
# 連接資訊
|
||||||
|
info_label = QLabel(f"{conn['name']} - {conn['url']}")
|
||||||
|
info_label.setStyleSheet("color: #DDD; font-size: 12px;")
|
||||||
|
|
||||||
|
# 狀態指示器
|
||||||
|
status_label = QLabel("●")
|
||||||
|
if conn.get('enabled', False):
|
||||||
|
status_label.setStyleSheet("color: #4CAF50; font-size: 16px;")
|
||||||
|
status_label.setToolTip("運行中")
|
||||||
|
else:
|
||||||
|
status_label.setStyleSheet("color: #888; font-size: 16px;")
|
||||||
|
status_label.setToolTip("已停止")
|
||||||
|
|
||||||
|
# 控制按鈕
|
||||||
|
toggle_btn = QPushButton("停止" if conn.get('enabled', False) else "啟動")
|
||||||
|
toggle_btn.setFixedWidth(60)
|
||||||
|
toggle_btn.clicked.connect(lambda: self.ws_connection_toggled.emit(conn, toggle_btn, status_label))
|
||||||
|
toggle_btn.setStyleSheet("""
|
||||||
|
QPushButton {
|
||||||
|
background-color: #444;
|
||||||
|
color: #DDD;
|
||||||
|
border: none;
|
||||||
|
padding: 4px 8px;
|
||||||
|
border-radius: 3px;
|
||||||
|
font-size: 11px;
|
||||||
|
}
|
||||||
|
QPushButton:hover { background-color: #555; }
|
||||||
|
""")
|
||||||
|
|
||||||
|
remove_btn = QPushButton("移除")
|
||||||
|
remove_btn.setFixedWidth(60)
|
||||||
|
remove_btn.clicked.connect(lambda: self.ws_connection_removed.emit(conn, panel))
|
||||||
|
remove_btn.setStyleSheet("""
|
||||||
|
QPushButton {
|
||||||
|
background-color: #d32f2f;
|
||||||
|
color: white;
|
||||||
|
border: none;
|
||||||
|
padding: 4px 8px;
|
||||||
|
border-radius: 3px;
|
||||||
|
font-size: 11px;
|
||||||
|
}
|
||||||
|
QPushButton:hover { background-color: #b71c1c; }
|
||||||
|
""")
|
||||||
|
|
||||||
|
layout.addWidget(status_label)
|
||||||
|
layout.addWidget(info_label)
|
||||||
|
layout.addStretch()
|
||||||
|
layout.addWidget(toggle_btn)
|
||||||
|
layout.addWidget(remove_btn)
|
||||||
|
|
||||||
|
# 儲存引用
|
||||||
|
panel.connection = conn
|
||||||
|
panel.toggle_btn = toggle_btn
|
||||||
|
panel.status_label = status_label
|
||||||
|
|
||||||
|
return panel
|
||||||
|
|
||||||
|
def add_udp_panel(self, conn):
|
||||||
|
"""添加 UDP 連接面板到列表"""
|
||||||
|
panel = self.create_udp_connection_panel(conn)
|
||||||
|
self.udp_list_layout.addWidget(panel)
|
||||||
|
self.udp_connections.append(conn)
|
||||||
|
return panel
|
||||||
|
|
||||||
|
def add_ws_panel(self, conn):
|
||||||
|
"""添加 WebSocket 連接面板到列表"""
|
||||||
|
panel = self.create_ws_connection_panel(conn)
|
||||||
|
self.ws_list_layout.addWidget(panel)
|
||||||
|
self.ws_connections.append(conn)
|
||||||
|
return panel
|
||||||
|
|
||||||
|
def remove_udp_connection_from_list(self, conn):
|
||||||
|
"""從列表中移除 UDP 連接"""
|
||||||
|
if conn in self.udp_connections:
|
||||||
|
self.udp_connections.remove(conn)
|
||||||
|
|
||||||
|
def remove_ws_connection_from_list(self, conn):
|
||||||
|
"""從列表中移除 WebSocket 連接"""
|
||||||
|
if conn in self.ws_connections:
|
||||||
|
self.ws_connections.remove(conn)
|
||||||
@ -0,0 +1,662 @@
|
|||||||
|
from rclpy.node import Node
|
||||||
|
from PyQt6.QtCore import QObject, pyqtSignal
|
||||||
|
import math
|
||||||
|
import re
|
||||||
|
import threading
|
||||||
|
from threading import Lock
|
||||||
|
import asyncio
|
||||||
|
import websockets
|
||||||
|
import json
|
||||||
|
import socket
|
||||||
|
from pymavlink import mavutil
|
||||||
|
from geometry_msgs.msg import Point, Vector3
|
||||||
|
from sensor_msgs.msg import BatteryState, NavSatFix, Imu
|
||||||
|
from std_msgs.msg import Float64
|
||||||
|
from mavros_msgs.msg import State, VfrHud
|
||||||
|
from mavros_msgs.srv import CommandBool, CommandTOL
|
||||||
|
|
||||||
|
class DroneSignals(QObject):
|
||||||
|
update_signal = pyqtSignal(str, str, object) # (msg_type, drone_id, data)
|
||||||
|
|
||||||
|
class UDPMavlinkReceiver(threading.Thread):
|
||||||
|
"""UDP MAVLink 接收器"""
|
||||||
|
def __init__(self, ip, port, signals, connection_name):
|
||||||
|
super().__init__(daemon=True)
|
||||||
|
self.ip = ip
|
||||||
|
self.port = port
|
||||||
|
self.signals = signals
|
||||||
|
self.connection_name = connection_name
|
||||||
|
self.running = False
|
||||||
|
self.sock = None
|
||||||
|
|
||||||
|
def run(self):
|
||||||
|
"""執行 UDP 接收循環"""
|
||||||
|
self.running = True
|
||||||
|
try:
|
||||||
|
print(f"UDP MAVLink receiver started on {self.ip}:{self.port}")
|
||||||
|
|
||||||
|
# 創建 MAVLink 連接
|
||||||
|
mav = mavutil.mavlink_connection(f'udpin:{self.ip}:{self.port}')
|
||||||
|
while self.running:
|
||||||
|
try:
|
||||||
|
msg = mav.recv_match(blocking=True, timeout=1.0)
|
||||||
|
if msg is None:
|
||||||
|
continue
|
||||||
|
|
||||||
|
self.process_mavlink_message(msg)
|
||||||
|
|
||||||
|
except socket.timeout:
|
||||||
|
continue
|
||||||
|
except Exception as e:
|
||||||
|
print(f"Error receiving MAVLink message: {e}")
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
print(f"UDP receiver error: {e}")
|
||||||
|
finally:
|
||||||
|
if self.sock:
|
||||||
|
self.sock.close()
|
||||||
|
|
||||||
|
def process_mavlink_message(self, msg):
|
||||||
|
"""處理 MAVLink 訊息"""
|
||||||
|
try:
|
||||||
|
msg_type = msg.get_type()
|
||||||
|
system_id = msg.get_srcSystem()
|
||||||
|
drone_id = f"s8_{system_id}" # 使用 s8_ 前綴表示 UDP 來源
|
||||||
|
|
||||||
|
if msg_type == "HEARTBEAT":
|
||||||
|
mode = mavutil.mode_string_v10(msg)
|
||||||
|
armed = bool(msg.base_mode & 128)
|
||||||
|
self.signals.update_signal.emit('state', drone_id, {
|
||||||
|
'mode': mode,
|
||||||
|
'armed': armed
|
||||||
|
})
|
||||||
|
|
||||||
|
elif msg_type == "BATTERY_STATUS":
|
||||||
|
voltage = msg.voltages[0] / 1000
|
||||||
|
self.signals.update_signal.emit('battery', drone_id, {
|
||||||
|
'voltage': voltage
|
||||||
|
})
|
||||||
|
|
||||||
|
elif msg_type == "GLOBAL_POSITION_INT":
|
||||||
|
latitude = msg.lat / 1e7
|
||||||
|
longitude = msg.lon / 1e7
|
||||||
|
self.signals.update_signal.emit('gps', drone_id, {
|
||||||
|
'lat': latitude,
|
||||||
|
'lon': longitude
|
||||||
|
})
|
||||||
|
|
||||||
|
elif msg_type == "GPS_RAW_INT":
|
||||||
|
fix_type = msg.fix_type
|
||||||
|
|
||||||
|
elif msg_type == "LOCAL_POSITION_NED":
|
||||||
|
x = msg.y
|
||||||
|
y = msg.x
|
||||||
|
z = -msg.z
|
||||||
|
self.signals.update_signal.emit('local_pose', drone_id, {
|
||||||
|
'x': x,
|
||||||
|
'y': y,
|
||||||
|
'z': z
|
||||||
|
})
|
||||||
|
self.signals.update_signal.emit('altitude', drone_id, {
|
||||||
|
'altitude': z
|
||||||
|
})
|
||||||
|
self.signals.update_signal.emit('velocity', drone_id, {
|
||||||
|
'vx': msg.vx,
|
||||||
|
'vy': msg.vy,
|
||||||
|
'vz': msg.vz
|
||||||
|
})
|
||||||
|
|
||||||
|
elif msg_type == "ATTITUDE":
|
||||||
|
pitch = math.degrees(msg.pitch)
|
||||||
|
self.signals.update_signal.emit('attitude', drone_id, {
|
||||||
|
'pitch': pitch,
|
||||||
|
'roll': 0,
|
||||||
|
'yaw': 0,
|
||||||
|
'rates': (0, 0, 0)
|
||||||
|
})
|
||||||
|
|
||||||
|
elif msg_type == "VFR_HUD":
|
||||||
|
groundspeed = msg.groundspeed
|
||||||
|
heading = msg.heading
|
||||||
|
self.signals.update_signal.emit('hud', drone_id, {
|
||||||
|
'heading': heading,
|
||||||
|
'groundspeed': groundspeed
|
||||||
|
})
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
print(f"Error processing MAVLink message: {e}")
|
||||||
|
|
||||||
|
def stop(self):
|
||||||
|
"""停止接收器"""
|
||||||
|
self.running = False
|
||||||
|
|
||||||
|
class SerialMavlinkReceiver(threading.Thread):
|
||||||
|
"""串口 MAVLink 接收器"""
|
||||||
|
def __init__(self, port, baudrate, signals, connection_name):
|
||||||
|
super().__init__(daemon=True)
|
||||||
|
self.port = port
|
||||||
|
self.baudrate = baudrate
|
||||||
|
self.signals = signals
|
||||||
|
self.connection_name = connection_name
|
||||||
|
self.running = False
|
||||||
|
self.mav = None
|
||||||
|
|
||||||
|
def run(self):
|
||||||
|
"""執行串口接收循環"""
|
||||||
|
self.running = True
|
||||||
|
try:
|
||||||
|
print(f"Serial MAVLink receiver started on {self.port} at {self.baudrate} baud")
|
||||||
|
|
||||||
|
# 創建 MAVLink 串口連接
|
||||||
|
self.mav = mavutil.mavlink_connection(
|
||||||
|
self.port,
|
||||||
|
baud=self.baudrate,
|
||||||
|
source_system=255
|
||||||
|
)
|
||||||
|
|
||||||
|
print(f"Waiting for heartbeat from {self.port}...")
|
||||||
|
self.mav.wait_heartbeat()
|
||||||
|
print(f"Heartbeat received from system {self.mav.target_system}, component {self.mav.target_component}")
|
||||||
|
|
||||||
|
while self.running:
|
||||||
|
try:
|
||||||
|
msg = self.mav.recv_match(blocking=True, timeout=1.0)
|
||||||
|
if msg is None:
|
||||||
|
continue
|
||||||
|
|
||||||
|
self.process_mavlink_message(msg)
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
if self.running:
|
||||||
|
print(f"Error receiving MAVLink message from serial: {e}")
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
print(f"Serial receiver error: {e}")
|
||||||
|
finally:
|
||||||
|
if self.mav:
|
||||||
|
try:
|
||||||
|
self.mav.close()
|
||||||
|
except:
|
||||||
|
pass
|
||||||
|
|
||||||
|
def process_mavlink_message(self, msg):
|
||||||
|
"""處理 MAVLink 訊息"""
|
||||||
|
try:
|
||||||
|
msg_type = msg.get_type()
|
||||||
|
system_id = msg.get_srcSystem()
|
||||||
|
drone_id = f"s5_{system_id}" # 使用 serial_ 前綴表示串口來源
|
||||||
|
|
||||||
|
if msg_type == "HEARTBEAT":
|
||||||
|
mode = mavutil.mode_string_v10(msg)
|
||||||
|
armed = bool(msg.base_mode & 128)
|
||||||
|
self.signals.update_signal.emit('state', drone_id, {
|
||||||
|
'mode': mode,
|
||||||
|
'armed': armed
|
||||||
|
})
|
||||||
|
|
||||||
|
elif msg_type == "BATTERY_STATUS":
|
||||||
|
voltage = msg.voltages[0] / 1000
|
||||||
|
self.signals.update_signal.emit('battery', drone_id, {
|
||||||
|
'voltage': voltage
|
||||||
|
})
|
||||||
|
|
||||||
|
elif msg_type == "GLOBAL_POSITION_INT":
|
||||||
|
latitude = msg.lat / 1e7
|
||||||
|
longitude = msg.lon / 1e7
|
||||||
|
self.signals.update_signal.emit('gps', drone_id, {
|
||||||
|
'lat': latitude,
|
||||||
|
'lon': longitude
|
||||||
|
})
|
||||||
|
|
||||||
|
elif msg_type == "GPS_RAW_INT":
|
||||||
|
fix_type = msg.fix_type
|
||||||
|
|
||||||
|
elif msg_type == "LOCAL_POSITION_NED":
|
||||||
|
x = msg.y
|
||||||
|
y = msg.x
|
||||||
|
z = -msg.z
|
||||||
|
self.signals.update_signal.emit('local_pose', drone_id, {
|
||||||
|
'x': x,
|
||||||
|
'y': y,
|
||||||
|
'z': z
|
||||||
|
})
|
||||||
|
self.signals.update_signal.emit('altitude', drone_id, {
|
||||||
|
'altitude': z
|
||||||
|
})
|
||||||
|
self.signals.update_signal.emit('velocity', drone_id, {
|
||||||
|
'vx': msg.vx,
|
||||||
|
'vy': msg.vy,
|
||||||
|
'vz': msg.vz
|
||||||
|
})
|
||||||
|
|
||||||
|
elif msg_type == "ATTITUDE":
|
||||||
|
pitch = math.degrees(msg.pitch)
|
||||||
|
self.signals.update_signal.emit('attitude', drone_id, {
|
||||||
|
'pitch': pitch,
|
||||||
|
'roll': 0,
|
||||||
|
'yaw': 0,
|
||||||
|
'rates': (0, 0, 0)
|
||||||
|
})
|
||||||
|
|
||||||
|
elif msg_type == "VFR_HUD":
|
||||||
|
groundspeed = msg.groundspeed
|
||||||
|
heading = msg.heading
|
||||||
|
self.signals.update_signal.emit('hud', drone_id, {
|
||||||
|
'heading': heading,
|
||||||
|
'groundspeed': groundspeed
|
||||||
|
})
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
print(f"Error processing MAVLink message from serial: {e}")
|
||||||
|
|
||||||
|
def stop(self):
|
||||||
|
"""停止接收器"""
|
||||||
|
self.running = False
|
||||||
|
|
||||||
|
class WebSocketMavlinkReceiver(threading.Thread):
|
||||||
|
"""WebSocket MAVLink 接收器"""
|
||||||
|
def __init__(self, url, signals, connection_name):
|
||||||
|
super().__init__(daemon=True)
|
||||||
|
self.url = url
|
||||||
|
self.signals = signals
|
||||||
|
self.connection_name = connection_name
|
||||||
|
self.running = False
|
||||||
|
self.max_retries = 5
|
||||||
|
self.base_delay = 1.0
|
||||||
|
|
||||||
|
def run(self):
|
||||||
|
"""執行 WebSocket 接收循環"""
|
||||||
|
self.running = True
|
||||||
|
asyncio.set_event_loop(asyncio.new_event_loop())
|
||||||
|
asyncio.get_event_loop().run_until_complete(self.ws_client_loop())
|
||||||
|
|
||||||
|
async def ws_client_loop(self):
|
||||||
|
"""WebSocket 連接的主循環"""
|
||||||
|
retry_count = 0
|
||||||
|
|
||||||
|
print(f"Starting WebSocket client for {self.connection_name} at {self.url}")
|
||||||
|
|
||||||
|
while self.running and retry_count < self.max_retries:
|
||||||
|
try:
|
||||||
|
async with websockets.connect(self.url) as websocket:
|
||||||
|
print(f"WebSocket {self.connection_name} connected to {self.url}")
|
||||||
|
retry_count = 0 # 重置重試計數
|
||||||
|
|
||||||
|
async for message in websocket:
|
||||||
|
if not self.running:
|
||||||
|
break
|
||||||
|
|
||||||
|
try:
|
||||||
|
data = json.loads(message)
|
||||||
|
data['_connection_source'] = self.connection_name
|
||||||
|
self.process_websocket_message(data)
|
||||||
|
except json.JSONDecodeError as e:
|
||||||
|
print(f"WebSocket {self.connection_name} JSON decode error: {e}")
|
||||||
|
except Exception as e:
|
||||||
|
print(f"WebSocket {self.connection_name} message processing error: {e}")
|
||||||
|
|
||||||
|
except websockets.exceptions.ConnectionClosedError:
|
||||||
|
print(f"WebSocket {self.connection_name} connection closed")
|
||||||
|
if self.running:
|
||||||
|
retry_count += 1
|
||||||
|
if retry_count < self.max_retries:
|
||||||
|
delay = self.base_delay * (2 ** min(retry_count, 4))
|
||||||
|
print(f"Reconnecting in {delay}s...")
|
||||||
|
await asyncio.sleep(delay)
|
||||||
|
else:
|
||||||
|
break
|
||||||
|
except Exception as e:
|
||||||
|
retry_count += 1
|
||||||
|
if retry_count < self.max_retries and self.running:
|
||||||
|
delay = self.base_delay * (2 ** min(retry_count, 4))
|
||||||
|
print(f"WebSocket {self.connection_name} connection error: {e}, retrying in {delay}s (attempt {retry_count}/{self.max_retries})")
|
||||||
|
await asyncio.sleep(delay)
|
||||||
|
else:
|
||||||
|
break
|
||||||
|
|
||||||
|
print(f"WebSocket client {self.connection_name} stopped")
|
||||||
|
|
||||||
|
def process_websocket_message(self, data):
|
||||||
|
"""處理 WebSocket 訊息"""
|
||||||
|
try:
|
||||||
|
drone_id = data.get('system_id')
|
||||||
|
drone_id = f"s9_{drone_id}" if drone_id else None
|
||||||
|
if not drone_id:
|
||||||
|
return
|
||||||
|
|
||||||
|
# 模式
|
||||||
|
if 'mode' in data:
|
||||||
|
self.signals.update_signal.emit('state', drone_id, {
|
||||||
|
'mode': data['mode'],
|
||||||
|
})
|
||||||
|
|
||||||
|
# 電池
|
||||||
|
if 'battery' in data:
|
||||||
|
self.signals.update_signal.emit('battery', drone_id, {
|
||||||
|
'percentage': data['battery']
|
||||||
|
})
|
||||||
|
|
||||||
|
# 位置
|
||||||
|
if 'position' in data:
|
||||||
|
pos = data['position']
|
||||||
|
self.signals.update_signal.emit('gps', drone_id, {
|
||||||
|
'lat': pos.get('lat', 0),
|
||||||
|
'lon': pos.get('lon', 0)
|
||||||
|
})
|
||||||
|
|
||||||
|
# Local position - 設定 x, y 為 0.0
|
||||||
|
self.signals.update_signal.emit('local_pose', drone_id, {
|
||||||
|
'x': 0.0,
|
||||||
|
'y': 0.0,
|
||||||
|
'z': 0.0
|
||||||
|
})
|
||||||
|
|
||||||
|
# Altitude - 設定為 0.0
|
||||||
|
self.signals.update_signal.emit('altitude', drone_id, {
|
||||||
|
'altitude': 0.0
|
||||||
|
})
|
||||||
|
|
||||||
|
# 航向
|
||||||
|
if 'heading' in data:
|
||||||
|
self.signals.update_signal.emit('hud', drone_id, {
|
||||||
|
'heading': data['heading'],
|
||||||
|
'groundspeed': 0.0
|
||||||
|
})
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
print(f"WebSocket message processing error: {e}")
|
||||||
|
|
||||||
|
def stop(self):
|
||||||
|
"""停止接收器"""
|
||||||
|
self.running = False
|
||||||
|
|
||||||
|
class DroneMonitor(Node):
|
||||||
|
def __init__(self):
|
||||||
|
super().__init__('drone_monitor')
|
||||||
|
self.signals = DroneSignals()
|
||||||
|
self.drone_topics = {}
|
||||||
|
self.lock = Lock()
|
||||||
|
|
||||||
|
self.arm_clients = {}
|
||||||
|
self.takeoff_clients = {}
|
||||||
|
self.setpoint_pubs = {}
|
||||||
|
self.selected_drones = set()
|
||||||
|
self.latest_data = {}
|
||||||
|
|
||||||
|
# 定義需要過濾的模式
|
||||||
|
self.filtered_modes = ['Mode(0x000000c0)']
|
||||||
|
|
||||||
|
# WebSocket 接收器列表
|
||||||
|
self.ws_receivers = []
|
||||||
|
|
||||||
|
# 串口接收器列表
|
||||||
|
|
||||||
|
# ================================================================================
|
||||||
|
# 【新增】儲存 GPS 資料的字典
|
||||||
|
# ================================================================================
|
||||||
|
self.drone_gps = {} # {drone_id: {'lat': ..., 'lon': ..., 'alt': ...}}
|
||||||
|
# ================================================================================
|
||||||
|
self.serial_receivers = []
|
||||||
|
|
||||||
|
# 主题检测定时器
|
||||||
|
self.create_timer(1.0, self.scan_topics)
|
||||||
|
|
||||||
|
def scan_topics(self):
|
||||||
|
topics = self.get_topic_names_and_types()
|
||||||
|
drone_pattern = re.compile(r'/MavLinkBus/(s\d+_\d+)/(\w+)')
|
||||||
|
|
||||||
|
found_drones = set()
|
||||||
|
for topic_name, _ in topics:
|
||||||
|
if match := drone_pattern.match(topic_name):
|
||||||
|
drone_id, topic_type = match.groups()
|
||||||
|
found_drones.add(drone_id)
|
||||||
|
with self.lock:
|
||||||
|
self.drone_topics.setdefault(drone_id, set()).add(topic_type)
|
||||||
|
|
||||||
|
for drone_id in found_drones:
|
||||||
|
if not hasattr(self, f'drone_{drone_id}_subs'):
|
||||||
|
self.setup_drone(drone_id)
|
||||||
|
|
||||||
|
def setup_drone(self, drone_id):
|
||||||
|
base_topic = f'/MavLinkBus/{drone_id}'
|
||||||
|
|
||||||
|
# Add service clients
|
||||||
|
self.arm_clients[drone_id] = self.create_client(
|
||||||
|
CommandBool,
|
||||||
|
f'{base_topic}/cmd/arming'
|
||||||
|
)
|
||||||
|
self.takeoff_clients[drone_id] = self.create_client(
|
||||||
|
CommandTOL,
|
||||||
|
f'{base_topic}/cmd/takeoff'
|
||||||
|
)
|
||||||
|
|
||||||
|
# Add setpoint publisher
|
||||||
|
self.setpoint_pubs[drone_id] = self.create_publisher(
|
||||||
|
Point,
|
||||||
|
f'{base_topic}/setpoint_position/local',
|
||||||
|
10
|
||||||
|
)
|
||||||
|
|
||||||
|
subs = {
|
||||||
|
'attitude': self.create_subscription(
|
||||||
|
Imu,
|
||||||
|
f'{base_topic}/attitude',
|
||||||
|
lambda msg, did=drone_id: self.attitude_callback(did, msg),
|
||||||
|
10
|
||||||
|
),
|
||||||
|
'battery': self.create_subscription(
|
||||||
|
BatteryState,
|
||||||
|
f'{base_topic}/battery',
|
||||||
|
lambda msg, did=drone_id: self.battery_callback(did, msg),
|
||||||
|
10
|
||||||
|
),
|
||||||
|
'global': self.create_subscription(
|
||||||
|
NavSatFix,
|
||||||
|
f'{base_topic}/global_position/global',
|
||||||
|
lambda msg, did=drone_id: self.gps_callback(did, msg),
|
||||||
|
10
|
||||||
|
),
|
||||||
|
'rel_alt': self.create_subscription(
|
||||||
|
Float64,
|
||||||
|
f'{base_topic}/global_position/rel_alt',
|
||||||
|
lambda msg, did=drone_id: self.altitude_callback(did, msg),
|
||||||
|
10
|
||||||
|
),
|
||||||
|
'local_pose': self.create_subscription(
|
||||||
|
Point,
|
||||||
|
f'{base_topic}/local_position/pose',
|
||||||
|
lambda msg, did=drone_id: self.local_pose_callback(did, msg),
|
||||||
|
10
|
||||||
|
),
|
||||||
|
'local_vel': self.create_subscription(
|
||||||
|
Vector3,
|
||||||
|
f'{base_topic}/local_position/velocity',
|
||||||
|
lambda msg, did=drone_id: self.local_vel_callback(did, msg),
|
||||||
|
10
|
||||||
|
),
|
||||||
|
'loss_rate': self.create_subscription(
|
||||||
|
Float64,
|
||||||
|
f'{base_topic}/packet_loss_rate',
|
||||||
|
lambda msg, did=drone_id: self.loss_rate_callback(did, msg),
|
||||||
|
10
|
||||||
|
),
|
||||||
|
'state': self.create_subscription(
|
||||||
|
State,
|
||||||
|
f'{base_topic}/state',
|
||||||
|
lambda msg, did=drone_id: self.state_callback(did, msg),
|
||||||
|
10
|
||||||
|
),
|
||||||
|
'ping': self.create_subscription(
|
||||||
|
Float64,
|
||||||
|
f'{base_topic}/ping',
|
||||||
|
lambda msg, did=drone_id: self.ping_callback(did, msg),
|
||||||
|
10
|
||||||
|
),
|
||||||
|
'vfr_hud': self.create_subscription(
|
||||||
|
VfrHud,
|
||||||
|
f'{base_topic}/vfr_hud',
|
||||||
|
lambda msg, did=drone_id: self.hud_callback(did, msg),
|
||||||
|
10
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
setattr(self, f'drone_{drone_id}_subs', subs)
|
||||||
|
|
||||||
|
async def arm_drone(self, drone_id, arm):
|
||||||
|
if drone_id not in self.arm_clients:
|
||||||
|
return False
|
||||||
|
|
||||||
|
client = self.arm_clients[drone_id]
|
||||||
|
if not client.wait_for_service(timeout_sec=1.0):
|
||||||
|
return False
|
||||||
|
|
||||||
|
request = CommandBool.Request()
|
||||||
|
request.value = arm
|
||||||
|
|
||||||
|
future = client.call_async(request)
|
||||||
|
try:
|
||||||
|
response = await future
|
||||||
|
return response.success
|
||||||
|
except Exception as e:
|
||||||
|
self.get_logger().error(f'Arm service call failed: {e}')
|
||||||
|
return False
|
||||||
|
|
||||||
|
async def takeoff_drone(self, drone_id, altitude=10.0):
|
||||||
|
if drone_id not in self.takeoff_clients:
|
||||||
|
return False
|
||||||
|
|
||||||
|
client = self.takeoff_clients[drone_id]
|
||||||
|
if not client.wait_for_service(timeout_sec=1.0):
|
||||||
|
return False
|
||||||
|
|
||||||
|
request = CommandTOL.Request()
|
||||||
|
request.altitude = altitude
|
||||||
|
request.min_pitch = 0.0
|
||||||
|
request.yaw = 0.0
|
||||||
|
|
||||||
|
future = client.call_async(request)
|
||||||
|
try:
|
||||||
|
response = await future
|
||||||
|
return response.success
|
||||||
|
except Exception as e:
|
||||||
|
self.get_logger().error(f'Takeoff service call failed: {e}')
|
||||||
|
return False
|
||||||
|
|
||||||
|
def send_setpoint(self, drone_id, x, y, z):
|
||||||
|
"""Send setpoint position command"""
|
||||||
|
if drone_id not in self.setpoint_pubs:
|
||||||
|
return False
|
||||||
|
|
||||||
|
msg = Point()
|
||||||
|
msg.x = float(x)
|
||||||
|
msg.y = float(y)
|
||||||
|
msg.z = float(z)
|
||||||
|
|
||||||
|
self.setpoint_pubs[drone_id].publish(msg)
|
||||||
|
return True
|
||||||
|
|
||||||
|
def quaternion_to_euler(self, q):
|
||||||
|
sinr_cosp = 2 * (q.w * q.x + q.y * q.z)
|
||||||
|
cosr_cosp = 1 - 2 * (q.x**2 + q.y**2)
|
||||||
|
roll = math.atan2(sinr_cosp, cosr_cosp)
|
||||||
|
|
||||||
|
sinp = 2 * (q.w * q.y - q.z * q.x)
|
||||||
|
pitch = math.asin(sinp) if abs(sinp) < 1 else math.copysign(math.pi/2, sinp)
|
||||||
|
|
||||||
|
siny_cosp = 2 * (q.w * q.z + q.x * q.y)
|
||||||
|
cosy_cosp = 1 - 2 * (q.y**2 + q.z**2)
|
||||||
|
yaw = math.atan2(siny_cosp, cosy_cosp)
|
||||||
|
|
||||||
|
return math.degrees(roll), math.degrees(pitch), math.degrees(yaw)
|
||||||
|
|
||||||
|
# callbacks
|
||||||
|
def attitude_callback(self, drone_id, msg):
|
||||||
|
if hasattr(msg, 'orientation'):
|
||||||
|
roll, pitch, yaw = self.quaternion_to_euler(msg.orientation)
|
||||||
|
self.latest_data[(drone_id, 'attitude')] = {
|
||||||
|
'roll': roll,
|
||||||
|
'pitch': pitch,
|
||||||
|
'yaw': yaw,
|
||||||
|
'rates': (msg.angular_velocity.x,
|
||||||
|
msg.angular_velocity.y,
|
||||||
|
msg.angular_velocity.z)
|
||||||
|
}
|
||||||
|
|
||||||
|
def battery_callback(self, drone_id, msg):
|
||||||
|
self.latest_data[(drone_id, 'battery')] = {
|
||||||
|
'voltage': msg.voltage
|
||||||
|
}
|
||||||
|
|
||||||
|
def state_callback(self, drone_id, msg):
|
||||||
|
mode = msg.mode
|
||||||
|
if mode in self.filtered_modes:
|
||||||
|
return
|
||||||
|
self.latest_data[(drone_id, 'state')] = {
|
||||||
|
'mode': msg.mode,
|
||||||
|
'armed': msg.armed
|
||||||
|
}
|
||||||
|
|
||||||
|
def gps_callback(self, drone_id, msg):
|
||||||
|
self.latest_data[(drone_id, 'gps')] = {
|
||||||
|
'lat': msg.latitude,
|
||||||
|
'lon': msg.longitude,
|
||||||
|
'alt': msg.altitude
|
||||||
|
}
|
||||||
|
|
||||||
|
# ================================================================================
|
||||||
|
# 【新增】儲存 GPS 資料到 drone_gps 字典
|
||||||
|
# ================================================================================
|
||||||
|
self.drone_gps[drone_id] = {
|
||||||
|
'lat': msg.latitude,
|
||||||
|
'lon': msg.longitude,
|
||||||
|
'alt': msg.altitude
|
||||||
|
}
|
||||||
|
# ================================================================================
|
||||||
|
|
||||||
|
def local_vel_callback(self, drone_id, msg):
|
||||||
|
self.latest_data[(drone_id, 'velocity')] = {
|
||||||
|
'vx': msg.x,
|
||||||
|
'vy': msg.y,
|
||||||
|
'vz': msg.z
|
||||||
|
}
|
||||||
|
|
||||||
|
def altitude_callback(self, drone_id, msg):
|
||||||
|
self.latest_data[(drone_id, 'altitude')] = {
|
||||||
|
'altitude': msg.data
|
||||||
|
}
|
||||||
|
|
||||||
|
def local_pose_callback(self, drone_id, msg):
|
||||||
|
self.latest_data[(drone_id, 'local_pose')] = {
|
||||||
|
'x': msg.x,
|
||||||
|
'y': msg.y,
|
||||||
|
'z': msg.z
|
||||||
|
}
|
||||||
|
|
||||||
|
def hud_callback(self, drone_id, msg):
|
||||||
|
self.latest_data[(drone_id, 'hud')] = {
|
||||||
|
'airspeed': msg.airspeed,
|
||||||
|
'groundspeed': msg.groundspeed,
|
||||||
|
'heading': msg.heading,
|
||||||
|
'throttle': msg.throttle,
|
||||||
|
'alt': msg.altitude,
|
||||||
|
'climb': msg.climb
|
||||||
|
}
|
||||||
|
|
||||||
|
def loss_rate_callback(self, drone_id, msg):
|
||||||
|
self.latest_data[(drone_id, 'loss_rate')] = {
|
||||||
|
'loss_rate': msg.data
|
||||||
|
}
|
||||||
|
|
||||||
|
def ping_callback(self, drone_id, msg):
|
||||||
|
self.latest_data[(drone_id, 'ping')] = {
|
||||||
|
'ping': msg.data
|
||||||
|
}
|
||||||
|
|
||||||
|
def start_serial_connection(self, port='/dev/ttyUSB0', baudrate=57600):
|
||||||
|
"""啟動串口 MAVLink 連接"""
|
||||||
|
connection_name = f"Serial_{port.replace('/', '_')}"
|
||||||
|
receiver = SerialMavlinkReceiver(port, baudrate, self.signals, connection_name)
|
||||||
|
receiver.start()
|
||||||
|
self.serial_receivers.append(receiver)
|
||||||
|
print(f"Started serial connection on {port} at {baudrate} baud")
|
||||||
|
return receiver
|
||||||
File diff suppressed because it is too large
Load Diff
@ -0,0 +1,89 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
from PyQt6.QtWidgets import QTableWidget, QTableWidgetItem, QHeaderView, QLabel
|
||||||
|
from PyQt6.QtCore import Qt
|
||||||
|
|
||||||
|
class OverviewTable(QTableWidget):
|
||||||
|
"""總覽表格,顯示所有無人機的狀態資訊"""
|
||||||
|
|
||||||
|
def __init__(self, info_types, info_type_map, parent=None):
|
||||||
|
super().__init__(parent)
|
||||||
|
|
||||||
|
self.info_types = info_types
|
||||||
|
self.info_type_map = info_type_map
|
||||||
|
self.drones = {} # 存儲無人機面板的引用
|
||||||
|
|
||||||
|
# 初始化表格
|
||||||
|
self.setColumnCount(1)
|
||||||
|
self.setRowCount(len(self.info_types))
|
||||||
|
self.setHorizontalHeaderLabels(["資訊"])
|
||||||
|
header = self.horizontalHeader()
|
||||||
|
header.setSectionResizeMode(0, QHeaderView.ResizeMode.ResizeToContents)
|
||||||
|
self.verticalHeader().setVisible(False)
|
||||||
|
|
||||||
|
# 設置第一列的資訊類型
|
||||||
|
for i, txt in enumerate(self.info_types):
|
||||||
|
item = QTableWidgetItem(txt)
|
||||||
|
item.setFlags(Qt.ItemFlag.ItemIsEnabled)
|
||||||
|
item.setTextAlignment(Qt.AlignmentFlag.AlignCenter)
|
||||||
|
self.setItem(i, 0, item)
|
||||||
|
|
||||||
|
def set_drones(self, drones):
|
||||||
|
"""設置無人機面板字典的引用"""
|
||||||
|
self.drones = drones
|
||||||
|
|
||||||
|
def update_table(self, drone_id=None, field=None, value=None):
|
||||||
|
"""更新總覽表格
|
||||||
|
|
||||||
|
Args:
|
||||||
|
drone_id: 無人機 ID
|
||||||
|
field: 欄位名稱 (如 'mode', 'altitude' 等)
|
||||||
|
value: 要更新的值
|
||||||
|
"""
|
||||||
|
# 更新特定儲存格
|
||||||
|
if drone_id and field and value:
|
||||||
|
if drone_id not in self.drones:
|
||||||
|
return
|
||||||
|
|
||||||
|
col = 1 + list(self.drones.keys()).index(drone_id)
|
||||||
|
row = self.info_type_map.get(field, -1)
|
||||||
|
|
||||||
|
if row == -1:
|
||||||
|
return # 無效的欄位
|
||||||
|
|
||||||
|
item = self.item(row, col)
|
||||||
|
if not item:
|
||||||
|
item = QTableWidgetItem()
|
||||||
|
self.setItem(row, col, item)
|
||||||
|
|
||||||
|
item.setText(value)
|
||||||
|
item.setTextAlignment(Qt.AlignmentFlag.AlignCenter)
|
||||||
|
|
||||||
|
# 如果沒有指定更新,刷新整個表格
|
||||||
|
if drone_id is None:
|
||||||
|
self.refresh_all()
|
||||||
|
|
||||||
|
def refresh_all(self):
|
||||||
|
"""刷新整個表格"""
|
||||||
|
cols = 1 + len(self.drones)
|
||||||
|
self.setColumnCount(cols)
|
||||||
|
headers = ["資訊"] + list(self.drones.keys())
|
||||||
|
self.setHorizontalHeaderLabels(headers)
|
||||||
|
|
||||||
|
for col, did in enumerate(self.drones, start=1):
|
||||||
|
panel = self.drones[did]
|
||||||
|
for field, row in self.info_type_map.items():
|
||||||
|
lbl = panel.findChild(QLabel, f"{did}_{field}")
|
||||||
|
val = lbl.text() if lbl else "--"
|
||||||
|
val_item = QTableWidgetItem(val)
|
||||||
|
val_item.setTextAlignment(Qt.AlignmentFlag.AlignCenter)
|
||||||
|
self.setItem(row, col, val_item)
|
||||||
|
|
||||||
|
def add_drone_column(self, drone_id):
|
||||||
|
"""當新增無人機時,添加一列"""
|
||||||
|
if drone_id in self.drones:
|
||||||
|
self.refresh_all()
|
||||||
|
|
||||||
|
def remove_drone_column(self, drone_id):
|
||||||
|
"""當移除無人機時,刷新表格"""
|
||||||
|
if drone_id in self.drones:
|
||||||
|
self.refresh_all()
|
||||||
Loading…
Reference in New Issue