|
|
<!DOCTYPE html>
|
|
|
<html lang="zh-CN">
|
|
|
|
|
|
<head>
|
|
|
<meta charset="UTF-8">
|
|
|
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
|
|
<title>无人机控制 - 距离判断系统</title>
|
|
|
|
|
|
<!-- 内联Bootstrap CSS - 避免CDN依赖 -->
|
|
|
<style>
|
|
|
/* Bootstrap 5.1.3 基础样式 - 简化版 */
|
|
|
*,
|
|
|
*::before,
|
|
|
*::after {
|
|
|
box-sizing: border-box;
|
|
|
}
|
|
|
|
|
|
body {
|
|
|
margin: 0;
|
|
|
font-family: system-ui, -apple-system, "Segoe UI", Roboto, "Helvetica Neue", Arial, sans-serif;
|
|
|
font-size: 1rem;
|
|
|
font-weight: 400;
|
|
|
line-height: 1.5;
|
|
|
color: #212529;
|
|
|
background-color: #f8f9fa;
|
|
|
}
|
|
|
|
|
|
.container-fluid {
|
|
|
width: 100%;
|
|
|
padding-right: 0.75rem;
|
|
|
padding-left: 0.75rem;
|
|
|
margin-right: auto;
|
|
|
margin-left: auto;
|
|
|
}
|
|
|
|
|
|
.row {
|
|
|
display: flex;
|
|
|
flex-wrap: wrap;
|
|
|
margin-right: -0.75rem;
|
|
|
margin-left: -0.75rem;
|
|
|
}
|
|
|
|
|
|
.col-12 {
|
|
|
flex: 0 0 auto;
|
|
|
width: 100%;
|
|
|
}
|
|
|
|
|
|
.col-lg-6 {
|
|
|
flex: 0 0 auto;
|
|
|
width: 50%;
|
|
|
}
|
|
|
|
|
|
@media (max-width: 991.98px) {
|
|
|
.col-lg-6 {
|
|
|
width: 100%;
|
|
|
}
|
|
|
}
|
|
|
|
|
|
.py-4 {
|
|
|
padding-top: 1.5rem !important;
|
|
|
padding-bottom: 1.5rem !important;
|
|
|
}
|
|
|
|
|
|
.mb-4 {
|
|
|
margin-bottom: 1.5rem !important;
|
|
|
}
|
|
|
|
|
|
.mb-3 {
|
|
|
margin-bottom: 1rem !important;
|
|
|
}
|
|
|
|
|
|
.mb-0 {
|
|
|
margin-bottom: 0 !important;
|
|
|
}
|
|
|
|
|
|
.me-2 {
|
|
|
margin-right: 0.5rem !important;
|
|
|
}
|
|
|
|
|
|
.me-3 {
|
|
|
margin-right: 1rem !important;
|
|
|
}
|
|
|
|
|
|
.ms-auto {
|
|
|
margin-left: auto !important;
|
|
|
}
|
|
|
|
|
|
.text-center {
|
|
|
text-align: center !important;
|
|
|
}
|
|
|
|
|
|
.text-muted {
|
|
|
color: #6c757d !important;
|
|
|
}
|
|
|
|
|
|
.d-flex {
|
|
|
display: flex !important;
|
|
|
}
|
|
|
|
|
|
.d-block {
|
|
|
display: block !important;
|
|
|
}
|
|
|
|
|
|
.align-items-center {
|
|
|
align-items: center !important;
|
|
|
}
|
|
|
|
|
|
.justify-content-center {
|
|
|
justify-content: center !important;
|
|
|
}
|
|
|
|
|
|
.btn {
|
|
|
display: inline-block;
|
|
|
padding: 0.375rem 0.75rem;
|
|
|
margin-bottom: 0;
|
|
|
font-size: 1rem;
|
|
|
font-weight: 400;
|
|
|
line-height: 1.5;
|
|
|
color: #212529;
|
|
|
text-align: center;
|
|
|
text-decoration: none;
|
|
|
vertical-align: middle;
|
|
|
cursor: pointer;
|
|
|
border: 1px solid transparent;
|
|
|
border-radius: 0.25rem;
|
|
|
transition: color 0.15s ease-in-out, background-color 0.15s ease-in-out;
|
|
|
}
|
|
|
|
|
|
.btn:hover {
|
|
|
text-decoration: none;
|
|
|
}
|
|
|
|
|
|
.btn:disabled {
|
|
|
pointer-events: none;
|
|
|
opacity: 0.65;
|
|
|
}
|
|
|
|
|
|
.btn-primary {
|
|
|
color: #fff;
|
|
|
background-color: #0d6efd;
|
|
|
border-color: #0d6efd;
|
|
|
}
|
|
|
|
|
|
.btn-primary:hover {
|
|
|
background-color: #0b5ed7;
|
|
|
border-color: #0a58ca;
|
|
|
}
|
|
|
|
|
|
.btn-success {
|
|
|
color: #fff;
|
|
|
background-color: #198754;
|
|
|
border-color: #198754;
|
|
|
}
|
|
|
|
|
|
.btn-success:hover {
|
|
|
background-color: #157347;
|
|
|
border-color: #146c43;
|
|
|
}
|
|
|
|
|
|
.btn-danger {
|
|
|
color: #fff;
|
|
|
background-color: #dc3545;
|
|
|
border-color: #dc3545;
|
|
|
}
|
|
|
|
|
|
.btn-danger:hover {
|
|
|
background-color: #bb2d3b;
|
|
|
border-color: #b02a37;
|
|
|
}
|
|
|
|
|
|
.btn-warning {
|
|
|
color: #000;
|
|
|
background-color: #ffc107;
|
|
|
border-color: #ffc107;
|
|
|
}
|
|
|
|
|
|
.btn-warning:hover {
|
|
|
background-color: #ffca2c;
|
|
|
border-color: #ffc720;
|
|
|
}
|
|
|
|
|
|
.btn-info {
|
|
|
color: #000;
|
|
|
background-color: #0dcaf0;
|
|
|
border-color: #0dcaf0;
|
|
|
}
|
|
|
|
|
|
.btn-info:hover {
|
|
|
background-color: #31d2f2;
|
|
|
border-color: #25cff2;
|
|
|
}
|
|
|
|
|
|
.btn-light {
|
|
|
color: #000;
|
|
|
background-color: #f8f9fa;
|
|
|
border-color: #f8f9fa;
|
|
|
}
|
|
|
|
|
|
.btn-light:hover {
|
|
|
background-color: #f9fafb;
|
|
|
border-color: #f9fafb;
|
|
|
}
|
|
|
|
|
|
.btn-outline-light {
|
|
|
color: #f8f9fa;
|
|
|
border-color: #f8f9fa;
|
|
|
}
|
|
|
|
|
|
.btn-outline-light:hover {
|
|
|
color: #000;
|
|
|
background-color: #f8f9fa;
|
|
|
border-color: #f8f9fa;
|
|
|
}
|
|
|
|
|
|
.btn-outline-warning {
|
|
|
color: #ffc107;
|
|
|
border-color: #ffc107;
|
|
|
}
|
|
|
|
|
|
.btn-outline-warning:hover {
|
|
|
color: #000;
|
|
|
background-color: #ffc107;
|
|
|
border-color: #ffc107;
|
|
|
}
|
|
|
|
|
|
.btn-lg {
|
|
|
padding: 0.5rem 1rem;
|
|
|
font-size: 1.25rem;
|
|
|
}
|
|
|
|
|
|
.badge {
|
|
|
display: inline-block;
|
|
|
padding: 0.35em 0.65em;
|
|
|
font-size: 0.75em;
|
|
|
font-weight: 700;
|
|
|
line-height: 1;
|
|
|
color: #fff;
|
|
|
text-align: center;
|
|
|
white-space: nowrap;
|
|
|
vertical-align: baseline;
|
|
|
border-radius: 0.25rem;
|
|
|
}
|
|
|
|
|
|
.bg-danger {
|
|
|
background-color: #dc3545 !important;
|
|
|
}
|
|
|
|
|
|
.bg-success {
|
|
|
background-color: #198754 !important;
|
|
|
}
|
|
|
|
|
|
.bg-warning {
|
|
|
background-color: #ffc107 !important;
|
|
|
color: #000 !important;
|
|
|
}
|
|
|
|
|
|
.fs-6 {
|
|
|
font-size: 1rem !important;
|
|
|
}
|
|
|
|
|
|
.fs-5 {
|
|
|
font-size: 1.25rem !important;
|
|
|
}
|
|
|
|
|
|
.display-5 {
|
|
|
font-size: 2.5rem;
|
|
|
font-weight: 300;
|
|
|
line-height: 1.2;
|
|
|
}
|
|
|
|
|
|
.alert {
|
|
|
position: relative;
|
|
|
padding: 0.75rem 1.25rem;
|
|
|
margin-bottom: 1rem;
|
|
|
border: 1px solid transparent;
|
|
|
border-radius: 0.25rem;
|
|
|
}
|
|
|
|
|
|
.alert-info {
|
|
|
color: #055160;
|
|
|
background-color: #d1ecf1;
|
|
|
border-color: #bee5eb;
|
|
|
}
|
|
|
|
|
|
.alert-success {
|
|
|
color: #0f5132;
|
|
|
background-color: #d1e7dd;
|
|
|
border-color: #badbcc;
|
|
|
}
|
|
|
|
|
|
.alert-warning {
|
|
|
color: #664d03;
|
|
|
background-color: #fff3cd;
|
|
|
border-color: #ffecb5;
|
|
|
}
|
|
|
|
|
|
.alert-danger {
|
|
|
color: #721c24;
|
|
|
background-color: #f8d7da;
|
|
|
border-color: #f5c6cb;
|
|
|
}
|
|
|
|
|
|
.alert-secondary {
|
|
|
color: #383d41;
|
|
|
background-color: #e2e3e5;
|
|
|
border-color: #d6d8db;
|
|
|
}
|
|
|
|
|
|
.form-range {
|
|
|
width: 100%;
|
|
|
height: 1.5rem;
|
|
|
padding: 0;
|
|
|
background-color: transparent;
|
|
|
appearance: none;
|
|
|
-webkit-appearance: none;
|
|
|
}
|
|
|
|
|
|
.form-range::-webkit-slider-track {
|
|
|
width: 100%;
|
|
|
height: 0.5rem;
|
|
|
color: transparent;
|
|
|
cursor: pointer;
|
|
|
background-color: #dee2e6;
|
|
|
border-color: transparent;
|
|
|
border-radius: 1rem;
|
|
|
}
|
|
|
|
|
|
.form-range::-webkit-slider-thumb {
|
|
|
width: 1rem;
|
|
|
height: 1rem;
|
|
|
margin-top: -0.25rem;
|
|
|
background-color: #0d6efd;
|
|
|
border: 0;
|
|
|
border-radius: 1rem;
|
|
|
-webkit-appearance: none;
|
|
|
appearance: none;
|
|
|
}
|
|
|
|
|
|
.form-label {
|
|
|
margin-bottom: 0.5rem;
|
|
|
font-weight: 500;
|
|
|
}
|
|
|
|
|
|
.img-fluid {
|
|
|
max-width: 100%;
|
|
|
height: auto;
|
|
|
}
|
|
|
|
|
|
.rounded {
|
|
|
border-radius: 0.25rem !important;
|
|
|
}
|
|
|
|
|
|
/* 自定义样式 */
|
|
|
.video-container {
|
|
|
background: #000;
|
|
|
min-height: 300px;
|
|
|
display: flex;
|
|
|
align-items: center;
|
|
|
justify-content: center;
|
|
|
border-radius: 0.375rem;
|
|
|
}
|
|
|
|
|
|
.btn-grid {
|
|
|
display: grid;
|
|
|
grid-template-columns: repeat(3, 1fr);
|
|
|
gap: 10px;
|
|
|
max-width: 300px;
|
|
|
margin: 0 auto;
|
|
|
}
|
|
|
|
|
|
.btn-grid .btn {
|
|
|
min-height: 50px;
|
|
|
font-size: 14px;
|
|
|
}
|
|
|
|
|
|
.status-badge {
|
|
|
position: fixed;
|
|
|
top: 20px;
|
|
|
right: 20px;
|
|
|
z-index: 1000;
|
|
|
}
|
|
|
|
|
|
.control-panel {
|
|
|
background: linear-gradient(135deg, #667eea 0%, #764ba2 100%);
|
|
|
color: white;
|
|
|
border-radius: 15px;
|
|
|
padding: 20px;
|
|
|
margin-bottom: 20px;
|
|
|
}
|
|
|
|
|
|
.video-panel {
|
|
|
background: linear-gradient(135deg, #f093fb 0%, #f5576c 100%);
|
|
|
color: white;
|
|
|
border-radius: 15px;
|
|
|
padding: 20px;
|
|
|
}
|
|
|
|
|
|
.status-panel {
|
|
|
background: linear-gradient(135deg, #4facfe 0%, #00f2fe 100%);
|
|
|
color: white;
|
|
|
border-radius: 15px;
|
|
|
padding: 15px;
|
|
|
}
|
|
|
|
|
|
/* Font Awesome图标替换为文字符号 */
|
|
|
.fa-wifi::before {
|
|
|
content: "📶";
|
|
|
}
|
|
|
|
|
|
.fa-play::before {
|
|
|
content: "▶";
|
|
|
}
|
|
|
|
|
|
.fa-stop::before {
|
|
|
content: "⏹";
|
|
|
}
|
|
|
|
|
|
.fa-paper-plane::before {
|
|
|
content: "✈";
|
|
|
}
|
|
|
|
|
|
.fa-rocket::before {
|
|
|
content: "🚀";
|
|
|
}
|
|
|
|
|
|
.fa-arrow-up::before {
|
|
|
content: "↑";
|
|
|
}
|
|
|
|
|
|
.fa-arrow-down::before {
|
|
|
content: "↓";
|
|
|
}
|
|
|
|
|
|
.fa-arrow-left::before {
|
|
|
content: "←";
|
|
|
}
|
|
|
|
|
|
.fa-arrow-right::before {
|
|
|
content: "→";
|
|
|
}
|
|
|
|
|
|
.fa-undo::before {
|
|
|
content: "↶";
|
|
|
}
|
|
|
|
|
|
.fa-redo::before {
|
|
|
content: "↷";
|
|
|
}
|
|
|
|
|
|
.fa-video::before {
|
|
|
content: "📹";
|
|
|
}
|
|
|
|
|
|
.fa-camera::before {
|
|
|
content: "📷";
|
|
|
}
|
|
|
|
|
|
.fa-ruler::before {
|
|
|
content: "📏";
|
|
|
}
|
|
|
|
|
|
.fa-sync-alt::before {
|
|
|
content: "🔄";
|
|
|
}
|
|
|
|
|
|
.fa-tachometer-alt::before {
|
|
|
content: "⚡";
|
|
|
}
|
|
|
|
|
|
.fa-home::before {
|
|
|
content: "🏠";
|
|
|
}
|
|
|
|
|
|
.fa-cog::before {
|
|
|
content: "⚙";
|
|
|
}
|
|
|
|
|
|
.fa-info-circle::before {
|
|
|
content: "ℹ";
|
|
|
}
|
|
|
|
|
|
.fa-check-circle::before {
|
|
|
content: "✅";
|
|
|
}
|
|
|
|
|
|
.fa-exclamation-triangle::before {
|
|
|
content: "⚠";
|
|
|
}
|
|
|
|
|
|
.fa-times-circle::before {
|
|
|
content: "❌";
|
|
|
}
|
|
|
|
|
|
.fa-minus-circle::before {
|
|
|
content: "⊖";
|
|
|
}
|
|
|
|
|
|
.fa-circle::before {
|
|
|
content: "●";
|
|
|
}
|
|
|
|
|
|
.text-success {
|
|
|
color: #198754 !important;
|
|
|
}
|
|
|
|
|
|
.text-secondary {
|
|
|
color: #6c757d !important;
|
|
|
}
|
|
|
|
|
|
.text-dark {
|
|
|
color: #212529 !important;
|
|
|
}
|
|
|
|
|
|
.text-white {
|
|
|
color: #fff !important;
|
|
|
}
|
|
|
</style>
|
|
|
</head>
|
|
|
|
|
|
<body class="bg-light">
|
|
|
<div class="container-fluid py-4">
|
|
|
<!-- 状态指示器 -->
|
|
|
<div class="status-badge">
|
|
|
<span id="connectionBadge" class="badge bg-danger fs-6">未连接</span>
|
|
|
</div>
|
|
|
|
|
|
<!-- 页面标题 -->
|
|
|
<div class="row mb-4">
|
|
|
<div class="col-12 text-center">
|
|
|
<h1 class="display-5">🚁 RoboMaster TT 无人机控制系统</h1>
|
|
|
<p class="text-muted fs-5">基于距离判断系统的无人机实时视频传输控制</p>
|
|
|
</div>
|
|
|
</div>
|
|
|
|
|
|
<div class="row">
|
|
|
<!-- 左侧控制面板 -->
|
|
|
<div class="col-lg-6">
|
|
|
<!-- 连接控制 -->
|
|
|
<div class="control-panel">
|
|
|
<h5 class="mb-3"><i class="fa-wifi me-2"></i>连接控制</h5>
|
|
|
<div class="d-flex mb-3 align-items-center">
|
|
|
<button id="connectBtn" class="btn btn-light me-2">
|
|
|
<i class="fa-play me-1"></i>连接无人机
|
|
|
</button>
|
|
|
<button id="disconnectBtn" class="btn btn-danger me-2" disabled>
|
|
|
<i class="fa-stop me-1"></i>断开连接
|
|
|
</button>
|
|
|
<button id="diagnoseBtn" class="btn btn-info me-2">
|
|
|
🔍 诊断连接
|
|
|
</button>
|
|
|
<div class="ms-auto">
|
|
|
<span id="batteryStatus" class="badge bg-warning text-dark fs-6">电量: --</span>
|
|
|
</div>
|
|
|
</div>
|
|
|
<div class="alert alert-info mb-0" id="connectionStatus">
|
|
|
<i class="fa-info-circle me-2"></i>未连接到无人机,请确保已连接到无人机WiFi (TELLO-xxxxxx)
|
|
|
</div>
|
|
|
</div>
|
|
|
|
|
|
<!-- 飞行控制 -->
|
|
|
<div class="control-panel">
|
|
|
<h5 class="mb-3"><i class="fa-paper-plane me-2"></i>飞行控制</h5>
|
|
|
|
|
|
<!-- 起飞降落 -->
|
|
|
<div class="d-flex justify-content-center mb-4">
|
|
|
<button id="takeoffBtn" class="btn btn-success btn-lg me-3" disabled>
|
|
|
<i class="fa-rocket me-1"></i>起飞
|
|
|
</button>
|
|
|
<button id="landBtn" class="btn btn-warning btn-lg" disabled>
|
|
|
<i class="fa-arrow-down me-1"></i>降落
|
|
|
</button>
|
|
|
</div>
|
|
|
|
|
|
<!-- 移动控制网格 -->
|
|
|
<div class="btn-grid mb-4">
|
|
|
<div></div>
|
|
|
<button id="upBtn" class="btn btn-outline-light" disabled>
|
|
|
<i class="fa-arrow-up d-block mb-1"></i>上升
|
|
|
</button>
|
|
|
<div></div>
|
|
|
|
|
|
<button id="leftBtn" class="btn btn-outline-light" disabled>
|
|
|
<i class="fa-arrow-left d-block mb-1"></i>左移
|
|
|
</button>
|
|
|
<button id="forwardBtn" class="btn btn-outline-light" disabled>
|
|
|
<i class="fa-arrow-up d-block mb-1"></i>前进
|
|
|
</button>
|
|
|
<button id="rightBtn" class="btn btn-outline-light" disabled>
|
|
|
<i class="fa-arrow-right d-block mb-1"></i>右移
|
|
|
</button>
|
|
|
|
|
|
<button id="rotateLeftBtn" class="btn btn-outline-warning" disabled>
|
|
|
<i class="fa-undo d-block mb-1"></i>左转
|
|
|
</button>
|
|
|
<button id="backBtn" class="btn btn-outline-light" disabled>
|
|
|
<i class="fa-arrow-down d-block mb-1"></i>后退
|
|
|
</button>
|
|
|
<button id="rotateRightBtn" class="btn btn-outline-warning" disabled>
|
|
|
<i class="fa-redo d-block mb-1"></i>右转
|
|
|
</button>
|
|
|
|
|
|
<div></div>
|
|
|
<button id="downBtn" class="btn btn-outline-light" disabled>
|
|
|
<i class="fa-arrow-down d-block mb-1"></i>下降
|
|
|
</button>
|
|
|
<div></div>
|
|
|
</div>
|
|
|
|
|
|
<!-- 参数控制 -->
|
|
|
<div class="row">
|
|
|
<div class="col-md-6">
|
|
|
<label for="moveDistance" class="form-label">
|
|
|
<i class="fa-ruler me-1"></i>移动距离 (厘米)
|
|
|
</label>
|
|
|
<input type="range" class="form-range" id="moveDistance" min="20" max="100" step="10"
|
|
|
value="30">
|
|
|
<div class="text-center"><span id="distanceValue">30</span> 厘米</div>
|
|
|
</div>
|
|
|
<div class="col-md-6">
|
|
|
<label for="rotateAngle" class="form-label">
|
|
|
<i class="fa-sync-alt me-1"></i>旋转角度 (度)
|
|
|
</label>
|
|
|
<input type="range" class="form-range" id="rotateAngle" min="15" max="360" step="15"
|
|
|
value="90">
|
|
|
<div class="text-center"><span id="angleValue">90</span> 度</div>
|
|
|
</div>
|
|
|
</div>
|
|
|
</div>
|
|
|
</div>
|
|
|
|
|
|
<!-- 右侧视频面板 -->
|
|
|
<div class="col-lg-6">
|
|
|
<!-- 视频流控制 -->
|
|
|
<div class="video-panel mb-3">
|
|
|
<h5 class="mb-3"><i class="fa-video me-2"></i>实时视频流</h5>
|
|
|
|
|
|
<div class="d-flex mb-3">
|
|
|
<button id="startVideoBtn" class="btn btn-light me-2" disabled>
|
|
|
<i class="fa-play me-1"></i>开始视频流
|
|
|
</button>
|
|
|
<button id="stopVideoBtn" class="btn btn-danger me-2" disabled>
|
|
|
<i class="fa-stop me-1"></i>停止视频流
|
|
|
</button>
|
|
|
<button id="captureBtn" class="btn btn-info" disabled>
|
|
|
<i class="fa-camera me-1"></i>捕获图像
|
|
|
</button>
|
|
|
</div>
|
|
|
|
|
|
<div class="video-container mb-3">
|
|
|
<img id="videoStream" class="img-fluid rounded" style="max-width: 100%; max-height: 350px;"
|
|
|
src=""
|
|
|
alt="无视频流">
|
|
|
</div>
|
|
|
|
|
|
<div id="videoStatus" class="text-center">
|
|
|
<small><i class="fa-circle text-secondary me-1"></i>视频流状态: 未启动</small>
|
|
|
</div>
|
|
|
</div>
|
|
|
|
|
|
<!-- 无人机状态 -->
|
|
|
<div class="status-panel">
|
|
|
<h6 class="mb-3"><i class="fa-tachometer-alt me-2"></i>无人机状态</h6>
|
|
|
<div class="row text-center">
|
|
|
<div class="col-3">
|
|
|
<div class="h4 mb-0" id="batteryLevel">--</div>
|
|
|
<small>电量</small>
|
|
|
</div>
|
|
|
<div class="col-3">
|
|
|
<div class="h4 mb-0" id="heightLevel">--</div>
|
|
|
<small>高度</small>
|
|
|
</div>
|
|
|
<div class="col-3">
|
|
|
<div class="h4 mb-0" id="speedLevel">--</div>
|
|
|
<small>速度</small>
|
|
|
</div>
|
|
|
<div class="col-3">
|
|
|
<div class="h4 mb-0" id="signalLevel">--</div>
|
|
|
<small>信号</small>
|
|
|
</div>
|
|
|
</div>
|
|
|
</div>
|
|
|
</div>
|
|
|
</div>
|
|
|
|
|
|
<!-- 快速访问链接 -->
|
|
|
<div class="row mt-4">
|
|
|
<div class="col-12 text-center">
|
|
|
<div class="btn-group" role="group">
|
|
|
<a href="/" class="btn btn-outline-primary">
|
|
|
<i class="fa-home me-1"></i>返回主页
|
|
|
</a>
|
|
|
<a href="/test_device_selector.html" class="btn btn-outline-info">
|
|
|
<i class="fa-cog me-1"></i>设备测试
|
|
|
</a>
|
|
|
</div>
|
|
|
</div>
|
|
|
</div>
|
|
|
</div>
|
|
|
|
|
|
<script>
|
|
|
// 全局变量
|
|
|
let isConnected = false;
|
|
|
let isVideoStreaming = false;
|
|
|
let videoUpdateInterval = null;
|
|
|
|
|
|
// DOM元素
|
|
|
const connectBtn = document.getElementById('connectBtn');
|
|
|
const disconnectBtn = document.getElementById('disconnectBtn');
|
|
|
const diagnoseBtn = document.getElementById('diagnoseBtn');
|
|
|
const connectionStatus = document.getElementById('connectionStatus');
|
|
|
const connectionBadge = document.getElementById('connectionBadge');
|
|
|
const batteryStatus = document.getElementById('batteryStatus');
|
|
|
|
|
|
const takeoffBtn = document.getElementById('takeoffBtn');
|
|
|
const landBtn = document.getElementById('landBtn');
|
|
|
const upBtn = document.getElementById('upBtn');
|
|
|
const downBtn = document.getElementById('downBtn');
|
|
|
const leftBtn = document.getElementById('leftBtn');
|
|
|
const rightBtn = document.getElementById('rightBtn');
|
|
|
const forwardBtn = document.getElementById('forwardBtn');
|
|
|
const backBtn = document.getElementById('backBtn');
|
|
|
const rotateLeftBtn = document.getElementById('rotateLeftBtn');
|
|
|
const rotateRightBtn = document.getElementById('rotateRightBtn');
|
|
|
|
|
|
const startVideoBtn = document.getElementById('startVideoBtn');
|
|
|
const stopVideoBtn = document.getElementById('stopVideoBtn');
|
|
|
const captureBtn = document.getElementById('captureBtn');
|
|
|
const videoStream = document.getElementById('videoStream');
|
|
|
const videoStatus = document.getElementById('videoStatus');
|
|
|
|
|
|
const moveDistance = document.getElementById('moveDistance');
|
|
|
const rotateAngle = document.getElementById('rotateAngle');
|
|
|
const distanceValue = document.getElementById('distanceValue');
|
|
|
const angleValue = document.getElementById('angleValue');
|
|
|
|
|
|
// 初始化
|
|
|
document.addEventListener('DOMContentLoaded', function () {
|
|
|
setupEventListeners();
|
|
|
|
|
|
// 🔧 确保状态变量正确初始化
|
|
|
isConnected = false;
|
|
|
isVideoStreaming = false;
|
|
|
videoUpdateInterval = null;
|
|
|
|
|
|
// 更新界面状态
|
|
|
updateConnectionState(false);
|
|
|
updateStatus('未连接到无人机,请确保已连接到无人机WiFi (TELLO-xxxxxx)', 'info');
|
|
|
|
|
|
console.log('🚁 无人机控制界面已加载');
|
|
|
console.log('📊 初始状态 - 连接:', isConnected, '视频流:', isVideoStreaming);
|
|
|
});
|
|
|
|
|
|
// 设置事件监听器
|
|
|
function setupEventListeners() {
|
|
|
connectBtn.addEventListener('click', connectDrone);
|
|
|
disconnectBtn.addEventListener('click', disconnectDrone);
|
|
|
diagnoseBtn.addEventListener('click', diagnoseDroneConnection);
|
|
|
|
|
|
takeoffBtn.addEventListener('click', () => sendCommand('takeoff'));
|
|
|
landBtn.addEventListener('click', () => sendCommand('land'));
|
|
|
|
|
|
upBtn.addEventListener('click', () => sendMoveCommand('up'));
|
|
|
downBtn.addEventListener('click', () => sendMoveCommand('down'));
|
|
|
leftBtn.addEventListener('click', () => sendMoveCommand('left'));
|
|
|
rightBtn.addEventListener('click', () => sendMoveCommand('right'));
|
|
|
forwardBtn.addEventListener('click', () => sendMoveCommand('forward'));
|
|
|
backBtn.addEventListener('click', () => sendMoveCommand('back'));
|
|
|
|
|
|
rotateLeftBtn.addEventListener('click', () => sendRotateCommand('ccw'));
|
|
|
rotateRightBtn.addEventListener('click', () => sendRotateCommand('cw'));
|
|
|
|
|
|
startVideoBtn.addEventListener('click', startVideo);
|
|
|
stopVideoBtn.addEventListener('click', stopVideo);
|
|
|
captureBtn.addEventListener('click', captureImage);
|
|
|
|
|
|
moveDistance.addEventListener('input', () => {
|
|
|
distanceValue.textContent = moveDistance.value;
|
|
|
});
|
|
|
|
|
|
rotateAngle.addEventListener('input', () => {
|
|
|
angleValue.textContent = rotateAngle.value;
|
|
|
});
|
|
|
}
|
|
|
|
|
|
// 连接无人机
|
|
|
async function connectDrone() {
|
|
|
try {
|
|
|
connectBtn.disabled = true;
|
|
|
updateStatus('正在连接无人机...', 'warning');
|
|
|
|
|
|
const response = await fetch('/api/drone/connect', {
|
|
|
method: 'POST',
|
|
|
headers: {
|
|
|
'Content-Type': 'application/json'
|
|
|
},
|
|
|
body: JSON.stringify({
|
|
|
drone_ip: '192.168.10.1'
|
|
|
})
|
|
|
});
|
|
|
|
|
|
const result = await response.json();
|
|
|
|
|
|
if (result.status === 'success') {
|
|
|
isConnected = true;
|
|
|
updateStatus('无人机连接成功!', 'success');
|
|
|
updateConnectionState(true);
|
|
|
updateDroneStatus(result.drone_info);
|
|
|
showToast('✅ 无人机连接成功!', 'success');
|
|
|
} else {
|
|
|
updateStatus('连接失败: ' + result.message, 'danger');
|
|
|
showToast('❌ 连接失败: ' + result.message, 'error');
|
|
|
}
|
|
|
} catch (error) {
|
|
|
updateStatus('连接错误: ' + error.message, 'danger');
|
|
|
showToast('🔥 连接错误: ' + error.message, 'error');
|
|
|
} finally {
|
|
|
connectBtn.disabled = false;
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// 断开无人机
|
|
|
async function disconnectDrone() {
|
|
|
try {
|
|
|
const response = await fetch('/api/drone/disconnect', {
|
|
|
method: 'POST'
|
|
|
});
|
|
|
|
|
|
const result = await response.json();
|
|
|
|
|
|
// 🔧 强制清理所有状态
|
|
|
isConnected = false;
|
|
|
isVideoStreaming = false;
|
|
|
|
|
|
// 停止视频帧更新
|
|
|
if (videoUpdateInterval) {
|
|
|
clearInterval(videoUpdateInterval);
|
|
|
videoUpdateInterval = null;
|
|
|
console.log('⏹️ 已停止视频帧更新定时器');
|
|
|
}
|
|
|
|
|
|
// 重置视频显示
|
|
|
videoStream.src = "";
|
|
|
videoStatus.innerHTML = '<small><i class="fa-circle text-secondary me-1"></i>视频流状态: 未启动</small>';
|
|
|
|
|
|
// 重置按钮状态
|
|
|
startVideoBtn.disabled = true;
|
|
|
stopVideoBtn.disabled = true;
|
|
|
captureBtn.disabled = true;
|
|
|
|
|
|
updateStatus('无人机已断开连接', 'secondary');
|
|
|
updateConnectionState(false);
|
|
|
|
|
|
showToast('📡 无人机已断开连接', 'info');
|
|
|
console.log('📊 断开后状态 - 连接:', isConnected, '视频流:', isVideoStreaming);
|
|
|
} catch (error) {
|
|
|
updateStatus('断开连接错误: ' + error.message, 'danger');
|
|
|
showToast('❌ 断开连接错误: ' + error.message, 'error');
|
|
|
|
|
|
// 即使出错也要清理状态
|
|
|
isConnected = false;
|
|
|
isVideoStreaming = false;
|
|
|
if (videoUpdateInterval) {
|
|
|
clearInterval(videoUpdateInterval);
|
|
|
videoUpdateInterval = null;
|
|
|
}
|
|
|
updateConnectionState(false);
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// 发送控制命令
|
|
|
async function sendCommand(command, params = {}) {
|
|
|
if (!isConnected) {
|
|
|
showToast('⚠️ 请先连接无人机', 'warning');
|
|
|
return;
|
|
|
}
|
|
|
|
|
|
try {
|
|
|
showToast('📡 正在发送命令: ' + command, 'info');
|
|
|
|
|
|
const response = await fetch('/api/drone/control', {
|
|
|
method: 'POST',
|
|
|
headers: {
|
|
|
'Content-Type': 'application/json'
|
|
|
},
|
|
|
body: JSON.stringify({
|
|
|
command: command,
|
|
|
params: params
|
|
|
})
|
|
|
});
|
|
|
|
|
|
const result = await response.json();
|
|
|
|
|
|
if (result.status === 'success') {
|
|
|
showToast('✅ 命令执行成功: ' + command, 'success');
|
|
|
} else {
|
|
|
showToast('❌ 命令执行失败: ' + result.message, 'error');
|
|
|
}
|
|
|
} catch (error) {
|
|
|
showToast('🔥 命令发送错误: ' + error.message, 'error');
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// 发送移动命令
|
|
|
function sendMoveCommand(direction) {
|
|
|
const distance = parseInt(moveDistance.value);
|
|
|
sendCommand('move', { direction: direction, distance: distance });
|
|
|
}
|
|
|
|
|
|
// 发送旋转命令
|
|
|
function sendRotateCommand(direction) {
|
|
|
const angle = parseInt(rotateAngle.value);
|
|
|
sendCommand('rotate', { direction: direction, angle: angle });
|
|
|
}
|
|
|
|
|
|
// 开始视频流
|
|
|
async function startVideo() {
|
|
|
if (!isConnected) {
|
|
|
showToast('⚠️ 请先连接无人机', 'warning');
|
|
|
return;
|
|
|
}
|
|
|
|
|
|
try {
|
|
|
const response = await fetch('/api/drone/start_video', {
|
|
|
method: 'POST'
|
|
|
});
|
|
|
|
|
|
const result = await response.json();
|
|
|
|
|
|
if (result.status === 'success') {
|
|
|
isVideoStreaming = true;
|
|
|
videoStatus.innerHTML = '<small><i class="fa-circle text-success me-1"></i>视频流状态: 正在接收</small>';
|
|
|
startVideoBtn.disabled = true;
|
|
|
stopVideoBtn.disabled = false;
|
|
|
captureBtn.disabled = false;
|
|
|
|
|
|
// 开始更新视频帧
|
|
|
videoUpdateInterval = setInterval(updateVideoFrame, 200); // 5 FPS
|
|
|
|
|
|
showToast('📹 视频流已启动', 'success');
|
|
|
} else {
|
|
|
showToast('❌ 视频流启动失败: ' + result.message, 'error');
|
|
|
}
|
|
|
} catch (error) {
|
|
|
showToast('🔥 视频流启动错误: ' + error.message, 'error');
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// 停止视频流
|
|
|
async function stopVideo() {
|
|
|
try {
|
|
|
const response = await fetch('/api/drone/stop_video', {
|
|
|
method: 'POST'
|
|
|
});
|
|
|
|
|
|
isVideoStreaming = false;
|
|
|
videoStatus.innerHTML = '<small><i class="fa-circle text-secondary me-1"></i>视频流状态: 已停止</small>';
|
|
|
startVideoBtn.disabled = false;
|
|
|
stopVideoBtn.disabled = true;
|
|
|
captureBtn.disabled = true;
|
|
|
|
|
|
if (videoUpdateInterval) {
|
|
|
clearInterval(videoUpdateInterval);
|
|
|
videoUpdateInterval = null;
|
|
|
}
|
|
|
|
|
|
// 重置视频显示
|
|
|
videoStream.src = "";
|
|
|
|
|
|
showToast('⏹️ 视频流已停止', 'info');
|
|
|
} catch (error) {
|
|
|
showToast('❌ 停止视频流错误: ' + error.message, 'error');
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// 更新视频帧
|
|
|
async function updateVideoFrame() {
|
|
|
// 双重检查确保视频流已启动
|
|
|
if (!isVideoStreaming || !isConnected) {
|
|
|
console.log('⚠️ 视频流未启动或无人机未连接,跳过帧更新');
|
|
|
return;
|
|
|
}
|
|
|
|
|
|
try {
|
|
|
const response = await fetch('/api/drone/video_frame');
|
|
|
|
|
|
// 检查HTTP响应状态
|
|
|
if (!response.ok) {
|
|
|
if (response.status === 404) {
|
|
|
// 没有视频帧可用,这是正常的,不显示错误
|
|
|
console.log('📹 等待视频帧...');
|
|
|
return;
|
|
|
}
|
|
|
throw new Error(`HTTP ${response.status}: ${response.statusText}`);
|
|
|
}
|
|
|
|
|
|
const result = await response.json();
|
|
|
|
|
|
if (result.status === 'success') {
|
|
|
videoStream.src = result.frame;
|
|
|
const fps = result.stats?.fps || 0;
|
|
|
videoStatus.innerHTML = `<small><i class="fa-circle text-success me-1"></i>视频流状态: 正在接收 (${fps.toFixed(1)} FPS)</small>`;
|
|
|
} else if (result.status === 'no_frame') {
|
|
|
// 暂时没有帧,这是正常的
|
|
|
console.log('📹 等待视频帧数据...');
|
|
|
} else {
|
|
|
console.warn('视频帧响应异常:', result.message);
|
|
|
}
|
|
|
} catch (error) {
|
|
|
// 只在非预期错误时显示日志
|
|
|
if (!error.message.includes('404')) {
|
|
|
console.log('获取视频帧失败:', error.message);
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// 捕获图像
|
|
|
function captureImage() {
|
|
|
if (!isVideoStreaming) {
|
|
|
showToast('⚠️ 请先启动视频流', 'warning');
|
|
|
return;
|
|
|
}
|
|
|
|
|
|
showToast('📸 图像捕获功能开发中...', 'info');
|
|
|
}
|
|
|
|
|
|
// 🔍 诊断无人机连接
|
|
|
async function diagnoseDroneConnection() {
|
|
|
try {
|
|
|
diagnoseBtn.disabled = true;
|
|
|
diagnoseBtn.innerHTML = '🔄 诊断中...';
|
|
|
|
|
|
showToast('🔍 正在诊断无人机连接状态...', 'info');
|
|
|
|
|
|
const response = await fetch('/api/drone/diagnose');
|
|
|
const result = await response.json();
|
|
|
|
|
|
if (result.status === 'success') {
|
|
|
const diagnosis = result.diagnosis;
|
|
|
|
|
|
// 🔧 显示诊断结果
|
|
|
let diagnosticHtml = '<div class="diagnostic-results">';
|
|
|
diagnosticHtml += '<h6>🔍 诊断结果:</h6>';
|
|
|
|
|
|
// 显示建议
|
|
|
diagnosis.recommendations.forEach(rec => {
|
|
|
diagnosticHtml += `<div class="mb-1">${rec}</div>`;
|
|
|
});
|
|
|
|
|
|
// 显示详细信息
|
|
|
diagnosticHtml += '<hr>';
|
|
|
diagnosticHtml += '<h6>📊 详细信息:</h6>';
|
|
|
|
|
|
if (diagnosis.network.ping_success !== undefined) {
|
|
|
diagnosticHtml += `<div>网络连通性: ${diagnosis.network.ping_success ? '✅ 正常' : '❌ 失败'}</div>`;
|
|
|
}
|
|
|
|
|
|
if (diagnosis.network.port_11111_available !== undefined) {
|
|
|
diagnosticHtml += `<div>UDP端口11111: ${diagnosis.network.port_11111_available ? '✅ 可用' : '❌ 被占用'}</div>`;
|
|
|
}
|
|
|
|
|
|
if (diagnosis.tello.connected !== undefined) {
|
|
|
diagnosticHtml += `<div>Tello连接: ${diagnosis.tello.connected ? '✅ 已连接' : '❌ 未连接'}</div>`;
|
|
|
if (diagnosis.tello.battery) {
|
|
|
diagnosticHtml += `<div>电池电量: ${diagnosis.tello.battery}%</div>`;
|
|
|
}
|
|
|
}
|
|
|
|
|
|
if (diagnosis.video_stream.receiver_exists !== undefined) {
|
|
|
diagnosticHtml += `<div>视频接收器: ${diagnosis.video_stream.receiver_exists ? '✅ 已创建' : '❌ 未创建'}</div>`;
|
|
|
if (diagnosis.video_stream.running !== undefined) {
|
|
|
diagnosticHtml += `<div>视频流状态: ${diagnosis.video_stream.running ? '🟢 运行中' : '⚪ 已停止'}</div>`;
|
|
|
}
|
|
|
if (diagnosis.video_stream.has_frames !== undefined) {
|
|
|
diagnosticHtml += `<div>视频帧接收: ${diagnosis.video_stream.has_frames ? '✅ 正常' : '⚠️ 无数据'}</div>`;
|
|
|
}
|
|
|
}
|
|
|
|
|
|
diagnosticHtml += `<div>系统平台: ${diagnosis.system.platform}</div>`;
|
|
|
diagnosticHtml += `<div>OpenCV版本: ${diagnosis.system.opencv_version}</div>`;
|
|
|
|
|
|
diagnosticHtml += '</div>';
|
|
|
|
|
|
// 创建诊断结果弹窗
|
|
|
const alertDiv = document.createElement('div');
|
|
|
alertDiv.className = 'alert alert-info alert-dismissible fade show position-fixed';
|
|
|
alertDiv.style.cssText = 'top: 100px; right: 20px; z-index: 1060; max-width: 450px; max-height: 400px; overflow-y: auto; box-shadow: 0 4px 6px rgba(0,0,0,0.1);';
|
|
|
alertDiv.innerHTML = `
|
|
|
${diagnosticHtml}
|
|
|
<button type="button" class="btn-close" onclick="this.parentElement.remove()"></button>
|
|
|
`;
|
|
|
|
|
|
document.body.appendChild(alertDiv);
|
|
|
|
|
|
// 10秒后自动删除
|
|
|
setTimeout(() => {
|
|
|
if (alertDiv.parentNode) {
|
|
|
alertDiv.parentNode.removeChild(alertDiv);
|
|
|
}
|
|
|
}, 10000);
|
|
|
|
|
|
showToast('✅ 诊断完成,请查看右侧详细信息', 'success');
|
|
|
} else {
|
|
|
showToast('❌ 诊断失败: ' + result.message, 'error');
|
|
|
}
|
|
|
} catch (error) {
|
|
|
showToast('🔥 诊断过程出错: ' + error.message, 'error');
|
|
|
} finally {
|
|
|
diagnoseBtn.disabled = false;
|
|
|
diagnoseBtn.innerHTML = '🔍 诊断连接';
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// 更新连接状态
|
|
|
function updateConnectionState(connected) {
|
|
|
isConnected = connected;
|
|
|
|
|
|
connectBtn.disabled = connected;
|
|
|
disconnectBtn.disabled = !connected;
|
|
|
|
|
|
// 更新所有控制按钮
|
|
|
const controlButtons = [
|
|
|
takeoffBtn, landBtn, upBtn, downBtn, leftBtn, rightBtn,
|
|
|
forwardBtn, backBtn, rotateLeftBtn, rotateRightBtn, startVideoBtn
|
|
|
];
|
|
|
|
|
|
controlButtons.forEach(btn => {
|
|
|
btn.disabled = !connected;
|
|
|
});
|
|
|
|
|
|
connectionBadge.textContent = connected ? '已连接' : '未连接';
|
|
|
connectionBadge.className = connected ? 'badge bg-success fs-6' : 'badge bg-danger fs-6';
|
|
|
}
|
|
|
|
|
|
// 更新状态显示
|
|
|
function updateStatus(message, type = 'info') {
|
|
|
const iconMap = {
|
|
|
'info': 'fa-info-circle',
|
|
|
'success': 'fa-check-circle',
|
|
|
'warning': 'fa-exclamation-triangle',
|
|
|
'danger': 'fa-times-circle',
|
|
|
'secondary': 'fa-minus-circle'
|
|
|
};
|
|
|
|
|
|
connectionStatus.innerHTML = `<i class="${iconMap[type] || iconMap.info} me-2"></i>${message}`;
|
|
|
connectionStatus.className = `alert alert-${type} mb-0`;
|
|
|
}
|
|
|
|
|
|
// 更新无人机状态
|
|
|
function updateDroneStatus(droneInfo) {
|
|
|
if (droneInfo) {
|
|
|
const battery = droneInfo.battery || 0;
|
|
|
const height = droneInfo.height || 0;
|
|
|
const speed = droneInfo.speed || 0;
|
|
|
const signal = droneInfo.signal_strength || 0;
|
|
|
|
|
|
batteryStatus.textContent = `电量: ${battery}%`;
|
|
|
batteryStatus.className = battery > 30 ? 'badge bg-success text-white fs-6' :
|
|
|
battery > 15 ? 'badge bg-warning text-dark fs-6' :
|
|
|
'badge bg-danger text-white fs-6';
|
|
|
|
|
|
document.getElementById('batteryLevel').textContent = `${battery}%`;
|
|
|
document.getElementById('heightLevel').textContent = `${height}cm`;
|
|
|
document.getElementById('speedLevel').textContent = `${speed}km/h`;
|
|
|
document.getElementById('signalLevel').textContent = `${signal}%`;
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// 显示提示信息
|
|
|
function showToast(message, type = 'info') {
|
|
|
const typeMap = {
|
|
|
'success': 'success',
|
|
|
'error': 'danger',
|
|
|
'warning': 'warning',
|
|
|
'info': 'info'
|
|
|
};
|
|
|
|
|
|
const alertDiv = document.createElement('div');
|
|
|
alertDiv.className = `alert alert-${typeMap[type] || 'info'} alert-dismissible fade show position-fixed`;
|
|
|
alertDiv.style.cssText = 'top: 80px; right: 20px; z-index: 1050; max-width: 350px; box-shadow: 0 4px 6px rgba(0,0,0,0.1);';
|
|
|
alertDiv.innerHTML = `
|
|
|
${message}
|
|
|
<button type="button" class="btn-close" data-bs-dismiss="alert"></button>
|
|
|
`;
|
|
|
|
|
|
document.body.appendChild(alertDiv);
|
|
|
|
|
|
// 3秒后自动删除
|
|
|
setTimeout(() => {
|
|
|
if (alertDiv.parentNode) {
|
|
|
alertDiv.parentNode.removeChild(alertDiv);
|
|
|
}
|
|
|
}, 3000);
|
|
|
}
|
|
|
|
|
|
// 定期更新无人机状态
|
|
|
setInterval(async () => {
|
|
|
if (isConnected) {
|
|
|
try {
|
|
|
const response = await fetch('/api/drone/status');
|
|
|
const result = await response.json();
|
|
|
if (result.status === 'success') {
|
|
|
updateDroneStatus(result.drone_state);
|
|
|
}
|
|
|
} catch (error) {
|
|
|
console.log('获取无人机状态失败:', error);
|
|
|
}
|
|
|
}
|
|
|
}, 2000); // 每2秒更新一次状态
|
|
|
|
|
|
// 简单的Bootstrap替代功能
|
|
|
document.addEventListener('click', function (e) {
|
|
|
// 处理alert关闭按钮
|
|
|
if (e.target.classList.contains('btn-close')) {
|
|
|
const alert = e.target.closest('.alert');
|
|
|
if (alert) {
|
|
|
alert.remove();
|
|
|
}
|
|
|
}
|
|
|
});
|
|
|
</script>
|
|
|
</body>
|
|
|
|
|
|
</html> |