/**
* 语音通话功能
*/
import { getSettings, splitAIMessages } from './config.js';
import { currentChatIndex } from './chat.js';
import { requestSave } from './save-manager.js';
import { refreshChatList } from './ui.js';
import { escapeHtml } from './utils.js';
// 通话状态
let callState = {
isActive: false,
isConnected: false,
isMuted: false,
isSpeakerOn: false,
startTime: null,
timerInterval: null,
dotsInterval: null,
connectTimeout: null, // 连接超时计时器
aiHangupTimeout: null, // AI主动挂断计时器
contactIndex: -1,
contactName: '',
contactAvatar: '',
messages: [], // 通话中的消息
contact: null,
initiator: 'user', // 谁发起的通话: 'user' 或 'ai'
rejectedByUser: false, // 是否被用户主动拒绝
rejectedByAI: false, // 是否被AI主动拒绝
hungUpByAI: false // 是否被AI主动挂断
};
// 开始语音通话
export function startVoiceCall(initiator = 'user', contactIndex = currentChatIndex) {
if (callState.isActive) return;
if (contactIndex < 0) return;
const settings = getSettings();
const contact = settings.contacts[contactIndex];
if (!contact) return;
callState.contactName = contact.name;
callState.contactAvatar = contact.avatar;
callState.contact = contact;
callState.contactIndex = contactIndex;
callState.isActive = true;
callState.isConnected = false;
callState.isMuted = false;
callState.isSpeakerOn = false;
callState.messages = []; // 重置消息
callState.initiator = initiator; // 记录谁发起的通话
callState.rejectedByUser = false; // 重置用户拒绝状态
callState.rejectedByAI = false; // 重置AI拒绝状态
callState.hungUpByAI = false; // 重置AI挂断状态
showCallPage();
startConnecting();
}
// 显示通话页面
function showCallPage() {
const page = document.getElementById('wechat-voice-call-page');
if (!page) return;
// 设置头像
const avatarEl = document.getElementById('wechat-voice-call-avatar');
if (avatarEl) {
const firstChar = callState.contactName ? callState.contactName.charAt(0) : '?';
if (callState.contactAvatar) {
avatarEl.innerHTML = ``;
} else {
avatarEl.textContent = firstChar;
}
}
// 设置名称
const nameEl = document.getElementById('wechat-voice-call-name');
if (nameEl) {
nameEl.textContent = callState.contactName;
}
// 设置状态 - 根据发起者显示不同文案
const statusEl = document.getElementById('wechat-voice-call-status');
if (statusEl) {
if (callState.initiator === 'ai') {
statusEl.textContent = '邀请你语音通话...';
} else {
statusEl.textContent = '等待对方接受邀请';
}
statusEl.classList.add('connecting');
}
// 重置时间显示 - 等待时隐藏
const timeEl = document.getElementById('wechat-voice-call-time');
if (timeEl) {
timeEl.textContent = '00:00';
timeEl.classList.add('hidden');
}
// 重置按钮状态
updateMuteButton();
updateSpeakerButton();
// 隐藏对话框并清空消息
const chatEl = document.getElementById('wechat-voice-call-chat');
if (chatEl) {
chatEl.classList.add('hidden');
}
// 隐藏输入框
const inputAreaEl = document.getElementById('wechat-voice-call-input-area');
if (inputAreaEl) {
inputAreaEl.classList.add('hidden');
}
const messagesEl = document.getElementById('wechat-voice-call-messages');
if (messagesEl) {
messagesEl.innerHTML = '';
}
// 根据发起者显示不同的操作按钮
const incomingActionsEl = document.getElementById('wechat-voice-call-incoming-actions');
const callActionsEl = document.getElementById('wechat-voice-call-actions');
if (callState.initiator === 'ai') {
// AI发起的来电:显示接听/拒绝按钮
if (incomingActionsEl) incomingActionsEl.classList.remove('hidden');
if (callActionsEl) callActionsEl.classList.add('hidden');
} else {
// 用户发起的呼叫:显示静音/挂断/扬声器按钮
if (incomingActionsEl) incomingActionsEl.classList.add('hidden');
if (callActionsEl) callActionsEl.classList.remove('hidden');
}
page.classList.remove('hidden');
bindCallEvents();
}
// 开始连接动画
async function startConnecting() {
const statusEl = document.getElementById('wechat-voice-call-status');
if (!statusEl) return;
let dotCount = 0;
clearInterval(callState.dotsInterval);
clearTimeout(callState.connectTimeout);
// 根据发起者显示不同的等待文案
const waitingText = callState.initiator === 'ai' ? '邀请你语音通话' : '等待对方接受邀请';
callState.dotsInterval = setInterval(() => {
dotCount = (dotCount + 1) % 4;
const dots = '.'.repeat(dotCount);
statusEl.textContent = waitingText + dots;
}, 500);
if (callState.initiator === 'user') {
// 用户发起:调用AI决策是否接听
const shouldAnswer = await askAIToAnswerCall(callState.contact, 'voice');
if (!callState.isActive) return; // 用户可能已经取消
if (shouldAnswer) {
// AI决定接听
if (callState.isActive && !callState.isConnected) {
onCallConnected();
}
} else {
// AI决定拒接
callState.rejectedByAI = true;
hangupCall();
}
} else {
// AI发起:15秒后如果用户没接就超时取消
callState.connectTimeout = setTimeout(() => {
if (callState.isActive && !callState.isConnected) {
// 超时,对方已取消(不是用户主动拒绝)
callState.rejectedByUser = false;
hangupCall();
}
}, 15000);
}
}
// AI决定是否接听用户的来电
async function askAIToAnswerCall(contact, callType = 'voice') {
if (!contact) return true;
try {
const { callAI } = await import('./ai.js');
const callTypeText = callType === 'video' ? '视频' : '语音';
const prompt = `[用户正在给你打${callTypeText}电话,你需要决定是否接听]
根据你的性格和当前心情决定:
- 如果你想接听,只回复:[接听]
- 如果你不想接听(比如在忙、生气、故意不接、想让用户着急等),只回复:[拒接]
【绝对禁止】
- 只能回复 [接听] 或 [拒接],不能有任何其他文字!
- [接听] 或 [拒接] 必须独立成行,前后不能有任何内容!
× 错误:好吧[接听] ← 有其他文字,错误!
× 错误:[拒接]哼 ← 有其他文字,错误!
√ 正确:[接听]
√ 正确:[拒接]
注意:大多数情况下你应该接听,只有特殊情况才拒接。`;
const response = await callAI(contact, prompt);
const trimmed = (response || '').trim();
console.log('[可乐] AI接听决策:', trimmed);
// 检查是否拒接
if (trimmed.includes('[拒接]') || trimmed.includes('拒接')) {
return false;
}
// 默认接听
return true;
} catch (err) {
console.error('[可乐] AI接听决策失败:', err);
// 出错时默认接听
return true;
}
}
// 通话接通
function onCallConnected() {
callState.isConnected = true;
callState.startTime = Date.now();
clearInterval(callState.dotsInterval);
clearTimeout(callState.connectTimeout);
const statusEl = document.getElementById('wechat-voice-call-status');
if (statusEl) {
statusEl.textContent = '通话中';
statusEl.classList.remove('connecting');
}
// 显示计时器
const timeEl = document.getElementById('wechat-voice-call-time');
if (timeEl) {
timeEl.classList.remove('hidden');
}
// 显示对话框
const chatEl = document.getElementById('wechat-voice-call-chat');
if (chatEl) {
chatEl.classList.remove('hidden');
}
// 显示输入框
const inputAreaEl = document.getElementById('wechat-voice-call-input-area');
if (inputAreaEl) {
inputAreaEl.classList.remove('hidden');
}
// 切换到通话中按钮(隐藏来电按钮,显示通话控制按钮)
const incomingActionsEl = document.getElementById('wechat-voice-call-incoming-actions');
const callActionsEl = document.getElementById('wechat-voice-call-actions');
if (incomingActionsEl) incomingActionsEl.classList.add('hidden');
if (callActionsEl) callActionsEl.classList.remove('hidden');
// 开始计时
startCallTimer();
// 如果是AI发起的通话,接通后AI自动发送第一条消息
if (callState.initiator === 'ai') {
triggerAIGreeting();
}
// 启动AI主动挂断检查(通话30秒后开始随机检查)
scheduleAIHangupCheck();
}
// 调度AI主动挂断检查
// 通话接通后30秒开始,每次用户发消息后AI回复时有5%概率挂断
// 同时设置一个180秒(3分钟)的保底挂断时间
function scheduleAIHangupCheck() {
// 清除已有的计时器
clearTimeout(callState.aiHangupTimeout);
// 设置保底挂断时间:通话3分钟后有50%概率挂断,超过5分钟必定挂断
const checkTime = 180000 + Math.random() * 120000; // 3-5分钟
callState.aiHangupTimeout = setTimeout(() => {
if (callState.isConnected) {
// 50%概率挂断,否则再等1-2分钟
if (Math.random() < 0.5) {
aiHangup();
} else {
// 再设置一个60-120秒后的必定挂断
callState.aiHangupTimeout = setTimeout(() => {
if (callState.isConnected) {
aiHangup();
}
}, 60000 + Math.random() * 60000);
}
}
}, checkTime);
}
// 每次AI回复后检查是否要挂断(5%概率,通话30秒后生效)
export function checkAIHangupAfterReply() {
if (!callState.isConnected || !callState.startTime) return false;
// 通话至少30秒后才开始随机挂断检查
const elapsed = Date.now() - callState.startTime;
if (elapsed < 30000) return false;
// 5%概率挂断
if (Math.random() < 0.05) {
// 延迟1-3秒后挂断,更自然
setTimeout(() => {
if (callState.isConnected) {
aiHangup();
}
}, 1000 + Math.random() * 2000);
return true;
}
return false;
}
// 检测AI是否有挂断意图
function detectHangupIntent(text) {
if (!text) return false;
// 常见的挂断表达
const hangupPatterns = [
/我(先)?挂了/,
/那我挂了/,
/先挂(了)?啊?/,
/挂了(啊|哈|呀|哦)?$/,
/我(要)?挂(电话|断)了/,
/拜拜.*挂/,
/挂.*拜拜/,
/再见.*挂/,
/不聊了.*挂/,
/不说了.*挂/,
/那就这样.*挂/,
/就这样吧.*挂/
];
return hangupPatterns.some(pattern => pattern.test(text));
}
// AI主动挂断电话
function aiHangup() {
if (!callState.isConnected) return;
console.log('[可乐] AI主动挂断电话');
callState.hungUpByAI = true;
hangupCall();
}
// 开始通话计时
function startCallTimer() {
clearInterval(callState.timerInterval);
callState.timerInterval = setInterval(() => {
if (!callState.isConnected || !callState.startTime) return;
const elapsed = Math.floor((Date.now() - callState.startTime) / 1000);
const minutes = Math.floor(elapsed / 60).toString().padStart(2, '0');
const seconds = (elapsed % 60).toString().padStart(2, '0');
const timeEl = document.getElementById('wechat-voice-call-time');
if (timeEl) {
timeEl.textContent = `${minutes}:${seconds}`;
}
}, 1000);
}
// 挂断电话
export function hangupCall() {
// 清除AI挂断计时器
clearTimeout(callState.aiHangupTimeout);
// 计算通话时长
let durationStr = '00:00';
if (callState.isConnected && callState.startTime) {
const elapsed = Math.floor((Date.now() - callState.startTime) / 1000);
const minutes = Math.floor(elapsed / 60).toString().padStart(2, '0');
const seconds = (elapsed % 60).toString().padStart(2, '0');
durationStr = `${minutes}:${seconds}`;
}
// 添加通话记录到聊天历史
if (callState.contact) {
const settings = getSettings();
const contact = callState.contact;
if (!contact.chatHistory) {
contact.chatHistory = [];
}
const now = new Date();
const timeStr = `${now.getFullYear()}-${(now.getMonth()+1).toString().padStart(2,'0')}-${now.getDate().toString().padStart(2,'0')} ${now.getHours().toString().padStart(2,'0')}:${now.getMinutes().toString().padStart(2,'0')}`;
let callContent;
let lastMessage;
if (callState.isConnected) {
// 已接通的通话
callContent = `[通话记录:${durationStr}]`;
lastMessage = `通话时长 ${durationStr}`;
} else {
// 未接通的通话
if (callState.initiator === 'user') {
if (callState.rejectedByAI) {
// 用户发起,AI拒接
callContent = '[通话记录:对方已拒绝]';
lastMessage = '对方已拒绝';
} else {
// 用户发起,用户取消
callContent = '[通话记录:已取消]';
lastMessage = '已取消';
}
} else if (callState.rejectedByUser) {
// AI发起,用户主动拒绝
callContent = '[通话记录:已拒绝]';
lastMessage = '已拒绝';
} else {
// AI发起,超时未接(对方取消)
callContent = '[通话记录:对方已取消]';
lastMessage = '对方已取消';
}
}
// 通话记录消息
const callRecord = {
role: callState.initiator === 'user' ? 'user' : 'assistant',
content: callContent,
time: timeStr,
timestamp: Date.now(),
isCallRecord: true
};
contact.chatHistory.push(callRecord);
// 通话内容只进"通话历史",不在主聊天界面展示(避免污染主界面/列表预览)
if (callState.messages && callState.messages.length > 0) {
const callStatusForHistory = callState.isConnected
? 'connected'
: (callState.initiator === 'user'
? (callState.rejectedByAI ? 'rejectedByAI' : 'cancelled')
: (callState.rejectedByUser ? 'rejected' : 'timeout'));
contact.callHistory = Array.isArray(contact.callHistory) ? contact.callHistory : [];
contact.callHistory.push({
type: 'voice',
initiator: callState.initiator,
status: callStatusForHistory,
duration: durationStr,
time: timeStr,
timestamp: Date.now(),
messages: callState.messages.map(m => ({ role: m.role, content: m.content }))
});
}
contact.lastMessage = lastMessage;
// 在聊天界面显示通话记录
// 传递状态类型: 'connected' | 'cancelled' | 'rejected' | 'rejectedByAI' | 'timeout'
let callStatus = 'connected';
if (!callState.isConnected) {
if (callState.initiator === 'user') {
callStatus = callState.rejectedByAI ? 'rejectedByAI' : 'cancelled';
} else if (callState.rejectedByUser) {
callStatus = 'rejected';
} else {
callStatus = 'timeout';
}
}
if (currentChatIndex === callState.contactIndex) {
appendCallRecordMessage(callState.initiator === 'user' ? 'user' : 'assistant', callStatus, durationStr, contact);
}
// AI 对通话结束做出反应(所有情况都触发)
triggerCallEndReaction(contact, callStatus, callState.initiator, callState.messages, callState.hungUpByAI);
requestSave();
refreshChatList();
}
callState.isActive = false;
callState.isConnected = false;
callState.startTime = null;
clearInterval(callState.timerInterval);
clearInterval(callState.dotsInterval);
const page = document.getElementById('wechat-voice-call-page');
if (page) {
page.classList.add('hidden');
}
}
// 在聊天界面显示通话记录消息
// status: 'connected' | 'cancelled' | 'rejected' | 'timeout'
function appendCallRecordMessage(role, status, duration, contact) {
const messagesContainer = document.getElementById('wechat-chat-messages');
if (!messagesContainer) return;
const messageDiv = document.createElement('div');
messageDiv.className = `wechat-message ${role === 'user' ? 'self' : ''}`;
const firstChar = contact?.name ? contact.name.charAt(0) : '?';
// 获取用户头像
let userAvatarContent = '我';
try {
const settings = getSettings();
if (settings.userAvatar) {
userAvatarContent = `
`;
}
} catch (e) {}
const avatarContent = role === 'user'
? userAvatarContent
: (contact?.avatar
? `
`
: firstChar);
// 通话记录卡片内容
// 线条电话图标
const phoneIconSVG = ``;
let callRecordHTML;
if (status === 'connected') {
// 已接通:显示通话时长
callRecordHTML = `