interface / index.html
joermd's picture
Update index.html
f0472c6 verified
raw
history blame
28.3 kB
<!DOCTYPE html>
<html lang="ar" dir="rtl">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Speedy Chat</title>
<link href="https://cdn.jsdelivr.net/npm/[email protected]/dist/tailwind.min.css" rel="stylesheet">
<link href="https://cdnjs.cloudflare.com/ajax/libs/prism/1.24.1/themes/prism.min.css" rel="stylesheet">
<script src="https://cdnjs.cloudflare.com/ajax/libs/prism/1.24.1/prism.min.js"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/prism/1.24.1/components/prism-python.min.js"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/prism/1.24.1/components/prism-javascript.min.js"></script>
<style>
@import url('https://fonts.googleapis.com/css2?family=IBM+Plex+Sans+Arabic:wght@100;200;300;400;500;600;700&display=swap');
@import url('https://fonts.googleapis.com/css2?family=Noto+Kufi+Arabic:wght@100;200;300;400;500;600;700;800;900&display=swap');
body {
font-family: 'IBM Plex Sans Arabic', sans-serif;
line-height: 1.6;
letter-spacing: -0.01em;
transition: background-color 0.3s, color 0.3s;
min-height: 100vh;
display: flex;
flex-direction: column;
}
body.light-mode {
background-color: #fafafa;
color: #374151;
}
body.dark-mode {
background-color: #1a1a1a;
color: #e5e5e5;
}
.dark-mode header {
background-color: #2d2d2d;
border-color: #404040;
}
.dark-mode .message-input {
background-color: #363636;
border-color: #404040;
color: #e5e5e5;
}
.dark-mode .style-select {
background-color: #363636;
border-color: #404040;
color: #e5e5e5;
}
.dark-mode .bot-message {
background-color: #363636;
color: #e5e5e5;
}
.dark-mode .action-button {
color: #e5e5e5;
background-color: #404040;
}
.dark-mode .modal-content {
background-color: #2d2d2d;
color: #e5e5e5;
}
.dark-mode .modal-content textarea {
background-color: #363636;
color: #e5e5e5;
border-color: #404040;
}
.welcome-text {
font-family: 'Noto Kufi Arabic', sans-serif;
font-size: 2rem;
font-weight: 700;
text-align: center;
margin: 1.5rem 0;
letter-spacing: -0.02em;
}
main {
flex: 1;
display: flex;
flex-direction: column;
padding-top: 4rem;
}
.messages-container {
flex: 1;
overflow-y: auto;
padding: 1rem;
}
.input-container {
margin-top: auto;
padding: 1rem;
background-color: transparent;
}
.message {
opacity: 0;
transform: translateY(20px);
animation: fadeIn 0.3s ease forwards;
}
.toggle-switch {
width: 32px;
height: 16px;
background-color: #e5e7eb;
border-radius: 999px;
position: relative;
cursor: pointer;
transition: background-color 0.3s;
}
.toggle-switch::after {
content: '';
position: absolute;
top: 2px;
right: 2px;
width: 12px;
height: 12px;
background-color: white;
border-radius: 50%;
transition: transform 0.3s;
}
.toggle-switch.active {
background-color: #3b82f6;
}
.toggle-switch.active::after {
transform: translateX(-16px);
}
.action-button {
padding: 0.25rem 0.5rem;
border-radius: 0.375rem;
transition: all 0.2s;
display: inline-flex;
align-items: center;
gap: 0.25rem;
font-size: 0.7rem;
color: #6B7280;
background-color: #F3F4F6;
}
.action-button:hover {
background-color: #E5E7EB;
}
.thinking-steps {
margin: 10px 0;
padding: 10px;
background-color: #f8f9fa;
border-radius: 8px;
font-size: 0.9em;
}
.chat-input {
max-height: 60px;
min-height: 32px;
}
@keyframes fadeIn {
to {
opacity: 1;
transform: translateY(0);
}
}
.message-actions {
margin-top: 0.5rem;
display: flex;
gap: 0.5rem;
}
/* Modal styles */
.modal {
display: none;
position: fixed;
top: 0;
left: 0;
right: 0;
bottom: 0;
background-color: rgba(0, 0, 0, 0.5);
align-items: center;
justify-content: center;
z-index: 1000;
}
.modal-content {
background-color: white;
padding: 2rem;
border-radius: 0.5rem;
width: 90%;
max-width: 500px;
}
.modal textarea {
width: 100%;
min-height: 200px;
padding: 0.5rem;
margin: 1rem 0;
border: 1px solid #e5e7eb;
border-radius: 0.375rem;
}
</style>
</head>
<body class="text-lg light-mode">
<header class="fixed top-0 left-0 right-0 bg-white border-b border-gray-100 z-50 shadow-sm">
<div class="flex items-center px-4 py-2">
<div class="flex items-center flex-1">
<span class="text-2xl font-bold text-indigo-600">Speedy</span>
</div>
<div class="flex items-center gap-2">
<a href="/home" class="header-button">
<svg class="w-6 h-6" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M3 12l2-2m0 0l7-7 7 7M5 10v10a1 1 0 001 1h3m10-11l2 2m-2-2v10a1 1 0 01-1 1h-3m-6 0a1 1 0 001-1v-4a1 1 0 011-1h2a1 1 0 011 1v4a1 1 0 001 1m-6 0h6"></path>
</svg>
</a>
<button id="darkModeToggle" class="header-button">
<svg class="w-6 h-6" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M20.354 15.354A9 9 0 018.646 3.646 9.003 9.003 0 0012 21a9.003 9.003 0 008.354-5.646z"></path>
</svg>
</button>
<button id="clearChat" class="header-button">
<svg class="w-6 h-6" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M19 7l-.867 12.142A2 2 0 0116.138 21H7.862a2 2 0 01-1.995-1.858L5 7m5 4v6m4-6v6m1-10V4a1 1 0 00-1-1h-4a1 1 0 00-1 1v3M4 7h16"></path>
</svg>
</button>
</div>
</div>
</header>
<main>
<div class="max-w-3xl mx-auto px-4 w-full flex flex-col flex-1">
<div class="welcome-text">ุณุจูŠุฏูŠ ู‡ู†ุง ู„ู…ุณุงุนุฏุชูƒ</div>
<div id="messagesContainer" class="messages-container space-y-4 flex-1"></div>
<div class="input-container">
<div class="bg-white rounded-xl border border-gray-200 p-2">
<textarea
id="messageInput"
class="chat-input w-full text-gray-600 text-lg p-2 rounded-lg resize-none border-none"
placeholder="ุงูƒุชุจ ุฑุณุงู„ุชูƒ ู‡ู†ุง..."
rows="1"
></textarea>
<div class="flex items-center justify-between mt-2">
<div class="flex gap-3">
<button class="text-gray-400 hover:text-gray-600" id="attachButton">
<svg class="w-5 h-5" viewBox="0 0 24 24" fill="none" stroke="currentColor">
<path d="M21.44 11.05l-9.19 9.19a6 6 0 01-8.49-8.49l9.19-9.19a4 4 0 015.66 5.66l-9.2 9.19a2 2 0 01-2.83-2.83l8.49-8.48"/>
</svg>
</button>
<a href="https://joermd-test11.static.hf.space" target="_blank" id="micButton" class="text-gray-400 hover:text-gray-600">
<svg class="w-5 h-5" viewBox="0 0 24 24" fill="none" stroke="currentColor">
<path d="M19 11a7 7 0 01-7 7m0 0a7 7 0 01-7-7m7 7v4m0 0H8m4 0h4m-4-8a3 3 0 01-3-3V5a3 3 0 116 0v6a3 3 0 01-3 3z"/>
</svg>
</a>
<button id="searchButton" class="text-gray-400 hover:text-gray-600">
<svg class="w-5 h-5" viewBox="0 0 24 24" fill="none" stroke="currentColor">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M21 21l-6-6m2-5a7 7 0 11-14 0 7 7 0 0114 0z"/>
</svg>
</button>
</div>
<div class="flex items-center gap-3">
<select id="styleSelect" class="style-select bg-gray-50 border border-gray-200 text-gray-600 text-sm rounded-lg pl-8 pr-2 py-2">
<option value="short">โšก ุฑุฏูˆุฏ ู‚ุตูŠุฑุฉ</option>
<option value="normal" selected>โ—ฏ ุนุงุฏูŠ</option>
<option value="long">โ†” ุฑุฏูˆุฏ ู…ูุตู„ุฉ</option>
</select>
<div class="flex items-center gap-2 bg-gray-50 px-2 py-1 rounded-lg">
<span class="text-gray-600">ุฐูƒุงุก ู…ุชู‚ุฏู…</span>
<div class="toggle-switch" id="aiToggle"></div>
</div>
<button id="sendMessage" class="bg-black text-white px-3 py-1.5 rounded-lg flex items-center gap-2">
ุฅุฑุณุงู„
<svg class="w-4 h-4 rotate-90" viewBox="0 0 24 24" fill="none" stroke="currentColor">
<path d="M12 19V5M5 12l7-7 7 7"/>
</svg>
</button>
</div>
</div>
</div>
</div>
</div>
</main>
<!-- Edit Modal -->
<div id="editModal" class="modal">
<div class="modal-content">
<h2 class="text-lg font-bold mb-2">ุชุนุฏูŠู„ ุงู„ุฑุณุงู„ุฉ</h2>
<textarea id="editText"></textarea>
<div class="flex justify-end gap-2">
<button id="cancelEdit" class="px-4 py-2 bg-gray-200 rounded-lg">ุฅู„ุบุงุก</button>
<button id="saveEdit" class="px-4 py-2 bg-black text-white rounded-lg">ุญูุธ</button>
</div>
</div>
</div>
<script>
const API_URL = 'https://975yry7ibyersz-7777.proxy.runpod.net/proxy/8000/chat';
const messagesContainer = document.getElementById('messagesContainer');
const messageInput = document.getElementById('messageInput');
const sendButton = document.getElementById('sendMessage');
const clearButton = document.getElementById('clearChat');
const styleSelect = document.getElementById('styleSelect');
const darkModeToggle = document.getElementById('darkModeToggle');
const aiToggle = document.getElementById('aiToggle');
const attachButton = document.getElementById('attachButton');
const searchButton = document.getElementById('searchButton');
const editModal = document.getElementById('editModal');
const editText = document.getElementById('editText');
const saveEdit = document.getElementById('saveEdit');
const cancelEdit = document.getElementById('cancelEdit');
let chatHistory = [];
let currentStyle = 'normal';
let currentController = null;
let isSearchEnabled = false;
let currentEditMessageId = null;
let uploadedFiles = [];
async function handleFileUpload(file) {
const formData = new FormData();
formData.append('file', file);
try {
const response = await fetch(`${API_URL}/upload`, {
method: 'POST',
body: formData
});
const data = await response.json();
uploadedFiles.push(data.filename);
return data.filename;
} catch (error) {
console.error('Error uploading file:', error);
return null;
}
}
function formatCode(text) {
const codeBlockRegex = /```(\w+)?\n([\s\S]*?)```/g;
return text.replace(codeBlockRegex, (match, lang, code) => {
const language = lang || 'plaintext';
return `<pre><code class="language-${language}">${code.trim()}</code></pre>`;
});
}
function createThinkingSteps(messageId) {
return `
<div id="${messageId}-thinking" class="thinking-steps">
<div>1. ุชุญู„ูŠู„ ุงู„ุณุคุงู„ ูˆูู‡ู… ุงู„ู…ุทู„ูˆุจ...</div>
<div>2. ุฌู…ุน ุงู„ู…ุนู„ูˆู…ุงุช ุฐุงุช ุงู„ุตู„ุฉ...</div>
<div>3. ุตูŠุงุบุฉ ุฅุฌุงุจุฉ ุดุงู…ู„ุฉ ูˆุฏู‚ูŠู‚ุฉ...</div>
</div>
`;
}
function createUserMessage(text) {
return `
<div class="message flex justify-end mb-4">
<div class="max-w-[80%]">
<div class="bg-black text-white rounded-lg p-3 shadow-sm">
<p class="text-sm">${text}</p>
</div>
</div>
</div>
`;
}
function createBotMessage(text, messageId) {
return `
<div class="message flex justify-start mb-4">
<div class="flex-shrink-0 mt-1">
<img src="https://ufastpro.com/wp-content/uploads/2024/12/3.png" alt="Bot Avatar" class="w-8 h-8 rounded-full">
</div>
<div class="max-w-[80%] mr-3">
<div class="bot-message bg-white rounded-lg p-3 shadow-sm">
<p id="${messageId}" class="text-sm"></p>
<div class="message-actions">
<button class="action-button copy-button" data-message-id="${messageId}">
<svg class="w-3 h-3" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M8 5H6a2 2 0 00-2 2v12a2 2 0 002 2h10a2 2 0 002-2v-1M8 5a2 2 0 002 2h2a2 2 0 002-2M8 5a2 2 0 012-2h2a2 2 0 012 2"></path>
</svg>
ู†ุณุฎ
</button>
<button class="action-button speak-button" data-message-id="${messageId}">
<svg class="w-3 h-3" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M19 11a7 7 0 01-7 7m0 0a7 7 0 01-7-7m7 7v4m0 0H8m4 0h4m-4-8a3 3 0 01-3-3V5a3 3 0 116 0v6a3 3 0 01-3 3z"></path>
</svg>
ู‚ุฑุงุกุฉ
</button>
<button class="action-button edit-button" data-message-id="${messageId}">
<svg class="w-3 h-3" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M11 5H6a2 2 0 00-2 2v11a2 2 0 002 2h11a2 2 0 002-2v-5m-1.414-9.414a2 2 0 112.828 2.828L11.828 15H9v-2.828l8.586-8.586z"></path>
</svg>
ุชุนุฏูŠู„
</button>
</div>
</div>
</div>
</div>
`;
}
async function typeText(elementId, text) {
const element = document.getElementById(elementId);
element.innerHTML = formatCode(text);
Prism.highlightAllUnder(element);
}
function speakText(text) {
const utterance = new SpeechSynthesisUtterance(text);
utterance.lang = 'ar';
speechSynthesis.speak(utterance);
}
function openEditModal(messageId) {
const messageElement = document.getElementById(messageId);
editText.value = messageElement.textContent;
editModal.style.display = 'flex';
currentEditMessageId = messageId;
}
function closeEditModal() {
editModal.style.display = 'none';
currentEditMessageId = null;
}
function toggleSendButton(isGenerating) {
if (isGenerating) {
sendButton.classList.add('stop-generation');
sendButton.innerHTML = `
<svg class="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M6 18L18 6M6 6l12 12"></path>
</svg>
ุฅูŠู‚ุงู
`;
} else {
sendButton.classList.remove('stop-generation');
sendButton.innerHTML = `
ุฅุฑุณุงู„
<svg class="w-4 h-4 rotate-90" viewBox="0 0 24 24" fill="none" stroke="currentColor">
<path d="M12 19V5M5 12l7-7 7 7"/>
</svg>
`;
}
}
async function sendMessage() {
const message = messageInput.value.trim();
if (!message) return;
if (currentController) {
currentController.abort();
currentController = null;
toggleSendButton(false);
return;
}
messageInput.value = '';
adjustTextareaHeight();
const messageId = 'msg-' + Date.now();
let actualMessage = message;
if (currentStyle === 'short') {
actualMessage = 'ุงุนุทู†ูŠ ุฑุฏ ุจุงุฎุชุตุงุฑ ูˆุณุฑุนุฉ: ' + message;
} else if (currentStyle === 'long') {
actualMessage = 'ุงุนุทู†ูŠ ุฑุฏ ู…ูุตู„ ูˆู…ูˆุณุน: ' + message;
}
messagesContainer.insertAdjacentHTML('beforeend', createUserMessage(message));
messagesContainer.insertAdjacentHTML('beforeend', createBotMessage('', messageId));
if (aiToggle.classList.contains('active')) {
document.getElementById(messageId).insertAdjacentHTML('beforebegin', createThinkingSteps(messageId));
}
scrollToBottom();
try {
currentController = new AbortController();
toggleSendButton(true);
const requestData = {
message: actualMessage,
history: chatHistory,
advanced: aiToggle.classList.contains('active'),
search: isSearchEnabled
};
if (uploadedFiles.length > 0) {
requestData.files = uploadedFiles;
}
const response = await fetch(API_URL, {
method: 'POST',
headers: { 'Content-Type': 'application/json' },
body: JSON.stringify(requestData),
signal: currentController.signal
});
const data = await response.json();
const thinkingSteps = document.getElementById(`${messageId}-thinking`);
if (thinkingSteps) {
thinkingSteps.remove();
}
await typeText(messageId, data.response);
chatHistory.push({
human: actualMessage,
assistant: data.response
});
uploadedFiles = [];
} catch (error) {
if (error.name === 'AbortError') {
document.getElementById(messageId).textContent = 'ุชู… ุฅูŠู‚ุงู ุงู„ุชูˆู„ูŠุฏ.';
} else {
document.getElementById(messageId).textContent = 'ุนุฐุฑุงู‹ุŒ ุญุฏุซ ุฎุทุฃ ููŠ ุงู„ู…ุนุงู„ุฌุฉ.';
}
} finally {
currentController = null;
toggleSendButton(false);
}
scrollToBottom();
}
function adjustTextareaHeight() {
messageInput.style.height = 'auto';
messageInput.style.height = Math.min(messageInput.scrollHeight, 60) + 'px';
}
function scrollToBottom() {
window.scrollTo({
top: document.documentElement.scrollHeight,
behavior: 'smooth'
});
}
// Event Listeners
document.addEventListener('click', async (e) => {
if (e.target.closest('.copy-button')) {
const messageId = e.target.closest('.copy-button').dataset.messageId;
const text = document.getElementById(messageId).textContent;
await navigator.clipboard.writeText(text);
const button = e.target.closest('.copy-button');
button.innerHTML = `
<svg class="w-3 h-3" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M5 13l4 4L19 7"></path>
</svg>
ุชู… ุงู„ู†ุณุฎ!
`;
setTimeout(() => {
button.innerHTML = `
<svg class="w-3 h-3" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M8 5H6a2 2 0 00-2 2v12a2 2 0 002 2h10a2 2 0 002-2v-1M8 5a2 2 0 002 2h2a2 2 0 002-2M8 5a2 2 0 012-2h2a2 2 0 012 2"></path>
</svg>
ู†ุณุฎ
`;
}, 2000);
}
if (e.target.closest('.speak-button')) {
const messageId = e.target.closest('.speak-button').dataset.messageId;
const text = document.getElementById(messageId).textContent;
speakText(text);
}
if (e.target.closest('.edit-button')) {
const messageId = e.target.closest('.edit-button').dataset.messageId;
openEditModal(messageId);
}
});
attachButton.addEventListener('click', () => {
const input = document.createElement('input');
input.type = 'file';
input.accept = '*/*';
input.onchange = async (e) => {
const file = e.target.files[0];
if (file) {
const filename = await handleFileUpload(file);
if (filename) {
messageInput.value += `\nู…ู„ู ู…ุฑูู‚: ${filename}`;
adjustTextareaHeight();
}
}
};
input.click();
});
// Event listeners for buttons and inputs
sendButton.addEventListener('click', sendMessage);
messageInput.addEventListener('keypress', (e) => {
if (e.key === 'Enter' && !e.shiftKey) {
e.preventDefault();
sendMessage();
}
});
messageInput.addEventListener('input', adjustTextareaHeight);
saveEdit.addEventListener('click', () => {
if (currentEditMessageId) {
const messageElement = document.getElementById(currentEditMessageId);
messageElement.innerHTML = formatCode(editText.value);
Prism.highlightAllUnder(messageElement);
}
closeEditModal();
});
cancelEdit.addEventListener('click', closeEditModal);
searchButton.addEventListener('click', () => {
isSearchEnabled = !isSearchEnabled;
searchButton.classList.toggle('active');
messageInput.placeholder = isSearchEnabled ? "ุงุจุญุซ ุนู†..." : "ุงูƒุชุจ ุฑุณุงู„ุชูƒ ู‡ู†ุง...";
});
darkModeToggle.addEventListener('click', () => {
document.body.classList.toggle('dark-mode');
document.body.classList.toggle('light-mode');
});
aiToggle.addEventListener('click', function() {
this.classList.toggle('active');
});
styleSelect.addEventListener('change', (e) => {
currentStyle = e.target.value;
});
clearButton.addEventListener('click', () => {
messagesContainer.innerHTML = '';
chatHistory = [];
uploadedFiles = [];
});
// Initial welcome message
const initialMessageId = 'msg-initial';
messagesContainer.insertAdjacentHTML('beforeend', createBotMessage('', initialMessageId));
typeText(initialMessageId, 'ู…ุฑุญุจุงู‹! ูƒูŠู ูŠู…ูƒู†ู†ูŠ ู…ุณุงุนุฏุชูƒ ุงู„ูŠูˆู…ุŸ');
// Initialize Prism.js
document.addEventListener('DOMContentLoaded', () => {
Prism.highlightAll();
});
// Support for paste events
messageInput.addEventListener('paste', (e) => {
const files = e.clipboardData.files;
if (files.length > 0) {
e.preventDefault();
handleFileUpload(files[0]).then(filename => {
if (filename) {
messageInput.value += `\nู…ู„ู ู…ุฑูู‚: ${filename}`;
adjustTextareaHeight();
}
});
}
});
// Support for drag and drop
messageInput.addEventListener('dragover', (e) => {
e.preventDefault();
e.stopPropagation();
messageInput.classList.add('dragover');
});
messageInput.addEventListener('dragleave', () => {
messageInput.classList.remove('dragover');
});
messageInput.addEventListener('drop', (e) => {
e.preventDefault();
e.stopPropagation();
messageInput.classList.remove('dragover');
const files = e.dataTransfer.files;
if (files.length > 0) {
handleFileUpload(files[0]).then(filename => {
if (filename) {
messageInput.value += `\nู…ู„ู ู…ุฑูู‚: ${filename}`;
adjustTextareaHeight();
}
});
}
});
</script>
</body>
</html>