仿生人AI服务端测试网页
This commit is contained in:
File diff suppressed because it is too large
Load Diff
@@ -0,0 +1,503 @@
|
|||||||
|
<!DOCTYPE html>
|
||||||
|
<html lang="en">
|
||||||
|
<head>
|
||||||
|
<meta charset="UTF-8">
|
||||||
|
<title>小智语音服务测试</title>
|
||||||
|
<style>
|
||||||
|
body {
|
||||||
|
font-family: Arial, sans-serif;
|
||||||
|
max-width: 800px;
|
||||||
|
margin: 0 auto;
|
||||||
|
padding: 20px;
|
||||||
|
}
|
||||||
|
.container {
|
||||||
|
background-color: #f5f5f5;
|
||||||
|
padding: 20px;
|
||||||
|
border-radius: 8px;
|
||||||
|
box-shadow: 0 2px 10px rgba(0, 0, 0, 0.1);
|
||||||
|
margin-bottom: 20px;
|
||||||
|
}
|
||||||
|
button {
|
||||||
|
padding: 8px 15px;
|
||||||
|
margin-right: 10px;
|
||||||
|
border: none;
|
||||||
|
border-radius: 5px;
|
||||||
|
background-color: #4285f4;
|
||||||
|
color: white;
|
||||||
|
cursor: pointer;
|
||||||
|
transition: background-color 0.2s;
|
||||||
|
}
|
||||||
|
button:hover {
|
||||||
|
background-color: #3367d6;
|
||||||
|
}
|
||||||
|
button:disabled {
|
||||||
|
background-color: #cccccc;
|
||||||
|
cursor: not-allowed;
|
||||||
|
}
|
||||||
|
#status {
|
||||||
|
font-weight: bold;
|
||||||
|
}
|
||||||
|
#scriptStatus {
|
||||||
|
position: fixed;
|
||||||
|
top: 10px;
|
||||||
|
left: 50%;
|
||||||
|
transform: translateX(-50%);
|
||||||
|
padding: 15px;
|
||||||
|
border-radius: 5px;
|
||||||
|
display: block;
|
||||||
|
z-index: 100;
|
||||||
|
transition: all 0.3s ease;
|
||||||
|
}
|
||||||
|
#scriptStatus.info {
|
||||||
|
background-color: #e8f0fe;
|
||||||
|
color: #4285f4;
|
||||||
|
border-left: 4px solid #4285f4;
|
||||||
|
}
|
||||||
|
#scriptStatus.success {
|
||||||
|
background-color: #e6f4ea;
|
||||||
|
color: #0f9d58;
|
||||||
|
border-left: 4px solid #0f9d58;
|
||||||
|
}
|
||||||
|
#scriptStatus.error {
|
||||||
|
background-color: #fce8e6;
|
||||||
|
color: #db4437;
|
||||||
|
border-left: 4px solid #db4437;
|
||||||
|
}
|
||||||
|
#debugInfo {
|
||||||
|
margin-top: 20px;
|
||||||
|
border: 1px solid #ccc;
|
||||||
|
border-radius: 5px;
|
||||||
|
padding: 10px;
|
||||||
|
font-family: monospace;
|
||||||
|
font-size: 12px;
|
||||||
|
max-height: 200px;
|
||||||
|
overflow-y: auto;
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
#showDebug {
|
||||||
|
margin-top: 10px;
|
||||||
|
background-color: #f5f5f5;
|
||||||
|
color: #444;
|
||||||
|
font-size: 12px;
|
||||||
|
}
|
||||||
|
#audioMeter {
|
||||||
|
margin-top: 10px;
|
||||||
|
height: 20px;
|
||||||
|
background-color: #eee;
|
||||||
|
border-radius: 4px;
|
||||||
|
overflow: hidden;
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
#audioLevel {
|
||||||
|
height: 100%;
|
||||||
|
width: 0%;
|
||||||
|
background-color: #4285f4;
|
||||||
|
transition: width 0.1s;
|
||||||
|
}
|
||||||
|
.conversation {
|
||||||
|
max-height: 300px;
|
||||||
|
overflow-y: auto;
|
||||||
|
border: 1px solid #ddd;
|
||||||
|
border-radius: 5px;
|
||||||
|
padding: 10px;
|
||||||
|
background-color: white;
|
||||||
|
margin-top: 10px;
|
||||||
|
}
|
||||||
|
.message {
|
||||||
|
margin-bottom: 10px;
|
||||||
|
padding: 8px 12px;
|
||||||
|
border-radius: 8px;
|
||||||
|
max-width: 80%;
|
||||||
|
}
|
||||||
|
.user {
|
||||||
|
background-color: #e2f2ff;
|
||||||
|
margin-left: auto;
|
||||||
|
margin-right: 10px;
|
||||||
|
text-align: right;
|
||||||
|
}
|
||||||
|
.server {
|
||||||
|
background-color: #f0f0f0;
|
||||||
|
margin-right: auto;
|
||||||
|
margin-left: 10px;
|
||||||
|
}
|
||||||
|
#serverUrl {
|
||||||
|
flex-grow: 1;
|
||||||
|
padding: 8px;
|
||||||
|
border: 1px solid #ddd;
|
||||||
|
border-radius: 5px;
|
||||||
|
width: 60%;
|
||||||
|
margin-right: 10px;
|
||||||
|
}
|
||||||
|
#messageInput {
|
||||||
|
flex-grow: 1;
|
||||||
|
padding: 8px;
|
||||||
|
border: 1px solid #ddd;
|
||||||
|
border-radius: 5px;
|
||||||
|
width: 70%;
|
||||||
|
margin-right: 10px;
|
||||||
|
}
|
||||||
|
.section {
|
||||||
|
margin-bottom: 15px;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<div class="container">
|
||||||
|
<h2>小智语音服务测试</h2>
|
||||||
|
|
||||||
|
<div id="scriptStatus" class="info">正在加载Opus库...</div>
|
||||||
|
|
||||||
|
<div class="section">
|
||||||
|
<h3>WebSocket连接 <span id="connectionStatus">未连接</span></h3>
|
||||||
|
<div style="display: flex; align-items: center; margin-bottom: 10px;">
|
||||||
|
<input type="text" id="serverUrl" value="ws://127.0.0.1:8000/xiaozhi/v1/" placeholder="WebSocket服务器地址">
|
||||||
|
<button id="connectButton">连接</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="section">
|
||||||
|
<h3>录音测试</h3>
|
||||||
|
<button id="initAudio" style="background-color: #34a853;">初始化音频</button>
|
||||||
|
<button id="testMic" style="background-color: #fbbc05;">测试麦克风</button>
|
||||||
|
<p></p>
|
||||||
|
<button id="start" disabled>开始录音</button>
|
||||||
|
<button id="stop" disabled>停止录音</button>
|
||||||
|
<button id="play" disabled>播放录音</button>
|
||||||
|
<p>录音状态: <span id="status">待机,正在初始化...</span></p>
|
||||||
|
<div id="audioMeter">
|
||||||
|
<div id="audioLevel"></div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="section">
|
||||||
|
<h3>文本消息</h3>
|
||||||
|
<div style="display: flex; align-items: center;">
|
||||||
|
<input type="text" id="messageInput" placeholder="输入消息..." disabled>
|
||||||
|
<button id="sendTextButton" disabled>发送</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="section">
|
||||||
|
<h3>会话记录</h3>
|
||||||
|
<div id="conversation" class="conversation"></div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<button id="showDebug">显示/隐藏调试信息</button>
|
||||||
|
<div id="debugInfo"></div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
// 定义全局变量以跟踪库加载状态
|
||||||
|
window.opusLoaded = false;
|
||||||
|
window.startButton = document.getElementById("start");
|
||||||
|
window.stopButton = document.getElementById("stop");
|
||||||
|
window.playButton = document.getElementById("play");
|
||||||
|
window.statusLabel = document.getElementById("status");
|
||||||
|
window.debugInfo = document.getElementById("debugInfo");
|
||||||
|
window.audioContextReady = false;
|
||||||
|
window.testMicActive = false;
|
||||||
|
|
||||||
|
// 显示/隐藏调试信息
|
||||||
|
document.getElementById("showDebug").addEventListener("click", function() {
|
||||||
|
if (debugInfo.style.display === "none" || !debugInfo.style.display) {
|
||||||
|
debugInfo.style.display = "block";
|
||||||
|
this.textContent = "隐藏调试信息";
|
||||||
|
} else {
|
||||||
|
debugInfo.style.display = "none";
|
||||||
|
this.textContent = "显示调试信息";
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// 添加初始化音频按钮事件
|
||||||
|
document.getElementById("initAudio").addEventListener("click", function() {
|
||||||
|
initializeAudioSystem();
|
||||||
|
});
|
||||||
|
|
||||||
|
// 添加测试麦克风按钮事件
|
||||||
|
document.getElementById("testMic").addEventListener("click", function() {
|
||||||
|
if (window.testMicActive) {
|
||||||
|
stopMicTest();
|
||||||
|
} else {
|
||||||
|
startMicTest();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// 初始化音频系统
|
||||||
|
function initializeAudioSystem() {
|
||||||
|
try {
|
||||||
|
log("初始化音频系统...");
|
||||||
|
// 创建临时AudioContext来触发用户授权
|
||||||
|
const tempContext = new (window.AudioContext || window.webkitAudioContext)({
|
||||||
|
sampleRate: 16000,
|
||||||
|
latencyHint: 'interactive'
|
||||||
|
});
|
||||||
|
|
||||||
|
// 创建振荡器并播放短促的声音
|
||||||
|
const oscillator = tempContext.createOscillator();
|
||||||
|
const gain = tempContext.createGain();
|
||||||
|
gain.gain.value = 0.1; // 很小的音量
|
||||||
|
oscillator.connect(gain);
|
||||||
|
gain.connect(tempContext.destination);
|
||||||
|
oscillator.frequency.value = 440; // A4
|
||||||
|
oscillator.start();
|
||||||
|
|
||||||
|
// 0.2秒后停止
|
||||||
|
setTimeout(() => {
|
||||||
|
oscillator.stop();
|
||||||
|
// 关闭上下文
|
||||||
|
tempContext.close().then(() => {
|
||||||
|
log("音频系统初始化成功", "success");
|
||||||
|
updateScriptStatus("音频系统已激活", "success");
|
||||||
|
document.getElementById("initAudio").disabled = true;
|
||||||
|
document.getElementById("initAudio").textContent = "音频已初始化";
|
||||||
|
window.audioContextReady = true;
|
||||||
|
|
||||||
|
// 如果Opus已加载,启用开始录音按钮
|
||||||
|
if (window.opusLoaded) {
|
||||||
|
startButton.disabled = false;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}, 200);
|
||||||
|
} catch (err) {
|
||||||
|
log("初始化音频系统失败: " + err.message, "error");
|
||||||
|
updateScriptStatus("初始化音频失败: " + err.message, "error");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 测试麦克风
|
||||||
|
function startMicTest() {
|
||||||
|
const audioMeter = document.getElementById("audioMeter");
|
||||||
|
const audioLevel = document.getElementById("audioLevel");
|
||||||
|
const testMicBtn = document.getElementById("testMic");
|
||||||
|
|
||||||
|
if (!navigator.mediaDevices || !navigator.mediaDevices.getUserMedia) {
|
||||||
|
log("浏览器不支持麦克风访问", "error");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
log("开始麦克风测试...");
|
||||||
|
audioMeter.style.display = "block";
|
||||||
|
testMicBtn.textContent = "停止测试";
|
||||||
|
testMicBtn.style.backgroundColor = "#ea4335";
|
||||||
|
window.testMicActive = true;
|
||||||
|
|
||||||
|
// 创建音频上下文
|
||||||
|
const testContext = new (window.AudioContext || window.webkitAudioContext)();
|
||||||
|
window.testContext = testContext;
|
||||||
|
|
||||||
|
// 获取麦克风权限
|
||||||
|
navigator.mediaDevices.getUserMedia({
|
||||||
|
audio: {
|
||||||
|
echoCancellation: true,
|
||||||
|
noiseSuppression: true,
|
||||||
|
autoGainControl: true
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.then(stream => {
|
||||||
|
log("已获取麦克风访问权限", "success");
|
||||||
|
|
||||||
|
// 保存流以便稍后关闭
|
||||||
|
window.testStream = stream;
|
||||||
|
|
||||||
|
// 创建音频分析器
|
||||||
|
const source = testContext.createMediaStreamSource(stream);
|
||||||
|
const analyser = testContext.createAnalyser();
|
||||||
|
analyser.fftSize = 256;
|
||||||
|
source.connect(analyser);
|
||||||
|
|
||||||
|
// 创建音量显示更新函数
|
||||||
|
const bufferLength = analyser.frequencyBinCount;
|
||||||
|
const dataArray = new Uint8Array(bufferLength);
|
||||||
|
|
||||||
|
function updateMeter() {
|
||||||
|
if (!window.testMicActive) return;
|
||||||
|
|
||||||
|
analyser.getByteFrequencyData(dataArray);
|
||||||
|
|
||||||
|
// 计算音量级别 (0-100)
|
||||||
|
let sum = 0;
|
||||||
|
for (let i = 0; i < bufferLength; i++) {
|
||||||
|
sum += dataArray[i];
|
||||||
|
}
|
||||||
|
const average = sum / bufferLength;
|
||||||
|
const level = Math.min(100, Math.max(0, average * 2));
|
||||||
|
|
||||||
|
// 更新音量计
|
||||||
|
audioLevel.style.width = level + "%";
|
||||||
|
|
||||||
|
// 如果有声音,记录日志
|
||||||
|
if (level > 10) {
|
||||||
|
log(`检测到声音: ${level.toFixed(1)}%`);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 循环更新
|
||||||
|
window.testMicAnimationFrame = requestAnimationFrame(updateMeter);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 开始更新
|
||||||
|
updateMeter();
|
||||||
|
})
|
||||||
|
.catch(err => {
|
||||||
|
log("麦克风测试失败: " + err.message, "error");
|
||||||
|
window.testMicActive = false;
|
||||||
|
testMicBtn.textContent = "测试麦克风";
|
||||||
|
testMicBtn.style.backgroundColor = "#fbbc05";
|
||||||
|
audioMeter.style.display = "none";
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// 停止麦克风测试
|
||||||
|
function stopMicTest() {
|
||||||
|
const audioMeter = document.getElementById("audioMeter");
|
||||||
|
const testMicBtn = document.getElementById("testMic");
|
||||||
|
|
||||||
|
log("停止麦克风测试");
|
||||||
|
window.testMicActive = false;
|
||||||
|
testMicBtn.textContent = "测试麦克风";
|
||||||
|
testMicBtn.style.backgroundColor = "#fbbc05";
|
||||||
|
|
||||||
|
// 停止分析器动画
|
||||||
|
if (window.testMicAnimationFrame) {
|
||||||
|
cancelAnimationFrame(window.testMicAnimationFrame);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 停止麦克风流
|
||||||
|
if (window.testStream) {
|
||||||
|
window.testStream.getTracks().forEach(track => track.stop());
|
||||||
|
}
|
||||||
|
|
||||||
|
// 关闭测试上下文
|
||||||
|
if (window.testContext) {
|
||||||
|
window.testContext.close();
|
||||||
|
}
|
||||||
|
|
||||||
|
// 隐藏音量计
|
||||||
|
audioMeter.style.display = "none";
|
||||||
|
}
|
||||||
|
|
||||||
|
// 添加调试日志
|
||||||
|
function log(message, type = "info") {
|
||||||
|
console.log(message);
|
||||||
|
const time = new Date().toLocaleTimeString();
|
||||||
|
const entry = document.createElement("div");
|
||||||
|
entry.textContent = `[${time}] ${message}`;
|
||||||
|
|
||||||
|
if (type === "error") {
|
||||||
|
entry.style.color = "#db4437";
|
||||||
|
} else if (type === "success") {
|
||||||
|
entry.style.color = "#0f9d58";
|
||||||
|
}
|
||||||
|
|
||||||
|
debugInfo.appendChild(entry);
|
||||||
|
debugInfo.scrollTop = debugInfo.scrollHeight;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 更新脚本状态显示
|
||||||
|
function updateScriptStatus(message, type) {
|
||||||
|
const statusElement = document.getElementById('scriptStatus');
|
||||||
|
if (statusElement) {
|
||||||
|
statusElement.textContent = message;
|
||||||
|
statusElement.className = type;
|
||||||
|
statusElement.style.display = 'block';
|
||||||
|
}
|
||||||
|
log(message, type);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 检查Opus库是否已加载
|
||||||
|
function checkOpusLoaded() {
|
||||||
|
try {
|
||||||
|
// 检查Module是否存在(本地库导出的全局变量)
|
||||||
|
if (typeof Module === 'undefined') {
|
||||||
|
log("Module对象不存在", "error");
|
||||||
|
throw new Error('Opus库未加载,Module对象不存在');
|
||||||
|
}
|
||||||
|
|
||||||
|
// 记录Module对象结构以便调试
|
||||||
|
log("Module对象结构: " + Object.keys(Module).join(", "));
|
||||||
|
|
||||||
|
// 尝试使用全局Module函数
|
||||||
|
if (typeof Module._opus_decoder_get_size === 'function') {
|
||||||
|
window.ModuleInstance = Module;
|
||||||
|
log('Opus库加载成功(使用全局Module)', "success");
|
||||||
|
updateScriptStatus('Opus库加载成功', 'success');
|
||||||
|
|
||||||
|
// 启用开始录音按钮
|
||||||
|
startButton.disabled = false;
|
||||||
|
statusLabel.textContent = "待机";
|
||||||
|
|
||||||
|
// 3秒后隐藏状态
|
||||||
|
setTimeout(() => {
|
||||||
|
const statusElement = document.getElementById('scriptStatus');
|
||||||
|
if (statusElement) statusElement.style.display = 'none';
|
||||||
|
}, 3000);
|
||||||
|
|
||||||
|
window.opusLoaded = true;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 尝试使用Module.instance
|
||||||
|
if (typeof Module.instance !== 'undefined' && typeof Module.instance._opus_decoder_get_size === 'function') {
|
||||||
|
window.ModuleInstance = Module.instance;
|
||||||
|
log('Opus库加载成功(使用Module.instance)', "success");
|
||||||
|
updateScriptStatus('Opus库加载成功', 'success');
|
||||||
|
|
||||||
|
// 启用开始录音按钮
|
||||||
|
startButton.disabled = false;
|
||||||
|
statusLabel.textContent = "待机";
|
||||||
|
|
||||||
|
// 3秒后隐藏状态
|
||||||
|
setTimeout(() => {
|
||||||
|
const statusElement = document.getElementById('scriptStatus');
|
||||||
|
if (statusElement) statusElement.style.display = 'none';
|
||||||
|
}, 3000);
|
||||||
|
|
||||||
|
window.opusLoaded = true;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 检查是否有其他导出方式
|
||||||
|
log("Module上可用方法: " + Object.getOwnPropertyNames(Module).filter(prop => typeof Module[prop] === 'function').join(", "));
|
||||||
|
|
||||||
|
if (typeof Module.onRuntimeInitialized === 'function' || typeof Module.onRuntimeInitialized === 'undefined') {
|
||||||
|
log("Module.onRuntimeInitialized 尚未执行,等待模块初始化完成...");
|
||||||
|
// Module可能未完成初始化,注册回调
|
||||||
|
Module.onRuntimeInitialized = function() {
|
||||||
|
log("Opus库运行时初始化完成,重新检查");
|
||||||
|
checkOpusLoaded();
|
||||||
|
};
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
throw new Error('Opus解码函数未找到,可能Module结构不正确');
|
||||||
|
} catch (err) {
|
||||||
|
log(`Opus库加载失败: ${err.message}`, "error");
|
||||||
|
updateScriptStatus(`Opus库加载失败: ${err.message}`, 'error');
|
||||||
|
statusLabel.textContent = "错误:Opus库加载失败";
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 页面加载完成后检查浏览器能力和Opus库
|
||||||
|
window.addEventListener('load', function() {
|
||||||
|
log("页面加载完成,开始初始化...");
|
||||||
|
updateScriptStatus('正在初始化录音环境...', 'info');
|
||||||
|
|
||||||
|
// 延迟一小段时间再检查,确保库有时间加载
|
||||||
|
setTimeout(function checkAndRetry() {
|
||||||
|
if (!checkOpusLoaded()) {
|
||||||
|
// 如果加载失败,5秒后重试
|
||||||
|
log("Opus库加载失败,5秒后重试...");
|
||||||
|
updateScriptStatus('Opus库加载失败,正在重试...', 'error');
|
||||||
|
setTimeout(checkAndRetry, 5000);
|
||||||
|
}
|
||||||
|
}, 1000);
|
||||||
|
});
|
||||||
|
|
||||||
|
// 防止按钮误操作
|
||||||
|
document.getElementById("stop").disabled = true;
|
||||||
|
document.getElementById("play").disabled = true;
|
||||||
|
</script>
|
||||||
|
<script src="../libopus.js"></script>
|
||||||
|
<script src="app.js"></script>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
@@ -0,0 +1,72 @@
|
|||||||
|
[
|
||||||
|
{
|
||||||
|
"name": "self.get_device_status",
|
||||||
|
"description": "Provides the real-time information of the device, including the current status of the audio speaker, screen, battery, network, etc.\nUse this tool for: \n1. Answering questions about current condition (e.g. what is the current volume of the audio speaker?)\n2. As the first step to control the device (e.g. turn up / down the volume of the audio speaker, etc.)",
|
||||||
|
"inputSchema": {
|
||||||
|
"type": "object",
|
||||||
|
"properties": {}
|
||||||
|
},
|
||||||
|
"mockResponse": {
|
||||||
|
"audio_speaker": {
|
||||||
|
"volume": 50,
|
||||||
|
"muted": false
|
||||||
|
},
|
||||||
|
"screen": {
|
||||||
|
"brightness": 80,
|
||||||
|
"theme": "light"
|
||||||
|
},
|
||||||
|
"battery": {
|
||||||
|
"level": 85,
|
||||||
|
"charging": false
|
||||||
|
},
|
||||||
|
"network": {
|
||||||
|
"connected": true,
|
||||||
|
"type": "wifi"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "self.audio_speaker.set_volume",
|
||||||
|
"description": "Set the volume of the audio speaker. If the current volume is unknown, you must call `self.get_device_status` tool first and then call this tool.",
|
||||||
|
"inputSchema": {
|
||||||
|
"type": "object",
|
||||||
|
"properties": {
|
||||||
|
"volume": {
|
||||||
|
"type": "integer",
|
||||||
|
"minimum": 0,
|
||||||
|
"maximum": 100
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"required": [
|
||||||
|
"volume"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"mockResponse": {
|
||||||
|
"success": true,
|
||||||
|
"volume": "${volume}",
|
||||||
|
"message": "音量已设置为 ${volume}"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "self.screen.set_brightness",
|
||||||
|
"description": "Set the brightness of the screen.",
|
||||||
|
"inputSchema": {
|
||||||
|
"type": "object",
|
||||||
|
"properties": {
|
||||||
|
"brightness": {
|
||||||
|
"type": "integer",
|
||||||
|
"minimum": 0,
|
||||||
|
"maximum": 100
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"required": [
|
||||||
|
"brightness"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"mockResponse": {
|
||||||
|
"success": true,
|
||||||
|
"brightness": "${brightness}",
|
||||||
|
"message": "亮度已设置为 ${brightness}"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
@@ -0,0 +1,149 @@
|
|||||||
|
import BlockingQueue from './utils/BlockingQueue.js';
|
||||||
|
import { log } from './utils/logger.js';
|
||||||
|
|
||||||
|
// 音频流播放上下文类
|
||||||
|
export class StreamingContext {
|
||||||
|
constructor(opusDecoder, audioContext, sampleRate, channels, minAudioDuration) {
|
||||||
|
this.opusDecoder = opusDecoder;
|
||||||
|
this.audioContext = audioContext;
|
||||||
|
|
||||||
|
// 音频参数
|
||||||
|
this.sampleRate = sampleRate;
|
||||||
|
this.channels = channels;
|
||||||
|
this.minAudioDuration = minAudioDuration;
|
||||||
|
|
||||||
|
// 初始化队列和状态
|
||||||
|
this.queue = []; // 已解码的PCM队列。正在播放
|
||||||
|
this.activeQueue = new BlockingQueue(); // 已解码的PCM队列。准备播放
|
||||||
|
this.pendingAudioBufferQueue = []; // 待处理的缓存队列
|
||||||
|
this.audioBufferQueue = new BlockingQueue(); // 缓存队列
|
||||||
|
this.playing = false; // 是否正在播放
|
||||||
|
this.endOfStream = false; // 是否收到结束信号
|
||||||
|
this.source = null; // 当前音频源
|
||||||
|
this.totalSamples = 0; // 累积的总样本数
|
||||||
|
this.lastPlayTime = 0; // 上次播放的时间戳
|
||||||
|
}
|
||||||
|
|
||||||
|
// 缓存音频数组
|
||||||
|
pushAudioBuffer(item) {
|
||||||
|
this.audioBufferQueue.enqueue(...item);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 获取需要处理缓存队列,单线程:在audioBufferQueue一直更新的状态下不会出现安全问题
|
||||||
|
async getPendingAudioBufferQueue() {
|
||||||
|
// 原子交换 + 清空
|
||||||
|
[this.pendingAudioBufferQueue, this.audioBufferQueue] = [await this.audioBufferQueue.dequeue(), new BlockingQueue()];
|
||||||
|
}
|
||||||
|
|
||||||
|
// 获取正在播放已解码的PCM队列,单线程:在activeQueue一直更新的状态下不会出现安全问题
|
||||||
|
async getQueue(minSamples) {
|
||||||
|
let TepArray = [];
|
||||||
|
const num = minSamples - this.queue.length > 0 ? minSamples - this.queue.length : 1;
|
||||||
|
// 原子交换 + 清空
|
||||||
|
[TepArray, this.activeQueue] = [await this.activeQueue.dequeue(num), new BlockingQueue()];
|
||||||
|
this.queue.push(...TepArray);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 将Int16音频数据转换为Float32音频数据
|
||||||
|
convertInt16ToFloat32(int16Data) {
|
||||||
|
const float32Data = new Float32Array(int16Data.length);
|
||||||
|
for (let i = 0; i < int16Data.length; i++) {
|
||||||
|
// 将[-32768,32767]范围转换为[-1,1]
|
||||||
|
float32Data[i] = int16Data[i] / (int16Data[i] < 0 ? 0x8000 : 0x7FFF);
|
||||||
|
}
|
||||||
|
return float32Data;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 将Opus数据解码为PCM
|
||||||
|
async decodeOpusFrames() {
|
||||||
|
if (!this.opusDecoder) {
|
||||||
|
log('Opus解码器未初始化,无法解码', 'error');
|
||||||
|
return;
|
||||||
|
} else {
|
||||||
|
log('Opus解码器启动', 'info');
|
||||||
|
}
|
||||||
|
|
||||||
|
while (true) {
|
||||||
|
let decodedSamples = [];
|
||||||
|
for (const frame of this.pendingAudioBufferQueue) {
|
||||||
|
try {
|
||||||
|
// 使用Opus解码器解码
|
||||||
|
const frameData = this.opusDecoder.decode(frame);
|
||||||
|
if (frameData && frameData.length > 0) {
|
||||||
|
// 转换为Float32
|
||||||
|
const floatData = this.convertInt16ToFloat32(frameData);
|
||||||
|
// 使用循环替代展开运算符
|
||||||
|
for (let i = 0; i < floatData.length; i++) {
|
||||||
|
decodedSamples.push(floatData[i]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
log("Opus解码失败: " + error.message, 'error');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (decodedSamples.length > 0) {
|
||||||
|
// 使用循环替代展开运算符
|
||||||
|
for (let i = 0; i < decodedSamples.length; i++) {
|
||||||
|
this.activeQueue.enqueue(decodedSamples[i]);
|
||||||
|
}
|
||||||
|
this.totalSamples += decodedSamples.length;
|
||||||
|
} else {
|
||||||
|
log('没有成功解码的样本', 'warning');
|
||||||
|
}
|
||||||
|
await this.getPendingAudioBufferQueue();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 开始播放音频
|
||||||
|
async startPlaying() {
|
||||||
|
while (true) {
|
||||||
|
// 如果累积了至少0.3秒的音频,开始播放
|
||||||
|
const minSamples = this.sampleRate * this.minAudioDuration * 3;
|
||||||
|
if (!this.playing && this.queue.length < minSamples) {
|
||||||
|
await this.getQueue(minSamples);
|
||||||
|
}
|
||||||
|
this.playing = true;
|
||||||
|
while (this.playing && this.queue.length) {
|
||||||
|
// 创建新的音频缓冲区
|
||||||
|
const minPlaySamples = Math.min(this.queue.length, this.sampleRate);
|
||||||
|
const currentSamples = this.queue.splice(0, minPlaySamples);
|
||||||
|
|
||||||
|
const audioBuffer = this.audioContext.createBuffer(this.channels, currentSamples.length, this.sampleRate);
|
||||||
|
audioBuffer.copyToChannel(new Float32Array(currentSamples), 0);
|
||||||
|
|
||||||
|
// 创建音频源
|
||||||
|
this.source = this.audioContext.createBufferSource();
|
||||||
|
this.source.buffer = audioBuffer;
|
||||||
|
|
||||||
|
// 创建增益节点用于平滑过渡
|
||||||
|
const gainNode = this.audioContext.createGain();
|
||||||
|
|
||||||
|
// 应用淡入淡出效果避免爆音
|
||||||
|
const fadeDuration = 0.02; // 20毫秒
|
||||||
|
gainNode.gain.setValueAtTime(0, this.audioContext.currentTime);
|
||||||
|
gainNode.gain.linearRampToValueAtTime(1, this.audioContext.currentTime + fadeDuration);
|
||||||
|
|
||||||
|
const duration = audioBuffer.duration;
|
||||||
|
if (duration > fadeDuration * 2) {
|
||||||
|
gainNode.gain.setValueAtTime(1, this.audioContext.currentTime + duration - fadeDuration);
|
||||||
|
gainNode.gain.linearRampToValueAtTime(0, this.audioContext.currentTime + duration);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 连接节点并开始播放
|
||||||
|
this.source.connect(gainNode);
|
||||||
|
gainNode.connect(this.audioContext.destination);
|
||||||
|
|
||||||
|
this.lastPlayTime = this.audioContext.currentTime;
|
||||||
|
log(`开始播放 ${currentSamples.length} 个样本,约 ${(currentSamples.length / this.sampleRate).toFixed(2)} 秒`, 'info');
|
||||||
|
this.source.start();
|
||||||
|
}
|
||||||
|
await this.getQueue(minSamples);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 创建streamingContext实例的工厂函数
|
||||||
|
export function createStreamingContext(opusDecoder, audioContext, sampleRate, channels, minAudioDuration) {
|
||||||
|
return new StreamingContext(opusDecoder, audioContext, sampleRate, channels, minAudioDuration);
|
||||||
|
}
|
||||||
@@ -0,0 +1,49 @@
|
|||||||
|
// DOM元素
|
||||||
|
const connectButton = document.getElementById('connectButton');
|
||||||
|
const serverUrlInput = document.getElementById('serverUrl');
|
||||||
|
const connectionStatus = document.getElementById('connectionStatus');
|
||||||
|
const messageInput = document.getElementById('messageInput');
|
||||||
|
const sendTextButton = document.getElementById('sendTextButton');
|
||||||
|
const recordButton = document.getElementById('recordButton');
|
||||||
|
const stopButton = document.getElementById('stopButton');
|
||||||
|
// 会话记录
|
||||||
|
const conversationDiv = document.getElementById('conversation');
|
||||||
|
const logContainer = document.getElementById('logContainer');
|
||||||
|
let visualizerCanvas = document.getElementById('audioVisualizer');
|
||||||
|
|
||||||
|
// ota 是否连接成功,修改成对应的样式
|
||||||
|
export function otaStatusStyle (flan) {
|
||||||
|
if(flan){
|
||||||
|
document.getElementById('otaStatus').textContent = 'ota已连接';
|
||||||
|
document.getElementById('otaStatus').style.color = 'green';
|
||||||
|
}else{
|
||||||
|
document.getElementById('otaStatus').textContent = 'ota未连接';
|
||||||
|
document.getElementById('otaStatus').style.color = 'red';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// ota 是否连接成功,修改成对应的样式
|
||||||
|
export function getLogContainer (flan) {
|
||||||
|
return logContainer;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 更新Opus库状态显示
|
||||||
|
export function updateScriptStatus(message, type) {
|
||||||
|
const statusElement = document.getElementById('scriptStatus');
|
||||||
|
if (statusElement) {
|
||||||
|
statusElement.textContent = message;
|
||||||
|
statusElement.className = `script-status ${type}`;
|
||||||
|
statusElement.style.display = 'block';
|
||||||
|
statusElement.style.width = 'auto';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 添加消息到会话记录
|
||||||
|
export function addMessage(text, isUser = false) {
|
||||||
|
const messageDiv = document.createElement('div');
|
||||||
|
messageDiv.className = `message ${isUser ? 'user' : 'server'}`;
|
||||||
|
messageDiv.textContent = text;
|
||||||
|
conversationDiv.appendChild(messageDiv);
|
||||||
|
conversationDiv.scrollTop = conversationDiv.scrollHeight;
|
||||||
|
}
|
||||||
|
|
||||||
@@ -0,0 +1,186 @@
|
|||||||
|
import { log } from './utils/logger.js';
|
||||||
|
import { updateScriptStatus } from './document.js'
|
||||||
|
|
||||||
|
|
||||||
|
// 检查Opus库是否已加载
|
||||||
|
export function checkOpusLoaded() {
|
||||||
|
try {
|
||||||
|
// 检查Module是否存在(本地库导出的全局变量)
|
||||||
|
if (typeof Module === 'undefined') {
|
||||||
|
throw new Error('Opus库未加载,Module对象不存在');
|
||||||
|
}
|
||||||
|
|
||||||
|
// 尝试先使用Module.instance(libopus.js最后一行导出方式)
|
||||||
|
if (typeof Module.instance !== 'undefined' && typeof Module.instance._opus_decoder_get_size === 'function') {
|
||||||
|
// 使用Module.instance对象替换全局Module对象
|
||||||
|
window.ModuleInstance = Module.instance;
|
||||||
|
log('Opus库加载成功(使用Module.instance)', 'success');
|
||||||
|
updateScriptStatus('Opus库加载成功', 'success');
|
||||||
|
|
||||||
|
// 3秒后隐藏状态
|
||||||
|
const statusElement = document.getElementById('scriptStatus');
|
||||||
|
if (statusElement) statusElement.style.display = 'none';
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 如果没有Module.instance,检查全局Module函数
|
||||||
|
if (typeof Module._opus_decoder_get_size === 'function') {
|
||||||
|
window.ModuleInstance = Module;
|
||||||
|
log('Opus库加载成功(使用全局Module)', 'success');
|
||||||
|
updateScriptStatus('Opus库加载成功', 'success');
|
||||||
|
|
||||||
|
// 3秒后隐藏状态
|
||||||
|
const statusElement = document.getElementById('scriptStatus');
|
||||||
|
if (statusElement) statusElement.style.display = 'none';
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
throw new Error('Opus解码函数未找到,可能Module结构不正确');
|
||||||
|
} catch (err) {
|
||||||
|
log(`Opus库加载失败,请检查libopus.js文件是否存在且正确: ${err.message}`, 'error');
|
||||||
|
updateScriptStatus('Opus库加载失败,请检查libopus.js文件是否存在且正确', 'error');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// 创建一个Opus编码器
|
||||||
|
let opusEncoder = null;
|
||||||
|
export function initOpusEncoder() {
|
||||||
|
try {
|
||||||
|
if (opusEncoder) {
|
||||||
|
return opusEncoder; // 已经初始化过
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!window.ModuleInstance) {
|
||||||
|
log('无法创建Opus编码器:ModuleInstance不可用', 'error');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 初始化一个Opus编码器
|
||||||
|
const mod = window.ModuleInstance;
|
||||||
|
const sampleRate = 16000; // 16kHz采样率
|
||||||
|
const channels = 1; // 单声道
|
||||||
|
const application = 2048; // OPUS_APPLICATION_VOIP = 2048
|
||||||
|
|
||||||
|
// 创建编码器
|
||||||
|
opusEncoder = {
|
||||||
|
channels: channels,
|
||||||
|
sampleRate: sampleRate,
|
||||||
|
frameSize: 960, // 60ms @ 16kHz = 60 * 16 = 960 samples
|
||||||
|
maxPacketSize: 4000, // 最大包大小
|
||||||
|
module: mod,
|
||||||
|
|
||||||
|
// 初始化编码器
|
||||||
|
init: function () {
|
||||||
|
try {
|
||||||
|
// 获取编码器大小
|
||||||
|
const encoderSize = mod._opus_encoder_get_size(this.channels);
|
||||||
|
log(`Opus编码器大小: ${encoderSize}字节`, 'info');
|
||||||
|
|
||||||
|
// 分配内存
|
||||||
|
this.encoderPtr = mod._malloc(encoderSize);
|
||||||
|
if (!this.encoderPtr) {
|
||||||
|
throw new Error("无法分配编码器内存");
|
||||||
|
}
|
||||||
|
|
||||||
|
// 初始化编码器
|
||||||
|
const err = mod._opus_encoder_init(
|
||||||
|
this.encoderPtr,
|
||||||
|
this.sampleRate,
|
||||||
|
this.channels,
|
||||||
|
application
|
||||||
|
);
|
||||||
|
|
||||||
|
if (err < 0) {
|
||||||
|
throw new Error(`Opus编码器初始化失败: ${err}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 设置位率 (16kbps)
|
||||||
|
mod._opus_encoder_ctl(this.encoderPtr, 4002, 16000); // OPUS_SET_BITRATE
|
||||||
|
|
||||||
|
// 设置复杂度 (0-10, 越高质量越好但CPU使用越多)
|
||||||
|
mod._opus_encoder_ctl(this.encoderPtr, 4010, 5); // OPUS_SET_COMPLEXITY
|
||||||
|
|
||||||
|
// 设置使用DTX (不传输静音帧)
|
||||||
|
mod._opus_encoder_ctl(this.encoderPtr, 4016, 1); // OPUS_SET_DTX
|
||||||
|
|
||||||
|
log("Opus编码器初始化成功", 'success');
|
||||||
|
return true;
|
||||||
|
} catch (error) {
|
||||||
|
if (this.encoderPtr) {
|
||||||
|
mod._free(this.encoderPtr);
|
||||||
|
this.encoderPtr = null;
|
||||||
|
}
|
||||||
|
log(`Opus编码器初始化失败: ${error.message}`, 'error');
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
// 编码PCM数据为Opus
|
||||||
|
encode: function (pcmData) {
|
||||||
|
if (!this.encoderPtr) {
|
||||||
|
if (!this.init()) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
const mod = this.module;
|
||||||
|
|
||||||
|
// 为PCM数据分配内存
|
||||||
|
const pcmPtr = mod._malloc(pcmData.length * 2); // 2字节/int16
|
||||||
|
|
||||||
|
// 将PCM数据复制到HEAP
|
||||||
|
for (let i = 0; i < pcmData.length; i++) {
|
||||||
|
mod.HEAP16[(pcmPtr >> 1) + i] = pcmData[i];
|
||||||
|
}
|
||||||
|
|
||||||
|
// 为输出分配内存
|
||||||
|
const outPtr = mod._malloc(this.maxPacketSize);
|
||||||
|
|
||||||
|
// 进行编码
|
||||||
|
const encodedLen = mod._opus_encode(
|
||||||
|
this.encoderPtr,
|
||||||
|
pcmPtr,
|
||||||
|
this.frameSize,
|
||||||
|
outPtr,
|
||||||
|
this.maxPacketSize
|
||||||
|
);
|
||||||
|
|
||||||
|
if (encodedLen < 0) {
|
||||||
|
throw new Error(`Opus编码失败: ${encodedLen}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 复制编码后的数据
|
||||||
|
const opusData = new Uint8Array(encodedLen);
|
||||||
|
for (let i = 0; i < encodedLen; i++) {
|
||||||
|
opusData[i] = mod.HEAPU8[outPtr + i];
|
||||||
|
}
|
||||||
|
|
||||||
|
// 释放内存
|
||||||
|
mod._free(pcmPtr);
|
||||||
|
mod._free(outPtr);
|
||||||
|
|
||||||
|
return opusData;
|
||||||
|
} catch (error) {
|
||||||
|
log(`Opus编码出错: ${error.message}`, 'error');
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
// 销毁编码器
|
||||||
|
destroy: function () {
|
||||||
|
if (this.encoderPtr) {
|
||||||
|
this.module._free(this.encoderPtr);
|
||||||
|
this.encoderPtr = null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
opusEncoder.init();
|
||||||
|
return opusEncoder;
|
||||||
|
} catch (error) {
|
||||||
|
log(`创建Opus编码器失败: ${error.message}`, 'error');
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,98 @@
|
|||||||
|
export default class BlockingQueue {
|
||||||
|
#items = [];
|
||||||
|
#waiters = []; // {resolve, reject, min, timer, onTimeout}
|
||||||
|
|
||||||
|
/* 空队列一次性闸门 */
|
||||||
|
#emptyPromise = null;
|
||||||
|
#emptyResolve = null;
|
||||||
|
|
||||||
|
/* 生产者:把数据塞进去 */
|
||||||
|
enqueue(item, ...restItems) {
|
||||||
|
if (restItems.length === 0) {
|
||||||
|
this.#items.push(item);
|
||||||
|
}
|
||||||
|
// 如果有额外参数,批量处理所有项
|
||||||
|
else {
|
||||||
|
const items = [item, ...restItems].filter(i => i);
|
||||||
|
if (items.length === 0) return;
|
||||||
|
this.#items.push(...items);
|
||||||
|
}
|
||||||
|
// 若有空队列闸门,一次性放行所有等待者
|
||||||
|
if (this.#emptyResolve) {
|
||||||
|
this.#emptyResolve();
|
||||||
|
this.#emptyResolve = null;
|
||||||
|
this.#emptyPromise = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 唤醒所有正在等的 waiter
|
||||||
|
this.#wakeWaiters();
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 消费者:min 条或 timeout ms 先到谁 */
|
||||||
|
async dequeue(min = 1, timeout = Infinity, onTimeout = null) {
|
||||||
|
// 1. 若空,等第一次数据到达(所有调用共享同一个 promise)
|
||||||
|
if (this.#items.length === 0) {
|
||||||
|
await this.#waitForFirstItem();
|
||||||
|
}
|
||||||
|
|
||||||
|
// 立即满足
|
||||||
|
if (this.#items.length >= min) {
|
||||||
|
return this.#flush();
|
||||||
|
}
|
||||||
|
|
||||||
|
// 需要等待
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
let timer = null;
|
||||||
|
const waiter = { resolve, reject, min, onTimeout, timer };
|
||||||
|
|
||||||
|
// 超时逻辑
|
||||||
|
if (Number.isFinite(timeout)) {
|
||||||
|
waiter.timer = setTimeout(() => {
|
||||||
|
this.#removeWaiter(waiter);
|
||||||
|
if (onTimeout) onTimeout(this.#items.length);
|
||||||
|
resolve(this.#flush());
|
||||||
|
}, timeout);
|
||||||
|
}
|
||||||
|
|
||||||
|
this.#waiters.push(waiter);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 空队列闸门生成器 */
|
||||||
|
#waitForFirstItem() {
|
||||||
|
if (!this.#emptyPromise) {
|
||||||
|
this.#emptyPromise = new Promise(r => (this.#emptyResolve = r));
|
||||||
|
}
|
||||||
|
return this.#emptyPromise;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 内部:每次数据变动后,检查哪些 waiter 已满足 */
|
||||||
|
#wakeWaiters() {
|
||||||
|
for (let i = this.#waiters.length - 1; i >= 0; i--) {
|
||||||
|
const w = this.#waiters[i];
|
||||||
|
if (this.#items.length >= w.min) {
|
||||||
|
this.#removeWaiter(w);
|
||||||
|
w.resolve(this.#flush());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#removeWaiter(waiter) {
|
||||||
|
const idx = this.#waiters.indexOf(waiter);
|
||||||
|
if (idx !== -1) {
|
||||||
|
this.#waiters.splice(idx, 1);
|
||||||
|
if (waiter.timer) clearTimeout(waiter.timer);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#flush() {
|
||||||
|
const snapshot = [...this.#items];
|
||||||
|
this.#items.length = 0;
|
||||||
|
return snapshot;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 当前缓存长度(不含等待者) */
|
||||||
|
get length() {
|
||||||
|
return this.#items.length;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,37 @@
|
|||||||
|
import { getLogContainer } from '../document.js'
|
||||||
|
|
||||||
|
const logContainer = getLogContainer();
|
||||||
|
// 日志记录函数
|
||||||
|
export function log(message, type = 'info') {
|
||||||
|
// 将消息按换行符分割成多行
|
||||||
|
const lines = message.split('\n');
|
||||||
|
const now = new Date();
|
||||||
|
// const timestamp = `[${now.toLocaleTimeString()}] `;
|
||||||
|
const timestamp = `[${now.toLocaleTimeString()}.${now.getMilliseconds().toString().padStart(3, '0')}] `;
|
||||||
|
// 为每一行创建日志条目
|
||||||
|
lines.forEach((line, index) => {
|
||||||
|
const logEntry = document.createElement('div');
|
||||||
|
logEntry.className = `log-entry log-${type}`;
|
||||||
|
// 如果是第一条日志,显示时间戳
|
||||||
|
const prefix = index === 0 ? timestamp : ' '.repeat(timestamp.length);
|
||||||
|
logEntry.textContent = `${prefix}${line}`;
|
||||||
|
// logEntry.textContent = `[${new Date().toLocaleTimeString()}] ${message}`;
|
||||||
|
// logEntry.style 保留起始的空格
|
||||||
|
logEntry.style.whiteSpace = 'pre';
|
||||||
|
if (type === 'error') {
|
||||||
|
logEntry.style.color = 'red';
|
||||||
|
} else if (type === 'debug') {
|
||||||
|
logEntry.style.color = 'gray';
|
||||||
|
return;
|
||||||
|
} else if (type === 'warning') {
|
||||||
|
logEntry.style.color = 'orange';
|
||||||
|
} else if (type === 'success') {
|
||||||
|
logEntry.style.color = 'green';
|
||||||
|
} else {
|
||||||
|
logEntry.style.color = 'black';
|
||||||
|
}
|
||||||
|
logContainer.appendChild(logEntry);
|
||||||
|
});
|
||||||
|
|
||||||
|
logContainer.scrollTop = logContainer.scrollHeight;
|
||||||
|
}
|
||||||
@@ -0,0 +1,124 @@
|
|||||||
|
import { otaStatusStyle } from './document.js';
|
||||||
|
import { log } from './utils/logger.js';
|
||||||
|
|
||||||
|
// WebSocket 连接
|
||||||
|
export async function webSocketConnect(otaUrl, config) {
|
||||||
|
|
||||||
|
if (!validateConfig(config)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 发送OTA请求并获取返回的websocket信息
|
||||||
|
const otaResult = await sendOTA(otaUrl, config);
|
||||||
|
if (!otaResult) {
|
||||||
|
log('无法从OTA服务器获取信息', 'error');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 从OTA响应中提取websocket信息
|
||||||
|
const { websocket } = otaResult;
|
||||||
|
if (!websocket || !websocket.url) {
|
||||||
|
log('OTA响应中缺少websocket信息', 'error');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 使用OTA返回的websocket URL
|
||||||
|
let connUrl = new URL(websocket.url);
|
||||||
|
|
||||||
|
// 添加token参数(从OTA响应中获取)
|
||||||
|
if (websocket.token) {
|
||||||
|
if (websocket.token.startsWith("Bearer ")) {
|
||||||
|
connUrl.searchParams.append('authorization', websocket.token);
|
||||||
|
} else {
|
||||||
|
connUrl.searchParams.append('authorization', 'Bearer ' + websocket.token);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 添加认证参数(保持原有逻辑)
|
||||||
|
connUrl.searchParams.append('device-id', config.deviceId);
|
||||||
|
connUrl.searchParams.append('client-id', config.clientId);
|
||||||
|
|
||||||
|
const wsurl = connUrl.toString()
|
||||||
|
|
||||||
|
log(`正在连接: ${wsurl}`, 'info');
|
||||||
|
|
||||||
|
if (wsurl) {
|
||||||
|
document.getElementById('serverUrl').value = wsurl;
|
||||||
|
}
|
||||||
|
|
||||||
|
return new WebSocket(connUrl.toString());
|
||||||
|
}
|
||||||
|
|
||||||
|
// 验证配置
|
||||||
|
function validateConfig(config) {
|
||||||
|
if (!config.deviceMac) {
|
||||||
|
log('设备MAC地址不能为空', 'error');
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (!config.clientId) {
|
||||||
|
log('客户端ID不能为空', 'error');
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 判断wsUrl路径是否存在错误
|
||||||
|
function validateWsUrl(wsUrl) {
|
||||||
|
if (wsUrl === '') return false;
|
||||||
|
// 检查URL格式
|
||||||
|
if (!wsUrl.startsWith('ws://') && !wsUrl.startsWith('wss://')) {
|
||||||
|
log('URL格式错误,必须以ws://或wss://开头', 'error');
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// OTA发送请求,验证状态,并返回响应数据
|
||||||
|
async function sendOTA(otaUrl, config) {
|
||||||
|
try {
|
||||||
|
const res = await fetch(otaUrl, {
|
||||||
|
method: 'POST',
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
'Device-Id': config.deviceId,
|
||||||
|
'Client-Id': config.clientId
|
||||||
|
},
|
||||||
|
body: JSON.stringify({
|
||||||
|
version: 0,
|
||||||
|
uuid: '',
|
||||||
|
application: {
|
||||||
|
name: 'xiaozhi-web-test',
|
||||||
|
version: '1.0.0',
|
||||||
|
compile_time: '2025-04-16 10:00:00',
|
||||||
|
idf_version: '4.4.3',
|
||||||
|
elf_sha256: '1234567890abcdef1234567890abcdef1234567890abcdef'
|
||||||
|
},
|
||||||
|
ota: { label: 'xiaozhi-web-test' },
|
||||||
|
board: {
|
||||||
|
type: 'xiaozhi-web-test',
|
||||||
|
ssid: 'xiaozhi-web-test',
|
||||||
|
rssi: 0,
|
||||||
|
channel: 0,
|
||||||
|
ip: '192.168.1.1',
|
||||||
|
mac: config.deviceMac
|
||||||
|
},
|
||||||
|
flash_size: 0,
|
||||||
|
minimum_free_heap_size: 0,
|
||||||
|
mac_address: config.deviceMac,
|
||||||
|
chip_model_name: '',
|
||||||
|
chip_info: { model: 0, cores: 0, revision: 0, features: 0 },
|
||||||
|
partition_table: [{ label: '', type: 0, subtype: 0, address: 0, size: 0 }]
|
||||||
|
})
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!res.ok) throw new Error(`${res.status} ${res.statusText}`);
|
||||||
|
|
||||||
|
const result = await res.json();
|
||||||
|
otaStatusStyle(true)
|
||||||
|
return result; // 返回完整的响应数据
|
||||||
|
} catch (err) {
|
||||||
|
otaStatusStyle(false)
|
||||||
|
return null; // 失败返回null
|
||||||
|
}
|
||||||
|
}
|
||||||
File diff suppressed because one or more lines are too long
@@ -0,0 +1,670 @@
|
|||||||
|
const SAMPLE_RATE = 16000;
|
||||||
|
const CHANNELS = 1;
|
||||||
|
const FRAME_SIZE = 960; // 对应于60ms帧大小 (16000Hz * 0.06s = 960 samples)
|
||||||
|
const OPUS_APPLICATION = 2049; // OPUS_APPLICATION_AUDIO
|
||||||
|
const BUFFER_SIZE = 4096;
|
||||||
|
|
||||||
|
let audioContext = new (window.AudioContext || window.webkitAudioContext)({ sampleRate: SAMPLE_RATE });
|
||||||
|
let mediaStream, mediaSource, audioProcessor;
|
||||||
|
let recordedPcmData = []; // 存储原始PCM数据
|
||||||
|
let recordedOpusData = []; // 存储Opus编码后的数据
|
||||||
|
let opusEncoder, opusDecoder;
|
||||||
|
let isRecording = false;
|
||||||
|
|
||||||
|
const startButton = document.getElementById("start");
|
||||||
|
const stopButton = document.getElementById("stop");
|
||||||
|
const playButton = document.getElementById("play");
|
||||||
|
const statusLabel = document.getElementById("status");
|
||||||
|
|
||||||
|
startButton.addEventListener("click", startRecording);
|
||||||
|
stopButton.addEventListener("click", stopRecording);
|
||||||
|
playButton.addEventListener("click", playRecording);
|
||||||
|
|
||||||
|
// 初始化Opus编码器与解码器
|
||||||
|
async function initOpus() {
|
||||||
|
if (typeof window.ModuleInstance === 'undefined') {
|
||||||
|
if (typeof Module !== 'undefined') {
|
||||||
|
// 尝试使用全局Module
|
||||||
|
window.ModuleInstance = Module;
|
||||||
|
console.log('使用全局Module作为ModuleInstance');
|
||||||
|
} else {
|
||||||
|
console.error("Opus库未加载,ModuleInstance和Module对象都不存在");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
const mod = window.ModuleInstance;
|
||||||
|
|
||||||
|
// 创建编码器
|
||||||
|
opusEncoder = {
|
||||||
|
channels: CHANNELS,
|
||||||
|
sampleRate: SAMPLE_RATE,
|
||||||
|
frameSize: FRAME_SIZE,
|
||||||
|
maxPacketSize: 4000,
|
||||||
|
module: mod,
|
||||||
|
|
||||||
|
// 初始化编码器
|
||||||
|
init: function() {
|
||||||
|
// 获取编码器大小
|
||||||
|
const encoderSize = mod._opus_encoder_get_size(this.channels);
|
||||||
|
console.log(`Opus编码器大小: ${encoderSize}字节`);
|
||||||
|
|
||||||
|
// 分配内存
|
||||||
|
this.encoderPtr = mod._malloc(encoderSize);
|
||||||
|
if (!this.encoderPtr) {
|
||||||
|
throw new Error("无法分配编码器内存");
|
||||||
|
}
|
||||||
|
|
||||||
|
// 初始化编码器
|
||||||
|
const err = mod._opus_encoder_init(
|
||||||
|
this.encoderPtr,
|
||||||
|
this.sampleRate,
|
||||||
|
this.channels,
|
||||||
|
OPUS_APPLICATION
|
||||||
|
);
|
||||||
|
|
||||||
|
if (err < 0) {
|
||||||
|
throw new Error(`Opus编码器初始化失败: ${err}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
},
|
||||||
|
|
||||||
|
// 编码方法
|
||||||
|
encode: function(pcmData) {
|
||||||
|
const mod = this.module;
|
||||||
|
|
||||||
|
// 为PCM数据分配内存
|
||||||
|
const pcmPtr = mod._malloc(pcmData.length * 2); // Int16 = 2字节
|
||||||
|
|
||||||
|
// 将数据复制到WASM内存
|
||||||
|
for (let i = 0; i < pcmData.length; i++) {
|
||||||
|
mod.HEAP16[(pcmPtr >> 1) + i] = pcmData[i];
|
||||||
|
}
|
||||||
|
|
||||||
|
// 为Opus编码数据分配内存
|
||||||
|
const maxEncodedSize = this.maxPacketSize;
|
||||||
|
const encodedPtr = mod._malloc(maxEncodedSize);
|
||||||
|
|
||||||
|
// 编码
|
||||||
|
const encodedBytes = mod._opus_encode(
|
||||||
|
this.encoderPtr,
|
||||||
|
pcmPtr,
|
||||||
|
this.frameSize,
|
||||||
|
encodedPtr,
|
||||||
|
maxEncodedSize
|
||||||
|
);
|
||||||
|
|
||||||
|
if (encodedBytes < 0) {
|
||||||
|
mod._free(pcmPtr);
|
||||||
|
mod._free(encodedPtr);
|
||||||
|
throw new Error(`Opus编码失败: ${encodedBytes}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 复制编码后的数据
|
||||||
|
const encodedData = new Uint8Array(encodedBytes);
|
||||||
|
for (let i = 0; i < encodedBytes; i++) {
|
||||||
|
encodedData[i] = mod.HEAPU8[encodedPtr + i];
|
||||||
|
}
|
||||||
|
|
||||||
|
// 释放内存
|
||||||
|
mod._free(pcmPtr);
|
||||||
|
mod._free(encodedPtr);
|
||||||
|
|
||||||
|
return encodedData;
|
||||||
|
},
|
||||||
|
|
||||||
|
// 销毁方法
|
||||||
|
destroy: function() {
|
||||||
|
if (this.encoderPtr) {
|
||||||
|
this.module._free(this.encoderPtr);
|
||||||
|
this.encoderPtr = null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// 创建解码器
|
||||||
|
opusDecoder = {
|
||||||
|
channels: CHANNELS,
|
||||||
|
rate: SAMPLE_RATE,
|
||||||
|
frameSize: FRAME_SIZE,
|
||||||
|
module: mod,
|
||||||
|
|
||||||
|
// 初始化解码器
|
||||||
|
init: function() {
|
||||||
|
// 获取解码器大小
|
||||||
|
const decoderSize = mod._opus_decoder_get_size(this.channels);
|
||||||
|
console.log(`Opus解码器大小: ${decoderSize}字节`);
|
||||||
|
|
||||||
|
// 分配内存
|
||||||
|
this.decoderPtr = mod._malloc(decoderSize);
|
||||||
|
if (!this.decoderPtr) {
|
||||||
|
throw new Error("无法分配解码器内存");
|
||||||
|
}
|
||||||
|
|
||||||
|
// 初始化解码器
|
||||||
|
const err = mod._opus_decoder_init(
|
||||||
|
this.decoderPtr,
|
||||||
|
this.rate,
|
||||||
|
this.channels
|
||||||
|
);
|
||||||
|
|
||||||
|
if (err < 0) {
|
||||||
|
throw new Error(`Opus解码器初始化失败: ${err}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
},
|
||||||
|
|
||||||
|
// 解码方法
|
||||||
|
decode: function(opusData) {
|
||||||
|
const mod = this.module;
|
||||||
|
|
||||||
|
// 为Opus数据分配内存
|
||||||
|
const opusPtr = mod._malloc(opusData.length);
|
||||||
|
mod.HEAPU8.set(opusData, opusPtr);
|
||||||
|
|
||||||
|
// 为PCM输出分配内存
|
||||||
|
const pcmPtr = mod._malloc(this.frameSize * 2); // Int16 = 2字节
|
||||||
|
|
||||||
|
// 解码
|
||||||
|
const decodedSamples = mod._opus_decode(
|
||||||
|
this.decoderPtr,
|
||||||
|
opusPtr,
|
||||||
|
opusData.length,
|
||||||
|
pcmPtr,
|
||||||
|
this.frameSize,
|
||||||
|
0 // 不使用FEC
|
||||||
|
);
|
||||||
|
|
||||||
|
if (decodedSamples < 0) {
|
||||||
|
mod._free(opusPtr);
|
||||||
|
mod._free(pcmPtr);
|
||||||
|
throw new Error(`Opus解码失败: ${decodedSamples}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 复制解码后的数据
|
||||||
|
const decodedData = new Int16Array(decodedSamples);
|
||||||
|
for (let i = 0; i < decodedSamples; i++) {
|
||||||
|
decodedData[i] = mod.HEAP16[(pcmPtr >> 1) + i];
|
||||||
|
}
|
||||||
|
|
||||||
|
// 释放内存
|
||||||
|
mod._free(opusPtr);
|
||||||
|
mod._free(pcmPtr);
|
||||||
|
|
||||||
|
return decodedData;
|
||||||
|
},
|
||||||
|
|
||||||
|
// 销毁方法
|
||||||
|
destroy: function() {
|
||||||
|
if (this.decoderPtr) {
|
||||||
|
this.module._free(this.decoderPtr);
|
||||||
|
this.decoderPtr = null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// 初始化编码器和解码器
|
||||||
|
if (opusEncoder.init() && opusDecoder.init()) {
|
||||||
|
console.log("Opus 编码器和解码器初始化成功。");
|
||||||
|
return true;
|
||||||
|
} else {
|
||||||
|
console.error("Opus 初始化失败");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Opus 初始化失败:", error);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 将Float32音频数据转换为Int16音频数据
|
||||||
|
function convertFloat32ToInt16(float32Data) {
|
||||||
|
const int16Data = new Int16Array(float32Data.length);
|
||||||
|
for (let i = 0; i < float32Data.length; i++) {
|
||||||
|
// 将[-1,1]范围转换为[-32768,32767]
|
||||||
|
const s = Math.max(-1, Math.min(1, float32Data[i]));
|
||||||
|
int16Data[i] = s < 0 ? s * 0x8000 : s * 0x7FFF;
|
||||||
|
}
|
||||||
|
return int16Data;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 将Int16音频数据转换为Float32音频数据
|
||||||
|
function convertInt16ToFloat32(int16Data) {
|
||||||
|
const float32Data = new Float32Array(int16Data.length);
|
||||||
|
for (let i = 0; i < int16Data.length; i++) {
|
||||||
|
// 将[-32768,32767]范围转换为[-1,1]
|
||||||
|
float32Data[i] = int16Data[i] / (int16Data[i] < 0 ? 0x8000 : 0x7FFF);
|
||||||
|
}
|
||||||
|
return float32Data;
|
||||||
|
}
|
||||||
|
|
||||||
|
function startRecording() {
|
||||||
|
if (isRecording) return;
|
||||||
|
|
||||||
|
// 确保有权限并且AudioContext是活跃的
|
||||||
|
if (audioContext.state === 'suspended') {
|
||||||
|
audioContext.resume().then(() => {
|
||||||
|
console.log("AudioContext已恢复");
|
||||||
|
continueStartRecording();
|
||||||
|
}).catch(err => {
|
||||||
|
console.error("恢复AudioContext失败:", err);
|
||||||
|
statusLabel.textContent = "无法激活音频上下文,请再次点击";
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
continueStartRecording();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 实际开始录音的逻辑
|
||||||
|
function continueStartRecording() {
|
||||||
|
// 重置录音数据
|
||||||
|
recordedPcmData = [];
|
||||||
|
recordedOpusData = [];
|
||||||
|
window.audioDataBuffer = new Int16Array(0); // 重置缓冲区
|
||||||
|
|
||||||
|
// 初始化Opus
|
||||||
|
initOpus().then(success => {
|
||||||
|
if (!success) {
|
||||||
|
statusLabel.textContent = "Opus初始化失败";
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
console.log("开始录音,参数:", {
|
||||||
|
sampleRate: SAMPLE_RATE,
|
||||||
|
channels: CHANNELS,
|
||||||
|
frameSize: FRAME_SIZE,
|
||||||
|
bufferSize: BUFFER_SIZE
|
||||||
|
});
|
||||||
|
|
||||||
|
// 请求麦克风权限
|
||||||
|
navigator.mediaDevices.getUserMedia({
|
||||||
|
audio: {
|
||||||
|
sampleRate: SAMPLE_RATE,
|
||||||
|
channelCount: CHANNELS,
|
||||||
|
echoCancellation: true,
|
||||||
|
noiseSuppression: true,
|
||||||
|
autoGainControl: true
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.then(stream => {
|
||||||
|
console.log("获取到麦克风流,实际参数:", stream.getAudioTracks()[0].getSettings());
|
||||||
|
|
||||||
|
// 检查流是否有效
|
||||||
|
if (!stream || !stream.getAudioTracks().length || !stream.getAudioTracks()[0].enabled) {
|
||||||
|
throw new Error("获取到的音频流无效");
|
||||||
|
}
|
||||||
|
|
||||||
|
mediaStream = stream;
|
||||||
|
mediaSource = audioContext.createMediaStreamSource(stream);
|
||||||
|
|
||||||
|
// 创建ScriptProcessor(虽然已弃用,但兼容性好)
|
||||||
|
// 在降级到ScriptProcessor之前尝试使用AudioWorklet
|
||||||
|
createAudioProcessor().then(processor => {
|
||||||
|
if (processor) {
|
||||||
|
console.log("使用AudioWorklet处理音频");
|
||||||
|
audioProcessor = processor;
|
||||||
|
// 连接音频处理链
|
||||||
|
mediaSource.connect(audioProcessor);
|
||||||
|
audioProcessor.connect(audioContext.destination);
|
||||||
|
} else {
|
||||||
|
console.log("回退到ScriptProcessor");
|
||||||
|
// 创建ScriptProcessor节点
|
||||||
|
audioProcessor = audioContext.createScriptProcessor(BUFFER_SIZE, CHANNELS, CHANNELS);
|
||||||
|
|
||||||
|
// 处理音频数据
|
||||||
|
audioProcessor.onaudioprocess = processAudioData;
|
||||||
|
|
||||||
|
// 连接音频处理链
|
||||||
|
mediaSource.connect(audioProcessor);
|
||||||
|
audioProcessor.connect(audioContext.destination);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 更新UI
|
||||||
|
isRecording = true;
|
||||||
|
statusLabel.textContent = "录音中...";
|
||||||
|
startButton.disabled = true;
|
||||||
|
stopButton.disabled = false;
|
||||||
|
playButton.disabled = true;
|
||||||
|
}).catch(error => {
|
||||||
|
console.error("创建音频处理器失败:", error);
|
||||||
|
statusLabel.textContent = "创建音频处理器失败";
|
||||||
|
});
|
||||||
|
})
|
||||||
|
.catch(error => {
|
||||||
|
console.error("获取麦克风失败:", error);
|
||||||
|
statusLabel.textContent = "获取麦克风失败: " + error.message;
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// 创建AudioWorklet处理器
|
||||||
|
async function createAudioProcessor() {
|
||||||
|
try {
|
||||||
|
// 尝试使用更现代的AudioWorklet API
|
||||||
|
if ('AudioWorklet' in window && 'AudioWorkletNode' in window) {
|
||||||
|
// 定义AudioWorklet处理器代码
|
||||||
|
const workletCode = `
|
||||||
|
class OpusRecorderProcessor extends AudioWorkletProcessor {
|
||||||
|
constructor() {
|
||||||
|
super();
|
||||||
|
this.buffers = [];
|
||||||
|
this.frameSize = ${FRAME_SIZE};
|
||||||
|
this.buffer = new Float32Array(this.frameSize);
|
||||||
|
this.bufferIndex = 0;
|
||||||
|
this.isRecording = false;
|
||||||
|
|
||||||
|
this.port.onmessage = (event) => {
|
||||||
|
if (event.data.command === 'start') {
|
||||||
|
this.isRecording = true;
|
||||||
|
} else if (event.data.command === 'stop') {
|
||||||
|
this.isRecording = false;
|
||||||
|
// 发送最后的缓冲区
|
||||||
|
if (this.bufferIndex > 0) {
|
||||||
|
const finalBuffer = this.buffer.slice(0, this.bufferIndex);
|
||||||
|
this.port.postMessage({ buffer: finalBuffer });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
process(inputs, outputs) {
|
||||||
|
if (!this.isRecording) return true;
|
||||||
|
|
||||||
|
// 获取输入数据
|
||||||
|
const input = inputs[0][0]; // mono channel
|
||||||
|
if (!input || input.length === 0) return true;
|
||||||
|
|
||||||
|
// 将输入数据添加到缓冲区
|
||||||
|
for (let i = 0; i < input.length; i++) {
|
||||||
|
this.buffer[this.bufferIndex++] = input[i];
|
||||||
|
|
||||||
|
// 当缓冲区填满时,发送给主线程
|
||||||
|
if (this.bufferIndex >= this.frameSize) {
|
||||||
|
this.port.postMessage({ buffer: this.buffer.slice() });
|
||||||
|
this.bufferIndex = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
registerProcessor('opus-recorder-processor', OpusRecorderProcessor);
|
||||||
|
`;
|
||||||
|
|
||||||
|
// 创建Blob URL
|
||||||
|
const blob = new Blob([workletCode], { type: 'application/javascript' });
|
||||||
|
const url = URL.createObjectURL(blob);
|
||||||
|
|
||||||
|
// 加载AudioWorklet模块
|
||||||
|
await audioContext.audioWorklet.addModule(url);
|
||||||
|
|
||||||
|
// 创建AudioWorkletNode
|
||||||
|
const workletNode = new AudioWorkletNode(audioContext, 'opus-recorder-processor');
|
||||||
|
|
||||||
|
// 处理从AudioWorklet接收的消息
|
||||||
|
workletNode.port.onmessage = (event) => {
|
||||||
|
if (event.data.buffer) {
|
||||||
|
// 使用与ScriptProcessor相同的处理逻辑
|
||||||
|
processAudioData({
|
||||||
|
inputBuffer: {
|
||||||
|
getChannelData: () => event.data.buffer
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// 启动录音
|
||||||
|
workletNode.port.postMessage({ command: 'start' });
|
||||||
|
|
||||||
|
// 保存停止函数
|
||||||
|
workletNode.stopRecording = () => {
|
||||||
|
workletNode.port.postMessage({ command: 'stop' });
|
||||||
|
};
|
||||||
|
|
||||||
|
console.log("AudioWorklet 音频处理器创建成功");
|
||||||
|
return workletNode;
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error("创建AudioWorklet失败,将使用ScriptProcessor:", error);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 如果AudioWorklet不可用或失败,返回null以便回退到ScriptProcessor
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 处理音频数据
|
||||||
|
function processAudioData(e) {
|
||||||
|
// 获取输入缓冲区
|
||||||
|
const inputBuffer = e.inputBuffer;
|
||||||
|
|
||||||
|
// 获取第一个通道的Float32数据
|
||||||
|
const inputData = inputBuffer.getChannelData(0);
|
||||||
|
|
||||||
|
// 添加调试信息
|
||||||
|
const nonZeroCount = Array.from(inputData).filter(x => Math.abs(x) > 0.001).length;
|
||||||
|
console.log(`接收到音频数据: ${inputData.length} 个样本, 非零样本数: ${nonZeroCount}`);
|
||||||
|
|
||||||
|
// 如果全是0,可能是麦克风没有正确获取声音
|
||||||
|
if (nonZeroCount < 5) {
|
||||||
|
console.warn("警告: 检测到大量静音样本,请检查麦克风是否正常工作");
|
||||||
|
// 继续处理,以防有些样本确实是静音
|
||||||
|
}
|
||||||
|
|
||||||
|
// 存储PCM数据用于调试
|
||||||
|
recordedPcmData.push(new Float32Array(inputData));
|
||||||
|
|
||||||
|
// 转换为Int16数据供Opus编码
|
||||||
|
const int16Data = convertFloat32ToInt16(inputData);
|
||||||
|
|
||||||
|
// 如果收集到的数据不是FRAME_SIZE的整数倍,需要进行处理
|
||||||
|
// 创建静态缓冲区来存储不足一帧的数据
|
||||||
|
if (!window.audioDataBuffer) {
|
||||||
|
window.audioDataBuffer = new Int16Array(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 合并之前缓存的数据和新数据
|
||||||
|
const combinedData = new Int16Array(window.audioDataBuffer.length + int16Data.length);
|
||||||
|
combinedData.set(window.audioDataBuffer);
|
||||||
|
combinedData.set(int16Data, window.audioDataBuffer.length);
|
||||||
|
|
||||||
|
// 处理完整帧
|
||||||
|
const frameCount = Math.floor(combinedData.length / FRAME_SIZE);
|
||||||
|
console.log(`可编码的完整帧数: ${frameCount}, 缓冲区总大小: ${combinedData.length}`);
|
||||||
|
|
||||||
|
for (let i = 0; i < frameCount; i++) {
|
||||||
|
const frameData = combinedData.subarray(i * FRAME_SIZE, (i + 1) * FRAME_SIZE);
|
||||||
|
|
||||||
|
try {
|
||||||
|
console.log(`编码第 ${i+1}/${frameCount} 帧, 帧大小: ${frameData.length}`);
|
||||||
|
const encodedData = opusEncoder.encode(frameData);
|
||||||
|
if (encodedData) {
|
||||||
|
console.log(`编码成功: ${encodedData.length} 字节`);
|
||||||
|
recordedOpusData.push(encodedData);
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error(`Opus编码帧 ${i+1} 失败:`, error);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 保存剩余不足一帧的数据
|
||||||
|
const remainingSamples = combinedData.length % FRAME_SIZE;
|
||||||
|
if (remainingSamples > 0) {
|
||||||
|
window.audioDataBuffer = combinedData.subarray(frameCount * FRAME_SIZE);
|
||||||
|
console.log(`保留 ${remainingSamples} 个样本到下一次处理`);
|
||||||
|
} else {
|
||||||
|
window.audioDataBuffer = new Int16Array(0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function stopRecording() {
|
||||||
|
if (!isRecording) return;
|
||||||
|
|
||||||
|
// 处理剩余的缓冲数据
|
||||||
|
if (window.audioDataBuffer && window.audioDataBuffer.length > 0) {
|
||||||
|
console.log(`停止录音,处理剩余的 ${window.audioDataBuffer.length} 个样本`);
|
||||||
|
// 如果剩余数据不足一帧,可以通过补零的方式凑成一帧
|
||||||
|
if (window.audioDataBuffer.length < FRAME_SIZE) {
|
||||||
|
const paddedFrame = new Int16Array(FRAME_SIZE);
|
||||||
|
paddedFrame.set(window.audioDataBuffer);
|
||||||
|
// 剩余部分填充为0
|
||||||
|
for (let i = window.audioDataBuffer.length; i < FRAME_SIZE; i++) {
|
||||||
|
paddedFrame[i] = 0;
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
console.log(`编码最后一帧(补零): ${paddedFrame.length} 样本`);
|
||||||
|
const encodedData = opusEncoder.encode(paddedFrame);
|
||||||
|
if (encodedData) {
|
||||||
|
recordedOpusData.push(encodedData);
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error("最后一帧Opus编码失败:", error);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// 如果数据超过一帧,按正常流程处理
|
||||||
|
processAudioData({
|
||||||
|
inputBuffer: {
|
||||||
|
getChannelData: () => convertInt16ToFloat32(window.audioDataBuffer)
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
window.audioDataBuffer = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 如果使用的是AudioWorklet,调用其特定的停止方法
|
||||||
|
if (audioProcessor && typeof audioProcessor.stopRecording === 'function') {
|
||||||
|
audioProcessor.stopRecording();
|
||||||
|
}
|
||||||
|
|
||||||
|
// 停止麦克风
|
||||||
|
if (mediaStream) {
|
||||||
|
mediaStream.getTracks().forEach(track => track.stop());
|
||||||
|
}
|
||||||
|
|
||||||
|
// 断开音频处理链
|
||||||
|
if (audioProcessor) {
|
||||||
|
try {
|
||||||
|
audioProcessor.disconnect();
|
||||||
|
if (mediaSource) mediaSource.disconnect();
|
||||||
|
} catch (error) {
|
||||||
|
console.warn("断开音频处理链时出错:", error);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 更新UI
|
||||||
|
isRecording = false;
|
||||||
|
statusLabel.textContent = "已停止录音,收集了 " + recordedOpusData.length + " 帧Opus数据";
|
||||||
|
startButton.disabled = false;
|
||||||
|
stopButton.disabled = true;
|
||||||
|
playButton.disabled = recordedOpusData.length === 0;
|
||||||
|
|
||||||
|
console.log("录制完成:",
|
||||||
|
"PCM帧数:", recordedPcmData.length,
|
||||||
|
"Opus帧数:", recordedOpusData.length);
|
||||||
|
}
|
||||||
|
|
||||||
|
function playRecording() {
|
||||||
|
if (!recordedOpusData.length) {
|
||||||
|
statusLabel.textContent = "没有可播放的录音";
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 将所有Opus数据解码为PCM
|
||||||
|
let allDecodedData = [];
|
||||||
|
|
||||||
|
for (const opusData of recordedOpusData) {
|
||||||
|
try {
|
||||||
|
// 解码为Int16数据
|
||||||
|
const decodedData = opusDecoder.decode(opusData);
|
||||||
|
|
||||||
|
if (decodedData && decodedData.length > 0) {
|
||||||
|
// 将Int16数据转换为Float32
|
||||||
|
const float32Data = convertInt16ToFloat32(decodedData);
|
||||||
|
|
||||||
|
// 添加到总解码数据中
|
||||||
|
allDecodedData.push(...float32Data);
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Opus解码失败:", error);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 如果没有解码出数据,返回
|
||||||
|
if (allDecodedData.length === 0) {
|
||||||
|
statusLabel.textContent = "解码失败,无法播放";
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 创建音频缓冲区
|
||||||
|
const audioBuffer = audioContext.createBuffer(CHANNELS, allDecodedData.length, SAMPLE_RATE);
|
||||||
|
audioBuffer.copyToChannel(new Float32Array(allDecodedData), 0);
|
||||||
|
|
||||||
|
// 创建音频源并播放
|
||||||
|
const source = audioContext.createBufferSource();
|
||||||
|
source.buffer = audioBuffer;
|
||||||
|
source.connect(audioContext.destination);
|
||||||
|
source.start();
|
||||||
|
|
||||||
|
// 更新UI
|
||||||
|
statusLabel.textContent = "正在播放...";
|
||||||
|
playButton.disabled = true;
|
||||||
|
|
||||||
|
// 播放结束后恢复UI
|
||||||
|
source.onended = () => {
|
||||||
|
statusLabel.textContent = "播放完毕";
|
||||||
|
playButton.disabled = false;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
// 模拟服务端返回的Opus数据进行解码播放
|
||||||
|
function playOpusFromServer(opusData) {
|
||||||
|
// 这个函数展示如何处理服务端返回的opus数据
|
||||||
|
// opusData应该是一个包含opus帧的数组
|
||||||
|
|
||||||
|
if (!opusDecoder) {
|
||||||
|
initOpus().then(success => {
|
||||||
|
if (success) {
|
||||||
|
decodeAndPlayOpusData(opusData);
|
||||||
|
} else {
|
||||||
|
statusLabel.textContent = "Opus解码器初始化失败";
|
||||||
|
}
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
decodeAndPlayOpusData(opusData);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function decodeAndPlayOpusData(opusData) {
|
||||||
|
let allDecodedData = [];
|
||||||
|
|
||||||
|
for (const frame of opusData) {
|
||||||
|
try {
|
||||||
|
const decodedData = opusDecoder.decode(frame);
|
||||||
|
if (decodedData && decodedData.length > 0) {
|
||||||
|
const float32Data = convertInt16ToFloat32(decodedData);
|
||||||
|
allDecodedData.push(...float32Data);
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error("服务端Opus数据解码失败:", error);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (allDecodedData.length === 0) {
|
||||||
|
statusLabel.textContent = "服务端数据解码失败";
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const audioBuffer = audioContext.createBuffer(CHANNELS, allDecodedData.length, SAMPLE_RATE);
|
||||||
|
audioBuffer.copyToChannel(new Float32Array(allDecodedData), 0);
|
||||||
|
|
||||||
|
const source = audioContext.createBufferSource();
|
||||||
|
source.buffer = audioBuffer;
|
||||||
|
source.connect(audioContext.destination);
|
||||||
|
source.start();
|
||||||
|
|
||||||
|
statusLabel.textContent = "正在播放服务端数据...";
|
||||||
|
source.onended = () => statusLabel.textContent = "服务端数据播放完毕";
|
||||||
|
}
|
||||||
@@ -0,0 +1,464 @@
|
|||||||
|
<!DOCTYPE html>
|
||||||
|
<html lang="en">
|
||||||
|
<head>
|
||||||
|
<meta charset="UTF-8">
|
||||||
|
<title>Opus 编解码测试</title>
|
||||||
|
<style>
|
||||||
|
body {
|
||||||
|
font-family: Arial, sans-serif;
|
||||||
|
max-width: 800px;
|
||||||
|
margin: 0 auto;
|
||||||
|
padding: 20px;
|
||||||
|
}
|
||||||
|
.container {
|
||||||
|
background-color: #f5f5f5;
|
||||||
|
padding: 20px;
|
||||||
|
border-radius: 8px;
|
||||||
|
box-shadow: 0 2px 10px rgba(0, 0, 0, 0.1);
|
||||||
|
}
|
||||||
|
button {
|
||||||
|
padding: 8px 15px;
|
||||||
|
margin-right: 10px;
|
||||||
|
border: none;
|
||||||
|
border-radius: 5px;
|
||||||
|
background-color: #4285f4;
|
||||||
|
color: white;
|
||||||
|
cursor: pointer;
|
||||||
|
transition: background-color 0.2s;
|
||||||
|
}
|
||||||
|
button:hover {
|
||||||
|
background-color: #3367d6;
|
||||||
|
}
|
||||||
|
button:disabled {
|
||||||
|
background-color: #cccccc;
|
||||||
|
cursor: not-allowed;
|
||||||
|
}
|
||||||
|
#status {
|
||||||
|
font-weight: bold;
|
||||||
|
}
|
||||||
|
#scriptStatus {
|
||||||
|
position: fixed;
|
||||||
|
top: 10px;
|
||||||
|
left: 50%;
|
||||||
|
transform: translateX(-50%);
|
||||||
|
padding: 15px;
|
||||||
|
border-radius: 5px;
|
||||||
|
display: block;
|
||||||
|
z-index: 100;
|
||||||
|
transition: all 0.3s ease;
|
||||||
|
}
|
||||||
|
#scriptStatus.info {
|
||||||
|
background-color: #e8f0fe;
|
||||||
|
color: #4285f4;
|
||||||
|
border-left: 4px solid #4285f4;
|
||||||
|
}
|
||||||
|
#scriptStatus.success {
|
||||||
|
background-color: #e6f4ea;
|
||||||
|
color: #0f9d58;
|
||||||
|
border-left: 4px solid #0f9d58;
|
||||||
|
}
|
||||||
|
#scriptStatus.error {
|
||||||
|
background-color: #fce8e6;
|
||||||
|
color: #db4437;
|
||||||
|
border-left: 4px solid #db4437;
|
||||||
|
}
|
||||||
|
#debugInfo {
|
||||||
|
margin-top: 20px;
|
||||||
|
border: 1px solid #ccc;
|
||||||
|
border-radius: 5px;
|
||||||
|
padding: 10px;
|
||||||
|
font-family: monospace;
|
||||||
|
font-size: 12px;
|
||||||
|
max-height: 200px;
|
||||||
|
overflow-y: auto;
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
#showDebug {
|
||||||
|
margin-top: 10px;
|
||||||
|
background-color: #f5f5f5;
|
||||||
|
color: #444;
|
||||||
|
font-size: 12px;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<div class="container">
|
||||||
|
<h2>Opus 编解码录音播放测试</h2>
|
||||||
|
|
||||||
|
<div id="scriptStatus" class="info">正在加载Opus库...</div>
|
||||||
|
|
||||||
|
<div>
|
||||||
|
<button id="initAudio" style="background-color: #34a853;">初始化音频</button>
|
||||||
|
<button id="testMic" style="background-color: #fbbc05;">测试麦克风</button>
|
||||||
|
<p></p>
|
||||||
|
<button id="start" disabled>开始录音</button>
|
||||||
|
<button id="stop" disabled>停止录音</button>
|
||||||
|
<button id="play" disabled>播放录音</button>
|
||||||
|
<p>录音状态: <span id="status">待机,正在初始化...</span></p>
|
||||||
|
<button id="showDebug">显示/隐藏调试信息</button>
|
||||||
|
<div id="debugInfo"></div>
|
||||||
|
<div id="audioMeter" style="margin-top: 10px; height: 20px; background-color: #eee; border-radius: 4px; overflow: hidden; display: none;">
|
||||||
|
<div id="audioLevel" style="height: 100%; width: 0%; background-color: #4285f4; transition: width 0.1s;"></div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Opus解码库 - 这个库会设置一个全局Module变量 -->
|
||||||
|
<script>
|
||||||
|
// 定义全局变量以跟踪库加载状态
|
||||||
|
window.opusLoaded = false;
|
||||||
|
window.startButton = document.getElementById("start");
|
||||||
|
window.stopButton = document.getElementById("stop");
|
||||||
|
window.playButton = document.getElementById("play");
|
||||||
|
window.statusLabel = document.getElementById("status");
|
||||||
|
window.debugInfo = document.getElementById("debugInfo");
|
||||||
|
window.audioContextReady = false;
|
||||||
|
window.testMicActive = false;
|
||||||
|
|
||||||
|
// 显示/隐藏调试信息
|
||||||
|
document.getElementById("showDebug").addEventListener("click", function() {
|
||||||
|
if (debugInfo.style.display === "none" || !debugInfo.style.display) {
|
||||||
|
debugInfo.style.display = "block";
|
||||||
|
this.textContent = "隐藏调试信息";
|
||||||
|
} else {
|
||||||
|
debugInfo.style.display = "none";
|
||||||
|
this.textContent = "显示调试信息";
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// 添加初始化音频按钮事件
|
||||||
|
document.getElementById("initAudio").addEventListener("click", function() {
|
||||||
|
initializeAudioSystem();
|
||||||
|
});
|
||||||
|
|
||||||
|
// 添加测试麦克风按钮事件
|
||||||
|
document.getElementById("testMic").addEventListener("click", function() {
|
||||||
|
if (window.testMicActive) {
|
||||||
|
stopMicTest();
|
||||||
|
} else {
|
||||||
|
startMicTest();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// 初始化音频系统
|
||||||
|
function initializeAudioSystem() {
|
||||||
|
try {
|
||||||
|
log("初始化音频系统...");
|
||||||
|
// 创建临时AudioContext来触发用户授权
|
||||||
|
const tempContext = new (window.AudioContext || window.webkitAudioContext)({
|
||||||
|
sampleRate: 16000,
|
||||||
|
latencyHint: 'interactive'
|
||||||
|
});
|
||||||
|
|
||||||
|
// 创建振荡器并播放短促的声音
|
||||||
|
const oscillator = tempContext.createOscillator();
|
||||||
|
const gain = tempContext.createGain();
|
||||||
|
gain.gain.value = 0.1; // 很小的音量
|
||||||
|
oscillator.connect(gain);
|
||||||
|
gain.connect(tempContext.destination);
|
||||||
|
oscillator.frequency.value = 440; // A4
|
||||||
|
oscillator.start();
|
||||||
|
|
||||||
|
// 0.2秒后停止
|
||||||
|
setTimeout(() => {
|
||||||
|
oscillator.stop();
|
||||||
|
// 关闭上下文
|
||||||
|
tempContext.close().then(() => {
|
||||||
|
log("音频系统初始化成功", "success");
|
||||||
|
updateScriptStatus("音频系统已激活", "success");
|
||||||
|
document.getElementById("initAudio").disabled = true;
|
||||||
|
document.getElementById("initAudio").textContent = "音频已初始化";
|
||||||
|
window.audioContextReady = true;
|
||||||
|
|
||||||
|
// 如果Opus已加载,启用开始录音按钮
|
||||||
|
if (window.opusLoaded) {
|
||||||
|
startButton.disabled = false;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}, 200);
|
||||||
|
} catch (err) {
|
||||||
|
log("初始化音频系统失败: " + err.message, "error");
|
||||||
|
updateScriptStatus("初始化音频失败: " + err.message, "error");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 测试麦克风
|
||||||
|
function startMicTest() {
|
||||||
|
const audioMeter = document.getElementById("audioMeter");
|
||||||
|
const audioLevel = document.getElementById("audioLevel");
|
||||||
|
const testMicBtn = document.getElementById("testMic");
|
||||||
|
|
||||||
|
if (!navigator.mediaDevices || !navigator.mediaDevices.getUserMedia) {
|
||||||
|
log("浏览器不支持麦克风访问", "error");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
log("开始麦克风测试...");
|
||||||
|
audioMeter.style.display = "block";
|
||||||
|
testMicBtn.textContent = "停止测试";
|
||||||
|
testMicBtn.style.backgroundColor = "#ea4335";
|
||||||
|
window.testMicActive = true;
|
||||||
|
|
||||||
|
// 创建音频上下文
|
||||||
|
const testContext = new (window.AudioContext || window.webkitAudioContext)();
|
||||||
|
window.testContext = testContext;
|
||||||
|
|
||||||
|
// 获取麦克风权限
|
||||||
|
navigator.mediaDevices.getUserMedia({
|
||||||
|
audio: {
|
||||||
|
echoCancellation: true,
|
||||||
|
noiseSuppression: true,
|
||||||
|
autoGainControl: true
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.then(stream => {
|
||||||
|
log("已获取麦克风访问权限", "success");
|
||||||
|
|
||||||
|
// 保存流以便稍后关闭
|
||||||
|
window.testStream = stream;
|
||||||
|
|
||||||
|
// 创建音频分析器
|
||||||
|
const source = testContext.createMediaStreamSource(stream);
|
||||||
|
const analyser = testContext.createAnalyser();
|
||||||
|
analyser.fftSize = 256;
|
||||||
|
source.connect(analyser);
|
||||||
|
|
||||||
|
// 创建音量显示更新函数
|
||||||
|
const bufferLength = analyser.frequencyBinCount;
|
||||||
|
const dataArray = new Uint8Array(bufferLength);
|
||||||
|
|
||||||
|
function updateMeter() {
|
||||||
|
if (!window.testMicActive) return;
|
||||||
|
|
||||||
|
analyser.getByteFrequencyData(dataArray);
|
||||||
|
|
||||||
|
// 计算音量级别 (0-100)
|
||||||
|
let sum = 0;
|
||||||
|
for (let i = 0; i < bufferLength; i++) {
|
||||||
|
sum += dataArray[i];
|
||||||
|
}
|
||||||
|
const average = sum / bufferLength;
|
||||||
|
const level = Math.min(100, Math.max(0, average * 2));
|
||||||
|
|
||||||
|
// 更新音量计
|
||||||
|
audioLevel.style.width = level + "%";
|
||||||
|
|
||||||
|
// 如果有声音,记录日志
|
||||||
|
if (level > 10) {
|
||||||
|
log(`检测到声音: ${level.toFixed(1)}%`);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 循环更新
|
||||||
|
window.testMicAnimationFrame = requestAnimationFrame(updateMeter);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 开始更新
|
||||||
|
updateMeter();
|
||||||
|
})
|
||||||
|
.catch(err => {
|
||||||
|
log("麦克风测试失败: " + err.message, "error");
|
||||||
|
window.testMicActive = false;
|
||||||
|
testMicBtn.textContent = "测试麦克风";
|
||||||
|
testMicBtn.style.backgroundColor = "#fbbc05";
|
||||||
|
audioMeter.style.display = "none";
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// 停止麦克风测试
|
||||||
|
function stopMicTest() {
|
||||||
|
const audioMeter = document.getElementById("audioMeter");
|
||||||
|
const testMicBtn = document.getElementById("testMic");
|
||||||
|
|
||||||
|
log("停止麦克风测试");
|
||||||
|
window.testMicActive = false;
|
||||||
|
testMicBtn.textContent = "测试麦克风";
|
||||||
|
testMicBtn.style.backgroundColor = "#fbbc05";
|
||||||
|
|
||||||
|
// 停止分析器动画
|
||||||
|
if (window.testMicAnimationFrame) {
|
||||||
|
cancelAnimationFrame(window.testMicAnimationFrame);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 停止麦克风流
|
||||||
|
if (window.testStream) {
|
||||||
|
window.testStream.getTracks().forEach(track => track.stop());
|
||||||
|
}
|
||||||
|
|
||||||
|
// 关闭测试上下文
|
||||||
|
if (window.testContext) {
|
||||||
|
window.testContext.close();
|
||||||
|
}
|
||||||
|
|
||||||
|
// 隐藏音量计
|
||||||
|
audioMeter.style.display = "none";
|
||||||
|
}
|
||||||
|
|
||||||
|
// 初始化AudioContext
|
||||||
|
function initAudioContext() {
|
||||||
|
try {
|
||||||
|
window.AudioContext = window.AudioContext || window.webkitAudioContext;
|
||||||
|
if (!window.AudioContext) {
|
||||||
|
throw new Error("浏览器不支持AudioContext");
|
||||||
|
}
|
||||||
|
|
||||||
|
// 我们在app.js中会创建AudioContext,这里只检查兼容性
|
||||||
|
log("AudioContext API 可用");
|
||||||
|
return true;
|
||||||
|
} catch (err) {
|
||||||
|
log("AudioContext初始化失败: " + err.message, "error");
|
||||||
|
updateScriptStatus("AudioContext初始化失败,录音功能不可用", "error");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 检查麦克风权限
|
||||||
|
function checkMicrophonePermission() {
|
||||||
|
if (!navigator.mediaDevices || !navigator.mediaDevices.getUserMedia) {
|
||||||
|
log("浏览器不支持getUserMedia API", "error");
|
||||||
|
updateScriptStatus("浏览器不支持录音功能", "error");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
log("getUserMedia API 可用");
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 添加调试日志
|
||||||
|
function log(message, type = "info") {
|
||||||
|
console.log(message);
|
||||||
|
const time = new Date().toLocaleTimeString();
|
||||||
|
const entry = document.createElement("div");
|
||||||
|
entry.textContent = `[${time}] ${message}`;
|
||||||
|
|
||||||
|
if (type === "error") {
|
||||||
|
entry.style.color = "#db4437";
|
||||||
|
} else if (type === "success") {
|
||||||
|
entry.style.color = "#0f9d58";
|
||||||
|
}
|
||||||
|
|
||||||
|
debugInfo.appendChild(entry);
|
||||||
|
debugInfo.scrollTop = debugInfo.scrollHeight;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 更新脚本状态显示
|
||||||
|
function updateScriptStatus(message, type) {
|
||||||
|
const statusElement = document.getElementById('scriptStatus');
|
||||||
|
if (statusElement) {
|
||||||
|
statusElement.textContent = message;
|
||||||
|
statusElement.className = type;
|
||||||
|
statusElement.style.display = 'block';
|
||||||
|
}
|
||||||
|
log(message, type);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 检查Opus库是否已加载
|
||||||
|
function checkOpusLoaded() {
|
||||||
|
try {
|
||||||
|
// 检查Module是否存在(本地库导出的全局变量)
|
||||||
|
if (typeof Module === 'undefined') {
|
||||||
|
log("Module对象不存在", "error");
|
||||||
|
throw new Error('Opus库未加载,Module对象不存在');
|
||||||
|
}
|
||||||
|
|
||||||
|
// 记录Module对象结构以便调试
|
||||||
|
log("Module对象结构: " + Object.keys(Module).join(", "));
|
||||||
|
|
||||||
|
// 尝试使用全局Module函数
|
||||||
|
if (typeof Module._opus_decoder_get_size === 'function') {
|
||||||
|
window.ModuleInstance = Module;
|
||||||
|
log('Opus库加载成功(使用全局Module)', "success");
|
||||||
|
updateScriptStatus('Opus库加载成功', 'success');
|
||||||
|
|
||||||
|
// 启用开始录音按钮
|
||||||
|
startButton.disabled = false;
|
||||||
|
statusLabel.textContent = "待机";
|
||||||
|
|
||||||
|
// 3秒后隐藏状态
|
||||||
|
setTimeout(() => {
|
||||||
|
const statusElement = document.getElementById('scriptStatus');
|
||||||
|
if (statusElement) statusElement.style.display = 'none';
|
||||||
|
}, 3000);
|
||||||
|
|
||||||
|
window.opusLoaded = true;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 尝试使用Module.instance
|
||||||
|
if (typeof Module.instance !== 'undefined' && typeof Module.instance._opus_decoder_get_size === 'function') {
|
||||||
|
window.ModuleInstance = Module.instance;
|
||||||
|
log('Opus库加载成功(使用Module.instance)', "success");
|
||||||
|
updateScriptStatus('Opus库加载成功', 'success');
|
||||||
|
|
||||||
|
// 启用开始录音按钮
|
||||||
|
startButton.disabled = false;
|
||||||
|
statusLabel.textContent = "待机";
|
||||||
|
|
||||||
|
// 3秒后隐藏状态
|
||||||
|
setTimeout(() => {
|
||||||
|
const statusElement = document.getElementById('scriptStatus');
|
||||||
|
if (statusElement) statusElement.style.display = 'none';
|
||||||
|
}, 3000);
|
||||||
|
|
||||||
|
window.opusLoaded = true;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 检查是否有其他导出方式
|
||||||
|
log("Module上可用方法: " + Object.getOwnPropertyNames(Module).filter(prop => typeof Module[prop] === 'function').join(", "));
|
||||||
|
|
||||||
|
if (typeof Module.onRuntimeInitialized === 'function' || typeof Module.onRuntimeInitialized === 'undefined') {
|
||||||
|
log("Module.onRuntimeInitialized 尚未执行,等待模块初始化完成...");
|
||||||
|
// Module可能未完成初始化,注册回调
|
||||||
|
Module.onRuntimeInitialized = function() {
|
||||||
|
log("Opus库运行时初始化完成,重新检查");
|
||||||
|
checkOpusLoaded();
|
||||||
|
};
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
throw new Error('Opus解码函数未找到,可能Module结构不正确');
|
||||||
|
} catch (err) {
|
||||||
|
log(`Opus库加载失败: ${err.message}`, "error");
|
||||||
|
updateScriptStatus(`Opus库加载失败: ${err.message}`, 'error');
|
||||||
|
statusLabel.textContent = "错误:Opus库加载失败";
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 页面加载完成后检查浏览器能力和Opus库
|
||||||
|
window.addEventListener('load', function() {
|
||||||
|
log("页面加载完成,开始初始化...");
|
||||||
|
updateScriptStatus('正在初始化录音环境...', 'info');
|
||||||
|
|
||||||
|
// 检查浏览器能力
|
||||||
|
const audioContextSupported = initAudioContext();
|
||||||
|
const microphoneSupported = checkMicrophonePermission();
|
||||||
|
|
||||||
|
if (!audioContextSupported || !microphoneSupported) {
|
||||||
|
log("浏览器不支持必要API,无法进行录音", "error");
|
||||||
|
updateScriptStatus('浏览器不支持录音功能', 'error');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
log("检查环境完成,开始加载Opus库");
|
||||||
|
updateScriptStatus('正在加载Opus库...', 'info');
|
||||||
|
|
||||||
|
// 延迟一小段时间再检查,确保库有时间加载
|
||||||
|
setTimeout(function checkAndRetry() {
|
||||||
|
if (!checkOpusLoaded()) {
|
||||||
|
// 如果加载失败,5秒后重试
|
||||||
|
log("Opus库加载失败,5秒后重试...");
|
||||||
|
updateScriptStatus('Opus库加载失败,正在重试...', 'error');
|
||||||
|
setTimeout(checkAndRetry, 5000);
|
||||||
|
}
|
||||||
|
}, 1000);
|
||||||
|
});
|
||||||
|
|
||||||
|
// 防止按钮误操作
|
||||||
|
document.getElementById("stop").disabled = true;
|
||||||
|
document.getElementById("play").disabled = true;
|
||||||
|
</script>
|
||||||
|
<script src="../libopus.js"></script>
|
||||||
|
<script src="app.js"></script>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
@@ -0,0 +1,425 @@
|
|||||||
|
body {
|
||||||
|
font-family: 'PingFang SC', 'Microsoft YaHei', sans-serif;
|
||||||
|
margin: 0;
|
||||||
|
padding: 20px;
|
||||||
|
background-color: #f5f5f5;
|
||||||
|
}
|
||||||
|
|
||||||
|
.container {
|
||||||
|
max-width: 1000px;
|
||||||
|
margin: 0 auto;
|
||||||
|
background-color: white;
|
||||||
|
border-radius: 10px;
|
||||||
|
box-shadow: 0 2px 10px rgba(0, 0, 0, 0.1);
|
||||||
|
padding: 10px 20px 10px 20px;
|
||||||
|
}
|
||||||
|
|
||||||
|
h1 {
|
||||||
|
color: #333;
|
||||||
|
text-align: center;
|
||||||
|
margin-bottom: 30px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.section {
|
||||||
|
margin-bottom: 5px;
|
||||||
|
padding: 10px;
|
||||||
|
border-radius: 10px;
|
||||||
|
background-color: #f9f9f9;
|
||||||
|
}
|
||||||
|
|
||||||
|
.section h2 {
|
||||||
|
margin-top: 0;
|
||||||
|
color: #444;
|
||||||
|
font-size: 18px;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 10px;
|
||||||
|
padding: 10px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.section h2 .toggle-button {
|
||||||
|
margin-left: auto;
|
||||||
|
padding: 4px 12px;
|
||||||
|
font-size: 12px;
|
||||||
|
border-radius: 4px;
|
||||||
|
cursor: pointer;
|
||||||
|
height: 28px;
|
||||||
|
line-height: 20px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.device-info {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 20px;
|
||||||
|
margin-left: 20px;
|
||||||
|
padding: 0 15px;
|
||||||
|
background-color: #f9f9f9;
|
||||||
|
border-radius: 4px;
|
||||||
|
height: 28px;
|
||||||
|
line-height: 28px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.device-info span {
|
||||||
|
color: #666;
|
||||||
|
font-size: 13px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.device-info strong {
|
||||||
|
color: #333;
|
||||||
|
font-weight: 500;
|
||||||
|
}
|
||||||
|
|
||||||
|
.config-panel {
|
||||||
|
display: none;
|
||||||
|
transition: all 0.3s ease;
|
||||||
|
margin-top: 5px;
|
||||||
|
padding: 5px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.config-panel.expanded {
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
|
||||||
|
.control-panel {
|
||||||
|
display: flex;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
gap: 10px;
|
||||||
|
margin-top: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
button {
|
||||||
|
padding: 8px 15px;
|
||||||
|
border: none;
|
||||||
|
border-radius: 5px;
|
||||||
|
background-color: #4285f4;
|
||||||
|
color: white;
|
||||||
|
cursor: pointer;
|
||||||
|
transition: background-color 0.2s;
|
||||||
|
}
|
||||||
|
|
||||||
|
button:hover {
|
||||||
|
background-color: #3367d6;
|
||||||
|
}
|
||||||
|
|
||||||
|
button:disabled {
|
||||||
|
background-color: #cccccc;
|
||||||
|
cursor: not-allowed;
|
||||||
|
}
|
||||||
|
|
||||||
|
#serverUrl,
|
||||||
|
#otaUrl {
|
||||||
|
flex-grow: 1;
|
||||||
|
padding: 8px;
|
||||||
|
border: 1px solid #ddd;
|
||||||
|
border-radius: 5px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-input {
|
||||||
|
display: flex;
|
||||||
|
gap: 10px;
|
||||||
|
margin-bottom: 20px;
|
||||||
|
}
|
||||||
|
|
||||||
|
#messageInput {
|
||||||
|
flex-grow: 1;
|
||||||
|
padding: 8px;
|
||||||
|
border: 1px solid #ddd;
|
||||||
|
border-radius: 5px;
|
||||||
|
}
|
||||||
|
|
||||||
|
#nfcCardId {
|
||||||
|
flex-grow: 1;
|
||||||
|
padding: 8px;
|
||||||
|
border: 1px solid #ddd;
|
||||||
|
border-radius: 5px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.conversation {
|
||||||
|
max-height: 300px;
|
||||||
|
overflow-y: auto;
|
||||||
|
border: 1px solid #ddd;
|
||||||
|
border-radius: 5px;
|
||||||
|
padding: 10px;
|
||||||
|
background-color: white;
|
||||||
|
flex: 1;
|
||||||
|
margin-right: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.message {
|
||||||
|
margin-bottom: 10px;
|
||||||
|
padding: 8px 12px;
|
||||||
|
border-radius: 8px;
|
||||||
|
max-width: 80%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.user {
|
||||||
|
background-color: #e2f2ff;
|
||||||
|
margin-left: auto;
|
||||||
|
margin-right: 10px;
|
||||||
|
text-align: right;
|
||||||
|
}
|
||||||
|
|
||||||
|
.server {
|
||||||
|
background-color: #f0f0f0;
|
||||||
|
margin-right: auto;
|
||||||
|
margin-left: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.status {
|
||||||
|
color: #666;
|
||||||
|
font-style: italic;
|
||||||
|
font-size: 14px;
|
||||||
|
margin: 0;
|
||||||
|
padding: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.audio-controls {
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
gap: 10px;
|
||||||
|
margin-top: 20px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.audio-visualizer {
|
||||||
|
height: 60px;
|
||||||
|
width: 100%;
|
||||||
|
margin-top: 10px;
|
||||||
|
border: 1px solid #ddd;
|
||||||
|
border-radius: 5px;
|
||||||
|
background-color: #fafafa;
|
||||||
|
}
|
||||||
|
|
||||||
|
.record-button {
|
||||||
|
background-color: #db4437;
|
||||||
|
}
|
||||||
|
|
||||||
|
.record-button:hover {
|
||||||
|
background-color: #c53929;
|
||||||
|
}
|
||||||
|
|
||||||
|
.record-button.recording {
|
||||||
|
animation: pulse 1.5s infinite;
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes pulse {
|
||||||
|
0% {
|
||||||
|
background-color: #db4437;
|
||||||
|
}
|
||||||
|
|
||||||
|
50% {
|
||||||
|
background-color: #ff6659;
|
||||||
|
}
|
||||||
|
|
||||||
|
100% {
|
||||||
|
background-color: #db4437;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#logContainer {
|
||||||
|
margin-top: 0;
|
||||||
|
padding: 10px;
|
||||||
|
background-color: #f0f0f0;
|
||||||
|
border-radius: 5px;
|
||||||
|
font-family: monospace;
|
||||||
|
height: 300px;
|
||||||
|
overflow-y: auto;
|
||||||
|
flex: 1;
|
||||||
|
margin-left: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.log-entry {
|
||||||
|
margin: 5px 0;
|
||||||
|
font-size: 12px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.log-info {
|
||||||
|
color: #333;
|
||||||
|
}
|
||||||
|
|
||||||
|
.log-error {
|
||||||
|
color: #db4437;
|
||||||
|
}
|
||||||
|
|
||||||
|
.log-success {
|
||||||
|
color: #0f9d58;
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes fadeIn {
|
||||||
|
from {
|
||||||
|
opacity: 0;
|
||||||
|
transform: translate(-50%, -60%);
|
||||||
|
}
|
||||||
|
|
||||||
|
to {
|
||||||
|
opacity: 1;
|
||||||
|
transform: translate(-50%, -50%);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.script-status {
|
||||||
|
display: inline-block;
|
||||||
|
width: 10px;
|
||||||
|
height: 10px;
|
||||||
|
border-radius: 50%;
|
||||||
|
margin-right: 5px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.script-loaded {
|
||||||
|
background-color: #0f9d58;
|
||||||
|
}
|
||||||
|
|
||||||
|
.script-loading {
|
||||||
|
background-color: #f4b400;
|
||||||
|
}
|
||||||
|
|
||||||
|
.script-error {
|
||||||
|
background-color: #db4437;
|
||||||
|
}
|
||||||
|
|
||||||
|
.script-list {
|
||||||
|
margin: 10px 0;
|
||||||
|
padding: 10px;
|
||||||
|
background-color: #f9f9f9;
|
||||||
|
border-radius: 5px;
|
||||||
|
font-family: monospace;
|
||||||
|
font-size: 11px;
|
||||||
|
}
|
||||||
|
|
||||||
|
#scriptStatus.success {
|
||||||
|
background-color: #e6f4ea;
|
||||||
|
color: #0f9d58;
|
||||||
|
border-left: 4px solid #0f9d58;
|
||||||
|
}
|
||||||
|
|
||||||
|
#scriptStatus.error {
|
||||||
|
background-color: #fce8e6;
|
||||||
|
color: #db4437;
|
||||||
|
border-left: 4px solid #db4437;
|
||||||
|
}
|
||||||
|
|
||||||
|
#scriptStatus.warning {
|
||||||
|
background-color: #fef7e0;
|
||||||
|
color: #f4b400;
|
||||||
|
border-left: 4px solid #f4b400;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 标签页样式 */
|
||||||
|
.tabs {
|
||||||
|
display: flex;
|
||||||
|
margin-bottom: 20px;
|
||||||
|
border-bottom: 2px solid #e0e0e0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tab {
|
||||||
|
padding: 10px 20px;
|
||||||
|
cursor: pointer;
|
||||||
|
border: none;
|
||||||
|
background: none;
|
||||||
|
font-size: 16px;
|
||||||
|
color: #666;
|
||||||
|
position: relative;
|
||||||
|
transition: all 0.3s ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tab:hover {
|
||||||
|
color: #4285f4;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tab.active {
|
||||||
|
color: #4285f4;
|
||||||
|
font-weight: bold;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tab.active::after {
|
||||||
|
content: '';
|
||||||
|
position: absolute;
|
||||||
|
bottom: -2px;
|
||||||
|
left: 0;
|
||||||
|
width: 100%;
|
||||||
|
height: 2px;
|
||||||
|
background-color: #4285f4;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tab-content {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tab-content.active {
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
|
||||||
|
.flex-container {
|
||||||
|
display: flex;
|
||||||
|
gap: 20px;
|
||||||
|
margin-top: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.config-item {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
margin-bottom: 8px;
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.config-item label {
|
||||||
|
width: 100px;
|
||||||
|
text-align: right;
|
||||||
|
margin-right: 10px;
|
||||||
|
color: #666;
|
||||||
|
}
|
||||||
|
|
||||||
|
.config-item input {
|
||||||
|
flex-grow: 1;
|
||||||
|
padding: 6px;
|
||||||
|
border: 1px solid #ddd;
|
||||||
|
border-radius: 5px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.control-panel {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 10px;
|
||||||
|
margin-top: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.connection-controls {
|
||||||
|
display: flex;
|
||||||
|
gap: 10px;
|
||||||
|
align-items: center;
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.connection-controls input {
|
||||||
|
flex: 1;
|
||||||
|
padding: 8px;
|
||||||
|
border: 1px solid #ddd;
|
||||||
|
border-radius: 5px;
|
||||||
|
min-width: 200px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.connection-controls button {
|
||||||
|
white-space: nowrap;
|
||||||
|
padding: 8px 15px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.connection-status {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 20px;
|
||||||
|
margin-left: 20px;
|
||||||
|
padding: 0 15px;
|
||||||
|
background-color: #f9f9f9;
|
||||||
|
border-radius: 4px;
|
||||||
|
height: 28px;
|
||||||
|
line-height: 28px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.connection-status span {
|
||||||
|
color: #666;
|
||||||
|
font-size: 13px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.connection-status .status {
|
||||||
|
color: #333;
|
||||||
|
font-weight: 500;
|
||||||
|
}
|
||||||
File diff suppressed because it is too large
Load Diff
Reference in New Issue
Block a user