|
|
#!/usr/bin/env python3
|
|
|
import rclpy
|
|
|
from PyQt6.QtWidgets import (QApplication, QMainWindow, QVBoxLayout, QHBoxLayout,
|
|
|
QWidget, QLabel, QSplitter, QScrollArea,
|
|
|
QSizePolicy, QTabWidget, QTableWidget, QTableWidgetItem,
|
|
|
QHeaderView, QPushButton, QCheckBox, QLineEdit,
|
|
|
QComboBox, QDialog)
|
|
|
from PyQt6.QtCore import Qt, QTimer
|
|
|
from PyQt6.QtGui import QColor
|
|
|
import sys
|
|
|
import asyncio
|
|
|
import json
|
|
|
import subprocess
|
|
|
import time
|
|
|
import traceback
|
|
|
|
|
|
# 導入分離的類別
|
|
|
from communication import DroneMonitor, UDPMavlinkReceiver, WebSocketMavlinkReceiver
|
|
|
from map_layout import DroneMap
|
|
|
from drone_panel import DronePanel, SocketGroupPanel
|
|
|
from comm_panel import CommPanel
|
|
|
from overview_table import OverviewTable
|
|
|
|
|
|
# ================================================================================
|
|
|
# 導入任務規劃器、執行器、發送器、群組
|
|
|
# ================================================================================
|
|
|
from mission_planner import FormationPlanner, MissionType
|
|
|
from command_sender import MavlinkSender, Ros2CommandSender
|
|
|
from mission_executor import MissionExecutor, MissionState
|
|
|
from mission_group import (
|
|
|
MissionGroup, GroupPanel, DroneAssignDialog, GROUP_COLORS
|
|
|
)
|
|
|
# ================================================================================
|
|
|
|
|
|
class ControlStationUI(QMainWindow):
|
|
|
VERSION = '2.0.8'
|
|
|
|
|
|
def __init__(self):
|
|
|
super().__init__()
|
|
|
self.setWindowTitle(f'GCS v{self.VERSION}')
|
|
|
self.resize(1400, 900)
|
|
|
|
|
|
# 初始化消息隊列(用於線程安全的 GUI 更新)
|
|
|
import queue
|
|
|
self.message_queue = queue.Queue()
|
|
|
|
|
|
# 初始化ROS2 Monitor(ROS2 本身在 main() 中已初始化)
|
|
|
self.monitor = DroneMonitor()
|
|
|
self.monitor.signals.update_signal.connect(self.update_ui)
|
|
|
|
|
|
# ROS执行器配置
|
|
|
self.executor = rclpy.executors.SingleThreadedExecutor()
|
|
|
self.executor.add_node(self.monitor)
|
|
|
|
|
|
# 將執行器註冊到 DroneMonitor,以便動態創建的 CommandLongClient 能被添加
|
|
|
self.monitor.executor = self.executor
|
|
|
|
|
|
# 定时处理ROS事件
|
|
|
self.timer = QTimer()
|
|
|
self.timer.timeout.connect(self.spin_ros)
|
|
|
self.timer.start(10)
|
|
|
|
|
|
# 驅動 asyncio 事件循環的定時器(新增 - 關鍵!)
|
|
|
# 這允許異步任務(如 arm_drone)能夠執行
|
|
|
self.asyncio_timer = QTimer()
|
|
|
self.asyncio_timer.timeout.connect(self._spin_asyncio)
|
|
|
self.asyncio_timer.start(10) # 每 10ms 驅動一次 asyncio
|
|
|
|
|
|
# 初始化 panel 和 map 更新(10Hz)
|
|
|
self.panel_map_timer = QTimer()
|
|
|
self.panel_map_timer.timeout.connect(self._update_panel_and_map)
|
|
|
self.panel_map_timer.start(100) # 10Hz
|
|
|
|
|
|
# 消息隊列處理定時器(來自線程的 GUI 更新)
|
|
|
self.msg_queue_timer = QTimer()
|
|
|
self.msg_queue_timer.timeout.connect(self._process_message_queue)
|
|
|
self.msg_queue_timer.start(50) # 每 50ms 檢查一次
|
|
|
|
|
|
# 快取消息數據,以便在沒有新消息時使用上一次的值
|
|
|
self._message_cache = {}
|
|
|
|
|
|
# 初始化UI
|
|
|
self.drones = {}
|
|
|
self.socket_groups = {}
|
|
|
self.socket_types = {}
|
|
|
|
|
|
self.socket_colors = {
|
|
|
'0': '#00BFFF', # 天藍色 (DeepSkyBlue)
|
|
|
'1': '#FFD700', # 金色 (Gold)
|
|
|
'2': '#FF6969', # 淺紅色 (Light Red)
|
|
|
'3': '#FF69B4', # 熱粉紅 (HotPink)
|
|
|
'4': '#00FA9A', # 中春綠 (MediumSpringGreen)
|
|
|
'5': '#9370DB', # 中紫色 (MediumPurple) - 串口
|
|
|
'6': '#FFA500', # 橙色 (Orange)
|
|
|
'7': '#20B2AA', # 淺海綠 (LightSeaGreen)
|
|
|
'8': '#7CFC00', # 草綠色 (LawnGreen)
|
|
|
'9': '#FF8C00', # 深橙色 (DarkOrange)
|
|
|
'default': '#AAAAAA' # 灰色
|
|
|
}
|
|
|
|
|
|
self.drone_positions = {}
|
|
|
self.drone_headings = {}
|
|
|
# 初始化地圖
|
|
|
self.drone_map = DroneMap()
|
|
|
# 初始化連接列表
|
|
|
self.udp_receivers = []
|
|
|
self.udp_connections = []
|
|
|
self.ws_connections = []
|
|
|
self.serial_receivers = []
|
|
|
self.serial_connections = []
|
|
|
|
|
|
# ================================================================================
|
|
|
# 初始化任務規劃器
|
|
|
# ================================================================================
|
|
|
self.mission_planner = FormationPlanner(
|
|
|
spacing=5.0, # 5 公尺間距
|
|
|
base_altitude=10.0, # 基準高度 10 公尺
|
|
|
altitude_diff=2.0 # 高低差 2 公尺
|
|
|
)
|
|
|
# ================================================================================
|
|
|
|
|
|
# ================================================================================
|
|
|
# 初始化指令發送器(所有群組共用)
|
|
|
# ================================================================================
|
|
|
# 走 ROS2 /fc_network/vehicle/pos_global_int service;per-drone client + asyncio polling
|
|
|
self.command_sender = Ros2CommandSender(monitor=self.monitor)
|
|
|
# ================================================================================
|
|
|
|
|
|
# ================================================================================
|
|
|
# 多任務群組管理
|
|
|
# ================================================================================
|
|
|
self.mission_groups = {} # group_id → MissionGroup
|
|
|
self.group_panels = {} # group_id → GroupPanel
|
|
|
self.active_group_id = None # 當前 active 的 group
|
|
|
self._group_counter = 0 # 用來產生 group_id
|
|
|
self._pending_box_assign = None # 框選後直接分配到的 group_id
|
|
|
# ================================================================================
|
|
|
|
|
|
self.init_ui()
|
|
|
|
|
|
def init_ui(self):
|
|
|
main_splitter = QSplitter(Qt.Orientation.Horizontal)
|
|
|
|
|
|
# 左側 TabWidget
|
|
|
self.left_tab = QTabWidget()
|
|
|
|
|
|
# — 分頁 1:Drone Panel
|
|
|
self.drone_panel_container = QWidget()
|
|
|
self.drone_panel_layout = QVBoxLayout(self.drone_panel_container)
|
|
|
self.drone_panel_layout.setAlignment(Qt.AlignmentFlag.AlignTop)
|
|
|
self.drone_panel_layout.setSpacing(0)
|
|
|
self.drone_panel_layout.setContentsMargins(10, 10, 10, 10)
|
|
|
|
|
|
scroll = QScrollArea()
|
|
|
scroll.setWidget(self.drone_panel_container)
|
|
|
scroll.setWidgetResizable(True)
|
|
|
self.left_tab.addTab(scroll, "無人載具")
|
|
|
|
|
|
# — 分頁 2:Overview Table
|
|
|
self.overview_table = OverviewTable()
|
|
|
self.left_tab.addTab(self.overview_table, "總覽")
|
|
|
|
|
|
# — 分頁 3:通訊設定
|
|
|
self.comm_panel = CommPanel()
|
|
|
self.comm_panel.udp_connection_added.connect(self.handle_udp_connection_added)
|
|
|
self.comm_panel.ws_connection_added.connect(self.handle_ws_connection_added)
|
|
|
self.comm_panel.serial_connection_added.connect(self.handle_serial_connection_added)
|
|
|
self.comm_panel.udp_connection_toggled.connect(self.toggle_udp_connection)
|
|
|
self.comm_panel.ws_connection_toggled.connect(self.toggle_ws_connection)
|
|
|
self.comm_panel.serial_connection_toggled.connect(self.toggle_serial_connection)
|
|
|
self.comm_panel.udp_connection_removed.connect(self.remove_udp_connection)
|
|
|
self.comm_panel.ws_connection_removed.connect(self.remove_ws_connection)
|
|
|
self.comm_panel.serial_connection_removed.connect(self.remove_serial_connection)
|
|
|
self.comm_panel.status_message.connect(lambda msg, timeout: self.statusBar().showMessage(msg, timeout))
|
|
|
|
|
|
self.left_tab.addTab(self.comm_panel, "通訊")
|
|
|
|
|
|
# 右侧容器
|
|
|
right_container = QWidget()
|
|
|
right_layout = QVBoxLayout(right_container)
|
|
|
right_layout.setContentsMargins(10, 10, 10, 10)
|
|
|
right_layout.setSpacing(10)
|
|
|
|
|
|
# ========== 任務群組 Tab ==========
|
|
|
group_header = QHBoxLayout()
|
|
|
|
|
|
# 標題 + 收起/展開按鈕
|
|
|
title_layout = QHBoxLayout()
|
|
|
group_title = QLabel("任務群組")
|
|
|
group_title.setStyleSheet(
|
|
|
"color: #DDD; font-size: 14px; font-weight: bold; padding: 2px;")
|
|
|
title_layout.addWidget(group_title)
|
|
|
|
|
|
# 收起/展開按鈕
|
|
|
self.toggle_group_btn = QPushButton("▼")
|
|
|
self.toggle_group_btn.setStyleSheet("""
|
|
|
QPushButton { background-color: #555; color: white; border: none;
|
|
|
padding: 3px 8px; border-radius: 3px; font-size: 12px; font-weight: bold;
|
|
|
min-width: 30px; max-width: 30px; }
|
|
|
QPushButton:hover { background-color: #666; }
|
|
|
""")
|
|
|
self.toggle_group_btn.setToolTip("收起/展開任務群組")
|
|
|
self.toggle_group_btn.clicked.connect(self._toggle_group_panel)
|
|
|
title_layout.addWidget(self.toggle_group_btn)
|
|
|
title_layout.addStretch()
|
|
|
|
|
|
group_header.addLayout(title_layout)
|
|
|
group_header.addStretch()
|
|
|
|
|
|
clear_traj_btn = QPushButton("清除軌跡")
|
|
|
clear_traj_btn.setStyleSheet("""
|
|
|
QPushButton { background-color: #EF5350; color: white; border: none;
|
|
|
padding: 5px 12px; border-radius: 4px; font-size: 12px; font-weight: bold; }
|
|
|
QPushButton:hover { background-color: #E53935; }
|
|
|
""")
|
|
|
clear_traj_btn.clicked.connect(self.drone_map.clear_trajectories)
|
|
|
group_header.addWidget(clear_traj_btn)
|
|
|
|
|
|
right_layout.addLayout(group_header)
|
|
|
|
|
|
self.group_tab_widget = QTabWidget()
|
|
|
self.group_tab_widget.setStyleSheet("""
|
|
|
QTabWidget::pane { border: 1px solid #444; background-color: #2B2B2B; }
|
|
|
QTabBar::tab {
|
|
|
background-color: #333; color: #AAA; border: 1px solid #444;
|
|
|
padding: 5px 12px; margin-right: 2px; border-top-left-radius: 4px;
|
|
|
border-top-right-radius: 4px; font-size: 12px;
|
|
|
}
|
|
|
QTabBar::tab:selected { background-color: #2B2B2B; color: #FFF; border-bottom-color: #2B2B2B; }
|
|
|
QTabBar::tab:hover { background-color: #3A3A3A; }
|
|
|
""")
|
|
|
self.group_tab_widget.setFixedHeight(150)
|
|
|
self.group_tab_widget.currentChanged.connect(self._on_group_tab_changed)
|
|
|
right_layout.addWidget(self.group_tab_widget)
|
|
|
|
|
|
# 🌟 新增:保存群組面板的展開狀態
|
|
|
self.group_panel_expanded = True
|
|
|
|
|
|
# 預設建立 Group A
|
|
|
self._add_mission_group()
|
|
|
|
|
|
# 添加地圖
|
|
|
right_layout.addWidget(self.drone_map.get_widget())
|
|
|
self.drone_map.get_gps_signal().connect(self.handle_map_click)
|
|
|
self.drone_map.get_drone_clicked_signal().connect(self.handle_drone_clicked)
|
|
|
self.drone_map.get_clear_all_drone_selection_signal().connect(self.handle_clear_all_drone_selection)
|
|
|
self.drone_map.get_toggle_select_all_drones_signal().connect(self.handle_toggle_select_all_drones)
|
|
|
|
|
|
# ================================================================================
|
|
|
# 連接地圖信號(任務模式改由 Group Tab 控制,不再從地圖下拉選單)
|
|
|
# ================================================================================
|
|
|
self.drone_map.get_rectangle_selected_signal().connect(self.handle_rectangle_selected)
|
|
|
self.drone_map.get_route_confirmed_signal().connect(self.handle_route_confirmed)
|
|
|
self.drone_map.get_drone_box_selected_signal().connect(self._handle_drone_box_selected)
|
|
|
# ================================================================================
|
|
|
|
|
|
main_splitter.addWidget(self.left_tab)
|
|
|
main_splitter.addWidget(right_container)
|
|
|
main_splitter.setSizes([400, 1000])
|
|
|
|
|
|
self.setCentralWidget(main_splitter)
|
|
|
|
|
|
|
|
|
# ================================================================================
|
|
|
# 連線管理
|
|
|
# ================================================================================
|
|
|
|
|
|
def handle_udp_connection_added(self, ip, port):
|
|
|
new_conn = {'name': f'UDP {len(self.udp_connections) + 1}', 'ip': ip, 'port': port, 'enabled': True}
|
|
|
receiver = UDPMavlinkReceiver(ip, port, self.monitor.signals, new_conn['name'], self.monitor)
|
|
|
receiver.start()
|
|
|
self.udp_receivers.append(receiver)
|
|
|
new_conn['receiver'] = receiver
|
|
|
self.udp_connections.append(new_conn)
|
|
|
self.comm_panel.add_udp_panel(new_conn)
|
|
|
self.statusBar().showMessage(f"已添加 UDP 連接: {ip}:{port}", 3000)
|
|
|
|
|
|
def handle_ws_connection_added(self, url):
|
|
|
new_conn = {'name': f'WS {len(self.ws_connections) + 1}', 'url': url, 'enabled': True}
|
|
|
receiver = WebSocketMavlinkReceiver(url, self.monitor.signals, new_conn['name'], self.monitor)
|
|
|
receiver.start()
|
|
|
self.monitor.ws_receivers.append(receiver)
|
|
|
new_conn['receiver'] = receiver
|
|
|
self.ws_connections.append(new_conn)
|
|
|
self.comm_panel.add_ws_panel(new_conn)
|
|
|
self.statusBar().showMessage(f"已添加 WebSocket 連接: {url}", 3000)
|
|
|
|
|
|
def create_drone_panel(self, drone_id):
|
|
|
panel = DronePanel(drone_id)
|
|
|
panel.mode_change_requested.connect(self.handle_mode_change)
|
|
|
panel.arm_requested.connect(self.handle_arm)
|
|
|
panel.takeoff_requested.connect(self.handle_takeoff)
|
|
|
panel.setpoint_requested.connect(self.handle_single_setpoint)
|
|
|
panel.selection_changed.connect(self.handle_drone_selection)
|
|
|
return panel
|
|
|
|
|
|
def toggle_ws_connection(self, conn, btn, status_label):
|
|
|
if conn.get('enabled', False):
|
|
|
if 'receiver' in conn and conn['receiver']:
|
|
|
conn['receiver'].stop()
|
|
|
conn['enabled'] = False
|
|
|
btn.setText("啟動")
|
|
|
status_label.setStyleSheet("color: #888; font-size: 16px;")
|
|
|
status_label.setToolTip("已停止")
|
|
|
self.statusBar().showMessage(f"已停止 WebSocket 連接: {conn['url']}", 3000)
|
|
|
else:
|
|
|
receiver = WebSocketMavlinkReceiver(conn['url'], self.monitor.signals, conn['name'])
|
|
|
receiver.start()
|
|
|
self.monitor.ws_receivers.append(receiver)
|
|
|
conn['receiver'] = receiver
|
|
|
conn['enabled'] = True
|
|
|
btn.setText("停止")
|
|
|
status_label.setStyleSheet("color: #4CAF50; font-size: 16px;")
|
|
|
status_label.setToolTip("運行中")
|
|
|
self.statusBar().showMessage(f"已啟動 WebSocket 連接: {conn['url']}", 3000)
|
|
|
|
|
|
def remove_ws_connection(self, conn, panel):
|
|
|
if 'receiver' in conn and conn['receiver']:
|
|
|
conn['receiver'].stop()
|
|
|
if conn['receiver'] in self.monitor.ws_receivers:
|
|
|
self.monitor.ws_receivers.remove(conn['receiver'])
|
|
|
if conn in self.ws_connections:
|
|
|
self.ws_connections.remove(conn)
|
|
|
self.comm_panel.remove_ws_connection_from_list(conn)
|
|
|
panel.setParent(None)
|
|
|
panel.deleteLater()
|
|
|
self.statusBar().showMessage(f"已移除 WebSocket 連接: {conn['url']}", 3000)
|
|
|
|
|
|
def toggle_udp_connection(self, conn, btn, status_label):
|
|
|
if conn.get('enabled', False):
|
|
|
if 'receiver' in conn and conn['receiver']:
|
|
|
conn['receiver'].stop()
|
|
|
conn['enabled'] = False
|
|
|
btn.setText("啟動")
|
|
|
status_label.setStyleSheet("color: #888; font-size: 16px;")
|
|
|
status_label.setToolTip("已停止")
|
|
|
self.statusBar().showMessage(f"已停止 UDP 連接: {conn['ip']}:{conn['port']}", 3000)
|
|
|
else:
|
|
|
receiver = UDPMavlinkReceiver(conn['ip'], conn['port'], self.monitor.signals, conn['name'])
|
|
|
receiver.start()
|
|
|
self.udp_receivers.append(receiver)
|
|
|
conn['receiver'] = receiver
|
|
|
conn['enabled'] = True
|
|
|
btn.setText("停止")
|
|
|
status_label.setStyleSheet("color: #4CAF50; font-size: 16px;")
|
|
|
status_label.setToolTip("運行中")
|
|
|
self.statusBar().showMessage(f"已啟動 UDP 連接: {conn['ip']}:{conn['port']}", 3000)
|
|
|
|
|
|
def remove_udp_connection(self, conn, panel):
|
|
|
if 'receiver' in conn and conn['receiver']:
|
|
|
conn['receiver'].stop()
|
|
|
if conn['receiver'] in self.udp_receivers:
|
|
|
self.udp_receivers.remove(conn['receiver'])
|
|
|
if conn in self.udp_connections:
|
|
|
self.udp_connections.remove(conn)
|
|
|
self.comm_panel.remove_udp_connection_from_list(conn)
|
|
|
panel.setParent(None)
|
|
|
panel.deleteLater()
|
|
|
self.statusBar().showMessage(f"已移除 UDP 連接: {conn['ip']}:{conn['port']}", 3000)
|
|
|
|
|
|
def handle_serial_connection_added(self, port, baudrate):
|
|
|
conn = {'name': 'Serial', 'port': port, 'baudrate': baudrate, 'enabled': False, 'receiver': None}
|
|
|
self.serial_connections.append(conn)
|
|
|
self.comm_panel.add_serial_panel(conn)
|
|
|
self.statusBar().showMessage(f"已添加 Serial 連接: {port} @ {baudrate}", 3000)
|
|
|
|
|
|
def toggle_serial_connection(self, conn, btn, status_label):
|
|
|
if conn.get('enabled', False):
|
|
|
if conn.get('receiver'):
|
|
|
conn['receiver'].stop()
|
|
|
conn['receiver'] = None
|
|
|
conn['enabled'] = False
|
|
|
btn.setText("啟動")
|
|
|
status_label.setStyleSheet("color: #888; font-size: 16px;")
|
|
|
status_label.setToolTip("已停止")
|
|
|
self.statusBar().showMessage(f"已停止 Serial 連接: {conn['port']}", 3000)
|
|
|
else:
|
|
|
try:
|
|
|
receiver = self.monitor.start_serial_connection(conn['port'], conn['baudrate'])
|
|
|
conn['receiver'] = receiver
|
|
|
conn['enabled'] = True
|
|
|
btn.setText("停止")
|
|
|
status_label.setStyleSheet("color: #4CAF50; font-size: 16px;")
|
|
|
status_label.setToolTip("運行中")
|
|
|
self.statusBar().showMessage(f"已啟動 Serial 連接: {conn['port']}", 3000)
|
|
|
except Exception as e:
|
|
|
print(f"❌ Serial 連接啟動失敗: {str(e)}")
|
|
|
traceback.print_exc()
|
|
|
self.statusBar().showMessage(f"啟動 Serial 連接失敗: {str(e)}", 5000)
|
|
|
|
|
|
def remove_serial_connection(self, conn, panel):
|
|
|
if conn.get('enabled', False) and conn.get('receiver'):
|
|
|
conn['receiver'].stop()
|
|
|
if conn in self.serial_connections:
|
|
|
self.serial_connections.remove(conn)
|
|
|
self.comm_panel.remove_serial_connection_from_list(conn)
|
|
|
panel.setParent(None)
|
|
|
panel.deleteLater()
|
|
|
self.statusBar().showMessage(f"已移除 Serial 連接: {conn['port']}", 3000)
|
|
|
|
|
|
def create_socket_group_panel(self, socket_id):
|
|
|
color = self.socket_colors.get(socket_id, self.socket_colors['default'])
|
|
|
socket_type = self.socket_types.get(socket_id, None)
|
|
|
panel = SocketGroupPanel(socket_id, color, socket_type)
|
|
|
panel.group_selection_changed.connect(self.handle_group_selection)
|
|
|
return panel
|
|
|
|
|
|
# ================================================================================
|
|
|
# 無人機操作
|
|
|
# ================================================================================
|
|
|
|
|
|
def handle_mode_change(self, drone_id):
|
|
|
print(f"\n📢 [GUI] handle_mode_change 被调用")
|
|
|
print(f" drone_id: {drone_id}")
|
|
|
|
|
|
# 從 active group 的 mode_combo 讀取模式
|
|
|
group = self._get_active_group()
|
|
|
if group:
|
|
|
panel = self.group_panels.get(group.group_id)
|
|
|
mode = panel.mode_combo.currentText() if panel else "GUIDED"
|
|
|
print(f" 从群组读取模式: {mode}")
|
|
|
else:
|
|
|
mode = "GUIDED"
|
|
|
print(f" 没有活跃群组,使用默认模式: {mode}")
|
|
|
|
|
|
loop = asyncio.get_event_loop()
|
|
|
future = self.monitor.set_mode(drone_id, mode)
|
|
|
loop.create_task(self.handle_service_response(future, f"切換模式 {mode} {drone_id}"))
|
|
|
|
|
|
def handle_arm(self, drone_id):
|
|
|
print(f"\n📢 [GUI] handle_arm 被調用")
|
|
|
print(f" drone_id: {drone_id}")
|
|
|
loop = asyncio.get_event_loop()
|
|
|
print(f" 獲得事件循環: {loop}")
|
|
|
arm_state = not self.monitor.get_arm_state(drone_id)
|
|
|
print(f" 當前鎖定狀態: {not arm_state}, 目標狀態: {arm_state}")
|
|
|
print(f" 準備調用 arm_drone(drone_id={drone_id}, arm={arm_state})")
|
|
|
coro = self.monitor.arm_drone(drone_id, arm_state)
|
|
|
print(f" arm_drone() 返回類型: {type(coro)}")
|
|
|
print(f" coroutine 對象: {coro}")
|
|
|
action_text = f"{'解鎖' if arm_state else '上鎖'} {drone_id}"
|
|
|
print(f" 準備提交到事件循環: {action_text}")
|
|
|
task = loop.create_task(self.handle_service_response(coro, action_text))
|
|
|
print(f" task 已創建: {task}")
|
|
|
print(f" 已提交到事件循環\n")
|
|
|
|
|
|
def handle_takeoff(self, drone_id):
|
|
|
loop = asyncio.get_event_loop()
|
|
|
future = self.monitor.takeoff_drone(drone_id, 10.0)
|
|
|
loop.create_task(self.handle_service_response(future, f"起飛 {drone_id}"))
|
|
|
|
|
|
def handle_setpoint_selected(self):
|
|
|
"""發送位置命令到 active group 的所有選中無人機"""
|
|
|
group = self._get_active_group()
|
|
|
if not group:
|
|
|
self.statusBar().showMessage("⚠ 請先建立任務群組", 3000)
|
|
|
return
|
|
|
|
|
|
try:
|
|
|
x = float(self.x_input.text() or '0')
|
|
|
y = float(self.y_input.text() or '0')
|
|
|
z = float(self.z_input.text() or '0')
|
|
|
for drone_id in group.selected_drone_ids:
|
|
|
if self.monitor.send_setpoint(drone_id, x, y, z):
|
|
|
self.statusBar().showMessage(f"發送位置命令到 {drone_id}: ({x}, {y}, {z})", 3000)
|
|
|
else:
|
|
|
self.statusBar().showMessage(f"發送位置命令失敗: {drone_id}", 3000)
|
|
|
except ValueError:
|
|
|
self.statusBar().showMessage("座標格式錯誤", 3000)
|
|
|
|
|
|
def handle_single_setpoint(self, drone_id):
|
|
|
try:
|
|
|
x = float(self.x_input.text() or '0')
|
|
|
y = float(self.y_input.text() or '0')
|
|
|
z = float(self.z_input.text() or '0')
|
|
|
if self.monitor.send_setpoint(drone_id, x, y, z):
|
|
|
self.statusBar().showMessage(f"發送位置命令到 {drone_id}: ({x}, {y}, {z})", 3000)
|
|
|
else:
|
|
|
self.statusBar().showMessage(f"發送位置命令失敗: {drone_id}", 3000)
|
|
|
except ValueError:
|
|
|
self.statusBar().showMessage("座標格式錯誤", 3000)
|
|
|
|
|
|
async def handle_service_response(self, future, action):
|
|
|
print(f"\n📥 [GUI] handle_service_response 開始處理: {action}")
|
|
|
print(f" Future/Coroutine 類型: {type(future)}")
|
|
|
print(f" Future/Coroutine 對象: {future}")
|
|
|
try:
|
|
|
print(f" ├─ 等待 future/coroutine 完成...")
|
|
|
print(f" └─ (這是一個阻塞等待,直到服務返回)\n")
|
|
|
result = await future
|
|
|
print(f"\n ✓ Future/Coroutine 完成,接收到返回值!")
|
|
|
print(f" ├─ 返回值類型: {type(result)}")
|
|
|
print(f" ├─ 返回值內容: {result}")
|
|
|
print(f" ├─ 返回值 == True: {result == True}")
|
|
|
print(f" ├─ 返回值 is True: {result is True}")
|
|
|
print(f" └─ bool(返回值): {bool(result)}")
|
|
|
|
|
|
# 詳細檢查 result 結構(用於調試 fc_network 回傳值)
|
|
|
if hasattr(result, '__dict__'):
|
|
|
print(f" └─ 返回值屬性: {result.__dict__}")
|
|
|
|
|
|
if result:
|
|
|
print(f"\n✅ {action} 成功 (result={result})")
|
|
|
self.statusBar().showMessage(f"{action} 成功", 3000)
|
|
|
else:
|
|
|
print(f"\n❌ {action} 失敗 (result={result})")
|
|
|
self.statusBar().showMessage(f"{action} 失敗", 3000)
|
|
|
except asyncio.CancelledError:
|
|
|
print(f"⚠️ {action} 被取消")
|
|
|
except Exception as e:
|
|
|
print(f"❌ {action} 錯誤: {str(e)}")
|
|
|
traceback.print_exc()
|
|
|
self.statusBar().showMessage(f"{action} 錯誤: {str(e)}", 3000)
|
|
|
|
|
|
def handle_arm_selected(self):
|
|
|
"""解鎖 active group 的所有選中無人機"""
|
|
|
group = self._get_active_group()
|
|
|
if not group:
|
|
|
self.statusBar().showMessage("⚠ 請先建立任務群組", 3000)
|
|
|
return
|
|
|
|
|
|
selected = list(group.selected_drone_ids)
|
|
|
if not selected:
|
|
|
self.statusBar().showMessage("⚠ 尚未選擇任何無人機", 3000)
|
|
|
return
|
|
|
|
|
|
print(f"\n📢 [GUI] handle_arm_selected 被調用")
|
|
|
print(f" 已選擇的無人機: {selected}")
|
|
|
loop = asyncio.get_event_loop()
|
|
|
for drone_id in selected:
|
|
|
print(f" 準備批次解鎖: {drone_id}")
|
|
|
coro = self.monitor.arm_drone(drone_id, True)
|
|
|
print(f" arm_drone 返回: {coro}")
|
|
|
# 使用 run_coroutine_threadsafe() 正確地在事件循環中運行
|
|
|
asyncio.run_coroutine_threadsafe(
|
|
|
self.handle_service_response(coro, f"批次解鎖 {drone_id}"),
|
|
|
loop
|
|
|
)
|
|
|
print(f" handle_arm_selected 完成\n")
|
|
|
|
|
|
def handle_takeoff_selected(self):
|
|
|
"""起飛 active group 的所有選中無人機"""
|
|
|
group = self._get_active_group()
|
|
|
if not group:
|
|
|
self.statusBar().showMessage("⚠ 請先建立任務群組", 3000)
|
|
|
return
|
|
|
|
|
|
selected = list(group.selected_drone_ids)
|
|
|
if not selected:
|
|
|
self.statusBar().showMessage("⚠ 尚未選擇任何無人機", 3000)
|
|
|
return
|
|
|
|
|
|
loop = asyncio.get_event_loop()
|
|
|
for drone_id in selected:
|
|
|
future = self.monitor.takeoff_drone(drone_id, 10.0)
|
|
|
loop.create_task(self.handle_service_response(future, f"批次起飛 {drone_id}"))
|
|
|
|
|
|
# ================================================================================
|
|
|
# 任務群組管理
|
|
|
# ================================================================================
|
|
|
|
|
|
def _next_group_id(self):
|
|
|
"""產生下一個群組 ID (A, B, C, ...)"""
|
|
|
gid = chr(ord('A') + self._group_counter)
|
|
|
self._group_counter += 1
|
|
|
return gid
|
|
|
|
|
|
def _toggle_group_panel(self):
|
|
|
"""🌟 收起/展開任務群組面板"""
|
|
|
if self.group_panel_expanded:
|
|
|
# 收起
|
|
|
self.group_tab_widget.setFixedHeight(0)
|
|
|
self.group_tab_widget.hide()
|
|
|
self.toggle_group_btn.setText("▶")
|
|
|
self.toggle_group_btn.setToolTip("展開任務群組")
|
|
|
self.group_panel_expanded = False
|
|
|
else:
|
|
|
# 展開
|
|
|
self.group_tab_widget.setFixedHeight(150)
|
|
|
self.group_tab_widget.show()
|
|
|
self.toggle_group_btn.setText("▼")
|
|
|
self.toggle_group_btn.setToolTip("收起任務群組")
|
|
|
self.group_panel_expanded = True
|
|
|
|
|
|
def _add_mission_group(self):
|
|
|
"""新增一個任務群組"""
|
|
|
gid = self._next_group_id()
|
|
|
color = GROUP_COLORS[(self._group_counter - 1) % len(GROUP_COLORS)]
|
|
|
group = MissionGroup(gid, color)
|
|
|
self.mission_groups[gid] = group
|
|
|
|
|
|
panel = GroupPanel(group)
|
|
|
panel.assign_drones_requested.connect(self._handle_assign_drones)
|
|
|
panel.mission_type_changed.connect(self._handle_mission_type_changed)
|
|
|
panel.start_requested.connect(self._handle_group_start)
|
|
|
panel.pause_requested.connect(self._handle_group_pause)
|
|
|
panel.stop_requested.connect(self._handle_group_stop)
|
|
|
panel.mode_change_requested.connect(self._handle_group_mode_change)
|
|
|
panel.arm_requested.connect(self._handle_group_arm)
|
|
|
panel.takeoff_requested.connect(self._handle_group_takeoff)
|
|
|
panel.box_select_requested.connect(self._handle_box_select)
|
|
|
panel.select_all_requested.connect(self._handle_select_all_for_group)
|
|
|
panel.clear_group_requested.connect(self._handle_clear_group)
|
|
|
panel.add_group_requested.connect(self._add_mission_group)
|
|
|
panel.delete_group_requested.connect(self._handle_delete_group)
|
|
|
self.group_panels[gid] = panel
|
|
|
|
|
|
# 用帶顏色的 tab 標題
|
|
|
scroll = QScrollArea()
|
|
|
scroll.setWidget(panel)
|
|
|
scroll.setWidgetResizable(True)
|
|
|
idx = self.group_tab_widget.addTab(scroll, f"Group {gid}")
|
|
|
self.group_tab_widget.tabBar().setTabTextColor(idx, QColor(color))
|
|
|
|
|
|
# 切換到新 group 的 tab
|
|
|
self.group_tab_widget.setCurrentIndex(idx)
|
|
|
|
|
|
self.statusBar().showMessage(f"已新增 Group {gid}", 2000)
|
|
|
|
|
|
# 更新刪除按鈕的啟用/禁用狀態
|
|
|
self._update_delete_buttons_state()
|
|
|
|
|
|
def _on_group_tab_changed(self, index):
|
|
|
"""切換 group tab — 只需切換 active group 並刷新 UI"""
|
|
|
if index < 0:
|
|
|
self.active_group_id = None
|
|
|
return
|
|
|
|
|
|
tab_text = self.group_tab_widget.tabText(index)
|
|
|
gid = tab_text.replace("Group ", "")
|
|
|
if gid not in self.mission_groups:
|
|
|
return
|
|
|
|
|
|
self.active_group_id = gid
|
|
|
group = self.mission_groups[gid]
|
|
|
|
|
|
# 同步地圖的任務模式
|
|
|
self.drone_map.set_mission_mode(group.mission_type)
|
|
|
|
|
|
# 統一刷新所有 UI
|
|
|
self.refresh_selection_ui()
|
|
|
|
|
|
def _get_active_group(self):
|
|
|
"""取得當前 active 的 MissionGroup"""
|
|
|
if self.active_group_id and self.active_group_id in self.mission_groups:
|
|
|
return self.mission_groups[self.active_group_id]
|
|
|
return None
|
|
|
|
|
|
def refresh_selection_ui(self):
|
|
|
"""統一刷新所有 UI 元素:左側 drone 勾選、左側 socket 勾選、右側 group panel"""
|
|
|
group = self._get_active_group()
|
|
|
selected = group.selected_drone_ids if group else set()
|
|
|
current_group_id = group.group_id if group else None
|
|
|
|
|
|
# 左側 drone checkbox
|
|
|
for drone_id, panel in self.drones.items():
|
|
|
checkbox = panel.get_checkbox()
|
|
|
if checkbox:
|
|
|
blocked = (
|
|
|
current_group_id is not None and
|
|
|
self._is_drone_selected_by_other_group(drone_id, current_group_id)
|
|
|
)
|
|
|
checkbox.blockSignals(True)
|
|
|
checkbox.setStyleSheet(self._get_drone_checkbox_style(blocked))
|
|
|
checkbox.setChecked(drone_id in selected)
|
|
|
checkbox.setEnabled(not blocked or drone_id in selected)
|
|
|
checkbox.blockSignals(False)
|
|
|
|
|
|
# 左側 socket checkbox
|
|
|
for socket_id in self.socket_groups.keys():
|
|
|
self.refresh_socket_checkbox(socket_id, selected, current_group_id)
|
|
|
|
|
|
# 右側 group panel
|
|
|
if group:
|
|
|
panel = self.group_panels.get(group.group_id)
|
|
|
if panel:
|
|
|
panel.update_drone_list()
|
|
|
panel.update_status()
|
|
|
|
|
|
def _get_drone_checkbox_style(self, blocked=False):
|
|
|
"""回傳 drone checkbox 樣式;被其他 group 佔用時顯示紅色填滿。"""
|
|
|
blocked_style = """
|
|
|
QCheckBox::indicator:disabled {
|
|
|
background-color: #D32F2F;
|
|
|
border: 2px solid #A32020;
|
|
|
}
|
|
|
""" if blocked else ""
|
|
|
return f"""
|
|
|
QCheckBox {{
|
|
|
color: #DDD;
|
|
|
}}
|
|
|
QCheckBox::indicator {{
|
|
|
width: 16px;
|
|
|
height: 16px;
|
|
|
border: 2px solid #888888;
|
|
|
border-radius: 3px;
|
|
|
background: transparent;
|
|
|
}}
|
|
|
QCheckBox::indicator:checked {{
|
|
|
background-color: #7FFFD4;
|
|
|
border: 2px solid #888888;
|
|
|
}}
|
|
|
{blocked_style}
|
|
|
"""
|
|
|
|
|
|
def refresh_socket_checkbox(self, socket_id, selected_ids, current_group_id=None):
|
|
|
"""根據 selected_ids 推導並更新 socket 的勾選狀態"""
|
|
|
drone_ids = [did for did in self.drones.keys() if self.get_socket_id(did) == socket_id]
|
|
|
if not drone_ids:
|
|
|
return
|
|
|
|
|
|
socket_widget = self.socket_groups.get(socket_id)
|
|
|
if not socket_widget:
|
|
|
return
|
|
|
|
|
|
checkbox = socket_widget.findChild(QCheckBox, f"socket_{socket_id}_checkbox")
|
|
|
if not checkbox:
|
|
|
return
|
|
|
|
|
|
effective_drone_ids = [
|
|
|
did for did in drone_ids
|
|
|
if current_group_id is None
|
|
|
or did in selected_ids
|
|
|
or not self._is_drone_selected_by_other_group(did, current_group_id)
|
|
|
]
|
|
|
checked_count = sum(1 for did in effective_drone_ids if did in selected_ids)
|
|
|
|
|
|
checkbox.blockSignals(True)
|
|
|
if checked_count == 0:
|
|
|
checkbox.setCheckState(Qt.CheckState.Unchecked)
|
|
|
elif checked_count == len(effective_drone_ids):
|
|
|
checkbox.setCheckState(Qt.CheckState.Checked)
|
|
|
else:
|
|
|
checkbox.setCheckState(Qt.CheckState.PartiallyChecked)
|
|
|
checkbox.setEnabled(len(effective_drone_ids) > 0)
|
|
|
checkbox.blockSignals(False)
|
|
|
|
|
|
|
|
|
# 【已遷移至 refresh_socket_checkbox()】舊的 update_group_checkbox_state() 已廢棄
|
|
|
|
|
|
def _handle_assign_drones(self, group_id):
|
|
|
"""開啟無人機分配對話框 — 過濾被占用的drone"""
|
|
|
group = self.mission_groups.get(group_id)
|
|
|
if not group:
|
|
|
return
|
|
|
all_ids = list(self.drones.keys())
|
|
|
|
|
|
# 預先選中:目前 group 的 selected_drone_ids
|
|
|
pre_selected = set(group.selected_drone_ids)
|
|
|
|
|
|
dialog = DroneAssignDialog(self, all_ids, pre_selected, {})
|
|
|
if dialog.exec() == QDialog.DialogCode.Accepted:
|
|
|
requested = dialog.get_selected()
|
|
|
|
|
|
allowed = set()
|
|
|
blocked = set()
|
|
|
|
|
|
for did in requested:
|
|
|
if self._is_drone_selected_by_other_group(did, group_id):
|
|
|
blocked.add(did)
|
|
|
else:
|
|
|
allowed.add(did)
|
|
|
|
|
|
group.selected_drone_ids = allowed
|
|
|
|
|
|
# 只有當操作目標組是 active 組時,才更新 UI
|
|
|
if group_id == self.active_group_id:
|
|
|
self.refresh_selection_ui()
|
|
|
|
|
|
panel = self.group_panels.get(group_id)
|
|
|
if panel:
|
|
|
panel.update_drone_list()
|
|
|
panel.update_status()
|
|
|
|
|
|
if blocked:
|
|
|
self.statusBar().showMessage(
|
|
|
f"⚠ Group {group_id}: {len(blocked)} 台已被其他 group 勾選,未加入", 4000
|
|
|
)
|
|
|
else:
|
|
|
self.statusBar().showMessage(
|
|
|
f"Group {group_id}: 已分配 {len(group.selected_drone_ids)} 台無人機", 3000
|
|
|
)
|
|
|
|
|
|
def _handle_mission_type_changed(self, group_id, mission_type):
|
|
|
"""群組任務類型切換"""
|
|
|
group = self.mission_groups.get(group_id)
|
|
|
if group:
|
|
|
group.mission_type = mission_type
|
|
|
# 如果是 active group,同步更新地圖的選擇模式
|
|
|
if group_id == self.active_group_id:
|
|
|
self.drone_map.set_mission_mode(mission_type)
|
|
|
|
|
|
def _create_executor_for_group(self, group):
|
|
|
"""為群組建立 MissionExecutor"""
|
|
|
executor = MissionExecutor(
|
|
|
sender=self.command_sender,
|
|
|
drone_gps=self.monitor.drone_gps,
|
|
|
monitor=self.monitor,
|
|
|
arrival_radius=2.0,
|
|
|
tick_rate_hz=2.0,
|
|
|
)
|
|
|
executor.drone_waypoint_reached.connect(self.on_drone_waypoint_reached)
|
|
|
executor.task_status_changed.connect(self._on_task_status_changed)
|
|
|
executor.mission_completed.connect(
|
|
|
lambda gid=group.group_id: self._on_group_mission_completed(gid))
|
|
|
group.executor = executor
|
|
|
|
|
|
def _handle_group_start(self, group_id):
|
|
|
"""啟動群組任務"""
|
|
|
group = self.mission_groups.get(group_id)
|
|
|
if not group:
|
|
|
return
|
|
|
if group.planned_waypoints is None:
|
|
|
self.statusBar().showMessage(f"⚠ Group {group_id}: 請先規劃任務", 3000)
|
|
|
return
|
|
|
if group.executor is None:
|
|
|
self._create_executor_for_group(group)
|
|
|
group.executor.start(group.planned_waypoints)
|
|
|
panel = self.group_panels.get(group_id)
|
|
|
if panel:
|
|
|
panel.update_status()
|
|
|
self.statusBar().showMessage(f"🚀 Group {group_id}: 任務已啟動", 3000)
|
|
|
|
|
|
def _handle_group_pause(self, group_id):
|
|
|
"""暫停/恢復群組任務"""
|
|
|
group = self.mission_groups.get(group_id)
|
|
|
if not group or not group.executor:
|
|
|
return
|
|
|
if group.executor.state == MissionState.RUNNING:
|
|
|
group.executor.pause()
|
|
|
self.statusBar().showMessage(f"⏸ Group {group_id}: 任務已暫停", 3000)
|
|
|
elif group.executor.state == MissionState.PAUSED:
|
|
|
group.executor.resume()
|
|
|
self.statusBar().showMessage(f"▶ Group {group_id}: 任務已恢復", 3000)
|
|
|
panel = self.group_panels.get(group_id)
|
|
|
if panel:
|
|
|
panel.update_status()
|
|
|
|
|
|
def _handle_group_stop(self, group_id):
|
|
|
"""停止群組任務"""
|
|
|
group = self.mission_groups.get(group_id)
|
|
|
if not group:
|
|
|
return
|
|
|
if group.executor:
|
|
|
group.executor.stop()
|
|
|
group.planned_waypoints = None
|
|
|
self.drone_map.clear_mission_plan_for_group(group_id)
|
|
|
panel = self.group_panels.get(group_id)
|
|
|
if panel:
|
|
|
panel.update_status()
|
|
|
panel.clear_mission_info()
|
|
|
self.statusBar().showMessage(f"■ Group {group_id}: 任務已停止", 3000)
|
|
|
|
|
|
def _handle_group_mode_change(self, group_id, mode):
|
|
|
"""切換群組內所有無人機的飛行模式"""
|
|
|
print(f"\n📢 [GUI] _handle_group_mode_change 被调用", flush=True)
|
|
|
print(f" group_id: {group_id}, mode: {mode}", flush=True)
|
|
|
|
|
|
group = self.mission_groups.get(group_id)
|
|
|
if not group:
|
|
|
print(f"❌ 找不到群組: {group_id}", flush=True)
|
|
|
return
|
|
|
|
|
|
if not group.selected_drone_ids:
|
|
|
print(f"⚠️ 群組中沒有無人機", flush=True)
|
|
|
self.statusBar().showMessage(f"群組 {group_id} 中沒有無人機", 3000)
|
|
|
return
|
|
|
|
|
|
print(f" 準備為 {len(group.selected_drone_ids)} 台無人機切換模式...", flush=True)
|
|
|
self.statusBar().showMessage(f"正在切換模式...", 1000)
|
|
|
|
|
|
# 使用 asyncio 執行(通過事件循環)
|
|
|
async def do_mode_changes_async():
|
|
|
print(f"\n 【異步任務】開始執行模式切換", flush=True)
|
|
|
|
|
|
for drone_id in group.selected_drone_ids:
|
|
|
print(f"\n 【切換】{drone_id} → {mode}", flush=True)
|
|
|
try:
|
|
|
result = await self.monitor.set_mode(drone_id, mode)
|
|
|
|
|
|
if result:
|
|
|
msg = f"✅ {drone_id} 切換成功"
|
|
|
print(f" {msg}", flush=True)
|
|
|
self.message_queue.put((msg, 2000))
|
|
|
else:
|
|
|
msg = f"❌ {drone_id} 切換失敗"
|
|
|
print(f" {msg}", flush=True)
|
|
|
self.message_queue.put((msg, 2000))
|
|
|
|
|
|
except Exception as e:
|
|
|
msg = f"❌ {drone_id} 錯誤: {str(e)}"
|
|
|
print(f" {msg}", flush=True)
|
|
|
traceback.print_exc()
|
|
|
self.message_queue.put((msg, 2000))
|
|
|
|
|
|
print(f"\n 【完成】模式切換任務結束\n", flush=True)
|
|
|
|
|
|
# 通過事件循環提交異步任務
|
|
|
print(f" 【排隊】將任務提交至事件循環", flush=True)
|
|
|
loop = asyncio.get_event_loop()
|
|
|
asyncio.run_coroutine_threadsafe(
|
|
|
do_mode_changes_async(),
|
|
|
loop
|
|
|
)
|
|
|
|
|
|
def _handle_group_arm(self, group_id):
|
|
|
"""解鎖群組內所有無人機"""
|
|
|
print(f"\n📢 [GUI] _handle_group_arm 被調用")
|
|
|
print(f" 群組 ID: {group_id}")
|
|
|
group = self.mission_groups.get(group_id)
|
|
|
print(f" 群組存在: {group is not None}")
|
|
|
if not group:
|
|
|
print(f" ⚠️ 群組不存在,返回\n")
|
|
|
return
|
|
|
print(f" 群組內無人機: {group.selected_drone_ids}")
|
|
|
loop = asyncio.get_event_loop()
|
|
|
print(f" 事件循環: {loop}")
|
|
|
|
|
|
for drone_id in group.selected_drone_ids:
|
|
|
print(f"\n ┌─ 處理無人機: {drone_id}")
|
|
|
print(f" ├─ 準備調用 arm_drone(drone_id={drone_id}, arm=True)")
|
|
|
coro = self.monitor.arm_drone(drone_id, True)
|
|
|
print(f" ├─ arm_drone 返回類型: {type(coro)}")
|
|
|
action_text = f"解鎖 {drone_id}"
|
|
|
print(f" ├─ 準備提交到事件循環: {action_text}")
|
|
|
# 使用 run_coroutine_threadsafe() 正確地在事件循環中運行 coroutine
|
|
|
# 這是 Qt + asyncio 整合的正確方式
|
|
|
future = asyncio.run_coroutine_threadsafe(
|
|
|
self.handle_service_response(coro, action_text),
|
|
|
loop
|
|
|
)
|
|
|
print(f" ├─ Future 已創建: {future}")
|
|
|
print(f" └─ Future 將在事件循環中執行")
|
|
|
|
|
|
print(f"\n _handle_group_arm 完成(coroutine 已提交至事件循環)\n")
|
|
|
|
|
|
def _handle_group_takeoff(self, group_id, altitude):
|
|
|
"""群組內所有無人機起飛"""
|
|
|
group = self.mission_groups.get(group_id)
|
|
|
if not group:
|
|
|
return
|
|
|
loop = asyncio.get_event_loop()
|
|
|
for drone_id in group.selected_drone_ids:
|
|
|
future = self.monitor.takeoff_drone(drone_id, altitude)
|
|
|
loop.create_task(self.handle_service_response(future, f"起飛 {drone_id} ({altitude}m)"))
|
|
|
|
|
|
def _handle_box_select(self, group_id):
|
|
|
"""觸發地圖框選 → 框選完成後直接分配到該群組"""
|
|
|
self._pending_box_assign = group_id
|
|
|
self.drone_map.toggle_drone_box_select()
|
|
|
self.statusBar().showMessage(
|
|
|
f"請在地圖上框選要分配到 Group {group_id} 的無人機", 5000)
|
|
|
|
|
|
def _handle_drone_box_selected(self, drone_ids_json):
|
|
|
group_id = self._pending_box_assign
|
|
|
self._pending_box_assign = None
|
|
|
if not group_id:
|
|
|
return
|
|
|
|
|
|
group = self.mission_groups.get(group_id)
|
|
|
if not group:
|
|
|
return
|
|
|
|
|
|
drone_ids = json.loads(drone_ids_json)
|
|
|
|
|
|
allowed = set()
|
|
|
blocked = set()
|
|
|
for did in drone_ids:
|
|
|
if self._is_drone_selected_by_other_group(did, group_id):
|
|
|
blocked.add(did)
|
|
|
else:
|
|
|
allowed.add(did)
|
|
|
|
|
|
group.selected_drone_ids = allowed
|
|
|
|
|
|
if group_id == self.active_group_id:
|
|
|
self.refresh_selection_ui()
|
|
|
|
|
|
panel = self.group_panels.get(group_id)
|
|
|
if panel:
|
|
|
panel.update_drone_list()
|
|
|
panel.update_status()
|
|
|
|
|
|
if blocked:
|
|
|
self.statusBar().showMessage(
|
|
|
f"⚠ Group {group_id}: 部分 drone 已被其他 group 勾選,僅加入 {len(allowed)} 台", 4000
|
|
|
)
|
|
|
else:
|
|
|
self.statusBar().showMessage(
|
|
|
f"Group {group_id}: 框選分配 {len(allowed)} 台無人機", 3000
|
|
|
)
|
|
|
|
|
|
def _handle_select_all_for_group(self, group_id):
|
|
|
"""全選/取消全選 — Toggle 邏輯"""
|
|
|
group = self.mission_groups.get(group_id)
|
|
|
if not group:
|
|
|
return
|
|
|
|
|
|
selectable_ids = self._get_selectable_drones_for_group(group_id) | set(group.selected_drone_ids)
|
|
|
|
|
|
# Toggle 邏輯:如果已全選,則清空;否則全選
|
|
|
if group.selected_drone_ids == selectable_ids:
|
|
|
# 已全選 → 清空
|
|
|
group.selected_drone_ids.clear()
|
|
|
msg_status = "已取消全選"
|
|
|
else:
|
|
|
# 未全選 → 全選
|
|
|
group.selected_drone_ids = set(selectable_ids)
|
|
|
msg_status = f"全選可用無人機 {len(selectable_ids)} 台"
|
|
|
|
|
|
# 只有當操作目標組是 active 組時,才更新 UI
|
|
|
if group_id == self.active_group_id:
|
|
|
self.refresh_selection_ui()
|
|
|
|
|
|
panel = self.group_panels.get(group_id)
|
|
|
if panel:
|
|
|
panel.update_drone_list()
|
|
|
panel.update_status()
|
|
|
|
|
|
self.statusBar().showMessage(
|
|
|
f"Group {group_id}: {msg_status}", 3000)
|
|
|
|
|
|
def _handle_clear_group(self, group_id):
|
|
|
"""清除群組的無人機選擇"""
|
|
|
group = self.mission_groups.get(group_id)
|
|
|
if not group:
|
|
|
return
|
|
|
|
|
|
group.selected_drone_ids.clear()
|
|
|
group.planned_waypoints = None
|
|
|
if group.executor:
|
|
|
group.executor.stop()
|
|
|
self.drone_map.clear_mission_plan_for_group(group_id)
|
|
|
|
|
|
# 只有當操作目標組是 active 組時,才更新 UI
|
|
|
if group_id == self.active_group_id:
|
|
|
self.refresh_selection_ui()
|
|
|
|
|
|
panel = self.group_panels.get(group_id)
|
|
|
if panel:
|
|
|
panel.update_drone_list()
|
|
|
panel.update_status()
|
|
|
panel.clear_mission_info()
|
|
|
|
|
|
self.statusBar().showMessage(
|
|
|
f"Group {group_id}: 已清除分組", 3000)
|
|
|
|
|
|
def _handle_delete_group(self, group_id):
|
|
|
"""刪除一個任務群組"""
|
|
|
# 檢查是否只有一個群組,如果是就禁止刪除
|
|
|
if len(self.mission_groups) <= 1:
|
|
|
self.statusBar().showMessage("⚠️ 至少需要保留一個群組!", 3000)
|
|
|
return
|
|
|
|
|
|
if group_id not in self.mission_groups:
|
|
|
self.statusBar().showMessage(f"Group {group_id} 不存在", 3000)
|
|
|
return
|
|
|
|
|
|
# 停止群組的執行(如果有)
|
|
|
group = self.mission_groups[group_id]
|
|
|
if group.executor:
|
|
|
group.executor.stop()
|
|
|
|
|
|
# 移除地圖上的任務計畫
|
|
|
self.drone_map.clear_mission_plan_for_group(group_id)
|
|
|
|
|
|
# 移除 tab
|
|
|
for i in range(self.group_tab_widget.count()):
|
|
|
if f"Group {group_id}" in self.group_tab_widget.tabText(i):
|
|
|
self.group_tab_widget.removeTab(i)
|
|
|
break
|
|
|
|
|
|
# 移除資料
|
|
|
del self.mission_groups[group_id]
|
|
|
if group_id in self.group_panels:
|
|
|
del self.group_panels[group_id]
|
|
|
|
|
|
# 更新 active group — 讓 currentChanged signal 自動處理
|
|
|
if self.active_group_id == group_id:
|
|
|
self.active_group_id = None
|
|
|
if self.group_tab_widget.count() > 0:
|
|
|
self.group_tab_widget.setCurrentIndex(0)
|
|
|
# _on_group_tab_changed 會自動設定新的 active group
|
|
|
|
|
|
self.statusBar().showMessage(f"已刪除 Group {group_id}", 3000)
|
|
|
|
|
|
# 更新刪除按鈕的啟用/禁用狀態
|
|
|
self._update_delete_buttons_state()
|
|
|
|
|
|
def _update_delete_buttons_state(self):
|
|
|
"""根據群組數量,更新所有群組的刪除按鈕啟用/禁用狀態"""
|
|
|
# 如果只有一個群組,禁用該群組的刪除按鈕
|
|
|
# 如果有多個群組,啟用所有刪除按鈕
|
|
|
should_enable = len(self.mission_groups) > 1
|
|
|
for gid, panel in self.group_panels.items():
|
|
|
panel.set_delete_enabled(should_enable)
|
|
|
|
|
|
def _on_group_mission_completed(self, group_id):
|
|
|
"""群組任務完成回呼"""
|
|
|
panel = self.group_panels.get(group_id)
|
|
|
if panel:
|
|
|
panel.update_status()
|
|
|
self.statusBar().showMessage(f"✅ Group {group_id}: 所有無人機已完成任務", 5000)
|
|
|
|
|
|
# ================================================================================
|
|
|
# UI 更新
|
|
|
# ================================================================================
|
|
|
|
|
|
def update_ui(self, msg_type, drone_id, data):
|
|
|
"""只做數據快取,不在這裡更新 UI"""
|
|
|
if msg_type == 'connection_type':
|
|
|
conn_type = data.get('type', 'Unknown')
|
|
|
parts = drone_id.split('_')
|
|
|
if len(parts) >= 2 and parts[0].startswith('s'):
|
|
|
socket_id = parts[0][1:]
|
|
|
if socket_id not in self.socket_types:
|
|
|
self.socket_types[socket_id] = conn_type
|
|
|
if socket_id in self.socket_groups:
|
|
|
self.socket_groups[socket_id].set_socket_type(conn_type)
|
|
|
return
|
|
|
|
|
|
if drone_id not in self.drones:
|
|
|
self.add_drone(drone_id)
|
|
|
return
|
|
|
|
|
|
# 只做資料快取,不更新 UI - 所有 UI 更新都在 _update_panel_and_map 中進行
|
|
|
if drone_id not in self._message_cache:
|
|
|
self._message_cache[drone_id] = {}
|
|
|
|
|
|
self._message_cache[drone_id][msg_type] = data
|
|
|
|
|
|
|
|
|
# ================================================================================
|
|
|
# 勾選管理
|
|
|
# ================================================================================
|
|
|
|
|
|
def handle_group_selection(self, socket_id, state):
|
|
|
"""Socket 群組勾選 — 更新該 socket 下所有 drone 的選擇狀態"""
|
|
|
group = self._get_active_group()
|
|
|
if not group:
|
|
|
return
|
|
|
|
|
|
drone_ids = [did for did in self.drones.keys() if self.get_socket_id(did) == socket_id]
|
|
|
selectable = []
|
|
|
blocked = []
|
|
|
for did in drone_ids:
|
|
|
if self._is_drone_selected_by_other_group(did, group.group_id):
|
|
|
blocked.append(did)
|
|
|
else:
|
|
|
selectable.append(did)
|
|
|
|
|
|
if not selectable:
|
|
|
if blocked:
|
|
|
self.statusBar().showMessage(
|
|
|
f"⚠ Socket {socket_id} 的 drone 已被其他 group 勾選,無法操作", 4000
|
|
|
)
|
|
|
self.refresh_selection_ui()
|
|
|
return
|
|
|
|
|
|
all_selectable_selected = all(did in group.selected_drone_ids for did in selectable)
|
|
|
|
|
|
if all_selectable_selected:
|
|
|
group.selected_drone_ids.difference_update(selectable)
|
|
|
else:
|
|
|
group.selected_drone_ids.update(selectable)
|
|
|
if blocked:
|
|
|
self.statusBar().showMessage(
|
|
|
f"⚠ 以下 drone 已被其他 group 勾選,略過: {', '.join(blocked)}", 4000
|
|
|
)
|
|
|
|
|
|
self.refresh_selection_ui()
|
|
|
|
|
|
def _is_drone_selected_by_other_group(self, drone_id, current_group_id):
|
|
|
"""
|
|
|
檢查該 drone 是否已被其他 group 勾選
|
|
|
|
|
|
Args:
|
|
|
drone_id: 無人機 ID
|
|
|
current_group_id: 當前 group 的 ID
|
|
|
|
|
|
Returns:
|
|
|
True 如果 drone 已被其他 group 選擇,否則 False
|
|
|
"""
|
|
|
for gid, group in self.mission_groups.items():
|
|
|
# 跳過當前 group
|
|
|
if gid == current_group_id:
|
|
|
continue
|
|
|
# 檢查該 drone 是否在其他 group 中被勾選
|
|
|
if drone_id in group.selected_drone_ids:
|
|
|
return True
|
|
|
return False
|
|
|
|
|
|
def _get_selectable_drones_for_group(self, group_id):
|
|
|
"""
|
|
|
取得指定 group 可以選擇的 drone 清單(不被其他 group 佔用的)
|
|
|
|
|
|
Args:
|
|
|
group_id: group 的 ID
|
|
|
|
|
|
Returns:
|
|
|
set: 可選擇的 drone ID 集合
|
|
|
"""
|
|
|
selectable = set()
|
|
|
for did in self.drones.keys():
|
|
|
if not self._is_drone_selected_by_other_group(did, group_id):
|
|
|
selectable.add(did)
|
|
|
return selectable
|
|
|
|
|
|
def handle_drone_selection(self, drone_id, state):
|
|
|
"""單台 drone 勾選 — 只修改 active group 的 selected_drone_ids"""
|
|
|
group = self._get_active_group()
|
|
|
if not group:
|
|
|
return
|
|
|
|
|
|
is_checked = (state == Qt.CheckState.Checked.value)
|
|
|
|
|
|
if is_checked:
|
|
|
if self._is_drone_selected_by_other_group(drone_id, group.group_id):
|
|
|
self.statusBar().showMessage(
|
|
|
f"⚠ {drone_id} 已被其他 group 勾選,無法加入 Group {group.group_id}", 3000
|
|
|
)
|
|
|
self.refresh_selection_ui()
|
|
|
return
|
|
|
group.selected_drone_ids.add(drone_id)
|
|
|
else:
|
|
|
group.selected_drone_ids.discard(drone_id)
|
|
|
|
|
|
self.refresh_selection_ui()
|
|
|
|
|
|
def handle_drone_clicked(self, drone_id):
|
|
|
"""地圖上點擊 drone — 切換其選擇狀態"""
|
|
|
group = self._get_active_group()
|
|
|
if not group:
|
|
|
return
|
|
|
|
|
|
if drone_id in group.selected_drone_ids:
|
|
|
group.selected_drone_ids.remove(drone_id)
|
|
|
else:
|
|
|
# 嘗試勾選前先檢查是否被其他 group 使用
|
|
|
if self._is_drone_selected_by_other_group(drone_id, group.group_id):
|
|
|
self.statusBar().showMessage(
|
|
|
f"⚠ {drone_id} 已被其他 group 勾選,無法加入 Group {group.group_id}", 3000
|
|
|
)
|
|
|
self.refresh_selection_ui()
|
|
|
return
|
|
|
group.selected_drone_ids.add(drone_id)
|
|
|
|
|
|
self.refresh_selection_ui()
|
|
|
|
|
|
def handle_toggle_select_all_drones(self):
|
|
|
"""全選 / 清空 — 切換 active group 的所有 drone"""
|
|
|
group = self._get_active_group()
|
|
|
if not group:
|
|
|
return
|
|
|
|
|
|
selectable_ids = self._get_selectable_drones_for_group(group.group_id) | set(group.selected_drone_ids)
|
|
|
|
|
|
if group.selected_drone_ids == selectable_ids:
|
|
|
# 已全選 → 清空
|
|
|
group.selected_drone_ids.clear()
|
|
|
else:
|
|
|
# 未全選 → 全選
|
|
|
group.selected_drone_ids = set(selectable_ids)
|
|
|
|
|
|
self.refresh_selection_ui()
|
|
|
|
|
|
def handle_clear_all_drone_selection(self):
|
|
|
"""清除 active group 的所有無人機選擇"""
|
|
|
group = self._get_active_group()
|
|
|
if not group:
|
|
|
return
|
|
|
group.selected_drone_ids.clear()
|
|
|
self.refresh_selection_ui()
|
|
|
self.statusBar().showMessage("已清除所有選擇", 2000)
|
|
|
|
|
|
# ================================================================================
|
|
|
# 任務規劃 — 點擊地圖(路由到 active group)
|
|
|
# ================================================================================
|
|
|
|
|
|
def _get_group_drones(self, group):
|
|
|
"""取得群組的無人機 ID 列表(排序後)"""
|
|
|
return sorted(group.selected_drone_ids, key=lambda x: (x.split('_')[0], int(x.split('_')[1])))
|
|
|
|
|
|
def handle_map_click(self, lat, lon):
|
|
|
"""處理地圖點擊事件 — 根據 active group 的任務類型規劃"""
|
|
|
group = self._get_active_group()
|
|
|
if not group:
|
|
|
self.statusBar().showMessage("⚠ 請先建立任務群組", 3000)
|
|
|
return
|
|
|
|
|
|
mode_map = {
|
|
|
'M_FORMATION': MissionType.M_FORMATION,
|
|
|
'CIRCLE_FORMATION': MissionType.CIRCLE_FORMATION,
|
|
|
}
|
|
|
mission_type = mode_map.get(group.mission_type)
|
|
|
if mission_type is None:
|
|
|
return # Grid Sweep / Leader-Follower 由各自的觸發方式處理
|
|
|
|
|
|
selected_drones = self._get_group_drones(group)
|
|
|
if len(selected_drones) == 0:
|
|
|
self.statusBar().showMessage(f"⚠ Group {group.group_id}: 請先分配無人機", 3000)
|
|
|
return
|
|
|
|
|
|
print(f"地圖點擊: {lat:.6f}, {lon:.6f} → Group {group.group_id} ({group.mission_type})")
|
|
|
panel = self.group_panels.get(group.group_id)
|
|
|
params = panel.get_mission_params() if panel else {}
|
|
|
base_alt = params.get('base_altitude', params.get('altitude', 10.0))
|
|
|
target_gps = (lat, lon, base_alt)
|
|
|
self.statusBar().showMessage(
|
|
|
f"⏳ Group {group.group_id}: 正在規劃 {group.mission_type} ({len(selected_drones)} 台) ...", 2000)
|
|
|
|
|
|
try:
|
|
|
drone_gps_positions = self._collect_drone_gps(selected_drones, base_alt)
|
|
|
if drone_gps_positions is None:
|
|
|
return
|
|
|
|
|
|
waypoints_per_drone, center_origin, rendezvous_indices = self.mission_planner.plan_formation_mission(
|
|
|
drone_gps_positions, target_gps, mission_type, params=params
|
|
|
)
|
|
|
|
|
|
group.planned_waypoints = {
|
|
|
'drone_ids': selected_drones,
|
|
|
'waypoints': waypoints_per_drone,
|
|
|
'rendezvous_indices': rendezvous_indices,
|
|
|
}
|
|
|
group.center_origin = center_origin
|
|
|
self.show_planned_waypoints(group)
|
|
|
|
|
|
center_lat, center_lon, _ = center_origin
|
|
|
self.drone_map.draw_mission_plan_for_group(
|
|
|
group.group_id, group.color,
|
|
|
center_lat, center_lon, lat, lon
|
|
|
)
|
|
|
|
|
|
self._launch_verification(
|
|
|
group.mission_type, drone_gps_positions, selected_drones,
|
|
|
waypoints_per_drone, center_origin, target_gps=target_gps
|
|
|
)
|
|
|
|
|
|
panel = self.group_panels.get(group.group_id)
|
|
|
if panel:
|
|
|
panel.update_status()
|
|
|
panel.update_mission_info(center_lat, center_lon, lat, lon)
|
|
|
|
|
|
total_wps = sum(len(wps) for wps in waypoints_per_drone)
|
|
|
self.statusBar().showMessage(
|
|
|
f"✓ Group {group.group_id}: {group.mission_type} 規劃完成!{len(selected_drones)} 台,共 {total_wps} 個航點", 5000
|
|
|
)
|
|
|
except Exception as e:
|
|
|
self.statusBar().showMessage(f"❌ Group {group.group_id}: 規劃失敗: {str(e)}", 5000)
|
|
|
traceback.print_exc()
|
|
|
|
|
|
# ================================================================================
|
|
|
# 任務規劃 — 矩形選取 (Grid Sweep)
|
|
|
# ================================================================================
|
|
|
|
|
|
def handle_rectangle_selected(self, points_json):
|
|
|
group = self._get_active_group()
|
|
|
if not group:
|
|
|
self.statusBar().showMessage("⚠ 請先建立任務群組", 3000)
|
|
|
return
|
|
|
if group.mission_type != 'GRID_SWEEP':
|
|
|
return # 非 Grid Sweep 模式不處理矩形選取
|
|
|
|
|
|
selected_drones = self._get_group_drones(group)
|
|
|
if len(selected_drones) == 0:
|
|
|
self.statusBar().showMessage(f"⚠ Group {group.group_id}: 請先分配無人機", 3000)
|
|
|
return
|
|
|
|
|
|
print(f"矩形選取 → Group {group.group_id}: {points_json}")
|
|
|
try:
|
|
|
rect_corners = [(p[0], p[1]) for p in json.loads(points_json)]
|
|
|
except (json.JSONDecodeError, IndexError):
|
|
|
self.statusBar().showMessage("❌ 矩形座標解析失敗", 3000)
|
|
|
return
|
|
|
|
|
|
panel = self.group_panels.get(group.group_id)
|
|
|
params = panel.get_mission_params() if panel else {}
|
|
|
base_alt = params.get('altitude', 10.0)
|
|
|
self.statusBar().showMessage(
|
|
|
f"⏳ Group {group.group_id}: 正在規劃 Grid Sweep ({len(selected_drones)} 台) ...", 2000)
|
|
|
try:
|
|
|
drone_gps_positions = self._collect_drone_gps(selected_drones, base_alt)
|
|
|
if drone_gps_positions is None:
|
|
|
return
|
|
|
|
|
|
target_lat = sum(c[0] for c in rect_corners) / 4
|
|
|
target_lon = sum(c[1] for c in rect_corners) / 4
|
|
|
target_gps = (target_lat, target_lon, base_alt)
|
|
|
|
|
|
params['rect_corners'] = rect_corners
|
|
|
waypoints_per_drone, center_origin, rendezvous_indices = self.mission_planner.plan_formation_mission(
|
|
|
drone_gps_positions, target_gps, MissionType.GRID_SWEEP,
|
|
|
params=params
|
|
|
)
|
|
|
|
|
|
group.planned_waypoints = {
|
|
|
'drone_ids': selected_drones,
|
|
|
'waypoints': waypoints_per_drone,
|
|
|
'rendezvous_indices': rendezvous_indices,
|
|
|
}
|
|
|
group.center_origin = center_origin
|
|
|
self.show_planned_waypoints(group)
|
|
|
|
|
|
center_lat, center_lon, _ = center_origin
|
|
|
self.drone_map.draw_mission_plan_for_group(
|
|
|
group.group_id, group.color,
|
|
|
center_lat, center_lon, target_lat, target_lon
|
|
|
)
|
|
|
self._launch_verification(
|
|
|
'grid_sweep', drone_gps_positions, selected_drones,
|
|
|
waypoints_per_drone, center_origin, rect_corners=rect_corners
|
|
|
)
|
|
|
|
|
|
panel = self.group_panels.get(group.group_id)
|
|
|
if panel:
|
|
|
panel.update_status()
|
|
|
panel.update_mission_info(center_lat, center_lon, target_lat, target_lon)
|
|
|
|
|
|
total_wps = sum(len(wps) for wps in waypoints_per_drone)
|
|
|
self.statusBar().showMessage(
|
|
|
f"✓ Group {group.group_id}: Grid Sweep 規劃完成!{len(selected_drones)} 台,共 {total_wps} 個航點", 5000
|
|
|
)
|
|
|
except Exception as e:
|
|
|
self.statusBar().showMessage(f"❌ Group {group.group_id}: Grid Sweep 規劃失敗: {str(e)}", 5000)
|
|
|
traceback.print_exc()
|
|
|
|
|
|
# ================================================================================
|
|
|
# 任務規劃 — 路徑確認 (Leader-Follower)
|
|
|
# ================================================================================
|
|
|
|
|
|
def handle_route_confirmed(self, points_json):
|
|
|
group = self._get_active_group()
|
|
|
if not group:
|
|
|
self.statusBar().showMessage("⚠ 請先建立任務群組", 3000)
|
|
|
return
|
|
|
if group.mission_type != 'LEADER_FOLLOWER':
|
|
|
return
|
|
|
|
|
|
selected_drones = self._get_group_drones(group)
|
|
|
if len(selected_drones) == 0:
|
|
|
self.statusBar().showMessage(f"⚠ Group {group.group_id}: 請先分配無人機", 3000)
|
|
|
return
|
|
|
|
|
|
# LEADER_FOLLOWER:把指定的領隊放到 rank 0
|
|
|
leader = getattr(group, 'leader_drone_id', None)
|
|
|
if leader and leader in selected_drones:
|
|
|
selected_drones = [leader] + [d for d in selected_drones if d != leader]
|
|
|
|
|
|
print(f"路徑確認 → Group {group.group_id}: {points_json}")
|
|
|
try:
|
|
|
route_points = json.loads(points_json)
|
|
|
route_waypoints = [(p[0], p[1]) for p in route_points]
|
|
|
except (json.JSONDecodeError, IndexError):
|
|
|
self.statusBar().showMessage("❌ 路徑座標解析失敗", 3000)
|
|
|
return
|
|
|
|
|
|
if len(route_waypoints) < 2:
|
|
|
self.statusBar().showMessage("⚠ 至少需要 2 個路徑點", 3000)
|
|
|
return
|
|
|
|
|
|
panel = self.group_panels.get(group.group_id)
|
|
|
params = panel.get_mission_params() if panel else {}
|
|
|
base_alt = params.get('altitude', 10.0)
|
|
|
self.statusBar().showMessage(
|
|
|
f"⏳ Group {group.group_id}: 正在規劃跟隨模式 ({len(selected_drones)} 台) ...", 2000)
|
|
|
|
|
|
try:
|
|
|
drone_gps_positions = self._collect_drone_gps(selected_drones, base_alt)
|
|
|
if drone_gps_positions is None:
|
|
|
return
|
|
|
|
|
|
target_lat = sum(p[0] for p in route_waypoints) / len(route_waypoints)
|
|
|
target_lon = sum(p[1] for p in route_waypoints) / len(route_waypoints)
|
|
|
target_gps = (target_lat, target_lon, base_alt)
|
|
|
|
|
|
params['route_waypoints'] = route_waypoints
|
|
|
waypoints_per_drone, center_origin, rendezvous_indices = self.mission_planner.plan_formation_mission(
|
|
|
drone_gps_positions, target_gps, MissionType.LEADER_FOLLOWER,
|
|
|
params=params
|
|
|
)
|
|
|
|
|
|
group.planned_waypoints = {
|
|
|
'drone_ids': selected_drones,
|
|
|
'waypoints': waypoints_per_drone,
|
|
|
'rendezvous_indices': rendezvous_indices,
|
|
|
}
|
|
|
group.center_origin = center_origin
|
|
|
self.show_planned_waypoints(group)
|
|
|
|
|
|
center_lat, center_lon, _ = center_origin
|
|
|
self.drone_map.draw_mission_plan_for_group(
|
|
|
group.group_id, group.color,
|
|
|
center_lat, center_lon, target_lat, target_lon
|
|
|
)
|
|
|
|
|
|
self._launch_verification(
|
|
|
'LEADER_FOLLOWER', drone_gps_positions, selected_drones,
|
|
|
waypoints_per_drone, center_origin,
|
|
|
target_gps=target_gps, route_waypoints=route_waypoints
|
|
|
)
|
|
|
|
|
|
panel = self.group_panels.get(group.group_id)
|
|
|
if panel:
|
|
|
panel.update_status()
|
|
|
panel.update_mission_info(center_lat, center_lon, target_lat, target_lon)
|
|
|
|
|
|
total_wps = sum(len(wps) for wps in waypoints_per_drone)
|
|
|
self.statusBar().showMessage(
|
|
|
f"✓ Group {group.group_id}: 跟隨模式規劃完成!{len(selected_drones)} 台,共 {total_wps} 個航點", 5000
|
|
|
)
|
|
|
except Exception as e:
|
|
|
self.statusBar().showMessage(f"❌ Group {group.group_id}: 跟隨模式規劃失敗: {str(e)}", 5000)
|
|
|
traceback.print_exc()
|
|
|
|
|
|
# ================================================================================
|
|
|
# 任務執行回呼
|
|
|
# ================================================================================
|
|
|
|
|
|
def on_drone_waypoint_reached(self, drone_id, wp_index, total):
|
|
|
if wp_index >= total:
|
|
|
self.statusBar().showMessage(f"📍 {drone_id} 完成所有航點", 3000)
|
|
|
else:
|
|
|
self.statusBar().showMessage(f"📍 {drone_id} → WP {wp_index}/{total}", 2000)
|
|
|
|
|
|
def _on_task_status_changed(self, drone_id, status, message):
|
|
|
"""MissionExecutor.task_status_changed slot:把 goto 失敗/重試/fallback/barrier 丟到 status bar"""
|
|
|
if status == "retrying":
|
|
|
self.statusBar().showMessage(f"⚠ {drone_id} {message}", 4000)
|
|
|
elif status == "fallback_loiter":
|
|
|
self.statusBar().showMessage(f"❌ {drone_id} {message}", 8000)
|
|
|
elif status == "waiting_at_barrier":
|
|
|
self.statusBar().showMessage(f"⏸ {drone_id} {message}", 3000)
|
|
|
elif status == "normal":
|
|
|
self.statusBar().showMessage(f"▶ {drone_id} {message or '已恢復'}", 2000)
|
|
|
|
|
|
# ================================================================================
|
|
|
# 輔助方法
|
|
|
# ================================================================================
|
|
|
|
|
|
def _collect_drone_gps(self, selected_drones, base_alt):
|
|
|
drone_gps_positions = []
|
|
|
for drone_id in selected_drones:
|
|
|
if hasattr(self.monitor, 'drone_gps') and drone_id in self.monitor.drone_gps:
|
|
|
gps_data = self.monitor.drone_gps[drone_id]
|
|
|
drone_gps_positions.append((gps_data['lat'], gps_data['lon'], gps_data.get('alt', base_alt)))
|
|
|
elif drone_id in self.drone_positions:
|
|
|
pos = self.drone_positions[drone_id]
|
|
|
lat_drone = 24.0 + pos[1] / 111000
|
|
|
lon_drone = 120.0 + pos[0] / (111000 * 0.9)
|
|
|
alt_drone = pos[2] if len(pos) > 2 else base_alt
|
|
|
drone_gps_positions.append((lat_drone, lon_drone, alt_drone))
|
|
|
else:
|
|
|
self.statusBar().showMessage(f"⚠ 找不到 {drone_id} 的位置資料", 3000)
|
|
|
return None
|
|
|
return drone_gps_positions
|
|
|
|
|
|
def _launch_verification(self, mission_type, drone_gps_positions,
|
|
|
selected_drones, waypoints_per_drone, origin,
|
|
|
target_gps=None, rect_corners=None, route_waypoints=None):
|
|
|
"""存 JSON + 啟動 matplotlib 視覺化驗證 (獨立 process)"""
|
|
|
import os
|
|
|
data = {
|
|
|
'mission_type': mission_type,
|
|
|
'drone_ids': selected_drones,
|
|
|
'drones_gps': drone_gps_positions,
|
|
|
'waypoints': waypoints_per_drone,
|
|
|
'origin': list(origin),
|
|
|
}
|
|
|
if target_gps:
|
|
|
data['target'] = list(target_gps)
|
|
|
if rect_corners:
|
|
|
data['rect_corners'] = rect_corners
|
|
|
if route_waypoints:
|
|
|
data['route_waypoints'] = route_waypoints
|
|
|
|
|
|
json_path = '/tmp/mission_plan.json'
|
|
|
with open(json_path, 'w') as f:
|
|
|
json.dump(data, f, indent=2)
|
|
|
|
|
|
script_path = os.path.join(os.path.dirname(os.path.abspath(__file__)), 'validation', 'verify_waypoints.py')
|
|
|
subprocess.Popen([sys.executable, script_path, '--file', json_path])
|
|
|
print(f"驗證視窗已啟動: {json_path}")
|
|
|
|
|
|
def show_planned_waypoints(self, group=None):
|
|
|
pw = group.planned_waypoints if group else None
|
|
|
if not pw:
|
|
|
return
|
|
|
gid = group.group_id if group else "?"
|
|
|
print(f"\n{'=' * 60}")
|
|
|
print(f"任務規劃結果 — Group {gid}")
|
|
|
print(f"{'=' * 60}")
|
|
|
drone_ids = pw['drone_ids']
|
|
|
waypoints = pw['waypoints']
|
|
|
print(f"\n共 {len(drone_ids)} 台無人機")
|
|
|
for i, drone_id in enumerate(drone_ids):
|
|
|
wps = waypoints[i]
|
|
|
print(f"\n【{drone_id}】({len(wps)} 個航點)")
|
|
|
for j, wp in enumerate(wps):
|
|
|
print(f" WP{j}: ({wp[0]:.6f}°, {wp[1]:.6f}°, {wp[2]:.1f}m)")
|
|
|
print(f"\n{'=' * 60}")
|
|
|
|
|
|
def update_field(self, panel, drone_id, field, text, color=None):
|
|
|
if isinstance(panel, DronePanel):
|
|
|
panel.update_field(field, text, color)
|
|
|
|
|
|
def update_overview_table(self, drone_id=None, field=None, value=None):
|
|
|
if not hasattr(self, 'overview_table') or self.overview_table is None: return
|
|
|
self.overview_table.set_drones(self.drones)
|
|
|
self.overview_table.update_table(drone_id, field, value)
|
|
|
|
|
|
def get_socket_id(self, drone_id):
|
|
|
import re
|
|
|
match = re.match(r's(\d+)_(\d+)', drone_id)
|
|
|
return match.group(1) if match else 'unknown'
|
|
|
|
|
|
def add_drone(self, drone_id):
|
|
|
if drone_id in self.drones: return
|
|
|
socket_id = self.get_socket_id(drone_id)
|
|
|
panel = self.create_drone_panel(drone_id)
|
|
|
self.drones[drone_id] = panel
|
|
|
if socket_id not in self.socket_groups:
|
|
|
self.socket_groups[socket_id] = self.create_socket_group_panel(socket_id)
|
|
|
self.socket_groups[socket_id].drones_layout.addWidget(panel)
|
|
|
self.reorganize_socket_groups()
|
|
|
self.update_overview_table()
|
|
|
# 同步新 drone 到 UI
|
|
|
self.refresh_selection_ui()
|
|
|
|
|
|
def reorganize_socket_groups(self):
|
|
|
while self.drone_panel_layout.count():
|
|
|
w = self.drone_panel_layout.takeAt(0).widget()
|
|
|
if w: w.setParent(None)
|
|
|
for socket_id, group_panel in self.socket_groups.items():
|
|
|
group_drones = [did for did in self.drones.keys() if self.get_socket_id(did) == socket_id]
|
|
|
while group_panel.drones_layout.count():
|
|
|
w = group_panel.drones_layout.takeAt(0).widget()
|
|
|
if w: w.setParent(None)
|
|
|
def sort_key(x):
|
|
|
parts = x[1:].split('_')
|
|
|
return (int(parts[0]), int(parts[1]))
|
|
|
for did in sorted(group_drones, key=sort_key):
|
|
|
group_panel.drones_layout.addWidget(self.drones[did])
|
|
|
for socket_id in sorted(self.socket_groups.keys(), key=lambda x: int(x)):
|
|
|
self.drone_panel_layout.addWidget(self.socket_groups[socket_id])
|
|
|
|
|
|
def _update_panel_and_map(self):
|
|
|
"""30Hz 定時更新 panel 和 map,批量更新 UI 以避免過度重繪"""
|
|
|
if not hasattr(self, '_message_cache') or not self._message_cache:
|
|
|
return
|
|
|
|
|
|
# 頻率監控
|
|
|
if not hasattr(self, '_map_update_time'):
|
|
|
self._map_update_time = time.time()
|
|
|
self._map_update_count = 0
|
|
|
|
|
|
self._map_update_count += 1
|
|
|
now = time.time()
|
|
|
if now - self._map_update_time >= 1.0:
|
|
|
self._map_update_time = now
|
|
|
self._map_update_count = 0
|
|
|
|
|
|
# ✅ 步驟 1: 暫停表格的即時重繪
|
|
|
if hasattr(self, 'overview_table') and self.overview_table:
|
|
|
self.overview_table.setUpdatesEnabled(False)
|
|
|
|
|
|
try:
|
|
|
start_time = time.time()
|
|
|
|
|
|
# ✅ 步驟 2: 遍歷快取中最新的資料來更新 UI
|
|
|
for drone_id in list(self._message_cache.keys()):
|
|
|
if drone_id not in self.drones:
|
|
|
continue
|
|
|
|
|
|
panel = self.drones[drone_id]
|
|
|
cached_data = self._message_cache[drone_id]
|
|
|
|
|
|
# 處理所有快取的消息類型
|
|
|
for msg_type, data in cached_data.items():
|
|
|
if msg_type == 'state':
|
|
|
mode = data.get('mode', 'UNKNOWN')
|
|
|
armed = data.get('armed', None)
|
|
|
mode_color = '#FF5555' if any(x in mode.upper() for x in ['RTL', '返航', 'EMERGENCY']) else '#55FF55'
|
|
|
if armed is True:
|
|
|
arm_text, arm_color = "ARMED", '#55FF55'
|
|
|
elif armed is False:
|
|
|
arm_text, arm_color = "DISARMED", '#FF5555'
|
|
|
else:
|
|
|
arm_text, arm_color = "--", '#AAAAAA'
|
|
|
self.update_field(panel, drone_id, 'mode', mode, mode_color)
|
|
|
self.update_field(panel, drone_id, 'armed', arm_text, arm_color)
|
|
|
self.update_overview_table(drone_id, 'mode', mode)
|
|
|
self.update_overview_table(drone_id, 'armed', arm_text)
|
|
|
|
|
|
elif msg_type == 'battery':
|
|
|
voltage = data.get('voltage', 16)
|
|
|
cells = round(voltage / 3.95)
|
|
|
percentage = (voltage / cells - 3.7) / 0.5 * 100 if cells > 0 else 0
|
|
|
if percentage < 20: voltage_color = '#FF6464'
|
|
|
elif percentage < 50: voltage_color = '#FFA500'
|
|
|
else: voltage_color = '#FFFFFF'
|
|
|
percentage = data.get('percentage', percentage)
|
|
|
self.update_field(panel, drone_id, 'battery_pct', f"{percentage:.0f}%", voltage_color)
|
|
|
self.update_field(panel, drone_id, 'battery_vol', f"{voltage:.2f}V")
|
|
|
self.update_field(panel, drone_id, 'battery_cells', f"{cells}S")
|
|
|
self.update_overview_table(drone_id, 'battery', f"{voltage:.2f}V")
|
|
|
|
|
|
elif msg_type == 'altitude':
|
|
|
altitude = data.get('altitude', 0)
|
|
|
text = f"{altitude:.1f} m"
|
|
|
self.update_field(panel, drone_id, 'altitude', text)
|
|
|
self.update_overview_table(drone_id, 'altitude', text)
|
|
|
|
|
|
elif msg_type == 'local_pose':
|
|
|
x, y = data.get('x', 0), data.get('y', 0)
|
|
|
if not hasattr(self.monitor, 'drone_local'):
|
|
|
self.monitor.drone_local = {}
|
|
|
self.monitor.drone_local[drone_id] = {'x': x, 'y': y}
|
|
|
self.update_overview_table(drone_id, 'local', f"{x:.1f}, {y:.1f}")
|
|
|
|
|
|
elif msg_type == 'loss_rate':
|
|
|
text = f"{data.get('loss_rate', 0):.1f}%"
|
|
|
self.update_field(panel, drone_id, 'loss_rate', text)
|
|
|
self.update_overview_table(drone_id, 'loss_rate', text)
|
|
|
|
|
|
elif msg_type == 'ping':
|
|
|
text = f"{data.get('ping', 0):.1f} ms"
|
|
|
self.update_field(panel, drone_id, 'ping', text)
|
|
|
self.update_overview_table(drone_id, 'ping', text)
|
|
|
|
|
|
elif msg_type == 'velocity':
|
|
|
self.update_overview_table(drone_id, 'velocity', f"{data['vx']:.1f}, {data['vy']:.1f}")
|
|
|
|
|
|
elif msg_type == 'attitude':
|
|
|
roll, pitch, yaw = data.get('roll', 0), data.get('pitch', 0), data.get('yaw', 0)
|
|
|
self.update_overview_table(drone_id, 'roll', f"{roll:.1f}°")
|
|
|
self.update_overview_table(drone_id, 'pitch', f"{pitch:.1f}°")
|
|
|
self.update_overview_table(drone_id, 'yaw', f"{yaw:.1f}°")
|
|
|
panel._last_roll = roll
|
|
|
panel._last_pitch = pitch
|
|
|
if hasattr(panel, 'update_attitude'):
|
|
|
heading = self.drone_headings.get(drone_id, 0)
|
|
|
panel.update_attitude(heading, roll, pitch)
|
|
|
|
|
|
elif msg_type == 'gps':
|
|
|
gps_data = data
|
|
|
lat, lon = gps_data.get('lat', 0), gps_data.get('lon', 0)
|
|
|
self.drone_positions[drone_id] = (lat, lon)
|
|
|
alt = gps_data.get('alt', 0)
|
|
|
if not hasattr(self.monitor, 'drone_gps'):
|
|
|
self.monitor.drone_gps = {}
|
|
|
self.monitor.drone_gps[drone_id] = {'lat': lat, 'lon': lon, 'alt': alt}
|
|
|
self.update_overview_table(drone_id, 'latitude', f"{lat:.6f}°")
|
|
|
self.update_overview_table(drone_id, 'longitude', f"{lon:.6f}°")
|
|
|
|
|
|
elif msg_type == 'hud':
|
|
|
hud_data = data
|
|
|
heading = hud_data.get('heading', 0)
|
|
|
self.drone_headings[drone_id] = heading
|
|
|
groundspeed = hud_data.get('groundspeed', 0)
|
|
|
airspeed = hud_data.get('airspeed', 0)
|
|
|
throttle = hud_data.get('throttle', 0)
|
|
|
hud_alt = hud_data.get('alt', 0)
|
|
|
climb = hud_data.get('climb', 0)
|
|
|
|
|
|
self.update_overview_table(drone_id, 'heading', f"{heading:.1f}°")
|
|
|
self.update_overview_table(drone_id, 'groundspeed', f"{groundspeed:.1f} m/s" if isinstance(groundspeed, (int, float)) else "--")
|
|
|
self.update_overview_table(drone_id, 'airspeed', f"{airspeed:.1f} m/s" if isinstance(airspeed, (int, float)) else "--")
|
|
|
self.update_overview_table(drone_id, 'throttle', f"{throttle:.0f}%" if isinstance(throttle, (int, float)) else "--")
|
|
|
self.update_overview_table(drone_id, 'hud_alt', f"{hud_alt:.1f} m" if isinstance(hud_alt, (int, float)) else "--")
|
|
|
self.update_overview_table(drone_id, 'climb', f"{climb:.1f} m/s" if isinstance(climb, (int, float)) else "--")
|
|
|
|
|
|
self.update_field(panel, drone_id, 'heading', f"{heading:.1f}°")
|
|
|
self.update_field(panel, drone_id, 'groundspeed', f"{groundspeed:.1f} m/s" if isinstance(groundspeed, (int, float)) else "--")
|
|
|
self.update_field(panel, drone_id, 'speed', f"{groundspeed:.1f} m/s" if isinstance(groundspeed, (int, float)) else "--")
|
|
|
|
|
|
if drone_id in self.drone_positions:
|
|
|
lat, lon = self.drone_positions[drone_id]
|
|
|
self.drone_map.update_drone_position(drone_id, lat, lon, heading)
|
|
|
|
|
|
elapsed = (time.time() - start_time) * 1000
|
|
|
if elapsed > 33:
|
|
|
print(f"[WARNING] UI update took {elapsed:.1f}ms (target: <33ms)")
|
|
|
|
|
|
finally:
|
|
|
# ✅ 步驟 3: 恢復表格重繪(所有資料已填好,一次性重繪)
|
|
|
if hasattr(self, 'overview_table') and self.overview_table:
|
|
|
self.overview_table.setUpdatesEnabled(True)
|
|
|
self.overview_table.viewport().update()
|
|
|
|
|
|
|
|
|
|
|
|
def _process_message_queue(self):
|
|
|
"""處理來自後台線程的消息隊列(更新 GUI 狀態欄)"""
|
|
|
try:
|
|
|
while True:
|
|
|
try:
|
|
|
message, duration = self.message_queue.get_nowait()
|
|
|
self.statusBar().showMessage(message, duration)
|
|
|
except:
|
|
|
break
|
|
|
except Exception as e:
|
|
|
print(f"❌ 消息隊列處理錯誤: {e}", flush=True)
|
|
|
traceback.print_exc()
|
|
|
|
|
|
def _spin_asyncio(self):
|
|
|
"""驅動 asyncio 事件循環,允許異步任務執行
|
|
|
|
|
|
關鍵修復:asyncio 事件循環不會自動運行。
|
|
|
這個定時器會定期執行事件循環,讓 run_coroutine_threadsafe() 提交的任務能夠執行。
|
|
|
"""
|
|
|
try:
|
|
|
loop = asyncio.get_event_loop()
|
|
|
if loop and not loop.is_closed() and not loop.is_running():
|
|
|
# 執行事件循環直到沒有掛起的任務或超時
|
|
|
# 使用 run_until_complete() 配合一個快速返回的 coroutine
|
|
|
loop.run_until_complete(asyncio.sleep(0))
|
|
|
except Exception as e:
|
|
|
# 靜默忽略任何錯誤,防止 Qt 定時器出現異常
|
|
|
# 但仍然打印詳細的堆棧跟踪以便調試
|
|
|
traceback.print_exc()
|
|
|
|
|
|
def spin_ros(self):
|
|
|
try:
|
|
|
# 仅在 ROS2 正常工作时才尝试 spin
|
|
|
if rclpy.ok():
|
|
|
self.executor.spin_once(timeout_sec=0.01)
|
|
|
for (drone_id, msg_type), data in self.monitor.latest_data.items():
|
|
|
self.monitor.signals.update_signal.emit(msg_type, drone_id, data)
|
|
|
self.monitor.latest_data.clear()
|
|
|
except RuntimeError as e:
|
|
|
# ROS2 context 被破坏或不可用
|
|
|
if "Context" in str(e) or "context" in str(e).lower():
|
|
|
print(f"⚠️ ROS2 context 錯誤(忽略): {e}", flush=True)
|
|
|
else:
|
|
|
print(f"ROS spin error: {e}", flush=True)
|
|
|
traceback.print_exc()
|
|
|
except Exception as e:
|
|
|
print(f"ROS spin error: {e}", flush=True)
|
|
|
traceback.print_exc()
|
|
|
|
|
|
def closeEvent(self, event):
|
|
|
try:
|
|
|
for group in self.mission_groups.values():
|
|
|
if group.executor:
|
|
|
group.executor.stop()
|
|
|
self.command_sender.close()
|
|
|
for receiver in self.udp_receivers:
|
|
|
receiver.stop()
|
|
|
for receiver in self.monitor.ws_receivers:
|
|
|
receiver.stop()
|
|
|
# Clean up serial receivers
|
|
|
for receiver in self.monitor.serial_receivers:
|
|
|
receiver.stop()
|
|
|
# Clean up all CommandLongClient instances
|
|
|
for drone_id, client in self.monitor.command_long_clients.items():
|
|
|
try:
|
|
|
client.destroy_node()
|
|
|
except:
|
|
|
pass
|
|
|
# Clean up all PositionTargetGlobalIntClient instances
|
|
|
for drone_id, client in getattr(self.monitor, 'position_target_clients', {}).items():
|
|
|
try:
|
|
|
client.destroy_node()
|
|
|
except:
|
|
|
pass
|
|
|
self.monitor.destroy_node()
|
|
|
self.executor.shutdown()
|
|
|
except Exception as e:
|
|
|
print(f"⚠️ 清理資源時出錯: {e}", flush=True)
|
|
|
traceback.print_exc()
|
|
|
|
|
|
# 安全地 shutdown ROS2
|
|
|
try:
|
|
|
if rclpy.ok():
|
|
|
rclpy.shutdown()
|
|
|
except RuntimeError as e:
|
|
|
print(f"⚠️ ROS2 shutdown 錯誤: {e}", flush=True)
|
|
|
traceback.print_exc()
|
|
|
|
|
|
event.accept()
|
|
|
|
|
|
|
|
|
def main():
|
|
|
"""
|
|
|
GUI 應用程式的主入口點
|
|
|
|
|
|
標準 ROS2 + Qt 架構:
|
|
|
1. 在最外層/最前面只做一次 rclpy.init()
|
|
|
2. 啟動 Qt 應用程式
|
|
|
3. 在 finally 中做 rclpy.shutdown()
|
|
|
|
|
|
這樣可以確保所有 ROS2 相關操作都共享同一個初始化狀態
|
|
|
"""
|
|
|
# 第一步:在最外層只初始化一次 ROS2(終極防護)
|
|
|
# 添加 rclpy.ok() 檢查,防止重複初始化導致 "Context.init() must only be called once" 錯誤
|
|
|
print("🚀 [GUI 主程式] 檢查 ROS2 初始化狀態...", flush=True)
|
|
|
if not rclpy.ok():
|
|
|
print("🚀 [GUI 主程式] ROS2 未初始化,開始初始化...", flush=True)
|
|
|
rclpy.init()
|
|
|
print("✅ [GUI 主程式] ROS2 已全局初始化(由 GUI 主程式霸佔)", flush=True)
|
|
|
else:
|
|
|
print("ℹ️ [GUI 主程式] ROS2 已初始化,跳過重複初始化", flush=True)
|
|
|
|
|
|
try:
|
|
|
# 第二步:啟動 Qt 應用程式
|
|
|
print("🚀 [GUI 主程式] 啟動 Qt 應用程式...", flush=True)
|
|
|
app = QApplication(sys.argv)
|
|
|
station = ControlStationUI()
|
|
|
station.show()
|
|
|
print("✓ [GUI 主程式] 應用程式已啟動", flush=True)
|
|
|
|
|
|
# 第三步:進入 Qt 事件循環(阻塞直到用戶關閉應用)
|
|
|
print("🎯 [GUI 主程式] 進入主事件循環,等待用戶操作...", flush=True)
|
|
|
sys.exit(app.exec())
|
|
|
|
|
|
finally:
|
|
|
# 第四步:只有當 GUI 視窗被關閉時,才做 ROS2 cleanup(終極防護)
|
|
|
# 這裡確保 ROS2 被正確且安全地關閉
|
|
|
print("\n🛑 [GUI 主程式] 應用程式關閉,正在清理 ROS2 資源...", flush=True)
|
|
|
if rclpy.ok():
|
|
|
rclpy.shutdown()
|
|
|
print("✓ [GUI 主程式] ROS2 已安全關閉", flush=True)
|
|
|
else:
|
|
|
print("ℹ️ [GUI 主程式] ROS2 已關閉或不可用,無需重複 shutdown", flush=True)
|
|
|
print("✓ [GUI 主程式] 應用程式完全退出", flush=True)
|
|
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
main()
|