|
<!DOCTYPE html> |
|
<html lang="fr"> |
|
<head> |
|
<meta charset="UTF-8"> |
|
<meta name="viewport" content="width=device-width, initial-scale=1.0"> |
|
<title>ChatBot Gemini</title> |
|
<style> |
|
* { |
|
margin: 0; |
|
padding: 0; |
|
box-sizing: border-box; |
|
} |
|
|
|
body { |
|
font-family: 'Google Sans', -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif; |
|
background: linear-gradient(135deg, #667eea 0%, #764ba2 100%); |
|
min-height: 100vh; |
|
display: flex; |
|
align-items: center; |
|
justify-content: center; |
|
} |
|
|
|
.container { |
|
width: 100%; |
|
max-width: 1200px; |
|
height: 100vh; |
|
background: white; |
|
border-radius: 16px; |
|
box-shadow: 0 8px 32px rgba(0, 0, 0, 0.1); |
|
overflow: hidden; |
|
display: flex; |
|
flex-direction: column; |
|
} |
|
|
|
|
|
.login-container { |
|
display: flex; |
|
align-items: center; |
|
justify-content: center; |
|
height: 100vh; |
|
background: white; |
|
} |
|
|
|
.login-form { |
|
background: white; |
|
padding: 48px; |
|
border-radius: 16px; |
|
box-shadow: 0 4px 24px rgba(0, 0, 0, 0.1); |
|
width: 100%; |
|
max-width: 400px; |
|
text-align: center; |
|
} |
|
|
|
.login-form h1 { |
|
color: #1a73e8; |
|
font-size: 32px; |
|
font-weight: 400; |
|
margin-bottom: 8px; |
|
} |
|
|
|
.login-form p { |
|
color: #5f6368; |
|
font-size: 16px; |
|
margin-bottom: 32px; |
|
} |
|
|
|
.input-group { |
|
margin-bottom: 24px; |
|
text-align: left; |
|
} |
|
|
|
.input-group label { |
|
display: block; |
|
color: #202124; |
|
font-size: 14px; |
|
font-weight: 500; |
|
margin-bottom: 8px; |
|
} |
|
|
|
.input-group input { |
|
width: 100%; |
|
padding: 16px; |
|
border: 2px solid #dadce0; |
|
border-radius: 8px; |
|
font-size: 16px; |
|
transition: all 0.2s ease; |
|
background: #fafafa; |
|
} |
|
|
|
.input-group input:focus { |
|
outline: none; |
|
border-color: #1a73e8; |
|
background: white; |
|
box-shadow: 0 0 0 3px rgba(26, 115, 232, 0.1); |
|
} |
|
|
|
.btn-primary { |
|
background: #1a73e8; |
|
color: white; |
|
border: none; |
|
padding: 16px 32px; |
|
border-radius: 8px; |
|
font-size: 16px; |
|
font-weight: 500; |
|
cursor: pointer; |
|
transition: all 0.2s ease; |
|
width: 100%; |
|
} |
|
|
|
.btn-primary:hover { |
|
background: #1557b0; |
|
box-shadow: 0 2px 8px rgba(26, 115, 232, 0.3); |
|
} |
|
|
|
|
|
.chat-container { |
|
display: none; |
|
height: 100vh; |
|
flex-direction: column; |
|
} |
|
|
|
.chat-header { |
|
background: #1a73e8; |
|
color: white; |
|
padding: 16px 24px; |
|
display: flex; |
|
align-items: center; |
|
justify-content: space-between; |
|
box-shadow: 0 2px 8px rgba(0, 0, 0, 0.1); |
|
} |
|
|
|
.header-left { |
|
display: flex; |
|
align-items: center; |
|
gap: 16px; |
|
} |
|
|
|
.header-title { |
|
font-size: 20px; |
|
font-weight: 500; |
|
} |
|
|
|
.header-controls { |
|
display: flex; |
|
align-items: center; |
|
gap: 16px; |
|
} |
|
|
|
.model-select { |
|
background: rgba(255, 255, 255, 0.1); |
|
border: 1px solid rgba(255, 255, 255, 0.2); |
|
color: white; |
|
padding: 8px 12px; |
|
border-radius: 8px; |
|
font-size: 14px; |
|
} |
|
|
|
.model-select option { |
|
background: #1a73e8; |
|
color: white; |
|
} |
|
|
|
.toggle-container { |
|
display: flex; |
|
align-items: center; |
|
gap: 8px; |
|
} |
|
|
|
.toggle { |
|
position: relative; |
|
width: 48px; |
|
height: 24px; |
|
background: rgba(255, 255, 255, 0.2); |
|
border-radius: 12px; |
|
cursor: pointer; |
|
transition: all 0.3s ease; |
|
} |
|
|
|
.toggle.active { |
|
background: #34a853; |
|
} |
|
|
|
.toggle-slider { |
|
position: absolute; |
|
top: 2px; |
|
left: 2px; |
|
width: 20px; |
|
height: 20px; |
|
background: white; |
|
border-radius: 50%; |
|
transition: all 0.3s ease; |
|
} |
|
|
|
.toggle.active .toggle-slider { |
|
transform: translateX(24px); |
|
} |
|
|
|
.logout-btn { |
|
background: rgba(255, 255, 255, 0.1); |
|
border: 1px solid rgba(255, 255, 255, 0.2); |
|
color: white; |
|
padding: 8px 16px; |
|
border-radius: 8px; |
|
cursor: pointer; |
|
font-size: 14px; |
|
transition: all 0.2s ease; |
|
} |
|
|
|
.logout-btn:hover { |
|
background: rgba(255, 255, 255, 0.2); |
|
} |
|
|
|
.chat-messages { |
|
flex: 1; |
|
padding: 24px; |
|
overflow-y: auto; |
|
background: #f8f9fa; |
|
display: flex; |
|
flex-direction: column; |
|
gap: 16px; |
|
} |
|
|
|
.message { |
|
display: flex; |
|
gap: 12px; |
|
max-width: 80%; |
|
animation: slideIn 0.3s ease; |
|
} |
|
|
|
.message.user { |
|
align-self: flex-end; |
|
flex-direction: row-reverse; |
|
} |
|
|
|
.message-avatar { |
|
width: 40px; |
|
height: 40px; |
|
border-radius: 50%; |
|
display: flex; |
|
align-items: center; |
|
justify-content: center; |
|
font-weight: 500; |
|
font-size: 14px; |
|
flex-shrink: 0; |
|
} |
|
|
|
.message.user .message-avatar { |
|
background: #1a73e8; |
|
color: white; |
|
} |
|
|
|
.message.assistant .message-avatar { |
|
background: #34a853; |
|
color: white; |
|
} |
|
|
|
.message-content { |
|
background: white; |
|
padding: 16px; |
|
border-radius: 16px; |
|
box-shadow: 0 2px 8px rgba(0, 0, 0, 0.1); |
|
line-height: 1.5; |
|
word-wrap: break-word; |
|
} |
|
|
|
.message.user .message-content { |
|
background: #1a73e8; |
|
color: white; |
|
} |
|
|
|
.chat-input-container { |
|
padding: 24px; |
|
background: white; |
|
border-top: 1px solid #e0e0e0; |
|
} |
|
|
|
.chat-input-wrapper { |
|
display: flex; |
|
gap: 12px; |
|
align-items: flex-end; |
|
} |
|
|
|
.chat-input { |
|
flex: 1; |
|
padding: 16px; |
|
border: 2px solid #dadce0; |
|
border-radius: 24px; |
|
font-size: 16px; |
|
resize: none; |
|
min-height: 24px; |
|
max-height: 120px; |
|
transition: all 0.2s ease; |
|
} |
|
|
|
.chat-input:focus { |
|
outline: none; |
|
border-color: #1a73e8; |
|
box-shadow: 0 0 0 3px rgba(26, 115, 232, 0.1); |
|
} |
|
|
|
.send-btn { |
|
background: #1a73e8; |
|
color: white; |
|
border: none; |
|
width: 48px; |
|
height: 48px; |
|
border-radius: 50%; |
|
cursor: pointer; |
|
display: flex; |
|
align-items: center; |
|
justify-content: center; |
|
transition: all 0.2s ease; |
|
flex-shrink: 0; |
|
} |
|
|
|
.send-btn:hover { |
|
background: #1557b0; |
|
transform: scale(1.05); |
|
} |
|
|
|
.send-btn:disabled { |
|
background: #dadce0; |
|
cursor: not-allowed; |
|
transform: none; |
|
} |
|
|
|
.typing-indicator { |
|
display: none; |
|
align-items: center; |
|
gap: 12px; |
|
max-width: 80%; |
|
animation: slideIn 0.3s ease; |
|
} |
|
|
|
.typing-dots { |
|
background: white; |
|
padding: 16px; |
|
border-radius: 16px; |
|
box-shadow: 0 2px 8px rgba(0, 0, 0, 0.1); |
|
display: flex; |
|
gap: 4px; |
|
} |
|
|
|
.typing-dot { |
|
width: 8px; |
|
height: 8px; |
|
border-radius: 50%; |
|
background: #dadce0; |
|
animation: typing 1.4s infinite ease-in-out; |
|
} |
|
|
|
.typing-dot:nth-child(2) { |
|
animation-delay: 0.2s; |
|
} |
|
|
|
.typing-dot:nth-child(3) { |
|
animation-delay: 0.4s; |
|
} |
|
|
|
@keyframes slideIn { |
|
from { |
|
opacity: 0; |
|
transform: translateY(20px); |
|
} |
|
to { |
|
opacity: 1; |
|
transform: translateY(0); |
|
} |
|
} |
|
|
|
@keyframes typing { |
|
0%, 60%, 100% { |
|
transform: scale(1); |
|
background: #dadce0; |
|
} |
|
30% { |
|
transform: scale(1.2); |
|
background: #1a73e8; |
|
} |
|
} |
|
|
|
.error-message { |
|
background: #fce8e6; |
|
color: #d93025; |
|
padding: 12px 16px; |
|
border-radius: 8px; |
|
margin-bottom: 16px; |
|
border-left: 4px solid #d93025; |
|
} |
|
|
|
@media (max-width: 768px) { |
|
.container { |
|
border-radius: 0; |
|
height: 100vh; |
|
} |
|
|
|
.login-form { |
|
padding: 32px 24px; |
|
margin: 16px; |
|
} |
|
|
|
.header-controls { |
|
flex-direction: column; |
|
gap: 8px; |
|
align-items: flex-end; |
|
} |
|
|
|
.message { |
|
max-width: 90%; |
|
} |
|
|
|
.chat-input-container { |
|
padding: 16px; |
|
} |
|
} |
|
</style> |
|
</head> |
|
<body> |
|
<div class="container"> |
|
|
|
<div id="loginPage" class="login-container"> |
|
<div class="login-form"> |
|
<h1>ChatBot Gemini</h1> |
|
<p>Login with your API key</p> |
|
<div id="loginError"></div> |
|
<form id="loginForm"> |
|
<div class="input-group"> |
|
<label for="apiKey">API Key</label> |
|
<input type="password" id="apiKey" placeholder="Votre clé API..." required> |
|
</div> |
|
<button type="submit" class="btn-primary">Log In</button> |
|
</form> |
|
</div> |
|
</div> |
|
|
|
|
|
<div id="chatPage" class="chat-container"> |
|
<div class="chat-header"> |
|
<div class="header-left"> |
|
<div class="header-title">ChatBot Gemini</div> |
|
</div> |
|
<div class="header-controls"> |
|
<select id="modelSelect" class="model-select"> |
|
|
|
</select> |
|
<div class="toggle-container"> |
|
<span style="font-size: 14px;">Tools</span> |
|
<div id="toolToggle" class="toggle active"> |
|
<div class="toggle-slider"></div> |
|
</div> |
|
</div> |
|
<button id="logoutBtn" class="logout-btn">Log Out</button> |
|
</div> |
|
</div> |
|
|
|
<div id="chatMessages" class="chat-messages"> |
|
<div class="message assistant"> |
|
<div class="message-avatar">AI</div> |
|
<div class="message-content"> |
|
Hi ! I'm your AI assistant. How can I help you today ? |
|
</div> |
|
</div> |
|
</div> |
|
|
|
<div class="typing-indicator" id="typingIndicator"> |
|
<div class="message-avatar" style="background: #34a853; color: white;">AI</div> |
|
<div class="typing-dots"> |
|
<div class="typing-dot"></div> |
|
<div class="typing-dot"></div> |
|
<div class="typing-dot"></div> |
|
</div> |
|
</div> |
|
|
|
<div class="chat-input-container"> |
|
<div class="chat-input-wrapper"> |
|
<textarea id="chatInput" class="chat-input" placeholder="Tapez votre message..." rows="1"></textarea> |
|
<button id="sendBtn" class="send-btn"> |
|
<svg width="24" height="24" viewBox="0 0 24 24" fill="currentColor"> |
|
<path d="M2.01 21L23 12 2.01 3 2 10l15 2-15 2z"/> |
|
</svg> |
|
</button> |
|
</div> |
|
</div> |
|
</div> |
|
</div> |
|
|
|
<script> |
|
class ChatBot { |
|
constructor() { |
|
this.sessionId = ''; |
|
this.currentModel = 'llama-3.3-70b-versatile'; |
|
this.toolsEnabled = true; |
|
this.isTyping = false; |
|
this.availableModels = []; |
|
this.baseUrl = window.location.origin; |
|
|
|
this.initializeElements(); |
|
this.bindEvents(); |
|
this.loadFromStorage(); |
|
} |
|
|
|
initializeElements() { |
|
|
|
this.loginPage = document.getElementById('loginPage'); |
|
this.loginForm = document.getElementById('loginForm'); |
|
this.apiKeyInput = document.getElementById('apiKey'); |
|
this.loginError = document.getElementById('loginError'); |
|
|
|
|
|
this.chatPage = document.getElementById('chatPage'); |
|
this.chatMessages = document.getElementById('chatMessages'); |
|
this.chatInput = document.getElementById('chatInput'); |
|
this.sendBtn = document.getElementById('sendBtn'); |
|
this.modelSelect = document.getElementById('modelSelect'); |
|
this.toolToggle = document.getElementById('toolToggle'); |
|
this.logoutBtn = document.getElementById('logoutBtn'); |
|
this.typingIndicator = document.getElementById('typingIndicator'); |
|
} |
|
|
|
bindEvents() { |
|
|
|
this.loginForm.addEventListener('submit', (e) => this.handleLogin(e)); |
|
|
|
|
|
this.sendBtn.addEventListener('click', () => this.sendMessage()); |
|
this.chatInput.addEventListener('keypress', (e) => { |
|
if (e.key === 'Enter' && !e.shiftKey) { |
|
e.preventDefault(); |
|
this.sendMessage(); |
|
} |
|
}); |
|
|
|
|
|
this.chatInput.addEventListener('input', () => { |
|
this.chatInput.style.height = 'auto'; |
|
this.chatInput.style.height = Math.min(this.chatInput.scrollHeight, 120) + 'px'; |
|
}); |
|
|
|
|
|
this.modelSelect.addEventListener('change', (e) => { |
|
this.currentModel = e.target.value; |
|
this.saveToStorage(); |
|
}); |
|
|
|
this.toolToggle.addEventListener('click', () => { |
|
this.toolsEnabled = !this.toolsEnabled; |
|
this.toolToggle.classList.toggle('active', this.toolsEnabled); |
|
this.saveToStorage(); |
|
}); |
|
|
|
this.logoutBtn.addEventListener('click', () => this.logout()); |
|
} |
|
|
|
async handleLogin(e) { |
|
e.preventDefault(); |
|
const apiKey = this.apiKeyInput.value.trim(); |
|
|
|
if (!apiKey) { |
|
this.showLoginError('Please enter a valid API key'); |
|
return; |
|
} |
|
|
|
try { |
|
const response = await fetch(`${this.baseUrl}/init`, { |
|
method: 'POST', |
|
headers: { |
|
'Content-Type': 'application/json' |
|
}, |
|
body: JSON.stringify({ |
|
api_key: apiKey |
|
}) |
|
}); |
|
|
|
const data = await response.json(); |
|
|
|
if (data.success) { |
|
this.sessionId = data.session_id; |
|
this.availableModels = data.models || []; |
|
this.populateModelSelect(); |
|
this.saveToStorage(); |
|
this.showChatInterface(); |
|
this.loginError.innerHTML = ''; |
|
} else { |
|
this.showLoginError(data.error || 'Connection error.'); |
|
} |
|
} catch (error) { |
|
this.showLoginError('Server connection error. Please retry.'); |
|
console.error('Login error:', error); |
|
} |
|
} |
|
|
|
populateModelSelect() { |
|
this.modelSelect.innerHTML = ''; |
|
this.availableModels.forEach(model => { |
|
const option = document.createElement('option'); |
|
option.value = model; |
|
option.textContent = model; |
|
this.modelSelect.appendChild(option); |
|
}); |
|
|
|
|
|
if (this.availableModels.includes(this.currentModel)) { |
|
this.modelSelect.value = this.currentModel; |
|
} else if (this.availableModels.length > 0) { |
|
this.currentModel = this.availableModels[0]; |
|
this.modelSelect.value = this.currentModel; |
|
} |
|
} |
|
|
|
showLoginError(message) { |
|
this.loginError.innerHTML = `<div class="error-message">${message}</div>`; |
|
} |
|
|
|
showChatInterface() { |
|
this.loginPage.style.display = 'none'; |
|
this.chatPage.style.display = 'flex'; |
|
this.chatInput.focus(); |
|
} |
|
|
|
async logout() { |
|
try { |
|
if (this.sessionId) { |
|
await fetch(`${this.baseUrl}/logout`, { |
|
method: 'POST', |
|
headers: { |
|
'Content-Type': 'application/json' |
|
}, |
|
body: JSON.stringify({ |
|
session_id: this.sessionId |
|
}) |
|
}); |
|
} |
|
} catch (error) { |
|
console.error('Logout error:', error); |
|
} |
|
|
|
this.sessionId = ''; |
|
this.availableModels = []; |
|
localStorage.removeItem('chatbot_data'); |
|
this.chatPage.style.display = 'none'; |
|
this.loginPage.style.display = 'flex'; |
|
this.apiKeyInput.value = ''; |
|
this.clearMessages(); |
|
} |
|
|
|
async sendMessage() { |
|
const message = this.chatInput.value.trim(); |
|
if (!message || this.isTyping || !this.sessionId) return; |
|
|
|
|
|
this.addMessage('user', message); |
|
this.chatInput.value = ''; |
|
this.chatInput.style.height = 'auto'; |
|
|
|
|
|
this.showTyping(true); |
|
|
|
try { |
|
const response = await fetch(`${this.baseUrl}/chat`, { |
|
method: 'POST', |
|
headers: { |
|
'Content-Type': 'application/json' |
|
}, |
|
body: JSON.stringify({ |
|
session_id: this.sessionId, |
|
query: message, |
|
tool_use: this.toolsEnabled, |
|
model: this.currentModel |
|
}) |
|
}); |
|
|
|
const data = await response.json(); |
|
this.showTyping(false); |
|
|
|
if (data.error) { |
|
this.addMessage('assistant', `Erreur: ${data.error}`); |
|
} else { |
|
this.addMessage('assistant', data.output); |
|
} |
|
} catch (error) { |
|
this.showTyping(false); |
|
this.addMessage('assistant', `Erreur de connexion: ${error.message}`); |
|
console.error('Chat error:', error); |
|
} |
|
|
|
this.saveToStorage(); |
|
} |
|
|
|
addMessage(role, content) { |
|
const messageDiv = document.createElement('div'); |
|
messageDiv.className = `message ${role}`; |
|
|
|
const avatar = document.createElement('div'); |
|
avatar.className = 'message-avatar'; |
|
avatar.textContent = role === 'user' ? 'U' : 'AI'; |
|
|
|
const messageContent = document.createElement('div'); |
|
messageContent.className = 'message-content'; |
|
messageContent.textContent = content; |
|
|
|
messageDiv.appendChild(avatar); |
|
messageDiv.appendChild(messageContent); |
|
|
|
|
|
if (this.typingIndicator && this.typingIndicator.parentNode === this.chatMessages) { |
|
this.chatMessages.insertBefore(messageDiv, this.typingIndicator); |
|
} else { |
|
|
|
this.chatMessages.appendChild(messageDiv); |
|
} |
|
|
|
this.scrollToBottom(); |
|
} |
|
|
|
showTyping(show) { |
|
this.isTyping = show; |
|
this.typingIndicator.style.display = show ? 'flex' : 'none'; |
|
this.sendBtn.disabled = show; |
|
this.scrollToBottom(); |
|
} |
|
|
|
scrollToBottom() { |
|
setTimeout(() => { |
|
this.chatMessages.scrollTop = this.chatMessages.scrollHeight; |
|
}, 100); |
|
} |
|
|
|
clearMessages() { |
|
const messages = this.chatMessages.querySelectorAll('.message'); |
|
messages.forEach((msg, index) => { |
|
if (index > 0) { |
|
msg.remove(); |
|
} |
|
}); |
|
} |
|
|
|
saveToStorage() { |
|
const data = { |
|
sessionId: this.sessionId, |
|
currentModel: this.currentModel, |
|
toolsEnabled: this.toolsEnabled, |
|
availableModels: this.availableModels |
|
}; |
|
localStorage.setItem('chatbot_data', JSON.stringify(data)); |
|
} |
|
|
|
loadFromStorage() { |
|
const saved = localStorage.getItem('chatbot_data'); |
|
if (saved) { |
|
try { |
|
const data = JSON.parse(saved); |
|
if (data.sessionId) { |
|
this.sessionId = data.sessionId; |
|
this.currentModel = data.currentModel || 'models/gemini-2.0-flash'; |
|
this.toolsEnabled = data.toolsEnabled !== undefined ? data.toolsEnabled : true; |
|
this.availableModels = data.availableModels || []; |
|
|
|
|
|
this.populateModelSelect(); |
|
this.toolToggle.classList.toggle('active', this.toolsEnabled); |
|
|
|
|
|
this.verifySession(); |
|
} |
|
} catch (error) { |
|
console.error('Error loading from storage:', error); |
|
localStorage.removeItem('chatbot_data'); |
|
} |
|
} |
|
} |
|
|
|
async verifySession() { |
|
try { |
|
|
|
const response = await fetch(`${this.baseUrl}/chat`, { |
|
method: 'POST', |
|
headers: { |
|
'Content-Type': 'application/json' |
|
}, |
|
body: JSON.stringify({ |
|
session_id: this.sessionId, |
|
query: 'test', |
|
tool_use: false, |
|
model: this.currentModel |
|
}) |
|
}); |
|
|
|
if (response.ok) { |
|
this.showChatInterface(); |
|
} else { |
|
|
|
localStorage.removeItem('chatbot_data'); |
|
this.sessionId = ''; |
|
} |
|
} catch (error) { |
|
|
|
localStorage.removeItem('chatbot_data'); |
|
this.sessionId = ''; |
|
} |
|
} |
|
} |
|
|
|
|
|
document.addEventListener('DOMContentLoaded', () => { |
|
new ChatBot(); |
|
}); |
|
</script> |
|
</body> |
|
</html> |
|
|