|
|
|
import { updateLastMessage, autoResizeTextarea } from './utils.js'; |
|
import { renderCurrentSession, sessions, currentSessionIndex } from './sessions.js'; |
|
import { callLLMStream, callLLMSummaryBatch } from './api.js'; |
|
|
|
export let attachedFiles = []; |
|
|
|
|
|
|
|
|
|
export async function fileToBase64(file) { |
|
return new Promise((resolve, reject) => { |
|
const reader = new FileReader(); |
|
reader.onload = () => { |
|
const base64 = reader.result.split(',')[1]; |
|
resolve({ |
|
name: file.name, |
|
path: file.webkitRelativePath || file.path || file.name, |
|
size: file.size, |
|
type: file.type, |
|
content: base64, |
|
}); |
|
}; |
|
reader.onerror = reject; |
|
reader.readAsDataURL(file); |
|
}); |
|
} |
|
|
|
|
|
|
|
|
|
export function updateFileAttachments() { |
|
const fileAttachments = document.getElementById('fileAttachments'); |
|
fileAttachments.innerHTML = ""; |
|
attachedFiles.forEach((file, index) => { |
|
const fileDiv = document.createElement("div"); |
|
fileDiv.className = "file-item"; |
|
fileDiv.innerHTML = `<span>${file.name}</span> <button data-index="${index}">×</button>`; |
|
fileAttachments.appendChild(fileDiv); |
|
}); |
|
document.querySelectorAll(".file-item button").forEach(btn => { |
|
btn.addEventListener("click", (e) => { |
|
const idx = e.target.getAttribute("data-index"); |
|
attachedFiles.splice(idx, 1); |
|
updateFileAttachments(); |
|
}); |
|
}); |
|
} |
|
|
|
|
|
|
|
|
|
export async function addConversation(userText) { |
|
if (userText.trim() === '' && attachedFiles.length === 0) return; |
|
const session = sessions[currentSessionIndex]; |
|
session.messages.push({ |
|
userText, |
|
aiResponse: "", |
|
attachments: await Promise.all(attachedFiles.map(fileToBase64)), |
|
model: session.settings.model, |
|
timestamp: new Date().toISOString(), |
|
maxTokens: session.settings.maxTokens, |
|
temperature: session.settings.temperature, |
|
persona: session.settings.persona, |
|
sessionId: session.id, |
|
}); |
|
|
|
|
|
attachedFiles.length = 0; |
|
updateFileAttachments(); |
|
renderCurrentSession(); |
|
|
|
|
|
const conversation = []; |
|
session.messages.forEach(msg => { |
|
conversation.push({ role: "user", content: msg.userText, attachments: msg.attachments, sessionId: msg.sessionId }); |
|
if (msg.aiResponse) { |
|
conversation.push({ role: "assistant", content: msg.aiResponse, sessionId: msg.sessionId }); |
|
} |
|
}); |
|
|
|
window.currentStreamController = new AbortController(); |
|
window.isStreaming = true; |
|
const sendBtn = document.getElementById('sendBtn'); |
|
sendBtn.innerHTML = `<img src="assets/stop.svg" alt="Stop Icon" class="svg-icon-non-white">`; |
|
|
|
try { |
|
const aiResponse = await callLLMStream(conversation, window.currentStreamController.signal); |
|
session.messages[session.messages.length - 1].aiResponse = aiResponse; |
|
renderCurrentSession(); |
|
if (session.settings.enableSummarization) { |
|
await callLLMSummaryBatch( |
|
session.id, |
|
session.messages, |
|
session.settings.model, |
|
session.settings.temperature, |
|
session.settings.maxTokens |
|
); |
|
} |
|
} catch (err) { |
|
if (err.name === 'AbortError') { |
|
console.log('Streaming aborted by user.'); |
|
} else { |
|
console.error(err); |
|
session.messages[session.messages.length - 1].aiResponse = "Error: " + err.message; |
|
renderCurrentSession(); |
|
} |
|
} finally { |
|
sendBtn.innerHTML = `<img src="assets/send.svg" alt="Send Icon" class="svg-icon-non-white">`; |
|
window.isStreaming = false; |
|
} |
|
} |
|
|