504 lines
19 KiB
HTML
Executable File
504 lines
19 KiB
HTML
Executable File
<!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>
|